UNPKG

react-aria

Version:
160 lines (152 loc) • 8.53 kB
import {getEventTarget as $d8ac7ed472840322$export$e58f029f0fbfdb29, nodeContains as $d8ac7ed472840322$export$4282f70798064fe0} from "../utils/shadowdom/DOMFunctions.js"; import {getOwnerDocument as $cc3c3666b64debad$export$b204af158042fbac} from "../utils/domHelpers.js"; import {useGlobalListeners as $0d742958be022209$export$4eaf04e54aa8eed6} from "../utils/useGlobalListeners.js"; import {useState as $4dh1a$useState, useRef as $4dh1a$useRef, useEffect as $4dh1a$useEffect, useMemo as $4dh1a$useMemo} from "react"; /* * 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. */ // Portions of the code in this file are based on code from react. // Original licensing for the following can be found in the // NOTICE file in the root directory of this source tree. // See https://github.com/facebook/react/tree/cc7c1aece46a6b69b41958d731e0fd27c94bfc6c/packages/react-interactions // iOS fires onPointerEnter twice: once with pointerType="touch" and again with pointerType="mouse". // We want to ignore these emulated events so they do not trigger hover behavior. // See https://bugs.webkit.org/show_bug.cgi?id=214609. let $f7f05710dfc01c4c$var$globalIgnoreEmulatedMouseEvents = false; let $f7f05710dfc01c4c$var$hoverCount = 0; function $f7f05710dfc01c4c$var$setGlobalIgnoreEmulatedMouseEvents() { $f7f05710dfc01c4c$var$globalIgnoreEmulatedMouseEvents = true; // Clear globalIgnoreEmulatedMouseEvents after a short timeout. iOS fires onPointerEnter // with pointerType="mouse" immediately after onPointerUp and before onFocus. On other // devices that don't have this quirk, we don't want to ignore a mouse hover sometime in // the distant future because a user previously touched the element. setTimeout(()=>{ $f7f05710dfc01c4c$var$globalIgnoreEmulatedMouseEvents = false; }, 500); } function $f7f05710dfc01c4c$var$handleGlobalPointerEvent(e) { if (e.pointerType === 'touch') $f7f05710dfc01c4c$var$setGlobalIgnoreEmulatedMouseEvents(); } function $f7f05710dfc01c4c$var$setupGlobalTouchEvents() { let ownerDocument = (0, $cc3c3666b64debad$export$b204af158042fbac)(null); if (typeof ownerDocument === 'undefined') return; if ($f7f05710dfc01c4c$var$hoverCount === 0) { if (typeof PointerEvent !== 'undefined') ownerDocument.addEventListener('pointerup', $f7f05710dfc01c4c$var$handleGlobalPointerEvent); else if (process.env.NODE_ENV === 'test') ownerDocument.addEventListener('touchend', $f7f05710dfc01c4c$var$setGlobalIgnoreEmulatedMouseEvents); } $f7f05710dfc01c4c$var$hoverCount++; return ()=>{ $f7f05710dfc01c4c$var$hoverCount--; if ($f7f05710dfc01c4c$var$hoverCount > 0) return; if (typeof PointerEvent !== 'undefined') ownerDocument.removeEventListener('pointerup', $f7f05710dfc01c4c$var$handleGlobalPointerEvent); else if (process.env.NODE_ENV === 'test') ownerDocument.removeEventListener('touchend', $f7f05710dfc01c4c$var$setGlobalIgnoreEmulatedMouseEvents); }; } function $f7f05710dfc01c4c$export$ae780daf29e6d456(props) { let { onHoverStart: onHoverStart, onHoverChange: onHoverChange, onHoverEnd: onHoverEnd, isDisabled: isDisabled } = props; let [isHovered, setHovered] = (0, $4dh1a$useState)(false); let state = (0, $4dh1a$useRef)({ isHovered: false, ignoreEmulatedMouseEvents: false, pointerType: '', target: null }).current; (0, $4dh1a$useEffect)($f7f05710dfc01c4c$var$setupGlobalTouchEvents, []); let { addGlobalListener: addGlobalListener, removeAllGlobalListeners: removeAllGlobalListeners } = (0, $0d742958be022209$export$4eaf04e54aa8eed6)(); let { hoverProps: hoverProps, triggerHoverEnd: triggerHoverEnd } = (0, $4dh1a$useMemo)(()=>{ let triggerHoverStart = (event, pointerType)=>{ state.pointerType = pointerType; if (isDisabled || pointerType === 'touch' || state.isHovered || !(0, $d8ac7ed472840322$export$4282f70798064fe0)(event.currentTarget, (0, $d8ac7ed472840322$export$e58f029f0fbfdb29)(event))) return; state.isHovered = true; let target = event.currentTarget; state.target = target; // When an element that is hovered over is removed, no pointerleave event is fired by the browser, // even though the originally hovered target may have shrunk in size so it is no longer hovered. // However, a pointerover event will be fired on the new target the mouse is over. // In Chrome this happens immediately. In Safari and Firefox, it happens upon moving the mouse one pixel. addGlobalListener((0, $cc3c3666b64debad$export$b204af158042fbac)((0, $d8ac7ed472840322$export$e58f029f0fbfdb29)(event)), 'pointerover', (e)=>{ if (state.isHovered && state.target && !(0, $d8ac7ed472840322$export$4282f70798064fe0)(state.target, (0, $d8ac7ed472840322$export$e58f029f0fbfdb29)(e))) triggerHoverEnd(e, e.pointerType); }, { capture: true }); if (onHoverStart) onHoverStart({ type: 'hoverstart', target: target, pointerType: pointerType }); if (onHoverChange) onHoverChange(true); setHovered(true); }; let triggerHoverEnd = (event, pointerType)=>{ let target = state.target; state.pointerType = ''; state.target = null; if (pointerType === 'touch' || !state.isHovered || !target) return; state.isHovered = false; removeAllGlobalListeners(); if (onHoverEnd) onHoverEnd({ type: 'hoverend', target: target, pointerType: pointerType }); if (onHoverChange) onHoverChange(false); setHovered(false); }; let hoverProps = {}; if (typeof PointerEvent !== 'undefined') { hoverProps.onPointerEnter = (e)=>{ if ($f7f05710dfc01c4c$var$globalIgnoreEmulatedMouseEvents && e.pointerType === 'mouse') return; triggerHoverStart(e, e.pointerType); }; hoverProps.onPointerLeave = (e)=>{ if (!isDisabled && (0, $d8ac7ed472840322$export$4282f70798064fe0)(e.currentTarget, (0, $d8ac7ed472840322$export$e58f029f0fbfdb29)(e))) triggerHoverEnd(e, e.pointerType); }; } else if (process.env.NODE_ENV === 'test') { hoverProps.onTouchStart = ()=>{ state.ignoreEmulatedMouseEvents = true; }; hoverProps.onMouseEnter = (e)=>{ if (!state.ignoreEmulatedMouseEvents && !$f7f05710dfc01c4c$var$globalIgnoreEmulatedMouseEvents) triggerHoverStart(e, 'mouse'); state.ignoreEmulatedMouseEvents = false; }; hoverProps.onMouseLeave = (e)=>{ if (!isDisabled && (0, $d8ac7ed472840322$export$4282f70798064fe0)(e.currentTarget, (0, $d8ac7ed472840322$export$e58f029f0fbfdb29)(e))) triggerHoverEnd(e, 'mouse'); }; } return { hoverProps: hoverProps, triggerHoverEnd: triggerHoverEnd }; }, [ onHoverStart, onHoverChange, onHoverEnd, isDisabled, state, addGlobalListener, removeAllGlobalListeners ]); (0, $4dh1a$useEffect)(()=>{ // Call the triggerHoverEnd as soon as isDisabled changes to true // Safe to call triggerHoverEnd, it will early return if we aren't currently hovering if (isDisabled) triggerHoverEnd({ currentTarget: state.target }, state.pointerType); // eslint-disable-next-line react-hooks/exhaustive-deps }, [ isDisabled ]); return { hoverProps: hoverProps, isHovered: isHovered }; } export {$f7f05710dfc01c4c$export$ae780daf29e6d456 as useHover}; //# sourceMappingURL=useHover.js.map