blob: fed94c718bad71c28bbd7d1adc057f913226fb83 [file] [log] [blame]
// Copyright 2017 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "cc/paint/paint_filter.h"
#include "cc/paint/filter_operations.h"
#include "cc/paint/paint_image_builder.h"
#include "cc/paint/paint_op_writer.h"
#include "cc/paint/paint_record.h"
#include "third_party/skia/include/core/SkColorFilter.h"
#include "third_party/skia/include/core/SkMath.h"
#include "third_party/skia/include/effects/SkAlphaThresholdFilter.h"
#include "third_party/skia/include/effects/SkColorFilterImageFilter.h"
#include "third_party/skia/include/effects/SkComposeImageFilter.h"
#include "third_party/skia/include/effects/SkImageSource.h"
#include "third_party/skia/include/effects/SkLightingImageFilter.h"
#include "third_party/skia/include/effects/SkMagnifierImageFilter.h"
#include "third_party/skia/include/effects/SkMergeImageFilter.h"
#include "third_party/skia/include/effects/SkMorphologyImageFilter.h"
#include "third_party/skia/include/effects/SkOffsetImageFilter.h"
#include "third_party/skia/include/effects/SkPaintImageFilter.h"
#include "third_party/skia/include/effects/SkPerlinNoiseShader.h"
#include "third_party/skia/include/effects/SkPictureImageFilter.h"
#include "third_party/skia/include/effects/SkTileImageFilter.h"
#include "third_party/skia/include/effects/SkXfermodeImageFilter.h"
namespace cc {
namespace {
const bool kHasNoDiscardableImages = false;
bool AreFiltersEqual(const PaintFilter* one, const PaintFilter* two) {
if (!one || !two)
return !one && !two;
return *one == *two;
}
bool AreScalarsEqual(SkScalar one, SkScalar two) {
return PaintOp::AreEqualEvenIfNaN(one, two);
}
bool HasDiscardableImages(const sk_sp<PaintFilter>& filter) {
return filter ? filter->has_discardable_images() : false;
}
bool HasDiscardableImages(const sk_sp<PaintFilter>* const filters, int count) {
for (int i = 0; i < count; ++i) {
if (filters[i] && filters[i]->has_discardable_images())
return true;
}
return false;
}
sk_sp<PaintFilter> Snapshot(const sk_sp<PaintFilter>& filter,
ImageProvider* image_provider) {
if (!filter)
return nullptr;
return filter->SnapshotWithImages(image_provider);
}
} // namespace
PaintFilter::PaintFilter(Type type,
const CropRect* crop_rect,
bool has_discardable_images)
: type_(type), has_discardable_images_(has_discardable_images) {
if (crop_rect)
crop_rect_.emplace(*crop_rect);
}
PaintFilter::~PaintFilter() = default;
// static
std::string PaintFilter::TypeToString(Type type) {
switch (type) {
case Type::kNullFilter:
return "kNullFilter";
case Type::kColorFilter:
return "kColorFilter";
case Type::kBlur:
return "kBlur";
case Type::kDropShadow:
return "kDropShadow";
case Type::kMagnifier:
return "kMagnifier";
case Type::kCompose:
return "kCompose";
case Type::kAlphaThreshold:
return "kAlphaThreshold";
case Type::kXfermode:
return "kXfermode";
case Type::kArithmetic:
return "kArithmetic";
case Type::kMatrixConvolution:
return "kMatrixConvolution";
case Type::kDisplacementMapEffect:
return "kDisplacementMapEffect";
case Type::kImage:
return "kImage";
case Type::kPaintRecord:
return "kPaintRecord";
case Type::kMerge:
return "kMerge";
case Type::kMorphology:
return "kMorphology";
case Type::kOffset:
return "kOffset";
case Type::kTile:
return "kTile";
case Type::kTurbulence:
return "kTurbulence";
case Type::kPaintFlags:
return "kPaintFlags";
case Type::kMatrix:
return "kMatrix";
case Type::kLightingDistant:
return "kLightingDistant";
case Type::kLightingPoint:
return "kLightingPoint";
case Type::kLightingSpot:
return "kLightingSpot";
}
NOTREACHED();
return "Unknown";
}
size_t PaintFilter::GetFilterSize(const PaintFilter* filter) {
// A null type is used to indicate no filter.
if (!filter)
return sizeof(uint32_t);
return filter->SerializedSize() + PaintOpWriter::Alignment();
}
size_t PaintFilter::BaseSerializedSize() const {
size_t total_size = 0u;
// Filter type.
total_size += sizeof(uint32_t);
// Bool to indicate whether crop exists.
total_size += sizeof(uint32_t);
if (crop_rect_) {
// CropRect.
total_size += sizeof(crop_rect_->flags());
total_size += sizeof(crop_rect_->rect());
}
return total_size;
}
sk_sp<PaintFilter> PaintFilter::SnapshotWithImages(
ImageProvider* image_provider) const {
if (!has_discardable_images_)
return sk_ref_sp<PaintFilter>(this);
return SnapshotWithImagesInternal(image_provider);
}
bool PaintFilter::operator==(const PaintFilter& other) const {
if (type_ != other.type_)
return false;
if (!!crop_rect_ != !!other.crop_rect_)
return false;
if (crop_rect_) {
if (crop_rect_->flags() != other.crop_rect_->flags() ||
!PaintOp::AreSkRectsEqual(crop_rect_->rect(),
other.crop_rect_->rect())) {
return false;
}
}
switch (type_) {
case Type::kNullFilter:
return true;
case Type::kColorFilter:
return *static_cast<const ColorFilterPaintFilter*>(this) ==
static_cast<const ColorFilterPaintFilter&>(other);
case Type::kBlur:
return *static_cast<const BlurPaintFilter*>(this) ==
static_cast<const BlurPaintFilter&>(other);
case Type::kDropShadow:
return *static_cast<const DropShadowPaintFilter*>(this) ==
static_cast<const DropShadowPaintFilter&>(other);
case Type::kMagnifier:
return *static_cast<const MagnifierPaintFilter*>(this) ==
static_cast<const MagnifierPaintFilter&>(other);
case Type::kCompose:
return *static_cast<const ComposePaintFilter*>(this) ==
static_cast<const ComposePaintFilter&>(other);
case Type::kAlphaThreshold:
return *static_cast<const AlphaThresholdPaintFilter*>(this) ==
static_cast<const AlphaThresholdPaintFilter&>(other);
case Type::kXfermode:
return *static_cast<const XfermodePaintFilter*>(this) ==
static_cast<const XfermodePaintFilter&>(other);
case Type::kArithmetic:
return *static_cast<const ArithmeticPaintFilter*>(this) ==
static_cast<const ArithmeticPaintFilter&>(other);
case Type::kMatrixConvolution:
return *static_cast<const MatrixConvolutionPaintFilter*>(this) ==
static_cast<const MatrixConvolutionPaintFilter&>(other);
case Type::kDisplacementMapEffect:
return *static_cast<const DisplacementMapEffectPaintFilter*>(this) ==
static_cast<const DisplacementMapEffectPaintFilter&>(other);
case Type::kImage:
return *static_cast<const ImagePaintFilter*>(this) ==
static_cast<const ImagePaintFilter&>(other);
case Type::kPaintRecord:
return *static_cast<const RecordPaintFilter*>(this) ==
static_cast<const RecordPaintFilter&>(other);
case Type::kMerge:
return *static_cast<const MergePaintFilter*>(this) ==
static_cast<const MergePaintFilter&>(other);
case Type::kMorphology:
return *static_cast<const MorphologyPaintFilter*>(this) ==
static_cast<const MorphologyPaintFilter&>(other);
case Type::kOffset:
return *static_cast<const OffsetPaintFilter*>(this) ==
static_cast<const OffsetPaintFilter&>(other);
case Type::kTile:
return *static_cast<const TilePaintFilter*>(this) ==
static_cast<const TilePaintFilter&>(other);
case Type::kTurbulence:
return *static_cast<const TurbulencePaintFilter*>(this) ==
static_cast<const TurbulencePaintFilter&>(other);
case Type::kPaintFlags:
return *static_cast<const PaintFlagsPaintFilter*>(this) ==
static_cast<const PaintFlagsPaintFilter&>(other);
case Type::kMatrix:
return *static_cast<const MatrixPaintFilter*>(this) ==
static_cast<const MatrixPaintFilter&>(other);
case Type::kLightingDistant:
return *static_cast<const LightingDistantPaintFilter*>(this) ==
static_cast<const LightingDistantPaintFilter&>(other);
case Type::kLightingPoint:
return *static_cast<const LightingPointPaintFilter*>(this) ==
static_cast<const LightingPointPaintFilter&>(other);
case Type::kLightingSpot:
return *static_cast<const LightingSpotPaintFilter*>(this) ==
static_cast<const LightingSpotPaintFilter&>(other);
}
NOTREACHED();
return true;
}
ColorFilterPaintFilter::ColorFilterPaintFilter(
sk_sp<SkColorFilter> color_filter,
sk_sp<PaintFilter> input,
const CropRect* crop_rect)
: PaintFilter(kType, crop_rect, HasDiscardableImages(input)),
color_filter_(std::move(color_filter)),
input_(std::move(input)) {
DCHECK(color_filter_);
cached_sk_filter_ = SkColorFilterImageFilter::Make(
color_filter_, GetSkFilter(input_.get()), crop_rect);
}
ColorFilterPaintFilter::~ColorFilterPaintFilter() = default;
size_t ColorFilterPaintFilter::SerializedSize() const {
base::CheckedNumeric<size_t> total_size = 0u;
total_size += BaseSerializedSize();
total_size += PaintOpWriter::GetFlattenableSize(color_filter_.get());
total_size += GetFilterSize(input_.get());
return total_size.ValueOrDefault(0u);
}
sk_sp<PaintFilter> ColorFilterPaintFilter::SnapshotWithImagesInternal(
ImageProvider* image_provider) const {
return sk_make_sp<ColorFilterPaintFilter>(
color_filter_, Snapshot(input_, image_provider), crop_rect());
}
bool ColorFilterPaintFilter::operator==(
const ColorFilterPaintFilter& other) const {
return PaintOp::AreSkFlattenablesEqual(color_filter_.get(),
other.color_filter_.get()) &&
AreFiltersEqual(input_.get(), other.input_.get());
}
BlurPaintFilter::BlurPaintFilter(SkScalar sigma_x,
SkScalar sigma_y,
TileMode tile_mode,
sk_sp<PaintFilter> input,
const CropRect* crop_rect)
: PaintFilter(kType, crop_rect, HasDiscardableImages(input)),
sigma_x_(sigma_x),
sigma_y_(sigma_y),
tile_mode_(tile_mode),
input_(std::move(input)) {
cached_sk_filter_ = SkBlurImageFilter::Make(
sigma_x_, sigma_y_, GetSkFilter(input_.get()), crop_rect, tile_mode_);
}
BlurPaintFilter::~BlurPaintFilter() = default;
size_t BlurPaintFilter::SerializedSize() const {
base::CheckedNumeric<size_t> total_size =
BaseSerializedSize() + sizeof(sigma_x_) + sizeof(sigma_y_) +
sizeof(tile_mode_);
total_size += GetFilterSize(input_.get());
return total_size.ValueOrDefault(0u);
}
sk_sp<PaintFilter> BlurPaintFilter::SnapshotWithImagesInternal(
ImageProvider* image_provider) const {
return sk_make_sp<BlurPaintFilter>(sigma_x_, sigma_y_, tile_mode_,
Snapshot(input_, image_provider),
crop_rect());
}
bool BlurPaintFilter::operator==(const BlurPaintFilter& other) const {
return PaintOp::AreEqualEvenIfNaN(sigma_x_, other.sigma_x_) &&
PaintOp::AreEqualEvenIfNaN(sigma_y_, other.sigma_y_) &&
tile_mode_ == other.tile_mode_ &&
AreFiltersEqual(input_.get(), other.input_.get());
}
DropShadowPaintFilter::DropShadowPaintFilter(SkScalar dx,
SkScalar dy,
SkScalar sigma_x,
SkScalar sigma_y,
SkColor color,
ShadowMode shadow_mode,
sk_sp<PaintFilter> input,
const CropRect* crop_rect)
: PaintFilter(kType, crop_rect, HasDiscardableImages(input)),
dx_(dx),
dy_(dy),
sigma_x_(sigma_x),
sigma_y_(sigma_y),
color_(color),
shadow_mode_(shadow_mode),
input_(std::move(input)) {
cached_sk_filter_ = SkDropShadowImageFilter::Make(
dx_, dy_, sigma_x_, sigma_y_, color_, shadow_mode_,
GetSkFilter(input_.get()), crop_rect);
}
DropShadowPaintFilter::~DropShadowPaintFilter() = default;
size_t DropShadowPaintFilter::SerializedSize() const {
base::CheckedNumeric<size_t> total_size =
BaseSerializedSize() + sizeof(dx_) + sizeof(dy_) + sizeof(sigma_x_) +
sizeof(sigma_y_) + sizeof(color_) + sizeof(shadow_mode_);
total_size += GetFilterSize(input_.get());
return total_size.ValueOrDefault(0u);
}
sk_sp<PaintFilter> DropShadowPaintFilter::SnapshotWithImagesInternal(
ImageProvider* image_provider) const {
return sk_make_sp<DropShadowPaintFilter>(
dx_, dy_, sigma_x_, sigma_y_, color_, shadow_mode_,
Snapshot(input_, image_provider), crop_rect());
}
bool DropShadowPaintFilter::operator==(
const DropShadowPaintFilter& other) const {
return PaintOp::AreEqualEvenIfNaN(dx_, other.dx_) &&
PaintOp::AreEqualEvenIfNaN(dy_, other.dy_) &&
PaintOp::AreEqualEvenIfNaN(sigma_x_, other.sigma_x_) &&
PaintOp::AreEqualEvenIfNaN(sigma_y_, other.sigma_y_) &&
color_ == other.color_ && shadow_mode_ == other.shadow_mode_ &&
AreFiltersEqual(input_.get(), other.input_.get());
}
MagnifierPaintFilter::MagnifierPaintFilter(const SkRect& src_rect,
SkScalar inset,
sk_sp<PaintFilter> input,
const CropRect* crop_rect)
: PaintFilter(kType, crop_rect, HasDiscardableImages(input)),
src_rect_(src_rect),
inset_(inset),
input_(std::move(input)) {
cached_sk_filter_ = SkMagnifierImageFilter::Make(
src_rect_, inset_, GetSkFilter(input_.get()), crop_rect);
}
MagnifierPaintFilter::~MagnifierPaintFilter() = default;
size_t MagnifierPaintFilter::SerializedSize() const {
base::CheckedNumeric<size_t> total_size =
BaseSerializedSize() + sizeof(src_rect_) + sizeof(inset_);
total_size += GetFilterSize(input_.get());
return total_size.ValueOrDefault(0u);
}
sk_sp<PaintFilter> MagnifierPaintFilter::SnapshotWithImagesInternal(
ImageProvider* image_provider) const {
return sk_make_sp<MagnifierPaintFilter>(
src_rect_, inset_, Snapshot(input_, image_provider), crop_rect());
}
bool MagnifierPaintFilter::operator==(const MagnifierPaintFilter& other) const {
return PaintOp::AreSkRectsEqual(src_rect_, other.src_rect_) &&
PaintOp::AreEqualEvenIfNaN(inset_, other.inset_) &&
AreFiltersEqual(input_.get(), other.input_.get());
}
ComposePaintFilter::ComposePaintFilter(sk_sp<PaintFilter> outer,
sk_sp<PaintFilter> inner)
: PaintFilter(Type::kCompose,
nullptr,
HasDiscardableImages(outer) || HasDiscardableImages(inner)),
outer_(std::move(outer)),
inner_(std::move(inner)) {
cached_sk_filter_ = SkComposeImageFilter::Make(GetSkFilter(outer_.get()),
GetSkFilter(inner_.get()));
}
ComposePaintFilter::~ComposePaintFilter() = default;
size_t ComposePaintFilter::SerializedSize() const {
base::CheckedNumeric<size_t> total_size = BaseSerializedSize();
total_size += GetFilterSize(outer_.get());
total_size += GetFilterSize(inner_.get());
return total_size.ValueOrDefault(0u);
}
sk_sp<PaintFilter> ComposePaintFilter::SnapshotWithImagesInternal(
ImageProvider* image_provider) const {
return sk_make_sp<ComposePaintFilter>(Snapshot(outer_, image_provider),
Snapshot(inner_, image_provider));
}
bool ComposePaintFilter::operator==(const ComposePaintFilter& other) const {
return AreFiltersEqual(outer_.get(), other.outer_.get()) &&
AreFiltersEqual(inner_.get(), other.inner_.get());
}
AlphaThresholdPaintFilter::AlphaThresholdPaintFilter(const SkRegion& region,
SkScalar inner_min,
SkScalar outer_max,
sk_sp<PaintFilter> input,
const CropRect* crop_rect)
: PaintFilter(kType, crop_rect, HasDiscardableImages(input)),
region_(region),
inner_min_(inner_min),
outer_max_(outer_max),
input_(std::move(input)) {
cached_sk_filter_ = SkAlphaThresholdFilter::Make(
region_, inner_min_, outer_max_, GetSkFilter(input_.get()), crop_rect);
}
AlphaThresholdPaintFilter::~AlphaThresholdPaintFilter() = default;
size_t AlphaThresholdPaintFilter::SerializedSize() const {
size_t region_size = region_.writeToMemory(nullptr);
base::CheckedNumeric<size_t> total_size;
total_size = BaseSerializedSize() + sizeof(uint64_t) + region_size +
sizeof(inner_min_) + sizeof(outer_max_);
total_size += GetFilterSize(input_.get());
return total_size.ValueOrDefault(0u);
}
sk_sp<PaintFilter> AlphaThresholdPaintFilter::SnapshotWithImagesInternal(
ImageProvider* image_provider) const {
return sk_make_sp<AlphaThresholdPaintFilter>(region_, inner_min_, outer_max_,
Snapshot(input_, image_provider),
crop_rect());
}
bool AlphaThresholdPaintFilter::operator==(
const AlphaThresholdPaintFilter& other) const {
return region_ == other.region_ &&
PaintOp::AreEqualEvenIfNaN(inner_min_, other.inner_min_) &&
PaintOp::AreEqualEvenIfNaN(outer_max_, other.outer_max_) &&
AreFiltersEqual(input_.get(), other.input_.get());
}
XfermodePaintFilter::XfermodePaintFilter(SkBlendMode blend_mode,
sk_sp<PaintFilter> background,
sk_sp<PaintFilter> foreground,
const CropRect* crop_rect)
: PaintFilter(
kType,
crop_rect,
HasDiscardableImages(background) || HasDiscardableImages(foreground)),
blend_mode_(blend_mode),
background_(std::move(background)),
foreground_(std::move(foreground)) {
cached_sk_filter_ =
SkXfermodeImageFilter::Make(blend_mode_, GetSkFilter(background_.get()),
GetSkFilter(foreground_.get()), crop_rect);
}
XfermodePaintFilter::~XfermodePaintFilter() = default;
size_t XfermodePaintFilter::SerializedSize() const {
base::CheckedNumeric<size_t> total_size =
BaseSerializedSize() + sizeof(blend_mode_);
total_size += GetFilterSize(background_.get());
total_size += GetFilterSize(foreground_.get());
return total_size.ValueOrDefault(0u);
}
sk_sp<PaintFilter> XfermodePaintFilter::SnapshotWithImagesInternal(
ImageProvider* image_provider) const {
return sk_make_sp<XfermodePaintFilter>(
blend_mode_, Snapshot(background_, image_provider),
Snapshot(foreground_, image_provider), crop_rect());
}
bool XfermodePaintFilter::operator==(const XfermodePaintFilter& other) const {
return blend_mode_ == other.blend_mode_ &&
AreFiltersEqual(background_.get(), other.background_.get()) &&
AreFiltersEqual(foreground_.get(), other.foreground_.get());
}
ArithmeticPaintFilter::ArithmeticPaintFilter(float k1,
float k2,
float k3,
float k4,
bool enforce_pm_color,
sk_sp<PaintFilter> background,
sk_sp<PaintFilter> foreground,
const CropRect* crop_rect)
: PaintFilter(
kType,
crop_rect,
HasDiscardableImages(background) || HasDiscardableImages(foreground)),
k1_(k1),
k2_(k2),
k3_(k3),
k4_(k4),
enforce_pm_color_(enforce_pm_color),
background_(std::move(background)),
foreground_(std::move(foreground)) {
cached_sk_filter_ = SkArithmeticImageFilter::Make(
k1_, k2_, k3_, k4_, enforce_pm_color_, GetSkFilter(background_.get()),
GetSkFilter(foreground_.get()), crop_rect);
}
ArithmeticPaintFilter::~ArithmeticPaintFilter() = default;
size_t ArithmeticPaintFilter::SerializedSize() const {
base::CheckedNumeric<size_t> total_size =
BaseSerializedSize() + sizeof(k1_) + sizeof(k2_) + sizeof(k3_) +
sizeof(k4_) + sizeof(enforce_pm_color_);
total_size += GetFilterSize(background_.get());
total_size += GetFilterSize(foreground_.get());
return total_size.ValueOrDefault(0u);
}
sk_sp<PaintFilter> ArithmeticPaintFilter::SnapshotWithImagesInternal(
ImageProvider* image_provider) const {
return sk_make_sp<ArithmeticPaintFilter>(
k1_, k2_, k3_, k4_, enforce_pm_color_,
Snapshot(background_, image_provider),
Snapshot(foreground_, image_provider), crop_rect());
}
bool ArithmeticPaintFilter::operator==(
const ArithmeticPaintFilter& other) const {
return PaintOp::AreEqualEvenIfNaN(k1_, other.k1_) &&
PaintOp::AreEqualEvenIfNaN(k2_, other.k2_) &&
PaintOp::AreEqualEvenIfNaN(k3_, other.k3_) &&
PaintOp::AreEqualEvenIfNaN(k4_, other.k4_) &&
enforce_pm_color_ == other.enforce_pm_color_ &&
AreFiltersEqual(background_.get(), other.background_.get()) &&
AreFiltersEqual(foreground_.get(), other.foreground_.get());
}
MatrixConvolutionPaintFilter::MatrixConvolutionPaintFilter(
const SkISize& kernel_size,
const SkScalar* kernel,
SkScalar gain,
SkScalar bias,
const SkIPoint& kernel_offset,
TileMode tile_mode,
bool convolve_alpha,
sk_sp<PaintFilter> input,
const CropRect* crop_rect)
: PaintFilter(kType, crop_rect, HasDiscardableImages(input)),
kernel_size_(kernel_size),
gain_(gain),
bias_(bias),
kernel_offset_(kernel_offset),
tile_mode_(tile_mode),
convolve_alpha_(convolve_alpha),
input_(std::move(input)) {
auto len = static_cast<size_t>(
sk_64_mul(kernel_size_.width(), kernel_size_.height()));
kernel_->reserve(len);
for (size_t i = 0; i < len; ++i)
kernel_->push_back(kernel[i]);
cached_sk_filter_ = SkMatrixConvolutionImageFilter::Make(
kernel_size_, kernel, gain_, bias_, kernel_offset_, tile_mode_,
convolve_alpha_, GetSkFilter(input_.get()), crop_rect);
}
MatrixConvolutionPaintFilter::~MatrixConvolutionPaintFilter() = default;
size_t MatrixConvolutionPaintFilter::SerializedSize() const {
base::CheckedNumeric<size_t> total_size =
BaseSerializedSize() + sizeof(kernel_size_) + sizeof(size_t) +
kernel_->size() * sizeof(SkScalar) + sizeof(gain_) + sizeof(bias_) +
sizeof(kernel_offset_) + sizeof(tile_mode_) + sizeof(convolve_alpha_);
total_size += GetFilterSize(input_.get());
return total_size.ValueOrDefault(0u);
}
sk_sp<PaintFilter> MatrixConvolutionPaintFilter::SnapshotWithImagesInternal(
ImageProvider* image_provider) const {
return sk_make_sp<MatrixConvolutionPaintFilter>(
kernel_size_, &kernel_[0], gain_, bias_, kernel_offset_, tile_mode_,
convolve_alpha_, Snapshot(input_, image_provider), crop_rect());
}
bool MatrixConvolutionPaintFilter::operator==(
const MatrixConvolutionPaintFilter& other) const {
return kernel_size_ == other.kernel_size_ &&
std::equal(kernel_.container().begin(), kernel_.container().end(),
other.kernel_.container().begin(), AreScalarsEqual) &&
PaintOp::AreEqualEvenIfNaN(gain_, other.gain_) &&
PaintOp::AreEqualEvenIfNaN(bias_, other.bias_) &&
kernel_offset_ == other.kernel_offset_ &&
tile_mode_ == other.tile_mode_ &&
convolve_alpha_ == other.convolve_alpha_ &&
AreFiltersEqual(input_.get(), other.input_.get());
}
DisplacementMapEffectPaintFilter::DisplacementMapEffectPaintFilter(
ChannelSelectorType channel_x,
ChannelSelectorType channel_y,
SkScalar scale,
sk_sp<PaintFilter> displacement,
sk_sp<PaintFilter> color,
const CropRect* crop_rect)
: PaintFilter(
kType,
crop_rect,
HasDiscardableImages(displacement) || HasDiscardableImages(color)),
channel_x_(channel_x),
channel_y_(channel_y),
scale_(scale),
displacement_(std::move(displacement)),
color_(std::move(color)) {
cached_sk_filter_ = SkDisplacementMapEffect::Make(
channel_x_, channel_y_, scale_, GetSkFilter(displacement_.get()),
GetSkFilter(color_.get()), crop_rect);
}
DisplacementMapEffectPaintFilter::~DisplacementMapEffectPaintFilter() = default;
size_t DisplacementMapEffectPaintFilter::SerializedSize() const {
base::CheckedNumeric<size_t> total_size = BaseSerializedSize() +
sizeof(uint32_t) +
sizeof(uint32_t) + sizeof(scale_);
total_size += GetFilterSize(displacement_.get());
total_size += GetFilterSize(color_.get());
return total_size.ValueOrDefault(0u);
}
sk_sp<PaintFilter> DisplacementMapEffectPaintFilter::SnapshotWithImagesInternal(
ImageProvider* image_provider) const {
return sk_make_sp<DisplacementMapEffectPaintFilter>(
channel_x_, channel_y_, scale_, Snapshot(displacement_, image_provider),
Snapshot(color_, image_provider), crop_rect());
}
bool DisplacementMapEffectPaintFilter::operator==(
const DisplacementMapEffectPaintFilter& other) const {
return channel_x_ == other.channel_x_ && channel_y_ == other.channel_y_ &&
PaintOp::AreEqualEvenIfNaN(scale_, other.scale_) &&
AreFiltersEqual(displacement_.get(), other.displacement_.get()) &&
AreFiltersEqual(color_.get(), other.color_.get());
}
ImagePaintFilter::ImagePaintFilter(PaintImage image,
const SkRect& src_rect,
const SkRect& dst_rect,
SkFilterQuality filter_quality)
: PaintFilter(kType, nullptr, !image.IsTextureBacked()),
image_(std::move(image)),
src_rect_(src_rect),
dst_rect_(dst_rect),
filter_quality_(filter_quality) {
cached_sk_filter_ = SkImageSource::Make(image_.GetSkImage(), src_rect_,
dst_rect_, filter_quality_);
}
ImagePaintFilter::~ImagePaintFilter() = default;
size_t ImagePaintFilter::SerializedSize() const {
base::CheckedNumeric<size_t> total_size =
BaseSerializedSize() + sizeof(src_rect_) + sizeof(dst_rect_) +
sizeof(filter_quality_);
total_size += PaintOpWriter::GetImageSize(image_);
return total_size.ValueOrDefault(0u);
}
sk_sp<PaintFilter> ImagePaintFilter::SnapshotWithImagesInternal(
ImageProvider* image_provider) const {
DrawImage draw_image(image_, SkIRect::MakeWH(image_.width(), image_.height()),
filter_quality_, SkMatrix::I());
auto scoped_result = image_provider->GetRasterContent(draw_image);
if (!scoped_result)
return nullptr;
auto decoded_sk_image = sk_ref_sp<SkImage>(
const_cast<SkImage*>(scoped_result.decoded_image().image().get()));
PaintImage decoded_paint_image =
PaintImageBuilder::WithDefault()
.set_id(image_.stable_id())
.set_image(decoded_sk_image, PaintImage::GetNextContentId())
.TakePaintImage();
return sk_make_sp<ImagePaintFilter>(std::move(decoded_paint_image), src_rect_,
dst_rect_, filter_quality_);
}
bool ImagePaintFilter::operator==(const ImagePaintFilter& other) const {
return !!image_ == !!other.image_ &&
PaintOp::AreSkRectsEqual(src_rect_, other.src_rect_) &&
PaintOp::AreSkRectsEqual(dst_rect_, other.dst_rect_) &&
filter_quality_ == other.filter_quality_;
}
RecordPaintFilter::RecordPaintFilter(sk_sp<PaintRecord> record,
const SkRect& record_bounds)
: RecordPaintFilter(std::move(record), record_bounds, nullptr) {}
RecordPaintFilter::RecordPaintFilter(sk_sp<PaintRecord> record,
const SkRect& record_bounds,
ImageProvider* image_provider)
: PaintFilter(kType, nullptr, record->HasDiscardableImages()),
record_(std::move(record)),
record_bounds_(record_bounds) {
cached_sk_filter_ = SkPictureImageFilter::Make(
ToSkPicture(record_, record_bounds_, image_provider));
}
RecordPaintFilter::~RecordPaintFilter() = default;
size_t RecordPaintFilter::SerializedSize() const {
base::CheckedNumeric<size_t> total_size =
BaseSerializedSize() + sizeof(record_bounds_);
total_size += PaintOpWriter::GetRecordSize(record_.get());
return total_size.ValueOrDefault(0u);
}
sk_sp<PaintFilter> RecordPaintFilter::SnapshotWithImagesInternal(
ImageProvider* image_provider) const {
return sk_sp<RecordPaintFilter>(
new RecordPaintFilter(record_, record_bounds_, image_provider));
}
bool RecordPaintFilter::operator==(const RecordPaintFilter& other) const {
return !!record_ == !!other.record_ &&
PaintOp::AreSkRectsEqual(record_bounds_, other.record_bounds_);
}
MergePaintFilter::MergePaintFilter(const sk_sp<PaintFilter>* const filters,
int count,
const CropRect* crop_rect)
: MergePaintFilter(filters, count, crop_rect, nullptr) {}
MergePaintFilter::MergePaintFilter(const sk_sp<PaintFilter>* const filters,
int count,
const CropRect* crop_rect,
ImageProvider* image_provider)
: PaintFilter(kType, crop_rect, HasDiscardableImages(filters, count)) {
std::vector<sk_sp<SkImageFilter>> sk_filters;
sk_filters.reserve(count);
for (int i = 0; i < count; ++i) {
auto filter =
image_provider ? Snapshot(filters[i], image_provider) : filters[i];
inputs_->push_back(std::move(filter));
sk_filters.push_back(GetSkFilter(inputs_->back().get()));
}
cached_sk_filter_ = SkMergeImageFilter::Make(
static_cast<sk_sp<SkImageFilter>*>(sk_filters.data()), count, crop_rect);
}
MergePaintFilter::~MergePaintFilter() = default;
size_t MergePaintFilter::SerializedSize() const {
base::CheckedNumeric<size_t> total_size = 0u;
for (size_t i = 0; i < input_count(); ++i)
total_size += GetFilterSize(input_at(i));
total_size += BaseSerializedSize();
total_size += sizeof(input_count());
return total_size.ValueOrDefault(0u);
}
sk_sp<PaintFilter> MergePaintFilter::SnapshotWithImagesInternal(
ImageProvider* image_provider) const {
return sk_sp<MergePaintFilter>(new MergePaintFilter(
&inputs_[0], inputs_->size(), crop_rect(), image_provider));
}
bool MergePaintFilter::operator==(const MergePaintFilter& other) const {
if (inputs_->size() != other.inputs_->size())
return false;
for (size_t i = 0; i < inputs_->size(); ++i) {
if (!AreFiltersEqual(inputs_[i].get(), other.inputs_[i].get()))
return false;
}
return true;
}
MorphologyPaintFilter::MorphologyPaintFilter(MorphType morph_type,
int radius_x,
int radius_y,
sk_sp<PaintFilter> input,
const CropRect* crop_rect)
: PaintFilter(kType, crop_rect, HasDiscardableImages(input)),
morph_type_(morph_type),
radius_x_(radius_x),
radius_y_(radius_y),
input_(std::move(input)) {
switch (morph_type_) {
case MorphType::kDilate:
cached_sk_filter_ = SkDilateImageFilter::Make(
radius_x_, radius_y_, GetSkFilter(input_.get()), crop_rect);
break;
case MorphType::kErode:
cached_sk_filter_ = SkErodeImageFilter::Make(
radius_x_, radius_y_, GetSkFilter(input_.get()), crop_rect);
break;
}
}
MorphologyPaintFilter::~MorphologyPaintFilter() = default;
size_t MorphologyPaintFilter::SerializedSize() const {
base::CheckedNumeric<size_t> total_size =
BaseSerializedSize() + sizeof(morph_type_) + sizeof(radius_x_) +
sizeof(radius_y_);
total_size += GetFilterSize(input_.get());
return total_size.ValueOrDefault(0u);
}
sk_sp<PaintFilter> MorphologyPaintFilter::SnapshotWithImagesInternal(
ImageProvider* image_provider) const {
return sk_make_sp<MorphologyPaintFilter>(morph_type_, radius_x_, radius_y_,
Snapshot(input_, image_provider),
crop_rect());
}
bool MorphologyPaintFilter::operator==(
const MorphologyPaintFilter& other) const {
return morph_type_ == other.morph_type_ && radius_x_ == other.radius_x_ &&
radius_y_ == other.radius_y_ &&
AreFiltersEqual(input_.get(), other.input_.get());
}
OffsetPaintFilter::OffsetPaintFilter(SkScalar dx,
SkScalar dy,
sk_sp<PaintFilter> input,
const CropRect* crop_rect)
: PaintFilter(kType, crop_rect, HasDiscardableImages(input)),
dx_(dx),
dy_(dy),
input_(std::move(input)) {
cached_sk_filter_ =
SkOffsetImageFilter::Make(dx_, dy_, GetSkFilter(input_.get()), crop_rect);
}
OffsetPaintFilter::~OffsetPaintFilter() = default;
size_t OffsetPaintFilter::SerializedSize() const {
base::CheckedNumeric<size_t> total_size =
BaseSerializedSize() + sizeof(dx_) + sizeof(dy_);
total_size += GetFilterSize(input_.get());
return total_size.ValueOrDefault(0u);
}
sk_sp<PaintFilter> OffsetPaintFilter::SnapshotWithImagesInternal(
ImageProvider* image_provider) const {
return sk_make_sp<OffsetPaintFilter>(
dx_, dy_, Snapshot(input_, image_provider), crop_rect());
}
bool OffsetPaintFilter::operator==(const OffsetPaintFilter& other) const {
return PaintOp::AreEqualEvenIfNaN(dx_, other.dx_) &&
PaintOp::AreEqualEvenIfNaN(dy_, other.dy_) &&
AreFiltersEqual(input_.get(), other.input_.get());
}
TilePaintFilter::TilePaintFilter(const SkRect& src,
const SkRect& dst,
sk_sp<PaintFilter> input)
: PaintFilter(kType, nullptr, HasDiscardableImages(input)),
src_(src),
dst_(dst),
input_(std::move(input)) {
cached_sk_filter_ =
SkTileImageFilter::Make(src_, dst_, GetSkFilter(input_.get()));
}
TilePaintFilter::~TilePaintFilter() = default;
size_t TilePaintFilter::SerializedSize() const {
base::CheckedNumeric<size_t> total_size =
BaseSerializedSize() + sizeof(src_) + sizeof(dst_);
total_size += GetFilterSize(input_.get());
return total_size.ValueOrDefault(0u);
}
sk_sp<PaintFilter> TilePaintFilter::SnapshotWithImagesInternal(
ImageProvider* image_provider) const {
return sk_make_sp<TilePaintFilter>(src_, dst_,
Snapshot(input_, image_provider));
}
bool TilePaintFilter::operator==(const TilePaintFilter& other) const {
return PaintOp::AreSkRectsEqual(src_, other.src_) &&
PaintOp::AreSkRectsEqual(dst_, other.dst_) &&
AreFiltersEqual(input_.get(), other.input_.get());
}
TurbulencePaintFilter::TurbulencePaintFilter(TurbulenceType turbulence_type,
SkScalar base_frequency_x,
SkScalar base_frequency_y,
int num_octaves,
SkScalar seed,
const SkISize* tile_size,
const CropRect* crop_rect)
: PaintFilter(kType, crop_rect, kHasNoDiscardableImages),
turbulence_type_(turbulence_type),
base_frequency_x_(base_frequency_x),
base_frequency_y_(base_frequency_y),
num_octaves_(num_octaves),
seed_(seed),
tile_size_(tile_size ? *tile_size : SkISize::MakeEmpty()) {
sk_sp<SkShader> shader;
switch (turbulence_type_) {
case TurbulenceType::kTurbulence:
shader = SkPerlinNoiseShader::MakeTurbulence(
base_frequency_x_, base_frequency_y_, num_octaves_, seed_,
&tile_size_);
break;
case TurbulenceType::kFractalNoise:
shader = SkPerlinNoiseShader::MakeFractalNoise(
base_frequency_x_, base_frequency_y_, num_octaves_, seed_,
&tile_size_);
break;
}
SkPaint paint;
paint.setShader(std::move(shader));
cached_sk_filter_ = SkPaintImageFilter::Make(paint, crop_rect);
}
TurbulencePaintFilter::~TurbulencePaintFilter() = default;
size_t TurbulencePaintFilter::SerializedSize() const {
return BaseSerializedSize() + sizeof(turbulence_type_) +
sizeof(base_frequency_x_) + sizeof(base_frequency_y_) +
sizeof(num_octaves_) + sizeof(seed_) + sizeof(tile_size_);
}
sk_sp<PaintFilter> TurbulencePaintFilter::SnapshotWithImagesInternal(
ImageProvider* image_provider) const {
return sk_make_sp<TurbulencePaintFilter>(turbulence_type_, base_frequency_x_,
base_frequency_y_, num_octaves_,
seed_, &tile_size_, crop_rect());
}
bool TurbulencePaintFilter::operator==(
const TurbulencePaintFilter& other) const {
return turbulence_type_ == other.turbulence_type_ &&
PaintOp::AreEqualEvenIfNaN(base_frequency_x_,
other.base_frequency_x_) &&
PaintOp::AreEqualEvenIfNaN(base_frequency_y_,
other.base_frequency_y_) &&
num_octaves_ == other.num_octaves_ &&
PaintOp::AreEqualEvenIfNaN(seed_, other.seed_) &&
tile_size_ == other.tile_size_;
}
PaintFlagsPaintFilter::PaintFlagsPaintFilter(PaintFlags flags,
const CropRect* crop_rect)
: PaintFlagsPaintFilter(std::move(flags), nullptr, crop_rect) {}
PaintFlagsPaintFilter::PaintFlagsPaintFilter(PaintFlags flags,
ImageProvider* image_provider,
const CropRect* crop_rect)
: PaintFilter(kType, crop_rect, flags.HasDiscardableImages()),
flags_(std::move(flags)) {
if (image_provider) {
raster_flags_.emplace(&flags_, image_provider, SkMatrix::I(), 0, 255u);
}
cached_sk_filter_ = SkPaintImageFilter::Make(
raster_flags_ ? raster_flags_->flags()->ToSkPaint() : flags_.ToSkPaint(),
crop_rect);
}
PaintFlagsPaintFilter::~PaintFlagsPaintFilter() = default;
size_t PaintFlagsPaintFilter::SerializedSize() const {
base::CheckedNumeric<size_t> total_size = BaseSerializedSize();
total_size += flags_.GetSerializedSize();
return total_size.ValueOrDefault(0u);
}
sk_sp<PaintFilter> PaintFlagsPaintFilter::SnapshotWithImagesInternal(
ImageProvider* image_provider) const {
return sk_sp<PaintFilter>(
new PaintFlagsPaintFilter(flags_, image_provider, crop_rect()));
}
bool PaintFlagsPaintFilter::operator==(
const PaintFlagsPaintFilter& other) const {
return flags_ == other.flags_;
}
MatrixPaintFilter::MatrixPaintFilter(const SkMatrix& matrix,
SkFilterQuality filter_quality,
sk_sp<PaintFilter> input)
: PaintFilter(Type::kMatrix, nullptr, HasDiscardableImages(input)),
matrix_(matrix),
filter_quality_(filter_quality),
input_(std::move(input)) {
cached_sk_filter_ = SkImageFilter::MakeMatrixFilter(
matrix_, filter_quality_, GetSkFilter(input_.get()));
}
MatrixPaintFilter::~MatrixPaintFilter() = default;
size_t MatrixPaintFilter::SerializedSize() const {
base::CheckedNumeric<size_t> total_size =
BaseSerializedSize() + sizeof(matrix_) + sizeof(filter_quality_);
total_size += GetFilterSize(input_.get());
return total_size.ValueOrDefault(0u);
}
sk_sp<PaintFilter> MatrixPaintFilter::SnapshotWithImagesInternal(
ImageProvider* image_provider) const {
return sk_make_sp<MatrixPaintFilter>(matrix_, filter_quality_,
Snapshot(input_, image_provider));
}
bool MatrixPaintFilter::operator==(const MatrixPaintFilter& other) const {
return PaintOp::AreSkMatricesEqual(matrix_, other.matrix_) &&
filter_quality_ == other.filter_quality_ &&
AreFiltersEqual(input_.get(), other.input_.get());
}
LightingDistantPaintFilter::LightingDistantPaintFilter(
LightingType lighting_type,
const SkPoint3& direction,
SkColor light_color,
SkScalar surface_scale,
SkScalar kconstant,
SkScalar shininess,
sk_sp<PaintFilter> input,
const CropRect* crop_rect)
: PaintFilter(kType, crop_rect, HasDiscardableImages(input)),
lighting_type_(lighting_type),
direction_(direction),
light_color_(light_color),
surface_scale_(surface_scale),
kconstant_(kconstant),
shininess_(shininess),
input_(std::move(input)) {
switch (lighting_type_) {
case LightingType::kDiffuse:
cached_sk_filter_ = SkLightingImageFilter::MakeDistantLitDiffuse(
direction_, light_color_, surface_scale_, kconstant_,
GetSkFilter(input_.get()), crop_rect);
break;
case LightingType::kSpecular:
cached_sk_filter_ = SkLightingImageFilter::MakeDistantLitSpecular(
direction_, light_color_, surface_scale_, kconstant_, shininess_,
GetSkFilter(input_.get()), crop_rect);
break;
}
}
LightingDistantPaintFilter::~LightingDistantPaintFilter() = default;
size_t LightingDistantPaintFilter::SerializedSize() const {
base::CheckedNumeric<size_t> total_size =
BaseSerializedSize() + sizeof(lighting_type_) + sizeof(direction_) +
sizeof(light_color_) + sizeof(surface_scale_) + sizeof(kconstant_) +
sizeof(shininess_);
total_size += GetFilterSize(input_.get());
return total_size.ValueOrDefault(0u);
}
sk_sp<PaintFilter> LightingDistantPaintFilter::SnapshotWithImagesInternal(
ImageProvider* image_provider) const {
return sk_make_sp<LightingDistantPaintFilter>(
lighting_type_, direction_, light_color_, surface_scale_, kconstant_,
shininess_, Snapshot(input_, image_provider), crop_rect());
}
bool LightingDistantPaintFilter::operator==(
const LightingDistantPaintFilter& other) const {
return lighting_type_ == other.lighting_type_ &&
PaintOp::AreSkPoint3sEqual(direction_, other.direction_) &&
light_color_ == other.light_color_ &&
PaintOp::AreEqualEvenIfNaN(surface_scale_, other.surface_scale_) &&
PaintOp::AreEqualEvenIfNaN(kconstant_, other.kconstant_) &&
PaintOp::AreEqualEvenIfNaN(shininess_, other.shininess_) &&
AreFiltersEqual(input_.get(), other.input_.get());
}
LightingPointPaintFilter::LightingPointPaintFilter(LightingType lighting_type,
const SkPoint3& location,
SkColor light_color,
SkScalar surface_scale,
SkScalar kconstant,
SkScalar shininess,
sk_sp<PaintFilter> input,
const CropRect* crop_rect)
: PaintFilter(kType, crop_rect, HasDiscardableImages(input)),
lighting_type_(lighting_type),
location_(location),
light_color_(light_color),
surface_scale_(surface_scale),
kconstant_(kconstant),
shininess_(shininess),
input_(std::move(input)) {
switch (lighting_type_) {
case LightingType::kDiffuse:
cached_sk_filter_ = SkLightingImageFilter::MakePointLitDiffuse(
location_, light_color_, surface_scale_, kconstant_,
GetSkFilter(input_.get()), crop_rect);
break;
case LightingType::kSpecular:
cached_sk_filter_ = SkLightingImageFilter::MakePointLitSpecular(
location_, light_color_, surface_scale_, kconstant_, shininess_,
GetSkFilter(input_.get()), crop_rect);
break;
}
}
LightingPointPaintFilter::~LightingPointPaintFilter() = default;
size_t LightingPointPaintFilter::SerializedSize() const {
base::CheckedNumeric<size_t> total_size =
BaseSerializedSize() + sizeof(lighting_type_) + sizeof(location_) +
sizeof(light_color_) + sizeof(surface_scale_) + sizeof(kconstant_) +
sizeof(shininess_);
total_size += GetFilterSize(input_.get());
return total_size.ValueOrDefault(0u);
}
sk_sp<PaintFilter> LightingPointPaintFilter::SnapshotWithImagesInternal(
ImageProvider* image_provider) const {
return sk_make_sp<LightingPointPaintFilter>(
lighting_type_, location_, light_color_, surface_scale_, kconstant_,
shininess_, Snapshot(input_, image_provider), crop_rect());
}
bool LightingPointPaintFilter::operator==(
const LightingPointPaintFilter& other) const {
return lighting_type_ == other.lighting_type_ &&
PaintOp::AreSkPoint3sEqual(location_, other.location_) &&
light_color_ == other.light_color_ &&
PaintOp::AreEqualEvenIfNaN(surface_scale_, other.surface_scale_) &&
PaintOp::AreEqualEvenIfNaN(kconstant_, other.kconstant_) &&
PaintOp::AreEqualEvenIfNaN(shininess_, other.shininess_) &&
AreFiltersEqual(input_.get(), other.input_.get());
}
LightingSpotPaintFilter::LightingSpotPaintFilter(LightingType lighting_type,
const SkPoint3& location,
const SkPoint3& target,
SkScalar specular_exponent,
SkScalar cutoff_angle,
SkColor light_color,
SkScalar surface_scale,
SkScalar kconstant,
SkScalar shininess,
sk_sp<PaintFilter> input,
const CropRect* crop_rect)
: PaintFilter(kType, crop_rect, HasDiscardableImages(input)),
lighting_type_(lighting_type),
location_(location),
target_(target),
specular_exponent_(specular_exponent),
cutoff_angle_(cutoff_angle),
light_color_(light_color),
surface_scale_(surface_scale),
kconstant_(kconstant),
shininess_(shininess),
input_(std::move(input)) {
switch (lighting_type_) {
case LightingType::kDiffuse:
cached_sk_filter_ = SkLightingImageFilter::MakeSpotLitDiffuse(
location_, target_, specular_exponent_, cutoff_angle_, light_color_,
surface_scale_, kconstant_, GetSkFilter(input_.get()), crop_rect);
break;
case LightingType::kSpecular:
cached_sk_filter_ = SkLightingImageFilter::MakeSpotLitSpecular(
location_, target_, specular_exponent_, cutoff_angle_, light_color_,
surface_scale_, kconstant_, shininess_, GetSkFilter(input_.get()),
crop_rect);
break;
}
}
LightingSpotPaintFilter::~LightingSpotPaintFilter() = default;
size_t LightingSpotPaintFilter::SerializedSize() const {
base::CheckedNumeric<size_t> total_size =
BaseSerializedSize() + sizeof(lighting_type_) + sizeof(location_) +
sizeof(target_) + sizeof(specular_exponent_) + sizeof(cutoff_angle_) +
sizeof(light_color_) + sizeof(surface_scale_) + sizeof(kconstant_) +
sizeof(shininess_);
total_size += GetFilterSize(input_.get());
return total_size.ValueOrDefault(0u);
}
sk_sp<PaintFilter> LightingSpotPaintFilter::SnapshotWithImagesInternal(
ImageProvider* image_provider) const {
return sk_make_sp<LightingSpotPaintFilter>(
lighting_type_, location_, target_, specular_exponent_, cutoff_angle_,
light_color_, surface_scale_, kconstant_, shininess_,
Snapshot(input_, image_provider), crop_rect());
}
bool LightingSpotPaintFilter::operator==(
const LightingSpotPaintFilter& other) const {
return lighting_type_ == other.lighting_type_ &&
PaintOp::AreSkPoint3sEqual(location_, other.location_) &&
PaintOp::AreSkPoint3sEqual(target_, other.target_) &&
PaintOp::AreEqualEvenIfNaN(specular_exponent_,
other.specular_exponent_) &&
PaintOp::AreEqualEvenIfNaN(cutoff_angle_, other.cutoff_angle_) &&
light_color_ == other.light_color_ &&
PaintOp::AreEqualEvenIfNaN(surface_scale_, other.surface_scale_) &&
PaintOp::AreEqualEvenIfNaN(kconstant_, other.kconstant_) &&
PaintOp::AreEqualEvenIfNaN(shininess_, other.shininess_) &&
AreFiltersEqual(input_.get(), other.input_.get());
}
} // namespace cc