blob: 267653006ce7e84bf5441b74d1cec03d8493ce0a [file] [log] [blame]
reed@android.com8a1c16f2008-12-17 15:59:43 +00001#include "SampleCode.h"
2#include "SkView.h"
3#include "SkCanvas.h"
4#include "Sk64.h"
5#include "SkGradientShader.h"
6#include "SkGraphics.h"
7#include "SkImageDecoder.h"
8#include "SkKernel33MaskFilter.h"
9#include "SkPath.h"
10#include "SkRandom.h"
11#include "SkRegion.h"
12#include "SkShader.h"
13#include "SkUtils.h"
reed@android.com8a1c16f2008-12-17 15:59:43 +000014#include "SkColorPriv.h"
15#include "SkColorFilter.h"
16#include "SkTime.h"
17#include "SkTypeface.h"
18#include "SkXfermode.h"
19
20#include "SkStream.h"
21#include "SkXMLParser.h"
22
23static const int gKernel[3][3] = {
24// { -1, -2, -1 }, { -2, 12, -2 }, { -1, -2, -1 }
25 { 1, 2, 1 }, { 2, 64-12, 2 }, { 1, 2, 1 }
26};
27static const int gShift = 6;
28
29class ReduceNoise : public SkKernel33ProcMaskFilter {
30public:
31 ReduceNoise(int percent256) : SkKernel33ProcMaskFilter(percent256) {}
32 virtual uint8_t computeValue(uint8_t* const* srcRows)
33 {
34 int c = srcRows[1][1];
35 int min = 255, max = 0;
36 for (int i = 0; i < 3; i++)
37 for (int j = 0; j < 3; j++)
38 if (i != 1 || j != 1)
39 {
40 int v = srcRows[i][j];
41 if (max < v)
42 max = v;
43 if (min > v)
44 min = v;
45 }
46 if (c > max) c = max;
47 // if (c < min) c = min;
48 return c;
49 }
50 virtual Factory getFactory() { return Create; }
51private:
52 ReduceNoise(SkFlattenableReadBuffer& rb) : SkKernel33ProcMaskFilter(rb) {}
reed@google.com9e39bb32011-05-18 12:17:53 +000053 static SkFlattenable* Create(SkFlattenableReadBuffer& rb) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000054 return new ReduceNoise(rb);
55 }
56};
57
58class Darken : public SkKernel33ProcMaskFilter {
59public:
60 Darken(int percent256) : SkKernel33ProcMaskFilter(percent256) {}
61 virtual uint8_t computeValue(uint8_t* const* srcRows)
62 {
63 int c = srcRows[1][1];
64 float f = c / 255.f;
reed@google.com82065d62011-02-07 15:30:46 +000065
reed@google.com9e39bb32011-05-18 12:17:53 +000066 if (c >= 0) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000067 f = sqrtf(f);
reed@google.com9e39bb32011-05-18 12:17:53 +000068 } else {
reed@android.com8a1c16f2008-12-17 15:59:43 +000069 f *= f;
70 }
71 SkASSERT(f >= 0 && f <= 1);
72 return (int)(f * 255);
73 }
74 virtual Factory getFactory() { return Create; }
75private:
76 Darken(SkFlattenableReadBuffer& rb) : SkKernel33ProcMaskFilter(rb) {}
reed@google.com9e39bb32011-05-18 12:17:53 +000077 static SkFlattenable* Create(SkFlattenableReadBuffer& rb) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000078 return new Darken(rb);
79 }
80};
81
82static SkMaskFilter* makemf() { return new Darken(0x30); }
83
reed@google.com9e39bb32011-05-18 12:17:53 +000084static void test_breakText() {
reed@android.com8a1c16f2008-12-17 15:59:43 +000085 SkPaint paint;
86 const char* text = "sdfkljAKLDFJKEWkldfjlk#$%&sdfs.dsj";
87 size_t length = strlen(text);
88 SkScalar width = paint.measureText(text, length);
reed@google.com82065d62011-02-07 15:30:46 +000089
reed@android.com8a1c16f2008-12-17 15:59:43 +000090 SkScalar mm = 0;
91 SkScalar nn = 0;
reed@google.com9e39bb32011-05-18 12:17:53 +000092 for (SkScalar w = 0; w <= width; w += SK_Scalar1) {
reed@android.com8a1c16f2008-12-17 15:59:43 +000093 SkScalar m;
94 size_t n = paint.breakText(text, length, w, &m,
95 SkPaint::kBackward_TextBufferDirection);
reed@google.com82065d62011-02-07 15:30:46 +000096
reed@android.com8a1c16f2008-12-17 15:59:43 +000097 SkASSERT(n <= length);
98 SkASSERT(m <= width);
reed@google.com82065d62011-02-07 15:30:46 +000099
reed@google.com9e39bb32011-05-18 12:17:53 +0000100 if (n == 0) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000101 SkASSERT(m == 0);
reed@google.com9e39bb32011-05-18 12:17:53 +0000102 } else {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000103 // now assert that we're monotonic
reed@google.com9e39bb32011-05-18 12:17:53 +0000104 if (n == nn) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000105 SkASSERT(m == mm);
reed@google.com9e39bb32011-05-18 12:17:53 +0000106 } else {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000107 SkASSERT(n > nn);
108 SkASSERT(m > mm);
109 }
110 }
reed@google.com261b8e22011-04-14 17:53:24 +0000111 nn = SkIntToScalar(n);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000112 mm = m;
113 }
114
senorblanco@chromium.org64cc5792011-05-19 19:58:58 +0000115 SkDEBUGCODE(size_t length2 =) paint.breakText(text, length, width, &mm);
reed@google.com261b8e22011-04-14 17:53:24 +0000116 SkASSERT(length2 == length);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000117 SkASSERT(mm == width);
118}
119
120static SkRandom gRand;
121
122class SkPowerMode : public SkXfermode {
123public:
124 SkPowerMode(SkScalar exponent) { this->init(exponent); }
125
reed@google.com9e39bb32011-05-18 12:17:53 +0000126 virtual void xfer16(uint16_t dst[], const SkPMColor src[], int count,
127 const SkAlpha aa[]);
reed@android.com8a1c16f2008-12-17 15:59:43 +0000128
129 typedef SkFlattenable* (*Factory)(SkFlattenableReadBuffer&);
reed@google.com82065d62011-02-07 15:30:46 +0000130
reed@android.com8a1c16f2008-12-17 15:59:43 +0000131 // overrides for SkFlattenable
132 virtual Factory getFactory() { return Create; }
reed@google.com9e39bb32011-05-18 12:17:53 +0000133 virtual void flatten(SkFlattenableWriteBuffer& b) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000134 // this->INHERITED::flatten(b); How can we know if this is legal????
135 b.write32(SkScalarToFixed(fExp));
136 }
reed@google.com82065d62011-02-07 15:30:46 +0000137
reed@android.com8a1c16f2008-12-17 15:59:43 +0000138private:
139 SkScalar fExp; // user's value
140 uint8_t fTable[256]; // cache
141
142 void init(SkScalar exponent);
reed@google.com9e39bb32011-05-18 12:17:53 +0000143 SkPowerMode(SkFlattenableReadBuffer& b) : SkXfermode(b) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000144 // read the exponent
145 this->init(SkFixedToScalar(b.readS32()));
146 }
reed@google.com9e39bb32011-05-18 12:17:53 +0000147 static SkFlattenable* Create(SkFlattenableReadBuffer& b) {
reed@android.com8a1c16f2008-12-17 15:59:43 +0000148 return SkNEW_ARGS(SkPowerMode, (b));
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