Chromium Code Reviews| Index: src/effects/SkAlphaThresholdFilter.cpp |
| diff --git a/src/effects/SkAlphaThresholdFilter.cpp b/src/effects/SkAlphaThresholdFilter.cpp |
| index 0b65188e83a52703974217045bfbf425d6c9b94e..7c7fcb014510016b471085144bfac9bc2d9ca054 100644 |
| --- a/src/effects/SkAlphaThresholdFilter.cpp |
| +++ b/src/effects/SkAlphaThresholdFilter.cpp |
| @@ -8,8 +8,8 @@ |
| #include "SkAlphaThresholdFilter.h" |
| #include "SkBitmap.h" |
| -#include "SkDevice.h" |
| #include "SkReadBuffer.h" |
| +#include "SkSpecialImage.h" |
| #include "SkWriteBuffer.h" |
| #include "SkRegion.h" |
| #if SK_SUPPORT_GPU |
| @@ -19,7 +19,8 @@ |
| class SK_API SkAlphaThresholdFilterImpl : public SkImageFilter { |
| public: |
| SkAlphaThresholdFilterImpl(const SkRegion& region, SkScalar innerThreshold, |
| - SkScalar outerThreshold, sk_sp<SkImageFilter> input); |
| + SkScalar outerThreshold, sk_sp<SkImageFilter> input, |
| + const CropRect* cropRect = nullptr); |
| SK_TO_STRING_OVERRIDE() |
| SK_DECLARE_PUBLIC_FLATTENABLE_DESERIALIZATION_PROCS(SkAlphaThresholdFilterImpl) |
| @@ -28,11 +29,11 @@ public: |
| protected: |
| void flatten(SkWriteBuffer&) const override; |
| - bool onFilterImageDeprecated(Proxy*, const SkBitmap& src, const Context&, |
| - SkBitmap* result, SkIPoint* offset) const override; |
| + sk_sp<SkSpecialImage> onFilterImage(SkSpecialImage* source, const Context&, |
| + SkIPoint* offset) const override; |
| + |
| #if SK_SUPPORT_GPU |
| - bool asFragmentProcessor(GrFragmentProcessor**, GrTexture*, const SkMatrix&, |
| - const SkIRect& bounds) const override; |
| + sk_sp<GrTexture> createMaskTexture(GrContext*, const SkMatrix&, const SkIRect& bounds) const; |
| #endif |
| private: |
| @@ -53,14 +54,17 @@ static SkScalar pin_0_1(SkScalar x) { |
| sk_sp<SkImageFilter> SkAlphaThresholdFilter::Make(const SkRegion& region, |
| SkScalar innerThreshold, |
| SkScalar outerThreshold, |
| - sk_sp<SkImageFilter> input) { |
| + sk_sp<SkImageFilter> input, |
| + const SkImageFilter::CropRect* cropRect) { |
| innerThreshold = pin_0_1(innerThreshold); |
| outerThreshold = pin_0_1(outerThreshold); |
| if (!SkScalarIsFinite(innerThreshold) || !SkScalarIsFinite(outerThreshold)) { |
| return nullptr; |
| } |
| return sk_sp<SkImageFilter>(new SkAlphaThresholdFilterImpl(region, innerThreshold, |
| - outerThreshold, std::move(input))); |
| + outerThreshold, |
| + std::move(input), |
|
Stephen White
2016/04/12 16:34:32
Nit: std::move() here seems to be overkill. Unless
robertphillips
2016/04/12 16:55:28
Yeah - this seems to be SOP for now. There is some
|
| + cropRect)); |
| } |
| #if SK_SUPPORT_GPU |
| @@ -277,60 +281,52 @@ sk_sp<SkFlattenable> SkAlphaThresholdFilterImpl::CreateProc(SkReadBuffer& buffer |
| SkAlphaThresholdFilterImpl::SkAlphaThresholdFilterImpl(const SkRegion& region, |
| SkScalar innerThreshold, |
| SkScalar outerThreshold, |
| - sk_sp<SkImageFilter> input) |
| - : INHERITED(&input, 1, nullptr) |
| + sk_sp<SkImageFilter> input, |
| + const CropRect* cropRect) |
| + : INHERITED(&input, 1, cropRect) |
| , fRegion(region) |
| , fInnerThreshold(innerThreshold) |
| , fOuterThreshold(outerThreshold) { |
| } |
| #if SK_SUPPORT_GPU |
| -bool SkAlphaThresholdFilterImpl::asFragmentProcessor(GrFragmentProcessor** fp, |
| - GrTexture* texture, |
| - const SkMatrix& inMatrix, |
| - const SkIRect& bounds) const { |
| - if (fp) { |
| - GrContext* context = texture->getContext(); |
| - GrSurfaceDesc maskDesc; |
| - if (context->caps()->isConfigRenderable(kAlpha_8_GrPixelConfig, false)) { |
| - maskDesc.fConfig = kAlpha_8_GrPixelConfig; |
| - } else { |
| - maskDesc.fConfig = kRGBA_8888_GrPixelConfig; |
| - } |
| - maskDesc.fFlags = kRenderTarget_GrSurfaceFlag; |
| - // Add one pixel of border to ensure that clamp mode will be all zeros |
| - // the outside. |
| - maskDesc.fWidth = bounds.width(); |
| - maskDesc.fHeight = bounds.height(); |
| - SkAutoTUnref<GrTexture> maskTexture( |
| - context->textureProvider()->createApproxTexture(maskDesc)); |
| - if (!maskTexture) { |
| - return false; |
| - } |
| +sk_sp<GrTexture> SkAlphaThresholdFilterImpl::createMaskTexture(GrContext* context, |
| + const SkMatrix& inMatrix, |
| + const SkIRect& bounds) const { |
| + GrSurfaceDesc maskDesc; |
| + if (context->caps()->isConfigRenderable(kAlpha_8_GrPixelConfig, false)) { |
| + maskDesc.fConfig = kAlpha_8_GrPixelConfig; |
| + } else { |
| + maskDesc.fConfig = kRGBA_8888_GrPixelConfig; |
| + } |
| + maskDesc.fFlags = kRenderTarget_GrSurfaceFlag; |
| + // Add one pixel of border to ensure that clamp mode will be all zeros |
| + // the outside. |
| + maskDesc.fWidth = bounds.width(); |
| + maskDesc.fHeight = bounds.height(); |
| + sk_sp<GrTexture> maskTexture(context->textureProvider()->createApproxTexture(maskDesc)); |
| + if (!maskTexture) { |
| + return nullptr; |
| + } |
| - SkAutoTUnref<GrDrawContext> drawContext( |
| - context->drawContext(maskTexture->asRenderTarget())); |
| - if (drawContext) { |
| - GrPaint grPaint; |
| - grPaint.setPorterDuffXPFactory(SkXfermode::kSrc_Mode); |
| - SkRegion::Iterator iter(fRegion); |
| - drawContext->clear(nullptr, 0x0, true); |
| - |
| - GrClip clip(SkRect::Make(SkIRect::MakeWH(bounds.width(), bounds.height()))); |
| - while (!iter.done()) { |
| - SkRect rect = SkRect::Make(iter.rect()); |
| - drawContext->drawRect(clip, grPaint, inMatrix, rect); |
| - iter.next(); |
| - } |
| - } |
| + sk_sp<GrDrawContext> drawContext(context->drawContext(maskTexture->asRenderTarget())); |
| + if (!drawContext) { |
| + return nullptr; |
| + } |
| - *fp = AlphaThresholdEffect::Create(texture, |
| - maskTexture, |
| - fInnerThreshold, |
| - fOuterThreshold, |
| - bounds); |
| + GrPaint grPaint; |
| + grPaint.setPorterDuffXPFactory(SkXfermode::kSrc_Mode); |
| + SkRegion::Iterator iter(fRegion); |
| + drawContext->clear(nullptr, 0x0, true); |
| + |
| + GrClip clip(SkRect::Make(SkIRect::MakeWH(bounds.width(), bounds.height()))); |
| + while (!iter.done()) { |
| + SkRect rect = SkRect::Make(iter.rect()); |
| + drawContext->drawRect(clip, grPaint, inMatrix, rect); |
| + iter.next(); |
| } |
| - return true; |
| + |
| + return maskTexture; |
| } |
| #endif |
| @@ -341,50 +337,109 @@ void SkAlphaThresholdFilterImpl::flatten(SkWriteBuffer& buffer) const { |
| buffer.writeRegion(fRegion); |
| } |
| -bool SkAlphaThresholdFilterImpl::onFilterImageDeprecated(Proxy* proxy, const SkBitmap& src, |
| - const Context& ctx, SkBitmap* dst, |
| - SkIPoint* offset) const { |
| +sk_sp<SkSpecialImage> SkAlphaThresholdFilterImpl::onFilterImage(SkSpecialImage* source, |
| + const Context& ctx, |
| + SkIPoint* offset) const { |
| + SkIPoint inputOffset = SkIPoint::Make(0, 0); |
| + sk_sp<SkSpecialImage> input(this->filterInput(0, source, ctx, &inputOffset)); |
| + if (!input) { |
| + return nullptr; |
| + } |
| + |
| + const SkIRect inputBounds = SkIRect::MakeXYWH(inputOffset.x(), inputOffset.y(), |
| + input->width(), input->height()); |
| + |
| + SkIRect bounds; |
| + if (!this->applyCropRect(ctx, inputBounds, &bounds)) { |
| + return nullptr; |
| + } |
| + |
| +#if SK_SUPPORT_GPU |
| + if (source->isTextureBacked()) { |
| + GrContext* context = source->getContext(); |
| + |
| + sk_sp<GrTexture> inputTexture(input->asTextureRef(context)); |
| + SkASSERT(inputTexture); |
| + |
| + offset->fX = bounds.left(); |
| + offset->fY = bounds.top(); |
| - if (src.colorType() != kN32_SkColorType) { |
| - return false; |
| + bounds.offset(-inputOffset); |
| + |
| + SkMatrix matrix(ctx.ctm()); |
| + matrix.postTranslate(SkIntToScalar(-bounds.left()), SkIntToScalar(-bounds.top())); |
| + |
| + sk_sp<GrTexture> maskTexture(this->createMaskTexture(context, matrix, bounds)); |
| + if (!maskTexture) { |
| + return nullptr; |
| + } |
| + |
| + // SRGBTODO: handle sRGB here |
| + sk_sp<GrFragmentProcessor> fp(AlphaThresholdEffect::Create(inputTexture.get(), |
| + maskTexture.get(), |
| + fInnerThreshold, |
| + fOuterThreshold, |
| + bounds)); |
| + if (!fp) { |
| + return nullptr; |
| + } |
| + |
| + return DrawWithFP(context, std::move(fp), bounds, source->internal_getProxy()); |
|
Stephen White
2016/04/12 16:34:32
Much nicer, thanks!
(Again, I'm not a fan with pe
robertphillips
2016/04/12 16:55:28
Acknowledged.
|
| + } |
| +#endif |
| + |
| + SkBitmap inputBM; |
| + |
| + if (!input->getROPixels(&inputBM)) { |
| + return nullptr; |
| } |
| + if (inputBM.colorType() != kN32_SkColorType) { |
| + return nullptr; |
| + } |
| + |
| + SkAutoLockPixels inputLock(inputBM); |
| + |
| + if (!inputBM.getPixels() || inputBM.width() <= 0 || inputBM.height() <= 0) { |
| + return nullptr; |
| + } |
| + |
| + |
| SkMatrix localInverse; |
| if (!ctx.ctm().invert(&localInverse)) { |
| - return false; |
| + return nullptr; |
| } |
| - SkAutoLockPixels alp(src); |
| - SkASSERT(src.getPixels()); |
| - if (!src.getPixels() || src.width() <= 0 || src.height() <= 0) { |
| - return false; |
| - } |
| + SkImageInfo info = SkImageInfo::MakeN32(bounds.width(), bounds.height(), |
| + inputBM.alphaType()); |
| - SkAutoTUnref<SkBaseDevice> device(proxy->createDevice(src.width(), src.height())); |
| - if (!device) { |
| - return false; |
| + SkBitmap dst; |
| + if (!dst.tryAllocPixels(info)) { |
| + return nullptr; |
| } |
| - *dst = device->accessBitmap(false); |
| - SkAutoLockPixels alp_dst(*dst); |
| + |
| + SkAutoLockPixels dstLock(dst); |
| U8CPU innerThreshold = (U8CPU)(fInnerThreshold * 0xFF); |
| U8CPU outerThreshold = (U8CPU)(fOuterThreshold * 0xFF); |
| - SkColor* sptr = src.getAddr32(0, 0); |
| - SkColor* dptr = dst->getAddr32(0, 0); |
| - int width = src.width(), height = src.height(); |
| - for (int y = 0; y < height; ++y) { |
| - for (int x = 0; x < width; ++x) { |
| - const SkColor& source = sptr[y * width + x]; |
| - SkColor output_color(source); |
| + SkColor* dptr = dst.getAddr32(0, 0); |
| + int dstWidth = dst.width(), dstHeight = dst.height(); |
| + for (int y = 0; y < dstHeight; ++y) { |
| + const SkColor* sptr = inputBM.getAddr32(bounds.fLeft, bounds.fTop+y); |
| + |
| + for (int x = 0; x < dstWidth; ++x) { |
| + const SkColor& source = sptr[x]; |
| + SkColor outputColor(source); |
| SkPoint position; |
| - localInverse.mapXY((SkScalar)x, (SkScalar)y, &position); |
| + localInverse.mapXY((SkScalar)x + bounds.fLeft, (SkScalar)y + bounds.fTop, &position); |
| if (fRegion.contains((int32_t)position.x(), (int32_t)position.y())) { |
| if (SkColorGetA(source) < innerThreshold) { |
| U8CPU alpha = SkColorGetA(source); |
| - if (alpha == 0) |
| + if (alpha == 0) { |
| alpha = 1; |
| + } |
| float scale = (float)innerThreshold / alpha; |
| - output_color = SkColorSetARGB(innerThreshold, |
| + outputColor = SkColorSetARGB(innerThreshold, |
| (U8CPU)(SkColorGetR(source) * scale), |
| (U8CPU)(SkColorGetG(source) * scale), |
| (U8CPU)(SkColorGetB(source) * scale)); |
| @@ -392,17 +447,21 @@ bool SkAlphaThresholdFilterImpl::onFilterImageDeprecated(Proxy* proxy, const SkB |
| } else { |
| if (SkColorGetA(source) > outerThreshold) { |
| float scale = (float)outerThreshold / SkColorGetA(source); |
| - output_color = SkColorSetARGB(outerThreshold, |
| + outputColor = SkColorSetARGB(outerThreshold, |
| (U8CPU)(SkColorGetR(source) * scale), |
| (U8CPU)(SkColorGetG(source) * scale), |
| (U8CPU)(SkColorGetB(source) * scale)); |
| } |
| } |
| - dptr[y * dst->width() + x] = output_color; |
| + dptr[y * dstWidth + x] = outputColor; |
| } |
| } |
| - return true; |
| + offset->fX = bounds.left(); |
| + offset->fY = bounds.top(); |
| + return SkSpecialImage::MakeFromRaster(source->internal_getProxy(), |
| + SkIRect::MakeWH(bounds.width(), bounds.height()), |
| + dst); |
| } |
| #ifndef SK_IGNORE_TO_STRING |