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