1 /* 2 * Copyright (C) 2016 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 #pragma once 18 19 #include "DisplayList.h" 20 #include "GLFunctorDrawable.h" 21 #include "RenderNodeDrawable.h" 22 23 #include <deque> 24 #include <SkLiteDL.h> 25 #include <SkLiteRecorder.h> 26 27 namespace android { 28 namespace uirenderer { 29 30 class Outline; 31 32 namespace skiapipeline { 33 34 /** 35 * This class is intended to be self contained, but still subclasses from 36 * DisplayList to make it easier to support switching between the two at 37 * runtime. The downside of this inheritance is that we pay for the overhead 38 * of the parent class construction/destruction without any real benefit. 39 */ 40 class SkiaDisplayList : public DisplayList { 41 public: SkiaDisplayList()42 SkiaDisplayList() { SkASSERT(projectionReceiveIndex == -1); } ~SkiaDisplayList()43 virtual ~SkiaDisplayList() { 44 /* Given that we are using a LinearStdAllocator to store some of the 45 * SkDrawable contents we must ensure that any other object that is 46 * holding a reference to those drawables is destroyed prior to their 47 * deletion. 48 */ 49 mDisplayList.reset(); 50 } 51 52 /** 53 * This resets the DisplayList so that it behaves as if the object were newly 54 * constructed. The reuse avoids any overhead associated with destroying 55 * the SkLiteDL as well as the deques and vectors. 56 */ 57 void reset(); 58 59 /** 60 * Use the linear allocator to create any SkDrawables needed by the display 61 * list. This could be dangerous as these objects are ref-counted, so we 62 * need to monitor that they don't extend beyond the lifetime of the class 63 * that creates them. 64 */ 65 template<class T, typename... Params> allocateDrawable(Params &&...params)66 SkDrawable* allocateDrawable(Params&&... params) { 67 return allocator.create<T>(std::forward<Params>(params)...); 68 } 69 isSkiaDL()70 bool isSkiaDL() const override { return true; } 71 72 /** 73 * Returns true if the DisplayList does not have any recorded content 74 */ isEmpty()75 bool isEmpty() const override { return mDisplayList.empty(); } 76 77 /** 78 * Returns true if this list directly contains a GLFunctor drawing command. 79 */ hasFunctor()80 bool hasFunctor() const override { return !mChildFunctors.empty(); } 81 82 /** 83 * Returns true if this list directly contains a VectorDrawable drawing command. 84 */ hasVectorDrawables()85 bool hasVectorDrawables() const override { return !mVectorDrawables.empty(); } 86 87 /** 88 * Attempts to reset and reuse this DisplayList. 89 * 90 * @return true if the displayList will be reused and therefore should not be deleted 91 */ 92 bool reuseDisplayList(RenderNode* node, renderthread::CanvasContext* context) override; 93 94 /** 95 * ONLY to be called by RenderNode::syncDisplayList so that we can notify any 96 * contained VectorDrawables or GLFunctors to sync their state. 97 * 98 * NOTE: This function can be folded into RenderNode when we no longer need 99 * to subclass from DisplayList 100 */ 101 void syncContents() override; 102 103 /** 104 * ONLY to be called by RenderNode::prepareTree in order to prepare this 105 * list while the UI thread is blocked. Here we can upload mutable bitmaps 106 * and notify our parent if any of our content has been invalidated and in 107 * need of a redraw. If the renderNode has any children then they are also 108 * call in order to prepare them. 109 * 110 * @return true if any content change requires the node to be invalidated 111 * 112 * NOTE: This function can be folded into RenderNode when we no longer need 113 * to subclass from DisplayList 114 */ 115 116 bool prepareListAndChildren(TreeObserver& observer, TreeInfo& info, bool functorsNeedLayer, 117 std::function<void(RenderNode*, TreeObserver&, TreeInfo&, bool)> childFn) override; 118 119 /** 120 * Calls the provided function once for each child of this DisplayList 121 */ 122 void updateChildren(std::function<void(RenderNode*)> updateFn) override; 123 124 /** 125 * Returns true if there is a child render node that is a projection receiver. 126 */ containsProjectionReceiver()127 inline bool containsProjectionReceiver() const { return mProjectionReceiver; } 128 attachRecorder(SkLiteRecorder * recorder,const SkIRect & bounds)129 void attachRecorder(SkLiteRecorder* recorder, const SkIRect& bounds) { 130 recorder->reset(&mDisplayList, bounds); 131 } 132 draw(SkCanvas * canvas)133 void draw(SkCanvas* canvas) { mDisplayList.draw(canvas); } 134 135 void output(std::ostream& output, uint32_t level) override; 136 137 /** 138 * We use std::deque here because (1) we need to iterate through these 139 * elements and (2) mDisplayList holds pointers to the elements, so they 140 * cannot relocate. 141 */ 142 std::deque<RenderNodeDrawable> mChildNodes; 143 std::deque<GLFunctorDrawable> mChildFunctors; 144 std::vector<SkImage*> mMutableImages; 145 std::vector<VectorDrawableRoot*> mVectorDrawables; 146 SkLiteDL mDisplayList; 147 148 //mProjectionReceiver points to a child node (stored in mChildNodes) that is as a projection 149 //receiver. It is set at record time and used at both prepare and draw tree traversals to 150 //make sure backward projected nodes are found and drawn immediately after mProjectionReceiver. 151 RenderNodeDrawable* mProjectionReceiver = nullptr; 152 153 //mProjectedOutline is valid only when render node tree is traversed during the draw pass. 154 //Render nodes that have a child receiver node, will store a pointer to their outline in 155 //mProjectedOutline. Child receiver node will apply the clip before any backward projected 156 //node is drawn. 157 const Outline* mProjectedOutline = nullptr; 158 159 //mProjectedReceiverParentMatrix is valid when render node tree is traversed during the draw 160 //pass. Render nodes that have a child receiver node, will store their matrix in 161 //mProjectedReceiverParentMatrix. Child receiver node will set the matrix and then clip with the 162 //outline of their parent. 163 SkMatrix mProjectedReceiverParentMatrix; 164 }; 165 166 }; // namespace skiapipeline 167 }; // namespace uirenderer 168 }; // namespace android 169