blob: 880c257006bdbcc4fd40bdf877907879e9bf602b [file] [log] [blame]
/*
* Copyright 2011 The Android Open Source Project
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "include/effects/SkImageFilters.h"
#include "include/core/SkFlattenable.h"
#include "include/core/SkImageFilter.h"
#include "include/core/SkRect.h"
#include "include/core/SkRefCnt.h"
#include "include/core/SkScalar.h"
#include "include/core/SkSize.h"
#include "include/core/SkTileMode.h"
#include "include/core/SkTypes.h"
#include "include/private/base/SkFloatingPoint.h"
#include "src/core/SkBlurEngine.h"
#include "src/core/SkImageFilterTypes.h"
#include "src/core/SkImageFilter_Base.h"
#include "src/core/SkReadBuffer.h"
#include "src/core/SkWriteBuffer.h"
#include <algorithm>
#include <optional>
#include <utility>
namespace {
class SkBlurImageFilter final : public SkImageFilter_Base {
public:
SkBlurImageFilter(SkSize sigma, sk_sp<SkImageFilter> input)
: SkImageFilter_Base(&input, 1)
, fSigma{sigma} {}
SkBlurImageFilter(SkSize sigma, SkTileMode legacyTileMode, sk_sp<SkImageFilter> input)
: SkImageFilter_Base(&input, 1)
, fSigma(sigma)
, fLegacyTileMode(legacyTileMode) {}
SkRect computeFastBounds(const SkRect&) const override;
protected:
void flatten(SkWriteBuffer&) const override;
private:
friend void ::SkRegisterBlurImageFilterFlattenable();
SK_FLATTENABLE_HOOKS(SkBlurImageFilter)
skif::FilterResult onFilterImage(const skif::Context& context) const override;
skif::LayerSpace<SkIRect> onGetInputLayerBounds(
const skif::Mapping& mapping,
const skif::LayerSpace<SkIRect>& desiredOutput,
std::optional<skif::LayerSpace<SkIRect>> contentBounds) const override;
std::optional<skif::LayerSpace<SkIRect>> onGetOutputLayerBounds(
const skif::Mapping& mapping,
std::optional<skif::LayerSpace<SkIRect>> contentBounds) const override;
skif::LayerSpace<SkSize> mapSigma(const skif::Mapping& mapping) const;
skif::LayerSpace<SkIRect> kernelBounds(const skif::Mapping& mapping,
skif::LayerSpace<SkIRect> bounds) const {
skif::LayerSpace<SkSize> sigma = this->mapSigma(mapping);
bounds.outset(skif::LayerSpace<SkSize>({3 * sigma.width(), 3 * sigma.height()}).ceil());
return bounds;
}
skif::ParameterSpace<SkSize> fSigma;
// kDecal means no legacy tiling, it will be handled by SkCropImageFilter instead. Legacy
// tiling occurs when there's no provided crop rect, and should be deleted once clients create
// their filters with defined tiling geometry.
SkTileMode fLegacyTileMode = SkTileMode::kDecal;
};
} // end namespace
sk_sp<SkImageFilter> SkImageFilters::Blur(
SkScalar sigmaX, SkScalar sigmaY, SkTileMode tileMode, sk_sp<SkImageFilter> input,
const CropRect& cropRect) {
if (!SkIsFinite(sigmaX, sigmaY) || sigmaX < 0.f || sigmaY < 0.f) {
// Non-finite or negative sigmas are error conditions. We allow 0 sigma for X and/or Y
// for 1D blurs; onFilterImage() will detect when no visible blurring would occur based on
// the Context mapping.
return nullptr;
}
// Temporarily allow tiling with no crop rect
if (tileMode != SkTileMode::kDecal && !cropRect) {
return sk_make_sp<SkBlurImageFilter>(SkSize{sigmaX, sigmaY}, tileMode, std::move(input));
}
// The 'tileMode' behavior is not well-defined if there is no crop. We only apply it if
// there is a provided 'cropRect'.
sk_sp<SkImageFilter> filter = std::move(input);
if (tileMode != SkTileMode::kDecal && cropRect) {
// Historically the input image was restricted to the cropRect when tiling was not
// kDecal, so that the kernel evaluated the tiled edge conditions, while a kDecal crop
// only affected the output.
filter = SkImageFilters::Crop(*cropRect, tileMode, std::move(filter));
}
filter = sk_make_sp<SkBlurImageFilter>(SkSize{sigmaX, sigmaY}, std::move(filter));
if (cropRect) {
// But regardless of the tileMode, the output is always decal cropped
filter = SkImageFilters::Crop(*cropRect, SkTileMode::kDecal, std::move(filter));
}
return filter;
}
void SkRegisterBlurImageFilterFlattenable() {
SK_REGISTER_FLATTENABLE(SkBlurImageFilter);
SkFlattenable::Register("SkBlurImageFilterImpl", SkBlurImageFilter::CreateProc);
}
sk_sp<SkFlattenable> SkBlurImageFilter::CreateProc(SkReadBuffer& buffer) {
SK_IMAGEFILTER_UNFLATTEN_COMMON(common, 1);
SkScalar sigmaX = buffer.readScalar();
SkScalar sigmaY = buffer.readScalar();
SkTileMode tileMode = buffer.read32LE(SkTileMode::kLastTileMode);
// NOTE: For new SKPs, 'tileMode' holds the "legacy" tile mode; any originally specified tile
// mode with valid tiling geometry is handled in the SkCropImageFilters that wrap the blur.
// In a new SKP, when 'tileMode' is not kDecal, common.cropRect() will be null and the blur
// will automatically emulate the legacy tiling.
//
// In old SKPs, the 'tileMode' and common.cropRect() may not be null. ::Blur() automatically
// detects when this is a legacy or valid tiling and constructs the DAG appropriately.
return SkImageFilters::Blur(
sigmaX, sigmaY, tileMode, common.getInput(0), common.cropRect());
}
void SkBlurImageFilter::flatten(SkWriteBuffer& buffer) const {
this->SkImageFilter_Base::flatten(buffer);
buffer.writeScalar(SkSize(fSigma).fWidth);
buffer.writeScalar(SkSize(fSigma).fHeight);
buffer.writeInt(static_cast<int>(fLegacyTileMode));
}
///////////////////////////////////////////////////////////////////////////////
namespace {
// This rather arbitrary-looking value results in a maximum box blur kernel size of 1000 pixels on
// the raster path, which matches the WebKit and Firefox implementations. Since the GPU path does
// not compute a box blur, putting the limit on sigma ensures consistent behaviour between the GPU
// and raster paths.
static constexpr SkScalar kMaxSigma = 532.f;
} // namespace
skif::FilterResult SkBlurImageFilter::onFilterImage(const skif::Context& ctx) const {
skif::Context inputCtx = ctx.withNewDesiredOutput(
this->kernelBounds(ctx.mapping(), ctx.desiredOutput()));
skif::FilterResult childOutput = this->getChildOutput(0, inputCtx);
skif::LayerSpace<SkSize> sigma = this->mapSigma(ctx.mapping());
if (sigma.width() == 0.f && sigma.height() == 0.f) {
// No actual blur, so just return the input unmodified
return childOutput;
}
SkASSERT(sigma.width() >= 0.f && sigma.width() <= kMaxSigma &&
sigma.height() >= 0.f && sigma.height() <= kMaxSigma);
// By default, FilterResult::blur() will calculate a more optimal output automatically, so
// convey the original output to it.
skif::LayerSpace<SkIRect> maxOutput = ctx.desiredOutput();
if (fLegacyTileMode != SkTileMode::kDecal) {
// Legacy tiling output is also dependent on the original child output bounds ignoring
// the tile mode's effect.
maxOutput = this->kernelBounds(ctx.mapping(), childOutput.layerBounds());
if (!maxOutput.intersect(ctx.desiredOutput())) {
return {};
}
}
if (fLegacyTileMode != SkTileMode::kDecal) {
// Legacy tiling applied to the input image when there was no explicit crop rect. Use the
// child's output image's layer bounds as the crop rectangle to adjust the edge tile mode
// without restricting the image.
childOutput = childOutput.applyCrop(inputCtx,
childOutput.layerBounds(),
fLegacyTileMode);
}
// For non-legacy tiling, 'maxOutput' is equal to the desired output. For decal's it matches
// what Builder::blur() calculates internally. For legacy tiling, however, it's dependent on
// the original child output's bounds ignoring the tile mode's effect.
skif::Context croppedOutput = ctx.withNewDesiredOutput(maxOutput);
skif::FilterResult::Builder builder{croppedOutput};
builder.add(childOutput);
return builder.blur(sigma);
}
skif::LayerSpace<SkSize> SkBlurImageFilter::mapSigma(const skif::Mapping& mapping) const {
skif::LayerSpace<SkSize> sigma = mapping.paramToLayer(fSigma);
// Clamp to the maximum sigma
sigma = skif::LayerSpace<SkSize>({std::min(sigma.width(), kMaxSigma),
std::min(sigma.height(), kMaxSigma)});
// Disable bluring on axes that are not finite, or that are small enough that the blur is
// effectively an identity.
if (!SkIsFinite(sigma.width()) || SkBlurEngine::IsEffectivelyIdentity(sigma.width())) {
sigma = skif::LayerSpace<SkSize>({0.f, sigma.height()});
}
if (!SkIsFinite(sigma.height()) || SkBlurEngine::IsEffectivelyIdentity(sigma.height())) {
sigma = skif::LayerSpace<SkSize>({sigma.width(), 0.f});
}
return sigma;
}
skif::LayerSpace<SkIRect> SkBlurImageFilter::onGetInputLayerBounds(
const skif::Mapping& mapping,
const skif::LayerSpace<SkIRect>& desiredOutput,
std::optional<skif::LayerSpace<SkIRect>> contentBounds) const {
skif::LayerSpace<SkIRect> requiredInput =
this->kernelBounds(mapping, desiredOutput);
return this->getChildInputLayerBounds(0, mapping, requiredInput, contentBounds);
}
std::optional<skif::LayerSpace<SkIRect>> SkBlurImageFilter::onGetOutputLayerBounds(
const skif::Mapping& mapping,
std::optional<skif::LayerSpace<SkIRect>> contentBounds) const {
auto childOutput = this->getChildOutputLayerBounds(0, mapping, contentBounds);
if (childOutput) {
return this->kernelBounds(mapping, *childOutput);
} else {
return skif::LayerSpace<SkIRect>::Unbounded();
}
}
SkRect SkBlurImageFilter::computeFastBounds(const SkRect& src) const {
SkRect bounds = this->getInput(0) ? this->getInput(0)->computeFastBounds(src) : src;
bounds.outset(SkSize(fSigma).width() * 3, SkSize(fSigma).height() * 3);
return bounds;
}