Home | History | Annotate | Download | only in samplecode
      1 
      2 /*
      3  * Copyright 2011 Google Inc.
      4  *
      5  * Use of this source code is governed by a BSD-style license that can be
      6  * found in the LICENSE file.
      7  */
      8 #include "SampleCode.h"
      9 #include "SkBlurMask.h"
     10 #include "SkBlurMaskFilter.h"
     11 #include "SkCanvas.h"
     12 #include "SkParsePath.h"
     13 #include "SkPath.h"
     14 #include "SkRandom.h"
     15 #include "SkView.h"
     16 
     17 
     18 static void test_huge_stroke(SkCanvas* canvas) {
     19     SkRect srcR = { 0, 0, 72000, 54000 };
     20     SkRect dstR = { 0, 0, 640, 480 };
     21 
     22     SkPath path;
     23     path.moveTo(17600, 8000);
     24     path.lineTo(52800, 8000);
     25     path.lineTo(52800, 41600);
     26     path.lineTo(17600, 41600);
     27     path.close();
     28 
     29     SkPaint paint;
     30     paint.setAntiAlias(true);
     31     paint.setStrokeWidth(8000);
     32     paint.setStrokeMiter(10);
     33     paint.setStrokeCap(SkPaint::kButt_Cap);
     34     paint.setStrokeJoin(SkPaint::kRound_Join);
     35     paint.setStyle(SkPaint::kStroke_Style);
     36 
     37     SkMatrix matrix;
     38     matrix.setRectToRect(srcR, dstR, SkMatrix::kCenter_ScaleToFit);
     39     canvas->concat(matrix);
     40 
     41     canvas->drawPath(path, paint);
     42 }
     43 
     44 #if 0
     45 static void test_blur() {
     46     uint8_t cell[9];
     47     memset(cell, 0xFF, sizeof(cell));
     48     SkMask src;
     49     src.fImage = cell;
     50     src.fFormat = SkMask::kA8_Format;
     51     SkMask dst;
     52 
     53     for (int y = 1; y <= 3; y++) {
     54         for (int x = 1; x <= 3; x++) {
     55             src.fBounds.set(0, 0, x, y);
     56             src.fRowBytes = src.fBounds.width();
     57 
     58             SkScalar radius = 1.f;
     59 
     60             printf("src [%d %d %d %d] radius %g\n", src.fBounds.fLeft, src.fBounds.fTop,
     61                    src.fBounds.fRight, src.fBounds.fBottom, radius);
     62 
     63             SkBlurMask::Blur(&dst, src, radius, SkBlurMask::kNormal_Style);
     64             uint8_t* dstPtr = dst.fImage;
     65 
     66             for (int y = 0; y < dst.fBounds.height(); y++) {
     67                 for (int x = 0; x < dst.fBounds.width(); x++) {
     68                     printf(" %02X", dstPtr[x]);
     69                 }
     70                 printf("\n");
     71                 dstPtr += dst.fRowBytes;
     72             }
     73         }
     74     }
     75 }
     76 #endif
     77 
     78 static void scale_to_width(SkPath* path, SkScalar dstWidth) {
     79     const SkRect& bounds = path->getBounds();
     80     SkScalar scale = dstWidth / bounds.width();
     81     SkMatrix matrix;
     82 
     83     matrix.setScale(scale, scale);
     84     path->transform(matrix);
     85 }
     86 
     87 static const struct {
     88     SkPaint::Style  fStyle;
     89     SkPaint::Join   fJoin;
     90     int             fStrokeWidth;
     91 } gRec[] = {
     92     { SkPaint::kFill_Style,             SkPaint::kMiter_Join,   0 },
     93     { SkPaint::kStroke_Style,           SkPaint::kMiter_Join,   0 },
     94     { SkPaint::kStroke_Style,           SkPaint::kMiter_Join,   10 },
     95     { SkPaint::kStrokeAndFill_Style,    SkPaint::kMiter_Join,   10 },
     96 };
     97 
     98 class StrokePathView : public SampleView {
     99     SkScalar    fWidth;
    100     SkPath      fPath;
    101 public:
    102     StrokePathView() {
    103 //        test_blur();
    104         fWidth = SkIntToScalar(120);
    105 
    106 #if 0
    107         const char str[] =
    108             "M 0, 3"
    109             "C 10, -10, 30, -10, 0, 28"
    110             "C -30, -10, -10, -10, 0, 3"
    111             "Z";
    112         SkParsePath::FromSVGString(str, &fPath);
    113 #else
    114         fPath.addCircle(0, 0, SkIntToScalar(50), SkPath::kCW_Direction);
    115         fPath.addCircle(0, SkIntToScalar(-50), SkIntToScalar(30), SkPath::kCW_Direction);
    116 #endif
    117 
    118         scale_to_width(&fPath, fWidth);
    119         const SkRect& bounds = fPath.getBounds();
    120         fPath.offset(-bounds.fLeft, -bounds.fTop);
    121 
    122         this->setBGColor(0xFFDDDDDD);
    123     }
    124 
    125 protected:
    126     // overrides from SkEventSink
    127     virtual bool onQuery(SkEvent* evt) {
    128         if (SampleCode::TitleQ(*evt)) {
    129             SampleCode::TitleR(evt, "StrokePath");
    130             return true;
    131         }
    132         return this->INHERITED::onQuery(evt);
    133     }
    134 
    135     SkRandom rand;
    136 
    137     void drawSet(SkCanvas* canvas, SkPaint* paint) {
    138         SkAutoCanvasRestore acr(canvas, true);
    139 
    140         for (size_t i = 0; i < SK_ARRAY_COUNT(gRec); i++) {
    141             paint->setStyle(gRec[i].fStyle);
    142             paint->setStrokeJoin(gRec[i].fJoin);
    143             paint->setStrokeWidth(SkIntToScalar(gRec[i].fStrokeWidth));
    144             canvas->drawPath(fPath, *paint);
    145             canvas->translate(fWidth * 5 / 4, 0);
    146         }
    147     }
    148 
    149     virtual void onDrawContent(SkCanvas* canvas) {
    150         test_huge_stroke(canvas); return;
    151         canvas->translate(SkIntToScalar(10), SkIntToScalar(10));
    152 
    153         SkPaint paint;
    154         paint.setAntiAlias(true);
    155 
    156         if (true) {
    157             canvas->drawColor(SK_ColorBLACK);
    158 
    159             paint.setTextSize(24);
    160             paint.setColor(SK_ColorWHITE);
    161             canvas->translate(10, 30);
    162 
    163             static const SkBlurStyle gStyle[] = {
    164                 kNormal_SkBlurStyle,
    165                 kInner_SkBlurStyle,
    166                 kOuter_SkBlurStyle,
    167                 kSolid_SkBlurStyle,
    168             };
    169             for (int x = 0; x < 5; x++) {
    170                 SkMaskFilter* mf;
    171                 SkScalar sigma = SkBlurMask::ConvertRadiusToSigma(SkIntToScalar(4));
    172                 for (int y = 0; y < 10; y++) {
    173                     if (x) {
    174                         mf = SkBlurMaskFilter::Create(gStyle[x - 1], sigma);
    175                         paint.setMaskFilter(mf)->unref();
    176                     }
    177                     canvas->drawText("Title Bar", 9, x*SkIntToScalar(100), y*SkIntToScalar(30), paint);
    178                     sigma *= 0.75f;
    179                 }
    180 
    181             }
    182             return;
    183         }
    184 
    185         paint.setColor(SK_ColorBLUE);
    186 
    187 #if 1
    188         SkPath p;
    189         float r = rand.nextUScalar1() + 0.5f;
    190         SkScalar x = 0, y = 0;
    191         p.moveTo(x, y);
    192 #if 0
    193         p.cubicTo(x-75*r, y+75*r, x-40*r, y+125*r, x, y+85*r);
    194         p.cubicTo(x+40*r, y+125*r, x+75*r, y+75*r, x, y);
    195 #else
    196         p.cubicTo(x+75*r, y+75*r, x+40*r, y+125*r, x, y+85*r);
    197         p.cubicTo(x-40*r, y+125*r, x-75*r, y+75*r, x, y);
    198 #endif
    199         p.close();
    200         fPath = p;
    201         fPath.offset(100, 0);
    202 #endif
    203 
    204         fPath.setFillType(SkPath::kWinding_FillType);
    205         drawSet(canvas, &paint);
    206 
    207         canvas->translate(0, fPath.getBounds().height() * 5 / 4);
    208         fPath.setFillType(SkPath::kEvenOdd_FillType);
    209         drawSet(canvas, &paint);
    210     }
    211 
    212     virtual SkView::Click* onFindClickHandler(SkScalar x, SkScalar y,
    213                                               unsigned modi) SK_OVERRIDE {
    214         this->inval(NULL);
    215         return this->INHERITED::onFindClickHandler(x, y, modi);
    216     }
    217 private:
    218     typedef SampleView INHERITED;
    219 };
    220 
    221 //////////////////////////////////////////////////////////////////////////////
    222 
    223 static SkView* MyFactory() { return new StrokePathView; }
    224 static SkViewRegister reg(MyFactory);
    225