blob: 6a0eba8f977996ed65881762e4d72f4615a7ecc3 [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 }
57 virtual Factory getFactory() { return Create; }
58private:
59 ReduceNoise(SkFlattenableReadBuffer& rb) : SkKernel33ProcMaskFilter(rb) {}
reed@google.com9e39bb32011-05-18 12:17:53 +000060 static SkFlattenable* Create(SkFlattenableReadBuffer& rb) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000061 return new ReduceNoise(rb);
62 }
63};
64
65class Darken : public SkKernel33ProcMaskFilter {
66public:
67 Darken(int percent256) : SkKernel33ProcMaskFilter(percent256) {}
68 virtual uint8_t computeValue(uint8_t* const* srcRows)
69 {
70 int c = srcRows[1][1];
71 float f = c / 255.f;
reed@google.com82065d62011-02-07 15:30:46 +000072
reed@google.com9e39bb32011-05-18 12:17:53 +000073 if (c >= 0) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000074 f = sqrtf(f);
reed@google.com9e39bb32011-05-18 12:17:53 +000075 } else {
reed@android.com8a1c16f2008-12-17 15:59:43 +000076 f *= f;
77 }
78 SkASSERT(f >= 0 && f <= 1);
79 return (int)(f * 255);
80 }
81 virtual Factory getFactory() { return Create; }
82private:
83 Darken(SkFlattenableReadBuffer& rb) : SkKernel33ProcMaskFilter(rb) {}
reed@google.com9e39bb32011-05-18 12:17:53 +000084 static SkFlattenable* Create(SkFlattenableReadBuffer& rb) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000085 return new Darken(rb);
86 }
87};
88
89static SkMaskFilter* makemf() { return new Darken(0x30); }
90
reed@google.com9e39bb32011-05-18 12:17:53 +000091static void test_breakText() {
reed@android.com8a1c16f2008-12-17 15:59:43 +000092 SkPaint paint;
93 const char* text = "sdfkljAKLDFJKEWkldfjlk#$%&sdfs.dsj";
94 size_t length = strlen(text);
95 SkScalar width = paint.measureText(text, length);
reed@google.com82065d62011-02-07 15:30:46 +000096
reed@android.com8a1c16f2008-12-17 15:59:43 +000097 SkScalar mm = 0;
98 SkScalar nn = 0;
reed@google.com9e39bb32011-05-18 12:17:53 +000099 for (SkScalar w = 0; w <= width; w += SK_Scalar1) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000100 SkScalar m;
101 size_t n = paint.breakText(text, length, w, &m,
102 SkPaint::kBackward_TextBufferDirection);
reed@google.com82065d62011-02-07 15:30:46 +0000103
reed@android.com8a1c16f2008-12-17 15:59:43 +0000104 SkASSERT(n <= length);
105 SkASSERT(m <= width);
reed@google.com82065d62011-02-07 15:30:46 +0000106
reed@google.com9e39bb32011-05-18 12:17:53 +0000107 if (n == 0) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000108 SkASSERT(m == 0);
reed@google.com9e39bb32011-05-18 12:17:53 +0000109 } else {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000110 // now assert that we're monotonic
reed@google.com9e39bb32011-05-18 12:17:53 +0000111 if (n == nn) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000112 SkASSERT(m == mm);
reed@google.com9e39bb32011-05-18 12:17:53 +0000113 } else {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000114 SkASSERT(n > nn);
115 SkASSERT(m > mm);
116 }
117 }
reed@google.com261b8e22011-04-14 17:53:24 +0000118 nn = SkIntToScalar(n);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000119 mm = m;
120 }
121
senorblanco@chromium.org64cc5792011-05-19 19:58:58 +0000122 SkDEBUGCODE(size_t length2 =) paint.breakText(text, length, width, &mm);
reed@google.com261b8e22011-04-14 17:53:24 +0000123 SkASSERT(length2 == length);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000124 SkASSERT(mm == width);
125}
126
127static SkRandom gRand;
128
129class SkPowerMode : public SkXfermode {
130public:
131 SkPowerMode(SkScalar exponent) { this->init(exponent); }
132
reed@google.com9e39bb32011-05-18 12:17:53 +0000133 virtual void xfer16(uint16_t dst[], const SkPMColor src[], int count,
134 const SkAlpha aa[]);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000135
136 typedef SkFlattenable* (*Factory)(SkFlattenableReadBuffer&);
reed@google.com82065d62011-02-07 15:30:46 +0000137
reed@android.com8a1c16f2008-12-17 15:59:43 +0000138 // overrides for SkFlattenable
139 virtual Factory getFactory() { return Create; }
reed@google.com9e39bb32011-05-18 12:17:53 +0000140 virtual void flatten(SkFlattenableWriteBuffer& b) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000141 // this->INHERITED::flatten(b); How can we know if this is legal????
142 b.write32(SkScalarToFixed(fExp));
143 }
reed@google.com82065d62011-02-07 15:30:46 +0000144
reed@android.com8a1c16f2008-12-17 15:59:43 +0000145private:
146 SkScalar fExp; // user's value
147 uint8_t fTable[256]; // cache
148
149 void init(SkScalar exponent);
reed@google.com9e39bb32011-05-18 12:17:53 +0000150 SkPowerMode(SkFlattenableReadBuffer& b) : SkXfermode(b) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000151 // read the exponent
152 this->init(SkFixedToScalar(b.readS32()));
153 }
reed@google.com9e39bb32011-05-18 12:17:53 +0000154 static SkFlattenable* Create(SkFlattenableReadBuffer& b) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000155 return SkNEW_ARGS(SkPowerMode, (b));
156 }
reed@google.com82065d62011-02-07 15:30:46 +0000157
reed@android.com8a1c16f2008-12-17 15:59:43 +0000158 typedef SkXfermode INHERITED;
159};
160
reed@google.com9e39bb32011-05-18 12:17:53 +0000161void SkPowerMode::init(SkScalar e) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000162 fExp = e;
163 float ee = SkScalarToFloat(e);
reed@google.com82065d62011-02-07 15:30:46 +0000164
reed@android.com8a1c16f2008-12-17 15:59:43 +0000165 printf("------ %g\n", ee);
reed@google.com9e39bb32011-05-18 12:17:53 +0000166 for (int i = 0; i < 256; i++) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000167 float x = i / 255.f;
168 // printf(" %d %g", i, x);
169 x = powf(x, ee);
170 // printf(" %g", x);
171 int xx = SkScalarRound(SkFloatToScalar(x * 255));
172 // printf(" %d\n", xx);
173 fTable[i] = SkToU8(xx);
174 }
175}
176
reed@google.com9e39bb32011-05-18 12:17:53 +0000177void SkPowerMode::xfer16(uint16_t dst[], const SkPMColor src[], int count,
178 const SkAlpha aa[]) {
179 for (int i = 0; i < count; i++) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000180 SkPMColor c = src[i];
181 int r = SkGetPackedR32(c);
182 int g = SkGetPackedG32(c);
183 int b = SkGetPackedB32(c);
184 r = fTable[r];
185 g = fTable[g];
186 b = fTable[b];
187 dst[i] = SkPack888ToRGB16(r, g, b);
188 }
189}
190
191static const struct {
192 const char* fName;
193 uint32_t fFlags;
194 bool fFlushCache;
195} gHints[] = {
196 { "Linear", SkPaint::kLinearText_Flag, false },
197 { "Normal", 0, true },
198 { "Subpixel", SkPaint::kSubpixelText_Flag, true }
199};
200
reed@google.com9e39bb32011-05-18 12:17:53 +0000201static int count_char_points(const SkPaint& paint, char c) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000202 SkPath path;
reed@google.com82065d62011-02-07 15:30:46 +0000203
reed@android.com8a1c16f2008-12-17 15:59:43 +0000204 paint.getTextPath(&c, 1, 0, 0, &path);
205 return path.getPoints(NULL, 0);
206}
207
208static int gOld, gNew, gCount;
209
reed@google.com9e39bb32011-05-18 12:17:53 +0000210static void dump(int c, int oldc, int newc) {
211 if (oldc != newc) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000212 gOld += oldc;
213 gNew += newc;
214 gCount += 1;
215 printf("char %c: old = %3d, new = %3d, reduction %g%%\n", c, oldc, newc, 100. * (oldc - newc) / oldc);
216 }
217}
218
reed@google.com9e39bb32011-05-18 12:17:53 +0000219static void tab(int n) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000220// printf("[%d] ", n); return;
221 SkASSERT(n >= 0);
222 for (int i = 0; i < n; i++)
223 printf(" ");
224}
225
reed@google.com9e39bb32011-05-18 12:17:53 +0000226static void draw_rgn(const SkRegion& rgn, SkCanvas* canvas, const SkPaint& paint) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000227 SkRect r;
228 SkRegion::Iterator iter(rgn);
reed@google.com82065d62011-02-07 15:30:46 +0000229
reed@google.com9e39bb32011-05-18 12:17:53 +0000230 for (; !iter.done(); iter.next()) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000231 r.set(iter.rect());
232 canvas->drawRect(r, paint);
233 }
234}
235
236static void test_break(SkCanvas* canvas, const char text[], size_t length,
237 SkScalar x, SkScalar y, const SkPaint& paint,
reed@google.com9e39bb32011-05-18 12:17:53 +0000238 SkScalar clickX) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000239 SkPaint linePaint;
reed@google.com82065d62011-02-07 15:30:46 +0000240
reed@android.com8a1c16f2008-12-17 15:59:43 +0000241 linePaint.setAntiAlias(true);
reed@google.com82065d62011-02-07 15:30:46 +0000242
reed@android.com8a1c16f2008-12-17 15:59:43 +0000243 SkScalar measured;
reed@google.com82065d62011-02-07 15:30:46 +0000244
reed@google.com9e39bb32011-05-18 12:17:53 +0000245 if (paint.breakText(text, length, clickX - x, &measured,
246 SkPaint::kForward_TextBufferDirection)) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000247 linePaint.setColor(SK_ColorRED);
248 canvas->drawLine(x, y, x + measured, y, linePaint);
249 }
250
251 x += paint.measureText(text, length);
reed@google.com9e39bb32011-05-18 12:17:53 +0000252 if (paint.breakText(text, length, x - clickX, &measured,
253 SkPaint::kBackward_TextBufferDirection)) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000254 linePaint.setColor(SK_ColorBLUE);
255 canvas->drawLine(x - measured, y, x, y, linePaint);
256 }
257}
258
reed@android.com8a1c16f2008-12-17 15:59:43 +0000259static void DrawTheText(SkCanvas* canvas, const char text[], size_t length,
260 SkScalar x, SkScalar y, const SkPaint& paint,
reed@google.com9e39bb32011-05-18 12:17:53 +0000261 SkScalar clickX, SkMaskFilter* mf) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000262 SkPaint p(paint);
263
264#if 0
265 canvas->drawText(text, length, x, y, paint);
266#else
267 {
268 SkPoint pts[1000];
269 SkScalar xpos = x;
270 SkASSERT(length <= SK_ARRAY_COUNT(pts));
reed@google.com9e39bb32011-05-18 12:17:53 +0000271 for (size_t i = 0; i < length; i++) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000272 pts[i].set(xpos, y), xpos += paint.getTextSize();
reed@google.com9e39bb32011-05-18 12:17:53 +0000273 }
reed@android.com8a1c16f2008-12-17 15:59:43 +0000274 canvas->drawPosText(text, length, pts, paint);
275 }
276#endif
277
278 p.setSubpixelText(true);
279 x += SkIntToScalar(180);
280 canvas->drawText(text, length, x, y, p);
281
reed@android.com8a1c16f2008-12-17 15:59:43 +0000282#ifdef SK_DEBUG
reed@google.com9e39bb32011-05-18 12:17:53 +0000283 if (true) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000284 // p.setMaskFilter(mf);
285 p.setSubpixelText(false);
286 p.setLinearText(true);
287 x += SkIntToScalar(180);
288 canvas->drawText(text, length, x, y, p);
289 }
290#endif
291}
292
reed@google.com9e39bb32011-05-18 12:17:53 +0000293class TextSpeedView : public SampleView {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000294public:
reed@google.com9e39bb32011-05-18 12:17:53 +0000295 TextSpeedView() {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000296 fMF = makemf();
297
298 fHints = 0;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000299 fClickX = 0;
300
reed@google.com82065d62011-02-07 15:30:46 +0000301 test_breakText();
reed@android.com8a1c16f2008-12-17 15:59:43 +0000302 }
reed@google.com82065d62011-02-07 15:30:46 +0000303
reed@google.com9e39bb32011-05-18 12:17:53 +0000304 virtual ~TextSpeedView() {
reed@google.com82065d62011-02-07 15:30:46 +0000305 SkSafeUnref(fMF);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000306 }
307
308protected:
309 // overrides from SkEventSink
reed@google.com9e39bb32011-05-18 12:17:53 +0000310 virtual bool onQuery(SkEvent* evt) {
311 if (SampleCode::TitleQ(*evt)) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000312 SampleCode::TitleR(evt, "Text");
313 return true;
314 }
315 return this->INHERITED::onQuery(evt);
316 }
reed@google.com82065d62011-02-07 15:30:46 +0000317
reed@google.com9e39bb32011-05-18 12:17:53 +0000318 static void make_textstrip(SkBitmap* bm) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000319 bm->setConfig(SkBitmap::kRGB_565_Config, 200, 18);
320 bm->allocPixels();
321 bm->eraseColor(SK_ColorWHITE);
reed@google.com82065d62011-02-07 15:30:46 +0000322
reed@android.com8a1c16f2008-12-17 15:59:43 +0000323 SkCanvas canvas(*bm);
324 SkPaint paint;
325 const char* s = "Lorem ipsum dolor sit amet, consectetuer adipiscing elit";
reed@google.com82065d62011-02-07 15:30:46 +0000326
reed@android.com8a1c16f2008-12-17 15:59:43 +0000327 paint.setFlags(paint.getFlags() | SkPaint::kAntiAlias_Flag
328 | SkPaint::kDevKernText_Flag);
329 paint.setTextSize(SkIntToScalar(14));
330 canvas.drawText(s, strlen(s), SkIntToScalar(8), SkIntToScalar(14), paint);
331 }
reed@google.com82065d62011-02-07 15:30:46 +0000332
reed@google.com9e39bb32011-05-18 12:17:53 +0000333 static void fill_pts(SkPoint pts[], size_t n, SkRandom* rand) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000334 for (size_t i = 0; i < n; i++)
335 pts[i].set(rand->nextUScalar1() * 640, rand->nextUScalar1() * 480);
336 }
reed@google.com82065d62011-02-07 15:30:46 +0000337
reed@google.com9e39bb32011-05-18 12:17:53 +0000338 virtual void onDrawContent(SkCanvas* canvas) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000339 SkAutoCanvasRestore restore(canvas, false);
340 {
341 SkRect r;
342 r.set(0, 0, SkIntToScalar(1000), SkIntToScalar(20));
343 // canvas->saveLayer(&r, NULL, SkCanvas::kHasAlphaLayer_SaveFlag);
344 }
345
346 SkPaint paint;
347// const uint16_t glyphs[] = { 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19 };
348 int index = fHints % SK_ARRAY_COUNT(gHints);
349 index = 1;
350// const char* style = gHints[index].fName;
reed@google.com82065d62011-02-07 15:30:46 +0000351
reed@android.com8a1c16f2008-12-17 15:59:43 +0000352// canvas->translate(0, SkIntToScalar(50));
353
354 // canvas->drawText(style, strlen(style), SkIntToScalar(20), SkIntToScalar(20), paint);
355
reed@android.com04d86c62010-01-25 22:02:44 +0000356 SkSafeUnref(paint.setTypeface(SkTypeface::CreateFromFile("/skimages/samplefont.ttf")));
reed@android.com8a1c16f2008-12-17 15:59:43 +0000357 paint.setAntiAlias(true);
358 paint.setFlags(paint.getFlags() | gHints[index].fFlags);
reed@google.com82065d62011-02-07 15:30:46 +0000359
reed@android.com8a1c16f2008-12-17 15:59:43 +0000360 SkRect clip;
361 clip.set(SkIntToScalar(25), SkIntToScalar(34), SkIntToScalar(88), SkIntToScalar(155));
reed@google.com82065d62011-02-07 15:30:46 +0000362
reed@android.com8a1c16f2008-12-17 15:59:43 +0000363 const char* text = "Hamburgefons";
364 size_t length = strlen(text);
365
reed@google.com2f3dc9d2011-05-02 17:33:45 +0000366 SkScalar y = SkIntToScalar(0);
367 for (int i = 9; i <= 24; i++) {
368 paint.setTextSize(SkIntToScalar(i) /*+ (gRand.nextU() & 0xFFFF)*/);
reed@google.com9e39bb32011-05-18 12:17:53 +0000369 for (SkScalar dx = 0; dx <= SkIntToScalar(3)/4;
370 dx += SkIntToScalar(1) /* /4 */) {
reed@google.com2f3dc9d2011-05-02 17:33:45 +0000371 y += paint.getFontSpacing();
reed@google.com9e39bb32011-05-18 12:17:53 +0000372 DrawTheText(canvas, text, length, SkIntToScalar(20) + dx, y,
373 paint, fClickX, fMF);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000374 }
375 }
reed@google.com2f3dc9d2011-05-02 17:33:45 +0000376 if (gHints[index].fFlushCache) {
377// SkGraphics::SetFontCacheUsed(0);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000378 }
379 }
reed@google.com82065d62011-02-07 15:30:46 +0000380
reed@google.com9e39bb32011-05-18 12:17:53 +0000381 virtual SkView::Click* onFindClickHandler(SkScalar x, SkScalar y) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000382 fClickX = x;
383 this->inval(NULL);
384 return this->INHERITED::onFindClickHandler(x, y);
385 }
reed@google.com82065d62011-02-07 15:30:46 +0000386
reed@google.com9e39bb32011-05-18 12:17:53 +0000387 virtual bool onClick(Click* click) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000388 return this->INHERITED::onClick(click);
389 }
reed@google.com82065d62011-02-07 15:30:46 +0000390
reed@android.com8a1c16f2008-12-17 15:59:43 +0000391private:
392 int fHints;
393 SkScalar fClickX;
394 SkMaskFilter* fMF;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000395
reed@google.com9e39bb32011-05-18 12:17:53 +0000396 typedef SampleView INHERITED;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000397};
398
399//////////////////////////////////////////////////////////////////////////////
400
401static SkView* MyFactory() { return new TextSpeedView; }
402static SkViewRegister reg(MyFactory);
403