• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 #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"
14 #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 
23 static 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 };
27 static const int gShift = 6;
28 
29 class ReduceNoise : public SkKernel33ProcMaskFilter {
30 public:
ReduceNoise(int percent256)31     ReduceNoise(int percent256) : SkKernel33ProcMaskFilter(percent256) {}
computeValue(uint8_t * const * srcRows)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     }
getFactory()50     virtual Factory getFactory() { return Create; }
51 private:
ReduceNoise(SkFlattenableReadBuffer & rb)52     ReduceNoise(SkFlattenableReadBuffer& rb) : SkKernel33ProcMaskFilter(rb) {}
Create(SkFlattenableReadBuffer & rb)53     static SkFlattenable* Create(SkFlattenableReadBuffer& rb) {
54         return new ReduceNoise(rb);
55     }
56 };
57 
58 class Darken : public SkKernel33ProcMaskFilter {
59 public:
Darken(int percent256)60     Darken(int percent256) : SkKernel33ProcMaskFilter(percent256) {}
computeValue(uint8_t * const * srcRows)61     virtual uint8_t computeValue(uint8_t* const* srcRows)
62     {
63         int c = srcRows[1][1];
64         float f = c / 255.f;
65 
66         if (c >= 0) {
67             f = sqrtf(f);
68         } else {
69             f *= f;
70         }
71         SkASSERT(f >= 0 && f <= 1);
72         return (int)(f * 255);
73     }
getFactory()74     virtual Factory getFactory() { return Create; }
75 private:
Darken(SkFlattenableReadBuffer & rb)76     Darken(SkFlattenableReadBuffer& rb) : SkKernel33ProcMaskFilter(rb) {}
Create(SkFlattenableReadBuffer & rb)77     static SkFlattenable* Create(SkFlattenableReadBuffer& rb) {
78         return new Darken(rb);
79     }
80 };
81 
makemf()82 static SkMaskFilter* makemf() { return new Darken(0x30); }
83 
test_breakText()84 static void test_breakText() {
85     SkPaint paint;
86     const char* text = "sdfkljAKLDFJKEWkldfjlk#$%&sdfs.dsj";
87     size_t length = strlen(text);
88     SkScalar width = paint.measureText(text, length);
89 
90     SkScalar mm = 0;
91     SkScalar nn = 0;
92     for (SkScalar w = 0; w <= width; w += SK_Scalar1) {
93         SkScalar m;
94         size_t n = paint.breakText(text, length, w, &m,
95                                     SkPaint::kBackward_TextBufferDirection);
96 
97         SkASSERT(n <= length);
98         SkASSERT(m <= width);
99 
100         if (n == 0) {
101             SkASSERT(m == 0);
102         } else {
103             // now assert that we're monotonic
104             if (n == nn) {
105                 SkASSERT(m == mm);
106             } else {
107                 SkASSERT(n > nn);
108                 SkASSERT(m > mm);
109             }
110         }
111         nn = SkIntToScalar(n);
112         mm = m;
113     }
114 
115     SkDEBUGCODE(size_t length2 =) paint.breakText(text, length, width, &mm);
116     SkASSERT(length2 == length);
117     SkASSERT(mm == width);
118 }
119 
120 static SkRandom gRand;
121 
122 class SkPowerMode : public SkXfermode {
123 public:
SkPowerMode(SkScalar exponent)124     SkPowerMode(SkScalar exponent) { this->init(exponent); }
125 
126     virtual void xfer16(uint16_t dst[], const SkPMColor src[], int count,
127                         const SkAlpha aa[]);
128 
129     typedef SkFlattenable* (*Factory)(SkFlattenableReadBuffer&);
130 
131     // overrides for SkFlattenable
getFactory()132     virtual Factory getFactory() { return Create; }
flatten(SkFlattenableWriteBuffer & b)133     virtual void flatten(SkFlattenableWriteBuffer& b) {
134     //    this->INHERITED::flatten(b);  How can we know if this is legal????
135         b.write32(SkScalarToFixed(fExp));
136     }
137 
138 private:
139     SkScalar fExp;          // user's value
140     uint8_t fTable[256];    // cache
141 
142     void init(SkScalar exponent);
SkPowerMode(SkFlattenableReadBuffer & b)143     SkPowerMode(SkFlattenableReadBuffer& b) : SkXfermode(b) {
144         // read the exponent
145         this->init(SkFixedToScalar(b.readS32()));
146     }
Create(SkFlattenableReadBuffer & b)147     static SkFlattenable* Create(SkFlattenableReadBuffer& b) {
148         return SkNEW_ARGS(SkPowerMode, (b));
149     }
150 
151     typedef SkXfermode INHERITED;
152 };
153 
init(SkScalar e)154 void SkPowerMode::init(SkScalar e) {
155     fExp = e;
156     float ee = SkScalarToFloat(e);
157 
158     printf("------ %g\n", ee);
159     for (int i = 0; i < 256; i++) {
160         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 
xfer16(uint16_t dst[],const SkPMColor src[],int count,const SkAlpha aa[])170 void SkPowerMode::xfer16(uint16_t dst[], const SkPMColor src[], int count,
171                          const SkAlpha aa[]) {
172     for (int i = 0; i < count; i++) {
173         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 
184 static 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 
count_char_points(const SkPaint & paint,char c)194 static int count_char_points(const SkPaint& paint, char c) {
195     SkPath  path;
196 
197     paint.getTextPath(&c, 1, 0, 0, &path);
198     return path.getPoints(NULL, 0);
199 }
200 
201 static int gOld, gNew, gCount;
202 
dump(int c,int oldc,int newc)203 static void dump(int c, int oldc, int newc) {
204     if (oldc != newc) {
205         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 
tab(int n)212 static void tab(int n) {
213 //    printf("[%d] ", n); return;
214     SkASSERT(n >= 0);
215     for (int i = 0; i < n; i++)
216         printf("    ");
217 }
218 
draw_rgn(const SkRegion & rgn,SkCanvas * canvas,const SkPaint & paint)219 static void draw_rgn(const SkRegion& rgn, SkCanvas* canvas, const SkPaint& paint) {
220     SkRect    r;
221     SkRegion::Iterator  iter(rgn);
222 
223     for (; !iter.done(); iter.next()) {
224         r.set(iter.rect());
225         canvas->drawRect(r, paint);
226     }
227 }
228 
test_break(SkCanvas * canvas,const char text[],size_t length,SkScalar x,SkScalar y,const SkPaint & paint,SkScalar clickX)229 static void test_break(SkCanvas* canvas, const char text[], size_t length,
230                         SkScalar x, SkScalar y, const SkPaint& paint,
231                         SkScalar clickX) {
232     SkPaint linePaint;
233 
234     linePaint.setAntiAlias(true);
235 
236     SkScalar measured;
237 
238     if (paint.breakText(text, length, clickX - x, &measured,
239                         SkPaint::kForward_TextBufferDirection)) {
240         linePaint.setColor(SK_ColorRED);
241         canvas->drawLine(x, y, x + measured, y, linePaint);
242     }
243 
244     x += paint.measureText(text, length);
245     if (paint.breakText(text, length, x - clickX, &measured,
246                         SkPaint::kBackward_TextBufferDirection)) {
247         linePaint.setColor(SK_ColorBLUE);
248         canvas->drawLine(x - measured, y, x, y, linePaint);
249     }
250 }
251 
DrawTheText(SkCanvas * canvas,const char text[],size_t length,SkScalar x,SkScalar y,const SkPaint & paint,SkScalar clickX,SkMaskFilter * mf)252 static void DrawTheText(SkCanvas* canvas, const char text[], size_t length,
253                         SkScalar x, SkScalar y, const SkPaint& paint,
254                         SkScalar clickX, SkMaskFilter* mf) {
255     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));
264         for (size_t i = 0; i < length; i++) {
265             pts[i].set(xpos, y), xpos += paint.getTextSize();
266         }
267         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 
275 #ifdef SK_DEBUG
276     if (true) {
277     //    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 
286 class TextSpeedView : public SampleView {
287 public:
TextSpeedView()288 	TextSpeedView() {
289         fMF = makemf();
290 
291         fHints = 0;
292         fClickX = 0;
293 
294         test_breakText();
295     }
296 
~TextSpeedView()297     virtual ~TextSpeedView() {
298         SkSafeUnref(fMF);
299     }
300 
301 protected:
302     // overrides from SkEventSink
onQuery(SkEvent * evt)303     virtual bool onQuery(SkEvent* evt) {
304         if (SampleCode::TitleQ(*evt)) {
305             SampleCode::TitleR(evt, "Text");
306             return true;
307         }
308         return this->INHERITED::onQuery(evt);
309     }
310 
make_textstrip(SkBitmap * bm)311     static void make_textstrip(SkBitmap* bm) {
312         bm->setConfig(SkBitmap::kRGB_565_Config, 200, 18);
313         bm->allocPixels();
314         bm->eraseColor(SK_ColorWHITE);
315 
316         SkCanvas    canvas(*bm);
317         SkPaint     paint;
318         const char* s = "Lorem ipsum dolor sit amet, consectetuer adipiscing elit";
319 
320         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     }
325 
fill_pts(SkPoint pts[],size_t n,SkRandom * rand)326     static void fill_pts(SkPoint pts[], size_t n, SkRandom* rand) {
327         for (size_t i = 0; i < n; i++)
328             pts[i].set(rand->nextUScalar1() * 640, rand->nextUScalar1() * 480);
329     }
330 
onDrawContent(SkCanvas * canvas)331     virtual void onDrawContent(SkCanvas* canvas) {
332         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;
344 
345 //        canvas->translate(0, SkIntToScalar(50));
346 
347   //      canvas->drawText(style, strlen(style), SkIntToScalar(20), SkIntToScalar(20), paint);
348 
349         SkSafeUnref(paint.setTypeface(SkTypeface::CreateFromFile("/skimages/samplefont.ttf")));
350         paint.setAntiAlias(true);
351         paint.setFlags(paint.getFlags() | gHints[index].fFlags);
352 
353         SkRect clip;
354         clip.set(SkIntToScalar(25), SkIntToScalar(34), SkIntToScalar(88), SkIntToScalar(155));
355 
356         const char* text = "Hamburgefons";
357         size_t length = strlen(text);
358 
359         SkScalar y = SkIntToScalar(0);
360         for (int i = 9; i <= 24; i++) {
361             paint.setTextSize(SkIntToScalar(i) /*+ (gRand.nextU() & 0xFFFF)*/);
362             for (SkScalar dx = 0; dx <= SkIntToScalar(3)/4;
363                                             dx += SkIntToScalar(1) /* /4 */) {
364                 y += paint.getFontSpacing();
365                 DrawTheText(canvas, text, length, SkIntToScalar(20) + dx, y,
366                             paint, fClickX, fMF);
367             }
368         }
369         if (gHints[index].fFlushCache) {
370 //                SkGraphics::SetFontCacheUsed(0);
371         }
372     }
373 
onFindClickHandler(SkScalar x,SkScalar y)374     virtual SkView::Click* onFindClickHandler(SkScalar x, SkScalar y) {
375         fClickX = x;
376         this->inval(NULL);
377         return this->INHERITED::onFindClickHandler(x, y);
378     }
379 
onClick(Click * click)380     virtual bool onClick(Click* click) {
381         return this->INHERITED::onClick(click);
382     }
383 
384 private:
385     int fHints;
386     SkScalar fClickX;
387     SkMaskFilter* fMF;
388 
389     typedef SampleView INHERITED;
390 };
391 
392 //////////////////////////////////////////////////////////////////////////////
393 
MyFactory()394 static SkView* MyFactory() { return new TextSpeedView; }
395 static SkViewRegister reg(MyFactory);
396 
397