| /* |
| * 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 "include/core/SkBlendMode.h" |
| #include "include/core/SkCanvas.h" |
| #include "include/core/SkFlattenable.h" |
| #include "include/core/SkImage.h" |
| #include "include/core/SkImageFilter.h" |
| #include "include/core/SkMatrix.h" |
| #include "include/core/SkPaint.h" |
| #include "include/core/SkPoint.h" |
| #include "include/core/SkRect.h" |
| #include "include/core/SkRefCnt.h" |
| #include "include/core/SkSamplingOptions.h" |
| #include "include/core/SkScalar.h" |
| #include "include/core/SkSurface.h" |
| #include "include/core/SkTileMode.h" |
| #include "include/core/SkTypes.h" |
| #include "include/effects/SkImageFilters.h" |
| #include "src/core/SkImageFilterTypes.h" |
| #include "src/core/SkImageFilter_Base.h" |
| #include "src/core/SkReadBuffer.h" |
| #include "src/core/SkSpecialImage.h" |
| #include "src/core/SkSpecialSurface.h" |
| #include "src/core/SkValidationUtils.h" |
| #include "src/core/SkWriteBuffer.h" |
| |
| #include <utility> |
| |
| namespace { |
| |
| class SkTileImageFilter final : public SkImageFilter_Base { |
| public: |
| SkTileImageFilter(const SkRect& srcRect, const SkRect& dstRect, sk_sp<SkImageFilter> input) |
| : INHERITED(&input, 1, nullptr) |
| , fSrcRect(srcRect) |
| , fDstRect(dstRect) {} |
| |
| SkIRect onFilterBounds(const SkIRect& src, const SkMatrix& ctm, |
| MapDirection, const SkIRect* inputRect) const override; |
| SkIRect onFilterNodeBounds(const SkIRect&, const SkMatrix& ctm, |
| MapDirection, const SkIRect* inputRect) const override; |
| SkRect computeFastBounds(const SkRect& src) const override; |
| |
| protected: |
| void flatten(SkWriteBuffer& buffer) const override; |
| |
| sk_sp<SkSpecialImage> onFilterImage(const Context&, SkIPoint* offset) const override; |
| |
| private: |
| friend void ::SkRegisterTileImageFilterFlattenable(); |
| SK_FLATTENABLE_HOOKS(SkTileImageFilter) |
| |
| SkRect fSrcRect; |
| SkRect fDstRect; |
| |
| using INHERITED = SkImageFilter_Base; |
| }; |
| |
| } // end namespace |
| |
| |
| sk_sp<SkImageFilter> SkImageFilters::Tile(const SkRect& src, |
| const SkRect& dst, |
| sk_sp<SkImageFilter> input) { |
| if (!SkIsValidRect(src) || !SkIsValidRect(dst)) { |
| return nullptr; |
| } |
| if (src.width() == dst.width() && src.height() == dst.height()) { |
| SkRect ir = dst; |
| if (!ir.intersect(src)) { |
| return input; |
| } |
| return SkImageFilters::Offset(dst.x() - src.x(), dst.y() - src.y(), |
| std::move(input), &ir); |
| } |
| return sk_sp<SkImageFilter>(new SkTileImageFilter(src, dst, std::move(input))); |
| } |
| |
| void SkRegisterTileImageFilterFlattenable() { |
| SK_REGISTER_FLATTENABLE(SkTileImageFilter); |
| // TODO (michaelludwig) - Remove after grace period for SKPs to stop using old name |
| SkFlattenable::Register("SkTileImageFilterImpl", SkTileImageFilter::CreateProc); |
| } |
| |
| sk_sp<SkFlattenable> SkTileImageFilter::CreateProc(SkReadBuffer& buffer) { |
| SK_IMAGEFILTER_UNFLATTEN_COMMON(common, 1); |
| SkRect src, dst; |
| buffer.readRect(&src); |
| buffer.readRect(&dst); |
| return SkImageFilters::Tile(src, dst, common.getInput(0)); |
| } |
| |
| void SkTileImageFilter::flatten(SkWriteBuffer& buffer) const { |
| this->INHERITED::flatten(buffer); |
| buffer.writeRect(fSrcRect); |
| buffer.writeRect(fDstRect); |
| } |
| |
| /////////////////////////////////////////////////////////////////////////////////////////////////// |
| |
| sk_sp<SkSpecialImage> SkTileImageFilter::onFilterImage(const Context& ctx, |
| SkIPoint* offset) const { |
| SkIPoint inputOffset = SkIPoint::Make(0, 0); |
| sk_sp<SkSpecialImage> input(this->filterInput(0, ctx, &inputOffset)); |
| if (!input) { |
| return nullptr; |
| } |
| |
| SkRect dstRect; |
| ctx.ctm().mapRect(&dstRect, fDstRect); |
| if (!dstRect.intersect(SkRect::Make(ctx.clipBounds()))) { |
| return nullptr; |
| } |
| |
| const SkIRect dstIRect = skif::RoundOut(dstRect); |
| if (!fSrcRect.width() || !fSrcRect.height() || !dstIRect.width() || !dstIRect.height()) { |
| return nullptr; |
| } |
| |
| SkRect srcRect; |
| ctx.ctm().mapRect(&srcRect, fSrcRect); |
| SkIRect srcIRect = skif::RoundOut(srcRect); |
| srcIRect.offset(-inputOffset); |
| const SkIRect inputBounds = SkIRect::MakeWH(input->width(), input->height()); |
| |
| if (!SkIRect::Intersects(srcIRect, inputBounds)) { |
| return nullptr; |
| } |
| |
| // We create an SkImage here b.c. it needs to be a tight fit for the tiling |
| sk_sp<SkImage> subset; |
| if (inputBounds.contains(srcIRect)) { |
| subset = input->asImage(&srcIRect); |
| } else { |
| sk_sp<SkSurface> surf(input->makeTightSurface(ctx.colorType(), ctx.colorSpace(), |
| srcIRect.size())); |
| if (!surf) { |
| return nullptr; |
| } |
| |
| SkCanvas* canvas = surf->getCanvas(); |
| SkASSERT(canvas); |
| |
| SkPaint paint; |
| paint.setBlendMode(SkBlendMode::kSrc); |
| |
| input->draw(canvas, |
| SkIntToScalar(inputOffset.x()), SkIntToScalar(inputOffset.y()), |
| SkSamplingOptions(), &paint); |
| |
| subset = surf->makeImageSnapshot(); |
| } |
| if (!subset) { |
| return nullptr; |
| } |
| SkASSERT(subset->width() == srcIRect.width()); |
| SkASSERT(subset->height() == srcIRect.height()); |
| |
| sk_sp<SkSpecialSurface> surf(ctx.makeSurface(dstIRect.size())); |
| if (!surf) { |
| return nullptr; |
| } |
| |
| SkCanvas* canvas = surf->getCanvas(); |
| SkASSERT(canvas); |
| |
| SkPaint paint; |
| paint.setBlendMode(SkBlendMode::kSrc); |
| paint.setShader(subset->makeShader(SkTileMode::kRepeat, SkTileMode::kRepeat, |
| SkSamplingOptions())); |
| canvas->translate(-dstRect.fLeft, -dstRect.fTop); |
| canvas->drawRect(dstRect, paint); |
| offset->fX = dstIRect.fLeft; |
| offset->fY = dstIRect.fTop; |
| return surf->makeImageSnapshot(); |
| } |
| |
| SkIRect SkTileImageFilter::onFilterNodeBounds( |
| const SkIRect& src, const SkMatrix& ctm, MapDirection dir, const SkIRect* inputRect) const { |
| SkRect rect = kReverse_MapDirection == dir ? fSrcRect : fDstRect; |
| ctm.mapRect(&rect); |
| return rect.roundOut(); |
| } |
| |
| SkIRect SkTileImageFilter::onFilterBounds(const SkIRect& src, const SkMatrix&, |
| MapDirection, const SkIRect* inputRect) const { |
| // Don't recurse into inputs. |
| return src; |
| } |
| |
| SkRect SkTileImageFilter::computeFastBounds(const SkRect& src) const { |
| return fDstRect; |
| } |