| Index: cc/layers/viewport.cc
|
| diff --git a/cc/layers/viewport.cc b/cc/layers/viewport.cc
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..406bb6d84c777f8f102322dd6136ae30eb5106a0
|
| --- /dev/null
|
| +++ b/cc/layers/viewport.cc
|
| @@ -0,0 +1,173 @@
|
| +// 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/viewport.h"
|
| +
|
| +#include "base/logging.h"
|
| +#include "cc/input/top_controls_manager.h"
|
| +#include "cc/trees/layer_tree_host_impl.h"
|
| +#include "cc/trees/layer_tree_impl.h"
|
| +#include "ui/gfx/geometry/vector2d_f.h"
|
| +
|
| +namespace cc {
|
| +
|
| +// static
|
| +scoped_ptr<Viewport> Viewport::Create(
|
| + LayerTreeHostImpl* host_impl) {
|
| + return make_scoped_ptr(new Viewport(host_impl));
|
| +}
|
| +
|
| +Viewport::Viewport(LayerTreeHostImpl* host_impl)
|
| + : host_impl_(host_impl) {
|
| + DCHECK(host_impl_);
|
| +}
|
| +
|
| +Viewport::~Viewport() {
|
| +}
|
| +
|
| +void Viewport::SetViewportLayers(LayerImpl* inner_viewport_scroll,
|
| + LayerImpl* outer_viewport_scroll) {
|
| + inner_scroll_layer_ = inner_viewport_scroll;
|
| + outer_scroll_layer_ = outer_viewport_scroll;
|
| +}
|
| +
|
| +
|
| +bool Viewport::IsViewportScrollingLayer(LayerImpl* layer) const {
|
| + if (!layer)
|
| + return false;
|
| +
|
| + if (outer_scroll_layer_)
|
| + return layer == outer_scroll_layer_;
|
| +
|
| + return layer == inner_scroll_layer_;
|
| +}
|
| +
|
| +bool Viewport::IsViewportLayer(LayerImpl* layer) const {
|
| + if (!layer)
|
| + return false;
|
| +
|
| + return layer == outer_scroll_layer_ ||
|
| + layer == inner_scroll_layer_;
|
| +}
|
| +
|
| +Viewport::ScrollResult Viewport::ScrollBy(const gfx::Vector2dF& delta,
|
| + const gfx::Point& viewport_point,
|
| + bool is_wheel_scroll) {
|
| + gfx::Vector2dF pending_delta = delta;
|
| + ScrollResult result;
|
| +
|
| + pending_delta -= ScrollTopControls(pending_delta);
|
| + pending_delta -= ScrollViewport(pending_delta,
|
| + viewport_point,
|
| + is_wheel_scroll);
|
| +
|
| + result.applied_delta = delta - pending_delta;
|
| + result.unused_scroll_delta = pending_delta;
|
| +
|
| + const float kEpsilon = 0.1f;
|
| + if (std::abs(result.unused_scroll_delta.x()) < kEpsilon)
|
| + result.unused_scroll_delta.set_x(0.0f);
|
| + if (std::abs(result.unused_scroll_delta.y()) < kEpsilon)
|
| + result.unused_scroll_delta.set_y(0.0f);
|
| +
|
| + // Disable overscroll on axes which are impossible to scroll.
|
| + if (host_impl_->settings().report_overscroll_only_for_scrollable_axes) {
|
| + if (std::abs(MaxTotalScrollOffset().x()) <= kEpsilon ||
|
| + !inner_scroll_layer_->user_scrollable_horizontal())
|
| + result.unused_scroll_delta.set_x(0.0f);
|
| + if (std::abs(MaxTotalScrollOffset().y()) <= kEpsilon ||
|
| + !inner_scroll_layer_->user_scrollable_vertical())
|
| + result.unused_scroll_delta.set_y(0.0f);
|
| + }
|
| +
|
| + return result;
|
| +}
|
| +
|
| +gfx::Vector2dF Viewport::ScrollViewport(const gfx::Vector2dF& delta,
|
| + const gfx::Point& viewport_point,
|
| + bool is_wheel_scroll) {
|
| + gfx::Vector2dF pending_delta = delta;
|
| +
|
| + if (outer_scroll_layer_) {
|
| + pending_delta -= ScrollLayer(outer_scroll_layer_,
|
| + delta,
|
| + viewport_point,
|
| + is_wheel_scroll);
|
| + }
|
| +
|
| + pending_delta -= ScrollLayer(inner_scroll_layer_,
|
| + delta,
|
| + viewport_point,
|
| + is_wheel_scroll);
|
| +
|
| + return delta - pending_delta;
|
| +}
|
| +
|
| +gfx::Vector2dF Viewport::ScrollLayer(LayerImpl* layer_impl,
|
| + const gfx::Vector2dF& delta,
|
| + const gfx::Point& viewport_point,
|
| + bool is_wheel_scroll) {
|
| + gfx::Vector2dF applied_delta;
|
| + if (!is_wheel_scroll) {
|
| + return host_impl_->ScrollLayerWithViewportSpaceDelta(inner_scroll_layer_,
|
| + viewport_point,
|
| + delta);
|
| + } else {
|
| + float page_scale_factor =
|
| + host_impl_->active_tree()->current_page_scale_factor();
|
| + return host_impl_->ScrollLayerWithLocalDelta(inner_scroll_layer_,
|
| + delta,
|
| + page_scale_factor);
|
| + }
|
| +}
|
| +
|
| +gfx::Vector2dF Viewport::ScrollTopControls(const gfx::Vector2dF& delta) {
|
| + if (!ShouldTopControlsConsumeScroll(delta))
|
| + return gfx::Vector2dF();
|
| +
|
| + gfx::Vector2dF excess_delta =
|
| + host_impl_->top_controls_manager()->ScrollBy(delta);
|
| +
|
| + return delta - excess_delta;
|
| +}
|
| +
|
| +bool Viewport::ShouldTopControlsConsumeScroll(
|
| + const gfx::Vector2dF& scroll_delta) const {
|
| + if (!host_impl_->top_controls_manager())
|
| + return false;
|
| +
|
| + // Always consume if it's in the direction to show the top controls.
|
| + if (scroll_delta.y() < 0)
|
| + return true;
|
| +
|
| + if (TotalScrollOffset().y() < MaxTotalScrollOffset().y())
|
| + return true;
|
| +
|
| + return false;
|
| +}
|
| +
|
| +gfx::ScrollOffset Viewport::MaxTotalScrollOffset() const {
|
| + gfx::ScrollOffset offset;
|
| +
|
| + offset += inner_scroll_layer_->MaxScrollOffset();
|
| +
|
| + if (outer_scroll_layer_)
|
| + offset += outer_scroll_layer_->MaxScrollOffset();
|
| +
|
| + return offset;
|
| +}
|
| +
|
| +gfx::ScrollOffset Viewport::TotalScrollOffset() const {
|
| + gfx::ScrollOffset offset;
|
| +
|
| + offset += inner_scroll_layer_->CurrentScrollOffset();
|
| +
|
| + if (outer_scroll_layer_)
|
| + offset += outer_scroll_layer_->CurrentScrollOffset();
|
| +
|
| + return offset;
|
| +}
|
| +
|
| +
|
| +} // namespace cc
|
|
|