blob: 7ed16dcc460837648bddd370cc4303da267e4bfe [file] [log] [blame]
epoger@google.comec3ed6a2011-07-28 14:26:00 +00001
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 */
reed@android.com8a1c16f2008-12-17 15:59:43 +00008#include "SampleCode.h"
9#include "SkView.h"
10#include "SkCanvas.h"
11#include "Sk64.h"
12#include "SkGradientShader.h"
13#include "SkGraphics.h"
14#include "SkImageDecoder.h"
15#include "SkKernel33MaskFilter.h"
16#include "SkPath.h"
17#include "SkRandom.h"
18#include "SkRegion.h"
19#include "SkShader.h"
20#include "SkUtils.h"
reed@android.com8a1c16f2008-12-17 15:59:43 +000021#include "SkColorPriv.h"
22#include "SkColorFilter.h"
23#include "SkTime.h"
24#include "SkTypeface.h"
25#include "SkXfermode.h"
26
27#include "SkStream.h"
28#include "SkXMLParser.h"
29
30static const int gKernel[3][3] = {
31// { -1, -2, -1 }, { -2, 12, -2 }, { -1, -2, -1 }
32 { 1, 2, 1 }, { 2, 64-12, 2 }, { 1, 2, 1 }
33};
34static const int gShift = 6;
35
36class ReduceNoise : public SkKernel33ProcMaskFilter {
37public:
38 ReduceNoise(int percent256) : SkKernel33ProcMaskFilter(percent256) {}
39 virtual uint8_t computeValue(uint8_t* const* srcRows)
40 {
41 int c = srcRows[1][1];
42 int min = 255, max = 0;
43 for (int i = 0; i < 3; i++)
44 for (int j = 0; j < 3; j++)
45 if (i != 1 || j != 1)
46 {
47 int v = srcRows[i][j];
48 if (max < v)
49 max = v;
50 if (min > v)
51 min = v;
52 }
53 if (c > max) c = max;
54 // if (c < min) c = min;
55 return c;
56 }
djsollen@google.comba28d032012-03-26 17:57:35 +000057 SK_DECLARE_PUBLIC_FLATTENABLE_DESERIALIZATION_PROCS(ReduceNoise)
58
reed@android.com8a1c16f2008-12-17 15:59:43 +000059private:
60 ReduceNoise(SkFlattenableReadBuffer& rb) : SkKernel33ProcMaskFilter(rb) {}
reed@android.com8a1c16f2008-12-17 15:59:43 +000061};
62
63class Darken : public SkKernel33ProcMaskFilter {
64public:
65 Darken(int percent256) : SkKernel33ProcMaskFilter(percent256) {}
66 virtual uint8_t computeValue(uint8_t* const* srcRows)
67 {
68 int c = srcRows[1][1];
69 float f = c / 255.f;
reed@google.com82065d62011-02-07 15:30:46 +000070
reed@google.com9e39bb32011-05-18 12:17:53 +000071 if (c >= 0) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000072 f = sqrtf(f);
reed@google.com9e39bb32011-05-18 12:17:53 +000073 } else {
reed@android.com8a1c16f2008-12-17 15:59:43 +000074 f *= f;
75 }
76 SkASSERT(f >= 0 && f <= 1);
77 return (int)(f * 255);
78 }
djsollen@google.comba28d032012-03-26 17:57:35 +000079 SK_DECLARE_PUBLIC_FLATTENABLE_DESERIALIZATION_PROCS(Darken)
80
reed@android.com8a1c16f2008-12-17 15:59:43 +000081private:
82 Darken(SkFlattenableReadBuffer& rb) : SkKernel33ProcMaskFilter(rb) {}
reed@android.com8a1c16f2008-12-17 15:59:43 +000083};
84
85static SkMaskFilter* makemf() { return new Darken(0x30); }
86
reed@google.com9e39bb32011-05-18 12:17:53 +000087static void test_breakText() {
reed@android.com8a1c16f2008-12-17 15:59:43 +000088 SkPaint paint;
89 const char* text = "sdfkljAKLDFJKEWkldfjlk#$%&sdfs.dsj";
90 size_t length = strlen(text);
91 SkScalar width = paint.measureText(text, length);
reed@google.com82065d62011-02-07 15:30:46 +000092
reed@android.com8a1c16f2008-12-17 15:59:43 +000093 SkScalar mm = 0;
94 SkScalar nn = 0;
reed@google.com9e39bb32011-05-18 12:17:53 +000095 for (SkScalar w = 0; w <= width; w += SK_Scalar1) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000096 SkScalar m;
97 size_t n = paint.breakText(text, length, w, &m,
98 SkPaint::kBackward_TextBufferDirection);
reed@google.com82065d62011-02-07 15:30:46 +000099
reed@android.com8a1c16f2008-12-17 15:59:43 +0000100 SkASSERT(n <= length);
101 SkASSERT(m <= width);
reed@google.com82065d62011-02-07 15:30:46 +0000102
reed@google.com9e39bb32011-05-18 12:17:53 +0000103 if (n == 0) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000104 SkASSERT(m == 0);
reed@google.com9e39bb32011-05-18 12:17:53 +0000105 } else {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000106 // now assert that we're monotonic
reed@google.com9e39bb32011-05-18 12:17:53 +0000107 if (n == nn) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000108 SkASSERT(m == mm);
reed@google.com9e39bb32011-05-18 12:17:53 +0000109 } else {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000110 SkASSERT(n > nn);
111 SkASSERT(m > mm);
112 }
113 }
reed@google.com261b8e22011-04-14 17:53:24 +0000114 nn = SkIntToScalar(n);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000115 mm = m;
116 }
117
senorblanco@chromium.org64cc5792011-05-19 19:58:58 +0000118 SkDEBUGCODE(size_t length2 =) paint.breakText(text, length, width, &mm);
reed@google.com261b8e22011-04-14 17:53:24 +0000119 SkASSERT(length2 == length);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000120 SkASSERT(mm == width);
121}
122
123static SkRandom gRand;
124
125class SkPowerMode : public SkXfermode {
126public:
127 SkPowerMode(SkScalar exponent) { this->init(exponent); }
128
reed@google.com9e39bb32011-05-18 12:17:53 +0000129 virtual void xfer16(uint16_t dst[], const SkPMColor src[], int count,
130 const SkAlpha aa[]);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000131
132 typedef SkFlattenable* (*Factory)(SkFlattenableReadBuffer&);
reed@google.com82065d62011-02-07 15:30:46 +0000133
djsollen@google.comba28d032012-03-26 17:57:35 +0000134 SK_DECLARE_PUBLIC_FLATTENABLE_DESERIALIZATION_PROCS(SkPowerMode)
reed@google.com82065d62011-02-07 15:30:46 +0000135
reed@android.com8a1c16f2008-12-17 15:59:43 +0000136private:
137 SkScalar fExp; // user's value
138 uint8_t fTable[256]; // cache
139
140 void init(SkScalar exponent);
djsollen@google.com54924242012-03-29 15:18:04 +0000141 SkPowerMode(SkFlattenableReadBuffer& b) : INHERITED(b) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000142 // read the exponent
143 this->init(SkFixedToScalar(b.readS32()));
144 }
djsollen@google.com54924242012-03-29 15:18:04 +0000145 virtual void flatten(SkFlattenableWriteBuffer& b) const SK_OVERRIDE {
146 this->INHERITED::flatten(b);
147 b.write32(SkScalarToFixed(fExp));
148 }
reed@google.com82065d62011-02-07 15:30:46 +0000149
reed@android.com8a1c16f2008-12-17 15:59:43 +0000150 typedef SkXfermode INHERITED;
151};
152
reed@google.com9e39bb32011-05-18 12:17:53 +0000153void SkPowerMode::init(SkScalar e) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000154 fExp = e;
155 float ee = SkScalarToFloat(e);
reed@google.com82065d62011-02-07 15:30:46 +0000156
reed@android.com8a1c16f2008-12-17 15:59:43 +0000157 printf("------ %g\n", ee);
reed@google.com9e39bb32011-05-18 12:17:53 +0000158 for (int i = 0; i < 256; i++) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000159 float x = i / 255.f;
160 // printf(" %d %g", i, x);
161 x = powf(x, ee);
162 // printf(" %g", x);
163 int xx = SkScalarRound(SkFloatToScalar(x * 255));
164 // printf(" %d\n", xx);
165 fTable[i] = SkToU8(xx);
166 }
167}
168
reed@google.com9e39bb32011-05-18 12:17:53 +0000169void SkPowerMode::xfer16(uint16_t dst[], const SkPMColor src[], int count,
170 const SkAlpha aa[]) {
171 for (int i = 0; i < count; i++) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000172 SkPMColor c = src[i];
173 int r = SkGetPackedR32(c);
174 int g = SkGetPackedG32(c);
175 int b = SkGetPackedB32(c);
176 r = fTable[r];
177 g = fTable[g];
178 b = fTable[b];
179 dst[i] = SkPack888ToRGB16(r, g, b);
180 }
181}
182
183static const struct {
184 const char* fName;
185 uint32_t fFlags;
186 bool fFlushCache;
187} gHints[] = {
188 { "Linear", SkPaint::kLinearText_Flag, false },
189 { "Normal", 0, true },
190 { "Subpixel", SkPaint::kSubpixelText_Flag, true }
191};
192
reed@google.com9e39bb32011-05-18 12:17:53 +0000193static int count_char_points(const SkPaint& paint, char c) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000194 SkPath path;
reed@google.com82065d62011-02-07 15:30:46 +0000195
reed@android.com8a1c16f2008-12-17 15:59:43 +0000196 paint.getTextPath(&c, 1, 0, 0, &path);
197 return path.getPoints(NULL, 0);
198}
199
200static int gOld, gNew, gCount;
201
reed@google.com9e39bb32011-05-18 12:17:53 +0000202static void dump(int c, int oldc, int newc) {
203 if (oldc != newc) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000204 gOld += oldc;
205 gNew += newc;
206 gCount += 1;
207 printf("char %c: old = %3d, new = %3d, reduction %g%%\n", c, oldc, newc, 100. * (oldc - newc) / oldc);
208 }
209}
210
reed@google.com9e39bb32011-05-18 12:17:53 +0000211static void tab(int n) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000212// printf("[%d] ", n); return;
213 SkASSERT(n >= 0);
214 for (int i = 0; i < n; i++)
215 printf(" ");
216}
217
reed@google.com9e39bb32011-05-18 12:17:53 +0000218static void draw_rgn(const SkRegion& rgn, SkCanvas* canvas, const SkPaint& paint) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000219 SkRect r;
220 SkRegion::Iterator iter(rgn);
reed@google.com82065d62011-02-07 15:30:46 +0000221
reed@google.com9e39bb32011-05-18 12:17:53 +0000222 for (; !iter.done(); iter.next()) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000223 r.set(iter.rect());
224 canvas->drawRect(r, paint);
225 }
226}
227
228static void test_break(SkCanvas* canvas, const char text[], size_t length,
229 SkScalar x, SkScalar y, const SkPaint& paint,
reed@google.com9e39bb32011-05-18 12:17:53 +0000230 SkScalar clickX) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000231 SkPaint linePaint;
reed@google.com82065d62011-02-07 15:30:46 +0000232
reed@android.com8a1c16f2008-12-17 15:59:43 +0000233 linePaint.setAntiAlias(true);
reed@google.com82065d62011-02-07 15:30:46 +0000234
reed@android.com8a1c16f2008-12-17 15:59:43 +0000235 SkScalar measured;
reed@google.com82065d62011-02-07 15:30:46 +0000236
reed@google.com9e39bb32011-05-18 12:17:53 +0000237 if (paint.breakText(text, length, clickX - x, &measured,
238 SkPaint::kForward_TextBufferDirection)) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000239 linePaint.setColor(SK_ColorRED);
240 canvas->drawLine(x, y, x + measured, y, linePaint);
241 }
242
243 x += paint.measureText(text, length);
reed@google.com9e39bb32011-05-18 12:17:53 +0000244 if (paint.breakText(text, length, x - clickX, &measured,
245 SkPaint::kBackward_TextBufferDirection)) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000246 linePaint.setColor(SK_ColorBLUE);
247 canvas->drawLine(x - measured, y, x, y, linePaint);
248 }
249}
250
reed@android.com8a1c16f2008-12-17 15:59:43 +0000251static void DrawTheText(SkCanvas* canvas, const char text[], size_t length,
252 SkScalar x, SkScalar y, const SkPaint& paint,
reed@google.com9e39bb32011-05-18 12:17:53 +0000253 SkScalar clickX, SkMaskFilter* mf) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000254 SkPaint p(paint);
255
256#if 0
257 canvas->drawText(text, length, x, y, paint);
258#else
259 {
260 SkPoint pts[1000];
261 SkScalar xpos = x;
262 SkASSERT(length <= SK_ARRAY_COUNT(pts));
reed@google.com9e39bb32011-05-18 12:17:53 +0000263 for (size_t i = 0; i < length; i++) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000264 pts[i].set(xpos, y), xpos += paint.getTextSize();
reed@google.com9e39bb32011-05-18 12:17:53 +0000265 }
reed@android.com8a1c16f2008-12-17 15:59:43 +0000266 canvas->drawPosText(text, length, pts, paint);
267 }
268#endif
269
270 p.setSubpixelText(true);
271 x += SkIntToScalar(180);
272 canvas->drawText(text, length, x, y, p);
273
reed@android.com8a1c16f2008-12-17 15:59:43 +0000274#ifdef SK_DEBUG
reed@google.com9e39bb32011-05-18 12:17:53 +0000275 if (true) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000276 // p.setMaskFilter(mf);
277 p.setSubpixelText(false);
278 p.setLinearText(true);
279 x += SkIntToScalar(180);
280 canvas->drawText(text, length, x, y, p);
281 }
282#endif
283}
284
reed@google.com9e39bb32011-05-18 12:17:53 +0000285class TextSpeedView : public SampleView {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000286public:
reed@google.com9e39bb32011-05-18 12:17:53 +0000287 TextSpeedView() {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000288 fMF = makemf();
289
290 fHints = 0;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000291 fClickX = 0;
292
reed@google.com82065d62011-02-07 15:30:46 +0000293 test_breakText();
reed@android.com8a1c16f2008-12-17 15:59:43 +0000294 }
reed@google.com82065d62011-02-07 15:30:46 +0000295
reed@google.com9e39bb32011-05-18 12:17:53 +0000296 virtual ~TextSpeedView() {
reed@google.com82065d62011-02-07 15:30:46 +0000297 SkSafeUnref(fMF);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000298 }
299
300protected:
301 // overrides from SkEventSink
reed@google.com9e39bb32011-05-18 12:17:53 +0000302 virtual bool onQuery(SkEvent* evt) {
303 if (SampleCode::TitleQ(*evt)) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000304 SampleCode::TitleR(evt, "Text");
305 return true;
306 }
307 return this->INHERITED::onQuery(evt);
308 }
reed@google.com82065d62011-02-07 15:30:46 +0000309
reed@google.com9e39bb32011-05-18 12:17:53 +0000310 static void make_textstrip(SkBitmap* bm) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000311 bm->setConfig(SkBitmap::kRGB_565_Config, 200, 18);
312 bm->allocPixels();
313 bm->eraseColor(SK_ColorWHITE);
reed@google.com82065d62011-02-07 15:30:46 +0000314
reed@android.com8a1c16f2008-12-17 15:59:43 +0000315 SkCanvas canvas(*bm);
316 SkPaint paint;
317 const char* s = "Lorem ipsum dolor sit amet, consectetuer adipiscing elit";
reed@google.com82065d62011-02-07 15:30:46 +0000318
reed@android.com8a1c16f2008-12-17 15:59:43 +0000319 paint.setFlags(paint.getFlags() | SkPaint::kAntiAlias_Flag
320 | SkPaint::kDevKernText_Flag);
321 paint.setTextSize(SkIntToScalar(14));
322 canvas.drawText(s, strlen(s), SkIntToScalar(8), SkIntToScalar(14), paint);
323 }
reed@google.com82065d62011-02-07 15:30:46 +0000324
reed@google.com9e39bb32011-05-18 12:17:53 +0000325 static void fill_pts(SkPoint pts[], size_t n, SkRandom* rand) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000326 for (size_t i = 0; i < n; i++)
327 pts[i].set(rand->nextUScalar1() * 640, rand->nextUScalar1() * 480);
328 }
reed@google.com82065d62011-02-07 15:30:46 +0000329
reed@google.com9e39bb32011-05-18 12:17:53 +0000330 virtual void onDrawContent(SkCanvas* canvas) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000331 SkAutoCanvasRestore restore(canvas, false);
332 {
333 SkRect r;
334 r.set(0, 0, SkIntToScalar(1000), SkIntToScalar(20));
335 // canvas->saveLayer(&r, NULL, SkCanvas::kHasAlphaLayer_SaveFlag);
336 }
337
338 SkPaint paint;
339// const uint16_t glyphs[] = { 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19 };
340 int index = fHints % SK_ARRAY_COUNT(gHints);
341 index = 1;
342// const char* style = gHints[index].fName;
reed@google.com82065d62011-02-07 15:30:46 +0000343
reed@android.com8a1c16f2008-12-17 15:59:43 +0000344// canvas->translate(0, SkIntToScalar(50));
345
346 // canvas->drawText(style, strlen(style), SkIntToScalar(20), SkIntToScalar(20), paint);
347
reed@android.com04d86c62010-01-25 22:02:44 +0000348 SkSafeUnref(paint.setTypeface(SkTypeface::CreateFromFile("/skimages/samplefont.ttf")));
reed@android.com8a1c16f2008-12-17 15:59:43 +0000349 paint.setAntiAlias(true);
350 paint.setFlags(paint.getFlags() | gHints[index].fFlags);
reed@google.com82065d62011-02-07 15:30:46 +0000351
reed@android.com8a1c16f2008-12-17 15:59:43 +0000352 SkRect clip;
353 clip.set(SkIntToScalar(25), SkIntToScalar(34), SkIntToScalar(88), SkIntToScalar(155));
reed@google.com82065d62011-02-07 15:30:46 +0000354
reed@android.com8a1c16f2008-12-17 15:59:43 +0000355 const char* text = "Hamburgefons";
356 size_t length = strlen(text);
357
reed@google.com2f3dc9d2011-05-02 17:33:45 +0000358 SkScalar y = SkIntToScalar(0);
359 for (int i = 9; i <= 24; i++) {
360 paint.setTextSize(SkIntToScalar(i) /*+ (gRand.nextU() & 0xFFFF)*/);
reed@google.com9e39bb32011-05-18 12:17:53 +0000361 for (SkScalar dx = 0; dx <= SkIntToScalar(3)/4;
362 dx += SkIntToScalar(1) /* /4 */) {
reed@google.com2f3dc9d2011-05-02 17:33:45 +0000363 y += paint.getFontSpacing();
reed@google.com9e39bb32011-05-18 12:17:53 +0000364 DrawTheText(canvas, text, length, SkIntToScalar(20) + dx, y,
365 paint, fClickX, fMF);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000366 }
367 }
reed@google.com2f3dc9d2011-05-02 17:33:45 +0000368 if (gHints[index].fFlushCache) {
369// SkGraphics::SetFontCacheUsed(0);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000370 }
371 }
reed@google.com82065d62011-02-07 15:30:46 +0000372
reed@google.com9e39bb32011-05-18 12:17:53 +0000373 virtual SkView::Click* onFindClickHandler(SkScalar x, SkScalar y) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000374 fClickX = x;
375 this->inval(NULL);
376 return this->INHERITED::onFindClickHandler(x, y);
377 }
reed@google.com82065d62011-02-07 15:30:46 +0000378
reed@google.com9e39bb32011-05-18 12:17:53 +0000379 virtual bool onClick(Click* click) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000380 return this->INHERITED::onClick(click);
381 }
reed@google.com82065d62011-02-07 15:30:46 +0000382
reed@android.com8a1c16f2008-12-17 15:59:43 +0000383private:
384 int fHints;
385 SkScalar fClickX;
386 SkMaskFilter* fMF;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000387
reed@google.com9e39bb32011-05-18 12:17:53 +0000388 typedef SampleView INHERITED;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000389};
390
391//////////////////////////////////////////////////////////////////////////////
392
393static SkView* MyFactory() { return new TextSpeedView; }
394static SkViewRegister reg(MyFactory);
395