// Copyright 2011 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. #ifndef CC_LAYER_IMPL_H_ #define CC_LAYER_IMPL_H_ #include #include "base/logging.h" #include "base/memory/scoped_ptr.h" #include "cc/cc_export.h" #include "cc/draw_properties.h" #include "cc/input_handler.h" #include "cc/layer_animation_controller.h" #include "cc/layer_animation_value_observer.h" #include "cc/region.h" #include "cc/render_pass.h" #include "cc/render_surface_impl.h" #include "cc/resource_provider.h" #include "cc/scoped_ptr_vector.h" #include "cc/shared_quad_state.h" #include "skia/ext/refptr.h" #include "third_party/WebKit/Source/Platform/chromium/public/WebFilterOperations.h" #include "third_party/skia/include/core/SkColor.h" #include "third_party/skia/include/core/SkImageFilter.h" #include "third_party/skia/include/core/SkPicture.h" #include "ui/gfx/rect.h" #include "ui/gfx/rect_f.h" #include "ui/gfx/transform.h" namespace base { class DictionaryValue; } namespace cc { class LayerTreeHostImpl; class LayerTreeImpl; class QuadSink; class Renderer; class ScrollbarAnimationController; class ScrollbarLayerImpl; class Layer; struct AppendQuadsData; class CC_EXPORT LayerImpl : LayerAnimationValueObserver { public: typedef ScopedPtrVector LayerList; static scoped_ptr create(LayerTreeImpl* treeImpl, int id) { return make_scoped_ptr(new LayerImpl(treeImpl, id)); } virtual ~LayerImpl(); int id() const; // LayerAnimationValueObserver implementation. virtual void OnOpacityAnimated(float) OVERRIDE; virtual void OnTransformAnimated(const gfx::Transform&) OVERRIDE; virtual bool IsActive() const OVERRIDE; // Tree structure. LayerImpl* parent() { return m_parent; } const LayerImpl* parent() const { return m_parent; } const LayerList& children() const { return m_children; } LayerList& children() { return m_children; } LayerImpl* childAt(size_t index) const; void addChild(scoped_ptr); scoped_ptr removeChild(LayerImpl* child); void removeAllChildren(); void setParent(LayerImpl* parent) { m_parent = parent; } void clearChildList(); // Warning: This does not preserve tree structure invariants. void setMaskLayer(scoped_ptr); LayerImpl* maskLayer() { return m_maskLayer.get(); } const LayerImpl* maskLayer() const { return m_maskLayer.get(); } scoped_ptr takeMaskLayer(); void setReplicaLayer(scoped_ptr); LayerImpl* replicaLayer() { return m_replicaLayer.get(); } const LayerImpl* replicaLayer() const { return m_replicaLayer.get(); } scoped_ptr takeReplicaLayer(); bool hasMask() const { return m_maskLayer; } bool hasReplica() const { return m_replicaLayer; } bool replicaHasMask() const { return m_replicaLayer && (m_maskLayer || m_replicaLayer->m_maskLayer); } LayerTreeImpl* layerTreeImpl() const { return m_layerTreeImpl; } scoped_ptr createSharedQuadState() const; // willDraw must be called before appendQuads. If willDraw is called, // didDraw is guaranteed to be called before another willDraw or before // the layer is destroyed. To enforce this, any class that overrides // willDraw/didDraw must call the base class version. virtual void willDraw(ResourceProvider*); virtual void appendQuads(QuadSink&, AppendQuadsData&) { } virtual void didDraw(ResourceProvider*); virtual ResourceProvider::ResourceId contentsResourceId() const; virtual bool hasDelegatedContent() const; virtual bool hasContributingDelegatedRenderPasses() const; virtual RenderPass::Id firstContributingRenderPassId() const; virtual RenderPass::Id nextContributingRenderPassId(RenderPass::Id) const; virtual void updateTilePriorities() { } virtual ScrollbarLayerImpl* toScrollbarLayer(); // Returns true if this layer has content to draw. void setDrawsContent(bool); bool drawsContent() const { return m_drawsContent; } bool forceRenderSurface() const { return m_forceRenderSurface; } void setForceRenderSurface(bool force) { m_forceRenderSurface = force; } void setAnchorPoint(const gfx::PointF&); const gfx::PointF& anchorPoint() const { return m_anchorPoint; } void setAnchorPointZ(float); float anchorPointZ() const { return m_anchorPointZ; } void setBackgroundColor(SkColor); SkColor backgroundColor() const { return m_backgroundColor; } void setFilters(const WebKit::WebFilterOperations&); const WebKit::WebFilterOperations& filters() const { return m_filters; } void setBackgroundFilters(const WebKit::WebFilterOperations&); const WebKit::WebFilterOperations& backgroundFilters() const { return m_backgroundFilters; } void setFilter(const skia::RefPtr&); skia::RefPtr filter() const { return m_filter; } void setMasksToBounds(bool); bool masksToBounds() const { return m_masksToBounds; } void setContentsOpaque(bool); bool contentsOpaque() const { return m_contentsOpaque; } void setOpacity(float); float opacity() const; bool opacityIsAnimating() const; void setPosition(const gfx::PointF&); const gfx::PointF& position() const { return m_position; } void setIsContainerForFixedPositionLayers(bool isContainerForFixedPositionLayers) { m_isContainerForFixedPositionLayers = isContainerForFixedPositionLayers; } bool isContainerForFixedPositionLayers() const { return m_isContainerForFixedPositionLayers; } void setFixedToContainerLayer(bool fixedToContainerLayer = true) { m_fixedToContainerLayer = fixedToContainerLayer;} bool fixedToContainerLayer() const { return m_fixedToContainerLayer; } void setPreserves3D(bool); bool preserves3D() const { return m_preserves3D; } void setUseParentBackfaceVisibility(bool useParentBackfaceVisibility) { m_useParentBackfaceVisibility = useParentBackfaceVisibility; } bool useParentBackfaceVisibility() const { return m_useParentBackfaceVisibility; } void setSublayerTransform(const gfx::Transform&); const gfx::Transform& sublayerTransform() const { return m_sublayerTransform; } // Debug layer name. void setDebugName(const std::string& debugName) { m_debugName = debugName; } std::string debugName() const { return m_debugName; } bool showDebugBorders() const; // These invalidate the host's render surface layer list. The caller // is responsible for calling setNeedsUpdateDrawProperties on the host // so that its list can be recreated. void createRenderSurface(); void clearRenderSurface() { m_drawProperties.render_surface.reset(); } DrawProperties& drawProperties() { return m_drawProperties; } const DrawProperties& drawProperties() const { return m_drawProperties; } // The following are shortcut accessors to get various information from m_drawProperties const gfx::Transform& drawTransform() const { return m_drawProperties.target_space_transform; } const gfx::Transform& screenSpaceTransform() const { return m_drawProperties.screen_space_transform; } float drawOpacity() const { return m_drawProperties.opacity; } bool drawOpacityIsAnimating() const { return m_drawProperties.opacity_is_animating; } bool drawTransformIsAnimating() const { return m_drawProperties.target_space_transform_is_animating; } bool screenSpaceTransformIsAnimating() const { return m_drawProperties.screen_space_transform_is_animating; } bool screenSpaceOpacityIsAnimating() const { return m_drawProperties.screen_space_opacity_is_animating; } bool canUseLCDText() const { return m_drawProperties.can_use_lcd_text; } bool isClipped() const { return m_drawProperties.is_clipped; } const gfx::Rect& clipRect() const { return m_drawProperties.clip_rect; } const gfx::Rect& drawableContentRect() const { return m_drawProperties.drawable_content_rect; } const gfx::Rect& visibleContentRect() const { return m_drawProperties.visible_content_rect; } LayerImpl* renderTarget() { DCHECK(!m_drawProperties.render_target || m_drawProperties.render_target->renderSurface()); return m_drawProperties.render_target; } const LayerImpl* renderTarget() const { DCHECK(!m_drawProperties.render_target || m_drawProperties.render_target->renderSurface()); return m_drawProperties.render_target; } RenderSurfaceImpl* renderSurface() const { return m_drawProperties.render_surface.get(); } // The client should be responsible for setting bounds, contentBounds and // contentsScale to appropriate values. LayerImpl doesn't calculate any of // them from the other values. void setBounds(const gfx::Size&); const gfx::Size& bounds() const { return m_bounds; } void setContentBounds(const gfx::Size&); gfx::Size contentBounds() const { return m_drawProperties.content_bounds; } float contentsScaleX() const { return m_drawProperties.contents_scale_x; } float contentsScaleY() const { return m_drawProperties.contents_scale_y; } void setContentsScale(float contentsScaleX, float contentsScaleY); virtual void calculateContentsScale( float idealContentsScale, bool animatingTransformToScreen, float* contentsScaleX, float* contentsScaleY, gfx::Size* contentBounds); gfx::Vector2d scrollOffset() const { return m_scrollOffset; } void setScrollOffset(gfx::Vector2d); gfx::Vector2d maxScrollOffset() const {return m_maxScrollOffset; } void setMaxScrollOffset(gfx::Vector2d); const gfx::Vector2dF& scrollDelta() const { return m_scrollDelta; } void setScrollDelta(const gfx::Vector2dF&); const gfx::Transform& implTransform() const { return m_implTransform; } void setImplTransform(const gfx::Transform& transform); const gfx::Vector2d& sentScrollDelta() const { return m_sentScrollDelta; } void setSentScrollDelta(const gfx::Vector2d& sentScrollDelta); // Returns the delta of the scroll that was outside of the bounds of the initial scroll gfx::Vector2dF scrollBy(const gfx::Vector2dF& scroll); bool scrollable() const { return m_scrollable; } void setScrollable(bool scrollable) { m_scrollable = scrollable; } bool shouldScrollOnMainThread() const { return m_shouldScrollOnMainThread; } void setShouldScrollOnMainThread(bool shouldScrollOnMainThread) { m_shouldScrollOnMainThread = shouldScrollOnMainThread; } bool haveWheelEventHandlers() const { return m_haveWheelEventHandlers; } void setHaveWheelEventHandlers(bool haveWheelEventHandlers) { m_haveWheelEventHandlers = haveWheelEventHandlers; } const Region& nonFastScrollableRegion() const { return m_nonFastScrollableRegion; } void setNonFastScrollableRegion(const Region& region) { m_nonFastScrollableRegion = region; } const Region& touchEventHandlerRegion() const { return m_touchEventHandlerRegion; } void setTouchEventHandlerRegion(const Region& region) { m_touchEventHandlerRegion = region; } void setDrawCheckerboardForMissingTiles(bool checkerboard) { m_drawCheckerboardForMissingTiles = checkerboard; } bool drawCheckerboardForMissingTiles() const; InputHandlerClient::ScrollStatus tryScroll(const gfx::PointF& screenSpacePoint, InputHandlerClient::ScrollInputType) const; bool doubleSided() const { return m_doubleSided; } void setDoubleSided(bool); void setTransform(const gfx::Transform&); const gfx::Transform& transform() const; bool transformIsAnimating() const; const gfx::RectF& updateRect() const { return m_updateRect; } void setUpdateRect(const gfx::RectF& updateRect) { m_updateRect = updateRect; } std::string layerTreeAsText() const; virtual base::DictionaryValue* layerTreeAsJson() const; void setStackingOrderChanged(bool); bool layerPropertyChanged() const { return m_layerPropertyChanged || layerIsAlwaysDamaged(); } bool layerSurfacePropertyChanged() const; void resetAllChangeTrackingForSubtree(); virtual bool layerIsAlwaysDamaged() const; LayerAnimationController* layerAnimationController() { return m_layerAnimationController.get(); } virtual Region visibleContentOpaqueRegion() const; virtual void didBecomeActive(); // Indicates that the surface previously used to render this layer // was lost and that a new one has been created. Won't be called // until the new surface has been created successfully. virtual void didLoseOutputSurface(); ScrollbarAnimationController* scrollbarAnimationController() const { return m_scrollbarAnimationController.get(); } void setScrollbarOpacity(float opacity); void setHorizontalScrollbarLayer(ScrollbarLayerImpl*); ScrollbarLayerImpl* horizontalScrollbarLayer() { return m_horizontalScrollbarLayer; } void setVerticalScrollbarLayer(ScrollbarLayerImpl*); ScrollbarLayerImpl* verticalScrollbarLayer() { return m_verticalScrollbarLayer; } gfx::Rect layerRectToContentRect(const gfx::RectF& layerRect) const; virtual skia::RefPtr getPicture(); virtual bool canClipSelf() const; virtual bool areVisibleResourcesReady() const; virtual scoped_ptr createLayerImpl(LayerTreeImpl*); virtual void pushPropertiesTo(LayerImpl*); protected: LayerImpl(LayerTreeImpl* layerImpl, int); // Get the color and size of the layer's debug border. virtual void getDebugBorderProperties(SkColor*, float* width) const; void appendDebugBorderQuad(QuadSink&, const SharedQuadState*, AppendQuadsData&) const; virtual void dumpLayerProperties(std::string*, int indent) const; static std::string indentString(int indent); private: void updateScrollbarPositions(); void noteLayerSurfacePropertyChanged(); void noteLayerPropertyChanged(); void noteLayerPropertyChangedForSubtree(); // Note carefully this does not affect the current layer. void noteLayerPropertyChangedForDescendants(); virtual const char* layerTypeAsString() const; void dumpLayer(std::string*, int indent) const; // Properties internal to LayerImpl LayerImpl* m_parent; LayerList m_children; // m_maskLayer can be temporarily stolen during tree sync, we need this ID to confirm newly assigned layer is still the previous one int m_maskLayerId; scoped_ptr m_maskLayer; int m_replicaLayerId; // ditto scoped_ptr m_replicaLayer; int m_layerId; LayerTreeImpl* m_layerTreeImpl; // Properties synchronized from the associated Layer. gfx::PointF m_anchorPoint; float m_anchorPointZ; gfx::Size m_bounds; gfx::Vector2d m_scrollOffset; bool m_scrollable; bool m_shouldScrollOnMainThread; bool m_haveWheelEventHandlers; Region m_nonFastScrollableRegion; Region m_touchEventHandlerRegion; SkColor m_backgroundColor; bool m_stackingOrderChanged; // Whether the "back" of this layer should draw. bool m_doubleSided; // Tracks if drawing-related properties have changed since last redraw. bool m_layerPropertyChanged; // Indicates that a property has changed on this layer that would not // affect the pixels on its target surface, but would require redrawing // the targetSurface onto its ancestor targetSurface. // For layers that do not own a surface this flag acts as m_layerPropertyChanged. bool m_layerSurfacePropertyChanged; bool m_masksToBounds; bool m_contentsOpaque; float m_opacity; gfx::PointF m_position; bool m_preserves3D; bool m_useParentBackfaceVisibility; bool m_drawCheckerboardForMissingTiles; gfx::Transform m_sublayerTransform; gfx::Transform m_transform; bool m_drawsContent; bool m_forceRenderSurface; // Set for the layer that other layers are fixed to. bool m_isContainerForFixedPositionLayers; // This is true if the layer should be fixed to the closest ancestor container. bool m_fixedToContainerLayer; gfx::Vector2dF m_scrollDelta; gfx::Vector2d m_sentScrollDelta; gfx::Vector2d m_maxScrollOffset; gfx::Transform m_implTransform; gfx::Vector2dF m_lastScrollOffset; // The global depth value of the center of the layer. This value is used // to sort layers from back to front. float m_drawDepth; // Debug layer name. std::string m_debugName; WebKit::WebFilterOperations m_filters; WebKit::WebFilterOperations m_backgroundFilters; skia::RefPtr m_filter; #ifndef NDEBUG bool m_betweenWillDrawAndDidDraw; #endif // Rect indicating what was repainted/updated during update. // Note that plugin layers bypass this and leave it empty. // Uses layer's content space. gfx::RectF m_updateRect; // Manages animations for this layer. scoped_refptr m_layerAnimationController; // Manages scrollbars for this layer scoped_ptr m_scrollbarAnimationController; // Weak pointers to this layer's scrollbars, if it has them. Updated during // tree synchronization. ScrollbarLayerImpl* m_horizontalScrollbarLayer; ScrollbarLayerImpl* m_verticalScrollbarLayer; // Group of properties that need to be computed based on the layer tree // hierarchy before layers can be drawn. DrawProperties m_drawProperties; DISALLOW_COPY_AND_ASSIGN(LayerImpl); }; } #endif // CC_LAYER_IMPL_H_