UNPKG

@react-aria/overlays

Version:
214 lines (208 loc) • 12.5 kB
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 = ` @layer { * { 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