epoger@google.com | ec3ed6a | 2011-07-28 14:26:00 +0000 | [diff] [blame] | 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 | */ |
reed@google.com | bf0001d | 2014-01-13 14:53:55 +0000 | [diff] [blame] | 7 | |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 8 | #include "include/core/SkBitmap.h" |
| 9 | #include "include/core/SkCanvas.h" |
| 10 | #include "include/core/SkColorFilter.h" |
| 11 | #include "include/core/SkColorPriv.h" |
| 12 | #include "include/core/SkGraphics.h" |
| 13 | #include "include/core/SkPath.h" |
| 14 | #include "include/core/SkRegion.h" |
| 15 | #include "include/core/SkShader.h" |
| 16 | #include "include/core/SkStream.h" |
| 17 | #include "include/core/SkTime.h" |
| 18 | #include "include/core/SkTypeface.h" |
| 19 | #include "include/effects/SkCornerPathEffect.h" |
| 20 | #include "include/effects/SkGradientShader.h" |
| 21 | #include "include/private/SkTo.h" |
| 22 | #include "include/utils/SkRandom.h" |
| 23 | #include "samplecode/Sample.h" |
| 24 | #include "src/utils/SkUTF.h" |
reed@android.com | 8898363 | 2009-03-23 16:05:19 +0000 | [diff] [blame] | 25 | |
| 26 | static SkRandom gRand; |
| 27 | |
reed@android.com | 8898363 | 2009-03-23 16:05:19 +0000 | [diff] [blame] | 28 | static void generate_pts(SkPoint pts[], int count, int w, int h) { |
| 29 | for (int i = 0; i < count; i++) { |
| 30 | pts[i].set(gRand.nextUScalar1() * 3 * w - SkIntToScalar(w), |
| 31 | gRand.nextUScalar1() * 3 * h - SkIntToScalar(h)); |
| 32 | } |
| 33 | } |
| 34 | |
| 35 | static bool check_zeros(const SkPMColor pixels[], int count, int skip) { |
| 36 | for (int i = 0; i < count; i++) { |
| 37 | if (*pixels) { |
| 38 | return false; |
| 39 | } |
| 40 | pixels += skip; |
| 41 | } |
| 42 | return true; |
| 43 | } |
| 44 | |
| 45 | static bool check_bitmap_margin(const SkBitmap& bm, int margin) { |
| 46 | size_t rb = bm.rowBytes(); |
| 47 | for (int i = 0; i < margin; i++) { |
| 48 | if (!check_zeros(bm.getAddr32(0, i), bm.width(), 1)) { |
| 49 | return false; |
| 50 | } |
| 51 | int bottom = bm.height() - i - 1; |
| 52 | if (!check_zeros(bm.getAddr32(0, bottom), bm.width(), 1)) { |
| 53 | return false; |
| 54 | } |
| 55 | // left column |
reed@google.com | 7fa2a65 | 2014-01-27 13:42:58 +0000 | [diff] [blame] | 56 | if (!check_zeros(bm.getAddr32(i, 0), bm.height(), SkToInt(rb >> 2))) { |
reed@android.com | 8898363 | 2009-03-23 16:05:19 +0000 | [diff] [blame] | 57 | return false; |
| 58 | } |
| 59 | int right = bm.width() - margin + i; |
reed@google.com | 7fa2a65 | 2014-01-27 13:42:58 +0000 | [diff] [blame] | 60 | if (!check_zeros(bm.getAddr32(right, 0), bm.height(), |
| 61 | SkToInt(rb >> 2))) { |
reed@android.com | 8898363 | 2009-03-23 16:05:19 +0000 | [diff] [blame] | 62 | return false; |
| 63 | } |
| 64 | } |
| 65 | return true; |
| 66 | } |
| 67 | |
reed@android.com | 2893728 | 2009-08-28 15:34:46 +0000 | [diff] [blame] | 68 | #define WIDTH 620 |
| 69 | #define HEIGHT 460 |
| 70 | #define MARGIN 10 |
| 71 | |
| 72 | static void line_proc(SkCanvas* canvas, const SkPaint& paint, |
| 73 | const SkBitmap& bm) { |
| 74 | const int N = 2; |
| 75 | SkPoint pts[N]; |
| 76 | for (int i = 0; i < 400; i++) { |
| 77 | generate_pts(pts, N, WIDTH, HEIGHT); |
| 78 | |
Hal Canary | 23e474c | 2017-05-15 13:35:35 -0400 | [diff] [blame] | 79 | canvas->drawLine(pts[0], pts[1], paint); |
reed@android.com | 2893728 | 2009-08-28 15:34:46 +0000 | [diff] [blame] | 80 | if (!check_bitmap_margin(bm, MARGIN)) { |
| 81 | SkDebugf("---- hairline failure (%g %g) (%g %g)\n", |
| 82 | pts[0].fX, pts[0].fY, pts[1].fX, pts[1].fY); |
| 83 | break; |
| 84 | } |
| 85 | } |
| 86 | } |
| 87 | |
| 88 | static void poly_proc(SkCanvas* canvas, const SkPaint& paint, |
| 89 | const SkBitmap& bm) { |
| 90 | const int N = 8; |
| 91 | SkPoint pts[N]; |
| 92 | for (int i = 0; i < 50; i++) { |
| 93 | generate_pts(pts, N, WIDTH, HEIGHT); |
rmistry@google.com | ae933ce | 2012-08-23 18:19:56 +0000 | [diff] [blame] | 94 | |
reed@android.com | 2893728 | 2009-08-28 15:34:46 +0000 | [diff] [blame] | 95 | SkPath path; |
| 96 | path.moveTo(pts[0]); |
| 97 | for (int j = 1; j < N; j++) { |
| 98 | path.lineTo(pts[j]); |
| 99 | } |
| 100 | canvas->drawPath(path, paint); |
| 101 | } |
| 102 | } |
| 103 | |
| 104 | static SkPoint ave(const SkPoint& a, const SkPoint& b) { |
| 105 | SkPoint c = a + b; |
| 106 | c.fX = SkScalarHalf(c.fX); |
| 107 | c.fY = SkScalarHalf(c.fY); |
| 108 | return c; |
| 109 | } |
| 110 | |
| 111 | static void quad_proc(SkCanvas* canvas, const SkPaint& paint, |
| 112 | const SkBitmap& bm) { |
| 113 | const int N = 30; |
| 114 | SkPoint pts[N]; |
| 115 | for (int i = 0; i < 10; i++) { |
| 116 | generate_pts(pts, N, WIDTH, HEIGHT); |
rmistry@google.com | ae933ce | 2012-08-23 18:19:56 +0000 | [diff] [blame] | 117 | |
reed@android.com | 2893728 | 2009-08-28 15:34:46 +0000 | [diff] [blame] | 118 | SkPath path; |
| 119 | path.moveTo(pts[0]); |
| 120 | for (int j = 1; j < N - 2; j++) { |
| 121 | path.quadTo(pts[j], ave(pts[j], pts[j+1])); |
| 122 | } |
| 123 | path.quadTo(pts[N - 2], pts[N - 1]); |
rmistry@google.com | ae933ce | 2012-08-23 18:19:56 +0000 | [diff] [blame] | 124 | |
reed@android.com | 2893728 | 2009-08-28 15:34:46 +0000 | [diff] [blame] | 125 | canvas->drawPath(path, paint); |
| 126 | } |
| 127 | } |
| 128 | |
| 129 | static void add_cubic(SkPath* path, const SkPoint& mid, const SkPoint& end) { |
| 130 | SkPoint start; |
| 131 | path->getLastPt(&start); |
| 132 | path->cubicTo(ave(start, mid), ave(mid, end), end); |
| 133 | } |
| 134 | |
| 135 | static void cube_proc(SkCanvas* canvas, const SkPaint& paint, |
| 136 | const SkBitmap& bm) { |
| 137 | const int N = 30; |
| 138 | SkPoint pts[N]; |
| 139 | for (int i = 0; i < 10; i++) { |
| 140 | generate_pts(pts, N, WIDTH, HEIGHT); |
rmistry@google.com | ae933ce | 2012-08-23 18:19:56 +0000 | [diff] [blame] | 141 | |
reed@android.com | 2893728 | 2009-08-28 15:34:46 +0000 | [diff] [blame] | 142 | SkPath path; |
| 143 | path.moveTo(pts[0]); |
| 144 | for (int j = 1; j < N - 2; j++) { |
| 145 | add_cubic(&path, pts[j], ave(pts[j], pts[j+1])); |
| 146 | } |
| 147 | add_cubic(&path, pts[N - 2], pts[N - 1]); |
rmistry@google.com | ae933ce | 2012-08-23 18:19:56 +0000 | [diff] [blame] | 148 | |
reed@android.com | 2893728 | 2009-08-28 15:34:46 +0000 | [diff] [blame] | 149 | canvas->drawPath(path, paint); |
| 150 | } |
| 151 | } |
| 152 | |
| 153 | typedef void (*HairProc)(SkCanvas*, const SkPaint&, const SkBitmap&); |
| 154 | |
| 155 | static const struct { |
| 156 | const char* fName; |
| 157 | HairProc fProc; |
| 158 | } gProcs[] = { |
| 159 | { "line", line_proc }, |
| 160 | { "poly", poly_proc }, |
| 161 | { "quad", quad_proc }, |
| 162 | { "cube", cube_proc }, |
| 163 | }; |
| 164 | |
| 165 | static int cycle_hairproc_index(int index) { |
| 166 | return (index + 1) % SK_ARRAY_COUNT(gProcs); |
| 167 | } |
reed@android.com | 8898363 | 2009-03-23 16:05:19 +0000 | [diff] [blame] | 168 | |
Ben Wagner | b2c4ea6 | 2018-08-08 11:36:17 -0400 | [diff] [blame] | 169 | class HairlineView : public Sample { |
reed@android.com | a3d9010 | 2009-11-30 12:48:33 +0000 | [diff] [blame] | 170 | SkMSec fNow; |
reed@android.com | 2893728 | 2009-08-28 15:34:46 +0000 | [diff] [blame] | 171 | int fProcIndex; |
| 172 | bool fDoAA; |
reed@android.com | 8898363 | 2009-03-23 16:05:19 +0000 | [diff] [blame] | 173 | public: |
rmistry@google.com | ae933ce | 2012-08-23 18:19:56 +0000 | [diff] [blame] | 174 | HairlineView() { |
reed@android.com | 2893728 | 2009-08-28 15:34:46 +0000 | [diff] [blame] | 175 | fProcIndex = 0; |
| 176 | fDoAA = true; |
reed@android.com | a3d9010 | 2009-11-30 12:48:33 +0000 | [diff] [blame] | 177 | fNow = 0; |
reed@android.com | 2893728 | 2009-08-28 15:34:46 +0000 | [diff] [blame] | 178 | } |
rmistry@google.com | ae933ce | 2012-08-23 18:19:56 +0000 | [diff] [blame] | 179 | |
reed@android.com | 8898363 | 2009-03-23 16:05:19 +0000 | [diff] [blame] | 180 | protected: |
Hal Canary | 8a02731 | 2019-07-03 10:55:44 -0400 | [diff] [blame] | 181 | SkString name() override { return SkStringPrintf("Hair-%s", gProcs[fProcIndex].fName); } |
rmistry@google.com | ae933ce | 2012-08-23 18:19:56 +0000 | [diff] [blame] | 182 | |
reed@android.com | 8898363 | 2009-03-23 16:05:19 +0000 | [diff] [blame] | 183 | void show_bitmaps(SkCanvas* canvas, const SkBitmap& b0, const SkBitmap& b1, |
| 184 | const SkIRect& inset) { |
halcanary | 96fcdcc | 2015-08-27 07:41:13 -0700 | [diff] [blame] | 185 | canvas->drawBitmap(b0, 0, 0, nullptr); |
| 186 | canvas->drawBitmap(b1, SkIntToScalar(b0.width()), 0, nullptr); |
reed@android.com | 8898363 | 2009-03-23 16:05:19 +0000 | [diff] [blame] | 187 | } |
| 188 | |
mtklein | 36352bf | 2015-03-25 18:17:31 -0700 | [diff] [blame] | 189 | void onDrawContent(SkCanvas* canvas) override { |
reed@android.com | a3d9010 | 2009-11-30 12:48:33 +0000 | [diff] [blame] | 190 | gRand.setSeed(fNow); |
rmistry@google.com | ae933ce | 2012-08-23 18:19:56 +0000 | [diff] [blame] | 191 | |
reed@android.com | 8898363 | 2009-03-23 16:05:19 +0000 | [diff] [blame] | 192 | SkBitmap bm, bm2; |
commit-bot@chromium.org | a8c1831 | 2014-02-17 02:55:57 +0000 | [diff] [blame] | 193 | bm.allocN32Pixels(WIDTH + MARGIN*2, HEIGHT + MARGIN*2); |
reed@android.com | 8898363 | 2009-03-23 16:05:19 +0000 | [diff] [blame] | 194 | // this will erase our margin, which we want to always stay 0 |
junov@google.com | dbfac8a | 2012-12-06 21:47:40 +0000 | [diff] [blame] | 195 | bm.eraseColor(SK_ColorTRANSPARENT); |
reed@android.com | 8898363 | 2009-03-23 16:05:19 +0000 | [diff] [blame] | 196 | |
commit-bot@chromium.org | a8c1831 | 2014-02-17 02:55:57 +0000 | [diff] [blame] | 197 | bm2.installPixels(SkImageInfo::MakeN32Premul(WIDTH, HEIGHT), |
commit-bot@chromium.org | 00f8d6c | 2014-05-29 15:57:20 +0000 | [diff] [blame] | 198 | bm.getAddr32(MARGIN, MARGIN), bm.rowBytes()); |
rmistry@google.com | ae933ce | 2012-08-23 18:19:56 +0000 | [diff] [blame] | 199 | |
reed@android.com | 8898363 | 2009-03-23 16:05:19 +0000 | [diff] [blame] | 200 | SkCanvas c2(bm2); |
| 201 | SkPaint paint; |
reed@android.com | 2893728 | 2009-08-28 15:34:46 +0000 | [diff] [blame] | 202 | paint.setAntiAlias(fDoAA); |
| 203 | paint.setStyle(SkPaint::kStroke_Style); |
| 204 | |
junov@google.com | dbfac8a | 2012-12-06 21:47:40 +0000 | [diff] [blame] | 205 | bm2.eraseColor(SK_ColorTRANSPARENT); |
reed@android.com | 2893728 | 2009-08-28 15:34:46 +0000 | [diff] [blame] | 206 | gProcs[fProcIndex].fProc(&c2, paint, bm); |
halcanary | 96fcdcc | 2015-08-27 07:41:13 -0700 | [diff] [blame] | 207 | canvas->drawBitmap(bm2, SkIntToScalar(10), SkIntToScalar(10), nullptr); |
reed@android.com | 8898363 | 2009-03-23 16:05:19 +0000 | [diff] [blame] | 208 | } |
| 209 | |
Hal Canary | 4124807 | 2019-07-11 16:32:53 -0400 | [diff] [blame] | 210 | bool onAnimate(double /*nanos*/) override { |
reed | d9adfe6 | 2015-02-01 19:01:04 -0800 | [diff] [blame] | 211 | if (fDoAA) { |
| 212 | fProcIndex = cycle_hairproc_index(fProcIndex); |
| 213 | // todo: signal that we want to rebuild our TITLE |
| 214 | } |
| 215 | fDoAA = !fDoAA; |
| 216 | return true; |
| 217 | } |
| 218 | |
Hal Canary | b1f411a | 2019-08-29 10:39:22 -0400 | [diff] [blame] | 219 | Sample::Click* onFindClickHandler(SkScalar x, SkScalar y, skui::ModifierKey modi) override { |
reed@android.com | 2893728 | 2009-08-28 15:34:46 +0000 | [diff] [blame] | 220 | fDoAA = !fDoAA; |
reed@google.com | 4d5c26d | 2013-01-08 16:17:50 +0000 | [diff] [blame] | 221 | return this->INHERITED::onFindClickHandler(x, y, modi); |
reed@android.com | 2893728 | 2009-08-28 15:34:46 +0000 | [diff] [blame] | 222 | } |
rmistry@google.com | ae933ce | 2012-08-23 18:19:56 +0000 | [diff] [blame] | 223 | |
reed@android.com | 2893728 | 2009-08-28 15:34:46 +0000 | [diff] [blame] | 224 | |
reed@android.com | 8898363 | 2009-03-23 16:05:19 +0000 | [diff] [blame] | 225 | private: |
Ben Wagner | b2c4ea6 | 2018-08-08 11:36:17 -0400 | [diff] [blame] | 226 | typedef Sample INHERITED; |
reed@android.com | 8898363 | 2009-03-23 16:05:19 +0000 | [diff] [blame] | 227 | }; |
| 228 | |
| 229 | ////////////////////////////////////////////////////////////////////////////// |
| 230 | |
Ben Wagner | b2c4ea6 | 2018-08-08 11:36:17 -0400 | [diff] [blame] | 231 | DEF_SAMPLE( return new HairlineView(); ) |