skia2/samplecode/SampleTextOnPath.cpp
reed@android.com da342a8917 update more sample code to new compose shader header
git-svn-id: http://skia.googlecode.com/svn/trunk@48 2bbb7eff-a529-9590-31e7-b0007b416f81
2008-12-23 18:37:08 +00:00

443 lines
13 KiB
C++

#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);