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