| #include "SampleCode.h" |
| #include "SkView.h" |
| #include "SkCanvas.h" |
| #include "SkGradientShader.h" |
| #include "SkGraphics.h" |
| #include "SkImageDecoder.h" |
| #include "SkPackBits.h" |
| #include "SkPath.h" |
| #include "SkPathMeasure.h" |
| #include "SkRandom.h" |
| #include "SkRegion.h" |
| #include "SkShader.h" |
| #include "SkUtils.h" |
| #include "SkColorPriv.h" |
| #include "SkColorFilter.h" |
| #include "SkTypeface.h" |
| #include "SkAvoidXfermode.h" |
| |
| #define REPEAT_COUNT 1 |
| |
| static const char gText[] = "Hamburgefons"; |
| |
| static bool gDevKern; |
| |
| static void rand_text(char text[], SkRandom& rand, size_t count) { |
| for (size_t i = 0; i < count; i++) { |
| text[i] = rand.nextU() & 0x7F; |
| } |
| } |
| |
| static SkScalar sum_widths(const SkScalar widths[], int count) { |
| SkScalar w = 0; |
| for (int i = 0; i < count; i++) { |
| w += widths[i]; |
| } |
| return w; |
| } |
| |
| static void test_measure(const SkPaint& paint) { |
| char text[256]; |
| SkScalar widths[256]; |
| SkRect rects[256]; |
| SkRect bounds; |
| int count = 256; |
| |
| SkRandom rand; |
| |
| for (int i = 0; i < 100; i++) { |
| rand_text(text, rand, 256); |
| paint.getTextWidths(text, count, widths, NULL); |
| SkScalar tw0 = sum_widths(widths, count); |
| paint.getTextWidths(text, count, widths, rects); |
| SkScalar tw1 = sum_widths(widths, count); |
| SkASSERT(tw0 == tw1); |
| |
| SkScalar w0 = paint.measureText(text, count, NULL); |
| SkScalar w1 = paint.measureText(text, count, &bounds); |
| SkASSERT(w0 == w1); |
| SkASSERT(w0 == tw0); |
| |
| SkRect r = rects[0]; |
| SkScalar x = 0; |
| for (int j = 1; j < count; j++) { |
| x += widths[j-1]; |
| rects[j].offset(x, 0); |
| r.join(rects[j]); |
| } |
| SkASSERT(r == bounds); |
| |
| if (r != bounds) { |
| printf("flags=%x i=%d [%g %g %g %g] [%g %g %g %g]\n", |
| paint.getFlags(), i, |
| SkScalarToFloat(r.fLeft), |
| SkScalarToFloat(r.fTop), |
| SkScalarToFloat(r.fRight), |
| SkScalarToFloat(r.fBottom), |
| SkScalarToFloat(bounds.fLeft), |
| SkScalarToFloat(bounds.fTop), |
| SkScalarToFloat(bounds.fRight), |
| SkScalarToFloat(bounds.fBottom)); |
| } |
| } |
| } |
| |
| static void test_measure() { |
| SkPaint paint; |
| |
| for (int i = 0; i <= SkPaint::kAllFlags; i++) { |
| paint.setFlags(i); |
| test_measure(paint); |
| } |
| } |
| |
| ////////////////////////////////////////////////////////////////////////////// |
| |
| static void test_textBounds(SkCanvas* canvas) { |
| // canvas->scale(SK_Scalar1/2, SK_Scalar1/2); |
| |
| // canvas->rotate(SkIntToScalar(30)); |
| |
| gDevKern = !gDevKern; |
| |
| SkScalar x = SkIntToScalar(50); |
| SkScalar y = SkIntToScalar(150); |
| SkScalar w[100]; |
| SkRect r[100], bounds; |
| |
| SkPaint paint; |
| paint.setTextSize(SkIntToScalar(64)); |
| paint.setAntiAlias(true); |
| paint.setDevKernText(gDevKern); |
| |
| (void)paint.measureText(gText, strlen(gText), &bounds, NULL); |
| paint.setColor(SK_ColorGREEN); |
| bounds.offset(x, y); |
| canvas->drawRect(bounds, paint); |
| |
| int count = paint.getTextWidths(gText, strlen(gText), w, r); |
| |
| paint.setColor(SK_ColorRED); |
| for (int i = 0; i < count; i++) { |
| r[i].offset(x, y); |
| canvas->drawRect(r[i], paint); |
| x += w[i]; |
| } |
| x = SkIntToScalar(50); |
| paint.setColor(gDevKern ? SK_ColorDKGRAY : SK_ColorBLACK); |
| canvas->drawText(gText, strlen(gText), x, y, paint); |
| } |
| |
| static void create_src(SkBitmap* bitmap, SkBitmap::Config config) { |
| bitmap->setConfig(config, 100, 100); |
| bitmap->allocPixels(); |
| bitmap->eraseColor(0); |
| |
| SkCanvas canvas(*bitmap); |
| SkPaint paint; |
| |
| paint.setAntiAlias(true); |
| canvas.drawCircle(SkIntToScalar(50), SkIntToScalar(50), |
| SkIntToScalar(50), paint); |
| } |
| |
| static void blur(SkBitmap* dst, const SkBitmap& src, SkScalar radius) { |
| *dst = src; |
| } |
| |
| static void test_bitmap_blur(SkCanvas* canvas) { |
| SkBitmap src, dst; |
| |
| create_src(&src, SkBitmap::kARGB_8888_Config); |
| blur(&dst, src, SkIntToScalar(4)); |
| |
| SkPaint paint; |
| |
| paint.setColor(SK_ColorRED); |
| |
| canvas->drawBitmap(dst, SkIntToScalar(30), SkIntToScalar(60), &paint); |
| } |
| |
| static SkScalar getpathlen(const SkPath& path) { |
| SkPathMeasure meas(path, false); |
| return meas.getLength(); |
| } |
| |
| static void test_textpathmatrix(SkCanvas* canvas) { |
| SkPaint paint; |
| SkPath path; |
| SkMatrix matrix; |
| |
| path.moveTo(SkIntToScalar(200), SkIntToScalar(300)); |
| path.quadTo(SkIntToScalar(400), SkIntToScalar(100), |
| SkIntToScalar(600), SkIntToScalar(300)); |
| |
| paint.setAntiAlias(true); |
| |
| paint.setStyle(SkPaint::kStroke_Style); |
| canvas->drawPath(path, paint); |
| paint.setStyle(SkPaint::kFill_Style); |
| paint.setTextSize(SkIntToScalar(48)); |
| paint.setTextAlign(SkPaint::kRight_Align); |
| |
| const char* text = "Android"; |
| size_t len = strlen(text); |
| SkScalar pathLen = getpathlen(path); |
| |
| canvas->drawTextOnPath(text, len, path, NULL, paint); |
| |
| paint.setColor(SK_ColorRED); |
| matrix.setScale(-SK_Scalar1, SK_Scalar1); |
| matrix.postTranslate(pathLen, 0); |
| canvas->drawTextOnPath(text, len, path, &matrix, paint); |
| |
| paint.setColor(SK_ColorBLUE); |
| matrix.setScale(SK_Scalar1, -SK_Scalar1); |
| canvas->drawTextOnPath(text, len, path, &matrix, paint); |
| |
| paint.setColor(SK_ColorGREEN); |
| matrix.setScale(-SK_Scalar1, -SK_Scalar1); |
| matrix.postTranslate(pathLen, 0); |
| canvas->drawTextOnPath(text, len, path, &matrix, paint); |
| } |
| |
| class TextOnPathView : public SkView { |
| public: |
| SkPath fPath; |
| SkScalar fHOffset; |
| |
| TextOnPathView() { |
| SkRect r; |
| r.set(SkIntToScalar(100), SkIntToScalar(100), |
| SkIntToScalar(300), SkIntToScalar(300)); |
| fPath.addOval(r); |
| |
| fHOffset = SkIntToScalar(50); |
| } |
| |
| protected: |
| // overrides from SkEventSink |
| virtual bool onQuery(SkEvent* evt) { |
| if (SampleCode::TitleQ(*evt)) { |
| SampleCode::TitleR(evt, "Text On Path"); |
| return true; |
| } |
| return this->INHERITED::onQuery(evt); |
| } |
| |
| void drawBG(SkCanvas* canvas) { |
| canvas->drawColor(SK_ColorWHITE); |
| #if 0 |
| SkRect r; |
| SkPaint p; |
| SkRandom rand; |
| p.setAntiAlias(true); |
| |
| for (int i = 0; i < 100; i++) { |
| SkScalar x = rand.nextUScalar1() * 300 + SkIntToScalar(50); |
| SkScalar y = rand.nextUScalar1() * 200 + SkIntToScalar(50); |
| SkScalar w = rand.nextUScalar1() * 10; |
| SkScalar h = rand.nextUScalar1() * 10; |
| r.set(x, y, x + w, y + h); |
| canvas->drawRect(r, p); |
| } |
| |
| test_textBounds(canvas); |
| // return; |
| |
| SkBitmap bm; |
| if (SkImageDecoder::DecodeFile("/loading_tile.png", |
| &bm, SkBitmap::kRGB_565_Config, true)) |
| canvas->drawBitmap(bm, 0, 0); |
| #endif |
| } |
| |
| virtual void onDraw(SkCanvas* canvas) { |
| this->drawBG(canvas); |
| |
| SkPaint paint; |
| |
| paint.setAntiAlias(true); |
| paint.setTextSize(SkIntToScalar(50)); |
| |
| for (int j = 0; j < REPEAT_COUNT; j++) { |
| SkScalar x = fHOffset; |
| |
| paint.setColor(SK_ColorBLACK); |
| canvas->drawTextOnPathHV(gText, sizeof(gText)-1, fPath, |
| x, paint.getTextSize()/2, paint); |
| |
| paint.setColor(SK_ColorRED); |
| canvas->drawTextOnPathHV(gText, sizeof(gText)-1, fPath, |
| x + SkIntToScalar(50), 0, paint); |
| |
| paint.setColor(SK_ColorBLUE); |
| canvas->drawTextOnPathHV(gText, sizeof(gText)-1, fPath, |
| x + SkIntToScalar(100), -paint.getTextSize()/2, paint); |
| } |
| |
| paint.setColor(SK_ColorGREEN); |
| paint.setStyle(SkPaint::kStroke_Style); |
| canvas->drawPath(fPath, paint); |
| |
| canvas->translate(SkIntToScalar(200), 0); |
| test_textpathmatrix(canvas); |
| |
| test_bitmap_blur(canvas); |
| |
| if (REPEAT_COUNT > 1) |
| this->inval(NULL); |
| } |
| |
| virtual SkView::Click* onFindClickHandler(SkScalar x, SkScalar y) { |
| fHints += 1; |
| this->inval(NULL); |
| return this->INHERITED::onFindClickHandler(x, y); |
| } |
| |
| virtual bool onClick(Click* click) { |
| return this->INHERITED::onClick(click); |
| } |
| |
| private: |
| int fHints; |
| typedef SkView INHERITED; |
| }; |
| |
| static const uint16_t gTest0[] = { 0, 0, 1, 1 }; |
| static const uint16_t gTest1[] = { 1, 2, 3, 4, 5, 6 }; |
| static const uint16_t gTest2[] = { 0, 0, 0, 1, 2, 3, 3, 3 }; |
| static const uint16_t gTest3[] = { 0, 0, 0, 0, 0, 0, 1, 2, 3, 3, 3, 0, 0, 1 }; |
| |
| #include "SkRandom.h" |
| static SkRandom gRand; |
| static void rand_fill(uint16_t buffer[], int count) { |
| for (int i = 0; i < count; i++) |
| buffer[i] = (uint16_t)gRand.nextU(); |
| } |
| |
| static void test_pack16() { |
| static const struct { |
| const uint16_t* fSrc; |
| int fCount; |
| } gTests[] = { |
| { gTest0, SK_ARRAY_COUNT(gTest0) }, |
| { gTest1, SK_ARRAY_COUNT(gTest1) }, |
| { gTest2, SK_ARRAY_COUNT(gTest2) }, |
| { gTest3, SK_ARRAY_COUNT(gTest3) } |
| }; |
| |
| for (size_t i = 0; i < SK_ARRAY_COUNT(gTests); i++) { |
| uint8_t dst[100]; |
| size_t dstSize = SkPackBits::Pack16(gTests[i].fSrc, |
| gTests[i].fCount, dst); |
| printf("Test[%d] orig size = %d, dst size = %d", |
| i, gTests[i].fCount, (int)dstSize); |
| uint16_t src[100]; |
| int srcCount = SkPackBits::Unpack16(dst, dstSize, src); |
| printf(", src size = %d", srcCount); |
| bool match = gTests[i].fCount == srcCount && memcmp(gTests[i].fSrc, src, |
| gTests[i].fCount * sizeof(uint16_t)) == 0; |
| printf(", match = %d\n", match); |
| } |
| |
| for (int n = 1000; n; n--) { |
| size_t size = 50; |
| uint16_t src[100], src2[100]; |
| uint8_t dst[200]; |
| rand_fill(src, size); |
| |
| size_t dstSize = SkPackBits::Pack16(src, size, dst); |
| size_t maxSize = SkPackBits::ComputeMaxSize16(size); |
| SkASSERT(maxSize >= dstSize); |
| |
| int srcCount = SkPackBits::Unpack16(dst, dstSize, src2); |
| SkASSERT(size == srcCount); |
| bool match = memcmp(src, src2, size * sizeof(uint16_t)) == 0; |
| SkASSERT(match); |
| } |
| } |
| |
| static const uint8_t gTest80[] = { 0, 0, 1, 1 }; |
| static const uint8_t gTest81[] = { 1, 2, 3, 4, 5, 6 }; |
| static const uint8_t gTest82[] = { 0, 0, 0, 1, 2, 3, 3, 3 }; |
| static const uint8_t gTest83[] = { 0, 0, 0, 0, 0, 0, 1, 2, 3, 3, 3, 0, 0, 1 }; |
| static const uint8_t gTest84[] = { 1, 0, 3, 0, 0, 0, 2, 1, 1, 2 }; |
| |
| static void rand_fill(uint8_t buffer[], int count) { |
| for (int i = 0; i < count; i++) |
| buffer[i] = (uint8_t)((gRand.nextU() >> 8) & 0x3); |
| } |
| |
| static void test_pack8() { |
| static const struct { |
| const uint8_t* fSrc; |
| int fCount; |
| } gTests[] = { |
| { gTest80, SK_ARRAY_COUNT(gTest80) }, |
| { gTest81, SK_ARRAY_COUNT(gTest81) }, |
| { gTest82, SK_ARRAY_COUNT(gTest82) }, |
| { gTest83, SK_ARRAY_COUNT(gTest83) }, |
| { gTest84, SK_ARRAY_COUNT(gTest84) } |
| }; |
| |
| for (size_t i = 4; i < SK_ARRAY_COUNT(gTests); i++) { |
| uint8_t dst[100]; |
| size_t maxSize = SkPackBits::ComputeMaxSize8(gTests[i].fCount); |
| size_t dstSize = SkPackBits::Pack8(gTests[i].fSrc, |
| gTests[i].fCount, dst); |
| SkASSERT(dstSize <= maxSize); |
| printf("Test[%d] orig size = %d, dst size = %d", i, |
| gTests[i].fCount, (int)dstSize); |
| uint8_t src[100]; |
| int srcCount = SkPackBits::Unpack8(dst, dstSize, src); |
| printf(", src size = %d", srcCount); |
| bool match = gTests[i].fCount == srcCount && |
| memcmp(gTests[i].fSrc, src, |
| gTests[i].fCount * sizeof(uint8_t)) == 0; |
| printf(", match = %d\n", match); |
| } |
| |
| for (size_t size = 1; size <= 512; size += 1) { |
| for (int n = 200; n; n--) { |
| uint8_t src[600], src2[600]; |
| uint8_t dst[600]; |
| rand_fill(src, size); |
| |
| size_t dstSize = SkPackBits::Pack8(src, size, dst); |
| size_t maxSize = SkPackBits::ComputeMaxSize8(size); |
| SkASSERT(maxSize >= dstSize); |
| |
| int srcCount = SkPackBits::Unpack8(dst, dstSize, src2); |
| SkASSERT(size == srcCount); |
| bool match = memcmp(src, src2, size * sizeof(uint8_t)) == 0; |
| SkASSERT(match); |
| |
| for (int j = 0; j < 200; j++) { |
| size_t skip = gRand.nextU() % size; |
| size_t write = gRand.nextU() % size; |
| if (skip + write > size) { |
| write = size - skip; |
| } |
| SkPackBits::Unpack8(src, skip, write, dst); |
| bool match = memcmp(src, src2 + skip, write) == 0; |
| SkASSERT(match); |
| } |
| } |
| } |
| } |
| |
| ////////////////////////////////////////////////////////////////////////////// |
| |
| static SkView* MyFactory() { |
| static bool gOnce; |
| if (!gOnce) { |
| // test_pack8(); |
| gOnce = true; |
| } |
| return new TextOnPathView; |
| } |
| |
| static SkViewRegister reg(MyFactory); |
| |