// Copyright 2012 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_TREES_LAYER_TREE_IMPL_H_ #define CC_TREES_LAYER_TREE_IMPL_H_ #include #include #include #include "base/containers/hash_tables.h" #include "base/values.h" #include "cc/base/scoped_ptr_vector.h" #include "cc/base/synced_property.h" #include "cc/input/layer_selection_bound.h" #include "cc/layers/layer_impl.h" #include "cc/output/begin_frame_args.h" #include "cc/output/renderer.h" #include "cc/output/swap_promise.h" #include "cc/resources/ui_resource_client.h" #include "cc/trees/layer_tree_host_impl.h" #include "cc/trees/property_tree.h" namespace base { namespace trace_event { class TracedValue; } } namespace cc { class ContextProvider; class DebugRectHistory; class FrameRateCounter; class HeadsUpDisplayLayerImpl; class LayerExternalScrollOffsetListener; class LayerScrollOffsetDelegate; class LayerTreeDebugState; class LayerTreeImpl; class LayerTreeSettings; class MemoryHistory; class OutputSurface; class PageScaleAnimation; class PictureLayerImpl; class Proxy; class ResourceProvider; class TileManager; class UIResourceRequest; class VideoFrameControllerClient; struct PendingPageScaleAnimation; struct RendererCapabilities; typedef std::vector UIResourceRequestQueue; typedef SyncedProperty> SyncedTopControls; typedef SyncedProperty> SyncedElasticOverscroll; class CC_EXPORT LayerTreeImpl { public: static scoped_ptr create( LayerTreeHostImpl* layer_tree_host_impl, scoped_refptr> page_scale_factor, scoped_refptr top_controls_shown_ratio, scoped_refptr elastic_overscroll) { return make_scoped_ptr( new LayerTreeImpl(layer_tree_host_impl, page_scale_factor, top_controls_shown_ratio, elastic_overscroll)); } virtual ~LayerTreeImpl(); void Shutdown(); void ReleaseResources(); void RecreateResources(); // Methods called by the layer tree that pass-through or access LTHI. // --------------------------------------------------------------------------- const LayerTreeSettings& settings() const; const LayerTreeDebugState& debug_state() const; const RendererCapabilitiesImpl& GetRendererCapabilities() const; ContextProvider* context_provider() const; OutputSurface* output_surface() const; ResourceProvider* resource_provider() const; TileManager* tile_manager() const; FrameRateCounter* frame_rate_counter() const; MemoryHistory* memory_history() const; gfx::Size device_viewport_size() const; DebugRectHistory* debug_rect_history() const; bool IsActiveTree() const; bool IsPendingTree() const; bool IsRecycleTree() const; bool IsSyncTree() const; LayerImpl* FindActiveTreeLayerById(int id); LayerImpl* FindPendingTreeLayerById(int id); bool PinchGestureActive() const; BeginFrameArgs CurrentBeginFrameArgs() const; base::TimeDelta CurrentBeginFrameInterval() const; void SetNeedsCommit(); gfx::Rect DeviceViewport() const; gfx::Size DrawViewportSize() const; const gfx::Rect ViewportRectForTilePriority() const; scoped_ptr CreateScrollbarAnimationController( LayerImpl* scrolling_layer); void DidAnimateScrollOffset(); void InputScrollAnimationFinished(); bool use_gpu_rasterization() const; GpuRasterizationStatus GetGpuRasterizationStatus() const; bool create_low_res_tiling() const; BlockingTaskRunner* BlockingMainThreadTaskRunner() const; bool RequiresHighResToDraw() const; bool SmoothnessTakesPriority() const; VideoFrameControllerClient* GetVideoFrameControllerClient() const; // Tree specific methods exposed to layer-impl tree. // --------------------------------------------------------------------------- void SetNeedsRedraw(); // Tracing methods. // --------------------------------------------------------------------------- void GetAllPrioritizedTilesForTracing( std::vector* prioritized_tiles) const; void AsValueInto(base::trace_event::TracedValue* dict) const; // Other public methods // --------------------------------------------------------------------------- LayerImpl* root_layer() const { return root_layer_.get(); } void SetRootLayer(scoped_ptr); scoped_ptr DetachLayerTree(); void SetPropertyTrees(const PropertyTrees& property_trees) { property_trees_ = property_trees; property_trees_.transform_tree.set_source_to_parent_updates_allowed(false); } PropertyTrees* property_trees() { return &property_trees_; } void UpdatePropertyTreesForBoundsDelta(); void PushPropertiesTo(LayerTreeImpl* tree_impl); int source_frame_number() const { return source_frame_number_; } void set_source_frame_number(int frame_number) { source_frame_number_ = frame_number; } HeadsUpDisplayLayerImpl* hud_layer() { return hud_layer_; } void set_hud_layer(HeadsUpDisplayLayerImpl* layer_impl) { hud_layer_ = layer_impl; } LayerImpl* InnerViewportScrollLayer() const; // This function may return NULL, it is the caller's responsibility to check. LayerImpl* OuterViewportScrollLayer() const; gfx::ScrollOffset TotalScrollOffset() const; gfx::ScrollOffset TotalMaxScrollOffset() const; LayerImpl* InnerViewportContainerLayer() const; LayerImpl* OuterViewportContainerLayer() const; LayerImpl* CurrentlyScrollingLayer() const; void SetCurrentlyScrollingLayer(LayerImpl* layer); void ClearCurrentlyScrollingLayer(); void SetViewportLayersFromIds(int overscroll_elasticity_layer, int page_scale_layer_id, int inner_viewport_scroll_layer_id, int outer_viewport_scroll_layer_id); void ClearViewportLayers(); LayerImpl* OverscrollElasticityLayer() { return LayerById(overscroll_elasticity_layer_id_); } LayerImpl* PageScaleLayer() { return LayerById(page_scale_layer_id_); } void ApplySentScrollAndScaleDeltasFromAbortedCommit(); SkColor background_color() const { return background_color_; } void set_background_color(SkColor color) { background_color_ = color; } bool has_transparent_background() const { return has_transparent_background_; } void set_has_transparent_background(bool transparent) { has_transparent_background_ = transparent; } void UpdatePropertyTreeScrollingAndAnimationFromMainThread(); void SetPageScaleOnActiveTree(float active_page_scale); void PushPageScaleFromMainThread(float page_scale_factor, float min_page_scale_factor, float max_page_scale_factor); float current_page_scale_factor() const { return page_scale_factor()->Current(IsActiveTree()); } float min_page_scale_factor() const { return min_page_scale_factor_; } float max_page_scale_factor() const { return max_page_scale_factor_; } float page_scale_delta() const { return page_scale_factor()->Delta(); } SyncedProperty* page_scale_factor(); const SyncedProperty* page_scale_factor() const; void SetDeviceScaleFactor(float device_scale_factor); float device_scale_factor() const { return device_scale_factor_; } void set_hide_pinch_scrollbars_near_min_scale(bool hide) { hide_pinch_scrollbars_near_min_scale_ = hide; } SyncedElasticOverscroll* elastic_overscroll() { return elastic_overscroll_.get(); } const SyncedElasticOverscroll* elastic_overscroll() const { return elastic_overscroll_.get(); } SyncedTopControls* top_controls_shown_ratio() { return top_controls_shown_ratio_.get(); } const SyncedTopControls* top_controls_shown_ratio() const { return top_controls_shown_ratio_.get(); } // Updates draw properties and render surface layer list, as well as tile // priorities. Returns false if it was unable to update. Updating lcd // text may cause invalidations, so should only be done after a commit. bool UpdateDrawProperties(bool update_lcd_text); void BuildPropertyTreesForTesting(); void set_needs_update_draw_properties() { needs_update_draw_properties_ = true; } bool needs_update_draw_properties() const { return needs_update_draw_properties_; } void set_needs_full_tree_sync(bool needs) { needs_full_tree_sync_ = needs; } bool needs_full_tree_sync() const { return needs_full_tree_sync_; } void ForceRedrawNextActivation() { next_activation_forces_redraw_ = true; } void set_has_ever_been_drawn(bool has_drawn) { has_ever_been_drawn_ = has_drawn; } bool has_ever_been_drawn() const { return has_ever_been_drawn_; } void set_ui_resource_request_queue(const UIResourceRequestQueue& queue); const LayerImplList& RenderSurfaceLayerList() const; const Region& UnoccludedScreenSpaceRegion() const; // These return the size of the root scrollable area and the size of // the user-visible scrolling viewport, in CSS layout coordinates. gfx::SizeF ScrollableSize() const; gfx::SizeF ScrollableViewportSize() const; gfx::Rect RootScrollLayerDeviceViewportBounds() const; LayerImpl* LayerById(int id) const; // These should be called by LayerImpl's ctor/dtor. void RegisterLayer(LayerImpl* layer); void UnregisterLayer(LayerImpl* layer); size_t NumLayers(); AnimationRegistrar* GetAnimationRegistrar() const; void DidBecomeActive(); // Set on the active tree when the viewport size recently changed // and the active tree's size is now out of date. bool ViewportSizeInvalid() const; void SetViewportSizeInvalid(); void ResetViewportSizeInvalid(); // Useful for debug assertions, probably shouldn't be used for anything else. Proxy* proxy() const; // Distribute the root scroll between outer and inner viewport scroll layer. // The outer viewport scroll layer scrolls first. void DistributeRootScrollOffset(const gfx::ScrollOffset& root_offset); void ApplyScroll(LayerImpl* layer, ScrollState* scroll_state) { layer_tree_host_impl_->ApplyScroll(layer, scroll_state); } // Call this function when you expect there to be a swap buffer. // See swap_promise.h for how to use SwapPromise. // // A swap promise queued by QueueSwapPromise travels with the layer // information currently associated with the tree. For example, when // a pending tree is activated, the swap promise is passed to the // active tree along with the layer information. Similarly, when a // new activation overwrites layer information on the active tree, // queued swap promises are broken. void QueueSwapPromise(scoped_ptr swap_promise); // Queue a swap promise, pinned to this tree. Pinned swap promises // may only be queued on the active tree. // // An active tree pinned swap promise will see only DidSwap() or // DidNotSwap(SWAP_FAILS). No DidActivate() will be seen because // that has already happened prior to queueing of the swap promise. // // Pinned active tree swap promises will not be broken prematurely // on the active tree if a new tree is activated. void QueuePinnedSwapPromise(scoped_ptr swap_promise); // Take the |new_swap_promise| and append it to |swap_promise_list_|. void PassSwapPromises(ScopedPtrVector* new_swap_promise); void FinishSwapPromises(CompositorFrameMetadata* metadata); void BreakSwapPromises(SwapPromise::DidNotSwapReason reason); void DidModifyTilePriorities(); ResourceId ResourceIdForUIResource(UIResourceId uid) const; void ProcessUIResourceRequestQueue(); bool IsUIResourceOpaque(UIResourceId uid) const; void RegisterPictureLayerImpl(PictureLayerImpl* layer); void UnregisterPictureLayerImpl(PictureLayerImpl* layer); const std::vector& picture_layers() const { return picture_layers_; } void AddLayerWithCopyOutputRequest(LayerImpl* layer); void RemoveLayerWithCopyOutputRequest(LayerImpl* layer); const std::vector& LayersWithCopyOutputRequest() const; int current_render_surface_list_id() const { return render_surface_layer_list_id_; } LayerImpl* FindFirstScrollingLayerThatIsHitByPoint( const gfx::PointF& screen_space_point); LayerImpl* FindLayerThatIsHitByPoint(const gfx::PointF& screen_space_point); LayerImpl* FindLayerWithWheelHandlerThatIsHitByPoint( const gfx::PointF& screen_space_point); LayerImpl* FindLayerThatIsHitByPointInTouchHandlerRegion( const gfx::PointF& screen_space_point); void RegisterSelection(const LayerSelection& selection); // Compute the current selection handle location and visbility with respect to // the viewport. void GetViewportSelection(ViewportSelection* selection); void set_top_controls_shrink_blink_size(bool shrink); bool top_controls_shrink_blink_size() const { return top_controls_shrink_blink_size_; } bool SetCurrentTopControlsShownRatio(float ratio); float CurrentTopControlsShownRatio() const { return top_controls_shown_ratio_->Current(IsActiveTree()); } void set_top_controls_height(float top_controls_height); float top_controls_height() const { return top_controls_height_; } void PushTopControlsFromMainThread(float top_controls_shown_ratio); void SetPendingPageScaleAnimation( scoped_ptr pending_animation); scoped_ptr TakePendingPageScaleAnimation(); void GatherFrameTimingRequestIds(std::vector* request_ids); bool IsAnimatingFilterProperty(const LayerImpl* layer) const; bool IsAnimatingOpacityProperty(const LayerImpl* layer) const; bool IsAnimatingTransformProperty(const LayerImpl* layer) const; bool HasPotentiallyRunningFilterAnimation(const LayerImpl* layer) const; bool HasPotentiallyRunningOpacityAnimation(const LayerImpl* layer) const; bool HasPotentiallyRunningTransformAnimation(const LayerImpl* layer) const; bool HasAnyAnimationTargetingProperty( const LayerImpl* layer, Animation::TargetProperty property) const; bool FilterIsAnimatingOnImplOnly(const LayerImpl* layer) const; bool OpacityIsAnimatingOnImplOnly(const LayerImpl* layer) const; bool TransformIsAnimatingOnImplOnly(const LayerImpl* layer) const; bool HasOnlyTranslationTransforms(const LayerImpl* layer) const; bool MaximumTargetScale(const LayerImpl* layer, float* max_scale) const; bool AnimationStartScale(const LayerImpl* layer, float* start_scale) const; bool HasFilterAnimationThatInflatesBounds(const LayerImpl* layer) const; bool HasTransformAnimationThatInflatesBounds(const LayerImpl* layer) const; bool HasAnimationThatInflatesBounds(const LayerImpl* layer) const; bool FilterAnimationBoundsForBox(const LayerImpl* layer, const gfx::BoxF& box, gfx::BoxF* bounds) const; bool TransformAnimationBoundsForBox(const LayerImpl* layer, const gfx::BoxF& box, gfx::BoxF* bounds) const; protected: explicit LayerTreeImpl( LayerTreeHostImpl* layer_tree_host_impl, scoped_refptr> page_scale_factor, scoped_refptr top_controls_shown_ratio, scoped_refptr elastic_overscroll); float ClampPageScaleFactorToLimits(float page_scale_factor) const; void PushPageScaleFactorAndLimits(const float* page_scale_factor, float min_page_scale_factor, float max_page_scale_factor); bool SetPageScaleFactorLimits(float min_page_scale_factor, float max_page_scale_factor); void DidUpdatePageScale(); void HideInnerViewportScrollbarsIfNeeded(); void PushTopControls(const float* top_controls_shown_ratio); LayerTreeHostImpl* layer_tree_host_impl_; int source_frame_number_; scoped_ptr root_layer_; HeadsUpDisplayLayerImpl* hud_layer_; PropertyTrees property_trees_; SkColor background_color_; bool has_transparent_background_; int currently_scrolling_layer_id_; int overscroll_elasticity_layer_id_; int page_scale_layer_id_; int inner_viewport_scroll_layer_id_; int outer_viewport_scroll_layer_id_; LayerSelection selection_; scoped_refptr> page_scale_factor_; float min_page_scale_factor_; float max_page_scale_factor_; bool hide_pinch_scrollbars_near_min_scale_; float device_scale_factor_; scoped_refptr elastic_overscroll_; typedef base::hash_map LayerIdMap; LayerIdMap layer_id_map_; std::vector picture_layers_; std::vector layers_with_copy_output_request_; // List of visible layers for the most recently prepared frame. LayerImplList render_surface_layer_list_; // After drawing the |render_surface_layer_list_| the areas in this region // would not be fully covered by opaque content. Region unoccluded_screen_space_region_; bool viewport_size_invalid_; bool needs_update_draw_properties_; // In impl-side painting mode, this is true when the tree may contain // structural differences relative to the active tree. bool needs_full_tree_sync_; bool next_activation_forces_redraw_; bool has_ever_been_drawn_; ScopedPtrVector swap_promise_list_; ScopedPtrVector pinned_swap_promise_list_; UIResourceRequestQueue ui_resource_request_queue_; int render_surface_layer_list_id_; // Whether or not Blink's viewport size was shrunk by the height of the top // controls at the time of the last layout. bool top_controls_shrink_blink_size_; float top_controls_height_; // The amount that the top controls are shown from 0 (hidden) to 1 (fully // shown). scoped_refptr top_controls_shown_ratio_; scoped_ptr pending_page_scale_animation_; private: DISALLOW_COPY_AND_ASSIGN(LayerTreeImpl); }; } // namespace cc #endif // CC_TREES_LAYER_TREE_IMPL_H_