--- /dev/null
+
+/*
+ * 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 make_isize(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 < SkBlurMaskFilter::kBlurStyleCount; ++j) {
+ SkMaskFilter* mf = SkBlurMaskFilter::Create((SkBlurMaskFilter::BlurStyle)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 * SkBlurMaskFilter::kBlurStyleCount * kCloseUpSize;
+
+ typedef GM INHERITED;
+};
+
+DEF_GM( return SkNEW(BigBlursGM); )
+
+}
\ No newline at end of file