UNPKG

6.61 kBJavaScriptView Raw
1var $82z6W$react = require("react");
2
3
4function $parcel$export(e, n, v, s) {
5 Object.defineProperty(e, n, {get: v, set: s, enumerable: true, configurable: true});
6}
7
8$parcel$export(module.exports, "useHover", () => $ffbc150311c75f01$export$ae780daf29e6d456);
9/*
10 * Copyright 2020 Adobe. All rights reserved.
11 * This file is licensed to you under the Apache License, Version 2.0 (the "License");
12 * you may not use this file except in compliance with the License. You may obtain a copy
13 * of the License at http://www.apache.org/licenses/LICENSE-2.0
14 *
15 * Unless required by applicable law or agreed to in writing, software distributed under
16 * the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS
17 * OF ANY KIND, either express or implied. See the License for the specific language
18 * governing permissions and limitations under the License.
19 */ // Portions of the code in this file are based on code from react.
20// Original licensing for the following can be found in the
21// NOTICE file in the root directory of this source tree.
22// See https://github.com/facebook/react/tree/cc7c1aece46a6b69b41958d731e0fd27c94bfc6c/packages/react-interactions
23
24// iOS fires onPointerEnter twice: once with pointerType="touch" and again with pointerType="mouse".
25// We want to ignore these emulated events so they do not trigger hover behavior.
26// See https://bugs.webkit.org/show_bug.cgi?id=214609.
27let $ffbc150311c75f01$var$globalIgnoreEmulatedMouseEvents = false;
28let $ffbc150311c75f01$var$hoverCount = 0;
29function $ffbc150311c75f01$var$setGlobalIgnoreEmulatedMouseEvents() {
30 $ffbc150311c75f01$var$globalIgnoreEmulatedMouseEvents = true;
31 // Clear globalIgnoreEmulatedMouseEvents after a short timeout. iOS fires onPointerEnter
32 // with pointerType="mouse" immediately after onPointerUp and before onFocus. On other
33 // devices that don't have this quirk, we don't want to ignore a mouse hover sometime in
34 // the distant future because a user previously touched the element.
35 setTimeout(()=>{
36 $ffbc150311c75f01$var$globalIgnoreEmulatedMouseEvents = false;
37 }, 50);
38}
39function $ffbc150311c75f01$var$handleGlobalPointerEvent(e) {
40 if (e.pointerType === 'touch') $ffbc150311c75f01$var$setGlobalIgnoreEmulatedMouseEvents();
41}
42function $ffbc150311c75f01$var$setupGlobalTouchEvents() {
43 if (typeof document === 'undefined') return;
44 if (typeof PointerEvent !== 'undefined') document.addEventListener('pointerup', $ffbc150311c75f01$var$handleGlobalPointerEvent);
45 else document.addEventListener('touchend', $ffbc150311c75f01$var$setGlobalIgnoreEmulatedMouseEvents);
46 $ffbc150311c75f01$var$hoverCount++;
47 return ()=>{
48 $ffbc150311c75f01$var$hoverCount--;
49 if ($ffbc150311c75f01$var$hoverCount > 0) return;
50 if (typeof PointerEvent !== 'undefined') document.removeEventListener('pointerup', $ffbc150311c75f01$var$handleGlobalPointerEvent);
51 else document.removeEventListener('touchend', $ffbc150311c75f01$var$setGlobalIgnoreEmulatedMouseEvents);
52 };
53}
54function $ffbc150311c75f01$export$ae780daf29e6d456(props) {
55 let { onHoverStart: onHoverStart, onHoverChange: onHoverChange, onHoverEnd: onHoverEnd, isDisabled: isDisabled } = props;
56 let [isHovered, setHovered] = (0, $82z6W$react.useState)(false);
57 let state = (0, $82z6W$react.useRef)({
58 isHovered: false,
59 ignoreEmulatedMouseEvents: false,
60 pointerType: '',
61 target: null
62 }).current;
63 (0, $82z6W$react.useEffect)($ffbc150311c75f01$var$setupGlobalTouchEvents, []);
64 let { hoverProps: hoverProps, triggerHoverEnd: triggerHoverEnd } = (0, $82z6W$react.useMemo)(()=>{
65 let triggerHoverStart = (event, pointerType)=>{
66 state.pointerType = pointerType;
67 if (isDisabled || pointerType === 'touch' || state.isHovered || !event.currentTarget.contains(event.target)) return;
68 state.isHovered = true;
69 let target = event.currentTarget;
70 state.target = target;
71 if (onHoverStart) onHoverStart({
72 type: 'hoverstart',
73 target: target,
74 pointerType: pointerType
75 });
76 if (onHoverChange) onHoverChange(true);
77 setHovered(true);
78 };
79 let triggerHoverEnd = (event, pointerType)=>{
80 state.pointerType = '';
81 state.target = null;
82 if (pointerType === 'touch' || !state.isHovered) return;
83 state.isHovered = false;
84 let target = event.currentTarget;
85 if (onHoverEnd) onHoverEnd({
86 type: 'hoverend',
87 target: target,
88 pointerType: pointerType
89 });
90 if (onHoverChange) onHoverChange(false);
91 setHovered(false);
92 };
93 let hoverProps = {};
94 if (typeof PointerEvent !== 'undefined') {
95 hoverProps.onPointerEnter = (e)=>{
96 if ($ffbc150311c75f01$var$globalIgnoreEmulatedMouseEvents && e.pointerType === 'mouse') return;
97 triggerHoverStart(e, e.pointerType);
98 };
99 hoverProps.onPointerLeave = (e)=>{
100 if (!isDisabled && e.currentTarget.contains(e.target)) triggerHoverEnd(e, e.pointerType);
101 };
102 } else {
103 hoverProps.onTouchStart = ()=>{
104 state.ignoreEmulatedMouseEvents = true;
105 };
106 hoverProps.onMouseEnter = (e)=>{
107 if (!state.ignoreEmulatedMouseEvents && !$ffbc150311c75f01$var$globalIgnoreEmulatedMouseEvents) triggerHoverStart(e, 'mouse');
108 state.ignoreEmulatedMouseEvents = false;
109 };
110 hoverProps.onMouseLeave = (e)=>{
111 if (!isDisabled && e.currentTarget.contains(e.target)) triggerHoverEnd(e, 'mouse');
112 };
113 }
114 return {
115 hoverProps: hoverProps,
116 triggerHoverEnd: triggerHoverEnd
117 };
118 }, [
119 onHoverStart,
120 onHoverChange,
121 onHoverEnd,
122 isDisabled,
123 state
124 ]);
125 (0, $82z6W$react.useEffect)(()=>{
126 // Call the triggerHoverEnd as soon as isDisabled changes to true
127 // Safe to call triggerHoverEnd, it will early return if we aren't currently hovering
128 if (isDisabled) triggerHoverEnd({
129 currentTarget: state.target
130 }, state.pointerType);
131 // eslint-disable-next-line react-hooks/exhaustive-deps
132 }, [
133 isDisabled
134 ]);
135 return {
136 hoverProps: hoverProps,
137 isHovered: isHovered
138 };
139}
140
141
142//# sourceMappingURL=useHover.main.js.map