blob: 7ce1abe65c464364a69715355d4e0299c29c2f8d [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
reed@android.com8a1c16f2008-12-17 15:59:43 +0000134 // overrides for SkFlattenable
reed@google.com9e39bb32011-05-18 12:17:53 +0000135 virtual void flatten(SkFlattenableWriteBuffer& b) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000136 // this->INHERITED::flatten(b); How can we know if this is legal????
137 b.write32(SkScalarToFixed(fExp));
138 }
djsollen@google.comba28d032012-03-26 17:57:35 +0000139 SK_DECLARE_PUBLIC_FLATTENABLE_DESERIALIZATION_PROCS(SkPowerMode)
reed@google.com82065d62011-02-07 15:30:46 +0000140
reed@android.com8a1c16f2008-12-17 15:59:43 +0000141private:
142 SkScalar fExp; // user's value
143 uint8_t fTable[256]; // cache
144
145 void init(SkScalar exponent);
reed@google.com9e39bb32011-05-18 12:17:53 +0000146 SkPowerMode(SkFlattenableReadBuffer& b) : SkXfermode(b) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000147 // read the exponent
148 this->init(SkFixedToScalar(b.readS32()));
149 }
reed@google.com82065d62011-02-07 15:30:46 +0000150
reed@android.com8a1c16f2008-12-17 15:59:43 +0000151 typedef SkXfermode INHERITED;
152};
153
reed@google.com9e39bb32011-05-18 12:17:53 +0000154void SkPowerMode::init(SkScalar e) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000155 fExp = e;
156 float ee = SkScalarToFloat(e);
reed@google.com82065d62011-02-07 15:30:46 +0000157
reed@android.com8a1c16f2008-12-17 15:59:43 +0000158 printf("------ %g\n", ee);
reed@google.com9e39bb32011-05-18 12:17:53 +0000159 for (int i = 0; i < 256; i++) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000160 float x = i / 255.f;
161 // printf(" %d %g", i, x);
162 x = powf(x, ee);
163 // printf(" %g", x);
164 int xx = SkScalarRound(SkFloatToScalar(x * 255));
165 // printf(" %d\n", xx);
166 fTable[i] = SkToU8(xx);
167 }
168}
169
reed@google.com9e39bb32011-05-18 12:17:53 +0000170void SkPowerMode::xfer16(uint16_t dst[], const SkPMColor src[], int count,
171 const SkAlpha aa[]) {
172 for (int i = 0; i < count; i++) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000173 SkPMColor c = src[i];
174 int r = SkGetPackedR32(c);
175 int g = SkGetPackedG32(c);
176 int b = SkGetPackedB32(c);
177 r = fTable[r];
178 g = fTable[g];
179 b = fTable[b];
180 dst[i] = SkPack888ToRGB16(r, g, b);
181 }
182}
183
184static const struct {
185 const char* fName;
186 uint32_t fFlags;
187 bool fFlushCache;
188} gHints[] = {
189 { "Linear", SkPaint::kLinearText_Flag, false },
190 { "Normal", 0, true },
191 { "Subpixel", SkPaint::kSubpixelText_Flag, true }
192};
193
reed@google.com9e39bb32011-05-18 12:17:53 +0000194static int count_char_points(const SkPaint& paint, char c) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000195 SkPath path;
reed@google.com82065d62011-02-07 15:30:46 +0000196
reed@android.com8a1c16f2008-12-17 15:59:43 +0000197 paint.getTextPath(&c, 1, 0, 0, &path);
198 return path.getPoints(NULL, 0);
199}
200
201static int gOld, gNew, gCount;
202
reed@google.com9e39bb32011-05-18 12:17:53 +0000203static void dump(int c, int oldc, int newc) {
204 if (oldc != newc) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000205 gOld += oldc;
206 gNew += newc;
207 gCount += 1;
208 printf("char %c: old = %3d, new = %3d, reduction %g%%\n", c, oldc, newc, 100. * (oldc - newc) / oldc);
209 }
210}
211
reed@google.com9e39bb32011-05-18 12:17:53 +0000212static void tab(int n) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000213// printf("[%d] ", n); return;
214 SkASSERT(n >= 0);
215 for (int i = 0; i < n; i++)
216 printf(" ");
217}
218
reed@google.com9e39bb32011-05-18 12:17:53 +0000219static void draw_rgn(const SkRegion& rgn, SkCanvas* canvas, const SkPaint& paint) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000220 SkRect r;
221 SkRegion::Iterator iter(rgn);
reed@google.com82065d62011-02-07 15:30:46 +0000222
reed@google.com9e39bb32011-05-18 12:17:53 +0000223 for (; !iter.done(); iter.next()) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000224 r.set(iter.rect());
225 canvas->drawRect(r, paint);
226 }
227}
228
229static void test_break(SkCanvas* canvas, const char text[], size_t length,
230 SkScalar x, SkScalar y, const SkPaint& paint,
reed@google.com9e39bb32011-05-18 12:17:53 +0000231 SkScalar clickX) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000232 SkPaint linePaint;
reed@google.com82065d62011-02-07 15:30:46 +0000233
reed@android.com8a1c16f2008-12-17 15:59:43 +0000234 linePaint.setAntiAlias(true);
reed@google.com82065d62011-02-07 15:30:46 +0000235
reed@android.com8a1c16f2008-12-17 15:59:43 +0000236 SkScalar measured;
reed@google.com82065d62011-02-07 15:30:46 +0000237
reed@google.com9e39bb32011-05-18 12:17:53 +0000238 if (paint.breakText(text, length, clickX - x, &measured,
239 SkPaint::kForward_TextBufferDirection)) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000240 linePaint.setColor(SK_ColorRED);
241 canvas->drawLine(x, y, x + measured, y, linePaint);
242 }
243
244 x += paint.measureText(text, length);
reed@google.com9e39bb32011-05-18 12:17:53 +0000245 if (paint.breakText(text, length, x - clickX, &measured,
246 SkPaint::kBackward_TextBufferDirection)) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000247 linePaint.setColor(SK_ColorBLUE);
248 canvas->drawLine(x - measured, y, x, y, linePaint);
249 }
250}
251
reed@android.com8a1c16f2008-12-17 15:59:43 +0000252static void DrawTheText(SkCanvas* canvas, const char text[], size_t length,
253 SkScalar x, SkScalar y, const SkPaint& paint,
reed@google.com9e39bb32011-05-18 12:17:53 +0000254 SkScalar clickX, SkMaskFilter* mf) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000255 SkPaint p(paint);
256
257#if 0
258 canvas->drawText(text, length, x, y, paint);
259#else
260 {
261 SkPoint pts[1000];
262 SkScalar xpos = x;
263 SkASSERT(length <= SK_ARRAY_COUNT(pts));
reed@google.com9e39bb32011-05-18 12:17:53 +0000264 for (size_t i = 0; i < length; i++) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000265 pts[i].set(xpos, y), xpos += paint.getTextSize();
reed@google.com9e39bb32011-05-18 12:17:53 +0000266 }
reed@android.com8a1c16f2008-12-17 15:59:43 +0000267 canvas->drawPosText(text, length, pts, paint);
268 }
269#endif
270
271 p.setSubpixelText(true);
272 x += SkIntToScalar(180);
273 canvas->drawText(text, length, x, y, p);
274
reed@android.com8a1c16f2008-12-17 15:59:43 +0000275#ifdef SK_DEBUG
reed@google.com9e39bb32011-05-18 12:17:53 +0000276 if (true) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000277 // p.setMaskFilter(mf);
278 p.setSubpixelText(false);
279 p.setLinearText(true);
280 x += SkIntToScalar(180);
281 canvas->drawText(text, length, x, y, p);
282 }
283#endif
284}
285
reed@google.com9e39bb32011-05-18 12:17:53 +0000286class TextSpeedView : public SampleView {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000287public:
reed@google.com9e39bb32011-05-18 12:17:53 +0000288 TextSpeedView() {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000289 fMF = makemf();
290
291 fHints = 0;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000292 fClickX = 0;
293
reed@google.com82065d62011-02-07 15:30:46 +0000294 test_breakText();
reed@android.com8a1c16f2008-12-17 15:59:43 +0000295 }
reed@google.com82065d62011-02-07 15:30:46 +0000296
reed@google.com9e39bb32011-05-18 12:17:53 +0000297 virtual ~TextSpeedView() {
reed@google.com82065d62011-02-07 15:30:46 +0000298 SkSafeUnref(fMF);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000299 }
300
301protected:
302 // overrides from SkEventSink
reed@google.com9e39bb32011-05-18 12:17:53 +0000303 virtual bool onQuery(SkEvent* evt) {
304 if (SampleCode::TitleQ(*evt)) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000305 SampleCode::TitleR(evt, "Text");
306 return true;
307 }
308 return this->INHERITED::onQuery(evt);
309 }
reed@google.com82065d62011-02-07 15:30:46 +0000310
reed@google.com9e39bb32011-05-18 12:17:53 +0000311 static void make_textstrip(SkBitmap* bm) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000312 bm->setConfig(SkBitmap::kRGB_565_Config, 200, 18);
313 bm->allocPixels();
314 bm->eraseColor(SK_ColorWHITE);
reed@google.com82065d62011-02-07 15:30:46 +0000315
reed@android.com8a1c16f2008-12-17 15:59:43 +0000316 SkCanvas canvas(*bm);
317 SkPaint paint;
318 const char* s = "Lorem ipsum dolor sit amet, consectetuer adipiscing elit";
reed@google.com82065d62011-02-07 15:30:46 +0000319
reed@android.com8a1c16f2008-12-17 15:59:43 +0000320 paint.setFlags(paint.getFlags() | SkPaint::kAntiAlias_Flag
321 | SkPaint::kDevKernText_Flag);
322 paint.setTextSize(SkIntToScalar(14));
323 canvas.drawText(s, strlen(s), SkIntToScalar(8), SkIntToScalar(14), paint);
324 }
reed@google.com82065d62011-02-07 15:30:46 +0000325
reed@google.com9e39bb32011-05-18 12:17:53 +0000326 static void fill_pts(SkPoint pts[], size_t n, SkRandom* rand) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000327 for (size_t i = 0; i < n; i++)
328 pts[i].set(rand->nextUScalar1() * 640, rand->nextUScalar1() * 480);
329 }
reed@google.com82065d62011-02-07 15:30:46 +0000330
reed@google.com9e39bb32011-05-18 12:17:53 +0000331 virtual void onDrawContent(SkCanvas* canvas) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000332 SkAutoCanvasRestore restore(canvas, false);
333 {
334 SkRect r;
335 r.set(0, 0, SkIntToScalar(1000), SkIntToScalar(20));
336 // canvas->saveLayer(&r, NULL, SkCanvas::kHasAlphaLayer_SaveFlag);
337 }
338
339 SkPaint paint;
340// const uint16_t glyphs[] = { 0,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19 };
341 int index = fHints % SK_ARRAY_COUNT(gHints);
342 index = 1;
343// const char* style = gHints[index].fName;
reed@google.com82065d62011-02-07 15:30:46 +0000344
reed@android.com8a1c16f2008-12-17 15:59:43 +0000345// canvas->translate(0, SkIntToScalar(50));
346
347 // canvas->drawText(style, strlen(style), SkIntToScalar(20), SkIntToScalar(20), paint);
348
reed@android.com04d86c62010-01-25 22:02:44 +0000349 SkSafeUnref(paint.setTypeface(SkTypeface::CreateFromFile("/skimages/samplefont.ttf")));
reed@android.com8a1c16f2008-12-17 15:59:43 +0000350 paint.setAntiAlias(true);
351 paint.setFlags(paint.getFlags() | gHints[index].fFlags);
reed@google.com82065d62011-02-07 15:30:46 +0000352
reed@android.com8a1c16f2008-12-17 15:59:43 +0000353 SkRect clip;
354 clip.set(SkIntToScalar(25), SkIntToScalar(34), SkIntToScalar(88), SkIntToScalar(155));
reed@google.com82065d62011-02-07 15:30:46 +0000355
reed@android.com8a1c16f2008-12-17 15:59:43 +0000356 const char* text = "Hamburgefons";
357 size_t length = strlen(text);
358
reed@google.com2f3dc9d2011-05-02 17:33:45 +0000359 SkScalar y = SkIntToScalar(0);
360 for (int i = 9; i <= 24; i++) {
361 paint.setTextSize(SkIntToScalar(i) /*+ (gRand.nextU() & 0xFFFF)*/);
reed@google.com9e39bb32011-05-18 12:17:53 +0000362 for (SkScalar dx = 0; dx <= SkIntToScalar(3)/4;
363 dx += SkIntToScalar(1) /* /4 */) {
reed@google.com2f3dc9d2011-05-02 17:33:45 +0000364 y += paint.getFontSpacing();
reed@google.com9e39bb32011-05-18 12:17:53 +0000365 DrawTheText(canvas, text, length, SkIntToScalar(20) + dx, y,
366 paint, fClickX, fMF);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000367 }
368 }
reed@google.com2f3dc9d2011-05-02 17:33:45 +0000369 if (gHints[index].fFlushCache) {
370// SkGraphics::SetFontCacheUsed(0);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000371 }
372 }
reed@google.com82065d62011-02-07 15:30:46 +0000373
reed@google.com9e39bb32011-05-18 12:17:53 +0000374 virtual SkView::Click* onFindClickHandler(SkScalar x, SkScalar y) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000375 fClickX = x;
376 this->inval(NULL);
377 return this->INHERITED::onFindClickHandler(x, y);
378 }
reed@google.com82065d62011-02-07 15:30:46 +0000379
reed@google.com9e39bb32011-05-18 12:17:53 +0000380 virtual bool onClick(Click* click) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000381 return this->INHERITED::onClick(click);
382 }
reed@google.com82065d62011-02-07 15:30:46 +0000383
reed@android.com8a1c16f2008-12-17 15:59:43 +0000384private:
385 int fHints;
386 SkScalar fClickX;
387 SkMaskFilter* fMF;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000388
reed@google.com9e39bb32011-05-18 12:17:53 +0000389 typedef SampleView INHERITED;
reed@android.com8a1c16f2008-12-17 15:59:43 +0000390};
391
392//////////////////////////////////////////////////////////////////////////////
393
394static SkView* MyFactory() { return new TextSpeedView; }
395static SkViewRegister reg(MyFactory);
396