#include "cc/input/scroll_snap_data.h"
#include <algorithm>
#include <cmath>
#include <limits>
#include <memory>
#include "base/check.h"
#include "base/notreached.h"
#include "cc/input/snap_selection_strategy.h"
#include "ui/gfx/geometry/vector2d_f.h"
namespace cc {
namespace {
gfx::Vector2dF DistanceFromCorridor(double dx,
double dy,
const gfx::RectF& area) {
gfx::Vector2dF distance;
if (dx < 0)
distance.set_x(-dx);
else if (dx > area.width())
distance.set_x(dx - area.width());
else
distance.set_x(0);
if (dy < 0)
distance.set_y(-dy);
else if (dy > area.height())
distance.set_y(dy - area.height());
else
distance.set_y(0);
return distance;
}
bool IsMutualVisible(const SnapSearchResult& a, const SnapSearchResult& b) {
return gfx::RangeF(b.snap_offset()).IsBoundedBy(a.visible_range()) &&
gfx::RangeF(a.snap_offset()).IsBoundedBy(b.visible_range());
}
void SetOrUpdateResult(const SnapSearchResult& candidate,
absl::optional<SnapSearchResult>* result,
const ElementId& active_element_id) {
if (result->has_value()) {
result->value().Union(candidate);
if (candidate.element_id() == active_element_id)
result->value().set_element_id(active_element_id);
} else {
*result = candidate;
}
}
const absl::optional<SnapSearchResult>& ClosestSearchResult(
const gfx::PointF reference_point,
SearchAxis axis,
const absl::optional<SnapSearchResult>& a,
const absl::optional<SnapSearchResult>& b) {
if (!a.has_value())
return b;
if (!b.has_value())
return a;
float reference_position =
axis == SearchAxis::kX ? reference_point.x() : reference_point.y();
float position_a = a.value().snap_offset();
float position_b = b.value().snap_offset();
DCHECK(
(reference_position <= position_a && reference_position <= position_b) ||
(reference_position >= position_a && reference_position >= position_b));
float distance_a = std::abs(position_a - reference_position);
float distance_b = std::abs(position_b - reference_position);
return distance_a < distance_b ? a : b;
}
}
SnapSearchResult::SnapSearchResult(float offset, const gfx::RangeF& range)
: snap_offset_(offset) {
set_visible_range(range);
}
void SnapSearchResult::set_visible_range(const gfx::RangeF& range) {
DCHECK(range.start() <= range.end());
visible_range_ = range;
}
void SnapSearchResult::Clip(float max_snap, float max_visible) {
snap_offset_ = std::clamp(snap_offset_, 0.0f, max_snap);
visible_range_ =
gfx::RangeF(std::clamp(visible_range_.start(), 0.0f, max_visible),
std::clamp(visible_range_.end(), 0.0f, max_visible));
}
void SnapSearchResult::Union(const SnapSearchResult& other) {
DCHECK(snap_offset_ == other.snap_offset_);
visible_range_ = gfx::RangeF(
std::min(visible_range_.start(), other.visible_range_.start()),
std::max(visible_range_.end(), other.visible_range_.end()));
}
SnapContainerData::SnapContainerData()
: proximity_range_(gfx::PointF(std::numeric_limits<float>::max(),
std::numeric_limits<float>::max())) {}
SnapContainerData::SnapContainerData(ScrollSnapType type)
: scroll_snap_type_(type),
proximity_range_(gfx::PointF(std::numeric_limits<float>::max(),
std::numeric_limits<float>::max())) {}
SnapContainerData::SnapContainerData(ScrollSnapType type,
const gfx::RectF& rect,
const gfx::PointF& max)
: scroll_snap_type_(type),
rect_(rect),
max_position_(max),
proximity_range_(gfx::PointF(std::numeric_limits<float>::max(),
std::numeric_limits<float>::max())) {}
SnapContainerData::SnapContainerData(const SnapContainerData& other) = default;
SnapContainerData::SnapContainerData(SnapContainerData&& other) = default;
SnapContainerData::~SnapContainerData() = default;
SnapContainerData& SnapContainerData::operator=(
const SnapContainerData& other) = default;
SnapContainerData& SnapContainerData::operator=(SnapContainerData&& other) =
default;
void SnapContainerData::AddSnapAreaData(SnapAreaData snap_area_data) {
snap_area_list_.push_back(snap_area_data);
}
bool SnapContainerData::FindSnapPosition(
const SnapSelectionStrategy& strategy,
gfx::PointF* snap_position,
TargetSnapAreaElementIds* target_element_ids,
const ElementId& active_element_id) const {
*target_element_ids = TargetSnapAreaElementIds();
if (scroll_snap_type_.is_none)
return false;
gfx::PointF base_position = strategy.base_position();
SnapAxis axis = scroll_snap_type_.axis;
bool should_snap_on_x = strategy.ShouldSnapOnX() &&
(axis == SnapAxis::kX || axis == SnapAxis::kBoth);
bool should_snap_on_y = strategy.ShouldSnapOnY() &&
(axis == SnapAxis::kY || axis == SnapAxis::kBoth);
if (!should_snap_on_x && !should_snap_on_y)
return false;
bool should_prioritize_x_target =
strategy.ShouldPrioritizeSnapTargets() &&
target_snap_area_element_ids_.x != ElementId();
bool should_prioritize_y_target =
strategy.ShouldPrioritizeSnapTargets() &&
target_snap_area_element_ids_.y != ElementId();
absl::optional<SnapSearchResult> selected_x, selected_y;
if (should_snap_on_x) {
if (should_prioritize_x_target) {
selected_x = GetTargetSnapAreaSearchResult(strategy, SearchAxis::kX);
}
if (!selected_x) {
SnapSearchResult initial_snap_position_y = {
std::clamp(base_position.y(), 0.f, max_position_.y()),
gfx::RangeF(0, max_position_.x())};
selected_x = FindClosestValidArea(
SearchAxis::kX, strategy, initial_snap_position_y, active_element_id);
}
}
if (should_snap_on_y) {
if (should_prioritize_y_target) {
selected_y = GetTargetSnapAreaSearchResult(strategy, SearchAxis::kY);
}
if (!selected_y) {
SnapSearchResult initial_snap_position_x = {
std::clamp(base_position.x(), 0.f, max_position_.x()),
gfx::RangeF(0, max_position_.y())};
selected_y = FindClosestValidArea(
SearchAxis::kY, strategy, initial_snap_position_x, active_element_id);
}
}
if (!selected_x.has_value() && !selected_y.has_value()) {
if (should_snap_on_x && should_snap_on_y &&
!strategy.ShouldRespectSnapStop())
return FindSnapPositionForMutualSnap(strategy, snap_position);
return false;
}
if (selected_x.has_value() && selected_y.has_value() &&
!IsMutualVisible(selected_x.value(), selected_y.value())) {
bool keep_candidate_on_x = should_prioritize_x_target;
if (should_prioritize_x_target == should_prioritize_y_target) {
keep_candidate_on_x =
std::abs(selected_x.value().snap_offset() - base_position.x()) <=
std::abs(selected_y.value().snap_offset() - base_position.y());
}
if (keep_candidate_on_x) {
selected_y = FindClosestValidArea(SearchAxis::kY, strategy,
selected_x.value(), active_element_id);
} else {
selected_x = FindClosestValidArea(SearchAxis::kX, strategy,
selected_y.value(), active_element_id);
}
}
*snap_position = strategy.current_position();
if (selected_x.has_value()) {
snap_position->set_x(selected_x.value().snap_offset());
target_element_ids->x = selected_x.value().element_id();
}
if (selected_y.has_value()) {
snap_position->set_y(selected_y.value().snap_offset());
target_element_ids->y = selected_y.value().element_id();
}
return true;
}
bool SnapContainerData::FindSnapPositionForMutualSnap(
const SnapSelectionStrategy& strategy,
gfx::PointF* snap_position) const {
DCHECK(strategy.ShouldSnapOnX() && strategy.ShouldSnapOnY());
bool found = false;
gfx::Vector2dF smallest_distance(std::numeric_limits<float>::max(),
std::numeric_limits<float>::max());
for (const SnapAreaData& area : snap_area_list_) {
if (!strategy.IsValidSnapArea(SearchAxis::kX, area))
continue;
if (!strategy.IsValidSnapArea(SearchAxis::kY, area))
continue;
SnapSearchResult x_candidate = GetSnapSearchResult(SearchAxis::kX, area);
float dx = x_candidate.snap_offset() - strategy.current_position().x();
if (std::abs(dx) > proximity_range_.x())
continue;
SnapSearchResult y_candidate = GetSnapSearchResult(SearchAxis::kY, area);
float dy = y_candidate.snap_offset() - strategy.current_position().y();
if (std::abs(dy) > proximity_range_.y())
continue;
gfx::Vector2dF distance = DistanceFromCorridor(dx, dy, rect_);
if (distance.y() < smallest_distance.y() ||
(distance.y() == smallest_distance.y() &&
distance.x() < smallest_distance.x())) {
smallest_distance = distance;
snap_position->set_x(x_candidate.snap_offset());
snap_position->set_y(y_candidate.snap_offset());
found = true;
}
}
return found;
}
absl::optional<SnapSearchResult>
SnapContainerData::GetTargetSnapAreaSearchResult(
const SnapSelectionStrategy& strategy,
SearchAxis axis) const {
ElementId target_id = axis == SearchAxis::kX
? target_snap_area_element_ids_.x
: target_snap_area_element_ids_.y;
if (target_id == ElementId())
return absl::nullopt;
for (const SnapAreaData& area : snap_area_list_) {
if (area.element_id == target_id && strategy.IsValidSnapArea(axis, area)) {
return GetSnapSearchResult(axis, area);
}
}
return absl::nullopt;
}
void SnapContainerData::UpdateSnapAreaForTesting(ElementId element_id,
SnapAreaData snap_area_data) {
for (SnapAreaData& area : snap_area_list_) {
if (area.element_id == element_id) {
area = snap_area_data;
}
}
}
const TargetSnapAreaElementIds& SnapContainerData::GetTargetSnapAreaElementIds()
const {
return target_snap_area_element_ids_;
}
bool SnapContainerData::SetTargetSnapAreaElementIds(
TargetSnapAreaElementIds ids) {
if (target_snap_area_element_ids_ == ids)
return false;
target_snap_area_element_ids_ = ids;
return true;
}
absl::optional<SnapSearchResult> SnapContainerData::FindClosestValidArea(
SearchAxis axis,
const SnapSelectionStrategy& strategy,
const SnapSearchResult& cross_axis_snap_result,
const ElementId& active_element_id) const {
absl::optional<SnapSearchResult> result = FindClosestValidAreaInternal(
axis, strategy, cross_axis_snap_result, active_element_id);
if (result.has_value() && strategy.ShouldRespectSnapStop()) {
std::unique_ptr<SnapSelectionStrategy> must_only_strategy =
SnapSelectionStrategy::CreateForDirection(
strategy.current_position(),
strategy.intended_position() - strategy.current_position(),
strategy.UsingFractionalOffsets(), SnapStopAlwaysFilter::kRequire);
absl::optional<SnapSearchResult> must_only_result =
FindClosestValidAreaInternal(axis, *must_only_strategy,
cross_axis_snap_result, active_element_id,
false);
result = ClosestSearchResult(strategy.current_position(), axis, result,
must_only_result);
}
if (result.has_value() ||
scroll_snap_type_.strictness == SnapStrictness::kProximity ||
!strategy.HasIntendedDirection())
return result;
std::unique_ptr<SnapSelectionStrategy> relaxed_strategy =
SnapSelectionStrategy::CreateForEndPosition(strategy.current_position(),
strategy.ShouldSnapOnX(),
strategy.ShouldSnapOnY());
return FindClosestValidAreaInternal(
axis, *relaxed_strategy, cross_axis_snap_result, active_element_id);
}
absl::optional<SnapSearchResult>
SnapContainerData::FindClosestValidAreaInternal(
SearchAxis axis,
const SnapSelectionStrategy& strategy,
const SnapSearchResult& cross_axis_snap_result,
const ElementId& active_element_id,
bool should_consider_covering) const {
DCHECK(cross_axis_snap_result.snap_offset() >= 0 &&
cross_axis_snap_result.snap_offset() <=
(axis == SearchAxis::kX ? max_position_.y() : max_position_.x()));
absl::optional<SnapSearchResult> closest;
absl::optional<SnapSearchResult> covering;
float prev = std::numeric_limits<float>::lowest();
float next = std::numeric_limits<float>::max();
float intended_position = axis == SearchAxis::kX
? strategy.intended_position().x()
: strategy.intended_position().y();
float base_position = axis == SearchAxis::kX ? strategy.base_position().x()
: strategy.base_position().y();
float smallest_distance =
axis == SearchAxis::kX ? proximity_range_.x() : proximity_range_.y();
for (const SnapAreaData& area : snap_area_list_) {
if (!strategy.IsValidSnapArea(axis, area))
continue;
SnapSearchResult candidate = GetSnapSearchResult(axis, area);
if (should_consider_covering &&
IsSnapportCoveredOnAxis(axis, intended_position, area.rect)) {
SnapSearchResult covering_candidate = candidate;
covering_candidate.set_snap_offset(intended_position);
if (IsMutualVisible(covering_candidate, cross_axis_snap_result))
SetOrUpdateResult(covering_candidate, &covering, active_element_id);
}
if (!IsMutualVisible(candidate, cross_axis_snap_result))
continue;
float distance = std::abs(candidate.snap_offset() - base_position);
if (strategy.IsValidSnapPosition(axis, candidate.snap_offset())) {
if (distance < smallest_distance ||
(candidate.element_id() == active_element_id &&
distance == smallest_distance)) {
smallest_distance = distance;
closest = candidate;
}
}
if (!should_consider_covering)
continue;
if (candidate.snap_offset() < intended_position &&
candidate.snap_offset() > prev) {
prev = candidate.snap_offset();
}
if (candidate.snap_offset() > intended_position &&
candidate.snap_offset() < next) {
next = candidate.snap_offset();
}
}
float size = axis == SearchAxis::kX ? rect_.width() : rect_.height();
if (prev != std::numeric_limits<float>::lowest() &&
next != std::numeric_limits<float>::max() && next - prev <= size) {
covering = absl::nullopt;
}
const absl::optional<SnapSearchResult>& picked =
strategy.PickBestResult(closest, covering);
return picked;
}
SnapSearchResult SnapContainerData::GetSnapSearchResult(
SearchAxis axis,
const SnapAreaData& area) const {
SnapSearchResult result;
if (axis == SearchAxis::kX) {
result.set_visible_range(gfx::RangeF(area.rect.y() - rect_.bottom(),
area.rect.bottom() - rect_.y()));
switch (area.scroll_snap_align.alignment_inline) {
case SnapAlignment::kStart:
result.set_snap_offset(area.rect.x() - rect_.x());
break;
case SnapAlignment::kCenter:
result.set_snap_offset(area.rect.CenterPoint().x() -
rect_.CenterPoint().x());
break;
case SnapAlignment::kEnd:
result.set_snap_offset(area.rect.right() - rect_.right());
break;
default:
NOTREACHED();
}
result.Clip(max_position_.x(), max_position_.y());
} else {
result.set_visible_range(gfx::RangeF(area.rect.x() - rect_.right(),
area.rect.right() - rect_.x()));
switch (area.scroll_snap_align.alignment_block) {
case SnapAlignment::kStart:
result.set_snap_offset(area.rect.y() - rect_.y());
break;
case SnapAlignment::kCenter:
result.set_snap_offset(area.rect.CenterPoint().y() -
rect_.CenterPoint().y());
break;
case SnapAlignment::kEnd:
result.set_snap_offset(area.rect.bottom() - rect_.bottom());
break;
default:
NOTREACHED();
}
result.Clip(max_position_.y(), max_position_.x());
}
result.set_element_id(area.element_id);
return result;
}
bool SnapContainerData::IsSnapportCoveredOnAxis(
SearchAxis axis,
float current_offset,
const gfx::RectF& area_rect) const {
if (axis == SearchAxis::kX) {
if (area_rect.width() < rect_.width())
return false;
float left = area_rect.x() - rect_.x();
float right = area_rect.right() - rect_.right();
return current_offset >= left && current_offset <= right;
} else {
if (area_rect.height() < rect_.height())
return false;
float top = area_rect.y() - rect_.y();
float bottom = area_rect.bottom() - rect_.bottom();
return current_offset >= top && current_offset <= bottom;
}
}
std::ostream& operator<<(std::ostream& ostream, const SnapAreaData& area_data) {
return ostream << area_data.rect.ToString();
}
std::ostream& operator<<(std::ostream& ostream,
const SnapContainerData& container_data) {
ostream << "container_rect: " << container_data.rect().ToString();
ostream << "area_rects: ";
for (size_t i = 0; i < container_data.size(); ++i) {
ostream << container_data.at(i) << "\n";
}
return ostream;
}
}