@react-aria/overlays
Version:
Spectrum UI components in React
214 lines (208 loc) • 12.5 kB
JavaScript
var $59kHH$reactariautils = require("@react-aria/utils");
function $parcel$export(e, n, v, s) {
Object.defineProperty(e, n, {get: v, set: s, enumerable: true, configurable: true});
}
$parcel$export(module.exports, "usePreventScroll", () => $5c2f5cd01815d369$export$ee0f7cc6afcd1c18);
/*
* Copyright 2020 Adobe. All rights reserved.
* This file is licensed to you under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. You may obtain a copy
* of the License at http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software distributed under
* the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS
* OF ANY KIND, either express or implied. See the License for the specific language
* governing permissions and limitations under the License.
*/
const $5c2f5cd01815d369$var$visualViewport = typeof document !== 'undefined' && window.visualViewport;
// The number of active usePreventScroll calls. Used to determine whether to revert back to the original page style/scroll position
let $5c2f5cd01815d369$var$preventScrollCount = 0;
let $5c2f5cd01815d369$var$restore;
function $5c2f5cd01815d369$export$ee0f7cc6afcd1c18(options = {}) {
let { isDisabled: isDisabled } = options;
(0, $59kHH$reactariautils.useLayoutEffect)(()=>{
if (isDisabled) return;
$5c2f5cd01815d369$var$preventScrollCount++;
if ($5c2f5cd01815d369$var$preventScrollCount === 1) {
if ((0, $59kHH$reactariautils.isIOS)()) $5c2f5cd01815d369$var$restore = $5c2f5cd01815d369$var$preventScrollMobileSafari();
else $5c2f5cd01815d369$var$restore = $5c2f5cd01815d369$var$preventScrollStandard();
}
return ()=>{
$5c2f5cd01815d369$var$preventScrollCount--;
if ($5c2f5cd01815d369$var$preventScrollCount === 0) $5c2f5cd01815d369$var$restore();
};
}, [
isDisabled
]);
}
// For most browsers, all we need to do is set `overflow: hidden` on the root element, and
// add some padding to prevent the page from shifting when the scrollbar is hidden.
function $5c2f5cd01815d369$var$preventScrollStandard() {
let scrollbarWidth = window.innerWidth - document.documentElement.clientWidth;
return (0, $59kHH$reactariautils.chain)(scrollbarWidth > 0 && // Use scrollbar-gutter when supported because it also works for fixed positioned elements.
('scrollbarGutter' in document.documentElement.style ? $5c2f5cd01815d369$var$setStyle(document.documentElement, 'scrollbarGutter', 'stable') : $5c2f5cd01815d369$var$setStyle(document.documentElement, 'paddingRight', `${scrollbarWidth}px`)), $5c2f5cd01815d369$var$setStyle(document.documentElement, 'overflow', 'hidden'));
}
// Mobile Safari is a whole different beast. Even with overflow: hidden,
// it still scrolls the page in many situations:
//
// 1. When the bottom toolbar and address bar are collapsed, page scrolling is always allowed.
// 2. When the keyboard is visible, the viewport does not resize. Instead, the keyboard covers part of
// it, so it becomes scrollable.
// 3. When tapping on an input, the page always scrolls so that the input is centered in the visual viewport.
// This may cause even fixed position elements to scroll off the screen.
// 4. When using the next/previous buttons in the keyboard to navigate between inputs, the whole page always
// scrolls, even if the input is inside a nested scrollable element that could be scrolled instead.
//
// In order to work around these cases, and prevent scrolling without jankiness, we do a few things:
//
// 1. Prevent default on `touchmove` events that are not in a scrollable element. This prevents touch scrolling
// on the window.
// 2. Set `overscroll-behavior: contain` on nested scrollable regions so they do not scroll the page when at
// the top or bottom. Work around a bug where this does not work when the element does not actually overflow
// by preventing default in a `touchmove` event. This is best effort: we can't prevent default when pinch
// zooming or when an element contains text selection, which may allow scrolling in some cases.
// 3. Prevent default on `touchend` events on input elements and handle focusing the element ourselves.
// 4. When focus moves to an input, create an off screen input and focus that temporarily. This prevents
// Safari from scrolling the page. After a small delay, focus the real input and scroll it into view
// ourselves, without scrolling the whole page.
function $5c2f5cd01815d369$var$preventScrollMobileSafari() {
let scrollable;
let allowTouchMove = false;
let onTouchStart = (e)=>{
// Store the nearest scrollable parent element from the element that the user touched.
let target = e.target;
scrollable = (0, $59kHH$reactariautils.isScrollable)(target) ? target : (0, $59kHH$reactariautils.getScrollParent)(target, true);
allowTouchMove = false;
// If the target is selected, don't preventDefault in touchmove to allow user to adjust selection.
let selection = target.ownerDocument.defaultView.getSelection();
if (selection && !selection.isCollapsed && selection.containsNode(target, true)) allowTouchMove = true;
// If this is a focused input element with a selected range, allow user to drag the selection handles.
if ('selectionStart' in target && 'selectionEnd' in target && target.selectionStart < target.selectionEnd && target.ownerDocument.activeElement === target) allowTouchMove = true;
};
// Prevent scrolling up when at the top and scrolling down when at the bottom
// of a nested scrollable area, otherwise mobile Safari will start scrolling
// the window instead.
// This must be applied before the touchstart event as of iOS 26, so inject it as a <style> element.
let style = document.createElement('style');
style.textContent = `
{
* {
overscroll-behavior: contain;
}
}`.trim();
document.head.prepend(style);
let onTouchMove = (e)=>{
// Allow pinch-zooming.
if (e.touches.length === 2 || allowTouchMove) return;
// Prevent scrolling the window.
if (!scrollable || scrollable === document.documentElement || scrollable === document.body) {
e.preventDefault();
return;
}
// overscroll-behavior should prevent scroll chaining, but currently does not
// if the element doesn't actually overflow. https://bugs.webkit.org/show_bug.cgi?id=243452
// This checks that both the width and height do not overflow, otherwise we might
// block horizontal scrolling too. In that case, adding `touch-action: pan-x` to
// the element will prevent vertical page scrolling. We can't add that automatically
// because it must be set before the touchstart event.
if (scrollable.scrollHeight === scrollable.clientHeight && scrollable.scrollWidth === scrollable.clientWidth) e.preventDefault();
};
let onBlur = (e)=>{
let target = e.target;
let relatedTarget = e.relatedTarget;
if (relatedTarget && (0, $59kHH$reactariautils.willOpenKeyboard)(relatedTarget)) {
// Focus without scrolling the whole page, and then scroll into view manually.
relatedTarget.focus({
preventScroll: true
});
$5c2f5cd01815d369$var$scrollIntoViewWhenReady(relatedTarget, (0, $59kHH$reactariautils.willOpenKeyboard)(target));
} else if (!relatedTarget) {
var _target_parentElement;
// When tapping the Done button on the keyboard, focus moves to the body.
// FocusScope will then restore focus back to the input. Later when tapping
// the same input again, it is already focused, so no blur event will fire,
// resulting in the flow above never running and Safari's native scrolling occurring.
// Instead, move focus to the parent focusable element (e.g. the dialog).
let focusable = (_target_parentElement = target.parentElement) === null || _target_parentElement === void 0 ? void 0 : _target_parentElement.closest('[tabindex]');
focusable === null || focusable === void 0 ? void 0 : focusable.focus({
preventScroll: true
});
}
};
// Override programmatic focus to scroll into view without scrolling the whole page.
let focus = HTMLElement.prototype.focus;
HTMLElement.prototype.focus = function(opts) {
// Track whether the keyboard was already visible before.
let wasKeyboardVisible = document.activeElement != null && (0, $59kHH$reactariautils.willOpenKeyboard)(document.activeElement);
// Focus the element without scrolling the page.
focus.call(this, {
...opts,
preventScroll: true
});
if (!opts || !opts.preventScroll) $5c2f5cd01815d369$var$scrollIntoViewWhenReady(this, wasKeyboardVisible);
};
let removeEvents = (0, $59kHH$reactariautils.chain)($5c2f5cd01815d369$var$addEvent(document, 'touchstart', onTouchStart, {
passive: false,
capture: true
}), $5c2f5cd01815d369$var$addEvent(document, 'touchmove', onTouchMove, {
passive: false,
capture: true
}), $5c2f5cd01815d369$var$addEvent(document, 'blur', onBlur, true));
return ()=>{
removeEvents();
style.remove();
HTMLElement.prototype.focus = focus;
};
}
// Sets a CSS property on an element, and returns a function to revert it to the previous value.
function $5c2f5cd01815d369$var$setStyle(element, style, value) {
let cur = element.style[style];
element.style[style] = value;
return ()=>{
element.style[style] = cur;
};
}
// Adds an event listener to an element, and returns a function to remove it.
function $5c2f5cd01815d369$var$addEvent(target, event, handler, options) {
// internal function, so it's ok to ignore the difficult to fix type error
// @ts-ignore
target.addEventListener(event, handler, options);
return ()=>{
// @ts-ignore
target.removeEventListener(event, handler, options);
};
}
function $5c2f5cd01815d369$var$scrollIntoViewWhenReady(target, wasKeyboardVisible) {
if (wasKeyboardVisible || !$5c2f5cd01815d369$var$visualViewport) // If the keyboard was already visible, scroll the target into view immediately.
$5c2f5cd01815d369$var$scrollIntoView(target);
else // Otherwise, wait for the visual viewport to resize before scrolling so we can
// measure the correct position to scroll to.
$5c2f5cd01815d369$var$visualViewport.addEventListener('resize', ()=>$5c2f5cd01815d369$var$scrollIntoView(target), {
once: true
});
}
function $5c2f5cd01815d369$var$scrollIntoView(target) {
let root = document.scrollingElement || document.documentElement;
let nextTarget = target;
while(nextTarget && nextTarget !== root){
// Find the parent scrollable element and adjust the scroll position if the target is not already in view.
let scrollable = (0, $59kHH$reactariautils.getScrollParent)(nextTarget);
if (scrollable !== document.documentElement && scrollable !== document.body && scrollable !== nextTarget) {
let scrollableRect = scrollable.getBoundingClientRect();
let targetRect = nextTarget.getBoundingClientRect();
if (targetRect.top < scrollableRect.top || targetRect.bottom > scrollableRect.top + nextTarget.clientHeight) {
let bottom = scrollableRect.bottom;
if ($5c2f5cd01815d369$var$visualViewport) bottom = Math.min(bottom, $5c2f5cd01815d369$var$visualViewport.offsetTop + $5c2f5cd01815d369$var$visualViewport.height);
// Center within the viewport.
let adjustment = targetRect.top - scrollableRect.top - ((bottom - scrollableRect.top) / 2 - targetRect.height / 2);
scrollable.scrollTo({
// Clamp to the valid range to prevent over-scrolling.
top: Math.max(0, Math.min(scrollable.scrollHeight - scrollable.clientHeight, scrollable.scrollTop + adjustment)),
behavior: 'smooth'
});
}
}
nextTarget = scrollable.parentElement;
}
}
//# sourceMappingURL=usePreventScroll.main.js.map