// Copyright 2013 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/layers/painted_scrollbar_layer.h" #include #include "base/auto_reset.h" #include "base/basictypes.h" #include "base/trace_event/trace_event.h" #include "cc/base/math_util.h" #include "cc/layers/painted_scrollbar_layer_impl.h" #include "cc/resources/ui_resource_bitmap.h" #include "cc/trees/layer_tree_host.h" #include "cc/trees/layer_tree_impl.h" #include "skia/ext/platform_canvas.h" #include "skia/ext/refptr.h" #include "third_party/skia/include/core/SkBitmap.h" #include "third_party/skia/include/core/SkCanvas.h" #include "third_party/skia/include/core/SkSize.h" #include "ui/gfx/geometry/size_conversions.h" #include "ui/gfx/skia_util.h" namespace cc { scoped_ptr PaintedScrollbarLayer::CreateLayerImpl( LayerTreeImpl* tree_impl) { return PaintedScrollbarLayerImpl::Create( tree_impl, id(), scrollbar_->Orientation()); } scoped_refptr PaintedScrollbarLayer::Create( scoped_ptr scrollbar, int scroll_layer_id) { return make_scoped_refptr( new PaintedScrollbarLayer(scrollbar.Pass(), scroll_layer_id)); } PaintedScrollbarLayer::PaintedScrollbarLayer(scoped_ptr scrollbar, int scroll_layer_id) : scrollbar_(scrollbar.Pass()), scroll_layer_id_(scroll_layer_id), clip_layer_id_(Layer::INVALID_ID), internal_contents_scale_(0.f), thumb_thickness_(scrollbar_->ThumbThickness()), thumb_length_(scrollbar_->ThumbLength()), is_overlay_(scrollbar_->IsOverlay()), has_thumb_(scrollbar_->HasThumb()) { if (!scrollbar_->IsOverlay()) SetShouldScrollOnMainThread(true); } PaintedScrollbarLayer::~PaintedScrollbarLayer() {} int PaintedScrollbarLayer::ScrollLayerId() const { return scroll_layer_id_; } void PaintedScrollbarLayer::SetScrollLayer(int layer_id) { if (layer_id == scroll_layer_id_) return; scroll_layer_id_ = layer_id; SetNeedsFullTreeSync(); } void PaintedScrollbarLayer::SetClipLayer(int layer_id) { if (layer_id == clip_layer_id_) return; clip_layer_id_ = layer_id; SetNeedsFullTreeSync(); } bool PaintedScrollbarLayer::OpacityCanAnimateOnImplThread() const { return scrollbar_->IsOverlay(); } ScrollbarOrientation PaintedScrollbarLayer::orientation() const { return scrollbar_->Orientation(); } int PaintedScrollbarLayer::MaxTextureSize() { DCHECK(layer_tree_host()); return layer_tree_host()->GetRendererCapabilities().max_texture_size; } float PaintedScrollbarLayer::ClampScaleToMaxTextureSize(float scale) { // If the scaled bounds() is bigger than the max texture size of the // device, we need to clamp it by rescaling, since this is used // below to set the texture size. gfx::Size scaled_bounds = gfx::ToCeiledSize(gfx::ScaleSize(bounds(), scale)); if (scaled_bounds.width() > MaxTextureSize() || scaled_bounds.height() > MaxTextureSize()) { if (scaled_bounds.width() > scaled_bounds.height()) return (MaxTextureSize() - 1) / static_cast(bounds().width()); else return (MaxTextureSize() - 1) / static_cast(bounds().height()); } return scale; } void PaintedScrollbarLayer::PushPropertiesTo(LayerImpl* layer) { Layer::PushPropertiesTo(layer); PushScrollClipPropertiesTo(layer); PaintedScrollbarLayerImpl* scrollbar_layer = static_cast(layer); scrollbar_layer->set_internal_contents_scale_and_bounds( internal_contents_scale_, internal_content_bounds_); scrollbar_layer->SetThumbThickness(thumb_thickness_); scrollbar_layer->SetThumbLength(thumb_length_); if (orientation() == HORIZONTAL) { scrollbar_layer->SetTrackStart( track_rect_.x() - location_.x()); scrollbar_layer->SetTrackLength(track_rect_.width()); } else { scrollbar_layer->SetTrackStart( track_rect_.y() - location_.y()); scrollbar_layer->SetTrackLength(track_rect_.height()); } if (track_resource_.get()) scrollbar_layer->set_track_ui_resource_id(track_resource_->id()); else scrollbar_layer->set_track_ui_resource_id(0); if (thumb_resource_.get()) scrollbar_layer->set_thumb_ui_resource_id(thumb_resource_->id()); else scrollbar_layer->set_thumb_ui_resource_id(0); scrollbar_layer->set_is_overlay_scrollbar(is_overlay_); } ScrollbarLayerInterface* PaintedScrollbarLayer::ToScrollbarLayer() { return this; } void PaintedScrollbarLayer::PushScrollClipPropertiesTo(LayerImpl* layer) { PaintedScrollbarLayerImpl* scrollbar_layer = static_cast(layer); scrollbar_layer->SetScrollLayerAndClipLayerByIds(scroll_layer_id_, clip_layer_id_); } void PaintedScrollbarLayer::SetLayerTreeHost(LayerTreeHost* host) { // When the LTH is set to null or has changed, then this layer should remove // all of its associated resources. if (!host || host != layer_tree_host()) { track_resource_ = nullptr; thumb_resource_ = nullptr; } Layer::SetLayerTreeHost(host); } gfx::Rect PaintedScrollbarLayer::ScrollbarLayerRectToContentRect( const gfx::Rect& layer_rect) const { // Don't intersect with the bounds as in LayerRectToContentRect() because // layer_rect here might be in coordinates of the containing layer. gfx::Rect expanded_rect = gfx::ScaleToEnclosingRect( layer_rect, internal_contents_scale_, internal_contents_scale_); // We should never return a rect bigger than the content bounds. gfx::Size clamped_size = expanded_rect.size(); clamped_size.SetToMin(internal_content_bounds_); expanded_rect.set_size(clamped_size); return expanded_rect; } gfx::Rect PaintedScrollbarLayer::OriginThumbRect() const { gfx::Size thumb_size; if (orientation() == HORIZONTAL) { thumb_size = gfx::Size(scrollbar_->ThumbLength(), scrollbar_->ThumbThickness()); } else { thumb_size = gfx::Size(scrollbar_->ThumbThickness(), scrollbar_->ThumbLength()); } return gfx::Rect(thumb_size); } void PaintedScrollbarLayer::UpdateThumbAndTrackGeometry() { UpdateProperty(scrollbar_->TrackRect(), &track_rect_); UpdateProperty(scrollbar_->Location(), &location_); UpdateProperty(scrollbar_->IsOverlay(), &is_overlay_); UpdateProperty(scrollbar_->HasThumb(), &has_thumb_); if (has_thumb_) { UpdateProperty(scrollbar_->ThumbThickness(), &thumb_thickness_); UpdateProperty(scrollbar_->ThumbLength(), &thumb_length_); } else { UpdateProperty(0, &thumb_thickness_); UpdateProperty(0, &thumb_length_); } } void PaintedScrollbarLayer::UpdateInternalContentScale() { float scale = layer_tree_host()->device_scale_factor(); if (layer_tree_host() ->settings() .layer_transforms_should_scale_layer_contents) { gfx::Vector2dF transform_scales = MathUtil::ComputeTransform2dScaleComponents(draw_transform(), scale); scale = std::max(transform_scales.x(), transform_scales.y()); } bool changed = false; changed |= UpdateProperty(ClampScaleToMaxTextureSize(scale), &internal_contents_scale_); changed |= UpdateProperty( gfx::ToCeiledSize(gfx::ScaleSize(bounds(), internal_contents_scale_)), &internal_content_bounds_); if (changed) { // If the content scale or bounds change, repaint. SetNeedsDisplay(); } } bool PaintedScrollbarLayer::Update(ResourceUpdateQueue* queue, const OcclusionTracker* occlusion) { { base::AutoReset ignore_set_needs_commit(&ignore_set_needs_commit_, true); Layer::Update(queue, occlusion); UpdateInternalContentScale(); } UpdateThumbAndTrackGeometry(); gfx::Rect track_layer_rect = gfx::Rect(location_, bounds()); gfx::Rect scaled_track_rect = ScrollbarLayerRectToContentRect( track_layer_rect); bool updated = false; if (track_rect_.IsEmpty() || scaled_track_rect.IsEmpty()) { if (track_resource_) { track_resource_ = nullptr; thumb_resource_ = nullptr; SetNeedsPushProperties(); updated = true; } return updated; } if (!has_thumb_ && thumb_resource_) { thumb_resource_ = nullptr; SetNeedsPushProperties(); updated = true; } if (update_rect_.IsEmpty() && track_resource_) return updated; track_resource_ = ScopedUIResource::Create( layer_tree_host(), RasterizeScrollbarPart(track_layer_rect, scaled_track_rect, TRACK)); gfx::Rect thumb_layer_rect = OriginThumbRect(); gfx::Rect scaled_thumb_rect = ScrollbarLayerRectToContentRect(thumb_layer_rect); if (has_thumb_ && !scaled_thumb_rect.IsEmpty()) { thumb_resource_ = ScopedUIResource::Create( layer_tree_host(), RasterizeScrollbarPart(thumb_layer_rect, scaled_thumb_rect, THUMB)); } // UI resources changed so push properties is needed. SetNeedsPushProperties(); updated = true; return updated; } UIResourceBitmap PaintedScrollbarLayer::RasterizeScrollbarPart( const gfx::Rect& layer_rect, const gfx::Rect& content_rect, ScrollbarPart part) { DCHECK(!content_rect.size().IsEmpty()); DCHECK(!layer_rect.size().IsEmpty()); SkBitmap skbitmap; skbitmap.allocN32Pixels(content_rect.width(), content_rect.height()); SkCanvas skcanvas(skbitmap); float scale_x = content_rect.width() / static_cast(layer_rect.width()); float scale_y = content_rect.height() / static_cast(layer_rect.height()); skcanvas.scale(SkFloatToScalar(scale_x), SkFloatToScalar(scale_y)); skcanvas.translate(SkFloatToScalar(-layer_rect.x()), SkFloatToScalar(-layer_rect.y())); SkRect layer_skrect = RectToSkRect(layer_rect); SkPaint paint; paint.setAntiAlias(false); paint.setXfermodeMode(SkXfermode::kClear_Mode); skcanvas.drawRect(layer_skrect, paint); skcanvas.clipRect(layer_skrect); scrollbar_->PaintPart(&skcanvas, part, layer_rect); // Make sure that the pixels are no longer mutable to unavoid unnecessary // allocation and copying. skbitmap.setImmutable(); return UIResourceBitmap(skbitmap); } } // namespace cc