/* * Copyright 2011 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include "SampleCode.h" #include "SkCanvas.h" #include "SkView.h" #include "Sk1DPathEffect.h" #include "Sk2DPathEffect.h" #include "SkAvoidXfermode.h" #include "SkBlurMaskFilter.h" #include "SkColorFilter.h" #include "SkColorPriv.h" #include "SkCornerPathEffect.h" #include "SkDashPathEffect.h" #include "SkDiscretePathEffect.h" #include "SkEmbossMaskFilter.h" #include "SkGradientShader.h" #include "SkImageDecoder.h" #include "SkLayerRasterizer.h" #include "SkMath.h" #include "SkPath.h" #include "SkRegion.h" #include "SkShader.h" #include "SkComposeShader.h" #include "SkCornerPathEffect.h" #include "SkPathMeasure.h" #include "SkPicture.h" #include "SkRandom.h" #include "SkTransparentShader.h" #include "SkTypeface.h" #include "SkUnitMappers.h" #include "SkUtils.h" #include "SkXfermode.h" #include <math.h> static inline SkPMColor rgb2gray(SkPMColor c) { unsigned r = SkGetPackedR32(c); unsigned g = SkGetPackedG32(c); unsigned b = SkGetPackedB32(c); unsigned x = (r * 5 + g * 7 + b * 4) >> 4; return SkPackARGB32(0, x, x, x) | (c & (SK_A32_MASK << SK_A32_SHIFT)); } class SkGrayScaleColorFilter : public SkColorFilter { public: virtual void filterSpan(const SkPMColor src[], int count, SkPMColor result[]) { for (int i = 0; i < count; i++) result[i] = rgb2gray(src[i]); } }; class SkChannelMaskColorFilter : public SkColorFilter { public: SkChannelMaskColorFilter(U8CPU redMask, U8CPU greenMask, U8CPU blueMask) { fMask = SkPackARGB32(0xFF, redMask, greenMask, blueMask); } virtual void filterSpan(const SkPMColor src[], int count, SkPMColor result[]) { SkPMColor mask = fMask; for (int i = 0; i < count; i++) { result[i] = src[i] & mask; } } private: SkPMColor fMask; }; /////////////////////////////////////////////////////////// static void r0(SkLayerRasterizer* rast, SkPaint& p) { p.setMaskFilter(SkBlurMaskFilter::Create(SkIntToScalar(3), SkBlurMaskFilter::kNormal_BlurStyle))->unref(); rast->addLayer(p, SkIntToScalar(3), SkIntToScalar(3)); p.setMaskFilter(NULL); p.setStyle(SkPaint::kStroke_Style); p.setStrokeWidth(SK_Scalar1); rast->addLayer(p); p.setAlpha(0x11); p.setStyle(SkPaint::kFill_Style); p.setXfermodeMode(SkXfermode::kSrc_Mode); rast->addLayer(p); } static void r1(SkLayerRasterizer* rast, SkPaint& p) { rast->addLayer(p); p.setAlpha(0x40); p.setXfermodeMode(SkXfermode::kSrc_Mode); p.setStyle(SkPaint::kStroke_Style); p.setStrokeWidth(SK_Scalar1*2); rast->addLayer(p); } static void r2(SkLayerRasterizer* rast, SkPaint& p) { p.setStyle(SkPaint::kStrokeAndFill_Style); p.setStrokeWidth(SK_Scalar1*4); rast->addLayer(p); p.setStyle(SkPaint::kStroke_Style); p.setStrokeWidth(SK_Scalar1*3/2); p.setXfermodeMode(SkXfermode::kClear_Mode); rast->addLayer(p); } static void r3(SkLayerRasterizer* rast, SkPaint& p) { p.setStyle(SkPaint::kStroke_Style); p.setStrokeWidth(SK_Scalar1*3); rast->addLayer(p); p.setAlpha(0x20); p.setStyle(SkPaint::kFill_Style); p.setXfermodeMode(SkXfermode::kSrc_Mode); rast->addLayer(p); } static void r4(SkLayerRasterizer* rast, SkPaint& p) { p.setAlpha(0x60); rast->addLayer(p, SkIntToScalar(3), SkIntToScalar(3)); p.setAlpha(0xFF); p.setXfermodeMode(SkXfermode::kClear_Mode); rast->addLayer(p, SK_Scalar1*3/2, SK_Scalar1*3/2); p.setXfermode(NULL); rast->addLayer(p); } static void r5(SkLayerRasterizer* rast, SkPaint& p) { rast->addLayer(p); p.setPathEffect(new SkDiscretePathEffect(SK_Scalar1*4, SK_Scalar1*3))->unref(); p.setXfermodeMode(SkXfermode::kSrcOut_Mode); rast->addLayer(p); } static void r6(SkLayerRasterizer* rast, SkPaint& p) { rast->addLayer(p); p.setAntiAlias(false); SkLayerRasterizer* rast2 = new SkLayerRasterizer; r5(rast2, p); p.setRasterizer(rast2)->unref(); p.setXfermodeMode(SkXfermode::kClear_Mode); rast->addLayer(p); } class Dot2DPathEffect : public Sk2DPathEffect { public: Dot2DPathEffect(SkScalar radius, const SkMatrix& matrix) : Sk2DPathEffect(matrix), fRadius(radius) {} virtual void flatten(SkFlattenableWriteBuffer& buffer) { this->INHERITED::flatten(buffer); buffer.writeScalar(fRadius); } virtual Factory getFactory() { return CreateProc; } protected: virtual void next(const SkPoint& loc, int u, int v, SkPath* dst) { dst->addCircle(loc.fX, loc.fY, fRadius); } Dot2DPathEffect(SkFlattenableReadBuffer& buffer) : Sk2DPathEffect(buffer) { fRadius = buffer.readScalar(); } private: SkScalar fRadius; static SkFlattenable* CreateProc(SkFlattenableReadBuffer& buffer) { return new Dot2DPathEffect(buffer); } typedef Sk2DPathEffect INHERITED; }; static void r7(SkLayerRasterizer* rast, SkPaint& p) { SkMatrix lattice; lattice.setScale(SK_Scalar1*6, SK_Scalar1*6, 0, 0); lattice.postSkew(SK_Scalar1/3, 0, 0, 0); p.setPathEffect(new Dot2DPathEffect(SK_Scalar1*4, lattice))->unref(); rast->addLayer(p); } static void r8(SkLayerRasterizer* rast, SkPaint& p) { rast->addLayer(p); SkMatrix lattice; lattice.setScale(SK_Scalar1*6, SK_Scalar1*6, 0, 0); lattice.postSkew(SK_Scalar1/3, 0, 0, 0); p.setPathEffect(new Dot2DPathEffect(SK_Scalar1*2, lattice))->unref(); p.setXfermodeMode(SkXfermode::kClear_Mode); rast->addLayer(p); p.setPathEffect(NULL); p.setXfermode(NULL); p.setStyle(SkPaint::kStroke_Style); p.setStrokeWidth(SK_Scalar1); rast->addLayer(p); } class Line2DPathEffect : public Sk2DPathEffect { public: Line2DPathEffect(SkScalar width, const SkMatrix& matrix) : Sk2DPathEffect(matrix), fWidth(width) {} virtual bool filterPath(SkPath* dst, const SkPath& src, SkScalar* width) { if (this->INHERITED::filterPath(dst, src, width)) { *width = fWidth; return true; } return false; } virtual Factory getFactory() { return CreateProc; } virtual void flatten(SkFlattenableWriteBuffer& buffer) { this->INHERITED::flatten(buffer); buffer.writeScalar(fWidth); } protected: virtual void nextSpan(int u, int v, int ucount, SkPath* dst) { if (ucount > 1) { SkPoint src[2], dstP[2]; src[0].set(SkIntToScalar(u) + SK_ScalarHalf, SkIntToScalar(v) + SK_ScalarHalf); src[1].set(SkIntToScalar(u+ucount) + SK_ScalarHalf, SkIntToScalar(v) + SK_ScalarHalf); this->getMatrix().mapPoints(dstP, src, 2); dst->moveTo(dstP[0]); dst->lineTo(dstP[1]); } } Line2DPathEffect(SkFlattenableReadBuffer& buffer) : Sk2DPathEffect(buffer) { fWidth = buffer.readScalar(); } private: SkScalar fWidth; static SkFlattenable* CreateProc(SkFlattenableReadBuffer& buffer) { return new Line2DPathEffect(buffer); } typedef Sk2DPathEffect INHERITED; }; static void r9(SkLayerRasterizer* rast, SkPaint& p) { rast->addLayer(p); SkMatrix lattice; lattice.setScale(SK_Scalar1, SK_Scalar1*6, 0, 0); lattice.postRotate(SkIntToScalar(30), 0, 0); p.setPathEffect(new Line2DPathEffect(SK_Scalar1*2, lattice))->unref(); p.setXfermodeMode(SkXfermode::kClear_Mode); rast->addLayer(p); p.setPathEffect(NULL); p.setXfermode(NULL); p.setStyle(SkPaint::kStroke_Style); p.setStrokeWidth(SK_Scalar1); rast->addLayer(p); } typedef void (*raster_proc)(SkLayerRasterizer*, SkPaint&); static const raster_proc gRastProcs[] = { r0, r1, r2, r3, r4, r5, r6, r7, r8, r9 }; static const struct { SkColor fMul, fAdd; } gLightingColors[] = { { 0x808080, 0x800000 }, // general case { 0x707070, 0x707070 }, // no-pin case { 0xFFFFFF, 0x800000 }, // just-add case { 0x808080, 0x000000 }, // just-mul case { 0xFFFFFF, 0x000000 } // identity case }; static unsigned color_dist16(uint16_t a, uint16_t b) { unsigned dr = SkAbs32(SkPacked16ToR32(a) - SkPacked16ToR32(b)); unsigned dg = SkAbs32(SkPacked16ToG32(a) - SkPacked16ToG32(b)); unsigned db = SkAbs32(SkPacked16ToB32(a) - SkPacked16ToB32(b)); return SkMax32(dr, SkMax32(dg, db)); } static unsigned scale_dist(unsigned dist, unsigned scale) { dist >>= 6; dist = (dist << 2) | dist; dist = (dist << 4) | dist; return dist; // return SkAlphaMul(dist, scale); } static void apply_shader(SkPaint* paint, int index) { raster_proc proc = gRastProcs[index]; if (proc) { SkPaint p; SkLayerRasterizer* rast = new SkLayerRasterizer; p.setAntiAlias(true); proc(rast, p); paint->setRasterizer(rast)->unref(); } #if 1 SkScalar dir[] = { SK_Scalar1, SK_Scalar1, SK_Scalar1 }; paint->setMaskFilter(SkBlurMaskFilter::CreateEmboss(dir, SK_Scalar1/4, SkIntToScalar(4), SkIntToScalar(3)))->unref(); paint->setColor(SK_ColorBLUE); #endif } class DemoView : public SampleView { public: DemoView() {} protected: // overrides from SkEventSink virtual bool onQuery(SkEvent* evt) { if (SampleCode::TitleQ(*evt)) { SampleCode::TitleR(evt, "Demo"); return true; } return this->INHERITED::onQuery(evt); } virtual bool onClick(Click* click) { return this->INHERITED::onClick(click); } void makePath(SkPath& path) { path.addCircle(SkIntToScalar(20), SkIntToScalar(20), SkIntToScalar(20), SkPath::kCCW_Direction); for (int index = 0; index < 10; index++) { SkScalar x = SkFloatToScalar(cos(index / 10.0f * 2 * 3.1415925358f)); SkScalar y = SkFloatToScalar(sin(index / 10.0f * 2 * 3.1415925358f)); x *= index & 1 ? 7 : 14; y *= index & 1 ? 7 : 14; x += SkIntToScalar(20); y += SkIntToScalar(20); if (index == 0) path.moveTo(x, y); else path.lineTo(x, y); } path.close(); } virtual void onDrawContent(SkCanvas* canvas) { canvas->save(); drawPicture(canvas, 0); canvas->restore(); { SkPicture picture; SkCanvas* record = picture.beginRecording(320, 480); drawPicture(record, 120); canvas->translate(0, SkIntToScalar(120)); SkRect clip; clip.set(0, 0, SkIntToScalar(160), SkIntToScalar(160)); do { canvas->save(); canvas->clipRect(clip); picture.draw(canvas); canvas->restore(); if (clip.fRight < SkIntToScalar(320)) clip.offset(SkIntToScalar(160), 0); else if (clip.fBottom < SkIntToScalar(480)) clip.offset(-SkIntToScalar(320), SkIntToScalar(160)); else break; } while (true); } } void drawPicture(SkCanvas* canvas, int spriteOffset) { SkMatrix matrix; matrix.reset(); SkPaint paint; SkPath path; SkPoint start = {0, 0}; SkPoint stop = { SkIntToScalar(40), SkIntToScalar(40) }; SkRect rect = {0, 0, SkIntToScalar(40), SkIntToScalar(40) }; SkRect rect2 = {0, 0, SkIntToScalar(65), SkIntToScalar(20) }; SkScalar left = 0, top = 0, x = 0, y = 0; size_t index; char ascii[] = "ascii..."; size_t asciiLength = sizeof(ascii) - 1; char utf8[] = "utf8" "\xe2\x80\xa6"; short utf16[] = {'u', 't', 'f', '1', '6', 0x2026 }; short utf16simple[] = {'u', 't', 'f', '1', '6', '!' }; makePath(path); SkTDArray<SkPoint>(pos); pos.setCount(asciiLength); for (index = 0; index < asciiLength; index++) pos[index].set(SkIntToScalar(index * 10), SkIntToScalar(index * 2)); SkTDArray<SkPoint>(pos2); pos2.setCount(asciiLength); for (index = 0; index < asciiLength; index++) pos2[index].set(SkIntToScalar(index * 10), SkIntToScalar(20)); // shaders SkPoint linearPoints[] = { { 0, 0, }, { SkIntToScalar(40), SkIntToScalar(40) } }; SkColor linearColors[] = { SK_ColorRED, SK_ColorBLUE }; SkScalar* linearPos = NULL; int linearCount = 2; SkShader::TileMode linearMode = SkShader::kMirror_TileMode; SkUnitMapper* linearMapper = new SkDiscreteMapper(3); SkAutoUnref unmapLinearMapper(linearMapper); SkShader* linear = SkGradientShader::CreateLinear(linearPoints, linearColors, linearPos, linearCount, linearMode, linearMapper); SkPoint radialCenter = { SkIntToScalar(25), SkIntToScalar(25) }; SkScalar radialRadius = SkIntToScalar(25); SkColor radialColors[] = { SK_ColorGREEN, SK_ColorGRAY, SK_ColorRED }; SkScalar radialPos[] = { 0, SkIntToScalar(3) / 5, SkIntToScalar(1)}; int radialCount = 3; SkShader::TileMode radialMode = SkShader::kRepeat_TileMode; SkUnitMapper* radialMapper = new SkCosineMapper(); SkAutoUnref unmapRadialMapper(radialMapper); SkShader* radial = SkGradientShader::CreateRadial(radialCenter, radialRadius, radialColors, radialPos, radialCount, radialMode, radialMapper); SkTransparentShader* transparentShader = new SkTransparentShader(); SkEmbossMaskFilter::Light light; light.fDirection[0] = SK_Scalar1/2; light.fDirection[1] = SK_Scalar1/2; light.fDirection[2] = SK_Scalar1/3; light.fAmbient = 0x48; light.fSpecular = 0x80; SkScalar radius = SkIntToScalar(12)/5; SkEmbossMaskFilter* embossFilter = new SkEmbossMaskFilter(light, radius); SkXfermode* xfermode = SkXfermode::Create(SkXfermode::kXor_Mode); SkColorFilter* lightingFilter = SkColorFilter::CreateLightingFilter( 0xff89bc45, 0xff112233); canvas->save(); canvas->translate(SkIntToScalar(0), SkIntToScalar(5)); paint.setFlags(SkPaint::kAntiAlias_Flag | SkPaint::kFilterBitmap_Flag); // !!! draw through a clip paint.setColor(SK_ColorLTGRAY); paint.setStyle(SkPaint::kFill_Style); SkRect clip = {0, 0, SkIntToScalar(320), SkIntToScalar(120)}; canvas->clipRect(clip); paint.setShader(SkShader::CreateBitmapShader(fTx, SkShader::kMirror_TileMode, SkShader::kRepeat_TileMode))->unref(); canvas->drawPaint(paint); canvas->save(); // line (exercises xfermode, colorShader, colorFilter, filterShader) paint.setColor(SK_ColorGREEN); paint.setStrokeWidth(SkIntToScalar(10)); paint.setStyle(SkPaint::kStroke_Style); paint.setXfermode(xfermode)->unref(); paint.setColorFilter(lightingFilter)->unref(); canvas->drawLine(start.fX, start.fY, stop.fX, stop.fY, paint); // should not be green paint.setXfermode(NULL); paint.setColorFilter(NULL); // rectangle paint.setStyle(SkPaint::kFill_Style); canvas->translate(SkIntToScalar(50), 0); paint.setColor(SK_ColorYELLOW); paint.setShader(linear)->unref(); paint.setPathEffect(pathEffectTest())->unref(); canvas->drawRect(rect, paint); paint.setPathEffect(NULL); // circle w/ emboss & transparent (exercises 3dshader) canvas->translate(SkIntToScalar(50), 0); paint.setMaskFilter(embossFilter)->unref(); canvas->drawOval(rect, paint); canvas->translate(SkIntToScalar(10), SkIntToScalar(10)); paint.setShader(transparentShader)->unref(); canvas->drawOval(rect, paint); canvas->translate(0, SkIntToScalar(-10)); // path canvas->translate(SkIntToScalar(50), 0); paint.setColor(SK_ColorRED); paint.setStyle(SkPaint::kStroke_Style); paint.setStrokeWidth(SkIntToScalar(5)); paint.setShader(radial)->unref(); paint.setMaskFilter(NULL); canvas->drawPath(path, paint); paint.setShader(NULL); // bitmap, sprite canvas->translate(SkIntToScalar(50), 0); paint.setStyle(SkPaint::kFill_Style); canvas->drawBitmap(fBug, left, top, &paint); canvas->translate(SkIntToScalar(30), 0); canvas->drawSprite(fTb, SkScalarRound(canvas->getTotalMatrix().getTranslateX()), spriteOffset + 10, &paint); canvas->translate(-SkIntToScalar(30), SkIntToScalar(30)); paint.setShader(shaderTest())->unref(); // test compose shader canvas->drawRect(rect2, paint); paint.setShader(NULL); canvas->restore(); // text canvas->translate(0, SkIntToScalar(60)); canvas->save(); paint.setColor(SK_ColorGRAY); canvas->drawPosText(ascii, asciiLength, pos.begin(), paint); canvas->drawPosText(ascii, asciiLength, pos2.begin(), paint); canvas->translate(SkIntToScalar(50), 0); paint.setColor(SK_ColorCYAN); canvas->drawText(utf8, sizeof(utf8) - 1, x, y, paint); canvas->translate(SkIntToScalar(30), 0); paint.setColor(SK_ColorMAGENTA); paint.setTextEncoding(SkPaint::kUTF16_TextEncoding); matrix.setTranslate(SkIntToScalar(10), SkIntToScalar(10)); canvas->drawTextOnPath((void*) utf16, sizeof(utf16), path, &matrix, paint); canvas->translate(0, SkIntToScalar(20)); canvas->drawTextOnPath((void*) utf16simple, sizeof(utf16simple), path, &matrix, paint); canvas->restore(); canvas->translate(0, SkIntToScalar(60)); paint.setTextEncoding(SkPaint::kUTF8_TextEncoding); canvas->restore(); } /* ./SkColorFilter.h:25:class SkColorFilter : public SkFlattenable { -- abstract static SkColorFilter* CreatXfermodeFilter() *** untested *** static SkColorFilter* CreatePorterDuffFilter() *** untested *** static SkColorFilter* CreateLightingFilter() -- tested ./SkDrawLooper.h:9:class SkDrawLooper : public SkFlattenable { -- virtually abstract ./SkBlurDrawLooper.h:9:class SkBlurDrawLooper : public SkDrawLooper { *** untested *** ./SkMaskFilter.h:41:class SkMaskFilter : public SkFlattenable { -- abstract chmod +w .h ./SkEmbossMaskFilter.h:27:class SkEmbossMaskFilter : public SkMaskFilter { -- tested ./SkPathEffect.h:33:class SkPathEffect : public SkFlattenable { -- abstract ./Sk1DPathEffect.h:27:class Sk1DPathEffect : public SkPathEffect { -- abstract ./Sk1DPathEffect.h:48:class SkPath1DPathEffect : public Sk1DPathEffect { -- tested ./Sk2DPathEffect.h:25:class Sk2DPathEffect : public SkPathEffect { *** untested *** ./SkCornerPathEffect.h:28:class SkCornerPathEffect : public SkPathEffect { *** untested *** ./SkDashPathEffect.h:27:class SkDashPathEffect : public SkPathEffect { ./SkDiscretePathEffect.h:27:class SkDiscretePathEffect : public SkPathEffect { ./SkPaint.h:760:class SkStrokePathEffect : public SkPathEffect { ./SkPathEffect.h:58:class SkPairPathEffect : public SkPathEffect { ./SkPathEffect.h:78:class SkComposePathEffect : public SkPairPathEffect { ./SkPathEffect.h:114:class SkSumPathEffect : public SkPairPathEffect { ./SkRasterizer.h:29:class SkRasterizer : public SkFlattenable { ./SkLayerRasterizer.h:27:class SkLayerRasterizer : public SkRasterizer { ./SkShader.h:36:class SkShader : public SkFlattenable { ./SkColorFilter.h:59:class SkFilterShader : public SkShader { ./SkColorShader.h:26:class SkColorShader : public SkShader { ./SkShaderExtras.h:31:class SkComposeShader : public SkShader { ./SkTransparentShader.h:23:class SkTransparentShader : public SkShader { ./SkUnitMapper.h:24:class SkUnitMapper : public SkFlattenable { ./SkUnitMapper.h:33:class SkDiscreteMapper : public SkUnitMapper { ./SkUnitMapper.h:51:class SkFlipCosineMapper : public SkUnitMapper { ./SkXfermode.h:32:class SkXfermode : public SkFlattenable { ./SkAvoidXfermode.h:28:class SkAvoidXfermode : public SkXfermode { *** not done *** chmod +w .h .cpp ./SkXfermode.h:54:class SkProcXfermode : public SkXfermode { */ /* ./SkBlurMaskFilter.h:25:class SkBlurMaskFilter { chmod +w SkBlurMaskFilter.cpp ./SkGradientShader.h:30:class SkGradientShader { */ // save layer, bounder, looper // matrix // clip /path/region // bitmap proc shader ? /* untested: SkCornerPathEffect.h:28:class SkCornerPathEffect : public SkPathEffect { */ virtual SkView::Click* onFindClickHandler(SkScalar x, SkScalar y) { fClickPt.set(x, y); this->inval(NULL); return this->INHERITED::onFindClickHandler(x, y); } SkPathEffect* pathEffectTest() { static const int gXY[] = { 1, 0, 0, -1, 2, -1, 3, 0, 2, 1, 0, 1 }; SkScalar gPhase = 0; SkPath path; path.moveTo(SkIntToScalar(gXY[0]), SkIntToScalar(gXY[1])); for (unsigned i = 2; i < SK_ARRAY_COUNT(gXY); i += 2) path.lineTo(SkIntToScalar(gXY[i]), SkIntToScalar(gXY[i+1])); path.close(); path.offset(SkIntToScalar(-6), 0); SkPathEffect* outer = new SkPath1DPathEffect(path, SkIntToScalar(12), gPhase, SkPath1DPathEffect::kRotate_Style); SkPathEffect* inner = new SkDiscretePathEffect(SkIntToScalar(2), SkIntToScalar(1)/10); // SkCornerPathEffect(SkIntToScalar(2)); SkPathEffect* result = new SkComposePathEffect(outer, inner); outer->unref(); inner->unref(); return result; } SkPathEffect* pathEffectTest2() { // unsure this works (has no visible effect) SkPathEffect* outer = new SkStrokePathEffect(SkIntToScalar(4), SkPaint::kStroke_Style, SkPaint::kMiter_Join, SkPaint::kButt_Cap); static const SkScalar intervals[] = {SkIntToScalar(1), SkIntToScalar(2), SkIntToScalar(2), SkIntToScalar(1)}; SkPathEffect* inner = new SkDashPathEffect(intervals, sizeof(intervals) / sizeof(intervals[0]), 0); SkPathEffect* result = new SkSumPathEffect(outer, inner); outer->unref(); inner->unref(); return result; } SkShader* shaderTest() { SkPoint pts[] = { { 0, 0, }, { SkIntToScalar(100), 0 } }; SkColor colors[] = { SK_ColorRED, SK_ColorBLUE }; SkShader* shaderA = SkGradientShader::CreateLinear(pts, colors, NULL, 2, SkShader::kClamp_TileMode); pts[1].set(0, SkIntToScalar(100)); SkColor colors2[] = {SK_ColorBLACK, SkColorSetARGB(0x80, 0, 0, 0)}; SkShader* shaderB = SkGradientShader::CreateLinear(pts, colors2, NULL, 2, SkShader::kClamp_TileMode); SkXfermode* mode = SkXfermode::Create(SkXfermode::kDstIn_Mode); SkShader* result = new SkComposeShader(shaderA, shaderB, mode); shaderA->unref(); shaderB->unref(); mode->unref(); return result; } virtual void startTest() { SkImageDecoder::DecodeFile("/Users/caryclark/Desktop/bugcirc.gif", &fBug); SkImageDecoder::DecodeFile("/Users/caryclark/Desktop/tbcirc.gif", &fTb); SkImageDecoder::DecodeFile("/Users/caryclark/Desktop/05psp04.gif", &fTx); } void drawRaster(SkCanvas* canvas) { for (size_t index = 0; index < SK_ARRAY_COUNT(gRastProcs); index++) drawOneRaster(canvas); } void drawOneRaster(SkCanvas* canvas) { canvas->save(); SkScalar x = SkIntToScalar(20); SkScalar y = SkIntToScalar(40); SkPaint paint; paint.setAntiAlias(true); paint.setTextSize(SkIntToScalar(48)); paint.setTypeface(SkTypeface::CreateFromName("sans-serif", SkTypeface::kBold)); SkString str("GOOGLE"); for (size_t i = 0; i < SK_ARRAY_COUNT(gRastProcs); i++) { apply_shader(&paint, i); // paint.setMaskFilter(NULL); // paint.setColor(SK_ColorBLACK); #if 01 int index = i % SK_ARRAY_COUNT(gLightingColors); paint.setColorFilter(SkColorFilter::CreateLightingFilter( gLightingColors[index].fMul, gLightingColors[index].fAdd))->unref(); #endif canvas->drawText(str.c_str(), str.size(), x, y, paint); SkRect oval = { x, y - SkIntToScalar(40), x + SkIntToScalar(40), y }; paint.setStyle(SkPaint::kStroke_Style); canvas->drawOval(oval, paint); paint.setStyle(SkPaint::kFill_Style); y += paint.getFontSpacing(); } canvas->restore(); if (1) { SkAvoidXfermode mode(SK_ColorWHITE, 0xFF, SkAvoidXfermode::kTargetColor_Mode); SkPaint paint; x += SkIntToScalar(20); SkRect r = { x, 0, x + SkIntToScalar(360), SkIntToScalar(700) }; paint.setXfermode(&mode); paint.setColor(SK_ColorGREEN); paint.setAntiAlias(true); canvas->drawOval(r, paint); } } private: SkPoint fClickPt; SkBitmap fBug, fTb, fTx; typedef SampleView INHERITED; }; ////////////////////////////////////////////////////////////////////////////// static SkView* MyFactory() { return new DemoView; } static SkViewRegister reg(MyFactory);