|  |  | 
|  | /* | 
|  | * 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 "SkBlurMask.h" | 
|  | #include "SkBlurMaskFilter.h" | 
|  |  | 
|  | namespace skiagm { | 
|  |  | 
|  | // This GM exercises the blurred rect nine-patching special cases when the | 
|  | // blurred rect is very large and/or very far from the origin. | 
|  | // It creates a large blurred rect/rectori then renders the 4 corners and the | 
|  | // middle. | 
|  | class BigBlursGM : public GM { | 
|  | public: | 
|  | BigBlursGM() { | 
|  | this->setBGColor(0xFFDDDDDD); | 
|  | } | 
|  |  | 
|  | protected: | 
|  | virtual SkString onShortName() SK_OVERRIDE { | 
|  | return SkString("bigblurs"); | 
|  | } | 
|  |  | 
|  | virtual SkISize onISize() SK_OVERRIDE { | 
|  | return SkISize::Make(kWidth, kHeight); | 
|  | } | 
|  |  | 
|  | virtual void onDraw(SkCanvas* canvas) SK_OVERRIDE { | 
|  | static const int kBig = 65536; | 
|  | static const SkScalar kSigma = SkBlurMask::ConvertRadiusToSigma(SkIntToScalar(4)); | 
|  |  | 
|  | const SkRect bigRect = SkRect::MakeWH(SkIntToScalar(kBig), SkIntToScalar(kBig)); | 
|  | SkRect insetRect = bigRect; | 
|  | insetRect.inset(20, 20); | 
|  |  | 
|  | SkPath rectori; | 
|  |  | 
|  | rectori.addRect(bigRect); | 
|  | rectori.addRect(insetRect, SkPath::kCCW_Direction); | 
|  |  | 
|  | // The blur extends 3*kSigma out from the big rect. | 
|  | // Offset the close-up windows so we get the entire blur | 
|  | static const SkScalar kLeftTopPad  = 3*kSigma;   // use on left & up of big rect | 
|  | static const SkScalar kRightBotPad = kCloseUpSize-3*kSigma; // use on right and bot sides | 
|  |  | 
|  | // UL hand corners of the rendered closeups | 
|  | const SkPoint origins[] = { | 
|  | { -kLeftTopPad,          -kLeftTopPad           }, // UL | 
|  | {  kBig-kRightBotPad,    -kLeftTopPad           }, // UR | 
|  | {  kBig-kRightBotPad,     kBig-kRightBotPad     }, // LR | 
|  | { -kLeftTopPad,           kBig-kRightBotPad     }, // LL | 
|  | {  kBig/2-kCloseUpSize/2, kBig/2-kCloseUpSize/2 }, // center | 
|  | }; | 
|  |  | 
|  | SkPaint outlinePaint; | 
|  | outlinePaint.setColor(SK_ColorRED); | 
|  | outlinePaint.setStyle(SkPaint::kStroke_Style); | 
|  |  | 
|  | SkPaint blurPaint; | 
|  | blurPaint.setAntiAlias(true); | 
|  | blurPaint.setColor(SK_ColorBLACK); | 
|  |  | 
|  | int desiredX = 0, desiredY = 0; | 
|  |  | 
|  | for (int i = 0; i < 2; ++i) { | 
|  | for (int j = 0; j <= kLastEnum_SkBlurStyle; ++j) { | 
|  | SkMaskFilter* mf = SkBlurMaskFilter::Create((SkBlurStyle)j, kSigma); | 
|  | blurPaint.setMaskFilter(mf)->unref(); | 
|  |  | 
|  | for (int k = 0; k < (int)SK_ARRAY_COUNT(origins); ++k) { | 
|  | canvas->save(); | 
|  |  | 
|  | SkRect clipRect = SkRect::MakeXYWH(SkIntToScalar(desiredX), | 
|  | SkIntToScalar(desiredY), | 
|  | SkIntToScalar(kCloseUpSize), | 
|  | SkIntToScalar(kCloseUpSize)); | 
|  |  | 
|  | canvas->clipRect(clipRect, SkRegion::kReplace_Op, false); | 
|  |  | 
|  | canvas->translate(desiredX-origins[k].fX, | 
|  | desiredY-origins[k].fY); | 
|  |  | 
|  | if (0 == i) { | 
|  | canvas->drawRect(bigRect, blurPaint); | 
|  | } else { | 
|  | canvas->drawPath(rectori, blurPaint); | 
|  | } | 
|  | canvas->restore(); | 
|  | canvas->drawRect(clipRect, outlinePaint); | 
|  |  | 
|  | desiredX += kCloseUpSize; | 
|  | } | 
|  |  | 
|  | desiredX = 0; | 
|  | desiredY += kCloseUpSize; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | private: | 
|  | static const int kCloseUpSize = 64; | 
|  | static const int kWidth = 5 * kCloseUpSize; | 
|  | static const int kHeight = 2 * (kLastEnum_SkBlurStyle + 1) * kCloseUpSize; | 
|  |  | 
|  | typedef GM INHERITED; | 
|  | }; | 
|  |  | 
|  | DEF_GM( return SkNEW(BigBlursGM); ) | 
|  |  | 
|  | } |