/* * Copyright 2011 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include "SampleCode.h" #include "SkView.h" #include "SkCanvas.h" #include "SkGraphics.h" #include "SkRandom.h" static void test_clearonlayers(SkCanvas* canvas) { SkCanvas& c = *canvas; SkPaint paint; paint.setColor(SK_ColorBLUE); paint.setStyle(SkPaint::kStrokeAndFill_Style); SkRect rect = SkRect::MakeXYWH(25, 25, 50, 50); c.drawRect(rect, paint); c.clipRect(rect); c.saveLayer(NULL, NULL); rect = SkRect::MakeXYWH(50, 10, 40, 80); c.clipRect(rect, SkRegion::kUnion_Op); rect = SkRect::MakeXYWH(50, 0, 50, 100); // You might draw something here, but it's not necessary. // paint.setColor(SK_ColorRED); // c.drawRect(rect, paint); paint.setXfermodeMode(SkXfermode::kClear_Mode); c.drawRect(rect, paint); c.restore(); } static void test_strokerect(SkCanvas* canvas, const SkRect& r) { SkPaint p; p.setAntiAlias(true); p.setStyle(SkPaint::kStroke_Style); p.setStrokeWidth(4); canvas->drawRect(r, p); SkPath path; SkRect r2(r); r2.offset(18, 0); path.addRect(r2); canvas->drawPath(path, p); } static void test_strokerect(SkCanvas* canvas) { canvas->drawColor(SK_ColorWHITE); SkRect r; r.set(10, 10, 14, 14); r.offset(0.25f, 0.3333f); test_strokerect(canvas, r); canvas->translate(0, 20); r.set(10, 10, 14.5f, 14.5f); r.offset(0.25f, 0.3333f); test_strokerect(canvas, r); canvas->translate(0, 20); r.set(10, 10, 14.5f, 20); r.offset(0.25f, 0.3333f); test_strokerect(canvas, r); canvas->translate(0, 20); r.set(10, 10, 20, 14.5f); r.offset(0.25f, 0.3333f); test_strokerect(canvas, r); canvas->translate(0, 20); r.set(10, 10, 20, 20); r.offset(0.25f, 0.3333f); test_strokerect(canvas, r); canvas->translate(0, 20); } class Draw : public SkRefCnt { public: Draw() : fFlags(0) {} enum Flags { kSelected_Flag = 1 << 0 }; int getFlags() const { return fFlags; } void setFlags(int flags); bool isSelected() const { return SkToBool(fFlags & kSelected_Flag); } void setSelected(bool pred) { if (pred) { fFlags |= kSelected_Flag; } else { fFlags &= ~kSelected_Flag; } } void draw(SkCanvas* canvas) { int sc = canvas->save(); this->onDraw(canvas); canvas->restoreToCount(sc); if (this->isSelected()) { this->drawSelection(canvas); } } void drawSelection(SkCanvas* canvas) { int sc = canvas->save(); this->onDrawSelection(canvas); canvas->restoreToCount(sc); } void getBounds(SkRect* bounds) { this->onGetBounds(bounds); } bool hitTest(SkScalar x, SkScalar y) { return this->onHitTest(x, y); } void offset(SkScalar dx, SkScalar dy) { if (dx || dy) { this->onOffset(dx, dy); } } protected: virtual void onDraw(SkCanvas*) = 0; virtual void onGetBounds(SkRect*) = 0; virtual void onOffset(SkScalar dx, SkScalar dy) = 0; virtual void onDrawSelection(SkCanvas* canvas) { SkRect r; this->getBounds(&r); SkPaint paint; SkPoint pts[4]; r.toQuad(pts); paint.setStrokeWidth(SkIntToScalar(10)); paint.setColor(0x80FF8844); paint.setStrokeCap(SkPaint::kRound_Cap); canvas->drawPoints(SkCanvas::kPoints_PointMode, 4, pts, paint); } virtual bool onHitTest(SkScalar x, SkScalar y) { SkRect bounds; this->getBounds(&bounds); return bounds.contains(x, y); } private: int fFlags; }; class RDraw : public Draw { public: enum Style { kRect_Style, kOval_Style, kRRect_Style, kFrame_Style }; RDraw(const SkRect& r, Style s) : fRect(r), fStyle(s) {} void setRect(const SkRect& r) { fRect = r; } void setPaint(const SkPaint& p) { fPaint = p; } protected: virtual void onDraw(SkCanvas* canvas) { switch (fStyle) { case kRect_Style: canvas->drawRect(fRect, fPaint); break; case kOval_Style: canvas->drawOval(fRect, fPaint); break; case kRRect_Style: { SkScalar rx = fRect.width() / 5; SkScalar ry = fRect.height() / 5; if (rx < ry) { ry = rx; } else { rx = ry; } canvas->drawRoundRect(fRect, rx, ry, fPaint); break; } case kFrame_Style: { SkPath path; path.addOval(fRect, SkPath::kCW_Direction); SkRect r = fRect; r.inset(fRect.width()/6, 0); path.addOval(r, SkPath::kCCW_Direction); canvas->drawPath(path, fPaint); break; } } } virtual void onGetBounds(SkRect* bounds) { *bounds = fRect; } virtual void onOffset(SkScalar dx, SkScalar dy) { fRect.offset(dx, dy); } private: SkRect fRect; SkPaint fPaint; Style fStyle; }; class DrawFactory { public: DrawFactory() { fPaint.setAntiAlias(true); } const SkPaint& getPaint() const { return fPaint; } void setPaint(const SkPaint& p) { fPaint = p; } virtual Draw* create(const SkPoint&, const SkPoint&) = 0; private: SkPaint fPaint; }; class RectFactory : public DrawFactory { public: virtual Draw* create(const SkPoint& p0, const SkPoint& p1) { SkRect r; r.set(p0.x(), p0.y(), p1.x(), p1.y()); r.sort(); // RDraw* d = new RDraw(r, RDraw::kRRect_Style); RDraw* d = new RDraw(r, RDraw::kFrame_Style); d->setPaint(this->getPaint()); return d; } }; class DrawView : public SkView { Draw* fDraw; DrawFactory* fFactory; SkRandom fRand; SkTDArray<Draw*> fList; public: DrawView() : fDraw(NULL) { fFactory = new RectFactory; } virtual ~DrawView() { fList.unrefAll(); SkSafeUnref(fDraw); delete fFactory; } Draw* setDraw(Draw* d) { SkRefCnt_SafeAssign(fDraw, d); return d; } SkColor randColor() { return (SkColor)fRand.nextU() | 0xFF000000; } Draw* hitTestList(SkScalar x, SkScalar y) const { Draw** first = fList.begin(); for (Draw** iter = fList.end(); iter > first;) { --iter; if ((*iter)->hitTest(x, y)) { return *iter; } } return NULL; } protected: // overrides from SkEventSink virtual bool onQuery(SkEvent* evt) { if (SampleCode::TitleQ(*evt)) { SampleCode::TitleR(evt, "Draw"); return true; } return this->INHERITED::onQuery(evt); } void drawBG(SkCanvas* canvas) { canvas->drawColor(0xFFDDDDDD); // canvas->drawColor(SK_ColorWHITE); } virtual void onDraw(SkCanvas* canvas) { this->drawBG(canvas); test_clearonlayers(canvas); return; // test_strokerect(canvas); return; for (Draw** iter = fList.begin(); iter < fList.end(); iter++) { (*iter)->draw(canvas); } if (fDraw) { fDraw->draw(canvas); } } virtual SkView::Click* onFindClickHandler(SkScalar x, SkScalar y) { for (Draw** iter = fList.begin(); iter < fList.end(); iter++) { (*iter)->setSelected(false); } Click* c = new Click(this); Draw* d = this->hitTestList(x, y); if (d) { d->setSelected(true); c->setType("dragger"); } else { c->setType("maker"); } return c; } virtual bool onClick(Click* click) { if (Click::kUp_State == click->fState) { if (click->isType("maker")) { if (SkPoint::Distance(click->fOrig, click->fCurr) > SkIntToScalar(3)) { *fList.append() = fDraw; } else { fDraw->unref(); } fDraw = NULL; } return true; } if (Click::kDown_State == click->fState) { SkPaint p = fFactory->getPaint(); p.setColor(this->randColor()); fFactory->setPaint(p); } if (click->isType("maker")) { this->setDraw(fFactory->create(click->fOrig, click->fCurr))->unref(); } else if (click->isType("dragger")) { for (Draw** iter = fList.begin(); iter < fList.end(); iter++) { if ((*iter)->isSelected()) { (*iter)->offset(click->fCurr.x() - click->fPrev.x(), click->fCurr.y() - click->fPrev.y()); } } } this->inval(NULL); return true; } private: typedef SkView INHERITED; }; ////////////////////////////////////////////////////////////////////////////// static SkView* MyFactory() { return new DrawView; } static SkViewRegister reg(MyFactory);