| // Copyright 2013 The Flutter Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #ifndef FLUTTER_FLOW_LAYERS_LAYER_STATE_STACK_H_ |
| #define FLUTTER_FLOW_LAYERS_LAYER_STATE_STACK_H_ |
| |
| #include "flutter/display_list/dl_canvas.h" |
| #include "flutter/flow/embedded_views.h" |
| #include "flutter/flow/paint_utils.h" |
| |
| namespace flutter { |
| |
| /// The LayerStateStack manages the inherited state passed down between |
| /// |Layer| objects in a |LayerTree| during |Preroll| and |Paint|. |
| /// |
| /// More specifically, it manages the clip and transform state during |
| /// recursive rendering and will hold and lazily apply opacity, ImageFilter |
| /// and ColorFilter attributes to recursive content. This is not a truly |
| /// general state management mechnanism as it makes assumptions that code |
| /// will be applying the attributes to rendered content that happens in |
| /// recursive calls. The automatic save/restore mechanisms only work in |
| /// a context where C++ auto-destruct calls will engage the restore at |
| /// the end of a code block and that any applied attributes will only |
| /// be applied to the content rendered inside that block. These restrictions |
| /// match the organization of the |LayerTree| methods precisely. |
| /// |
| /// The stack can manage a single state delegate. The delegate will provide |
| /// tracking of the current transform and clip and will also execute |
| /// saveLayer calls at the appropriate time if it is a rendering delegate. |
| /// The delegate can be swapped out on the fly (as is typically done by |
| /// PlatformViewLayer when recording the state for multiple "overlay" |
| /// layers that occur between embedded view subtrees. The old delegate |
| /// will be restored to its original state before it became a delegate |
| /// and the new delegate will have all of the state recorded by the stack |
| /// replayed into it to bring it up to speed with the current rendering |
| /// context. |
| /// |
| /// The delegate can be any one of: |
| /// - Preroll delegate: used during Preroll to remember the outstanding |
| /// state for embedded platform layers |
| /// - DlCanvas: used during Paint for rendering output |
| /// The stack will know which state needs to be conveyed to any of these |
| /// delegates and when is the best time to convey that state (i.e. lazy |
| /// saveLayer calls for example). |
| /// |
| /// The rendering state attributes will be automatically applied to the |
| /// nested content using a |saveLayer| call at the point at which we |
| /// encounter rendered content (i.e. various nested layers that exist only |
| /// to apply new state will not trigger the |saveLayer| and the attributes |
| /// can accumulate until we reach actual content that is rendered.) Some |
| /// rendered content can avoid the |saveLayer| if it reports to the object |
| /// that it is able to apply all of the attributes that happen to be |
| /// outstanding (accumulated from parent state-modifiers). A |ContainerLayer| |
| /// can also monitor the attribute rendering capabilities of a list of |
| /// children and can ask the object to apply a protective |saveLayer| or |
| /// not based on the negotiated capabilities of the entire group. |
| /// |
| /// Any code that is planning to modify the clip, transform, or rendering |
| /// attributes for its child content must start by calling the |save| method |
| /// which returns a MutatorContext object. The methods that modify such |
| /// state only exist on the MutatorContext object so it is difficult to get |
| /// that wrong, but the caller must make sure that the call happens within |
| /// a C++ code block that will define the "rendering scope" of those |
| /// state changes as they will be automatically restored on exit from that |
| /// block. Note that the layer might make similar state calls directly on |
| /// the canvas or builder during the Paint cycle (via saveLayer, transform, |
| /// or clip calls), but should avoid doing so if there is any nested content |
| /// that needs to track or react to those state calls. |
| /// |
| /// Code that needs to render content can simply inform the parent of their |
| /// abilities by setting the |PrerollContext::renderable_state_flags| during |
| /// |Preroll| and then render with those attributes during |Paint| by |
| /// requesting the outstanding values of those attributes from the state_stack |
| /// object. Individual leaf layers can ignore this feature as the default |
| /// behavior during |Preroll| will have their parent |ContainerLayer| assume |
| /// that they cannot render any outstanding state attributes and will apply |
| /// the protective saveLayer on their behalf if needed. As such, this object |
| /// only provides "opt-in" features for leaf layers and no responsibilities |
| /// otherwise. |
| /// See |LayerStateStack::fill| |
| /// See |LayerStateStack::outstanding_opacity| |
| /// See |LayerStateStack::outstanding_color_filter| |
| /// See |LayerStateStack::outstanding_image_filter| |
| /// |
| /// State-modifying layers should contain code similar to this pattern in both |
| /// their |Preroll| and |Paint| methods. |
| /// |
| /// void [LayerType]::[Preroll/Paint](context) { |
| /// auto mutator = context.state_stack.save(); |
| /// mutator.translate(origin.x, origin.y); |
| /// mutator.applyOpacity(content_bounds, opacity_value); |
| /// mutator.applyColorFilter(content_bounds, color_filter); |
| /// // or any of the mutator transform, clip or attribute methods |
| /// |
| /// // Children will react to the state applied above during their |
| /// // Preroll/Paint methods or ContainerLayer will protect them |
| /// // conservatively by default. |
| /// [Preroll/Paint]Children(context); |
| /// |
| /// // here the mutator will be auto-destructed and the state accumulated |
| /// // by it will be restored out of the state_stack and its associated |
| /// // delegates. |
| /// } |
| class LayerStateStack { |
| public: |
| LayerStateStack(); |
| |
| // Clears out any old delegate to make room for a new one. |
| void clear_delegate(); |
| |
| // Return the DlCanvas delegate if the state stack has such a delegate. |
| // The state stack will only have one delegate at a time holding either |
| // a DlCanvas or a preroll accumulator. |
| DlCanvas* canvas_delegate() { return delegate_->canvas(); } |
| |
| // Clears the old delegate and sets the canvas delegate to the indicated |
| // DL canvas (if not nullptr). This ensures that only one delegate - either |
| // a DlCanvas or a preroll accumulator - is present at any one time. |
| void set_delegate(DlCanvas* canvas); |
| |
| // Clears the old delegate and sets the state stack up to accumulate |
| // clip and transform information for a Preroll phase. This ensures |
| // that only one delegate - either a DlCanvas or a preroll accumulator - |
| // is present at any one time. |
| void set_preroll_delegate(const SkRect& cull_rect, const SkMatrix& matrix); |
| void set_preroll_delegate(const SkRect& cull_rect); |
| void set_preroll_delegate(const SkMatrix& matrix); |
| |
| // Fills the supplied MatatorsStack object with the mutations recorded |
| // by this LayerStateStack in the order encountered. |
| void fill(MutatorsStack* mutators); |
| |
| // Sets up a checkerboard function that will be used to checkerboard the |
| // contents of any saveLayer executed by the state stack. |
| CheckerboardFunc checkerboard_func() const { return checkerboard_func_; } |
| void set_checkerboard_func(CheckerboardFunc checkerboard_func) { |
| checkerboard_func_ = checkerboard_func; |
| } |
| |
| class AutoRestore { |
| public: |
| ~AutoRestore() { |
| layer_state_stack_->restore_to_count(stack_restore_count_); |
| } |
| |
| private: |
| AutoRestore(LayerStateStack* stack, const SkRect& bounds, int flags) |
| : layer_state_stack_(stack), |
| stack_restore_count_(stack->stack_count()) { |
| if (stack->needs_save_layer(flags)) { |
| stack->save_layer(bounds); |
| } |
| } |
| friend class LayerStateStack; |
| |
| LayerStateStack* layer_state_stack_; |
| const size_t stack_restore_count_; |
| |
| FML_DISALLOW_COPY_ASSIGN_AND_MOVE(AutoRestore); |
| }; |
| |
| class MutatorContext { |
| public: |
| ~MutatorContext() { |
| layer_state_stack_->restore_to_count(stack_restore_count_); |
| } |
| |
| // Immediately executes a saveLayer with all accumulated state |
| // onto the canvas or builder to be applied at the next matching |
| // restore. A saveLayer is always executed by this method even if |
| // there are no outstanding attributes. |
| void saveLayer(const SkRect& bounds); |
| |
| // Records the opacity for application at the next call to |
| // saveLayer or applyState. A saveLayer may be executed at |
| // this time if the opacity cannot be batched with other |
| // outstanding attributes. |
| void applyOpacity(const SkRect& bounds, SkScalar opacity); |
| |
| // Records the image filter for application at the next call to |
| // saveLayer or applyState. A saveLayer may be executed at |
| // this time if the image filter cannot be batched with other |
| // outstanding attributes. |
| // (Currently only opacity is recorded for batching) |
| void applyImageFilter(const SkRect& bounds, |
| const std::shared_ptr<const DlImageFilter>& filter); |
| |
| // Records the color filter for application at the next call to |
| // saveLayer or applyState. A saveLayer may be executed at |
| // this time if the color filter cannot be batched with other |
| // outstanding attributes. |
| // (Currently only opacity is recorded for batching) |
| void applyColorFilter(const SkRect& bounds, |
| const std::shared_ptr<const DlColorFilter>& filter); |
| |
| // Saves the state stack and immediately executes a saveLayer |
| // with the indicated backdrop filter and any outstanding |
| // state attributes. Since the backdrop filter only applies |
| // to the pixels alrady on the screen when this call is made, |
| // the backdrop filter will only be applied to the canvas or |
| // builder installed at the time that this call is made, and |
| // subsequent canvas or builder objects that are made delegates |
| // will only see a saveLayer with the indicated blend_mode. |
| void applyBackdropFilter(const SkRect& bounds, |
| const std::shared_ptr<const DlImageFilter>& filter, |
| DlBlendMode blend_mode); |
| |
| void translate(SkScalar tx, SkScalar ty); |
| void translate(SkPoint tp) { translate(tp.fX, tp.fY); } |
| void transform(const SkM44& m44); |
| void transform(const SkMatrix& matrix); |
| void integralTransform(); |
| |
| void clipRect(const SkRect& rect, bool is_aa); |
| void clipRRect(const SkRRect& rrect, bool is_aa); |
| void clipPath(const SkPath& path, bool is_aa); |
| |
| private: |
| MutatorContext(LayerStateStack* stack) |
| : layer_state_stack_(stack), |
| stack_restore_count_(stack->stack_count()), |
| save_needed_(true) {} |
| friend class LayerStateStack; |
| |
| LayerStateStack* layer_state_stack_; |
| const size_t stack_restore_count_; |
| bool save_needed_; |
| |
| FML_DISALLOW_COPY_ASSIGN_AND_MOVE(MutatorContext); |
| }; |
| |
| static constexpr int kCallerCanApplyOpacity = 0x1; |
| static constexpr int kCallerCanApplyColorFilter = 0x2; |
| static constexpr int kCallerCanApplyImageFilter = 0x4; |
| static constexpr int kCallerCanApplyAnything = |
| (kCallerCanApplyOpacity | kCallerCanApplyColorFilter | |
| kCallerCanApplyImageFilter); |
| |
| // Apply the outstanding state via saveLayer if necessary, |
| // respecting the flags representing which potentially |
| // outstanding attributes the calling layer can apply |
| // themselves. |
| // |
| // A saveLayer may or may not be sent to the delegates depending |
| // on how the outstanding state intersects with the flags supplied |
| // by the caller. |
| // |
| // An AutoRestore instance will always be returned even if there |
| // was no saveLayer applied. |
| [[nodiscard]] inline AutoRestore applyState(const SkRect& bounds, |
| int can_apply_flags) { |
| return AutoRestore(this, bounds, can_apply_flags); |
| } |
| |
| SkScalar outstanding_opacity() const { return outstanding_.opacity; } |
| |
| std::shared_ptr<const DlColorFilter> outstanding_color_filter() const { |
| return outstanding_.color_filter; |
| } |
| |
| std::shared_ptr<const DlImageFilter> outstanding_image_filter() const { |
| return outstanding_.image_filter; |
| } |
| |
| // The outstanding bounds are the bounds recorded during the last |
| // attribute applied to this state stack. The assumption is that |
| // the nested calls to the state stack will each supply bounds relative |
| // to the content of that single attribute and the bounds of the content |
| // of any outstanding attributes will include the output bounds of |
| // applying any nested attributes. Thus, only the innermost content |
| // bounds received will be sufficient to apply all outstanding attributes. |
| SkRect outstanding_bounds() const { return outstanding_.save_layer_bounds; } |
| |
| // Fill the provided paint object with any oustanding attributes and |
| // return a pointer to it, or return a nullptr if there were no |
| // outstanding attributes to paint with. |
| DlPaint* fill(DlPaint& paint) const { return outstanding_.fill(paint); } |
| |
| // The cull_rect (not the exact clip) relative to the device pixels. |
| // This rectangle may be a conservative estimate of the true clip region. |
| SkRect device_cull_rect() const { return delegate_->device_cull_rect(); } |
| |
| // The cull_rect (not the exact clip) relative to the local coordinates. |
| // This rectangle may be a conservative estimate of the true clip region. |
| SkRect local_cull_rect() const { return delegate_->local_cull_rect(); } |
| |
| // The transform from the local coordinates to the device coordinates |
| // in the most capable 4x4 matrix representation. This matrix may be |
| // more information than is needed to compute bounds for a 2D rendering |
| // primitive, but it will accurately concatenate with other 4x4 matrices |
| // without losing information. |
| SkM44 transform_4x4() const { return delegate_->matrix_4x4(); } |
| |
| // The transform from the local coordinates to the device coordinates |
| // in a more compact 3x3 matrix represenation that provides enough |
| // information to accurately transform 2D primitives into their |
| // resulting 2D bounds. This matrix also has enough information to |
| // concat with other 2D affine transforms, but does not carry enough |
| // information to accurately concat with fully perspective matrics. |
| SkMatrix transform_3x3() const { return delegate_->matrix_3x3(); } |
| |
| // Tests if painting content with the current outstanding attributes |
| // will produce any content. This method does not check the current |
| // transform or clip for being singular or empty. |
| // See |content_culled| |
| bool painting_is_nop() const { return outstanding_.opacity <= 0; } |
| |
| // Tests if painting content with the given bounds will produce any output. |
| // This method does not check the outstanding attributes to verify that |
| // they produce visible results. |
| // See |painting_is_nop| |
| bool content_culled(const SkRect& content_bounds) const { |
| return delegate_->content_culled(content_bounds); |
| } |
| |
| // Saves the current state of the state stack and returns a |
| // MutatorContext which can be used to manipulate the state. |
| // The state stack will be restored to its current state |
| // when the MutatorContext object goes out of scope. |
| [[nodiscard]] inline MutatorContext save() { return MutatorContext(this); } |
| |
| // Returns true if the state stack is in, or has returned to, |
| // its initial state. |
| bool is_empty() const { return state_stack_.empty(); } |
| |
| private: |
| size_t stack_count() const { return state_stack_.size(); } |
| void restore_to_count(size_t restore_count); |
| void reapply_all(); |
| |
| void apply_last_entry() { state_stack_.back()->apply(this); } |
| |
| // The push methods simply push an associated StateEntry on the stack |
| // and then apply it to the current canvas and builder. |
| // --------------------- |
| // void push_attributes(); |
| void push_opacity(const SkRect& rect, SkScalar opacity); |
| void push_color_filter(const SkRect& bounds, |
| const std::shared_ptr<const DlColorFilter>& filter); |
| void push_image_filter(const SkRect& bounds, |
| const std::shared_ptr<const DlImageFilter>& filter); |
| void push_backdrop(const SkRect& bounds, |
| const std::shared_ptr<const DlImageFilter>& filter, |
| DlBlendMode blend_mode); |
| |
| void push_translate(SkScalar tx, SkScalar ty); |
| void push_transform(const SkM44& matrix); |
| void push_transform(const SkMatrix& matrix); |
| void push_integral_transform(); |
| |
| void push_clip_rect(const SkRect& rect, bool is_aa); |
| void push_clip_rrect(const SkRRect& rrect, bool is_aa); |
| void push_clip_path(const SkPath& path, bool is_aa); |
| // --------------------- |
| |
| // The maybe/needs_save_layer methods will determine if the indicated |
| // attribute can be incorporated into the outstanding attributes as is, |
| // or if the apply_flags are compatible with the outstanding attributes. |
| // If the oustanding attributes are incompatible with the new attribute |
| // or the apply flags, then a protective saveLayer will be executed. |
| // --------------------- |
| bool needs_save_layer(int flags) const; |
| void do_save(); |
| void save_layer(const SkRect& bounds); |
| void maybe_save_layer_for_transform(bool needs_save); |
| void maybe_save_layer_for_clip(bool needs_save); |
| void maybe_save_layer(int apply_flags); |
| void maybe_save_layer(SkScalar opacity); |
| void maybe_save_layer(const std::shared_ptr<const DlColorFilter>& filter); |
| void maybe_save_layer(const std::shared_ptr<const DlImageFilter>& filter); |
| // --------------------- |
| |
| struct RenderingAttributes { |
| // We need to record the last bounds we received for the last |
| // attribute that we recorded so that we can perform a saveLayer |
| // on the proper area. When an attribute is applied that cannot |
| // be merged with the existing attributes, it will be submitted |
| // with a bounds for its own source content, not the bounds for |
| // the content that will be included in the saveLayer that applies |
| // the existing outstanding attributes - thus we need to record |
| // the bounds that were supplied with the most recent previous |
| // attribute to be applied. |
| SkRect save_layer_bounds{0, 0, 0, 0}; |
| |
| SkScalar opacity = SK_Scalar1; |
| std::shared_ptr<const DlColorFilter> color_filter; |
| std::shared_ptr<const DlImageFilter> image_filter; |
| |
| DlPaint* fill(DlPaint& paint, |
| DlBlendMode mode = DlBlendMode::kSrcOver) const; |
| |
| bool operator==(const RenderingAttributes& other) const { |
| return save_layer_bounds == other.save_layer_bounds && |
| opacity == other.opacity && |
| Equals(color_filter, other.color_filter) && |
| Equals(image_filter, other.image_filter); |
| } |
| }; |
| |
| class StateEntry { |
| public: |
| virtual ~StateEntry() = default; |
| |
| virtual void apply(LayerStateStack* stack) const = 0; |
| virtual void reapply(LayerStateStack* stack) const { apply(stack); } |
| virtual void restore(LayerStateStack* stack) const {} |
| virtual void update_mutators(MutatorsStack* mutators_stack) const {} |
| |
| protected: |
| StateEntry() = default; |
| |
| FML_DISALLOW_COPY_ASSIGN_AND_MOVE(StateEntry); |
| }; |
| friend class SaveEntry; |
| friend class SaveLayerEntry; |
| friend class BackdropFilterEntry; |
| friend class OpacityEntry; |
| friend class ImageFilterEntry; |
| friend class ColorFilterEntry; |
| friend class TranslateEntry; |
| friend class TransformMatrixEntry; |
| friend class TransformM44Entry; |
| friend class IntegralTransformEntry; |
| friend class ClipEntry; |
| friend class ClipRectEntry; |
| friend class ClipRRectEntry; |
| friend class ClipPathEntry; |
| |
| class Delegate { |
| protected: |
| using ClipOp = DlCanvas::ClipOp; |
| |
| public: |
| virtual ~Delegate() = default; |
| |
| // Mormally when a |Paint| or |Preroll| cycle is completed, the |
| // delegate will have been rewound to its initial state by the |
| // trailing recursive actions of the paint and preroll methods. |
| // When a delegate is swapped out, there may be unresolved state |
| // that the delegate received. This method is called when the |
| // delegate is cleared or swapped out to inform it to rewind its |
| // state and finalize all outstanding save or saveLayer operations. |
| virtual void decommission() = 0; |
| |
| virtual DlCanvas* canvas() const { return nullptr; } |
| |
| virtual SkRect local_cull_rect() const = 0; |
| virtual SkRect device_cull_rect() const = 0; |
| virtual SkM44 matrix_4x4() const = 0; |
| virtual SkMatrix matrix_3x3() const = 0; |
| virtual bool content_culled(const SkRect& content_bounds) const = 0; |
| |
| virtual void save() = 0; |
| virtual void saveLayer(const SkRect& bounds, |
| RenderingAttributes& attributes, |
| DlBlendMode blend, |
| const DlImageFilter* backdrop) = 0; |
| virtual void restore() = 0; |
| |
| virtual void translate(SkScalar tx, SkScalar ty) = 0; |
| virtual void transform(const SkM44& m44) = 0; |
| virtual void transform(const SkMatrix& matrix) = 0; |
| virtual void integralTransform() = 0; |
| |
| virtual void clipRect(const SkRect& rect, ClipOp op, bool is_aa) = 0; |
| virtual void clipRRect(const SkRRect& rrect, ClipOp op, bool is_aa) = 0; |
| virtual void clipPath(const SkPath& path, ClipOp op, bool is_aa) = 0; |
| }; |
| friend class DummyDelegate; |
| friend class DlCanvasDelegate; |
| friend class PrerollDelegate; |
| |
| std::vector<std::unique_ptr<StateEntry>> state_stack_; |
| friend class MutatorContext; |
| |
| std::shared_ptr<Delegate> delegate_; |
| RenderingAttributes outstanding_; |
| CheckerboardFunc checkerboard_func_ = nullptr; |
| |
| friend class SaveLayerEntry; |
| }; |
| |
| } // namespace flutter |
| |
| #endif // FLUTTER_FLOW_LAYERS_LAYER_STATE_STACK_H_ |