aboutsummaryrefslogtreecommitdiffhomepage
diff options
context:
space:
mode:
authorGravatar reed@google.com <reed@google.com@2bbb7eff-a529-9590-31e7-b0007b416f81>2012-11-19 16:45:14 +0000
committerGravatar reed@google.com <reed@google.com@2bbb7eff-a529-9590-31e7-b0007b416f81>2012-11-19 16:45:14 +0000
commitdab9b4fe035c1e8a79e110139953c19bd48d66f9 (patch)
tree3704060c53f91edf25f130a5ae5601f16d3743ca
parent4f65a77a1f90f3a52d4da2ed72ca4ea8cb088a16 (diff)
use SkPath::isNestedRects() to apply blurred nine-patch
Review URL: https://codereview.appspot.com/6855063 git-svn-id: http://skia.googlecode.com/svn/trunk@6483 2bbb7eff-a529-9590-31e7-b0007b416f81
-rw-r--r--include/core/SkMaskFilter.h14
-rw-r--r--src/core/SkMaskFilter.cpp56
-rw-r--r--src/effects/SkBlurMaskFilter.cpp105
3 files changed, 114 insertions, 61 deletions
diff --git a/include/core/SkMaskFilter.h b/include/core/SkMaskFilter.h
index 6bdc5335e8..3060e2f542 100644
--- a/include/core/SkMaskFilter.h
+++ b/include/core/SkMaskFilter.h
@@ -111,6 +111,12 @@ protected:
kUnimplemented_FilterReturn
};
+ struct NinePatch {
+ SkMask fMask; // fBounds must have [0,0] in its top-left
+ SkIRect fOuterRect; // width/height must be >= fMask.fBounds'
+ SkIPoint fCenter; // identifies center row/col for stretching
+ };
+
/**
* Override if your subclass can filter a rect, and return the answer as
* a ninepatch mask to be stretched over the returned outerRect. On success
@@ -126,10 +132,10 @@ protected:
* the caller will call mask.fBounds.centerX() and centerY() to find the
* strips that will be replicated.
*/
- virtual FilterReturn filterRectToNine(const SkRect&, const SkMatrix&,
- const SkIRect& clipBounds,
- SkMask* ninePatchMask,
- SkIRect* outerRect);
+ virtual FilterReturn filterRectsToNine(const SkRect[], int count,
+ const SkMatrix&,
+ const SkIRect& clipBounds,
+ NinePatch*);
private:
friend class SkDraw;
diff --git a/src/core/SkMaskFilter.cpp b/src/core/SkMaskFilter.cpp
index 90925f1ff0..e02717824e 100644
--- a/src/core/SkMaskFilter.cpp
+++ b/src/core/SkMaskFilter.cpp
@@ -66,9 +66,10 @@ static void dump(const SkMask& mask) {
#endif
static void draw_nine_clipped(const SkMask& mask, const SkIRect& outerR,
+ const SkIPoint& center, bool fillCenter,
const SkIRect& clipR, SkBlitter* blitter) {
- int cx = mask.fBounds.centerX();
- int cy = mask.fBounds.centerY();
+ int cx = center.x();
+ int cy = center.y();
SkMask m;
// top-left
@@ -109,7 +110,9 @@ static void draw_nine_clipped(const SkMask& mask, const SkIRect& outerR,
outerR.top() + cy - mask.fBounds.top(),
outerR.right() + (cx + 1 - mask.fBounds.right()),
outerR.bottom() + (cy + 1 - mask.fBounds.bottom()));
- blitClippedRect(blitter, innerR, clipR);
+ if (fillCenter) {
+ blitClippedRect(blitter, innerR, clipR);
+ }
const int innerW = innerR.width();
size_t storageSize = (innerW + 1) * (sizeof(int16_t) + sizeof(uint8_t));
@@ -169,6 +172,7 @@ static void draw_nine_clipped(const SkMask& mask, const SkIRect& outerR,
}
static void draw_nine(const SkMask& mask, const SkIRect& outerR,
+ const SkIPoint& center, bool fillCenter,
const SkRasterClip& clip, SkBounder* bounder,
SkBlitter* blitter) {
// if we get here, we need to (possibly) resolve the clip and blitter
@@ -180,33 +184,45 @@ static void draw_nine(const SkMask& mask, const SkIRect& outerR,
if (!clipper.done() && (!bounder || bounder->doIRect(outerR))) {
const SkIRect& cr = clipper.rect();
do {
- draw_nine_clipped(mask, outerR, cr, blitter);
+ draw_nine_clipped(mask, outerR, center, fillCenter, cr, blitter);
clipper.next();
} while (!clipper.done());
}
}
+static int countNestedRects(const SkPath& path, SkRect rects[2]) {
+ if (path.isNestedRects(rects)) {
+ return 2;
+ }
+ return path.isRect(&rects[0]);
+}
+
bool SkMaskFilter::filterPath(const SkPath& devPath, const SkMatrix& matrix,
const SkRasterClip& clip, SkBounder* bounder,
SkBlitter* blitter, SkPaint::Style style) {
- SkRect rect;
- if (!SK_IGNORE_FAST_BLURRECT &&
- devPath.isRect(&rect) && SkPaint::kFill_Style == style) {
- SkMask mask;
- SkIRect outerBounds;
-
- mask.fImage = NULL;
- switch (this->filterRectToNine(rect, matrix, clip.getBounds(), &mask,
- &outerBounds)) {
+ SkRect rects[2];
+ int rectCount = 0;
+ if (!SK_IGNORE_FAST_BLURRECT && SkPaint::kFill_Style == style) {
+ rectCount = countNestedRects(devPath, rects);
+ }
+ if (rectCount > 0) {
+ NinePatch patch;
+
+ patch.fMask.fImage = NULL;
+ switch (this->filterRectsToNine(rects, rectCount, matrix,
+ clip.getBounds(), &patch)) {
case kFalse_FilterReturn:
- SkASSERT(!mask.fImage);
+ SkASSERT(NULL == patch.fMask.fImage);
return false;
+
case kTrue_FilterReturn:
- draw_nine(mask, outerBounds, clip, bounder, blitter);
- SkMask::FreeImage(mask.fImage);
+ draw_nine(patch.fMask, patch.fOuterRect, patch.fCenter,
+ 1 == rectCount, clip, bounder, blitter);
+ SkMask::FreeImage(patch.fMask.fImage);
return true;
+
case kUnimplemented_FilterReturn:
- SkASSERT(!mask.fImage);
+ SkASSERT(NULL == patch.fMask.fImage);
// fall through
break;
}
@@ -244,10 +260,8 @@ bool SkMaskFilter::filterPath(const SkPath& devPath, const SkMatrix& matrix,
}
SkMaskFilter::FilterReturn
-SkMaskFilter::filterRectToNine(const SkRect&, const SkMatrix&,
- const SkIRect& clipBounds,
- SkMask* ninePatchMask,
- SkIRect* outerRect) {
+SkMaskFilter::filterRectsToNine(const SkRect[], int count, const SkMatrix&,
+ const SkIRect& clipBounds, NinePatch*) {
return kUnimplemented_FilterReturn;
}
diff --git a/src/effects/SkBlurMaskFilter.cpp b/src/effects/SkBlurMaskFilter.cpp
index c42bd8c9ed..467af23204 100644
--- a/src/effects/SkBlurMaskFilter.cpp
+++ b/src/effects/SkBlurMaskFilter.cpp
@@ -27,10 +27,9 @@ public:
SK_DECLARE_PUBLIC_FLATTENABLE_DESERIALIZATION_PROCS(SkBlurMaskFilterImpl)
protected:
- virtual FilterReturn filterRectToNine(const SkRect&, const SkMatrix&,
- const SkIRect& clipBounds,
- SkMask* ninePatchMask,
- SkIRect* outerRect) SK_OVERRIDE;
+ virtual FilterReturn filterRectsToNine(const SkRect[], int count, const SkMatrix&,
+ const SkIRect& clipBounds,
+ NinePatch*) SK_OVERRIDE;
private:
SkScalar fRadius;
@@ -104,8 +103,8 @@ bool SkBlurMaskFilterImpl::filterMask(SkMask* dst, const SkMask& src,
#include "SkCanvas.h"
-static bool drawRectIntoMask(const SkRect& r, SkMask* mask) {
- r.roundOut(&mask->fBounds);
+static bool drawRectsIntoMask(const SkRect rects[], int count, SkMask* mask) {
+ rects[0].roundOut(&mask->fBounds);
mask->fRowBytes = SkAlign4(mask->fBounds.width());
mask->fFormat = SkMask::kA8_Format;
size_t size = mask->computeImageSize();
@@ -122,24 +121,37 @@ static bool drawRectIntoMask(const SkRect& r, SkMask* mask) {
bitmap.setPixels(mask->fImage);
SkCanvas canvas(bitmap);
- canvas.translate(-SkScalarFloorToScalar(r.left()),
- -SkScalarFloorToScalar(r.top()));
+ canvas.translate(-SkIntToScalar(mask->fBounds.left()),
+ -SkIntToScalar(mask->fBounds.top()));
SkPaint paint;
paint.setAntiAlias(true);
- canvas.drawRect(r, paint);
+ if (1 == count) {
+ canvas.drawRect(rects[0], paint);
+ } else {
+ // todo: do I need a fast way to do this?
+ SkPath path;
+ path.addRect(rects[0]);
+ path.addRect(rects[1]);
+ path.setFillType(SkPath::kEvenOdd_FillType);
+ canvas.drawPath(path, paint);
+ }
return true;
}
SkMaskFilter::FilterReturn
-SkBlurMaskFilterImpl::filterRectToNine(const SkRect& rect, const SkMatrix& matrix,
- const SkIRect& clipBounds,
- SkMask* ninePatchMask,
- SkIRect* outerRect) {
+SkBlurMaskFilterImpl::filterRectsToNine(const SkRect rects[], int count,
+ const SkMatrix& matrix,
+ const SkIRect& clipBounds,
+ NinePatch* patch) {
+ if (count < 1 || count > 2) {
+ return kUnimplemented_FilterReturn;
+ }
+
SkIPoint margin;
SkMask srcM, dstM;
- rect.roundOut(&srcM.fBounds);
+ rects[0].roundOut(&srcM.fBounds);
srcM.fImage = NULL;
srcM.fFormat = SkMask::kA8_Format;
srcM.fRowBytes = 0;
@@ -161,35 +173,56 @@ SkBlurMaskFilterImpl::filterRectToNine(const SkRect& rect, const SkMatrix& matri
* Thus, in this case, we inset by a total of 5 (on each side) beginning
* with our outer-rect (dstM.fBounds)
*/
- SkRect smallR = rect;
- {
- // +3 is from +1 for each edge (to account for possible fractional pixel
- // edges, and +1 to make room for a center rol/col.
- int smallW = dstM.fBounds.width() - srcM.fBounds.width() + 3;
- int smallH = dstM.fBounds.height() - srcM.fBounds.height() + 3;
- // we want the inset amounts to be integral, so we don't change any
- // fractional phase on the fRight or fBottom of our smallR.
- SkScalar dx = SkIntToScalar(srcM.fBounds.width() - smallW);
- SkScalar dy = SkIntToScalar(srcM.fBounds.height() - smallH);
- if (dx < 0 || dy < 0) {
- // we're too small, relative to our blur, to break into nine-patch,
- // so we ask to have our normal filterMask() be called.
- return kUnimplemented_FilterReturn;
- }
- SkASSERT(dx >= 0 && dy >= 0);
- smallR.set(rect.left(), rect.top(), rect.right() - dx, rect.bottom() - dy);
- SkASSERT(!smallR.isEmpty());
+ SkRect smallR[2];
+ SkIPoint center;
+
+ // +2 is from +1 for each edge (to account for possible fractional edges
+ int smallW = dstM.fBounds.width() - srcM.fBounds.width() + 2;
+ int smallH = dstM.fBounds.height() - srcM.fBounds.height() + 2;
+ SkIRect innerIR;
+
+ if (1 == count) {
+ innerIR = srcM.fBounds;
+ center.set(smallW, smallH);
+ } else {
+ SkASSERT(2 == count);
+ rects[1].roundIn(&innerIR);
+ center.set(smallW + (innerIR.left() - srcM.fBounds.left()),
+ smallH + (innerIR.top() - srcM.fBounds.top()));
+ }
+
+ // +1 so we get a clean, stretchable, center row/col
+ smallW += 1;
+ smallH += 1;
+
+ // we want the inset amounts to be integral, so we don't change any
+ // fractional phase on the fRight or fBottom of our smallR.
+ const SkScalar dx = SkIntToScalar(innerIR.width() - smallW);
+ const SkScalar dy = SkIntToScalar(innerIR.height() - smallH);
+ if (dx < 0 || dy < 0) {
+ // we're too small, relative to our blur, to break into nine-patch,
+ // so we ask to have our normal filterMask() be called.
+ return kUnimplemented_FilterReturn;
+ }
+
+ smallR[0].set(rects[0].left(), rects[0].top(), rects[0].right() - dx, rects[0].bottom() - dy);
+ SkASSERT(!smallR[0].isEmpty());
+ if (2 == count) {
+ smallR[1].set(rects[1].left(), rects[1].top(),
+ rects[1].right() - dx, rects[1].bottom() - dy);
+ SkASSERT(!smallR[1].isEmpty());
}
- if (!drawRectIntoMask(smallR, &srcM)) {
+ if (!drawRectsIntoMask(smallR, count, &srcM)) {
return kFalse_FilterReturn;
}
- if (!this->filterMask(ninePatchMask, srcM, matrix, &margin)) {
+ if (!this->filterMask(&patch->fMask, srcM, matrix, &margin)) {
return kFalse_FilterReturn;
}
- ninePatchMask->fBounds.offsetTo(0, 0);
- *outerRect = dstM.fBounds;
+ patch->fMask.fBounds.offsetTo(0, 0);
+ patch->fOuterRect = dstM.fBounds;
+ patch->fCenter = center;
return kTrue_FilterReturn;
}