b161a7d965
Currently, shader effects are applied via layers for non-atomic draw fragments. This is incorrect, as shaders on layer paints are ignored. Update to always propagate the shader to leaf node paints. Change-Id: I59f5a5b874e87827f61844969550600fa39e22a3 Reviewed-on: https://skia-review.googlesource.com/c/skia/+/205931 Reviewed-by: Mike Reed <reed@google.com> Commit-Queue: Florin Malita <fmalita@chromium.org>
119 lines
3.8 KiB
C++
119 lines
3.8 KiB
C++
/*
|
|
* Copyright 2017 Google Inc.
|
|
*
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
* found in the LICENSE file.
|
|
*/
|
|
|
|
#ifndef SkSGRenderNode_DEFINED
|
|
#define SkSGRenderNode_DEFINED
|
|
|
|
#include "SkSGNode.h"
|
|
|
|
#include "SkBlendMode.h"
|
|
#include "SkColorFilter.h"
|
|
#include "SkShader.h"
|
|
|
|
class SkCanvas;
|
|
class SkImageFilter;
|
|
class SkPaint;
|
|
|
|
namespace sksg {
|
|
|
|
/**
|
|
* Base class for nodes which can render to a canvas.
|
|
*/
|
|
class RenderNode : public Node {
|
|
protected:
|
|
struct RenderContext;
|
|
|
|
public:
|
|
// Render the node and its descendants to the canvas.
|
|
void render(SkCanvas*, const RenderContext* = nullptr) const;
|
|
|
|
// Perform a front-to-back hit-test, and return the RenderNode located at |point|.
|
|
// Normally, hit-testing stops at leaf Draw nodes.
|
|
const RenderNode* nodeAt(const SkPoint& point) const;
|
|
|
|
protected:
|
|
explicit RenderNode(uint32_t inval_traits = 0);
|
|
|
|
virtual void onRender(SkCanvas*, const RenderContext*) const = 0;
|
|
virtual const RenderNode* onNodeAt(const SkPoint& p) const = 0;
|
|
|
|
// Paint property overrides.
|
|
// These are deferred until we can determine whether they can be applied to the individual
|
|
// draw paints, or whether they require content isolation (applied to a layer).
|
|
struct RenderContext {
|
|
sk_sp<SkColorFilter> fColorFilter;
|
|
sk_sp<SkShader> fShader;
|
|
SkMatrix fShaderCTM = SkMatrix::I();
|
|
float fOpacity = 1;
|
|
SkBlendMode fBlendMode = SkBlendMode::kSrcOver;
|
|
|
|
// Returns true if the paint overrides require a layer when applied to non-atomic draws.
|
|
bool requiresIsolation() const;
|
|
|
|
void modulatePaint(const SkMatrix& ctm, SkPaint*) const;
|
|
};
|
|
|
|
class ScopedRenderContext final {
|
|
public:
|
|
ScopedRenderContext(SkCanvas*, const RenderContext*);
|
|
~ScopedRenderContext();
|
|
|
|
ScopedRenderContext(ScopedRenderContext&& that) { *this = std::move(that); }
|
|
|
|
ScopedRenderContext& operator=(ScopedRenderContext&& that) {
|
|
fCanvas = that.fCanvas;
|
|
fCtx = std::move(that.fCtx);
|
|
fRestoreCount = that.fRestoreCount;
|
|
|
|
// scope ownership is being transferred
|
|
that.fRestoreCount = -1;
|
|
|
|
return *this;
|
|
}
|
|
|
|
operator const RenderContext* () const { return &fCtx; }
|
|
|
|
// Add (cumulative) paint overrides to a render node sub-DAG.
|
|
ScopedRenderContext&& modulateOpacity(float opacity);
|
|
ScopedRenderContext&& modulateColorFilter(sk_sp<SkColorFilter>);
|
|
ScopedRenderContext&& modulateShader(sk_sp<SkShader>, const SkMatrix& shader_ctm);
|
|
ScopedRenderContext&& modulateBlendMode(SkBlendMode);
|
|
|
|
// Force content isolation for a node sub-DAG by applying the RenderContext
|
|
// overrides via a layer.
|
|
ScopedRenderContext&& setIsolation(const SkRect& bounds, const SkMatrix& ctm,
|
|
bool do_isolate);
|
|
|
|
// Similarly, force content isolation by applying the RenderContext overrides and
|
|
// an image filter via a single layer.
|
|
ScopedRenderContext&& setFilterIsolation(const SkRect& bounds, const SkMatrix& ctm,
|
|
sk_sp<SkImageFilter>);
|
|
|
|
private:
|
|
// stack-only
|
|
void* operator new(size_t) = delete;
|
|
void* operator new(size_t, void*) = delete;
|
|
|
|
// Scopes cannot be copied.
|
|
ScopedRenderContext(const ScopedRenderContext&) = delete;
|
|
ScopedRenderContext& operator=(const ScopedRenderContext&) = delete;
|
|
|
|
SkCanvas* fCanvas;
|
|
RenderContext fCtx;
|
|
int fRestoreCount;
|
|
};
|
|
|
|
private:
|
|
friend class ImageFilterEffect;
|
|
|
|
typedef Node INHERITED;
|
|
};
|
|
|
|
} // namespace sksg
|
|
|
|
#endif // SkSGRenderNode_DEFINED
|