@zag-js/dismissable 0.49.0 → 0.51.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/index.d.mts +10 -2
- package/dist/index.d.ts +10 -2
- package/dist/index.js +1 -245
- package/dist/index.js.map +1 -1
- package/dist/index.mjs +1 -219
- package/dist/index.mjs.map +1 -1
- package/package.json +5 -5
- package/src/dismissable-layer.ts +14 -3
- package/src/escape-keydown.ts +4 -1
- package/src/index.ts +3 -3
- package/src/pointer-event-outside.ts +22 -10
package/dist/index.d.mts
CHANGED
|
@@ -11,7 +11,15 @@ interface DismissableElementHandlers extends InteractOutsideHandlers {
|
|
|
11
11
|
*/
|
|
12
12
|
onEscapeKeyDown?: (event: KeyboardEvent) => void;
|
|
13
13
|
}
|
|
14
|
-
interface
|
|
14
|
+
interface PersistentElementOptions {
|
|
15
|
+
/**
|
|
16
|
+
* Returns the persistent elements that:
|
|
17
|
+
* - should not have pointer-events disabled
|
|
18
|
+
* - should not trigger the dismiss event
|
|
19
|
+
*/
|
|
20
|
+
persistentElements?: Array<() => Element | null>;
|
|
21
|
+
}
|
|
22
|
+
interface DismissableElementOptions extends DismissableElementHandlers, PersistentElementOptions {
|
|
15
23
|
/**
|
|
16
24
|
* Whether to log debug information
|
|
17
25
|
*/
|
|
@@ -38,4 +46,4 @@ declare function trackDismissableBranch(nodeOrFn: NodeOrFn, options?: {
|
|
|
38
46
|
defer?: boolean;
|
|
39
47
|
}): () => void;
|
|
40
48
|
|
|
41
|
-
export { type DismissableElementHandlers, type DismissableElementOptions, trackDismissableBranch, trackDismissableElement };
|
|
49
|
+
export { type DismissableElementHandlers, type DismissableElementOptions, type PersistentElementOptions, trackDismissableBranch, trackDismissableElement };
|
package/dist/index.d.ts
CHANGED
|
@@ -11,7 +11,15 @@ interface DismissableElementHandlers extends InteractOutsideHandlers {
|
|
|
11
11
|
*/
|
|
12
12
|
onEscapeKeyDown?: (event: KeyboardEvent) => void;
|
|
13
13
|
}
|
|
14
|
-
interface
|
|
14
|
+
interface PersistentElementOptions {
|
|
15
|
+
/**
|
|
16
|
+
* Returns the persistent elements that:
|
|
17
|
+
* - should not have pointer-events disabled
|
|
18
|
+
* - should not trigger the dismiss event
|
|
19
|
+
*/
|
|
20
|
+
persistentElements?: Array<() => Element | null>;
|
|
21
|
+
}
|
|
22
|
+
interface DismissableElementOptions extends DismissableElementHandlers, PersistentElementOptions {
|
|
15
23
|
/**
|
|
16
24
|
* Whether to log debug information
|
|
17
25
|
*/
|
|
@@ -38,4 +46,4 @@ declare function trackDismissableBranch(nodeOrFn: NodeOrFn, options?: {
|
|
|
38
46
|
defer?: boolean;
|
|
39
47
|
}): () => void;
|
|
40
48
|
|
|
41
|
-
export { type DismissableElementHandlers, type DismissableElementOptions, trackDismissableBranch, trackDismissableElement };
|
|
49
|
+
export { type DismissableElementHandlers, type DismissableElementOptions, type PersistentElementOptions, trackDismissableBranch, trackDismissableElement };
|
package/dist/index.js
CHANGED
|
@@ -1,246 +1,2 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
var __defProp = Object.defineProperty;
|
|
3
|
-
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
4
|
-
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
5
|
-
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
6
|
-
var __export = (target, all) => {
|
|
7
|
-
for (var name in all)
|
|
8
|
-
__defProp(target, name, { get: all[name], enumerable: true });
|
|
9
|
-
};
|
|
10
|
-
var __copyProps = (to, from, except, desc) => {
|
|
11
|
-
if (from && typeof from === "object" || typeof from === "function") {
|
|
12
|
-
for (let key of __getOwnPropNames(from))
|
|
13
|
-
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
14
|
-
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
15
|
-
}
|
|
16
|
-
return to;
|
|
17
|
-
};
|
|
18
|
-
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
19
|
-
|
|
20
|
-
// src/index.ts
|
|
21
|
-
var src_exports = {};
|
|
22
|
-
__export(src_exports, {
|
|
23
|
-
trackDismissableBranch: () => trackDismissableBranch,
|
|
24
|
-
trackDismissableElement: () => trackDismissableElement
|
|
25
|
-
});
|
|
26
|
-
module.exports = __toCommonJS(src_exports);
|
|
27
|
-
|
|
28
|
-
// src/dismissable-layer.ts
|
|
29
|
-
var import_dom_query4 = require("@zag-js/dom-query");
|
|
30
|
-
var import_interact_outside = require("@zag-js/interact-outside");
|
|
31
|
-
var import_utils = require("@zag-js/utils");
|
|
32
|
-
|
|
33
|
-
// src/escape-keydown.ts
|
|
34
|
-
var import_dom_event = require("@zag-js/dom-event");
|
|
35
|
-
var import_dom_query = require("@zag-js/dom-query");
|
|
36
|
-
function trackEscapeKeydown(node, fn) {
|
|
37
|
-
const handleKeyDown = (event) => {
|
|
38
|
-
if (event.key === "Escape")
|
|
39
|
-
fn?.(event);
|
|
40
|
-
};
|
|
41
|
-
return (0, import_dom_event.addDomEvent)((0, import_dom_query.getDocument)(node), "keydown", handleKeyDown, { capture: true });
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
// src/layer-stack.ts
|
|
45
|
-
var import_dom_query2 = require("@zag-js/dom-query");
|
|
46
|
-
var layerStack = {
|
|
47
|
-
layers: [],
|
|
48
|
-
branches: [],
|
|
49
|
-
count() {
|
|
50
|
-
return this.layers.length;
|
|
51
|
-
},
|
|
52
|
-
pointerBlockingLayers() {
|
|
53
|
-
return this.layers.filter((layer) => layer.pointerBlocking);
|
|
54
|
-
},
|
|
55
|
-
topMostPointerBlockingLayer() {
|
|
56
|
-
return [...this.pointerBlockingLayers()].slice(-1)[0];
|
|
57
|
-
},
|
|
58
|
-
hasPointerBlockingLayer() {
|
|
59
|
-
return this.pointerBlockingLayers().length > 0;
|
|
60
|
-
},
|
|
61
|
-
isBelowPointerBlockingLayer(node) {
|
|
62
|
-
const index = this.indexOf(node);
|
|
63
|
-
const highestBlockingIndex = this.topMostPointerBlockingLayer() ? this.indexOf(this.topMostPointerBlockingLayer()?.node) : -1;
|
|
64
|
-
return index < highestBlockingIndex;
|
|
65
|
-
},
|
|
66
|
-
isTopMost(node) {
|
|
67
|
-
const layer = this.layers[this.count() - 1];
|
|
68
|
-
return layer?.node === node;
|
|
69
|
-
},
|
|
70
|
-
getNestedLayers(node) {
|
|
71
|
-
return Array.from(this.layers).slice(this.indexOf(node) + 1);
|
|
72
|
-
},
|
|
73
|
-
isInNestedLayer(node, target) {
|
|
74
|
-
return this.getNestedLayers(node).some((layer) => (0, import_dom_query2.contains)(layer.node, target));
|
|
75
|
-
},
|
|
76
|
-
isInBranch(target) {
|
|
77
|
-
return Array.from(this.branches).some((branch) => (0, import_dom_query2.contains)(branch, target));
|
|
78
|
-
},
|
|
79
|
-
add(layer) {
|
|
80
|
-
const num = this.layers.push(layer);
|
|
81
|
-
layer.node.style.setProperty("--layer-index", `${num}`);
|
|
82
|
-
},
|
|
83
|
-
addBranch(node) {
|
|
84
|
-
this.branches.push(node);
|
|
85
|
-
},
|
|
86
|
-
remove(node) {
|
|
87
|
-
const index = this.indexOf(node);
|
|
88
|
-
if (index < 0)
|
|
89
|
-
return;
|
|
90
|
-
if (index < this.count() - 1) {
|
|
91
|
-
const _layers = this.getNestedLayers(node);
|
|
92
|
-
_layers.forEach((layer) => layer.dismiss());
|
|
93
|
-
}
|
|
94
|
-
this.layers.splice(index, 1);
|
|
95
|
-
node.style.removeProperty("--layer-index");
|
|
96
|
-
},
|
|
97
|
-
removeBranch(node) {
|
|
98
|
-
const index = this.branches.indexOf(node);
|
|
99
|
-
if (index >= 0)
|
|
100
|
-
this.branches.splice(index, 1);
|
|
101
|
-
},
|
|
102
|
-
indexOf(node) {
|
|
103
|
-
return this.layers.findIndex((layer) => layer.node === node);
|
|
104
|
-
},
|
|
105
|
-
dismiss(node) {
|
|
106
|
-
this.layers[this.indexOf(node)]?.dismiss();
|
|
107
|
-
},
|
|
108
|
-
clear() {
|
|
109
|
-
this.remove(this.layers[0].node);
|
|
110
|
-
}
|
|
111
|
-
};
|
|
112
|
-
|
|
113
|
-
// src/pointer-event-outside.ts
|
|
114
|
-
var import_dom_query3 = require("@zag-js/dom-query");
|
|
115
|
-
var originalBodyPointerEvents;
|
|
116
|
-
function assignPointerEventToLayers() {
|
|
117
|
-
layerStack.layers.forEach(({ node }) => {
|
|
118
|
-
node.style.pointerEvents = layerStack.isBelowPointerBlockingLayer(node) ? "none" : "auto";
|
|
119
|
-
});
|
|
120
|
-
}
|
|
121
|
-
function clearPointerEvent(node) {
|
|
122
|
-
node.style.pointerEvents = "";
|
|
123
|
-
}
|
|
124
|
-
var DATA_ATTR = "data-inert";
|
|
125
|
-
function disablePointerEventsOutside(node) {
|
|
126
|
-
const doc = (0, import_dom_query3.getDocument)(node);
|
|
127
|
-
if (layerStack.hasPointerBlockingLayer() && !doc.body.hasAttribute(DATA_ATTR)) {
|
|
128
|
-
originalBodyPointerEvents = document.body.style.pointerEvents;
|
|
129
|
-
doc.body.style.pointerEvents = "none";
|
|
130
|
-
doc.body.setAttribute(DATA_ATTR, "");
|
|
131
|
-
}
|
|
132
|
-
return () => {
|
|
133
|
-
if (layerStack.hasPointerBlockingLayer())
|
|
134
|
-
return;
|
|
135
|
-
doc.body.style.pointerEvents = originalBodyPointerEvents;
|
|
136
|
-
doc.body.removeAttribute(DATA_ATTR);
|
|
137
|
-
if (doc.body.style.length === 0)
|
|
138
|
-
doc.body.removeAttribute("style");
|
|
139
|
-
};
|
|
140
|
-
}
|
|
141
|
-
|
|
142
|
-
// src/dismissable-layer.ts
|
|
143
|
-
function trackDismissableElementImpl(node, options) {
|
|
144
|
-
if (!node) {
|
|
145
|
-
(0, import_utils.warn)("[@zag-js/dismissable] node is `null` or `undefined`");
|
|
146
|
-
return;
|
|
147
|
-
}
|
|
148
|
-
const { onDismiss, pointerBlocking, exclude: excludeContainers, debug } = options;
|
|
149
|
-
const layer = { dismiss: onDismiss, node, pointerBlocking };
|
|
150
|
-
layerStack.add(layer);
|
|
151
|
-
assignPointerEventToLayers();
|
|
152
|
-
function onPointerDownOutside(event) {
|
|
153
|
-
const target = (0, import_dom_query4.getEventTarget)(event.detail.originalEvent);
|
|
154
|
-
if (layerStack.isBelowPointerBlockingLayer(node) || layerStack.isInBranch(target))
|
|
155
|
-
return;
|
|
156
|
-
options.onPointerDownOutside?.(event);
|
|
157
|
-
options.onInteractOutside?.(event);
|
|
158
|
-
if (event.defaultPrevented)
|
|
159
|
-
return;
|
|
160
|
-
if (debug) {
|
|
161
|
-
console.log("onPointerDownOutside:", event.detail.originalEvent);
|
|
162
|
-
}
|
|
163
|
-
onDismiss?.();
|
|
164
|
-
}
|
|
165
|
-
function onFocusOutside(event) {
|
|
166
|
-
const target = (0, import_dom_query4.getEventTarget)(event.detail.originalEvent);
|
|
167
|
-
if (layerStack.isInBranch(target))
|
|
168
|
-
return;
|
|
169
|
-
options.onFocusOutside?.(event);
|
|
170
|
-
options.onInteractOutside?.(event);
|
|
171
|
-
if (event.defaultPrevented)
|
|
172
|
-
return;
|
|
173
|
-
if (debug) {
|
|
174
|
-
console.log("onFocusOutside:", event.detail.originalEvent);
|
|
175
|
-
}
|
|
176
|
-
onDismiss?.();
|
|
177
|
-
}
|
|
178
|
-
function onEscapeKeyDown(event) {
|
|
179
|
-
if (!layerStack.isTopMost(node))
|
|
180
|
-
return;
|
|
181
|
-
options.onEscapeKeyDown?.(event);
|
|
182
|
-
if (!event.defaultPrevented && onDismiss) {
|
|
183
|
-
event.preventDefault();
|
|
184
|
-
onDismiss();
|
|
185
|
-
}
|
|
186
|
-
}
|
|
187
|
-
function exclude(target) {
|
|
188
|
-
if (!node)
|
|
189
|
-
return false;
|
|
190
|
-
const containers = typeof excludeContainers === "function" ? excludeContainers() : excludeContainers;
|
|
191
|
-
const _containers = Array.isArray(containers) ? containers : [containers];
|
|
192
|
-
return _containers.some((node2) => (0, import_dom_query4.contains)(node2, target)) || layerStack.isInNestedLayer(node, target);
|
|
193
|
-
}
|
|
194
|
-
const cleanups = [
|
|
195
|
-
pointerBlocking ? disablePointerEventsOutside(node) : void 0,
|
|
196
|
-
trackEscapeKeydown(node, onEscapeKeyDown),
|
|
197
|
-
(0, import_interact_outside.trackInteractOutside)(node, { exclude, onFocusOutside, onPointerDownOutside, defer: options.defer })
|
|
198
|
-
];
|
|
199
|
-
return () => {
|
|
200
|
-
layerStack.remove(node);
|
|
201
|
-
assignPointerEventToLayers();
|
|
202
|
-
clearPointerEvent(node);
|
|
203
|
-
cleanups.forEach((fn) => fn?.());
|
|
204
|
-
};
|
|
205
|
-
}
|
|
206
|
-
function trackDismissableElement(nodeOrFn, options) {
|
|
207
|
-
const { defer } = options;
|
|
208
|
-
const func = defer ? import_dom_query4.raf : (v) => v();
|
|
209
|
-
const cleanups = [];
|
|
210
|
-
cleanups.push(
|
|
211
|
-
func(() => {
|
|
212
|
-
const node = (0, import_utils.isFunction)(nodeOrFn) ? nodeOrFn() : nodeOrFn;
|
|
213
|
-
cleanups.push(trackDismissableElementImpl(node, options));
|
|
214
|
-
})
|
|
215
|
-
);
|
|
216
|
-
return () => {
|
|
217
|
-
cleanups.forEach((fn) => fn?.());
|
|
218
|
-
};
|
|
219
|
-
}
|
|
220
|
-
function trackDismissableBranch(nodeOrFn, options = {}) {
|
|
221
|
-
const { defer } = options;
|
|
222
|
-
const func = defer ? import_dom_query4.raf : (v) => v();
|
|
223
|
-
const cleanups = [];
|
|
224
|
-
cleanups.push(
|
|
225
|
-
func(() => {
|
|
226
|
-
const node = (0, import_utils.isFunction)(nodeOrFn) ? nodeOrFn() : nodeOrFn;
|
|
227
|
-
if (!node) {
|
|
228
|
-
(0, import_utils.warn)("[@zag-js/dismissable] branch node is `null` or `undefined`");
|
|
229
|
-
return;
|
|
230
|
-
}
|
|
231
|
-
layerStack.addBranch(node);
|
|
232
|
-
cleanups.push(() => {
|
|
233
|
-
layerStack.removeBranch(node);
|
|
234
|
-
});
|
|
235
|
-
})
|
|
236
|
-
);
|
|
237
|
-
return () => {
|
|
238
|
-
cleanups.forEach((fn) => fn?.());
|
|
239
|
-
};
|
|
240
|
-
}
|
|
241
|
-
// Annotate the CommonJS export names for ESM import in node:
|
|
242
|
-
0 && (module.exports = {
|
|
243
|
-
trackDismissableBranch,
|
|
244
|
-
trackDismissableElement
|
|
245
|
-
});
|
|
1
|
+
"use strict";var __defProp=Object.defineProperty;var __getOwnPropDesc=Object.getOwnPropertyDescriptor;var __getOwnPropNames=Object.getOwnPropertyNames;var __hasOwnProp=Object.prototype.hasOwnProperty;var __export=(target,all)=>{for(var name in all)__defProp(target,name,{get:all[name],enumerable:true})};var __copyProps=(to,from,except,desc)=>{if(from&&typeof from==="object"||typeof from==="function"){for(let key of __getOwnPropNames(from))if(!__hasOwnProp.call(to,key)&&key!==except)__defProp(to,key,{get:()=>from[key],enumerable:!(desc=__getOwnPropDesc(from,key))||desc.enumerable})}return to};var __toCommonJS=mod=>__copyProps(__defProp({},"__esModule",{value:true}),mod);var src_exports={};__export(src_exports,{trackDismissableBranch:()=>trackDismissableBranch,trackDismissableElement:()=>trackDismissableElement});module.exports=__toCommonJS(src_exports);var import_dom_query4=require("@zag-js/dom-query");var import_interact_outside=require("@zag-js/interact-outside");var import_utils=require("@zag-js/utils");var import_dom_event=require("@zag-js/dom-event");var import_dom_query=require("@zag-js/dom-query");function trackEscapeKeydown(node,fn){const handleKeyDown=event=>{if(event.key!=="Escape")return;if(event.isComposing)return;fn?.(event)};return(0,import_dom_event.addDomEvent)((0,import_dom_query.getDocument)(node),"keydown",handleKeyDown,{capture:true})}var import_dom_query2=require("@zag-js/dom-query");var layerStack={layers:[],branches:[],count(){return this.layers.length},pointerBlockingLayers(){return this.layers.filter(layer=>layer.pointerBlocking)},topMostPointerBlockingLayer(){return[...this.pointerBlockingLayers()].slice(-1)[0]},hasPointerBlockingLayer(){return this.pointerBlockingLayers().length>0},isBelowPointerBlockingLayer(node){const index=this.indexOf(node);const highestBlockingIndex=this.topMostPointerBlockingLayer()?this.indexOf(this.topMostPointerBlockingLayer()?.node):-1;return index<highestBlockingIndex},isTopMost(node){const layer=this.layers[this.count()-1];return layer?.node===node},getNestedLayers(node){return Array.from(this.layers).slice(this.indexOf(node)+1)},isInNestedLayer(node,target){return this.getNestedLayers(node).some(layer=>(0,import_dom_query2.contains)(layer.node,target))},isInBranch(target){return Array.from(this.branches).some(branch=>(0,import_dom_query2.contains)(branch,target))},add(layer){const num=this.layers.push(layer);layer.node.style.setProperty("--layer-index",`${num}`)},addBranch(node){this.branches.push(node)},remove(node){const index=this.indexOf(node);if(index<0)return;if(index<this.count()-1){const _layers=this.getNestedLayers(node);_layers.forEach(layer=>layer.dismiss())}this.layers.splice(index,1);node.style.removeProperty("--layer-index")},removeBranch(node){const index=this.branches.indexOf(node);if(index>=0)this.branches.splice(index,1)},indexOf(node){return this.layers.findIndex(layer=>layer.node===node)},dismiss(node){this.layers[this.indexOf(node)]?.dismiss()},clear(){this.remove(this.layers[0].node)}};var import_dom_query3=require("@zag-js/dom-query");var originalBodyPointerEvents;function assignPointerEventToLayers(){layerStack.layers.forEach(({node})=>{node.style.pointerEvents=layerStack.isBelowPointerBlockingLayer(node)?"none":"auto"})}function clearPointerEvent(node){node.style.pointerEvents=""}function disablePointerEventsOutside(node,peristentElements){const doc=(0,import_dom_query3.getDocument)(node);const cleanups=[];if(layerStack.hasPointerBlockingLayer()&&!doc.body.hasAttribute("data-inert")){originalBodyPointerEvents=document.body.style.pointerEvents;queueMicrotask(()=>{doc.body.style.pointerEvents="none";doc.body.setAttribute("data-inert","")})}if(peristentElements){const persistedCleanup=(0,import_dom_query3.waitForElements)(peristentElements,el=>{cleanups.push((0,import_dom_query3.setStyle)(el,{pointerEvents:"auto"}))});cleanups.push(persistedCleanup)}return()=>{if(layerStack.hasPointerBlockingLayer())return;queueMicrotask(()=>{doc.body.style.pointerEvents=originalBodyPointerEvents;doc.body.removeAttribute("data-inert");if(doc.body.style.length===0)doc.body.removeAttribute("style")});cleanups.forEach(fn=>fn())}}function trackDismissableElementImpl(node,options){if(!node){(0,import_utils.warn)("[@zag-js/dismissable] node is `null` or `undefined`");return}const{onDismiss,pointerBlocking,exclude:excludeContainers,debug}=options;const layer={dismiss:onDismiss,node,pointerBlocking};layerStack.add(layer);assignPointerEventToLayers();function onPointerDownOutside(event){const target=(0,import_dom_query4.getEventTarget)(event.detail.originalEvent);if(layerStack.isBelowPointerBlockingLayer(node)||layerStack.isInBranch(target))return;options.onPointerDownOutside?.(event);options.onInteractOutside?.(event);if(event.defaultPrevented)return;if(debug){console.log("onPointerDownOutside:",event.detail.originalEvent)}onDismiss?.()}function onFocusOutside(event){const target=(0,import_dom_query4.getEventTarget)(event.detail.originalEvent);if(layerStack.isInBranch(target))return;options.onFocusOutside?.(event);options.onInteractOutside?.(event);if(event.defaultPrevented)return;if(debug){console.log("onFocusOutside:",event.detail.originalEvent)}onDismiss?.()}function onEscapeKeyDown(event){if(!layerStack.isTopMost(node))return;options.onEscapeKeyDown?.(event);if(!event.defaultPrevented&&onDismiss){event.preventDefault();onDismiss()}}function exclude(target){if(!node)return false;const containers=typeof excludeContainers==="function"?excludeContainers():excludeContainers;const _containers=Array.isArray(containers)?containers:[containers];const persistentElements=options.persistentElements?.map(fn=>fn()).filter(import_dom_query4.isHTMLElement);if(persistentElements)_containers.push(...persistentElements);return _containers.some(node2=>(0,import_dom_query4.contains)(node2,target))||layerStack.isInNestedLayer(node,target)}const cleanups=[pointerBlocking?disablePointerEventsOutside(node,options.persistentElements):void 0,trackEscapeKeydown(node,onEscapeKeyDown),(0,import_interact_outside.trackInteractOutside)(node,{exclude,onFocusOutside,onPointerDownOutside,defer:options.defer})];return()=>{layerStack.remove(node);assignPointerEventToLayers();clearPointerEvent(node);cleanups.forEach(fn=>fn?.())}}function trackDismissableElement(nodeOrFn,options){const{defer}=options;const func=defer?import_dom_query4.raf:v=>v();const cleanups=[];cleanups.push(func(()=>{const node=(0,import_utils.isFunction)(nodeOrFn)?nodeOrFn():nodeOrFn;cleanups.push(trackDismissableElementImpl(node,options))}));return()=>{cleanups.forEach(fn=>fn?.())}}function trackDismissableBranch(nodeOrFn,options={}){const{defer}=options;const func=defer?import_dom_query4.raf:v=>v();const cleanups=[];cleanups.push(func(()=>{const node=(0,import_utils.isFunction)(nodeOrFn)?nodeOrFn():nodeOrFn;if(!node){(0,import_utils.warn)("[@zag-js/dismissable] branch node is `null` or `undefined`");return}layerStack.addBranch(node);cleanups.push(()=>{layerStack.removeBranch(node)})}));return()=>{cleanups.forEach(fn=>fn?.())}}0&&(module.exports={trackDismissableBranch,trackDismissableElement});
|
|
246
2
|
//# sourceMappingURL=index.js.map
|
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/index.ts","../src/dismissable-layer.ts","../src/escape-keydown.ts","../src/layer-stack.ts","../src/pointer-event-outside.ts"],"sourcesContent":["export * from \"./dismissable-layer\"\nexport type {\n InteractOutsideEvent,\n PointerDownOutsideEvent,\n FocusOutsideEvent,\n InteractOutsideHandlers,\n} from \"@zag-js/interact-outside\"\n","import { contains, getEventTarget, raf } from \"@zag-js/dom-query\"\nimport {\n trackInteractOutside,\n type FocusOutsideEvent,\n type InteractOutsideHandlers,\n type PointerDownOutsideEvent,\n} from \"@zag-js/interact-outside\"\nimport { isFunction, warn, type MaybeFunction } from \"@zag-js/utils\"\nimport { trackEscapeKeydown } from \"./escape-keydown\"\nimport { layerStack, type Layer } from \"./layer-stack\"\nimport { assignPointerEventToLayers, clearPointerEvent, disablePointerEventsOutside } from \"./pointer-event-outside\"\n\ntype MaybeElement = HTMLElement | null\ntype Container = MaybeElement | Array<MaybeElement>\ntype NodeOrFn = MaybeFunction<MaybeElement>\n\nexport interface DismissableElementHandlers extends InteractOutsideHandlers {\n /**\n * Function called when the escape key is pressed\n */\n onEscapeKeyDown?: (event: KeyboardEvent) => void\n}\n\nexport interface DismissableElementOptions extends DismissableElementHandlers {\n /**\n * Whether to log debug information\n */\n debug?: boolean\n /**\n * Whether to block pointer events outside the dismissable element\n */\n pointerBlocking?: boolean\n /**\n * Function called when the dismissable element is dismissed\n */\n onDismiss: VoidFunction\n /**\n * Exclude containers from the interact outside event\n */\n exclude?: MaybeFunction<Container>\n /**\n * Defer the interact outside event to the next frame\n */\n defer?: boolean\n}\n\nfunction trackDismissableElementImpl(node: MaybeElement, options: DismissableElementOptions) {\n if (!node) {\n warn(\"[@zag-js/dismissable] node is `null` or `undefined`\")\n return\n }\n\n const { onDismiss, pointerBlocking, exclude: excludeContainers, debug } = options\n\n const layer: Layer = { dismiss: onDismiss, node, pointerBlocking }\n\n layerStack.add(layer)\n assignPointerEventToLayers()\n\n function onPointerDownOutside(event: PointerDownOutsideEvent) {\n const target = getEventTarget(event.detail.originalEvent)\n if (layerStack.isBelowPointerBlockingLayer(node!) || layerStack.isInBranch(target)) return\n options.onPointerDownOutside?.(event)\n options.onInteractOutside?.(event)\n if (event.defaultPrevented) return\n if (debug) {\n console.log(\"onPointerDownOutside:\", event.detail.originalEvent)\n }\n onDismiss?.()\n }\n\n function onFocusOutside(event: FocusOutsideEvent) {\n const target = getEventTarget(event.detail.originalEvent)\n if (layerStack.isInBranch(target)) return\n options.onFocusOutside?.(event)\n options.onInteractOutside?.(event)\n if (event.defaultPrevented) return\n if (debug) {\n console.log(\"onFocusOutside:\", event.detail.originalEvent)\n }\n onDismiss?.()\n }\n\n function onEscapeKeyDown(event: KeyboardEvent) {\n if (!layerStack.isTopMost(node!)) return\n options.onEscapeKeyDown?.(event)\n if (!event.defaultPrevented && onDismiss) {\n event.preventDefault()\n onDismiss()\n }\n }\n\n function exclude(target: Element) {\n if (!node) return false\n const containers = typeof excludeContainers === \"function\" ? excludeContainers() : excludeContainers\n const _containers = Array.isArray(containers) ? containers : [containers]\n return _containers.some((node) => contains(node, target)) || layerStack.isInNestedLayer(node, target)\n }\n\n const cleanups = [\n pointerBlocking ? disablePointerEventsOutside(node) : undefined,\n trackEscapeKeydown(node, onEscapeKeyDown),\n trackInteractOutside(node, { exclude, onFocusOutside, onPointerDownOutside, defer: options.defer }),\n ]\n\n return () => {\n layerStack.remove(node!)\n // re-assign pointer event to remaining layers\n assignPointerEventToLayers()\n // remove pointer event from removed layer\n clearPointerEvent(node!)\n cleanups.forEach((fn) => fn?.())\n }\n}\n\nexport function trackDismissableElement(nodeOrFn: NodeOrFn, options: DismissableElementOptions) {\n const { defer } = options\n const func = defer ? raf : (v: any) => v()\n const cleanups: (VoidFunction | undefined)[] = []\n cleanups.push(\n func(() => {\n const node = isFunction(nodeOrFn) ? nodeOrFn() : nodeOrFn\n cleanups.push(trackDismissableElementImpl(node, options))\n }),\n )\n return () => {\n cleanups.forEach((fn) => fn?.())\n }\n}\n\nexport function trackDismissableBranch(nodeOrFn: NodeOrFn, options: { defer?: boolean } = {}) {\n const { defer } = options\n const func = defer ? raf : (v: any) => v()\n const cleanups: (VoidFunction | undefined)[] = []\n\n cleanups.push(\n func(() => {\n const node = isFunction(nodeOrFn) ? nodeOrFn() : nodeOrFn\n if (!node) {\n warn(\"[@zag-js/dismissable] branch node is `null` or `undefined`\")\n return\n }\n layerStack.addBranch(node)\n cleanups.push(() => {\n layerStack.removeBranch(node)\n })\n }),\n )\n\n return () => {\n cleanups.forEach((fn) => fn?.())\n }\n}\n","import { addDomEvent } from \"@zag-js/dom-event\"\nimport { getDocument } from \"@zag-js/dom-query\"\n\nexport function trackEscapeKeydown(node: HTMLElement, fn?: (event: KeyboardEvent) => void) {\n const handleKeyDown = (event: KeyboardEvent) => {\n if (event.key === \"Escape\") fn?.(event)\n }\n return addDomEvent(getDocument(node), \"keydown\", handleKeyDown, { capture: true })\n}\n","import { contains } from \"@zag-js/dom-query\"\n\nexport interface Layer {\n dismiss: VoidFunction\n node: HTMLElement\n pointerBlocking?: boolean\n}\n\nexport const layerStack = {\n layers: [] as Layer[],\n branches: [] as HTMLElement[],\n count(): number {\n return this.layers.length\n },\n pointerBlockingLayers(): Layer[] {\n return this.layers.filter((layer) => layer.pointerBlocking)\n },\n topMostPointerBlockingLayer(): Layer | undefined {\n return [...this.pointerBlockingLayers()].slice(-1)[0]\n },\n hasPointerBlockingLayer(): boolean {\n return this.pointerBlockingLayers().length > 0\n },\n isBelowPointerBlockingLayer(node: HTMLElement) {\n const index = this.indexOf(node)\n const highestBlockingIndex = this.topMostPointerBlockingLayer()\n ? this.indexOf(this.topMostPointerBlockingLayer()?.node)\n : -1\n return index < highestBlockingIndex\n },\n isTopMost(node: HTMLElement | null) {\n const layer = this.layers[this.count() - 1]\n return layer?.node === node\n },\n getNestedLayers(node: HTMLElement) {\n return Array.from(this.layers).slice(this.indexOf(node) + 1)\n },\n isInNestedLayer(node: HTMLElement, target: HTMLElement | EventTarget | null) {\n return this.getNestedLayers(node).some((layer) => contains(layer.node, target))\n },\n isInBranch(target: HTMLElement | EventTarget | null) {\n return Array.from(this.branches).some((branch) => contains(branch, target))\n },\n add(layer: Layer) {\n const num = this.layers.push(layer)\n layer.node.style.setProperty(\"--layer-index\", `${num}`)\n },\n addBranch(node: HTMLElement) {\n this.branches.push(node)\n },\n remove(node: HTMLElement) {\n const index = this.indexOf(node)\n if (index < 0) return\n\n // dismiss nested layers\n if (index < this.count() - 1) {\n const _layers = this.getNestedLayers(node)\n _layers.forEach((layer) => layer.dismiss())\n }\n // remove this layer\n this.layers.splice(index, 1)\n node.style.removeProperty(\"--layer-index\")\n },\n removeBranch(node: HTMLElement) {\n const index = this.branches.indexOf(node)\n if (index >= 0) this.branches.splice(index, 1)\n },\n indexOf(node: HTMLElement | undefined) {\n return this.layers.findIndex((layer) => layer.node === node)\n },\n dismiss(node: HTMLElement) {\n this.layers[this.indexOf(node)]?.dismiss()\n },\n clear() {\n this.remove(this.layers[0].node)\n },\n}\n","import { getDocument } from \"@zag-js/dom-query\"\nimport { layerStack } from \"./layer-stack\"\n\nlet originalBodyPointerEvents: string\n\nexport function assignPointerEventToLayers() {\n layerStack.layers.forEach(({ node }) => {\n node.style.pointerEvents = layerStack.isBelowPointerBlockingLayer(node) ? \"none\" : \"auto\"\n })\n}\n\nexport function clearPointerEvent(node: HTMLElement) {\n node.style.pointerEvents = \"\"\n}\n\nconst DATA_ATTR = \"data-inert\"\n\nexport function disablePointerEventsOutside(node: HTMLElement) {\n const doc = getDocument(node)\n\n if (layerStack.hasPointerBlockingLayer() && !doc.body.hasAttribute(DATA_ATTR)) {\n originalBodyPointerEvents = document.body.style.pointerEvents\n doc.body.style.pointerEvents = \"none\"\n doc.body.setAttribute(DATA_ATTR, \"\")\n }\n\n return () => {\n if (layerStack.hasPointerBlockingLayer()) return\n doc.body.style.pointerEvents = originalBodyPointerEvents\n doc.body.removeAttribute(DATA_ATTR)\n if (doc.body.style.length === 0) doc.body.removeAttribute(\"style\")\n }\n}\n"],"mappings":"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","names":["import_dom_query","import_dom_query","import_dom_query","node"]}
|
|
1
|
+
{"version":3,"sources":["../src/index.ts","../src/dismissable-layer.ts","../src/escape-keydown.ts","../src/layer-stack.ts","../src/pointer-event-outside.ts"],"sourcesContent":["export type {\n FocusOutsideEvent,\n InteractOutsideEvent,\n InteractOutsideHandlers,\n PointerDownOutsideEvent,\n} from \"@zag-js/interact-outside\"\nexport * from \"./dismissable-layer\"\n","import { contains, getEventTarget, isHTMLElement, raf } from \"@zag-js/dom-query\"\nimport {\n trackInteractOutside,\n type FocusOutsideEvent,\n type InteractOutsideHandlers,\n type PointerDownOutsideEvent,\n} from \"@zag-js/interact-outside\"\nimport { isFunction, warn, type MaybeFunction } from \"@zag-js/utils\"\nimport { trackEscapeKeydown } from \"./escape-keydown\"\nimport { layerStack, type Layer } from \"./layer-stack\"\nimport { assignPointerEventToLayers, clearPointerEvent, disablePointerEventsOutside } from \"./pointer-event-outside\"\n\ntype MaybeElement = HTMLElement | null\ntype Container = MaybeElement | Array<MaybeElement>\ntype NodeOrFn = MaybeFunction<MaybeElement>\n\nexport interface DismissableElementHandlers extends InteractOutsideHandlers {\n /**\n * Function called when the escape key is pressed\n */\n onEscapeKeyDown?: (event: KeyboardEvent) => void\n}\n\nexport interface PersistentElementOptions {\n /**\n * Returns the persistent elements that:\n * - should not have pointer-events disabled\n * - should not trigger the dismiss event\n */\n persistentElements?: Array<() => Element | null>\n}\n\nexport interface DismissableElementOptions extends DismissableElementHandlers, PersistentElementOptions {\n /**\n * Whether to log debug information\n */\n debug?: boolean\n /**\n * Whether to block pointer events outside the dismissable element\n */\n pointerBlocking?: boolean\n /**\n * Function called when the dismissable element is dismissed\n */\n onDismiss: VoidFunction\n /**\n * Exclude containers from the interact outside event\n */\n exclude?: MaybeFunction<Container>\n /**\n * Defer the interact outside event to the next frame\n */\n defer?: boolean\n}\n\nfunction trackDismissableElementImpl(node: MaybeElement, options: DismissableElementOptions) {\n if (!node) {\n warn(\"[@zag-js/dismissable] node is `null` or `undefined`\")\n return\n }\n\n const { onDismiss, pointerBlocking, exclude: excludeContainers, debug } = options\n\n const layer: Layer = { dismiss: onDismiss, node, pointerBlocking }\n\n layerStack.add(layer)\n assignPointerEventToLayers()\n\n function onPointerDownOutside(event: PointerDownOutsideEvent) {\n const target = getEventTarget(event.detail.originalEvent)\n if (layerStack.isBelowPointerBlockingLayer(node!) || layerStack.isInBranch(target)) return\n options.onPointerDownOutside?.(event)\n options.onInteractOutside?.(event)\n if (event.defaultPrevented) return\n if (debug) {\n console.log(\"onPointerDownOutside:\", event.detail.originalEvent)\n }\n onDismiss?.()\n }\n\n function onFocusOutside(event: FocusOutsideEvent) {\n const target = getEventTarget(event.detail.originalEvent)\n if (layerStack.isInBranch(target)) return\n options.onFocusOutside?.(event)\n options.onInteractOutside?.(event)\n if (event.defaultPrevented) return\n if (debug) {\n console.log(\"onFocusOutside:\", event.detail.originalEvent)\n }\n onDismiss?.()\n }\n\n function onEscapeKeyDown(event: KeyboardEvent) {\n if (!layerStack.isTopMost(node!)) return\n options.onEscapeKeyDown?.(event)\n if (!event.defaultPrevented && onDismiss) {\n event.preventDefault()\n onDismiss()\n }\n }\n\n function exclude(target: Element) {\n if (!node) return false\n const containers = typeof excludeContainers === \"function\" ? excludeContainers() : excludeContainers\n const _containers = Array.isArray(containers) ? containers : [containers]\n const persistentElements = options.persistentElements?.map((fn) => fn()).filter(isHTMLElement)\n if (persistentElements) _containers.push(...persistentElements)\n return _containers.some((node) => contains(node, target)) || layerStack.isInNestedLayer(node, target)\n }\n\n const cleanups = [\n pointerBlocking ? disablePointerEventsOutside(node, options.persistentElements) : undefined,\n trackEscapeKeydown(node, onEscapeKeyDown),\n trackInteractOutside(node, { exclude, onFocusOutside, onPointerDownOutside, defer: options.defer }),\n ]\n\n return () => {\n layerStack.remove(node!)\n // re-assign pointer event to remaining layers\n assignPointerEventToLayers()\n // remove pointer event from removed layer\n clearPointerEvent(node!)\n cleanups.forEach((fn) => fn?.())\n }\n}\n\nexport function trackDismissableElement(nodeOrFn: NodeOrFn, options: DismissableElementOptions) {\n const { defer } = options\n const func = defer ? raf : (v: any) => v()\n const cleanups: (VoidFunction | undefined)[] = []\n cleanups.push(\n func(() => {\n const node = isFunction(nodeOrFn) ? nodeOrFn() : nodeOrFn\n cleanups.push(trackDismissableElementImpl(node, options))\n }),\n )\n return () => {\n cleanups.forEach((fn) => fn?.())\n }\n}\n\nexport function trackDismissableBranch(nodeOrFn: NodeOrFn, options: { defer?: boolean } = {}) {\n const { defer } = options\n const func = defer ? raf : (v: any) => v()\n const cleanups: (VoidFunction | undefined)[] = []\n\n cleanups.push(\n func(() => {\n const node = isFunction(nodeOrFn) ? nodeOrFn() : nodeOrFn\n if (!node) {\n warn(\"[@zag-js/dismissable] branch node is `null` or `undefined`\")\n return\n }\n layerStack.addBranch(node)\n cleanups.push(() => {\n layerStack.removeBranch(node)\n })\n }),\n )\n\n return () => {\n cleanups.forEach((fn) => fn?.())\n }\n}\n","import { addDomEvent } from \"@zag-js/dom-event\"\nimport { getDocument } from \"@zag-js/dom-query\"\n\nexport function trackEscapeKeydown(node: HTMLElement, fn?: (event: KeyboardEvent) => void) {\n const handleKeyDown = (event: KeyboardEvent) => {\n if (event.key !== \"Escape\") return\n if (event.isComposing) return\n fn?.(event)\n }\n\n return addDomEvent(getDocument(node), \"keydown\", handleKeyDown, { capture: true })\n}\n","import { contains } from \"@zag-js/dom-query\"\n\nexport interface Layer {\n dismiss: VoidFunction\n node: HTMLElement\n pointerBlocking?: boolean\n}\n\nexport const layerStack = {\n layers: [] as Layer[],\n branches: [] as HTMLElement[],\n count(): number {\n return this.layers.length\n },\n pointerBlockingLayers(): Layer[] {\n return this.layers.filter((layer) => layer.pointerBlocking)\n },\n topMostPointerBlockingLayer(): Layer | undefined {\n return [...this.pointerBlockingLayers()].slice(-1)[0]\n },\n hasPointerBlockingLayer(): boolean {\n return this.pointerBlockingLayers().length > 0\n },\n isBelowPointerBlockingLayer(node: HTMLElement) {\n const index = this.indexOf(node)\n const highestBlockingIndex = this.topMostPointerBlockingLayer()\n ? this.indexOf(this.topMostPointerBlockingLayer()?.node)\n : -1\n return index < highestBlockingIndex\n },\n isTopMost(node: HTMLElement | null) {\n const layer = this.layers[this.count() - 1]\n return layer?.node === node\n },\n getNestedLayers(node: HTMLElement) {\n return Array.from(this.layers).slice(this.indexOf(node) + 1)\n },\n isInNestedLayer(node: HTMLElement, target: HTMLElement | EventTarget | null) {\n return this.getNestedLayers(node).some((layer) => contains(layer.node, target))\n },\n isInBranch(target: HTMLElement | EventTarget | null) {\n return Array.from(this.branches).some((branch) => contains(branch, target))\n },\n add(layer: Layer) {\n const num = this.layers.push(layer)\n layer.node.style.setProperty(\"--layer-index\", `${num}`)\n },\n addBranch(node: HTMLElement) {\n this.branches.push(node)\n },\n remove(node: HTMLElement) {\n const index = this.indexOf(node)\n if (index < 0) return\n\n // dismiss nested layers\n if (index < this.count() - 1) {\n const _layers = this.getNestedLayers(node)\n _layers.forEach((layer) => layer.dismiss())\n }\n // remove this layer\n this.layers.splice(index, 1)\n node.style.removeProperty(\"--layer-index\")\n },\n removeBranch(node: HTMLElement) {\n const index = this.branches.indexOf(node)\n if (index >= 0) this.branches.splice(index, 1)\n },\n indexOf(node: HTMLElement | undefined) {\n return this.layers.findIndex((layer) => layer.node === node)\n },\n dismiss(node: HTMLElement) {\n this.layers[this.indexOf(node)]?.dismiss()\n },\n clear() {\n this.remove(this.layers[0].node)\n },\n}\n","import { getDocument, setStyle, waitForElements } from \"@zag-js/dom-query\"\nimport { layerStack } from \"./layer-stack\"\n\nlet originalBodyPointerEvents: string\n\nexport function assignPointerEventToLayers() {\n layerStack.layers.forEach(({ node }) => {\n node.style.pointerEvents = layerStack.isBelowPointerBlockingLayer(node) ? \"none\" : \"auto\"\n })\n}\n\nexport function clearPointerEvent(node: HTMLElement) {\n node.style.pointerEvents = \"\"\n}\n\nexport function disablePointerEventsOutside(node: HTMLElement, peristentElements?: Array<() => Element | null>) {\n const doc = getDocument(node)\n\n const cleanups: VoidFunction[] = []\n\n if (layerStack.hasPointerBlockingLayer() && !doc.body.hasAttribute(\"data-inert\")) {\n originalBodyPointerEvents = document.body.style.pointerEvents\n queueMicrotask(() => {\n doc.body.style.pointerEvents = \"none\"\n doc.body.setAttribute(\"data-inert\", \"\")\n })\n }\n\n if (peristentElements) {\n const persistedCleanup = waitForElements(peristentElements, (el) => {\n cleanups.push(setStyle(el, { pointerEvents: \"auto\" }))\n })\n cleanups.push(persistedCleanup)\n }\n\n return () => {\n if (layerStack.hasPointerBlockingLayer()) return\n queueMicrotask(() => {\n doc.body.style.pointerEvents = originalBodyPointerEvents\n doc.body.removeAttribute(\"data-inert\")\n if (doc.body.style.length === 0) doc.body.removeAttribute(\"style\")\n })\n cleanups.forEach((fn) => fn())\n }\n}\n"],"mappings":"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","names":["import_dom_query","import_dom_query","import_dom_query","node"]}
|
package/dist/index.mjs
CHANGED
|
@@ -1,220 +1,2 @@
|
|
|
1
|
-
|
|
2
|
-
import { contains as contains2, getEventTarget, raf } from "@zag-js/dom-query";
|
|
3
|
-
import {
|
|
4
|
-
trackInteractOutside
|
|
5
|
-
} from "@zag-js/interact-outside";
|
|
6
|
-
import { isFunction, warn } from "@zag-js/utils";
|
|
7
|
-
|
|
8
|
-
// src/escape-keydown.ts
|
|
9
|
-
import { addDomEvent } from "@zag-js/dom-event";
|
|
10
|
-
import { getDocument } from "@zag-js/dom-query";
|
|
11
|
-
function trackEscapeKeydown(node, fn) {
|
|
12
|
-
const handleKeyDown = (event) => {
|
|
13
|
-
if (event.key === "Escape")
|
|
14
|
-
fn?.(event);
|
|
15
|
-
};
|
|
16
|
-
return addDomEvent(getDocument(node), "keydown", handleKeyDown, { capture: true });
|
|
17
|
-
}
|
|
18
|
-
|
|
19
|
-
// src/layer-stack.ts
|
|
20
|
-
import { contains } from "@zag-js/dom-query";
|
|
21
|
-
var layerStack = {
|
|
22
|
-
layers: [],
|
|
23
|
-
branches: [],
|
|
24
|
-
count() {
|
|
25
|
-
return this.layers.length;
|
|
26
|
-
},
|
|
27
|
-
pointerBlockingLayers() {
|
|
28
|
-
return this.layers.filter((layer) => layer.pointerBlocking);
|
|
29
|
-
},
|
|
30
|
-
topMostPointerBlockingLayer() {
|
|
31
|
-
return [...this.pointerBlockingLayers()].slice(-1)[0];
|
|
32
|
-
},
|
|
33
|
-
hasPointerBlockingLayer() {
|
|
34
|
-
return this.pointerBlockingLayers().length > 0;
|
|
35
|
-
},
|
|
36
|
-
isBelowPointerBlockingLayer(node) {
|
|
37
|
-
const index = this.indexOf(node);
|
|
38
|
-
const highestBlockingIndex = this.topMostPointerBlockingLayer() ? this.indexOf(this.topMostPointerBlockingLayer()?.node) : -1;
|
|
39
|
-
return index < highestBlockingIndex;
|
|
40
|
-
},
|
|
41
|
-
isTopMost(node) {
|
|
42
|
-
const layer = this.layers[this.count() - 1];
|
|
43
|
-
return layer?.node === node;
|
|
44
|
-
},
|
|
45
|
-
getNestedLayers(node) {
|
|
46
|
-
return Array.from(this.layers).slice(this.indexOf(node) + 1);
|
|
47
|
-
},
|
|
48
|
-
isInNestedLayer(node, target) {
|
|
49
|
-
return this.getNestedLayers(node).some((layer) => contains(layer.node, target));
|
|
50
|
-
},
|
|
51
|
-
isInBranch(target) {
|
|
52
|
-
return Array.from(this.branches).some((branch) => contains(branch, target));
|
|
53
|
-
},
|
|
54
|
-
add(layer) {
|
|
55
|
-
const num = this.layers.push(layer);
|
|
56
|
-
layer.node.style.setProperty("--layer-index", `${num}`);
|
|
57
|
-
},
|
|
58
|
-
addBranch(node) {
|
|
59
|
-
this.branches.push(node);
|
|
60
|
-
},
|
|
61
|
-
remove(node) {
|
|
62
|
-
const index = this.indexOf(node);
|
|
63
|
-
if (index < 0)
|
|
64
|
-
return;
|
|
65
|
-
if (index < this.count() - 1) {
|
|
66
|
-
const _layers = this.getNestedLayers(node);
|
|
67
|
-
_layers.forEach((layer) => layer.dismiss());
|
|
68
|
-
}
|
|
69
|
-
this.layers.splice(index, 1);
|
|
70
|
-
node.style.removeProperty("--layer-index");
|
|
71
|
-
},
|
|
72
|
-
removeBranch(node) {
|
|
73
|
-
const index = this.branches.indexOf(node);
|
|
74
|
-
if (index >= 0)
|
|
75
|
-
this.branches.splice(index, 1);
|
|
76
|
-
},
|
|
77
|
-
indexOf(node) {
|
|
78
|
-
return this.layers.findIndex((layer) => layer.node === node);
|
|
79
|
-
},
|
|
80
|
-
dismiss(node) {
|
|
81
|
-
this.layers[this.indexOf(node)]?.dismiss();
|
|
82
|
-
},
|
|
83
|
-
clear() {
|
|
84
|
-
this.remove(this.layers[0].node);
|
|
85
|
-
}
|
|
86
|
-
};
|
|
87
|
-
|
|
88
|
-
// src/pointer-event-outside.ts
|
|
89
|
-
import { getDocument as getDocument2 } from "@zag-js/dom-query";
|
|
90
|
-
var originalBodyPointerEvents;
|
|
91
|
-
function assignPointerEventToLayers() {
|
|
92
|
-
layerStack.layers.forEach(({ node }) => {
|
|
93
|
-
node.style.pointerEvents = layerStack.isBelowPointerBlockingLayer(node) ? "none" : "auto";
|
|
94
|
-
});
|
|
95
|
-
}
|
|
96
|
-
function clearPointerEvent(node) {
|
|
97
|
-
node.style.pointerEvents = "";
|
|
98
|
-
}
|
|
99
|
-
var DATA_ATTR = "data-inert";
|
|
100
|
-
function disablePointerEventsOutside(node) {
|
|
101
|
-
const doc = getDocument2(node);
|
|
102
|
-
if (layerStack.hasPointerBlockingLayer() && !doc.body.hasAttribute(DATA_ATTR)) {
|
|
103
|
-
originalBodyPointerEvents = document.body.style.pointerEvents;
|
|
104
|
-
doc.body.style.pointerEvents = "none";
|
|
105
|
-
doc.body.setAttribute(DATA_ATTR, "");
|
|
106
|
-
}
|
|
107
|
-
return () => {
|
|
108
|
-
if (layerStack.hasPointerBlockingLayer())
|
|
109
|
-
return;
|
|
110
|
-
doc.body.style.pointerEvents = originalBodyPointerEvents;
|
|
111
|
-
doc.body.removeAttribute(DATA_ATTR);
|
|
112
|
-
if (doc.body.style.length === 0)
|
|
113
|
-
doc.body.removeAttribute("style");
|
|
114
|
-
};
|
|
115
|
-
}
|
|
116
|
-
|
|
117
|
-
// src/dismissable-layer.ts
|
|
118
|
-
function trackDismissableElementImpl(node, options) {
|
|
119
|
-
if (!node) {
|
|
120
|
-
warn("[@zag-js/dismissable] node is `null` or `undefined`");
|
|
121
|
-
return;
|
|
122
|
-
}
|
|
123
|
-
const { onDismiss, pointerBlocking, exclude: excludeContainers, debug } = options;
|
|
124
|
-
const layer = { dismiss: onDismiss, node, pointerBlocking };
|
|
125
|
-
layerStack.add(layer);
|
|
126
|
-
assignPointerEventToLayers();
|
|
127
|
-
function onPointerDownOutside(event) {
|
|
128
|
-
const target = getEventTarget(event.detail.originalEvent);
|
|
129
|
-
if (layerStack.isBelowPointerBlockingLayer(node) || layerStack.isInBranch(target))
|
|
130
|
-
return;
|
|
131
|
-
options.onPointerDownOutside?.(event);
|
|
132
|
-
options.onInteractOutside?.(event);
|
|
133
|
-
if (event.defaultPrevented)
|
|
134
|
-
return;
|
|
135
|
-
if (debug) {
|
|
136
|
-
console.log("onPointerDownOutside:", event.detail.originalEvent);
|
|
137
|
-
}
|
|
138
|
-
onDismiss?.();
|
|
139
|
-
}
|
|
140
|
-
function onFocusOutside(event) {
|
|
141
|
-
const target = getEventTarget(event.detail.originalEvent);
|
|
142
|
-
if (layerStack.isInBranch(target))
|
|
143
|
-
return;
|
|
144
|
-
options.onFocusOutside?.(event);
|
|
145
|
-
options.onInteractOutside?.(event);
|
|
146
|
-
if (event.defaultPrevented)
|
|
147
|
-
return;
|
|
148
|
-
if (debug) {
|
|
149
|
-
console.log("onFocusOutside:", event.detail.originalEvent);
|
|
150
|
-
}
|
|
151
|
-
onDismiss?.();
|
|
152
|
-
}
|
|
153
|
-
function onEscapeKeyDown(event) {
|
|
154
|
-
if (!layerStack.isTopMost(node))
|
|
155
|
-
return;
|
|
156
|
-
options.onEscapeKeyDown?.(event);
|
|
157
|
-
if (!event.defaultPrevented && onDismiss) {
|
|
158
|
-
event.preventDefault();
|
|
159
|
-
onDismiss();
|
|
160
|
-
}
|
|
161
|
-
}
|
|
162
|
-
function exclude(target) {
|
|
163
|
-
if (!node)
|
|
164
|
-
return false;
|
|
165
|
-
const containers = typeof excludeContainers === "function" ? excludeContainers() : excludeContainers;
|
|
166
|
-
const _containers = Array.isArray(containers) ? containers : [containers];
|
|
167
|
-
return _containers.some((node2) => contains2(node2, target)) || layerStack.isInNestedLayer(node, target);
|
|
168
|
-
}
|
|
169
|
-
const cleanups = [
|
|
170
|
-
pointerBlocking ? disablePointerEventsOutside(node) : void 0,
|
|
171
|
-
trackEscapeKeydown(node, onEscapeKeyDown),
|
|
172
|
-
trackInteractOutside(node, { exclude, onFocusOutside, onPointerDownOutside, defer: options.defer })
|
|
173
|
-
];
|
|
174
|
-
return () => {
|
|
175
|
-
layerStack.remove(node);
|
|
176
|
-
assignPointerEventToLayers();
|
|
177
|
-
clearPointerEvent(node);
|
|
178
|
-
cleanups.forEach((fn) => fn?.());
|
|
179
|
-
};
|
|
180
|
-
}
|
|
181
|
-
function trackDismissableElement(nodeOrFn, options) {
|
|
182
|
-
const { defer } = options;
|
|
183
|
-
const func = defer ? raf : (v) => v();
|
|
184
|
-
const cleanups = [];
|
|
185
|
-
cleanups.push(
|
|
186
|
-
func(() => {
|
|
187
|
-
const node = isFunction(nodeOrFn) ? nodeOrFn() : nodeOrFn;
|
|
188
|
-
cleanups.push(trackDismissableElementImpl(node, options));
|
|
189
|
-
})
|
|
190
|
-
);
|
|
191
|
-
return () => {
|
|
192
|
-
cleanups.forEach((fn) => fn?.());
|
|
193
|
-
};
|
|
194
|
-
}
|
|
195
|
-
function trackDismissableBranch(nodeOrFn, options = {}) {
|
|
196
|
-
const { defer } = options;
|
|
197
|
-
const func = defer ? raf : (v) => v();
|
|
198
|
-
const cleanups = [];
|
|
199
|
-
cleanups.push(
|
|
200
|
-
func(() => {
|
|
201
|
-
const node = isFunction(nodeOrFn) ? nodeOrFn() : nodeOrFn;
|
|
202
|
-
if (!node) {
|
|
203
|
-
warn("[@zag-js/dismissable] branch node is `null` or `undefined`");
|
|
204
|
-
return;
|
|
205
|
-
}
|
|
206
|
-
layerStack.addBranch(node);
|
|
207
|
-
cleanups.push(() => {
|
|
208
|
-
layerStack.removeBranch(node);
|
|
209
|
-
});
|
|
210
|
-
})
|
|
211
|
-
);
|
|
212
|
-
return () => {
|
|
213
|
-
cleanups.forEach((fn) => fn?.());
|
|
214
|
-
};
|
|
215
|
-
}
|
|
216
|
-
export {
|
|
217
|
-
trackDismissableBranch,
|
|
218
|
-
trackDismissableElement
|
|
219
|
-
};
|
|
1
|
+
import{contains as contains2,getEventTarget,isHTMLElement,raf}from"@zag-js/dom-query";import{trackInteractOutside}from"@zag-js/interact-outside";import{isFunction,warn}from"@zag-js/utils";import{addDomEvent}from"@zag-js/dom-event";import{getDocument}from"@zag-js/dom-query";function trackEscapeKeydown(node,fn){const handleKeyDown=event=>{if(event.key!=="Escape")return;if(event.isComposing)return;fn?.(event)};return addDomEvent(getDocument(node),"keydown",handleKeyDown,{capture:true})}import{contains}from"@zag-js/dom-query";var layerStack={layers:[],branches:[],count(){return this.layers.length},pointerBlockingLayers(){return this.layers.filter(layer=>layer.pointerBlocking)},topMostPointerBlockingLayer(){return[...this.pointerBlockingLayers()].slice(-1)[0]},hasPointerBlockingLayer(){return this.pointerBlockingLayers().length>0},isBelowPointerBlockingLayer(node){const index=this.indexOf(node);const highestBlockingIndex=this.topMostPointerBlockingLayer()?this.indexOf(this.topMostPointerBlockingLayer()?.node):-1;return index<highestBlockingIndex},isTopMost(node){const layer=this.layers[this.count()-1];return layer?.node===node},getNestedLayers(node){return Array.from(this.layers).slice(this.indexOf(node)+1)},isInNestedLayer(node,target){return this.getNestedLayers(node).some(layer=>contains(layer.node,target))},isInBranch(target){return Array.from(this.branches).some(branch=>contains(branch,target))},add(layer){const num=this.layers.push(layer);layer.node.style.setProperty("--layer-index",`${num}`)},addBranch(node){this.branches.push(node)},remove(node){const index=this.indexOf(node);if(index<0)return;if(index<this.count()-1){const _layers=this.getNestedLayers(node);_layers.forEach(layer=>layer.dismiss())}this.layers.splice(index,1);node.style.removeProperty("--layer-index")},removeBranch(node){const index=this.branches.indexOf(node);if(index>=0)this.branches.splice(index,1)},indexOf(node){return this.layers.findIndex(layer=>layer.node===node)},dismiss(node){this.layers[this.indexOf(node)]?.dismiss()},clear(){this.remove(this.layers[0].node)}};import{getDocument as getDocument2,setStyle,waitForElements}from"@zag-js/dom-query";var originalBodyPointerEvents;function assignPointerEventToLayers(){layerStack.layers.forEach(({node})=>{node.style.pointerEvents=layerStack.isBelowPointerBlockingLayer(node)?"none":"auto"})}function clearPointerEvent(node){node.style.pointerEvents=""}function disablePointerEventsOutside(node,peristentElements){const doc=getDocument2(node);const cleanups=[];if(layerStack.hasPointerBlockingLayer()&&!doc.body.hasAttribute("data-inert")){originalBodyPointerEvents=document.body.style.pointerEvents;queueMicrotask(()=>{doc.body.style.pointerEvents="none";doc.body.setAttribute("data-inert","")})}if(peristentElements){const persistedCleanup=waitForElements(peristentElements,el=>{cleanups.push(setStyle(el,{pointerEvents:"auto"}))});cleanups.push(persistedCleanup)}return()=>{if(layerStack.hasPointerBlockingLayer())return;queueMicrotask(()=>{doc.body.style.pointerEvents=originalBodyPointerEvents;doc.body.removeAttribute("data-inert");if(doc.body.style.length===0)doc.body.removeAttribute("style")});cleanups.forEach(fn=>fn())}}function trackDismissableElementImpl(node,options){if(!node){warn("[@zag-js/dismissable] node is `null` or `undefined`");return}const{onDismiss,pointerBlocking,exclude:excludeContainers,debug}=options;const layer={dismiss:onDismiss,node,pointerBlocking};layerStack.add(layer);assignPointerEventToLayers();function onPointerDownOutside(event){const target=getEventTarget(event.detail.originalEvent);if(layerStack.isBelowPointerBlockingLayer(node)||layerStack.isInBranch(target))return;options.onPointerDownOutside?.(event);options.onInteractOutside?.(event);if(event.defaultPrevented)return;if(debug){console.log("onPointerDownOutside:",event.detail.originalEvent)}onDismiss?.()}function onFocusOutside(event){const target=getEventTarget(event.detail.originalEvent);if(layerStack.isInBranch(target))return;options.onFocusOutside?.(event);options.onInteractOutside?.(event);if(event.defaultPrevented)return;if(debug){console.log("onFocusOutside:",event.detail.originalEvent)}onDismiss?.()}function onEscapeKeyDown(event){if(!layerStack.isTopMost(node))return;options.onEscapeKeyDown?.(event);if(!event.defaultPrevented&&onDismiss){event.preventDefault();onDismiss()}}function exclude(target){if(!node)return false;const containers=typeof excludeContainers==="function"?excludeContainers():excludeContainers;const _containers=Array.isArray(containers)?containers:[containers];const persistentElements=options.persistentElements?.map(fn=>fn()).filter(isHTMLElement);if(persistentElements)_containers.push(...persistentElements);return _containers.some(node2=>contains2(node2,target))||layerStack.isInNestedLayer(node,target)}const cleanups=[pointerBlocking?disablePointerEventsOutside(node,options.persistentElements):void 0,trackEscapeKeydown(node,onEscapeKeyDown),trackInteractOutside(node,{exclude,onFocusOutside,onPointerDownOutside,defer:options.defer})];return()=>{layerStack.remove(node);assignPointerEventToLayers();clearPointerEvent(node);cleanups.forEach(fn=>fn?.())}}function trackDismissableElement(nodeOrFn,options){const{defer}=options;const func=defer?raf:v=>v();const cleanups=[];cleanups.push(func(()=>{const node=isFunction(nodeOrFn)?nodeOrFn():nodeOrFn;cleanups.push(trackDismissableElementImpl(node,options))}));return()=>{cleanups.forEach(fn=>fn?.())}}function trackDismissableBranch(nodeOrFn,options={}){const{defer}=options;const func=defer?raf:v=>v();const cleanups=[];cleanups.push(func(()=>{const node=isFunction(nodeOrFn)?nodeOrFn():nodeOrFn;if(!node){warn("[@zag-js/dismissable] branch node is `null` or `undefined`");return}layerStack.addBranch(node);cleanups.push(()=>{layerStack.removeBranch(node)})}));return()=>{cleanups.forEach(fn=>fn?.())}}export{trackDismissableBranch,trackDismissableElement};
|
|
220
2
|
//# sourceMappingURL=index.mjs.map
|
package/dist/index.mjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/dismissable-layer.ts","../src/escape-keydown.ts","../src/layer-stack.ts","../src/pointer-event-outside.ts"],"sourcesContent":["import { contains, getEventTarget, raf } from \"@zag-js/dom-query\"\nimport {\n trackInteractOutside,\n type FocusOutsideEvent,\n type InteractOutsideHandlers,\n type PointerDownOutsideEvent,\n} from \"@zag-js/interact-outside\"\nimport { isFunction, warn, type MaybeFunction } from \"@zag-js/utils\"\nimport { trackEscapeKeydown } from \"./escape-keydown\"\nimport { layerStack, type Layer } from \"./layer-stack\"\nimport { assignPointerEventToLayers, clearPointerEvent, disablePointerEventsOutside } from \"./pointer-event-outside\"\n\ntype MaybeElement = HTMLElement | null\ntype Container = MaybeElement | Array<MaybeElement>\ntype NodeOrFn = MaybeFunction<MaybeElement>\n\nexport interface DismissableElementHandlers extends InteractOutsideHandlers {\n /**\n * Function called when the escape key is pressed\n */\n onEscapeKeyDown?: (event: KeyboardEvent) => void\n}\n\nexport interface DismissableElementOptions extends DismissableElementHandlers {\n /**\n * Whether to log debug information\n */\n debug?: boolean\n /**\n * Whether to block pointer events outside the dismissable element\n */\n pointerBlocking?: boolean\n /**\n * Function called when the dismissable element is dismissed\n */\n onDismiss: VoidFunction\n /**\n * Exclude containers from the interact outside event\n */\n exclude?: MaybeFunction<Container>\n /**\n * Defer the interact outside event to the next frame\n */\n defer?: boolean\n}\n\nfunction trackDismissableElementImpl(node: MaybeElement, options: DismissableElementOptions) {\n if (!node) {\n warn(\"[@zag-js/dismissable] node is `null` or `undefined`\")\n return\n }\n\n const { onDismiss, pointerBlocking, exclude: excludeContainers, debug } = options\n\n const layer: Layer = { dismiss: onDismiss, node, pointerBlocking }\n\n layerStack.add(layer)\n assignPointerEventToLayers()\n\n function onPointerDownOutside(event: PointerDownOutsideEvent) {\n const target = getEventTarget(event.detail.originalEvent)\n if (layerStack.isBelowPointerBlockingLayer(node!) || layerStack.isInBranch(target)) return\n options.onPointerDownOutside?.(event)\n options.onInteractOutside?.(event)\n if (event.defaultPrevented) return\n if (debug) {\n console.log(\"onPointerDownOutside:\", event.detail.originalEvent)\n }\n onDismiss?.()\n }\n\n function onFocusOutside(event: FocusOutsideEvent) {\n const target = getEventTarget(event.detail.originalEvent)\n if (layerStack.isInBranch(target)) return\n options.onFocusOutside?.(event)\n options.onInteractOutside?.(event)\n if (event.defaultPrevented) return\n if (debug) {\n console.log(\"onFocusOutside:\", event.detail.originalEvent)\n }\n onDismiss?.()\n }\n\n function onEscapeKeyDown(event: KeyboardEvent) {\n if (!layerStack.isTopMost(node!)) return\n options.onEscapeKeyDown?.(event)\n if (!event.defaultPrevented && onDismiss) {\n event.preventDefault()\n onDismiss()\n }\n }\n\n function exclude(target: Element) {\n if (!node) return false\n const containers = typeof excludeContainers === \"function\" ? excludeContainers() : excludeContainers\n const _containers = Array.isArray(containers) ? containers : [containers]\n return _containers.some((node) => contains(node, target)) || layerStack.isInNestedLayer(node, target)\n }\n\n const cleanups = [\n pointerBlocking ? disablePointerEventsOutside(node) : undefined,\n trackEscapeKeydown(node, onEscapeKeyDown),\n trackInteractOutside(node, { exclude, onFocusOutside, onPointerDownOutside, defer: options.defer }),\n ]\n\n return () => {\n layerStack.remove(node!)\n // re-assign pointer event to remaining layers\n assignPointerEventToLayers()\n // remove pointer event from removed layer\n clearPointerEvent(node!)\n cleanups.forEach((fn) => fn?.())\n }\n}\n\nexport function trackDismissableElement(nodeOrFn: NodeOrFn, options: DismissableElementOptions) {\n const { defer } = options\n const func = defer ? raf : (v: any) => v()\n const cleanups: (VoidFunction | undefined)[] = []\n cleanups.push(\n func(() => {\n const node = isFunction(nodeOrFn) ? nodeOrFn() : nodeOrFn\n cleanups.push(trackDismissableElementImpl(node, options))\n }),\n )\n return () => {\n cleanups.forEach((fn) => fn?.())\n }\n}\n\nexport function trackDismissableBranch(nodeOrFn: NodeOrFn, options: { defer?: boolean } = {}) {\n const { defer } = options\n const func = defer ? raf : (v: any) => v()\n const cleanups: (VoidFunction | undefined)[] = []\n\n cleanups.push(\n func(() => {\n const node = isFunction(nodeOrFn) ? nodeOrFn() : nodeOrFn\n if (!node) {\n warn(\"[@zag-js/dismissable] branch node is `null` or `undefined`\")\n return\n }\n layerStack.addBranch(node)\n cleanups.push(() => {\n layerStack.removeBranch(node)\n })\n }),\n )\n\n return () => {\n cleanups.forEach((fn) => fn?.())\n }\n}\n","import { addDomEvent } from \"@zag-js/dom-event\"\nimport { getDocument } from \"@zag-js/dom-query\"\n\nexport function trackEscapeKeydown(node: HTMLElement, fn?: (event: KeyboardEvent) => void) {\n const handleKeyDown = (event: KeyboardEvent) => {\n if (event.key === \"Escape\") fn?.(event)\n }\n return addDomEvent(getDocument(node), \"keydown\", handleKeyDown, { capture: true })\n}\n","import { contains } from \"@zag-js/dom-query\"\n\nexport interface Layer {\n dismiss: VoidFunction\n node: HTMLElement\n pointerBlocking?: boolean\n}\n\nexport const layerStack = {\n layers: [] as Layer[],\n branches: [] as HTMLElement[],\n count(): number {\n return this.layers.length\n },\n pointerBlockingLayers(): Layer[] {\n return this.layers.filter((layer) => layer.pointerBlocking)\n },\n topMostPointerBlockingLayer(): Layer | undefined {\n return [...this.pointerBlockingLayers()].slice(-1)[0]\n },\n hasPointerBlockingLayer(): boolean {\n return this.pointerBlockingLayers().length > 0\n },\n isBelowPointerBlockingLayer(node: HTMLElement) {\n const index = this.indexOf(node)\n const highestBlockingIndex = this.topMostPointerBlockingLayer()\n ? this.indexOf(this.topMostPointerBlockingLayer()?.node)\n : -1\n return index < highestBlockingIndex\n },\n isTopMost(node: HTMLElement | null) {\n const layer = this.layers[this.count() - 1]\n return layer?.node === node\n },\n getNestedLayers(node: HTMLElement) {\n return Array.from(this.layers).slice(this.indexOf(node) + 1)\n },\n isInNestedLayer(node: HTMLElement, target: HTMLElement | EventTarget | null) {\n return this.getNestedLayers(node).some((layer) => contains(layer.node, target))\n },\n isInBranch(target: HTMLElement | EventTarget | null) {\n return Array.from(this.branches).some((branch) => contains(branch, target))\n },\n add(layer: Layer) {\n const num = this.layers.push(layer)\n layer.node.style.setProperty(\"--layer-index\", `${num}`)\n },\n addBranch(node: HTMLElement) {\n this.branches.push(node)\n },\n remove(node: HTMLElement) {\n const index = this.indexOf(node)\n if (index < 0) return\n\n // dismiss nested layers\n if (index < this.count() - 1) {\n const _layers = this.getNestedLayers(node)\n _layers.forEach((layer) => layer.dismiss())\n }\n // remove this layer\n this.layers.splice(index, 1)\n node.style.removeProperty(\"--layer-index\")\n },\n removeBranch(node: HTMLElement) {\n const index = this.branches.indexOf(node)\n if (index >= 0) this.branches.splice(index, 1)\n },\n indexOf(node: HTMLElement | undefined) {\n return this.layers.findIndex((layer) => layer.node === node)\n },\n dismiss(node: HTMLElement) {\n this.layers[this.indexOf(node)]?.dismiss()\n },\n clear() {\n this.remove(this.layers[0].node)\n },\n}\n","import { getDocument } from \"@zag-js/dom-query\"\nimport { layerStack } from \"./layer-stack\"\n\nlet originalBodyPointerEvents: string\n\nexport function assignPointerEventToLayers() {\n layerStack.layers.forEach(({ node }) => {\n node.style.pointerEvents = layerStack.isBelowPointerBlockingLayer(node) ? \"none\" : \"auto\"\n })\n}\n\nexport function clearPointerEvent(node: HTMLElement) {\n node.style.pointerEvents = \"\"\n}\n\nconst DATA_ATTR = \"data-inert\"\n\nexport function disablePointerEventsOutside(node: HTMLElement) {\n const doc = getDocument(node)\n\n if (layerStack.hasPointerBlockingLayer() && !doc.body.hasAttribute(DATA_ATTR)) {\n originalBodyPointerEvents = document.body.style.pointerEvents\n doc.body.style.pointerEvents = \"none\"\n doc.body.setAttribute(DATA_ATTR, \"\")\n }\n\n return () => {\n if (layerStack.hasPointerBlockingLayer()) return\n doc.body.style.pointerEvents = originalBodyPointerEvents\n doc.body.removeAttribute(DATA_ATTR)\n if (doc.body.style.length === 0) doc.body.removeAttribute(\"style\")\n }\n}\n"],"mappings":"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","names":["contains","getDocument","getDocument","node","contains"]}
|
|
1
|
+
{"version":3,"sources":["../src/dismissable-layer.ts","../src/escape-keydown.ts","../src/layer-stack.ts","../src/pointer-event-outside.ts"],"sourcesContent":["import { contains, getEventTarget, isHTMLElement, raf } from \"@zag-js/dom-query\"\nimport {\n trackInteractOutside,\n type FocusOutsideEvent,\n type InteractOutsideHandlers,\n type PointerDownOutsideEvent,\n} from \"@zag-js/interact-outside\"\nimport { isFunction, warn, type MaybeFunction } from \"@zag-js/utils\"\nimport { trackEscapeKeydown } from \"./escape-keydown\"\nimport { layerStack, type Layer } from \"./layer-stack\"\nimport { assignPointerEventToLayers, clearPointerEvent, disablePointerEventsOutside } from \"./pointer-event-outside\"\n\ntype MaybeElement = HTMLElement | null\ntype Container = MaybeElement | Array<MaybeElement>\ntype NodeOrFn = MaybeFunction<MaybeElement>\n\nexport interface DismissableElementHandlers extends InteractOutsideHandlers {\n /**\n * Function called when the escape key is pressed\n */\n onEscapeKeyDown?: (event: KeyboardEvent) => void\n}\n\nexport interface PersistentElementOptions {\n /**\n * Returns the persistent elements that:\n * - should not have pointer-events disabled\n * - should not trigger the dismiss event\n */\n persistentElements?: Array<() => Element | null>\n}\n\nexport interface DismissableElementOptions extends DismissableElementHandlers, PersistentElementOptions {\n /**\n * Whether to log debug information\n */\n debug?: boolean\n /**\n * Whether to block pointer events outside the dismissable element\n */\n pointerBlocking?: boolean\n /**\n * Function called when the dismissable element is dismissed\n */\n onDismiss: VoidFunction\n /**\n * Exclude containers from the interact outside event\n */\n exclude?: MaybeFunction<Container>\n /**\n * Defer the interact outside event to the next frame\n */\n defer?: boolean\n}\n\nfunction trackDismissableElementImpl(node: MaybeElement, options: DismissableElementOptions) {\n if (!node) {\n warn(\"[@zag-js/dismissable] node is `null` or `undefined`\")\n return\n }\n\n const { onDismiss, pointerBlocking, exclude: excludeContainers, debug } = options\n\n const layer: Layer = { dismiss: onDismiss, node, pointerBlocking }\n\n layerStack.add(layer)\n assignPointerEventToLayers()\n\n function onPointerDownOutside(event: PointerDownOutsideEvent) {\n const target = getEventTarget(event.detail.originalEvent)\n if (layerStack.isBelowPointerBlockingLayer(node!) || layerStack.isInBranch(target)) return\n options.onPointerDownOutside?.(event)\n options.onInteractOutside?.(event)\n if (event.defaultPrevented) return\n if (debug) {\n console.log(\"onPointerDownOutside:\", event.detail.originalEvent)\n }\n onDismiss?.()\n }\n\n function onFocusOutside(event: FocusOutsideEvent) {\n const target = getEventTarget(event.detail.originalEvent)\n if (layerStack.isInBranch(target)) return\n options.onFocusOutside?.(event)\n options.onInteractOutside?.(event)\n if (event.defaultPrevented) return\n if (debug) {\n console.log(\"onFocusOutside:\", event.detail.originalEvent)\n }\n onDismiss?.()\n }\n\n function onEscapeKeyDown(event: KeyboardEvent) {\n if (!layerStack.isTopMost(node!)) return\n options.onEscapeKeyDown?.(event)\n if (!event.defaultPrevented && onDismiss) {\n event.preventDefault()\n onDismiss()\n }\n }\n\n function exclude(target: Element) {\n if (!node) return false\n const containers = typeof excludeContainers === \"function\" ? excludeContainers() : excludeContainers\n const _containers = Array.isArray(containers) ? containers : [containers]\n const persistentElements = options.persistentElements?.map((fn) => fn()).filter(isHTMLElement)\n if (persistentElements) _containers.push(...persistentElements)\n return _containers.some((node) => contains(node, target)) || layerStack.isInNestedLayer(node, target)\n }\n\n const cleanups = [\n pointerBlocking ? disablePointerEventsOutside(node, options.persistentElements) : undefined,\n trackEscapeKeydown(node, onEscapeKeyDown),\n trackInteractOutside(node, { exclude, onFocusOutside, onPointerDownOutside, defer: options.defer }),\n ]\n\n return () => {\n layerStack.remove(node!)\n // re-assign pointer event to remaining layers\n assignPointerEventToLayers()\n // remove pointer event from removed layer\n clearPointerEvent(node!)\n cleanups.forEach((fn) => fn?.())\n }\n}\n\nexport function trackDismissableElement(nodeOrFn: NodeOrFn, options: DismissableElementOptions) {\n const { defer } = options\n const func = defer ? raf : (v: any) => v()\n const cleanups: (VoidFunction | undefined)[] = []\n cleanups.push(\n func(() => {\n const node = isFunction(nodeOrFn) ? nodeOrFn() : nodeOrFn\n cleanups.push(trackDismissableElementImpl(node, options))\n }),\n )\n return () => {\n cleanups.forEach((fn) => fn?.())\n }\n}\n\nexport function trackDismissableBranch(nodeOrFn: NodeOrFn, options: { defer?: boolean } = {}) {\n const { defer } = options\n const func = defer ? raf : (v: any) => v()\n const cleanups: (VoidFunction | undefined)[] = []\n\n cleanups.push(\n func(() => {\n const node = isFunction(nodeOrFn) ? nodeOrFn() : nodeOrFn\n if (!node) {\n warn(\"[@zag-js/dismissable] branch node is `null` or `undefined`\")\n return\n }\n layerStack.addBranch(node)\n cleanups.push(() => {\n layerStack.removeBranch(node)\n })\n }),\n )\n\n return () => {\n cleanups.forEach((fn) => fn?.())\n }\n}\n","import { addDomEvent } from \"@zag-js/dom-event\"\nimport { getDocument } from \"@zag-js/dom-query\"\n\nexport function trackEscapeKeydown(node: HTMLElement, fn?: (event: KeyboardEvent) => void) {\n const handleKeyDown = (event: KeyboardEvent) => {\n if (event.key !== \"Escape\") return\n if (event.isComposing) return\n fn?.(event)\n }\n\n return addDomEvent(getDocument(node), \"keydown\", handleKeyDown, { capture: true })\n}\n","import { contains } from \"@zag-js/dom-query\"\n\nexport interface Layer {\n dismiss: VoidFunction\n node: HTMLElement\n pointerBlocking?: boolean\n}\n\nexport const layerStack = {\n layers: [] as Layer[],\n branches: [] as HTMLElement[],\n count(): number {\n return this.layers.length\n },\n pointerBlockingLayers(): Layer[] {\n return this.layers.filter((layer) => layer.pointerBlocking)\n },\n topMostPointerBlockingLayer(): Layer | undefined {\n return [...this.pointerBlockingLayers()].slice(-1)[0]\n },\n hasPointerBlockingLayer(): boolean {\n return this.pointerBlockingLayers().length > 0\n },\n isBelowPointerBlockingLayer(node: HTMLElement) {\n const index = this.indexOf(node)\n const highestBlockingIndex = this.topMostPointerBlockingLayer()\n ? this.indexOf(this.topMostPointerBlockingLayer()?.node)\n : -1\n return index < highestBlockingIndex\n },\n isTopMost(node: HTMLElement | null) {\n const layer = this.layers[this.count() - 1]\n return layer?.node === node\n },\n getNestedLayers(node: HTMLElement) {\n return Array.from(this.layers).slice(this.indexOf(node) + 1)\n },\n isInNestedLayer(node: HTMLElement, target: HTMLElement | EventTarget | null) {\n return this.getNestedLayers(node).some((layer) => contains(layer.node, target))\n },\n isInBranch(target: HTMLElement | EventTarget | null) {\n return Array.from(this.branches).some((branch) => contains(branch, target))\n },\n add(layer: Layer) {\n const num = this.layers.push(layer)\n layer.node.style.setProperty(\"--layer-index\", `${num}`)\n },\n addBranch(node: HTMLElement) {\n this.branches.push(node)\n },\n remove(node: HTMLElement) {\n const index = this.indexOf(node)\n if (index < 0) return\n\n // dismiss nested layers\n if (index < this.count() - 1) {\n const _layers = this.getNestedLayers(node)\n _layers.forEach((layer) => layer.dismiss())\n }\n // remove this layer\n this.layers.splice(index, 1)\n node.style.removeProperty(\"--layer-index\")\n },\n removeBranch(node: HTMLElement) {\n const index = this.branches.indexOf(node)\n if (index >= 0) this.branches.splice(index, 1)\n },\n indexOf(node: HTMLElement | undefined) {\n return this.layers.findIndex((layer) => layer.node === node)\n },\n dismiss(node: HTMLElement) {\n this.layers[this.indexOf(node)]?.dismiss()\n },\n clear() {\n this.remove(this.layers[0].node)\n },\n}\n","import { getDocument, setStyle, waitForElements } from \"@zag-js/dom-query\"\nimport { layerStack } from \"./layer-stack\"\n\nlet originalBodyPointerEvents: string\n\nexport function assignPointerEventToLayers() {\n layerStack.layers.forEach(({ node }) => {\n node.style.pointerEvents = layerStack.isBelowPointerBlockingLayer(node) ? \"none\" : \"auto\"\n })\n}\n\nexport function clearPointerEvent(node: HTMLElement) {\n node.style.pointerEvents = \"\"\n}\n\nexport function disablePointerEventsOutside(node: HTMLElement, peristentElements?: Array<() => Element | null>) {\n const doc = getDocument(node)\n\n const cleanups: VoidFunction[] = []\n\n if (layerStack.hasPointerBlockingLayer() && !doc.body.hasAttribute(\"data-inert\")) {\n originalBodyPointerEvents = document.body.style.pointerEvents\n queueMicrotask(() => {\n doc.body.style.pointerEvents = \"none\"\n doc.body.setAttribute(\"data-inert\", \"\")\n })\n }\n\n if (peristentElements) {\n const persistedCleanup = waitForElements(peristentElements, (el) => {\n cleanups.push(setStyle(el, { pointerEvents: \"auto\" }))\n })\n cleanups.push(persistedCleanup)\n }\n\n return () => {\n if (layerStack.hasPointerBlockingLayer()) return\n queueMicrotask(() => {\n doc.body.style.pointerEvents = originalBodyPointerEvents\n doc.body.removeAttribute(\"data-inert\")\n if (doc.body.style.length === 0) doc.body.removeAttribute(\"style\")\n })\n cleanups.forEach((fn) => fn())\n }\n}\n"],"mappings":"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","names":["contains","getDocument","getDocument","node","contains"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@zag-js/dismissable",
|
|
3
|
-
"version": "0.
|
|
3
|
+
"version": "0.51.0",
|
|
4
4
|
"description": "Dismissable layer utilities for the DOM",
|
|
5
5
|
"keywords": [
|
|
6
6
|
"js",
|
|
@@ -24,10 +24,10 @@
|
|
|
24
24
|
"access": "public"
|
|
25
25
|
},
|
|
26
26
|
"dependencies": {
|
|
27
|
-
"@zag-js/interact-outside": "0.
|
|
28
|
-
"@zag-js/dom-query": "0.
|
|
29
|
-
"@zag-js/dom-event": "0.
|
|
30
|
-
"@zag-js/utils": "0.
|
|
27
|
+
"@zag-js/interact-outside": "0.51.0",
|
|
28
|
+
"@zag-js/dom-query": "0.51.0",
|
|
29
|
+
"@zag-js/dom-event": "0.51.0",
|
|
30
|
+
"@zag-js/utils": "0.51.0"
|
|
31
31
|
},
|
|
32
32
|
"devDependencies": {
|
|
33
33
|
"clean-package": "2.2.0"
|
package/src/dismissable-layer.ts
CHANGED
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { contains, getEventTarget, raf } from "@zag-js/dom-query"
|
|
1
|
+
import { contains, getEventTarget, isHTMLElement, raf } from "@zag-js/dom-query"
|
|
2
2
|
import {
|
|
3
3
|
trackInteractOutside,
|
|
4
4
|
type FocusOutsideEvent,
|
|
@@ -21,7 +21,16 @@ export interface DismissableElementHandlers extends InteractOutsideHandlers {
|
|
|
21
21
|
onEscapeKeyDown?: (event: KeyboardEvent) => void
|
|
22
22
|
}
|
|
23
23
|
|
|
24
|
-
export interface
|
|
24
|
+
export interface PersistentElementOptions {
|
|
25
|
+
/**
|
|
26
|
+
* Returns the persistent elements that:
|
|
27
|
+
* - should not have pointer-events disabled
|
|
28
|
+
* - should not trigger the dismiss event
|
|
29
|
+
*/
|
|
30
|
+
persistentElements?: Array<() => Element | null>
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
export interface DismissableElementOptions extends DismissableElementHandlers, PersistentElementOptions {
|
|
25
34
|
/**
|
|
26
35
|
* Whether to log debug information
|
|
27
36
|
*/
|
|
@@ -94,11 +103,13 @@ function trackDismissableElementImpl(node: MaybeElement, options: DismissableEle
|
|
|
94
103
|
if (!node) return false
|
|
95
104
|
const containers = typeof excludeContainers === "function" ? excludeContainers() : excludeContainers
|
|
96
105
|
const _containers = Array.isArray(containers) ? containers : [containers]
|
|
106
|
+
const persistentElements = options.persistentElements?.map((fn) => fn()).filter(isHTMLElement)
|
|
107
|
+
if (persistentElements) _containers.push(...persistentElements)
|
|
97
108
|
return _containers.some((node) => contains(node, target)) || layerStack.isInNestedLayer(node, target)
|
|
98
109
|
}
|
|
99
110
|
|
|
100
111
|
const cleanups = [
|
|
101
|
-
pointerBlocking ? disablePointerEventsOutside(node) : undefined,
|
|
112
|
+
pointerBlocking ? disablePointerEventsOutside(node, options.persistentElements) : undefined,
|
|
102
113
|
trackEscapeKeydown(node, onEscapeKeyDown),
|
|
103
114
|
trackInteractOutside(node, { exclude, onFocusOutside, onPointerDownOutside, defer: options.defer }),
|
|
104
115
|
]
|
package/src/escape-keydown.ts
CHANGED
|
@@ -3,7 +3,10 @@ import { getDocument } from "@zag-js/dom-query"
|
|
|
3
3
|
|
|
4
4
|
export function trackEscapeKeydown(node: HTMLElement, fn?: (event: KeyboardEvent) => void) {
|
|
5
5
|
const handleKeyDown = (event: KeyboardEvent) => {
|
|
6
|
-
if (event.key
|
|
6
|
+
if (event.key !== "Escape") return
|
|
7
|
+
if (event.isComposing) return
|
|
8
|
+
fn?.(event)
|
|
7
9
|
}
|
|
10
|
+
|
|
8
11
|
return addDomEvent(getDocument(node), "keydown", handleKeyDown, { capture: true })
|
|
9
12
|
}
|
package/src/index.ts
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
|
-
export * from "./dismissable-layer"
|
|
2
1
|
export type {
|
|
3
|
-
InteractOutsideEvent,
|
|
4
|
-
PointerDownOutsideEvent,
|
|
5
2
|
FocusOutsideEvent,
|
|
3
|
+
InteractOutsideEvent,
|
|
6
4
|
InteractOutsideHandlers,
|
|
5
|
+
PointerDownOutsideEvent,
|
|
7
6
|
} from "@zag-js/interact-outside"
|
|
7
|
+
export * from "./dismissable-layer"
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { getDocument } from "@zag-js/dom-query"
|
|
1
|
+
import { getDocument, setStyle, waitForElements } from "@zag-js/dom-query"
|
|
2
2
|
import { layerStack } from "./layer-stack"
|
|
3
3
|
|
|
4
4
|
let originalBodyPointerEvents: string
|
|
@@ -13,21 +13,33 @@ export function clearPointerEvent(node: HTMLElement) {
|
|
|
13
13
|
node.style.pointerEvents = ""
|
|
14
14
|
}
|
|
15
15
|
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
export function disablePointerEventsOutside(node: HTMLElement) {
|
|
16
|
+
export function disablePointerEventsOutside(node: HTMLElement, peristentElements?: Array<() => Element | null>) {
|
|
19
17
|
const doc = getDocument(node)
|
|
20
18
|
|
|
21
|
-
|
|
19
|
+
const cleanups: VoidFunction[] = []
|
|
20
|
+
|
|
21
|
+
if (layerStack.hasPointerBlockingLayer() && !doc.body.hasAttribute("data-inert")) {
|
|
22
22
|
originalBodyPointerEvents = document.body.style.pointerEvents
|
|
23
|
-
|
|
24
|
-
|
|
23
|
+
queueMicrotask(() => {
|
|
24
|
+
doc.body.style.pointerEvents = "none"
|
|
25
|
+
doc.body.setAttribute("data-inert", "")
|
|
26
|
+
})
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
if (peristentElements) {
|
|
30
|
+
const persistedCleanup = waitForElements(peristentElements, (el) => {
|
|
31
|
+
cleanups.push(setStyle(el, { pointerEvents: "auto" }))
|
|
32
|
+
})
|
|
33
|
+
cleanups.push(persistedCleanup)
|
|
25
34
|
}
|
|
26
35
|
|
|
27
36
|
return () => {
|
|
28
37
|
if (layerStack.hasPointerBlockingLayer()) return
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
38
|
+
queueMicrotask(() => {
|
|
39
|
+
doc.body.style.pointerEvents = originalBodyPointerEvents
|
|
40
|
+
doc.body.removeAttribute("data-inert")
|
|
41
|
+
if (doc.body.style.length === 0) doc.body.removeAttribute("style")
|
|
42
|
+
})
|
|
43
|
+
cleanups.forEach((fn) => fn())
|
|
32
44
|
}
|
|
33
45
|
}
|