// Copyright 2014 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. #include "cc/blink/web_layer_impl.h" #include #include #include #include #include "base/bind.h" #include "base/strings/string_util.h" #include "base/threading/thread_checker.h" #include "base/trace_event/trace_event_impl.h" #include "cc/base/region.h" #include "cc/base/switches.h" #include "cc/blink/web_blend_mode.h" #include "cc/layers/layer.h" #include "cc/layers/layer_position_constraint.h" #include "cc/trees/layer_tree_host.h" #include "third_party/WebKit/public/platform/WebFloatPoint.h" #include "third_party/WebKit/public/platform/WebFloatRect.h" #include "third_party/WebKit/public/platform/WebLayerPositionConstraint.h" #include "third_party/WebKit/public/platform/WebLayerScrollClient.h" #include "third_party/WebKit/public/platform/WebSize.h" #include "third_party/skia/include/utils/SkMatrix44.h" #include "ui/gfx/geometry/rect_conversions.h" #include "ui/gfx/geometry/vector2d_conversions.h" using cc::Layer; using blink::WebLayer; using blink::WebFloatPoint; using blink::WebVector; using blink::WebRect; using blink::WebSize; using blink::WebColor; namespace cc_blink { WebLayerImpl::WebLayerImpl() : layer_(Layer::Create()), contents_opaque_is_fixed_(false) {} WebLayerImpl::WebLayerImpl(scoped_refptr layer) : layer_(layer), contents_opaque_is_fixed_(false) { } WebLayerImpl::~WebLayerImpl() { layer_->SetLayerClient(nullptr); } int WebLayerImpl::id() const { return layer_->id(); } void WebLayerImpl::invalidateRect(const blink::WebRect& rect) { layer_->SetNeedsDisplayRect(rect); } void WebLayerImpl::invalidate() { layer_->SetNeedsDisplay(); } void WebLayerImpl::addChild(WebLayer* child) { layer_->AddChild(static_cast(child)->layer()); } void WebLayerImpl::insertChild(WebLayer* child, size_t index) { layer_->InsertChild(static_cast(child)->layer(), index); } void WebLayerImpl::replaceChild(WebLayer* reference, WebLayer* new_layer) { layer_->ReplaceChild(static_cast(reference)->layer(), static_cast(new_layer)->layer()); } void WebLayerImpl::removeFromParent() { layer_->RemoveFromParent(); } void WebLayerImpl::removeAllChildren() { layer_->RemoveAllChildren(); } void WebLayerImpl::setBounds(const WebSize& size) { layer_->SetBounds(size); } WebSize WebLayerImpl::bounds() const { return layer_->bounds(); } void WebLayerImpl::setMasksToBounds(bool masks_to_bounds) { layer_->SetMasksToBounds(masks_to_bounds); } bool WebLayerImpl::masksToBounds() const { return layer_->masks_to_bounds(); } void WebLayerImpl::setMaskLayer(WebLayer* maskLayer) { layer_->SetMaskLayer( maskLayer ? static_cast(maskLayer)->layer() : 0); } void WebLayerImpl::setReplicaLayer(WebLayer* replica_layer) { layer_->SetReplicaLayer( replica_layer ? static_cast(replica_layer)->layer() : 0); } void WebLayerImpl::setOpacity(float opacity) { layer_->SetOpacity(opacity); } float WebLayerImpl::opacity() const { return layer_->opacity(); } void WebLayerImpl::setBlendMode(blink::WebBlendMode blend_mode) { layer_->SetBlendMode(BlendModeToSkia(blend_mode)); } blink::WebBlendMode WebLayerImpl::blendMode() const { return BlendModeFromSkia(layer_->blend_mode()); } void WebLayerImpl::setIsRootForIsolatedGroup(bool isolate) { layer_->SetIsRootForIsolatedGroup(isolate); } bool WebLayerImpl::isRootForIsolatedGroup() { return layer_->is_root_for_isolated_group(); } void WebLayerImpl::setOpaque(bool opaque) { if (contents_opaque_is_fixed_) return; layer_->SetContentsOpaque(opaque); } bool WebLayerImpl::opaque() const { return layer_->contents_opaque(); } void WebLayerImpl::setPosition(const WebFloatPoint& position) { layer_->SetPosition(position); } WebFloatPoint WebLayerImpl::position() const { return layer_->position(); } void WebLayerImpl::setTransform(const SkMatrix44& matrix) { gfx::Transform transform; transform.matrix() = matrix; layer_->SetTransform(transform); } void WebLayerImpl::setTransformOrigin(const blink::WebFloatPoint3D& point) { gfx::Point3F gfx_point = point; layer_->SetTransformOrigin(gfx_point); } blink::WebFloatPoint3D WebLayerImpl::transformOrigin() const { return layer_->transform_origin(); } SkMatrix44 WebLayerImpl::transform() const { return layer_->transform().matrix(); } void WebLayerImpl::setDrawsContent(bool draws_content) { layer_->SetIsDrawable(draws_content); } bool WebLayerImpl::drawsContent() const { return layer_->DrawsContent(); } void WebLayerImpl::setDoubleSided(bool double_sided) { layer_->SetDoubleSided(double_sided); } void WebLayerImpl::setShouldFlattenTransform(bool flatten) { layer_->SetShouldFlattenTransform(flatten); } void WebLayerImpl::setRenderingContext(int context) { layer_->Set3dSortingContextId(context); } void WebLayerImpl::setUseParentBackfaceVisibility( bool use_parent_backface_visibility) { layer_->SetUseParentBackfaceVisibility(use_parent_backface_visibility); } void WebLayerImpl::setBackgroundColor(WebColor color) { layer_->SetBackgroundColor(color); } WebColor WebLayerImpl::backgroundColor() const { return layer_->background_color(); } void WebLayerImpl::setFilters(const cc::FilterOperations& filters) { layer_->SetFilters(filters); } void WebLayerImpl::setBackgroundFilters(const cc::FilterOperations& filters) { layer_->SetBackgroundFilters(filters); } bool WebLayerImpl::hasActiveAnimationForTesting() { return layer_->HasActiveAnimationForTesting(); } void WebLayerImpl::setForceRenderSurface(bool force_render_surface) { layer_->SetForceRenderSurface(force_render_surface); } void WebLayerImpl::setScrollPositionDouble(blink::WebDoublePoint position) { layer_->SetScrollOffset(gfx::ScrollOffset(position.x, position.y)); } blink::WebDoublePoint WebLayerImpl::scrollPositionDouble() const { return blink::WebDoublePoint(layer_->scroll_offset().x(), layer_->scroll_offset().y()); } void WebLayerImpl::setScrollClipLayer(WebLayer* clip_layer) { if (!clip_layer) { layer_->SetScrollClipLayerId(Layer::INVALID_ID); return; } layer_->SetScrollClipLayerId(clip_layer->id()); } bool WebLayerImpl::scrollable() const { return layer_->scrollable(); } void WebLayerImpl::setUserScrollable(bool horizontal, bool vertical) { layer_->SetUserScrollable(horizontal, vertical); } bool WebLayerImpl::userScrollableHorizontal() const { return layer_->user_scrollable_horizontal(); } bool WebLayerImpl::userScrollableVertical() const { return layer_->user_scrollable_vertical(); } void WebLayerImpl::addMainThreadScrollingReasons( uint32_t main_thread_scrolling_reasons) { // WebLayerImpl should only know about non-transient scrolling // reasons. Transient scrolling reasons are computed per hit test. DCHECK(main_thread_scrolling_reasons); DCHECK(cc::MainThreadScrollingReason::MainThreadCanSetScrollReasons( main_thread_scrolling_reasons)); layer_->AddMainThreadScrollingReasons(main_thread_scrolling_reasons); } void WebLayerImpl::clearMainThreadScrollingReasons( uint32_t main_thread_scrolling_reasons_to_clear) { layer_->ClearMainThreadScrollingReasons( main_thread_scrolling_reasons_to_clear); } uint32_t WebLayerImpl::mainThreadScrollingReasons() { return layer_->main_thread_scrolling_reasons(); } bool WebLayerImpl::shouldScrollOnMainThread() const { return layer_->should_scroll_on_main_thread(); } void WebLayerImpl::setNonFastScrollableRegion(const WebVector& rects) { cc::Region region; for (size_t i = 0; i < rects.size(); ++i) region.Union(rects[i]); layer_->SetNonFastScrollableRegion(region); } WebVector WebLayerImpl::nonFastScrollableRegion() const { size_t num_rects = 0; for (cc::Region::Iterator region_rects(layer_->non_fast_scrollable_region()); region_rects.has_rect(); region_rects.next()) ++num_rects; WebVector result(num_rects); size_t i = 0; for (cc::Region::Iterator region_rects(layer_->non_fast_scrollable_region()); region_rects.has_rect(); region_rects.next()) { result[i] = region_rects.rect(); ++i; } return result; } void WebLayerImpl::setFrameTimingRequests( const WebVector>& requests) { std::vector frame_timing_requests(requests.size()); for (size_t i = 0; i < requests.size(); ++i) { frame_timing_requests[i] = cc::FrameTimingRequest( requests[i].first, gfx::Rect(requests[i].second)); } layer_->SetFrameTimingRequests(frame_timing_requests); } WebVector> WebLayerImpl::frameTimingRequests() const { const std::vector& frame_timing_requests = layer_->FrameTimingRequests(); size_t num_requests = frame_timing_requests.size(); WebVector> result(num_requests); for (size_t i = 0; i < num_requests; ++i) { result[i] = std::make_pair(frame_timing_requests[i].id(), frame_timing_requests[i].rect()); } return result; } void WebLayerImpl::setTouchEventHandlerRegion(const WebVector& rects) { cc::Region region; for (size_t i = 0; i < rects.size(); ++i) region.Union(rects[i]); layer_->SetTouchEventHandlerRegion(region); } WebVector WebLayerImpl::touchEventHandlerRegion() const { size_t num_rects = 0; for (cc::Region::Iterator region_rects(layer_->touch_event_handler_region()); region_rects.has_rect(); region_rects.next()) ++num_rects; WebVector result(num_rects); size_t i = 0; for (cc::Region::Iterator region_rects(layer_->touch_event_handler_region()); region_rects.has_rect(); region_rects.next()) { result[i] = region_rects.rect(); ++i; } return result; } void WebLayerImpl::setIsContainerForFixedPositionLayers(bool enable) { layer_->SetIsContainerForFixedPositionLayers(enable); } bool WebLayerImpl::isContainerForFixedPositionLayers() const { return layer_->IsContainerForFixedPositionLayers(); } static blink::WebLayerPositionConstraint ToWebLayerPositionConstraint( const cc::LayerPositionConstraint& constraint) { blink::WebLayerPositionConstraint web_constraint; web_constraint.isFixedPosition = constraint.is_fixed_position(); web_constraint.isFixedToRightEdge = constraint.is_fixed_to_right_edge(); web_constraint.isFixedToBottomEdge = constraint.is_fixed_to_bottom_edge(); return web_constraint; } static cc::LayerPositionConstraint ToLayerPositionConstraint( const blink::WebLayerPositionConstraint& web_constraint) { cc::LayerPositionConstraint constraint; constraint.set_is_fixed_position(web_constraint.isFixedPosition); constraint.set_is_fixed_to_right_edge(web_constraint.isFixedToRightEdge); constraint.set_is_fixed_to_bottom_edge(web_constraint.isFixedToBottomEdge); return constraint; } void WebLayerImpl::setPositionConstraint( const blink::WebLayerPositionConstraint& constraint) { layer_->SetPositionConstraint(ToLayerPositionConstraint(constraint)); } blink::WebLayerPositionConstraint WebLayerImpl::positionConstraint() const { return ToWebLayerPositionConstraint(layer_->position_constraint()); } void WebLayerImpl::setScrollClient(blink::WebLayerScrollClient* scroll_client) { if (scroll_client) { layer_->set_did_scroll_callback( base::Bind(&blink::WebLayerScrollClient::didScroll, base::Unretained(scroll_client))); } else { layer_->set_did_scroll_callback(base::Closure()); } } void WebLayerImpl::setLayerClient(cc::LayerClient* client) { layer_->SetLayerClient(client); } const cc::Layer* WebLayerImpl::ccLayer() const { return layer_.get(); } cc::Layer* WebLayerImpl::ccLayer() { return layer_.get(); } void WebLayerImpl::setElementId(uint64_t id) { layer_->SetElementId(id); } uint64_t WebLayerImpl::elementId() const { return layer_->element_id(); } void WebLayerImpl::setCompositorMutableProperties(uint32_t properties) { layer_->SetMutableProperties(properties); } uint32_t WebLayerImpl::compositorMutableProperties() const { return layer_->mutable_properties(); } void WebLayerImpl::setScrollParent(blink::WebLayer* parent) { cc::Layer* scroll_parent = nullptr; if (parent) scroll_parent = static_cast(parent)->layer(); layer_->SetScrollParent(scroll_parent); } void WebLayerImpl::setClipParent(blink::WebLayer* parent) { cc::Layer* clip_parent = nullptr; if (parent) clip_parent = static_cast(parent)->layer(); layer_->SetClipParent(clip_parent); } Layer* WebLayerImpl::layer() const { return layer_.get(); } void WebLayerImpl::SetContentsOpaqueIsFixed(bool fixed) { contents_opaque_is_fixed_ = fixed; } } // namespace cc_blink