| Index: gm/textbloblooper.cpp
|
| diff --git a/gm/textbloblooper.cpp b/gm/textbloblooper.cpp
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..253945f123be7a24d59744f24e0a37709158033e
|
| --- /dev/null
|
| +++ b/gm/textbloblooper.cpp
|
| @@ -0,0 +1,257 @@
|
| +/*
|
| + * Copyright 2013 Google Inc.
|
| + *
|
| + * Use of this source code is governed by a BSD-style license that can be
|
| + * found in the LICENSE file.
|
| + */
|
| +
|
| +#include "gm.h"
|
| +
|
| +#include "Sk2DPathEffect.h"
|
| +#include "SkBlurMask.h"
|
| +#include "SkBlurMaskFilter.h"
|
| +#include "SkColorFilter.h"
|
| +#include "SkCanvas.h"
|
| +#include "SkGradientShader.h"
|
| +#include "SkGraphics.h"
|
| +#include "SkLayerDrawLooper.h"
|
| +#include "SkRandom.h"
|
| +#include "SkTextBlob.h"
|
| +
|
| +namespace skiagm {
|
| +
|
| +static const int kWidth = 1250;
|
| +static const int kHeight = 700;
|
| +
|
| +static void add_to_text_blob(SkTextBlobBuilder* builder, const char* text, const SkPaint& origPaint,
|
| + SkScalar x, SkScalar y) {
|
| + SkPaint paint(origPaint);
|
| + SkTDArray<uint16_t> glyphs;
|
| +
|
| + size_t len = strlen(text);
|
| + glyphs.append(paint.textToGlyphs(text, len, NULL));
|
| + paint.textToGlyphs(text, len, glyphs.begin());
|
| +
|
| + const SkScalar advanceX = paint.getTextSize() * 0.85f;
|
| + const SkScalar advanceY = paint.getTextSize() * 1.5f;
|
| +
|
| + SkTDArray<SkScalar> pos;
|
| + for (unsigned i = 0; i < len; ++i) {
|
| + *pos.append() = x + i * advanceX;
|
| + *pos.append() = y + i * (advanceY / len);
|
| + }
|
| +
|
| + paint.setTextEncoding(SkPaint::kGlyphID_TextEncoding);
|
| + const SkTextBlobBuilder::RunBuffer& run = builder->allocRunPos(paint, glyphs.count());
|
| + memcpy(run.glyphs, glyphs.begin(), glyphs.count() * sizeof(uint16_t));
|
| + memcpy(run.pos, pos.begin(), len * sizeof(SkScalar) * 2);
|
| +}
|
| +
|
| +typedef void (*LooperProc)(SkPaint*);
|
| +
|
| +struct LooperSettings {
|
| + SkXfermode::Mode fMode;
|
| + SkColor fColor;
|
| + SkPaint::Style fStyle;
|
| + SkScalar fWidth;
|
| + SkScalar fOffset;
|
| + SkScalar fSkewX;
|
| + bool fEffect;
|
| +};
|
| +
|
| +static void mask_filter(SkPaint* paint) {
|
| + SkMaskFilter* mf = SkBlurMaskFilter::Create(kNormal_SkBlurStyle,
|
| + SkBlurMask::ConvertRadiusToSigma(3.f));
|
| + paint->setMaskFilter(mf)->unref();
|
| +}
|
| +
|
| +static SkPathEffect* make_tile_effect() {
|
| + SkMatrix m;
|
| + m.setScale(1.f, 1.f);
|
| +
|
| + SkPath path;
|
| + path.addCircle(0, 0, SkIntToScalar(5));
|
| +
|
| + return SkPath2DPathEffect::Create(m, path);
|
| +}
|
| +
|
| +static void path_effect(SkPaint* paint) {
|
| + paint->setPathEffect(make_tile_effect())->unref();
|
| +}
|
| +
|
| +static SkShader* make_shader(const SkRect& bounds) {
|
| + const SkPoint pts[] = {
|
| + { bounds.left(), bounds.top() },
|
| + { bounds.right(), bounds.bottom() },
|
| + };
|
| + const SkColor colors[] = {
|
| + SK_ColorRED, SK_ColorGREEN, SK_ColorBLUE, SK_ColorBLACK,
|
| + SK_ColorCYAN, SK_ColorMAGENTA, SK_ColorYELLOW,
|
| + };
|
| + return SkGradientShader::CreateLinear(pts,
|
| + colors, NULL, SK_ARRAY_COUNT(colors),
|
| + SkShader::kClamp_TileMode);
|
| +}
|
| +
|
| +static void color_filter(SkPaint* paint) {
|
| + SkRect r;
|
| + r.setWH(SkIntToScalar(kWidth), 50);
|
| + paint->setShader(make_shader(r))->unref();
|
| + paint->setColorFilter(SkColorFilter::CreateLightingFilter(0xF0F0F0, 0))->unref();
|
| +}
|
| +
|
| +static void kitchen_sink(SkPaint* paint) {
|
| + color_filter(paint);
|
| + path_effect(paint);
|
| + mask_filter(paint);
|
| +
|
| +}
|
| +
|
| +static SkLayerDrawLooper* setupLooper(SkLayerDrawLooper::BitFlags bits,
|
| + LooperProc proc,
|
| + const LooperSettings settings[],
|
| + size_t size) {
|
| + SkLayerDrawLooper::Builder looperBuilder;
|
| +
|
| + SkLayerDrawLooper::LayerInfo info;
|
| + info.fPaintBits = bits;
|
| +
|
| + info.fColorMode = SkXfermode::kSrc_Mode;
|
| +
|
| + for (size_t i = 0; i < size; i++) {
|
| + info.fOffset.set(settings[i].fOffset, settings[i].fOffset);
|
| + SkPaint* paint = looperBuilder.addLayer(info);
|
| + paint->setXfermodeMode(settings[i].fMode);
|
| + paint->setColor(settings[i].fColor);
|
| + paint->setStyle(settings[i].fStyle);
|
| + paint->setStrokeWidth(settings[i].fWidth);
|
| + if (settings[i].fEffect) {
|
| + (*proc)(paint);
|
| + }
|
| + }
|
| + return looperBuilder.detachLooper();
|
| +}
|
| +
|
| +class TextBlobLooperGM : public GM {
|
| +public:
|
| + TextBlobLooperGM() {}
|
| +
|
| +protected:
|
| + void onOnceBeforeDraw() override {
|
| + SkTextBlobBuilder builder;
|
| +
|
| + // LCD
|
| + SkPaint paint;
|
| + paint.setTextSize(32);
|
| + const char* text = "The quick brown fox jumps over the lazy dog";
|
| + paint.setSubpixelText(true);
|
| + paint.setLCDRenderText(true);
|
| + paint.setAntiAlias(true);
|
| + add_to_text_blob(&builder, text, paint, 0, 0);
|
| + fBlob.reset(builder.build());
|
| +
|
| + // create a looper which sandwhiches an effect in two normal draws
|
| + LooperSettings looperSandwhich[] = {
|
| + { SkXfermode::kSrc_Mode, SK_ColorMAGENTA, SkPaint::kFill_Style, 0, 0, 0, false },
|
| + { SkXfermode::kSrcOver_Mode, 0x88000000, SkPaint::kFill_Style, 0, 10.f, 0, true },
|
| + { SkXfermode::kSrcOver_Mode, 0x50FF00FF, SkPaint::kFill_Style, 0, 20.f, 0, false },
|
| + };
|
| +
|
| + LooperSettings compound[] = {
|
| + { SkXfermode::kSrc_Mode, SK_ColorWHITE, SkPaint::kStroke_Style, 1.f * 3/4, 0, 0, false },
|
| + { SkXfermode::kSrc_Mode, SK_ColorRED, SkPaint::kStroke_Style, 4.f, 0, 0, false },
|
| + { SkXfermode::kSrc_Mode, SK_ColorBLUE, SkPaint::kFill_Style, 0, 0, 0, false },
|
| + { SkXfermode::kSrcOver_Mode, 0x88000000, SkPaint::kFill_Style, 0, 10.f, 0, true }
|
| + };
|
| +
|
| + LooperSettings xfermode[] = {
|
| + { SkXfermode::kDifference_Mode, SK_ColorWHITE, SkPaint::kFill_Style, 0, 0, 0, false },
|
| + { SkXfermode::kSrcOver_Mode, 0xFF000000, SkPaint::kFill_Style, 0, 1.f, 0, true },
|
| + { SkXfermode::kSrcOver_Mode, 0x50FF00FF, SkPaint::kFill_Style, 0, 2.f, 0, false },
|
| + };
|
| +
|
| + // NOTE, this should be ignored by textblobs
|
| + LooperSettings skew[] = {
|
| + { SkXfermode::kSrc_Mode, SK_ColorRED, SkPaint::kFill_Style, 0, 0, -1.f, false },
|
| + { SkXfermode::kSrc_Mode, SK_ColorGREEN, SkPaint::kFill_Style, 0, 10.f, -1.f, false },
|
| + { SkXfermode::kSrc_Mode, SK_ColorBLUE, SkPaint::kFill_Style, 0, 20.f, -1.f, false },
|
| + };
|
| +
|
| + LooperSettings kitchenSink[] = {
|
| + { SkXfermode::kSrc_Mode, SK_ColorWHITE, SkPaint::kStroke_Style, 1.f * 3/4, 0, 0, false },
|
| + { SkXfermode::kSrc_Mode, SK_ColorBLACK, SkPaint::kFill_Style, 0, 0, 0, false },
|
| + { SkXfermode::kDifference_Mode, SK_ColorWHITE, SkPaint::kFill_Style, 1.f, 10.f, 0, false },
|
| + { SkXfermode::kSrc_Mode, SK_ColorWHITE, SkPaint::kFill_Style, 0, 10.f, 0, true },
|
| + { SkXfermode::kSrcOver_Mode, 0x50FF00FF, SkPaint::kFill_Style, 0, 20.f, 0, false },
|
| + };
|
| +
|
| + fLoopers.push_back().reset(setupLooper(SkLayerDrawLooper::kMaskFilter_Bit |
|
| + SkLayerDrawLooper::kXfermode_Bit |
|
| + SkLayerDrawLooper::kStyle_Bit, &mask_filter,
|
| + compound, SK_ARRAY_COUNT(compound)));
|
| + fLoopers.push_back().reset(setupLooper(SkLayerDrawLooper::kPathEffect_Bit |
|
| + SkLayerDrawLooper::kXfermode_Bit, &path_effect,
|
| + looperSandwhich, SK_ARRAY_COUNT(looperSandwhich)));
|
| + fLoopers.push_back().reset(setupLooper(SkLayerDrawLooper::kShader_Bit |
|
| + SkLayerDrawLooper::kColorFilter_Bit |
|
| + SkLayerDrawLooper::kXfermode_Bit, &color_filter,
|
| + looperSandwhich, SK_ARRAY_COUNT(looperSandwhich)));
|
| + fLoopers.push_back().reset(setupLooper(SkLayerDrawLooper::kShader_Bit |
|
| + SkLayerDrawLooper::kColorFilter_Bit |
|
| + SkLayerDrawLooper::kXfermode_Bit, &color_filter,
|
| + xfermode, SK_ARRAY_COUNT(xfermode)));
|
| + fLoopers.push_back().reset(setupLooper(0, NULL, skew, SK_ARRAY_COUNT(skew)));
|
| + fLoopers.push_back().reset(setupLooper(SkLayerDrawLooper::kMaskFilter_Bit |
|
| + SkLayerDrawLooper::kShader_Bit |
|
| + SkLayerDrawLooper::kColorFilter_Bit |
|
| + SkLayerDrawLooper::kPathEffect_Bit |
|
| + SkLayerDrawLooper::kStyle_Bit |
|
| + SkLayerDrawLooper::kXfermode_Bit, &kitchen_sink,
|
| + kitchenSink, SK_ARRAY_COUNT(kitchenSink)));
|
| +
|
| + // Test we respect overrides
|
| + fLoopers.push_back().reset(setupLooper(0, &kitchen_sink,
|
| + kitchenSink, SK_ARRAY_COUNT(kitchenSink)));
|
| + }
|
| +
|
| + SkString onShortName() override {
|
| + return SkString("textbloblooper");
|
| + }
|
| +
|
| + SkISize onISize() override {
|
| + return SkISize::Make(kWidth, kHeight);
|
| + }
|
| +
|
| + void onDraw(SkCanvas* canvas) override {
|
| +
|
| + canvas->drawColor(SK_ColorGRAY);
|
| +
|
| + SkPaint paint;
|
| + canvas->translate(10, 40);
|
| +
|
| + paint.setTextSize(40);
|
| +
|
| + SkRect bounds = fBlob->bounds();
|
| +
|
| + int y = 0;
|
| + for (int looper = 0; looper < fLoopers.count(); looper++) {
|
| + paint.setLooper(fLoopers[looper]);
|
| + canvas->save();
|
| + canvas->translate(0, SkIntToScalar(y));
|
| + canvas->drawTextBlob(fBlob, 0, 0, paint);
|
| + canvas->restore();
|
| + y += SkScalarFloorToInt(bounds.height());
|
| + }
|
| + }
|
| +
|
| +private:
|
| + SkAutoTUnref<const SkTextBlob> fBlob;
|
| + SkTArray<SkAutoTUnref<SkLayerDrawLooper>, true> fLoopers;
|
| +
|
| + typedef GM INHERITED;
|
| +};
|
| +
|
| +//////////////////////////////////////////////////////////////////////////////
|
| +
|
| +DEF_GM( return SkNEW(TextBlobLooperGM); )
|
| +}
|
|
|