#include "cc/input/input_handler.h"
#include <string>
#include <utility>
#include <vector>
#include "base/feature_list.h"
#include "base/notreached.h"
#include "build/build_config.h"
#include "cc/base/features.h"
#include "cc/input/scroll_elasticity_helper.h"
#include "cc/input/scroll_utils.h"
#include "cc/input/scrollbar_controller.h"
#include "cc/input/snap_selection_strategy.h"
#include "cc/layers/viewport.h"
#include "cc/trees/compositor_commit_data.h"
#include "cc/trees/latency_info_swap_promise_monitor.h"
#include "cc/trees/layer_tree_host_impl.h"
#include "cc/trees/layer_tree_impl.h"
#include "cc/trees/layer_tree_settings.h"
#include "cc/trees/property_tree.h"
#include "cc/trees/scroll_node.h"
#include "ui/gfx/geometry/point.h"
#include "ui/gfx/geometry/point_conversions.h"
#include "ui/gfx/geometry/point_f.h"
#include "ui/gfx/geometry/vector2d_f.h"
namespace cc {
namespace {
enum SlowScrollMetricThread { MAIN_THREAD, CC_THREAD };
void RecordCompositorSlowScrollMetric(ui::ScrollInputType type,
SlowScrollMetricThread scroll_thread) {
bool scroll_on_main_thread = (scroll_thread == MAIN_THREAD);
if (type == ui::ScrollInputType::kWheel) {
UMA_HISTOGRAM_BOOLEAN("Renderer4.CompositorWheelScrollUpdateThread",
scroll_on_main_thread);
} else if (type == ui::ScrollInputType::kTouchscreen) {
UMA_HISTOGRAM_BOOLEAN("Renderer4.CompositorTouchScrollUpdateThread",
scroll_on_main_thread);
}
}
InputHandlerClient::ScrollEventDispatchMode GetScrollEventDispatchMode() {
const std::string mode_name = ::features::kScrollEventDispatchMode.Get();
if (mode_name ==
::features::kScrollEventDispatchModeDispatchScrollEventsImmediately) {
return InputHandlerClient::ScrollEventDispatchMode::
kDispatchScrollEventsImmediately;
} else if (mode_name ==
::features::
kScrollEventDispatchModeUseScrollPredictorForEmptyQueue) {
return InputHandlerClient::ScrollEventDispatchMode::
kUseScrollPredictorForEmptyQueue;
} else if (mode_name ==
::features::
kScrollEventDispatchModeUseScrollPredictorForDeadline) {
return InputHandlerClient::ScrollEventDispatchMode::
kUseScrollPredictorForDeadline;
} else if (mode_name ==
::features::
kScrollEventDispatchModeEnqueueScrollEvents) {
return InputHandlerClient::ScrollEventDispatchMode::
kEnqueueScrollEvents;
}
return InputHandlerClient::ScrollEventDispatchMode::kUseScrollPredictorForDeadline;
}
}
InputHandlerCommitData::InputHandlerCommitData() = default;
InputHandlerCommitData::~InputHandlerCommitData() = default;
base::WeakPtr<InputHandler> InputHandler::Create(
CompositorDelegateForInput& compositor_delegate) {
auto input_handler = std::make_unique<InputHandler>(compositor_delegate);
base::WeakPtr<InputHandler> input_handler_weak = input_handler->AsWeakPtr();
compositor_delegate.BindToInputHandler(std::move(input_handler));
return input_handler_weak;
}
InputHandler::InputHandler(CompositorDelegateForInput& compositor_delegate)
: compositor_delegate_(compositor_delegate),
scrollbar_controller_(std::make_unique<ScrollbarController>(
&compositor_delegate_->GetImplDeprecated())) {}
InputHandler::~InputHandler() = default;
base::WeakPtr<InputHandler> InputHandler::AsWeakPtr() {
return weak_factory_.GetWeakPtr();
}
void InputHandler::BindToClient(InputHandlerClient* client) {
DCHECK(input_handler_client_ == nullptr);
input_handler_client_ = client;
input_handler_client_->SetPrefersReducedMotion(prefers_reduced_motion_);
input_handler_client_->SetScrollEventDispatchMode(
GetScrollEventDispatchMode());
}
InputHandler::ScrollStatus InputHandler::ScrollBegin(ScrollState* scroll_state,
ui::ScrollInputType type) {
DCHECK(scroll_state);
DCHECK(scroll_state->delta_x() == 0 && scroll_state->delta_y() == 0);
InputHandler::ScrollStatus scroll_status;
scroll_status.main_thread_scrolling_reasons =
MainThreadScrollingReason::kNotScrollingOnMain;
TRACE_EVENT0("cc", "InputHandler::ScrollBegin");
if (scroll_state->delta_granularity() ==
ui::ScrollGranularity::kScrollByPrecisePixel) {
compositor_delegate_->GetImplDeprecated()
.mutator_host()
->ScrollAnimationAbort();
scroll_animating_snap_target_ids_ = TargetSnapAreaElementIds();
}
if (CurrentlyScrollingNode() && type == latched_scroll_type_) {
DCHECK(deferred_scroll_end_);
deferred_scroll_end_ = false;
return scroll_status;
}
ScrollNode* scrolling_node = nullptr;
bool scroll_on_main_thread = false;
ClearCurrentlyScrollingNode();
ElementId target_element_id = scroll_state->target_element_id();
ScrollTree& scroll_tree = GetScrollTree();
bool unification_enabled =
base::FeatureList::IsEnabled(features::kScrollUnification);
if (target_element_id && (!scroll_state->main_thread_hit_tested_reasons() ||
scroll_state->is_scrollbar_interaction())) {
TRACE_EVENT_INSTANT0("cc", "Latched scroll node provided",
TRACE_EVENT_SCOPE_THREAD);
scrolling_node = scroll_tree.FindNodeFromElementId(target_element_id);
if (!unification_enabled) {
if (scrolling_node) {
scroll_status = TryScroll(scroll_tree, scrolling_node);
if (IsMainThreadScrolling(scroll_status, scrolling_node))
scroll_on_main_thread = true;
}
}
} else {
ScrollNode* starting_node = nullptr;
if (target_element_id) {
TRACE_EVENT_INSTANT0("cc", "Unlatched scroll node provided",
TRACE_EVENT_SCOPE_THREAD);
DCHECK(scroll_state->main_thread_hit_tested_reasons());
DCHECK(unification_enabled);
starting_node = scroll_tree.FindNodeFromElementId(target_element_id);
if (!starting_node) {
scroll_status.thread = InputHandler::ScrollThread::SCROLL_IGNORED;
return scroll_status;
}
} else {
TRACE_EVENT_INSTANT0("cc", "Hit Testing for ScrollNode",
TRACE_EVENT_SCOPE_THREAD);
gfx::Point viewport_point(scroll_state->position_x(),
scroll_state->position_y());
gfx::PointF device_viewport_point =
gfx::ScalePoint(gfx::PointF(viewport_point),
compositor_delegate_->DeviceScaleFactor());
if (unification_enabled) {
if (scroll_state->main_thread_hit_tested_reasons()) {
NOTREACHED();
scroll_status.thread = InputHandler::ScrollThread::SCROLL_IGNORED;
return scroll_status;
}
ScrollHitTestResult scroll_hit_test =
HitTestScrollNode(device_viewport_point);
if (!scroll_hit_test.hit_test_successful) {
TRACE_EVENT_INSTANT0("cc", "Request Main Thread Hit Test",
TRACE_EVENT_SCOPE_THREAD);
scroll_status.thread =
InputHandler::ScrollThread::SCROLL_ON_IMPL_THREAD;
DCHECK(scroll_hit_test.main_thread_hit_test_reasons);
scroll_status.main_thread_hit_test_reasons =
scroll_hit_test.main_thread_hit_test_reasons;
return scroll_status;
}
starting_node = scroll_hit_test.scroll_node;
} else {
LayerImpl* layer_impl =
ActiveTree().FindLayerThatIsHitByPoint(device_viewport_point);
if (layer_impl) {
LayerImpl* first_scrolling_layer_or_scrollbar =
ActiveTree().FindFirstScrollingLayerOrScrollbarThatIsHitByPoint(
device_viewport_point);
if (IsTouchDraggingScrollbar(first_scrolling_layer_or_scrollbar,
type)) {
TRACE_EVENT_INSTANT0("cc", "Scrollbar Scrolling",
TRACE_EVENT_SCOPE_THREAD);
scroll_status.thread =
InputHandler::ScrollThread::SCROLL_ON_MAIN_THREAD;
scroll_status.main_thread_scrolling_reasons =
MainThreadScrollingReason::kScrollbarScrolling;
return scroll_status;
} else if (!IsInitialScrollHitTestReliable(
layer_impl, first_scrolling_layer_or_scrollbar)) {
TRACE_EVENT_INSTANT0("cc", "Failed Hit Test",
TRACE_EVENT_SCOPE_THREAD);
scroll_status.thread =
InputHandler::ScrollThread::SCROLL_ON_MAIN_THREAD;
scroll_status.main_thread_scrolling_reasons =
MainThreadScrollingReason::kFailedHitTest;
return scroll_status;
}
}
starting_node = FindScrollNodeForCompositedScrolling(
device_viewport_point, layer_impl, &scroll_on_main_thread,
&scroll_status.main_thread_scrolling_reasons);
}
}
scrolling_node = FindNodeToLatch(scroll_state, starting_node, type);
}
if (scroll_on_main_thread) {
DCHECK(!unification_enabled);
RecordCompositorSlowScrollMetric(type, MAIN_THREAD);
scroll_status.thread = InputHandler::ScrollThread::SCROLL_ON_MAIN_THREAD;
return scroll_status;
} else if (!scrolling_node) {
if (compositor_delegate_->GetSettings().is_for_embedded_frame) {
TRACE_EVENT_INSTANT0("cc",
"Ignored - No ScrollNode (OOPIF or FencedFrame)",
TRACE_EVENT_SCOPE_THREAD);
} else {
TRACE_EVENT_INSTANT0("cc", "Ignored - No ScrollNode",
TRACE_EVENT_SCOPE_THREAD);
}
scroll_status.thread = InputHandler::ScrollThread::SCROLL_IGNORED;
return scroll_status;
}
DCHECK_EQ(scroll_status.main_thread_scrolling_reasons,
MainThreadScrollingReason::kNotScrollingOnMain);
DCHECK_EQ(scroll_status.thread,
InputHandler::ScrollThread::SCROLL_ON_IMPL_THREAD);
DCHECK(scrolling_node);
ActiveTree().SetCurrentlyScrollingNode(scrolling_node);
if (unification_enabled)
scroll_status.main_thread_repaint_reasons =
scroll_tree.GetMainThreadRepaintReasons(*scrolling_node);
DidLatchToScroller(*scroll_state, type);
if (GetViewport().ShouldScroll(*CurrentlyScrollingNode())) {
outer_viewport_consumed_delta_ = false;
if (!GetViewport().CanScroll(*CurrentlyScrollingNode(), *scroll_state)) {
scroll_status.viewport_cannot_scroll = true;
}
}
return scroll_status;
}
InputHandler::ScrollStatus InputHandler::RootScrollBegin(
ScrollState* scroll_state,
ui::ScrollInputType type) {
TRACE_EVENT0("cc", "InputHandler::RootScrollBegin");
if (!OuterViewportScrollNode()) {
InputHandler::ScrollStatus scroll_status;
scroll_status.thread = InputHandler::ScrollThread::SCROLL_IGNORED;
return scroll_status;
}
scroll_state->data()->set_current_native_scrolling_element(
OuterViewportScrollNode()->element_id);
InputHandler::ScrollStatus scroll_status = ScrollBegin(scroll_state, type);
DCHECK(!scroll_status.main_thread_hit_test_reasons);
return scroll_status;
}
InputHandlerScrollResult InputHandler::ScrollUpdate(
ScrollState* scroll_state,
base::TimeDelta delayed_by) {
DCHECK(scroll_state);
DCHECK(!scroll_state->data()->current_native_scrolling_element());
OHOS_TRACE_EVENT2("cc", "InputHandler::ScrollUpdate", "dx",
scroll_state->delta_x(), "dy", scroll_state->delta_y());
#if BUILDFLAG(IS_OHOS)
SetHandledTouchEvent(false);
#endif
if (!CurrentlyScrollingNode())
return InputHandlerScrollResult();
const ScrollNode& scroll_node = *CurrentlyScrollingNode();
last_scroll_update_state_ = *scroll_state;
if (scroll_state->is_scrollbar_interaction() &&
scroll_state->delta_granularity() !=
ui::ScrollGranularity::kScrollByPrecisePixel) {
AdjustScrollDeltaForScrollbarSnap(scroll_state);
}
gfx::Vector2dF resolvedScrollDelta = ResolveScrollGranularityToPixels(
scroll_node,
gfx::Vector2dF(scroll_state->delta_x(), scroll_state->delta_y()),
scroll_state->delta_granularity());
scroll_state->data()->delta_x = resolvedScrollDelta.x();
scroll_state->data()->delta_y = resolvedScrollDelta.y();
if (scroll_state->delta_granularity() !=
ui::ScrollGranularity::kScrollByPrecisePixel) {
scroll_state->data()->delta_granularity =
ui::ScrollGranularity::kScrollByPixel;
}
compositor_delegate_->AccumulateScrollDeltaForTracing(
gfx::Vector2dF(scroll_state->delta_x(), scroll_state->delta_y()));
compositor_delegate_->WillScrollContent(scroll_node.element_id);
float initial_top_controls_offset = compositor_delegate_->GetImplDeprecated()
.browser_controls_manager()
->ControlsTopOffset();
ScrollLatchedScroller(scroll_state, delayed_by);
bool did_scroll_x = scroll_state->caused_scroll_x();
bool did_scroll_y = scroll_state->caused_scroll_y();
did_scroll_x_for_scroll_gesture_ |= did_scroll_x;
did_scroll_y_for_scroll_gesture_ |= did_scroll_y;
delta_consumed_for_scroll_gesture_ |=
scroll_state->delta_consumed_for_scroll_sequence();
bool did_scroll_content = did_scroll_x || did_scroll_y;
if (did_scroll_content) {
bool is_animated_scroll = ShouldAnimateScroll(*scroll_state);
compositor_delegate_->DidScrollContent(scroll_node.element_id,
is_animated_scroll);
} else {
overscroll_delta_for_main_thread_ +=
gfx::Vector2dF(scroll_state->delta_x(), scroll_state->delta_y());
}
SetNeedsCommit();
if (did_scroll_x)
accumulated_root_overscroll_.set_x(0);
if (did_scroll_y)
accumulated_root_overscroll_.set_y(0);
gfx::Vector2dF unused_root_delta;
if (GetViewport().ShouldScroll(scroll_node)) {
unused_root_delta =
gfx::Vector2dF(scroll_state->delta_x(), scroll_state->delta_y());
}
if (auto* inner_viewport_scroll_node = InnerViewportScrollNode()) {
unused_root_delta =
UserScrollableDelta(*inner_viewport_scroll_node, unused_root_delta);
}
accumulated_root_overscroll_ += unused_root_delta;
bool did_scroll_top_controls =
initial_top_controls_offset != compositor_delegate_->GetImplDeprecated()
.browser_controls_manager()
->ControlsTopOffset();
InputHandlerScrollResult scroll_result;
scroll_result.did_scroll = did_scroll_content || did_scroll_top_controls;
scroll_result.did_overscroll_root = !unused_root_delta.IsZero();
scroll_result.accumulated_root_overscroll = accumulated_root_overscroll_;
scroll_result.unused_scroll_delta = unused_root_delta;
scroll_result.overscroll_behavior =
scroll_state->is_scroll_chain_cut()
? OverscrollBehavior(OverscrollBehavior::Type::kNone)
: ActiveTree().overscroll_behavior();
if (scroll_result.did_scroll) {
UpdateRootLayerStateForSynchronousInputHandler();
}
scroll_result.current_visual_offset = GetVisualScrollOffset(scroll_node);
float scale_factor = ActiveTree().page_scale_factor_for_scroll();
scroll_result.current_visual_offset.Scale(scale_factor);
if (base::FeatureList::IsEnabled(features::kScrollUnification) &&
!GetScrollTree().CanRealizeScrollsOnCompositor(scroll_node)) {
scroll_result.needs_main_thread_repaint = true;
}
compositor_delegate_->GetImplDeprecated()
.mutator_host()
->TickScrollAnimations(compositor_delegate_->GetImplDeprecated()
.CurrentBeginFrameArgs()
.frame_time,
GetScrollTree());
return scroll_result;
}
void InputHandler::AdjustScrollDeltaForScrollbarSnap(
ScrollState* scroll_state) {
ScrollNode* scroll_node = CurrentlyScrollingNode();
if (!scroll_node || !scroll_node->snap_container_data)
return;
gfx::PointF current_position = GetVisualScrollOffset(*scroll_node);
const SnapContainerData& data = scroll_node->snap_container_data.value();
std::unique_ptr<SnapSelectionStrategy> strategy =
SnapSelectionStrategy::CreateForDirection(
current_position,
gfx::Vector2dF(scroll_state->delta_x(), scroll_state->delta_y()),
true);
gfx::PointF snap_position;
TargetSnapAreaElementIds snap_target_ids;
if (!data.FindSnapPosition(*strategy, &snap_position, &snap_target_ids))
return;
scroll_state->data()->delta_x = snap_position.x() - current_position.x();
scroll_state->data()->delta_y = snap_position.y() - current_position.y();
}
void InputHandler::ScrollEnd(bool should_snap) {
scrollbar_controller_->ResetState();
if (!CurrentlyScrollingNode())
return;
if (compositor_delegate_->GetImplDeprecated()
.mutator_host()
->ImplOnlyScrollAnimatingElement()) {
DCHECK(!deferred_scroll_end_);
deferred_scroll_end_ = true;
return;
}
if (should_snap && SnapAtScrollEnd(SnapReason::kGestureScrollEnd)) {
deferred_scroll_end_ = true;
return;
}
DCHECK(latched_scroll_type_.has_value());
compositor_delegate_->GetImplDeprecated()
.browser_controls_manager()
->ScrollEnd();
if (did_scroll_x_for_scroll_gesture_ || did_scroll_y_for_scroll_gesture_) {
scroll_gesture_did_end_ = true;
}
ClearCurrentlyScrollingNode();
deferred_scroll_end_ = false;
SetNeedsCommit();
}
void InputHandler::RecordScrollBegin(
ui::ScrollInputType input_type,
ScrollBeginThreadState scroll_start_state) {
auto tracker_type = GetTrackerTypeForScroll(input_type);
DCHECK_NE(tracker_type, FrameSequenceTrackerType::kMaxType);
FrameInfo::SmoothEffectDrivingThread scrolling_thread;
switch (scroll_start_state) {
case ScrollBeginThreadState::kScrollingOnCompositor:
scrolling_thread = FrameInfo::SmoothEffectDrivingThread::kCompositor;
break;
case ScrollBeginThreadState::kScrollingOnMain:
case ScrollBeginThreadState::kScrollingOnCompositorBlockedOnMain:
scrolling_thread = FrameInfo::SmoothEffectDrivingThread::kMain;
break;
}
compositor_delegate_->GetImplDeprecated()
.frame_trackers()
.StartScrollSequence(tracker_type, scrolling_thread);
}
void InputHandler::RecordScrollEnd(ui::ScrollInputType input_type) {
compositor_delegate_->GetImplDeprecated().frame_trackers().StopSequence(
GetTrackerTypeForScroll(input_type));
}
InputHandlerPointerResult InputHandler::MouseMoveAt(
const gfx::Point& viewport_point) {
InputHandlerPointerResult result =
scrollbar_controller_->HandlePointerMove(gfx::PointF(viewport_point));
if (!compositor_delegate_->HasAnimatedScrollbars())
return result;
gfx::PointF device_viewport_point = gfx::ScalePoint(
gfx::PointF(viewport_point), compositor_delegate_->DeviceScaleFactor());
ScrollHitTestResult hit_test = HitTestScrollNode(device_viewport_point);
ScrollNode* scroll_node = hit_test.scroll_node;
if (!hit_test.hit_test_successful || !scroll_node)
return result;
if (scroll_node->scrolls_inner_viewport)
scroll_node = OuterViewportScrollNode();
ElementId scroll_element_id = scroll_node->element_id;
ScrollbarAnimationController* new_animation_controller =
compositor_delegate_->GetImplDeprecated()
.ScrollbarAnimationControllerForElementId(scroll_element_id);
if (scroll_element_id != scroll_element_id_mouse_currently_over_) {
ScrollbarAnimationController* old_animation_controller =
compositor_delegate_->GetImplDeprecated()
.ScrollbarAnimationControllerForElementId(
scroll_element_id_mouse_currently_over_);
if (old_animation_controller)
old_animation_controller->DidMouseLeave();
scroll_element_id_mouse_currently_over_ = scroll_element_id;
}
if (!new_animation_controller)
return result;
new_animation_controller->DidMouseMove(device_viewport_point);
return result;
}
PointerResultType InputHandler::HitTest(const gfx::PointF& viewport_point) {
return scrollbar_controller_->HitTest(viewport_point);
}
InputHandlerPointerResult InputHandler::MouseDown(
const gfx::PointF& viewport_point,
bool shift_modifier) {
ScrollbarAnimationController* animation_controller =
compositor_delegate_->GetImplDeprecated()
.ScrollbarAnimationControllerForElementId(
scroll_element_id_mouse_currently_over_);
if (animation_controller) {
animation_controller->DidMouseDown();
scroll_element_id_mouse_currently_captured_ =
scroll_element_id_mouse_currently_over_;
}
return scrollbar_controller_->HandlePointerDown(viewport_point,
shift_modifier);
}
InputHandlerPointerResult InputHandler::MouseUp(
const gfx::PointF& viewport_point) {
if (scroll_element_id_mouse_currently_captured_) {
ScrollbarAnimationController* animation_controller =
compositor_delegate_->GetImplDeprecated()
.ScrollbarAnimationControllerForElementId(
scroll_element_id_mouse_currently_captured_);
scroll_element_id_mouse_currently_captured_ = ElementId();
if (animation_controller)
animation_controller->DidMouseUp();
}
return scrollbar_controller_->HandlePointerUp(viewport_point);
}
void InputHandler::MouseLeave() {
compositor_delegate_->DidMouseLeave();
scroll_element_id_mouse_currently_over_ = ElementId();
}
ElementId InputHandler::FindFrameElementIdAtPoint(
const gfx::PointF& viewport_point) {
gfx::PointF device_viewport_point = gfx::ScalePoint(
gfx::PointF(viewport_point), compositor_delegate_->DeviceScaleFactor());
return ActiveTree().FindFrameElementIdAtPoint(device_viewport_point);
}
void InputHandler::RequestUpdateForSynchronousInputHandler() {
UpdateRootLayerStateForSynchronousInputHandler();
}
void InputHandler::SetSynchronousInputHandlerRootScrollOffset(
const gfx::PointF& root_content_offset) {
TRACE_EVENT2("cc", "InputHandler::SetSynchronousInputHandlerRootScrollOffset",
"offset_x", root_content_offset.x(), "offset_y",
root_content_offset.y());
#if BUILDFLAG(IS_OHOS)
gfx::Vector2dF physical_delta =
gfx::Vector2dF(root_content_offset.x(), root_content_offset.y());
#else
gfx::Vector2dF physical_delta =
root_content_offset - GetViewport().TotalScrollOffset();
#endif
physical_delta.Scale(ActiveTree().page_scale_factor_for_scroll());
bool changed = !GetViewport()
.ScrollBy(physical_delta,
gfx::Point(),
false,
false,
true)
.consumed_delta.IsZero();
if (!changed)
return;
compositor_delegate_->DidScrollContent(OuterViewportScrollNode()->element_id,
false);
SetNeedsCommit();
UpdateRootLayerStateForSynchronousInputHandler();
compositor_delegate_->SetNeedsFullViewportRedraw();
}
void InputHandler::PinchGestureBegin(const gfx::Point& anchor,
ui::ScrollInputType source) {
DCHECK(source == ui::ScrollInputType::kTouchscreen ||
source == ui::ScrollInputType::kWheel);
pinch_gesture_active_ = true;
pinch_gesture_end_should_clear_scrolling_node_ = !CurrentlyScrollingNode();
TRACE_EVENT_INSTANT1("cc", "SetCurrentlyScrollingNode PinchGestureBegin",
TRACE_EVENT_SCOPE_THREAD, "isNull",
OuterViewportScrollNode() ? false : true);
if (OuterViewportScrollNode()) {
ActiveTree().SetCurrentlyScrollingNode(OuterViewportScrollNode());
ScrollStateData scroll_state_data;
scroll_state_data.position_x = anchor.x();
scroll_state_data.position_y = anchor.y();
scroll_state_data.is_beginning = true;
scroll_state_data.delta_granularity =
ui::ScrollGranularity::kScrollByPrecisePixel;
scroll_state_data.is_direct_manipulation =
source == ui::ScrollInputType::kTouchscreen;
ScrollState state(scroll_state_data);
DidLatchToScroller(state, source);
}
compositor_delegate_->GetImplDeprecated()
.browser_controls_manager()
->PinchBegin();
compositor_delegate_->DidStartPinchZoom();
}
void InputHandler::PinchGestureUpdate(float magnify_delta,
const gfx::Point& anchor) {
TRACE_EVENT0("cc", "InputHandler::PinchGestureUpdate");
if (!InnerViewportScrollNode())
return;
has_pinch_zoomed_ = true;
GetViewport().PinchUpdate(magnify_delta, anchor);
SetNeedsCommit();
compositor_delegate_->DidUpdatePinchZoom();
UpdateRootLayerStateForSynchronousInputHandler();
}
void InputHandler::PinchGestureEnd(const gfx::Point& anchor) {
DCHECK(latched_scroll_type_.has_value() || !CurrentlyScrollingNode());
bool snap_to_min = latched_scroll_type_.has_value() &&
latched_scroll_type_ == ui::ScrollInputType::kWheel;
pinch_gesture_active_ = false;
if (pinch_gesture_end_should_clear_scrolling_node_) {
pinch_gesture_end_should_clear_scrolling_node_ = false;
ClearCurrentlyScrollingNode();
}
GetViewport().PinchEnd(anchor, snap_to_min);
compositor_delegate_->GetImplDeprecated()
.browser_controls_manager()
->PinchEnd();
SetNeedsCommit();
compositor_delegate_->DidEndPinchZoom();
}
void InputHandler::SetNeedsAnimateInput() {
compositor_delegate_->GetImplDeprecated().SetNeedsAnimateInput();
}
bool InputHandler::IsCurrentlyScrollingViewport() const {
auto* node = CurrentlyScrollingNode();
if (!node)
return false;
return GetViewport().ShouldScroll(*node);
}
EventListenerProperties InputHandler::GetEventListenerProperties(
EventListenerClass event_class) const {
return ActiveTree().event_listener_properties(event_class);
}
bool InputHandler::HasBlockingWheelEventHandlerAt(
const gfx::Point& viewport_point) const {
gfx::PointF device_viewport_point = gfx::ScalePoint(
gfx::PointF(viewport_point), compositor_delegate_->DeviceScaleFactor());
LayerImpl* layer_impl_with_wheel_event_handler =
ActiveTree().FindLayerThatIsHitByPointInWheelEventHandlerRegion(
device_viewport_point);
return layer_impl_with_wheel_event_handler;
}
InputHandler::TouchStartOrMoveEventListenerType
InputHandler::EventListenerTypeForTouchStartOrMoveAt(
const gfx::Point& viewport_point,
TouchAction* out_touch_action) {
gfx::PointF device_viewport_point = gfx::ScalePoint(
gfx::PointF(viewport_point), compositor_delegate_->DeviceScaleFactor());
LayerImpl* layer_impl_with_touch_handler =
ActiveTree().FindLayerThatIsHitByPointInTouchHandlerRegion(
device_viewport_point);
if (layer_impl_with_touch_handler == nullptr) {
if (out_touch_action)
*out_touch_action = TouchAction::kAuto;
#if BUILDFLAG(IS_OHOS)
LayerImpl* layer_impl = ActiveTree().FindLayerThatIsHitByPoint(device_viewport_point);
if (layer_impl) {
if (layer_impl->may_contain_native()) {
return InputHandler::TouchStartOrMoveEventListenerType::HANDLER;
}
}
#endif
return InputHandler::TouchStartOrMoveEventListenerType::NO_HANDLER;
}
if (out_touch_action) {
gfx::Transform layer_screen_space_transform =
layer_impl_with_touch_handler->ScreenSpaceTransform();
gfx::Transform inverse_layer_screen_space =
layer_screen_space_transform.GetCheckedInverse();
bool clipped = false;
gfx::Point3F planar_point = MathUtil::ProjectPoint3D(
inverse_layer_screen_space, device_viewport_point, &clipped);
gfx::PointF hit_test_point_in_layer_space =
gfx::PointF(planar_point.x(), planar_point.y());
const auto& region = layer_impl_with_touch_handler->touch_action_region();
gfx::Point point = gfx::ToRoundedPoint(hit_test_point_in_layer_space);
*out_touch_action = region.GetAllowedTouchAction(point);
}
if (!CurrentlyScrollingNode())
return InputHandler::TouchStartOrMoveEventListenerType::HANDLER;
LayerImpl* layer_impl =
ActiveTree().FindLayerThatIsHitByPoint(device_viewport_point);
bool is_ancestor = IsScrolledBy(layer_impl, CurrentlyScrollingNode());
return is_ancestor ? InputHandler::TouchStartOrMoveEventListenerType::
HANDLER_ON_SCROLLING_LAYER
: InputHandler::TouchStartOrMoveEventListenerType::HANDLER;
}
std::unique_ptr<LatencyInfoSwapPromiseMonitor>
InputHandler::CreateLatencyInfoSwapPromiseMonitor(ui::LatencyInfo* latency) {
return compositor_delegate_->GetImplDeprecated()
.CreateLatencyInfoSwapPromiseMonitor(latency);
}
std::unique_ptr<EventsMetricsManager::ScopedMonitor>
InputHandler::GetScopedEventMetricsMonitor(
EventsMetricsManager::ScopedMonitor::DoneCallback done_callback) {
return compositor_delegate_->GetImplDeprecated().GetScopedEventMetricsMonitor(
std::move(done_callback));
}
ScrollElasticityHelper* InputHandler::CreateScrollElasticityHelper() {
DCHECK(!scroll_elasticity_helper_);
if (compositor_delegate_->GetSettings().enable_elastic_overscroll) {
scroll_elasticity_helper_.reset(
ScrollElasticityHelper::CreateForLayerTreeHostImpl(
&compositor_delegate_->GetImplDeprecated()));
}
return scroll_elasticity_helper_.get();
}
void InputHandler::DestroyScrollElasticityHelper() {
scroll_elasticity_helper_->SetStretchAmount(gfx::Vector2dF());
scroll_elasticity_helper_.reset();
}
bool InputHandler::GetScrollOffsetForLayer(ElementId element_id,
gfx::PointF* offset) {
ScrollTree& scroll_tree = GetScrollTree();
ScrollNode* scroll_node = scroll_tree.FindNodeFromElementId(element_id);
if (!scroll_node)
return false;
*offset = scroll_tree.current_scroll_offset(element_id);
return true;
}
bool InputHandler::ScrollLayerTo(ElementId element_id,
const gfx::PointF& offset) {
ScrollTree& scroll_tree = GetScrollTree();
ScrollNode* scroll_node = scroll_tree.FindNodeFromElementId(element_id);
if (!scroll_node)
return false;
scroll_tree.ScrollBy(*scroll_node,
offset - scroll_tree.current_scroll_offset(element_id),
&ActiveTree());
return true;
}
bool InputHandler::ScrollingShouldSwitchtoMainThread() {
DCHECK(!base::FeatureList::IsEnabled(features::kScrollUnification));
ScrollTree& scroll_tree = GetScrollTree();
ScrollNode* scroll_node = scroll_tree.CurrentlyScrollingNode();
if (!scroll_node)
return true;
for (; scroll_tree.parent(scroll_node);
scroll_node = scroll_tree.parent(scroll_node)) {
if (!!scroll_node->main_thread_scrolling_reasons)
return true;
}
return false;
}
bool InputHandler::GetSnapFlingInfoAndSetAnimatingSnapTarget(
const gfx::Vector2dF& natural_displacement_in_viewport,
gfx::PointF* out_initial_position,
gfx::PointF* out_target_position) {
ScrollNode* scroll_node = CurrentlyScrollingNode();
if (!scroll_node || !scroll_node->snap_container_data.has_value())
return false;
const SnapContainerData& data = scroll_node->snap_container_data.value();
float scale_factor = ActiveTree().page_scale_factor_for_scroll();
gfx::Vector2dF natural_displacement_in_content =
gfx::ScaleVector2d(natural_displacement_in_viewport, 1.f / scale_factor);
gfx::PointF current_offset = GetVisualScrollOffset(*scroll_node);
*out_initial_position = current_offset;
bool use_fractional_offsets = true;
TargetSnapAreaElementIds snap_target_ids;
std::unique_ptr<SnapSelectionStrategy> strategy =
SnapSelectionStrategy::CreateForEndAndDirection(
current_offset, natural_displacement_in_content,
use_fractional_offsets);
if (!data.FindSnapPosition(*strategy, out_target_position, &snap_target_ids))
return false;
scroll_animating_snap_target_ids_ = snap_target_ids;
out_target_position->Scale(scale_factor);
out_initial_position->Scale(scale_factor);
return true;
}
void InputHandler::ScrollEndForSnapFling(bool did_finish) {
ScrollNode* scroll_node = CurrentlyScrollingNode();
if (did_finish && scroll_node &&
scroll_node->snap_container_data.has_value()) {
scroll_node->snap_container_data.value().SetTargetSnapAreaElementIds(
scroll_animating_snap_target_ids_);
updated_snapped_elements_[scroll_node->element_id] =
scroll_animating_snap_target_ids_;
SetNeedsCommit();
}
scroll_animating_snap_target_ids_ = TargetSnapAreaElementIds();
ScrollEnd(false );
}
void InputHandler::NotifyInputEvent() {
compositor_delegate_->GetImplDeprecated().NotifyInputEvent();
}
void InputHandler::ProcessCommitDeltas(
CompositorCommitData* commit_data,
const MutatorHost* main_thread_mutator_host) {
DCHECK(commit_data);
if (ActiveTree().LayerListIsEmpty())
return;
ElementId inner_viewport_scroll_element_id =
InnerViewportScrollNode() ? InnerViewportScrollNode()->element_id
: ElementId();
base::flat_map<ElementId, TargetSnapAreaElementIds> snapped_elements;
updated_snapped_elements_.swap(snapped_elements);
GetScrollTree().CollectScrollDeltas(
commit_data, inner_viewport_scroll_element_id,
compositor_delegate_->GetSettings().commit_fractional_scroll_deltas,
snapped_elements, main_thread_mutator_host);
DCHECK(!commit_data->manipulation_info);
if (has_scrolled_by_wheel_)
commit_data->manipulation_info |= kManipulationInfoWheel;
if (has_scrolled_by_touch_)
commit_data->manipulation_info |= kManipulationInfoTouch;
if (has_scrolled_by_precisiontouchpad_)
commit_data->manipulation_info |= kManipulationInfoPrecisionTouchPad;
if (has_pinch_zoomed_)
commit_data->manipulation_info |= kManipulationInfoPinchZoom;
if (has_scrolled_by_scrollbar_)
commit_data->manipulation_info |= kManipulationInfoScrollbar;
has_scrolled_by_wheel_ = false;
has_scrolled_by_touch_ = false;
has_scrolled_by_precisiontouchpad_ = false;
has_pinch_zoomed_ = false;
has_scrolled_by_scrollbar_ = false;
commit_data->scroll_end_data.scroll_gesture_did_end = scroll_gesture_did_end_;
scroll_gesture_did_end_ = false;
commit_data->overscroll_delta = overscroll_delta_for_main_thread_;
overscroll_delta_for_main_thread_ = gfx::Vector2dF();
commit_data->scroll_latched_element_id = last_latched_scroller_;
if (commit_data->scroll_end_data.scroll_gesture_did_end) {
last_latched_scroller_ = ElementId();
commit_data->scroll_end_data.gesture_affects_outer_viewport_scroll =
outer_viewport_consumed_delta_;
outer_viewport_consumed_delta_ = false;
}
}
void InputHandler::TickAnimations(base::TimeTicks monotonic_time) {
if (input_handler_client_) {
input_handler_client_->Animate(monotonic_time);
}
}
void InputHandler::WillShutdown() {
if (input_handler_client_) {
input_handler_client_.ExtractAsDangling()->WillShutdown();
}
if (scroll_elasticity_helper_)
scroll_elasticity_helper_.reset();
}
void InputHandler::WillDraw() {
if (input_handler_client_)
input_handler_client_->ReconcileElasticOverscrollAndRootScroll();
}
void InputHandler::WillBeginImplFrame(const viz::BeginFrameArgs& args) {
if (input_handler_client_) {
scrollbar_controller_->WillBeginImplFrame();
input_handler_client_->DeliverInputForBeginFrame(args);
}
}
void InputHandler::DidCommit() {
if (input_handler_client_ && compositor_delegate_->IsInHighLatencyMode())
input_handler_client_->DeliverInputForHighLatencyMode();
}
void InputHandler::DidActivatePendingTree() {
if (!CurrentlyScrollingNode())
ClearCurrentlyScrollingNode();
UpdateRootLayerStateForSynchronousInputHandler();
}
void InputHandler::DidFinishImplFrame() {
if (input_handler_client_) {
input_handler_client_->DidFinishImplFrame();
}
}
void InputHandler::OnBeginImplFrameDeadline() {
if (!IsCurrentlyScrolling()) {
return;
}
if (input_handler_client_) {
input_handler_client_->DeliverInputForDeadline();
}
}
void InputHandler::RootLayerStateMayHaveChanged() {
UpdateRootLayerStateForSynchronousInputHandler();
}
void InputHandler::DidRegisterScrollbar(ElementId scroll_element_id,
ScrollbarOrientation orientation) {
scrollbar_controller_->DidRegisterScrollbar(scroll_element_id, orientation);
}
void InputHandler::DidUnregisterScrollbar(ElementId scroll_element_id,
ScrollbarOrientation orientation) {
scrollbar_controller_->DidUnregisterScrollbar(scroll_element_id, orientation);
}
void InputHandler::ScrollOffsetAnimationFinished() {
TRACE_EVENT0("cc", "InputHandler::ScrollOffsetAnimationFinished");
if (!IsAnimatingForSnap() &&
SnapAtScrollEnd(SnapReason::kScrollOffsetAnimationFinished))
return;
ScrollNode* scroll_node = CurrentlyScrollingNode();
if (scroll_node && scroll_node->snap_container_data.has_value()) {
scroll_node->snap_container_data.value().SetTargetSnapAreaElementIds(
scroll_animating_snap_target_ids_);
updated_snapped_elements_[scroll_node->element_id] =
scroll_animating_snap_target_ids_;
SetNeedsCommit();
}
scroll_animating_snap_target_ids_ = TargetSnapAreaElementIds();
if (deferred_scroll_end_) {
ScrollEnd(false);
return;
}
}
void InputHandler::SetPrefersReducedMotion(bool prefers_reduced_motion) {
if (prefers_reduced_motion_ == prefers_reduced_motion)
return;
prefers_reduced_motion_ = prefers_reduced_motion;
if (input_handler_client_)
input_handler_client_->SetPrefersReducedMotion(prefers_reduced_motion_);
}
bool InputHandler::IsCurrentlyScrolling() const {
return CurrentlyScrollingNode();
}
ActivelyScrollingType InputHandler::GetActivelyScrollingType() const {
if (!CurrentlyScrollingNode())
return ActivelyScrollingType::kNone;
if (!last_scroll_update_state_)
return ActivelyScrollingType::kNone;
if (!delta_consumed_for_scroll_gesture_)
return ActivelyScrollingType::kNone;
if (ShouldAnimateScroll(last_scroll_update_state_.value()))
return ActivelyScrollingType::kAnimated;
return ActivelyScrollingType::kPrecise;
}
#if BUILDFLAG(IS_OHOS)
void InputHandler::HandleScrollUpdateForInternalBeginFrame(const viz::BeginFrameArgs& args) {
TRACE_EVENT0("cc", "ThreadedInputHandler::HandleScrollUpdateForInternalBeginFrame");
if (input_handler_client_) {
input_handler_client_->WillHandleScrollUpdateForInternalBeginFrame(args);
scrollbar_controller_->WillBeginImplFrame();
input_handler_client_->DeliverInputForBeginFrame(args);
}
}
LayerImpl* InputHandler::GetLayerImplIsHitByPoint(const gfx::Point& viewport_point) {
gfx::PointF device_viewport_point =
gfx::ScalePoint(gfx::PointF(viewport_point),
compositor_delegate_->DeviceScaleFactor());
return ActiveTree().FindLayerThatIsHitByPoint(device_viewport_point);
}
void InputHandler::TriggerVsyncImplTask() {
TRACE_EVENT0("cc", "InputHandler::TriggerVsyncImplTask");
compositor_delegate_->GetImplDeprecated().TriggerVsyncImplTask();
}
LayerImpl* InputHandler::GetLayerImplById(int id) {
return ActiveTree().LayerById(id);
}
LayerImpl* InputHandler::GetNativeLayerImpl(const gfx::Point& viewport_point) {
gfx::PointF device_viewport_point =
gfx::ScalePoint(gfx::PointF(viewport_point),
compositor_delegate_->DeviceScaleFactor());
return ActiveTree().FindLayerThatIsHitByPointNative(device_viewport_point);
}
void InputHandler::SetHandledTouchEvent(bool handledTouchEvent) {
compositor_delegate_->GetImplDeprecated().SetHandledTouchEvent(handledTouchEvent);
}
#endif
bool InputHandler::IsCurrentScrollMainRepainted() const {
const ScrollNode* scroll_node = CurrentlyScrollingNode();
if (!scroll_node)
return false;
uint32_t repaint_reasons =
GetScrollTree().GetMainThreadRepaintReasons(*scroll_node);
return repaint_reasons != MainThreadScrollingReason::kNotScrollingOnMain;
}
bool InputHandler::HasQueuedInput() const {
if (input_handler_client_) {
return input_handler_client_->HasQueuedInput();
}
return false;
}
ScrollNode* InputHandler::CurrentlyScrollingNode() {
return GetScrollTree().CurrentlyScrollingNode();
}
const ScrollNode* InputHandler::CurrentlyScrollingNode() const {
return GetScrollTree().CurrentlyScrollingNode();
}
ScrollTree& InputHandler::GetScrollTree() {
return compositor_delegate_->GetScrollTree();
}
ScrollTree& InputHandler::GetScrollTree() const {
return compositor_delegate_->GetScrollTree();
}
ScrollNode* InputHandler::InnerViewportScrollNode() const {
return ActiveTree().InnerViewportScrollNode();
}
ScrollNode* InputHandler::OuterViewportScrollNode() const {
return ActiveTree().OuterViewportScrollNode();
}
Viewport& InputHandler::GetViewport() const {
return compositor_delegate_->GetImplDeprecated().viewport();
}
void InputHandler::SetNeedsCommit() {
compositor_delegate_->SetNeedsCommit();
}
LayerTreeImpl& InputHandler::ActiveTree() {
DCHECK(compositor_delegate_->GetImplDeprecated().active_tree());
return *compositor_delegate_->GetImplDeprecated().active_tree();
}
LayerTreeImpl& InputHandler::ActiveTree() const {
DCHECK(compositor_delegate_->GetImplDeprecated().active_tree());
return *compositor_delegate_->GetImplDeprecated().active_tree();
}
FrameSequenceTrackerType InputHandler::GetTrackerTypeForScroll(
ui::ScrollInputType input_type) const {
switch (input_type) {
case ui::ScrollInputType::kWheel:
return FrameSequenceTrackerType::kWheelScroll;
case ui::ScrollInputType::kTouchscreen:
return FrameSequenceTrackerType::kTouchScroll;
case ui::ScrollInputType::kScrollbar:
return FrameSequenceTrackerType::kScrollbarScroll;
case ui::ScrollInputType::kAutoscroll:
return FrameSequenceTrackerType::kMaxType;
}
}
bool InputHandler::IsMainThreadScrolling(
const InputHandler::ScrollStatus& status,
const ScrollNode* scroll_node) const {
if (status.thread == InputHandler::ScrollThread::SCROLL_ON_MAIN_THREAD) {
if (!!scroll_node->main_thread_scrolling_reasons) {
DCHECK(MainThreadScrollingReason::MainThreadCanSetScrollReasons(
status.main_thread_scrolling_reasons));
} else {
DCHECK(MainThreadScrollingReason::CompositorCanSetScrollReasons(
status.main_thread_scrolling_reasons));
}
return true;
}
return false;
}
float InputHandler::LineStep() const {
return kPixelsPerLineStep * ActiveTree().painted_device_scale_factor();
}
gfx::Vector2dF InputHandler::ResolveScrollGranularityToPixels(
const ScrollNode& scroll_node,
const gfx::Vector2dF& scroll_delta,
ui::ScrollGranularity granularity) {
gfx::Vector2dF pixel_delta = scroll_delta;
if (granularity == ui::ScrollGranularity::kScrollByPage) {
granularity = ui::ScrollGranularity::kScrollByPercentage;
pixel_delta.Scale(kMinFractionToStepWhenPaging);
}
if (granularity == ui::ScrollGranularity::kScrollByPercentage) {
gfx::SizeF scroller_size = gfx::SizeF(scroll_node.container_bounds);
gfx::SizeF viewport_size(compositor_delegate_->VisualDeviceViewportSize());
scroller_size.Scale(compositor_delegate_->PageScaleFactor());
viewport_size.InvScale(compositor_delegate_->DeviceScaleFactor());
pixel_delta = ScrollUtils::ResolveScrollPercentageToPixels(
pixel_delta, scroller_size, viewport_size);
}
if (granularity == ui::ScrollGranularity::kScrollByLine) {
pixel_delta.Scale(LineStep(), LineStep());
}
return pixel_delta;
}
InputHandler::ScrollStatus InputHandler::TryScroll(
const ScrollTree& scroll_tree,
ScrollNode* scroll_node) const {
DCHECK(!base::FeatureList::IsEnabled(features::kScrollUnification));
DCHECK(scroll_node->transform_id != kInvalidPropertyNodeId);
InputHandler::ScrollStatus scroll_status;
scroll_status.main_thread_scrolling_reasons =
MainThreadScrollingReason::kNotScrollingOnMain;
if (scroll_node->main_thread_scrolling_reasons) {
TRACE_EVENT1("cc", "LayerImpl::TryScroll: Failed ShouldScrollOnMainThread",
"MainThreadScrollingReason",
scroll_node->main_thread_scrolling_reasons);
scroll_status.thread = InputHandler::ScrollThread::SCROLL_ON_MAIN_THREAD;
scroll_status.main_thread_scrolling_reasons =
scroll_node->main_thread_scrolling_reasons;
return scroll_status;
}
gfx::Transform screen_space_transform =
scroll_tree.ScreenSpaceTransform(scroll_node->id);
if (!screen_space_transform.IsInvertible()) {
TRACE_EVENT0("cc", "LayerImpl::TryScroll: Ignored NonInvertibleTransform");
scroll_status.thread = InputHandler::ScrollThread::SCROLL_IGNORED;
return scroll_status;
}
if (!scroll_node->scrollable) {
TRACE_EVENT0("cc", "LayerImpl::tryScroll: Ignored not scrollable");
scroll_status.thread = InputHandler::ScrollThread::SCROLL_IGNORED;
return scroll_status;
}
if (scroll_node->id != kRootPropertyNodeId &&
scroll_node->id != kSecondaryRootPropertyNodeId &&
!scroll_node->scrolls_inner_viewport &&
!ActiveTree().LayerByElementId(scroll_node->element_id)) {
TRACE_EVENT0("cc",
"LayerImpl::tryScroll: Failed due to no scrolling layer");
scroll_status.thread = InputHandler::ScrollThread::SCROLL_ON_MAIN_THREAD;
scroll_status.main_thread_scrolling_reasons =
MainThreadScrollingReason::kNoScrollingLayer;
return scroll_status;
}
gfx::PointF max_scroll_offset = scroll_tree.MaxScrollOffset(scroll_node->id);
if (max_scroll_offset.x() <= 0 && max_scroll_offset.y() <= 0 &&
!GetViewport().ShouldScroll(*scroll_node)) {
TRACE_EVENT0("cc",
"LayerImpl::tryScroll: Ignored. Technically scrollable,"
" but has no affordance in either direction.");
scroll_status.thread = InputHandler::ScrollThread::SCROLL_IGNORED;
return scroll_status;
}
scroll_status.thread = InputHandler::ScrollThread::SCROLL_ON_IMPL_THREAD;
return scroll_status;
}
base::flat_set<int> InputHandler::NonFastScrollableNodes(
const gfx::PointF& device_viewport_point) const {
base::flat_set<int> non_fast_scrollable_nodes;
const auto& non_fast_layers =
ActiveTree().FindLayersHitByPointInNonFastScrollableRegion(
device_viewport_point);
for (const auto* layer : non_fast_layers)
non_fast_scrollable_nodes.insert(layer->scroll_tree_index());
return non_fast_scrollable_nodes;
}
ScrollNode* InputHandler::FindScrollNodeForCompositedScrolling(
const gfx::PointF& device_viewport_point,
LayerImpl* layer_impl,
bool* scroll_on_main_thread,
uint32_t* main_thread_scrolling_reasons) {
DCHECK(!base::FeatureList::IsEnabled(features::kScrollUnification));
DCHECK(scroll_on_main_thread);
DCHECK(main_thread_scrolling_reasons);
*main_thread_scrolling_reasons =
MainThreadScrollingReason::kNotScrollingOnMain;
const auto& non_fast_scrollable_nodes =
NonFastScrollableNodes(device_viewport_point);
if (layer_impl && layer_impl->IsScrollbarLayer()) {
layer_impl = ActiveTree().LayerByElementId(
ToScrollbarLayer(layer_impl)->scroll_element_id());
}
ScrollTree& scroll_tree = GetScrollTree();
ScrollNode* impl_scroll_node = nullptr;
if (layer_impl) {
ScrollNode* scroll_node = scroll_tree.Node(layer_impl->scroll_tree_index());
for (; scroll_tree.parent(scroll_node);
scroll_node = scroll_tree.parent(scroll_node)) {
InputHandler::ScrollStatus status = TryScroll(scroll_tree, scroll_node);
if (IsMainThreadScrolling(status, scroll_node)) {
*scroll_on_main_thread = true;
*main_thread_scrolling_reasons = status.main_thread_scrolling_reasons;
return scroll_node;
}
if (non_fast_scrollable_nodes.contains(scroll_node->id)) {
*scroll_on_main_thread = true;
*main_thread_scrolling_reasons =
MainThreadScrollingReason::kNonFastScrollableRegion;
return scroll_node;
}
if (status.thread == InputHandler::ScrollThread::SCROLL_ON_IMPL_THREAD &&
!impl_scroll_node) {
impl_scroll_node = scroll_node;
}
}
}
if (!impl_scroll_node)
impl_scroll_node = InnerViewportScrollNode();
if (!impl_scroll_node)
return nullptr;
impl_scroll_node = GetNodeToScroll(impl_scroll_node);
InputHandler::ScrollStatus status = TryScroll(scroll_tree, impl_scroll_node);
if (IsMainThreadScrolling(status, impl_scroll_node)) {
*scroll_on_main_thread = true;
*main_thread_scrolling_reasons = status.main_thread_scrolling_reasons;
} else if (non_fast_scrollable_nodes.contains(impl_scroll_node->id)) {
*scroll_on_main_thread = true;
*main_thread_scrolling_reasons =
MainThreadScrollingReason::kNonFastScrollableRegion;
}
return impl_scroll_node;
}
InputHandler::ScrollHitTestResult InputHandler::HitTestScrollNode(
const gfx::PointF& device_viewport_point) const {
ScrollHitTestResult result;
result.scroll_node = nullptr;
result.hit_test_successful = false;
std::vector<const LayerImpl*> layers =
ActiveTree().FindAllLayersUpToAndIncludingFirstScrollable(
device_viewport_point);
const LayerImpl* scroller_layer =
(!layers.empty() && layers.back()->IsScrollerOrScrollbar())
? layers.back()
: nullptr;
for (const auto* layer_impl : layers) {
if (!IsInitialScrollHitTestReliable(layer_impl, scroller_layer)) {
TRACE_EVENT_INSTANT0("cc", "Failed Hit Test", TRACE_EVENT_SCOPE_THREAD);
result.main_thread_hit_test_reasons =
MainThreadScrollingReason::kFailedHitTest;
return result;
}
if (ActiveTree().PointHitsNonFastScrollableRegion(device_viewport_point,
*layer_impl)) {
result.main_thread_hit_test_reasons =
MainThreadScrollingReason::kNonFastScrollableRegion;
return result;
}
}
if (scroller_layer && scroller_layer->IsScrollbarLayer()) {
scroller_layer = ActiveTree().LayerByElementId(
ToScrollbarLayer(scroller_layer)->scroll_element_id());
}
if (!scroller_layer) {
result.hit_test_successful = true;
if (InnerViewportScrollNode())
result.scroll_node = GetNodeToScroll(InnerViewportScrollNode());
return result;
}
ScrollNode* scroll_node =
GetScrollTree().Node(scroller_layer->scroll_tree_index());
result.scroll_node = GetNodeToScroll(scroll_node);
result.hit_test_successful = true;
return result;
}
bool InputHandler::IsTouchDraggingScrollbar(
LayerImpl* first_scrolling_layer_or_scrollbar,
ui::ScrollInputType type) {
return first_scrolling_layer_or_scrollbar &&
first_scrolling_layer_or_scrollbar->IsScrollbarLayer() &&
type == ui::ScrollInputType::kTouchscreen;
}
ScrollNode* InputHandler::GetNodeToScroll(ScrollNode* node) const {
DCHECK(!node->prevent_viewport_scrolling_from_inner ||
node->scrolls_inner_viewport);
if (node->scrolls_inner_viewport &&
!node->prevent_viewport_scrolling_from_inner) {
DCHECK(OuterViewportScrollNode());
return OuterViewportScrollNode();
}
return node;
}
bool InputHandler::IsInitialScrollHitTestReliable(
const LayerImpl* layer_impl,
const LayerImpl* first_scrolling_layer_or_scrollbar) const {
if (layer_impl->IsScrollbarLayer()) {
DCHECK(layer_impl == first_scrolling_layer_or_scrollbar);
return true;
}
ScrollNode* closest_scroll_node = nullptr;
auto& scroll_tree = GetScrollTree();
ScrollNode* scroll_node = scroll_tree.Node(layer_impl->scroll_tree_index());
for (; scroll_tree.parent(scroll_node);
scroll_node = scroll_tree.parent(scroll_node)) {
if (scroll_node->scrollable) {
closest_scroll_node = GetNodeToScroll(scroll_node);
break;
}
}
if ((first_scrolling_layer_or_scrollbar && !closest_scroll_node) ||
(closest_scroll_node && !first_scrolling_layer_or_scrollbar)) {
return false;
}
if (!first_scrolling_layer_or_scrollbar && !closest_scroll_node) {
return true;
}
if (!first_scrolling_layer_or_scrollbar->IsScrollbarLayer()) {
return closest_scroll_node->id ==
first_scrolling_layer_or_scrollbar->scroll_tree_index();
}
return false;
}
gfx::Vector2dF InputHandler::ComputeScrollDelta(const ScrollNode& scroll_node,
const gfx::Vector2dF& delta) {
ScrollTree& scroll_tree = GetScrollTree();
float scale_factor = compositor_delegate_->PageScaleFactor();
gfx::Vector2dF adjusted_scroll(delta);
adjusted_scroll.InvScale(scale_factor);
adjusted_scroll = UserScrollableDelta(scroll_node, adjusted_scroll);
gfx::PointF old_offset =
scroll_tree.current_scroll_offset(scroll_node.element_id);
gfx::PointF new_offset = scroll_tree.ClampScrollOffsetToLimits(
old_offset + adjusted_scroll, scroll_node);
return new_offset - old_offset;
}
bool InputHandler::CalculateLocalScrollDeltaAndStartPoint(
const ScrollNode& scroll_node,
const gfx::PointF& viewport_point,
const gfx::Vector2dF& viewport_delta,
gfx::Vector2dF* out_local_scroll_delta,
gfx::PointF* out_local_start_point ) {
if (scroll_node.transform_id == kInvalidPropertyNodeId) {
return false;
}
const gfx::Transform screen_space_transform =
GetScrollTree().ScreenSpaceTransform(scroll_node.id);
gfx::Transform inverse_screen_space_transform =
screen_space_transform.GetCheckedInverse();
float scale_from_viewport_to_screen_space =
compositor_delegate_->DeviceScaleFactor();
gfx::PointF screen_space_point =
gfx::ScalePoint(viewport_point, scale_from_viewport_to_screen_space);
gfx::Vector2dF screen_space_delta = viewport_delta;
screen_space_delta.Scale(scale_from_viewport_to_screen_space);
bool start_clipped, end_clipped;
gfx::PointF screen_space_end_point = screen_space_point + screen_space_delta;
gfx::PointF local_start_point = MathUtil::ProjectPoint(
inverse_screen_space_transform, screen_space_point, &start_clipped);
gfx::PointF local_end_point = MathUtil::ProjectPoint(
inverse_screen_space_transform, screen_space_end_point, &end_clipped);
DCHECK(out_local_scroll_delta);
*out_local_scroll_delta = local_end_point - local_start_point;
if (out_local_start_point)
*out_local_start_point = local_start_point;
if (start_clipped || end_clipped)
return false;
return true;
}
gfx::Vector2dF InputHandler::ScrollNodeWithViewportSpaceDelta(
const ScrollNode& scroll_node,
const gfx::PointF& viewport_point,
const gfx::Vector2dF& viewport_delta) {
ScrollTree& scroll_tree = GetScrollTree();
gfx::PointF local_start_point;
gfx::Vector2dF local_scroll_delta;
if (!CalculateLocalScrollDeltaAndStartPoint(
scroll_node, viewport_point, viewport_delta, &local_scroll_delta,
&local_start_point)) {
return gfx::Vector2dF();
}
bool scrolls_outer_viewport = scroll_node.scrolls_outer_viewport;
TRACE_EVENT2("cc", "ScrollNodeWithViewportSpaceDelta", "delta_y",
local_scroll_delta.y(), "is_outer", scrolls_outer_viewport);
gfx::PointF previous_offset =
scroll_tree.current_scroll_offset(scroll_node.element_id);
scroll_tree.ScrollBy(scroll_node, local_scroll_delta, &ActiveTree());
gfx::Vector2dF scrolled =
scroll_tree.current_scroll_offset(scroll_node.element_id) -
previous_offset;
TRACE_EVENT_INSTANT1("cc", "ConsumedDelta", TRACE_EVENT_SCOPE_THREAD, "y",
scrolled.y());
gfx::PointF actual_local_end_point = local_start_point + scrolled;
bool end_clipped;
const gfx::Transform screen_space_transform =
scroll_tree.ScreenSpaceTransform(scroll_node.id);
gfx::PointF actual_screen_space_end_point = MathUtil::MapPoint(
screen_space_transform, actual_local_end_point, &end_clipped);
DCHECK(!end_clipped);
if (end_clipped)
return gfx::Vector2dF();
float scale_from_viewport_to_screen_space =
compositor_delegate_->DeviceScaleFactor();
gfx::PointF actual_viewport_end_point = gfx::ScalePoint(
actual_screen_space_end_point, 1.f / scale_from_viewport_to_screen_space);
return actual_viewport_end_point - viewport_point;
}
gfx::Vector2dF InputHandler::ScrollNodeWithLocalDelta(
const ScrollNode& scroll_node,
const gfx::Vector2dF& local_delta) const {
bool scrolls_outer_viewport = scroll_node.scrolls_outer_viewport;
TRACE_EVENT2("cc", "ScrollNodeWithLocalDelta", "delta_y", local_delta.y(),
"is_outer", scrolls_outer_viewport);
float page_scale_factor = compositor_delegate_->PageScaleFactor();
ScrollTree& scroll_tree = GetScrollTree();
gfx::PointF previous_offset =
scroll_tree.current_scroll_offset(scroll_node.element_id);
gfx::Vector2dF delta = local_delta;
delta.InvScale(page_scale_factor);
scroll_tree.ScrollBy(scroll_node, delta, &ActiveTree());
gfx::Vector2dF scrolled =
scroll_tree.current_scroll_offset(scroll_node.element_id) -
previous_offset;
gfx::Vector2dF consumed_scroll(scrolled.x(), scrolled.y());
consumed_scroll.Scale(page_scale_factor);
TRACE_EVENT_INSTANT1("cc", "ConsumedDelta", TRACE_EVENT_SCOPE_THREAD, "y",
consumed_scroll.y());
return consumed_scroll;
}
gfx::Vector2dF InputHandler::ScrollSingleNode(const ScrollNode& scroll_node,
const gfx::Vector2dF& delta,
const gfx::Point& viewport_point,
bool is_direct_manipulation) {
gfx::Vector2dF adjusted_delta = UserScrollableDelta(scroll_node, delta);
if (is_direct_manipulation) {
gfx::Vector2dF scaled_delta(adjusted_delta);
scaled_delta.InvScale(ActiveTree().external_page_scale_factor());
return ScrollNodeWithViewportSpaceDelta(
scroll_node, gfx::PointF(viewport_point), scaled_delta);
}
return ScrollNodeWithLocalDelta(scroll_node, adjusted_delta);
}
void InputHandler::ScrollLatchedScroller(ScrollState* scroll_state,
base::TimeDelta delayed_by) {
DCHECK(CurrentlyScrollingNode());
DCHECK(scroll_state);
DCHECK(latched_scroll_type_.has_value());
ScrollNode& scroll_node = *CurrentlyScrollingNode();
const gfx::Vector2dF delta(scroll_state->delta_x(), scroll_state->delta_y());
TRACE_EVENT2("cc", "InputHandler::ScrollLatchedScroller", "delta_x",
delta.x(), "delta_y", delta.y());
gfx::Vector2dF applied_delta;
gfx::Vector2dF delta_applied_to_content;
const float kEpsilon = 0.1f;
if (ShouldAnimateScroll(*scroll_state)) {
DCHECK(!scroll_state->is_in_inertial_phase());
if (ElementId id = compositor_delegate_->GetImplDeprecated()
.mutator_host()
->ImplOnlyScrollAnimatingElement()) {
TRACE_EVENT_INSTANT0("cc", "UpdateExistingAnimation",
TRACE_EVENT_SCOPE_THREAD);
ScrollNode* animating_scroll_node =
GetScrollTree().FindNodeFromElementId(id);
DCHECK(animating_scroll_node);
DCHECK(animating_scroll_node->id == scroll_node.id ||
animating_scroll_node->scrolls_inner_viewport);
bool animation_updated = ScrollAnimationUpdateTarget(
*animating_scroll_node, delta, delayed_by);
if (animation_updated) {
applied_delta = delta;
} else {
TRACE_EVENT_INSTANT0("cc", "Didn't Update Animation",
TRACE_EVENT_SCOPE_THREAD);
}
} else {
TRACE_EVENT_INSTANT0("cc", "CreateNewAnimation",
TRACE_EVENT_SCOPE_THREAD);
if (scroll_node.scrolls_outer_viewport) {
auto result = GetViewport().ScrollAnimated(delta, delayed_by);
applied_delta = result.consumed_delta;
if (std::abs(result.outer_viewport_scrolled_delta.x()) > kEpsilon ||
std::abs(result.outer_viewport_scrolled_delta.y()) > kEpsilon) {
outer_viewport_consumed_delta_ = true;
}
} else {
applied_delta = ComputeScrollDelta(scroll_node, delta);
compositor_delegate_->GetImplDeprecated().ScrollAnimationCreate(
scroll_node, applied_delta, delayed_by);
}
}
delta_applied_to_content = delta;
} else {
gfx::Point viewport_point(scroll_state->position_x(),
scroll_state->position_y());
if (GetViewport().ShouldScroll(scroll_node)) {
Viewport::ScrollResult result = GetViewport().ScrollBy(
delta, viewport_point, scroll_state->is_direct_manipulation(),
latched_scroll_type_ != ui::ScrollInputType::kWheel,
scroll_node.scrolls_outer_viewport);
applied_delta = result.consumed_delta;
delta_applied_to_content = result.content_scrolled_delta;
if (std::abs(result.outer_viewport_scrolled_delta.x()) > kEpsilon ||
std::abs(result.outer_viewport_scrolled_delta.y()) > kEpsilon) {
outer_viewport_consumed_delta_ = true;
}
} else {
applied_delta = ScrollSingleNode(scroll_node, delta, viewport_point,
scroll_state->is_direct_manipulation());
}
}
bool scrolled = std::abs(applied_delta.x()) > kEpsilon;
scrolled = scrolled || std::abs(applied_delta.y()) > kEpsilon;
if (!scrolled) {
if (scroll_node.scrolls_outer_viewport)
scroll_state->ConsumeDelta(applied_delta.x(), applied_delta.y());
return;
}
if (!GetViewport().ShouldScroll(scroll_node)) {
float angle_threshold = 45;
if (MathUtil::SmallestAngleBetweenVectors(applied_delta, delta) <
angle_threshold) {
applied_delta = delta;
} else {
applied_delta = MathUtil::ProjectVector(delta, applied_delta);
}
delta_applied_to_content = applied_delta;
}
scroll_state->set_caused_scroll(
std::abs(delta_applied_to_content.x()) > kEpsilon,
std::abs(delta_applied_to_content.y()) > kEpsilon);
scroll_state->ConsumeDelta(applied_delta.x(), applied_delta.y());
}
bool InputHandler::CanPropagate(ScrollNode* scroll_node, float x, float y) {
return (x == 0 || scroll_node->overscroll_behavior.x ==
OverscrollBehavior::Type::kAuto) &&
(y == 0 || scroll_node->overscroll_behavior.y ==
OverscrollBehavior::Type::kAuto);
}
ScrollNode* InputHandler::FindNodeToLatch(ScrollState* scroll_state,
ScrollNode* starting_node,
ui::ScrollInputType type) {
ScrollTree& scroll_tree = GetScrollTree();
ScrollNode* scroll_node = nullptr;
ScrollNode* first_scrollable_node = nullptr;
for (ScrollNode* cur_node = starting_node; cur_node;
cur_node = scroll_tree.parent(cur_node)) {
if (GetViewport().ShouldScroll(*cur_node)) {
scroll_node = GetNodeToScroll(cur_node);
break;
}
if (!cur_node->scrollable)
continue;
if (!first_scrollable_node) {
first_scrollable_node = cur_node;
}
if (CanConsumeDelta(*scroll_state, *cur_node)) {
scroll_node = cur_node;
break;
}
float delta_x = scroll_state->is_beginning() ? scroll_state->delta_x_hint()
: scroll_state->delta_x();
float delta_y = scroll_state->is_beginning() ? scroll_state->delta_y_hint()
: scroll_state->delta_y();
if (!CanPropagate(cur_node, delta_x, delta_y)) {
scroll_node = cur_node;
scroll_state->set_is_scroll_chain_cut(true);
break;
}
}
if ((type == ui::ScrollInputType::kAutoscroll) && first_scrollable_node) {
if (!(scroll_node && CanConsumeDelta(*scroll_state, *scroll_node)))
scroll_node = first_scrollable_node;
}
return scroll_node;
}
void InputHandler::UpdateRootLayerStateForSynchronousInputHandler() {
if (!input_handler_client_)
return;
input_handler_client_->UpdateRootLayerStateForSynchronousInputHandler(
ActiveTree().TotalScrollOffset(), ActiveTree().TotalMaxScrollOffset(),
ActiveTree().ScrollableSize(), ActiveTree().current_page_scale_factor(),
ActiveTree().min_page_scale_factor(),
ActiveTree().max_page_scale_factor());
}
void InputHandler::DidLatchToScroller(const ScrollState& scroll_state,
ui::ScrollInputType type) {
DCHECK(CurrentlyScrollingNode());
deferred_scroll_end_ = false;
compositor_delegate_->GetImplDeprecated()
.browser_controls_manager()
->ScrollBegin();
compositor_delegate_->GetImplDeprecated()
.mutator_host()
->ScrollAnimationAbort();
scroll_animating_snap_target_ids_ = TargetSnapAreaElementIds();
last_latched_scroller_ = CurrentlyScrollingNode()->element_id;
latched_scroll_type_ = type;
last_scroll_begin_state_ = scroll_state;
compositor_delegate_->DidStartScroll();
RecordCompositorSlowScrollMetric(type, CC_THREAD);
UpdateScrollSourceInfo(scroll_state, type);
}
bool InputHandler::CanConsumeDelta(const ScrollState& scroll_state,
const ScrollNode& scroll_node) {
gfx::Vector2dF delta_to_scroll;
if (scroll_state.is_beginning()) {
delta_to_scroll = gfx::Vector2dF(scroll_state.delta_x_hint(),
scroll_state.delta_y_hint());
} else {
delta_to_scroll =
gfx::Vector2dF(scroll_state.delta_x(), scroll_state.delta_y());
}
if (delta_to_scroll == gfx::Vector2dF())
return true;
if (scroll_state.is_direct_manipulation()) {
gfx::Vector2dF local_scroll_delta;
if (!CalculateLocalScrollDeltaAndStartPoint(
scroll_node,
gfx::PointF(scroll_state.position_x(), scroll_state.position_y()),
delta_to_scroll, &local_scroll_delta)) {
return false;
}
delta_to_scroll = local_scroll_delta;
} else {
delta_to_scroll = ResolveScrollGranularityToPixels(
scroll_node, delta_to_scroll, scroll_state.delta_granularity());
}
if (ComputeScrollDelta(scroll_node, delta_to_scroll) != gfx::Vector2dF())
return true;
return false;
}
bool InputHandler::ShouldAnimateScroll(const ScrollState& scroll_state) const {
if (!compositor_delegate_->GetSettings().enable_smooth_scroll)
return false;
bool has_precise_scroll_deltas = scroll_state.delta_granularity() ==
ui::ScrollGranularity::kScrollByPrecisePixel;
return !has_precise_scroll_deltas;
}
bool InputHandler::SnapAtScrollEnd(SnapReason reason) {
ScrollNode* scroll_node = CurrentlyScrollingNode();
if (!scroll_node || !scroll_node->snap_container_data.has_value())
return false;
SnapContainerData& data = scroll_node->snap_container_data.value();
gfx::PointF current_position = GetVisualScrollOffset(*scroll_node);
DCHECK(last_scroll_update_state_ || last_scroll_begin_state_);
ScrollState& last_scroll_state = last_scroll_update_state_
? *last_scroll_update_state_
: *last_scroll_begin_state_;
bool imprecise_wheel_scrolling =
latched_scroll_type_ == ui::ScrollInputType::kWheel &&
last_scroll_state.delta_granularity() !=
ui::ScrollGranularity::kScrollByPrecisePixel;
gfx::Vector2dF last_scroll_delta = last_scroll_state.DeltaOrHint();
std::unique_ptr<SnapSelectionStrategy> strategy;
if (imprecise_wheel_scrolling && !last_scroll_delta.IsZero() &&
reason == SnapReason::kScrollOffsetAnimationFinished) {
strategy = SnapSelectionStrategy::CreateForDirection(
current_position, last_scroll_delta, true);
} else {
strategy = SnapSelectionStrategy::CreateForEndPosition(
current_position, did_scroll_x_for_scroll_gesture_,
did_scroll_y_for_scroll_gesture_);
}
gfx::PointF snap_position;
TargetSnapAreaElementIds snap_target_ids;
if (!data.FindSnapPosition(*strategy, &snap_position, &snap_target_ids))
return false;
if (GetViewport().ShouldScroll(*scroll_node)) {
compositor_delegate_->WillScrollContent(scroll_node->element_id);
}
gfx::Vector2dF delta = snap_position - current_position;
bool did_animate = false;
if (scroll_node->scrolls_outer_viewport) {
gfx::Vector2dF scaled_delta(delta);
scaled_delta.Scale(compositor_delegate_->PageScaleFactor());
gfx::Vector2dF consumed_delta =
GetViewport()
.ScrollAnimated(scaled_delta, base::TimeDelta())
.consumed_delta;
did_animate = !consumed_delta.IsZero();
} else {
did_animate =
compositor_delegate_->GetImplDeprecated().ScrollAnimationCreate(
*scroll_node, delta, base::TimeDelta());
}
DCHECK(!IsAnimatingForSnap());
if (did_animate) {
scroll_animating_snap_target_ids_ = snap_target_ids;
} else if (data.SetTargetSnapAreaElementIds(snap_target_ids)) {
updated_snapped_elements_[scroll_node->element_id] = snap_target_ids;
SetNeedsCommit();
}
return did_animate;
}
bool InputHandler::IsAnimatingForSnap() const {
return scroll_animating_snap_target_ids_ != TargetSnapAreaElementIds();
}
gfx::PointF InputHandler::GetVisualScrollOffset(
const ScrollNode& scroll_node) const {
if (scroll_node.scrolls_outer_viewport)
return GetViewport().TotalScrollOffset();
return GetScrollTree().current_scroll_offset(scroll_node.element_id);
}
void InputHandler::ClearCurrentlyScrollingNode() {
TRACE_EVENT0("cc", "InputHandler::ClearCurrentlyScrollingNode");
ActiveTree().ClearCurrentlyScrollingNode();
accumulated_root_overscroll_ = gfx::Vector2dF();
did_scroll_x_for_scroll_gesture_ = false;
did_scroll_y_for_scroll_gesture_ = false;
delta_consumed_for_scroll_gesture_ = false;
scroll_animating_snap_target_ids_ = TargetSnapAreaElementIds();
latched_scroll_type_.reset();
last_scroll_update_state_.reset();
last_scroll_begin_state_.reset();
compositor_delegate_->DidEndScroll();
}
bool InputHandler::ScrollAnimationUpdateTarget(
const ScrollNode& scroll_node,
const gfx::Vector2dF& scroll_delta,
base::TimeDelta delayed_by) {
DCHECK_EQ(scroll_node.element_id, compositor_delegate_->GetImplDeprecated()
.mutator_host()
->ImplOnlyScrollAnimatingElement());
float scale_factor = compositor_delegate_->PageScaleFactor();
gfx::Vector2dF adjusted_delta =
gfx::ScaleVector2d(scroll_delta, 1.f / scale_factor);
adjusted_delta = UserScrollableDelta(scroll_node, adjusted_delta);
bool animation_updated =
compositor_delegate_->GetImplDeprecated()
.mutator_host()
->ImplOnlyScrollAnimationUpdateTarget(
adjusted_delta, GetScrollTree().MaxScrollOffset(scroll_node.id),
compositor_delegate_->GetImplDeprecated()
.CurrentBeginFrameArgs()
.frame_time,
delayed_by);
if (animation_updated) {
compositor_delegate_->DidUpdateScrollAnimationCurve();
scroll_animating_snap_target_ids_ = TargetSnapAreaElementIds();
}
return animation_updated;
}
void InputHandler::UpdateScrollSourceInfo(const ScrollState& scroll_state,
ui::ScrollInputType type) {
if (type == ui::ScrollInputType::kWheel &&
scroll_state.delta_granularity() ==
ui::ScrollGranularity::kScrollByPrecisePixel) {
has_scrolled_by_precisiontouchpad_ = true;
} else if (type == ui::ScrollInputType::kWheel) {
has_scrolled_by_wheel_ = true;
} else if (type == ui::ScrollInputType::kTouchscreen) {
has_scrolled_by_touch_ = true;
} else if (type == ui::ScrollInputType::kScrollbar) {
has_scrolled_by_scrollbar_ = true;
}
}
bool InputHandler::IsScrolledBy(LayerImpl* child, ScrollNode* ancestor) {
DCHECK(ancestor && ancestor->scrollable);
if (!child)
return false;
DCHECK_EQ(child->layer_tree_impl(), &ActiveTree());
ScrollTree& scroll_tree = GetScrollTree();
for (ScrollNode* scroll_node = scroll_tree.Node(child->scroll_tree_index());
scroll_node; scroll_node = scroll_tree.parent(scroll_node)) {
if (scroll_node->id == ancestor->id)
return true;
}
return false;
}
gfx::Vector2dF InputHandler::UserScrollableDelta(
const ScrollNode& node,
const gfx::Vector2dF& delta) const {
gfx::Vector2dF adjusted_delta = delta;
if (!node.user_scrollable_horizontal)
adjusted_delta.set_x(0);
if (!node.user_scrollable_vertical)
adjusted_delta.set_y(0);
return adjusted_delta;
}
bool InputHandler::ScrollbarScrollIsActive() {
return scrollbar_controller_->ScrollbarScrollIsActive();
}
void InputHandler::SetDeferBeginMainFrame(bool defer_begin_main_frame) const {
compositor_delegate_->SetDeferBeginMainFrame(defer_begin_main_frame);
}
void InputHandler::UpdateBrowserControlsState(BrowserControlsState constraints,
BrowserControlsState current,
bool animate) {
compositor_delegate_->UpdateBrowserControlsState(constraints, current,
animate);
}
bool InputHandler::CurrentScrollNeedsFrameAlignment() const {
if (const ScrollNode* node = CurrentlyScrollingNode()) {
if (compositor_delegate_->HasScrollLinkedAnimation(node->element_id)) {
return true;
}
}
return false;
}
}