summaryrefslogtreecommitdiffstats
path: root/layout/generic/ScrollSnap.cpp
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--layout/generic/ScrollSnap.cpp788
1 files changed, 788 insertions, 0 deletions
diff --git a/layout/generic/ScrollSnap.cpp b/layout/generic/ScrollSnap.cpp
new file mode 100644
index 0000000000..cfe9fb1cbb
--- /dev/null
+++ b/layout/generic/ScrollSnap.cpp
@@ -0,0 +1,788 @@
+/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
+/* vim: set ts=8 sts=2 et sw=2 tw=80: */
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+#include "ScrollSnap.h"
+
+#include "FrameMetrics.h"
+
+#include "mozilla/ScrollSnapInfo.h"
+#include "mozilla/ServoStyleConsts.h"
+#include "nsIFrame.h"
+#include "nsIScrollableFrame.h"
+#include "nsLayoutUtils.h"
+#include "nsPresContext.h"
+#include "nsTArray.h"
+#include "mozilla/StaticPrefs_layout.h"
+
+namespace mozilla {
+
+/**
+ * Keeps track of the current best edge to snap to. The criteria for
+ * adding an edge depends on the scrolling unit.
+ */
+class CalcSnapPoints final {
+ using SnapTarget = ScrollSnapInfo::SnapTarget;
+
+ public:
+ CalcSnapPoints(ScrollUnit aUnit, ScrollSnapFlags aSnapFlags,
+ const nsPoint& aDestination, const nsPoint& aStartPos);
+ struct SnapPosition : public SnapTarget {
+ SnapPosition(const SnapTarget& aSnapTarget, nscoord aPosition,
+ nscoord aDistanceOnOtherAxis)
+ : SnapTarget(aSnapTarget),
+ mPosition(aPosition),
+ mDistanceOnOtherAxis(aDistanceOnOtherAxis) {}
+
+ nscoord mPosition;
+ // The distance from the scroll destination to this snap position on the
+ // other axis. This value is used if there are multiple SnapPositions on
+ // this axis, but the positions on the other axis are different.
+ nscoord mDistanceOnOtherAxis;
+ };
+
+ void AddHorizontalEdge(const SnapTarget& aTarget);
+ void AddVerticalEdge(const SnapTarget& aTarget);
+
+ struct CandidateTracker {
+ // keeps track of the position of the current second best edge on the
+ // opposite side of the best edge on this axis.
+ // We use NSCoordSaturatingSubtract to calculate the distance between a
+ // given position and this second best edge position so that it can be an
+ // uninitialized value as the maximum possible value, because the first
+ // distance calculation would always be nscoord_MAX.
+ nscoord mSecondBestEdge = nscoord_MAX;
+
+ // Assuming in most cases there's no multiple coincide snap points.
+ AutoTArray<ScrollSnapTargetId, 1> mTargetIds;
+ // keeps track of the positions of the current best edge on this axis.
+ // NOTE: Each SnapPosition.mPosition points the same snap position on this
+ // axis but other member variables of SnapPosition may have different
+ // values.
+ AutoTArray<SnapPosition, 1> mBestEdges;
+ bool EdgeFound() const { return !mBestEdges.IsEmpty(); }
+ };
+ void AddEdge(const SnapPosition& aEdge, nscoord aDestination,
+ nscoord aStartPos, nscoord aScrollingDirection,
+ CandidateTracker* aCandidateTracker);
+ SnapDestination GetBestEdge(const nsSize& aSnapportSize) const;
+ nscoord XDistanceBetweenBestAndSecondEdge() const {
+ return std::abs(NSCoordSaturatingSubtract(
+ mTrackerOnX.mSecondBestEdge,
+ mTrackerOnX.EdgeFound() ? mTrackerOnX.mBestEdges[0].mPosition
+ : mDestination.x,
+ nscoord_MAX));
+ }
+ nscoord YDistanceBetweenBestAndSecondEdge() const {
+ return std::abs(NSCoordSaturatingSubtract(
+ mTrackerOnY.mSecondBestEdge,
+ mTrackerOnY.EdgeFound() ? mTrackerOnY.mBestEdges[0].mPosition
+ : mDestination.y,
+ nscoord_MAX));
+ }
+ const nsPoint& Destination() const { return mDestination; }
+
+ protected:
+ ScrollUnit mUnit;
+ ScrollSnapFlags mSnapFlags;
+ nsPoint mDestination; // gives the position after scrolling but before
+ // snapping
+ nsPoint mStartPos; // gives the position before scrolling
+ nsIntPoint mScrollingDirection; // always -1, 0, or 1
+ CandidateTracker mTrackerOnX;
+ CandidateTracker mTrackerOnY;
+};
+
+CalcSnapPoints::CalcSnapPoints(ScrollUnit aUnit, ScrollSnapFlags aSnapFlags,
+ const nsPoint& aDestination,
+ const nsPoint& aStartPos)
+ : mUnit(aUnit),
+ mSnapFlags(aSnapFlags),
+ mDestination(aDestination),
+ mStartPos(aStartPos) {
+ MOZ_ASSERT(aSnapFlags != ScrollSnapFlags::Disabled);
+
+ nsPoint direction = aDestination - aStartPos;
+ mScrollingDirection = nsIntPoint(0, 0);
+ if (direction.x < 0) {
+ mScrollingDirection.x = -1;
+ }
+ if (direction.x > 0) {
+ mScrollingDirection.x = 1;
+ }
+ if (direction.y < 0) {
+ mScrollingDirection.y = -1;
+ }
+ if (direction.y > 0) {
+ mScrollingDirection.y = 1;
+ }
+}
+
+SnapDestination CalcSnapPoints::GetBestEdge(const nsSize& aSnapportSize) const {
+ if (mTrackerOnX.EdgeFound() && mTrackerOnY.EdgeFound()) {
+ nsPoint bestCandidate(mTrackerOnX.mBestEdges[0].mPosition,
+ mTrackerOnY.mBestEdges[0].mPosition);
+ nsRect snappedPort = nsRect(bestCandidate, aSnapportSize);
+
+ // If we've found the candidates on both axes, it's possible some of
+ // candidates will be outside of the snapport if we snap to the point
+ // (mTrackerOnX.mBestEdges[0].mPosition,
+ // mTrackerOnY.mBestEdges[0].mPosition). So we need to get the intersection
+ // of the snap area of each snap target element on each axis and the
+ // snapport to tell whether it's outside of the snapport or not.
+ //
+ // Also if at least either one of the elements will be outside of the
+ // snapport if we snap to (mTrackerOnX.mBestEdges[0].mPosition,
+ // mTrackerOnY.mBestEdges[0].mPosition). We need to choose one of
+ // combinations of the candidates which is closest to the destination.
+ //
+ // So here we iterate over mTrackerOnX and mTrackerOnY just once
+ // respectively for both purposes to avoid iterating over them again and
+ // again.
+ //
+ // NOTE: Ideally we have to iterate over every possible combinations of
+ // (mTrackerOnX.mBestEdges[i].mSnapPoint.mY,
+ // mTrackerOnY.mBestEdges[j].mSnapPoint.mX) and tell whether the given
+ // combination will be visible in the snapport or not (maybe we should
+ // choose the one that the visible area, i.e., the intersection area of
+ // the snap target elements and the snapport, is the largest one rather than
+ // the closest one?). But it will be inefficient, so here we will not
+ // iterate all the combinations, we just iterate all the snap target
+ // elements in each axis respectively.
+
+ AutoTArray<ScrollSnapTargetId, 1> visibleTargetIdsOnX;
+ nscoord minimumDistanceOnY = nscoord_MAX;
+ size_t minimumXIndex = 0;
+ AutoTArray<ScrollSnapTargetId, 1> minimumDistanceTargetIdsOnX;
+ for (size_t i = 0; i < mTrackerOnX.mBestEdges.Length(); i++) {
+ const auto& targetX = mTrackerOnX.mBestEdges[i];
+ if (targetX.mSnapArea.Intersects(snappedPort)) {
+ visibleTargetIdsOnX.AppendElement(targetX.mTargetId);
+ }
+
+ if (targetX.mDistanceOnOtherAxis < minimumDistanceOnY) {
+ minimumDistanceOnY = targetX.mDistanceOnOtherAxis;
+ minimumXIndex = i;
+ minimumDistanceTargetIdsOnX =
+ AutoTArray<ScrollSnapTargetId, 1>{targetX.mTargetId};
+ } else if (minimumDistanceOnY != nscoord_MAX &&
+ targetX.mDistanceOnOtherAxis == minimumDistanceOnY) {
+ minimumDistanceTargetIdsOnX.AppendElement(targetX.mTargetId);
+ }
+ }
+
+ AutoTArray<ScrollSnapTargetId, 1> visibleTargetIdsOnY;
+ nscoord minimumDistanceOnX = nscoord_MAX;
+ size_t minimumYIndex = 0;
+ AutoTArray<ScrollSnapTargetId, 1> minimumDistanceTargetIdsOnY;
+ for (size_t i = 0; i < mTrackerOnY.mBestEdges.Length(); i++) {
+ const auto& targetY = mTrackerOnY.mBestEdges[i];
+ if (targetY.mSnapArea.Intersects(snappedPort)) {
+ visibleTargetIdsOnY.AppendElement(targetY.mTargetId);
+ }
+
+ if (targetY.mDistanceOnOtherAxis < minimumDistanceOnX) {
+ minimumDistanceOnX = targetY.mDistanceOnOtherAxis;
+ minimumYIndex = i;
+ minimumDistanceTargetIdsOnY =
+ AutoTArray<ScrollSnapTargetId, 1>{targetY.mTargetId};
+ } else if (minimumDistanceOnX != nscoord_MAX &&
+ targetY.mDistanceOnOtherAxis == minimumDistanceOnX) {
+ minimumDistanceTargetIdsOnY.AppendElement(targetY.mTargetId);
+ }
+ }
+
+ // If we have the target ids on both axes, it means the target elements
+ // (ids) specifying the best edge on X axis and the target elements
+ // specifying the best edge on Y axis are visible if we snap to the best
+ // edge. Thus they are valid snap positions.
+ if (!visibleTargetIdsOnX.IsEmpty() && !visibleTargetIdsOnY.IsEmpty()) {
+ return SnapDestination{
+ bestCandidate,
+ ScrollSnapTargetIds{visibleTargetIdsOnX, visibleTargetIdsOnY}};
+ }
+
+ // Now we've already known that snapping to
+ // (mTrackerOnX.mBestEdges[0].mPosition,
+ // mTrackerOnY.mBestEdges[0].mPosition) will make all candidates of
+ // mTrackerX or mTrackerY (or both) outside of the snapport. We need to
+ // choose another combination where candidates of both mTrackerX/Y are
+ // inside the snapport.
+
+ // There are three possibilities;
+ // 1) There's no candidate on X axis in mTrackerOnY (that means
+ // each candidate's scroll-snap-align is `none` on X axis), but there's
+ // any candidate in mTrackerOnX, the closest candidates of mTrackerOnX
+ // should be used.
+ // 2) There's no candidate on Y axis in mTrackerOnX (that means
+ // each candidate's scroll-snap-align is `none` on Y axis), but there's
+ // any candidate in mTrackerOnY, the closest candidates of mTrackerOnY
+ // should be used.
+ // 3) There are candidates on both axes. Choosing a combination such as
+ // (mTrackerOnX.mBestEdges[i].mSnapPoint.mX,
+ // mTrackerOnY.mBestEdges[i].mSnapPoint.mY)
+ // would require us to iterate over the candidates again if the
+ // combination position is outside the snapport, which we don't want to
+ // do. Instead, we choose either one of the axis' candidates.
+ if ((minimumDistanceOnX == nscoord_MAX) &&
+ minimumDistanceOnY != nscoord_MAX) {
+ bestCandidate.y = *mTrackerOnX.mBestEdges[minimumXIndex].mSnapPoint.mY;
+ return SnapDestination{bestCandidate,
+ ScrollSnapTargetIds{minimumDistanceTargetIdsOnX,
+ minimumDistanceTargetIdsOnX}};
+ }
+
+ if (minimumDistanceOnX != nscoord_MAX &&
+ minimumDistanceOnY == nscoord_MAX) {
+ bestCandidate.x = *mTrackerOnY.mBestEdges[minimumYIndex].mSnapPoint.mX;
+ return SnapDestination{bestCandidate,
+ ScrollSnapTargetIds{minimumDistanceTargetIdsOnY,
+ minimumDistanceTargetIdsOnY}};
+ }
+
+ if (minimumDistanceOnX != nscoord_MAX &&
+ minimumDistanceOnY != nscoord_MAX) {
+ // If we've found candidates on both axes, choose the closest point either
+ // on X axis or Y axis from the scroll destination. I.e. choose
+ // `minimumXIndex` one or `minimumYIndex` one to make at least one of
+ // snap target elements visible inside the snapport.
+ //
+ // For example,
+ // [bestCandidate.x, mTrackerOnX.mBestEdges[minimumXIndex].mSnapPoint.mY]
+ // is a candidate generated from a single element, thus snapping to the
+ // point would definitely make the element visible inside the snapport.
+ if (hypotf(NSCoordToFloat(mDestination.x -
+ mTrackerOnX.mBestEdges[0].mPosition),
+ NSCoordToFloat(minimumDistanceOnY)) <
+ hypotf(NSCoordToFloat(minimumDistanceOnX),
+ NSCoordToFloat(mDestination.y -
+ mTrackerOnY.mBestEdges[0].mPosition))) {
+ bestCandidate.y = *mTrackerOnX.mBestEdges[minimumXIndex].mSnapPoint.mY;
+ } else {
+ bestCandidate.x = *mTrackerOnY.mBestEdges[minimumYIndex].mSnapPoint.mX;
+ }
+ return SnapDestination{bestCandidate,
+ ScrollSnapTargetIds{minimumDistanceTargetIdsOnX,
+ minimumDistanceTargetIdsOnY}};
+ }
+ MOZ_ASSERT_UNREACHABLE("There's at least one candidate on either axis");
+ // `minimumDistanceOnX == nscoord_MAX && minimumDistanceOnY == nscoord_MAX`
+ // should not happen but we fall back for safety.
+ }
+
+ return SnapDestination{
+ nsPoint(
+ mTrackerOnX.EdgeFound() ? mTrackerOnX.mBestEdges[0].mPosition
+ // In the case of IntendedEndPosition (i.e. the destination point is
+ // explicitely specied, e.g. scrollTo) use the destination point if we
+ // didn't find any candidates.
+ : !(mSnapFlags & ScrollSnapFlags::IntendedDirection) ? mDestination.x
+ : mStartPos.x,
+ mTrackerOnY.EdgeFound() ? mTrackerOnY.mBestEdges[0].mPosition
+ // Same as above X axis case, use the destination point if we didn't
+ // find any candidates.
+ : !(mSnapFlags & ScrollSnapFlags::IntendedDirection) ? mDestination.y
+ : mStartPos.y),
+ ScrollSnapTargetIds{mTrackerOnX.mTargetIds, mTrackerOnY.mTargetIds}};
+}
+
+void CalcSnapPoints::AddHorizontalEdge(const SnapTarget& aTarget) {
+ MOZ_ASSERT(aTarget.mSnapPoint.mY);
+ AddEdge(SnapPosition{aTarget, *aTarget.mSnapPoint.mY,
+ aTarget.mSnapPoint.mX
+ ? std::abs(mDestination.x - *aTarget.mSnapPoint.mX)
+ : nscoord_MAX},
+ mDestination.y, mStartPos.y, mScrollingDirection.y, &mTrackerOnY);
+}
+
+void CalcSnapPoints::AddVerticalEdge(const SnapTarget& aTarget) {
+ MOZ_ASSERT(aTarget.mSnapPoint.mX);
+ AddEdge(SnapPosition{aTarget, *aTarget.mSnapPoint.mX,
+ aTarget.mSnapPoint.mY
+ ? std::abs(mDestination.y - *aTarget.mSnapPoint.mY)
+ : nscoord_MAX},
+ mDestination.x, mStartPos.x, mScrollingDirection.x, &mTrackerOnX);
+}
+
+void CalcSnapPoints::AddEdge(const SnapPosition& aEdge, nscoord aDestination,
+ nscoord aStartPos, nscoord aScrollingDirection,
+ CandidateTracker* aCandidateTracker) {
+ if (mSnapFlags & ScrollSnapFlags::IntendedDirection) {
+ // In the case of intended direction, we only want to snap to points ahead
+ // of the direction we are scrolling.
+ if (aScrollingDirection == 0 ||
+ (aEdge.mPosition - aStartPos) * aScrollingDirection <= 0) {
+ // The scroll direction is neutral - will not hit a snap point, or the
+ // edge is not in the direction we are scrolling, skip it.
+ return;
+ }
+ }
+
+ if (!aCandidateTracker->EdgeFound()) {
+ aCandidateTracker->mBestEdges = AutoTArray<SnapPosition, 1>{aEdge};
+ aCandidateTracker->mTargetIds =
+ AutoTArray<ScrollSnapTargetId, 1>{aEdge.mTargetId};
+ return;
+ }
+
+ auto isPreferredStopAlways = [&](const SnapPosition& aSnapPosition) -> bool {
+ MOZ_ASSERT(mSnapFlags & ScrollSnapFlags::IntendedDirection);
+ // In the case of intended direction scroll operations, `scroll-snap-stop:
+ // always` snap points in between the start point and the scroll destination
+ // are preferable preferable. In other words any `scroll-snap-stop: always`
+ // snap points can be handled as if it's `scroll-snap-stop: normal`.
+ return aSnapPosition.mScrollSnapStop == StyleScrollSnapStop::Always &&
+ std::abs(aSnapPosition.mPosition - aStartPos) <
+ std::abs(aDestination - aStartPos);
+ };
+
+ const bool isOnOppositeSide =
+ ((aEdge.mPosition - aDestination) > 0) !=
+ ((aCandidateTracker->mBestEdges[0].mPosition - aDestination) > 0);
+ const nscoord distanceFromStart = aEdge.mPosition - aStartPos;
+ // A utility function to update the best and the second best edges in the
+ // given conditions.
+ // |aIsCloserThanBest| True if the current candidate is closer than the best
+ // edge.
+ // |aIsCloserThanSecond| True if the current candidate is closer than
+ // the second best edge.
+ const nscoord distanceFromDestination = aEdge.mPosition - aDestination;
+ auto updateBestEdges = [&](bool aIsCloserThanBest, bool aIsCloserThanSecond) {
+ if (aIsCloserThanBest) {
+ if (mSnapFlags & ScrollSnapFlags::IntendedDirection &&
+ isPreferredStopAlways(aEdge)) {
+ // In the case of intended direction scroll operations and the new best
+ // candidate is `scroll-snap-stop: always` and if it's closer to the
+ // start position than the destination, thus we won't use the second
+ // best edge since even if the snap port of the best edge covers entire
+ // snapport, the `scroll-snap-stop: always` snap point is preferred than
+ // any points.
+ // NOTE: We've already ignored snap points behind start points so that
+ // we can use std::abs here in the comparison.
+ //
+ // For example, if there's a `scroll-snap-stop: always` in between the
+ // start point and destination, no `snap-overflow` mechanism should
+ // happen, if there's `scroll-snap-stop: always` further than the
+ // destination, `snap-overflow` might happen something like below
+ // diagram.
+ // start always dest other always
+ // |------------|---------|------|
+ aCandidateTracker->mSecondBestEdge = aEdge.mPosition;
+ } else if (isOnOppositeSide) {
+ // Replace the second best edge with the current best edge only if the
+ // new best edge (aEdge) is on the opposite side of the current best
+ // edge.
+ aCandidateTracker->mSecondBestEdge =
+ aCandidateTracker->mBestEdges[0].mPosition;
+ }
+ aCandidateTracker->mBestEdges = AutoTArray<SnapPosition, 1>{aEdge};
+ aCandidateTracker->mTargetIds =
+ AutoTArray<ScrollSnapTargetId, 1>{aEdge.mTargetId};
+ } else {
+ if (aEdge.mPosition == aCandidateTracker->mBestEdges[0].mPosition) {
+ aCandidateTracker->mTargetIds.AppendElement(aEdge.mTargetId);
+ aCandidateTracker->mBestEdges.AppendElement(aEdge);
+ }
+ if (aIsCloserThanSecond && isOnOppositeSide) {
+ aCandidateTracker->mSecondBestEdge = aEdge.mPosition;
+ }
+ }
+ };
+
+ bool isCandidateOfBest = false;
+ bool isCandidateOfSecondBest = false;
+ switch (mUnit) {
+ case ScrollUnit::DEVICE_PIXELS:
+ case ScrollUnit::LINES:
+ case ScrollUnit::WHOLE: {
+ isCandidateOfBest =
+ std::abs(distanceFromDestination) <
+ std::abs(aCandidateTracker->mBestEdges[0].mPosition - aDestination);
+ isCandidateOfSecondBest =
+ std::abs(distanceFromDestination) <
+ std::abs(NSCoordSaturatingSubtract(aCandidateTracker->mSecondBestEdge,
+ aDestination, nscoord_MAX));
+ break;
+ }
+ case ScrollUnit::PAGES: {
+ // distance to the edge from the scrolling destination in the direction of
+ // scrolling
+ nscoord overshoot = distanceFromDestination * aScrollingDirection;
+ // distance to the current best edge from the scrolling destination in the
+ // direction of scrolling
+ nscoord curOvershoot =
+ (aCandidateTracker->mBestEdges[0].mPosition - aDestination) *
+ aScrollingDirection;
+
+ nscoord secondOvershoot =
+ NSCoordSaturatingSubtract(aCandidateTracker->mSecondBestEdge,
+ aDestination, nscoord_MAX) *
+ aScrollingDirection;
+
+ // edges between the current position and the scrolling destination are
+ // favoured to preserve context
+ if (overshoot < 0) {
+ isCandidateOfBest = overshoot > curOvershoot || curOvershoot >= 0;
+ isCandidateOfSecondBest =
+ overshoot > secondOvershoot || secondOvershoot >= 0;
+ }
+ // if there are no edges between the current position and the scrolling
+ // destination the closest edge beyond the destination is used
+ if (overshoot > 0) {
+ isCandidateOfBest = overshoot < curOvershoot;
+ isCandidateOfSecondBest = overshoot < secondOvershoot;
+ }
+ }
+ }
+
+ if (mSnapFlags & ScrollSnapFlags::IntendedDirection) {
+ if (isPreferredStopAlways(aEdge)) {
+ // If the given position is `scroll-snap-stop: always` and if the position
+ // is in between the start and the destination positions, update the best
+ // position based on the distance from the __start__ point.
+ isCandidateOfBest =
+ std::abs(distanceFromStart) <
+ std::abs(aCandidateTracker->mBestEdges[0].mPosition - aStartPos);
+ } else if (isPreferredStopAlways(aCandidateTracker->mBestEdges[0])) {
+ // If we've found a preferable `scroll-snap-stop:always` position as the
+ // best, do not update it unless the given position is also
+ // `scroll-snap-stop: always`.
+ isCandidateOfBest = false;
+ }
+ }
+
+ updateBestEdges(isCandidateOfBest, isCandidateOfSecondBest);
+}
+
+static void ProcessSnapPositions(CalcSnapPoints& aCalcSnapPoints,
+ const ScrollSnapInfo& aSnapInfo) {
+ aSnapInfo.ForEachValidTargetFor(
+ aCalcSnapPoints.Destination(), [&](const auto& aTarget) -> bool {
+ if (aTarget.mSnapPoint.mX && aSnapInfo.mScrollSnapStrictnessX !=
+ StyleScrollSnapStrictness::None) {
+ aCalcSnapPoints.AddVerticalEdge(aTarget);
+ }
+ if (aTarget.mSnapPoint.mY && aSnapInfo.mScrollSnapStrictnessY !=
+ StyleScrollSnapStrictness::None) {
+ aCalcSnapPoints.AddHorizontalEdge(aTarget);
+ }
+ return true;
+ });
+}
+
+Maybe<SnapDestination> ScrollSnapUtils::GetSnapPointForDestination(
+ const ScrollSnapInfo& aSnapInfo, ScrollUnit aUnit,
+ ScrollSnapFlags aSnapFlags, const nsRect& aScrollRange,
+ const nsPoint& aStartPos, const nsPoint& aDestination) {
+ if (aSnapInfo.mScrollSnapStrictnessY == StyleScrollSnapStrictness::None &&
+ aSnapInfo.mScrollSnapStrictnessX == StyleScrollSnapStrictness::None) {
+ return Nothing();
+ }
+
+ if (!aSnapInfo.HasSnapPositions()) {
+ return Nothing();
+ }
+
+ CalcSnapPoints calcSnapPoints(aUnit, aSnapFlags, aDestination, aStartPos);
+
+ ProcessSnapPositions(calcSnapPoints, aSnapInfo);
+
+ // If the distance between the first and the second candidate snap points
+ // is larger than the snapport size and the snapport is covered by larger
+ // elements, any points inside the covering area should be valid snap
+ // points.
+ // https://drafts.csswg.org/css-scroll-snap-1/#snap-overflow
+ // NOTE: |aDestination| sometimes points outside of the scroll range, e.g.
+ // by the APZC fling, so for the overflow checks we need to clamp it.
+ nsPoint clampedDestination = aScrollRange.ClampPoint(aDestination);
+ for (auto range : aSnapInfo.mXRangeWiderThanSnapport) {
+ if (range.IsValid(clampedDestination.x, aSnapInfo.mSnapportSize.width) &&
+ calcSnapPoints.XDistanceBetweenBestAndSecondEdge() >
+ aSnapInfo.mSnapportSize.width) {
+ calcSnapPoints.AddVerticalEdge(ScrollSnapInfo::SnapTarget{
+ Some(clampedDestination.x), Nothing(), range.mSnapArea,
+ StyleScrollSnapStop::Normal, range.mTargetId});
+ break;
+ }
+ }
+ for (auto range : aSnapInfo.mYRangeWiderThanSnapport) {
+ if (range.IsValid(clampedDestination.y, aSnapInfo.mSnapportSize.height) &&
+ calcSnapPoints.YDistanceBetweenBestAndSecondEdge() >
+ aSnapInfo.mSnapportSize.height) {
+ calcSnapPoints.AddHorizontalEdge(ScrollSnapInfo::SnapTarget{
+ Nothing(), Some(clampedDestination.y), range.mSnapArea,
+ StyleScrollSnapStop::Normal, range.mTargetId});
+ break;
+ }
+ }
+
+ bool snapped = false;
+ auto finalPos = calcSnapPoints.GetBestEdge(aSnapInfo.mSnapportSize);
+ constexpr float proximityRatio = 0.3;
+ if (aSnapInfo.mScrollSnapStrictnessY ==
+ StyleScrollSnapStrictness::Proximity &&
+ std::abs(aDestination.y - finalPos.mPosition.y) >
+ aSnapInfo.mSnapportSize.height * proximityRatio) {
+ finalPos.mPosition.y = aDestination.y;
+ } else if (aSnapInfo.mScrollSnapStrictnessY !=
+ StyleScrollSnapStrictness::None &&
+ aDestination.y != finalPos.mPosition.y) {
+ snapped = true;
+ }
+ if (aSnapInfo.mScrollSnapStrictnessX ==
+ StyleScrollSnapStrictness::Proximity &&
+ std::abs(aDestination.x - finalPos.mPosition.x) >
+ aSnapInfo.mSnapportSize.width * proximityRatio) {
+ finalPos.mPosition.x = aDestination.x;
+ } else if (aSnapInfo.mScrollSnapStrictnessX !=
+ StyleScrollSnapStrictness::None &&
+ aDestination.x != finalPos.mPosition.x) {
+ snapped = true;
+ }
+ return snapped ? Some(finalPos) : Nothing();
+}
+
+ScrollSnapTargetId ScrollSnapUtils::GetTargetIdFor(const nsIFrame* aFrame) {
+ MOZ_ASSERT(aFrame && aFrame->GetContent());
+ return ScrollSnapTargetId{reinterpret_cast<uintptr_t>(aFrame->GetContent())};
+}
+
+static std::pair<Maybe<nscoord>, Maybe<nscoord>> GetCandidateInLastTargets(
+ const ScrollSnapInfo& aSnapInfo, const nsPoint& aCurrentPosition,
+ const UniquePtr<ScrollSnapTargetIds>& aLastSnapTargetIds,
+ const nsIContent* aFocusedContent) {
+ ScrollSnapTargetId targetIdForFocusedContent = ScrollSnapTargetId::None;
+ if (aFocusedContent && aFocusedContent->GetPrimaryFrame()) {
+ targetIdForFocusedContent =
+ ScrollSnapUtils::GetTargetIdFor(aFocusedContent->GetPrimaryFrame());
+ }
+
+ // Note: Below algorithm doesn't care about cases where the last snap point
+ // was on an element larger than the snapport since it's not clear to us
+ // what we should do for now.
+ // https://github.com/w3c/csswg-drafts/issues/7438
+ const ScrollSnapInfo::SnapTarget* focusedTarget = nullptr;
+ Maybe<nscoord> x, y;
+ aSnapInfo.ForEachValidTargetFor(
+ aCurrentPosition, [&](const auto& aTarget) -> bool {
+ if (aTarget.mSnapPoint.mX && aSnapInfo.mScrollSnapStrictnessX !=
+ StyleScrollSnapStrictness::None) {
+ if (aLastSnapTargetIds->mIdsOnX.Contains(aTarget.mTargetId)) {
+ if (targetIdForFocusedContent == aTarget.mTargetId) {
+ // If we've already found the candidate on Y axis, but if snapping
+ // to the point results this target is scrolled out, we can't use
+ // it.
+ if ((y && !aTarget.mSnapArea.Intersects(
+ nsRect(nsPoint(*aTarget.mSnapPoint.mX, *y),
+ aSnapInfo.mSnapportSize)))) {
+ y.reset();
+ }
+
+ focusedTarget = &aTarget;
+ // If the focused one is valid, then it's the candidate.
+ x = aTarget.mSnapPoint.mX;
+ }
+
+ if (!x) {
+ // Update the candidate on X axis only if
+ // 1) we haven't yet found the candidate on Y axis
+ // 2) or if we've found the candiate on Y axis and if snapping to
+ // the
+ // candidate position result the target element is visible
+ // inside the snapport.
+ if (!y || (y && aTarget.mSnapArea.Intersects(
+ nsRect(nsPoint(*aTarget.mSnapPoint.mX, *y),
+ aSnapInfo.mSnapportSize)))) {
+ x = aTarget.mSnapPoint.mX;
+ }
+ }
+ }
+ }
+ if (aTarget.mSnapPoint.mY && aSnapInfo.mScrollSnapStrictnessY !=
+ StyleScrollSnapStrictness::None) {
+ if (aLastSnapTargetIds->mIdsOnY.Contains(aTarget.mTargetId)) {
+ if (targetIdForFocusedContent == aTarget.mTargetId) {
+ NS_ASSERTION(
+ !focusedTarget || focusedTarget == &aTarget,
+ "If the focused target has been found on X axis, the "
+ "target should be same");
+ // If we've already found the candidate on X axis other than the
+ // focused one, but if snapping to the point results this target
+ // is scrolled out, we can't use it.
+ if (!focusedTarget &&
+ (x && !aTarget.mSnapArea.Intersects(
+ nsRect(nsPoint(*x, *aTarget.mSnapPoint.mY),
+ aSnapInfo.mSnapportSize)))) {
+ x.reset();
+ }
+
+ focusedTarget = &aTarget;
+ y = aTarget.mSnapPoint.mY;
+ }
+
+ if (!y) {
+ if (!x || (x && aTarget.mSnapArea.Intersects(
+ nsRect(nsPoint(*x, *aTarget.mSnapPoint.mY),
+ aSnapInfo.mSnapportSize)))) {
+ y = aTarget.mSnapPoint.mY;
+ }
+ }
+ }
+ }
+
+ // If we found candidates on both axes, it's the one we need.
+ if (x && y &&
+ // If we haven't found the focused target, it's possible that we
+ // haven't iterated it, don't break in such case.
+ (targetIdForFocusedContent == ScrollSnapTargetId::None ||
+ focusedTarget)) {
+ return false;
+ }
+ return true;
+ });
+
+ return {x, y};
+}
+
+Maybe<SnapDestination> ScrollSnapUtils::GetSnapPointForResnap(
+ const ScrollSnapInfo& aSnapInfo, const nsRect& aScrollRange,
+ const nsPoint& aCurrentPosition,
+ const UniquePtr<ScrollSnapTargetIds>& aLastSnapTargetIds,
+ const nsIContent* aFocusedContent) {
+ if (!aLastSnapTargetIds) {
+ return GetSnapPointForDestination(aSnapInfo, ScrollUnit::DEVICE_PIXELS,
+ ScrollSnapFlags::IntendedEndPosition,
+ aScrollRange, aCurrentPosition,
+ aCurrentPosition);
+ }
+
+ auto [x, y] = GetCandidateInLastTargets(aSnapInfo, aCurrentPosition,
+ aLastSnapTargetIds, aFocusedContent);
+ if (!x && !y) {
+ // In the worst case there's no longer valid snap points previously snapped,
+ // try to find new valid snap points.
+ return GetSnapPointForDestination(aSnapInfo, ScrollUnit::DEVICE_PIXELS,
+ ScrollSnapFlags::IntendedEndPosition,
+ aScrollRange, aCurrentPosition,
+ aCurrentPosition);
+ }
+
+ // If there's no candidate on one of the axes in the last snap points, try
+ // to find a new candidate.
+ if (!x || !y) {
+ nsPoint newPosition =
+ nsPoint(x ? *x : aCurrentPosition.x, y ? *y : aCurrentPosition.y);
+ CalcSnapPoints calcSnapPoints(ScrollUnit::DEVICE_PIXELS,
+ ScrollSnapFlags::IntendedEndPosition,
+ newPosition, newPosition);
+
+ aSnapInfo.ForEachValidTargetFor(
+ newPosition, [&, &x = x, &y = y](const auto& aTarget) -> bool {
+ if (!x && aTarget.mSnapPoint.mX &&
+ aSnapInfo.mScrollSnapStrictnessX !=
+ StyleScrollSnapStrictness::None) {
+ calcSnapPoints.AddVerticalEdge(aTarget);
+ }
+ if (!y && aTarget.mSnapPoint.mY &&
+ aSnapInfo.mScrollSnapStrictnessY !=
+ StyleScrollSnapStrictness::None) {
+ calcSnapPoints.AddHorizontalEdge(aTarget);
+ }
+ return true;
+ });
+
+ auto finalPos = calcSnapPoints.GetBestEdge(aSnapInfo.mSnapportSize);
+ if (!x) {
+ x = Some(finalPos.mPosition.x);
+ }
+ if (!y) {
+ y = Some(finalPos.mPosition.y);
+ }
+ }
+
+ SnapDestination snapTarget{nsPoint(*x, *y)};
+ // Collect snap points where the position is still same as the new snap
+ // position.
+ aSnapInfo.ForEachValidTargetFor(
+ snapTarget.mPosition, [&, &x = x, &y = y](const auto& aTarget) -> bool {
+ if (aTarget.mSnapPoint.mX &&
+ aSnapInfo.mScrollSnapStrictnessX !=
+ StyleScrollSnapStrictness::None &&
+ aTarget.mSnapPoint.mX == x) {
+ snapTarget.mTargetIds.mIdsOnX.AppendElement(aTarget.mTargetId);
+ }
+
+ if (aTarget.mSnapPoint.mY &&
+ aSnapInfo.mScrollSnapStrictnessY !=
+ StyleScrollSnapStrictness::None &&
+ aTarget.mSnapPoint.mY == y) {
+ snapTarget.mTargetIds.mIdsOnY.AppendElement(aTarget.mTargetId);
+ }
+ return true;
+ });
+ return Some(snapTarget);
+}
+
+void ScrollSnapUtils::PostPendingResnapIfNeededFor(nsIFrame* aFrame) {
+ ScrollSnapTargetId id = GetTargetIdFor(aFrame);
+ if (id == ScrollSnapTargetId::None) {
+ return;
+ }
+
+ if (nsIScrollableFrame* sf = nsLayoutUtils::GetNearestScrollableFrame(
+ aFrame, nsLayoutUtils::SCROLLABLE_SAME_DOC |
+ nsLayoutUtils::SCROLLABLE_INCLUDE_HIDDEN)) {
+ sf->PostPendingResnapIfNeeded(aFrame);
+ }
+}
+
+void ScrollSnapUtils::PostPendingResnapFor(nsIFrame* aFrame) {
+ if (nsIScrollableFrame* sf = nsLayoutUtils::GetNearestScrollableFrame(
+ aFrame, nsLayoutUtils::SCROLLABLE_SAME_DOC |
+ nsLayoutUtils::SCROLLABLE_INCLUDE_HIDDEN)) {
+ sf->PostPendingResnap();
+ }
+}
+
+bool ScrollSnapUtils::NeedsToRespectTargetWritingMode(
+ const nsSize& aSnapAreaSize, const nsSize& aSnapportSize) {
+ // Use the writing-mode on the target element if the snap area is larger than
+ // the snapport.
+ // https://drafts.csswg.org/css-scroll-snap/#snap-scope
+ //
+ // It's unclear `larger` means that the size is larger than only on the target
+ // axis. If it doesn't, it will pick the same axis in the case where only one
+ // axis is larger. For example, if an element size is (200 x 10) and the
+ // snapport size is (100 x 100) and if the element's writing mode is different
+ // from the scroller's writing mode, then `scroll-snap-align: start start`
+ // will be conflict.
+ return aSnapAreaSize.width > aSnapportSize.width ||
+ aSnapAreaSize.height > aSnapportSize.height;
+}
+
+static nsRect InflateByScrollMargin(const nsRect& aTargetRect,
+ const nsMargin& aScrollMargin,
+ const nsRect& aScrolledRect) {
+ // Inflate the rect by scroll-margin.
+ nsRect result = aTargetRect;
+ result.Inflate(aScrollMargin);
+
+ // But don't be beyond the limit boundary.
+ return result.Intersect(aScrolledRect);
+}
+
+nsRect ScrollSnapUtils::GetSnapAreaFor(const nsIFrame* aFrame,
+ const nsIFrame* aScrolledFrame,
+ const nsRect& aScrolledRect) {
+ nsRect targetRect = nsLayoutUtils::TransformFrameRectToAncestor(
+ aFrame, aFrame->GetRectRelativeToSelf(), aScrolledFrame);
+
+ // The snap area contains scroll-margin values.
+ // https://drafts.csswg.org/css-scroll-snap-1/#scroll-snap-area
+ nsMargin scrollMargin = aFrame->StyleMargin()->GetScrollMargin();
+ return InflateByScrollMargin(targetRect, scrollMargin, aScrolledRect);
+}
+
+} // namespace mozilla