@aquera/nile-elements 1.7.9 → 1.8.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/README.md +3 -0
- package/dist/index.cjs.js +1 -1
- package/dist/index.esm.js +1 -1
- package/dist/index.js +697 -502
- package/dist/nile-context-menu/index.cjs.js +2 -0
- package/dist/nile-context-menu/index.cjs.js.map +1 -0
- package/dist/nile-context-menu/index.esm.js +1 -0
- package/dist/nile-context-menu/nile-context-menu.cjs.js +2 -0
- package/dist/nile-context-menu/nile-context-menu.cjs.js.map +1 -0
- package/dist/nile-context-menu/nile-context-menu.css.cjs.js +2 -0
- package/dist/nile-context-menu/nile-context-menu.css.cjs.js.map +1 -0
- package/dist/nile-context-menu/nile-context-menu.css.esm.js +51 -0
- package/dist/nile-context-menu/nile-context-menu.esm.js +25 -0
- package/dist/nile-context-menu-group/index.cjs.js +2 -0
- package/dist/nile-context-menu-group/index.cjs.js.map +1 -0
- package/dist/nile-context-menu-group/index.esm.js +1 -0
- package/dist/nile-context-menu-group/nile-context-menu-group.cjs.js +2 -0
- package/dist/nile-context-menu-group/nile-context-menu-group.cjs.js.map +1 -0
- package/dist/nile-context-menu-group/nile-context-menu-group.css.cjs.js +2 -0
- package/dist/nile-context-menu-group/nile-context-menu-group.css.cjs.js.map +1 -0
- package/dist/nile-context-menu-group/nile-context-menu-group.css.esm.js +20 -0
- package/dist/nile-context-menu-group/nile-context-menu-group.esm.js +11 -0
- package/dist/nile-context-menu-item/index.cjs.js +2 -0
- package/dist/nile-context-menu-item/index.cjs.js.map +1 -0
- package/dist/nile-context-menu-item/index.esm.js +1 -0
- package/dist/nile-context-menu-item/nile-context-menu-item.cjs.js +2 -0
- package/dist/nile-context-menu-item/nile-context-menu-item.cjs.js.map +1 -0
- package/dist/nile-context-menu-item/nile-context-menu-item.css.cjs.js +2 -0
- package/dist/nile-context-menu-item/nile-context-menu-item.css.cjs.js.map +1 -0
- package/dist/nile-context-menu-item/nile-context-menu-item.css.esm.js +72 -0
- package/dist/nile-context-menu-item/nile-context-menu-item.esm.js +20 -0
- package/dist/nile-context-submenu/index.cjs.js +2 -0
- package/dist/nile-context-submenu/index.cjs.js.map +1 -0
- package/dist/nile-context-submenu/index.esm.js +1 -0
- package/dist/nile-context-submenu/nile-context-submenu.cjs.js +2 -0
- package/dist/nile-context-submenu/nile-context-submenu.cjs.js.map +1 -0
- package/dist/nile-context-submenu/nile-context-submenu.esm.js +3 -0
- package/dist/src/index.d.ts +3 -0
- package/dist/src/index.js +3 -0
- package/dist/src/index.js.map +1 -1
- package/dist/src/nile-context-menu/index.d.ts +3 -0
- package/dist/src/nile-context-menu/index.js +4 -0
- package/dist/src/nile-context-menu/index.js.map +1 -0
- package/dist/src/nile-context-menu/nile-context-menu.css.d.ts +10 -0
- package/dist/src/nile-context-menu/nile-context-menu.css.js +127 -0
- package/dist/src/nile-context-menu/nile-context-menu.css.js.map +1 -0
- package/dist/src/nile-context-menu/nile-context-menu.d.ts +123 -0
- package/dist/src/nile-context-menu/nile-context-menu.js +625 -0
- package/dist/src/nile-context-menu/nile-context-menu.js.map +1 -0
- package/dist/src/nile-context-menu-group/index.d.ts +1 -0
- package/dist/src/nile-context-menu-group/index.js +2 -0
- package/dist/src/nile-context-menu-group/index.js.map +1 -0
- package/dist/src/nile-context-menu-group/nile-context-menu-group.css.d.ts +9 -0
- package/dist/src/nile-context-menu-group/nile-context-menu-group.css.js +29 -0
- package/dist/src/nile-context-menu-group/nile-context-menu-group.css.js.map +1 -0
- package/dist/src/nile-context-menu-group/nile-context-menu-group.d.ts +28 -0
- package/dist/src/nile-context-menu-group/nile-context-menu-group.js +55 -0
- package/dist/src/nile-context-menu-group/nile-context-menu-group.js.map +1 -0
- package/dist/src/nile-context-menu-item/index.d.ts +1 -0
- package/dist/src/nile-context-menu-item/index.js +2 -0
- package/dist/src/nile-context-menu-item/index.js.map +1 -0
- package/dist/src/nile-context-menu-item/nile-context-menu-item.css.d.ts +9 -0
- package/dist/src/nile-context-menu-item/nile-context-menu-item.css.js +81 -0
- package/dist/src/nile-context-menu-item/nile-context-menu-item.css.js.map +1 -0
- package/dist/src/nile-context-menu-item/nile-context-menu-item.d.ts +45 -0
- package/dist/src/nile-context-menu-item/nile-context-menu-item.js +96 -0
- package/dist/src/nile-context-menu-item/nile-context-menu-item.js.map +1 -0
- package/dist/src/nile-context-submenu/index.d.ts +1 -0
- package/dist/src/nile-context-submenu/index.js +2 -0
- package/dist/src/nile-context-submenu/index.js.map +1 -0
- package/dist/src/nile-context-submenu/nile-context-submenu.d.ts +60 -0
- package/dist/src/nile-context-submenu/nile-context-submenu.js +324 -0
- package/dist/src/nile-context-submenu/nile-context-submenu.js.map +1 -0
- package/dist/src/version.js +1 -1
- package/dist/src/version.js.map +1 -1
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/package.json +1 -1
- package/src/index.ts +3 -0
- package/src/nile-context-menu/index.ts +12 -0
- package/src/nile-context-menu/nile-context-menu.css.ts +130 -0
- package/src/nile-context-menu/nile-context-menu.ts +698 -0
- package/src/nile-context-menu-group/index.ts +1 -0
- package/src/nile-context-menu-group/nile-context-menu-group.css.ts +31 -0
- package/src/nile-context-menu-group/nile-context-menu-group.ts +55 -0
- package/src/nile-context-menu-item/index.ts +5 -0
- package/src/nile-context-menu-item/nile-context-menu-item.css.ts +83 -0
- package/src/nile-context-menu-item/nile-context-menu-item.ts +112 -0
- package/src/nile-context-submenu/index.ts +1 -0
- package/src/nile-context-submenu/nile-context-submenu.ts +322 -0
- package/vscode-html-custom-data.json +82 -4
|
@@ -0,0 +1,698 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Copyright Aquera Inc 2026
|
|
3
|
+
*
|
|
4
|
+
* This source code is licensed under the BSD-3-Clause license found in the
|
|
5
|
+
* LICENSE file in the root directory of this source tree.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import { html, CSSResultArray, TemplateResult } from 'lit';
|
|
9
|
+
import type { PropertyValues } from 'lit';
|
|
10
|
+
import { customElement, property, query, state } from 'lit/decorators.js';
|
|
11
|
+
import { styles } from './nile-context-menu.css';
|
|
12
|
+
import NileElement from '../internal/nile-element';
|
|
13
|
+
import type { NileContextMenuItem } from '../nile-context-menu-item';
|
|
14
|
+
import type { NileContextMenuGroup } from '../nile-context-menu-group';
|
|
15
|
+
import '../nile-floating-panel';
|
|
16
|
+
import '../nile-context-menu-group';
|
|
17
|
+
import '../nile-context-menu-item';
|
|
18
|
+
import '../nile-context-submenu';
|
|
19
|
+
|
|
20
|
+
// Data-driven menu entry
|
|
21
|
+
export interface NileContextMenuItemData {
|
|
22
|
+
type: 'item';
|
|
23
|
+
id: string;
|
|
24
|
+
label: string;
|
|
25
|
+
value?: string;
|
|
26
|
+
disabled?: boolean;
|
|
27
|
+
icon?: string;
|
|
28
|
+
iconSet?: string;
|
|
29
|
+
iconSize?: string;
|
|
30
|
+
iconMethod?: 'fill' | 'stroke' | string;
|
|
31
|
+
iconColor?: string;
|
|
32
|
+
submenu?: NileContextMenuData[];
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
export interface NileContextMenuGroupData {
|
|
36
|
+
type: 'group';
|
|
37
|
+
name?: string;
|
|
38
|
+
data: NileContextMenuItemData[];
|
|
39
|
+
}
|
|
40
|
+
|
|
41
|
+
export type NileContextMenuData = NileContextMenuGroupData;
|
|
42
|
+
|
|
43
|
+
const ITEM_TAG = 'nile-context-menu-item';
|
|
44
|
+
const SUBMENU_TAG = 'nile-context-submenu';
|
|
45
|
+
const MENU_CONTAINER_CLASS = 'nile-context-menu__menu';
|
|
46
|
+
const OBSERVED_ITEM_ATTRS = ['value', 'disabled'];
|
|
47
|
+
|
|
48
|
+
export type NileContextMenuTrigger = 'right' | 'left' | 'both' | 'manual' | 'global';
|
|
49
|
+
|
|
50
|
+
export interface NileContextMenuOpenOptions {
|
|
51
|
+
x: number;
|
|
52
|
+
y: number;
|
|
53
|
+
target?: Element;
|
|
54
|
+
originalEvent?: Event;
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
export type NileContextMenuCloseReason =
|
|
58
|
+
| 'select'
|
|
59
|
+
| 'outside-click'
|
|
60
|
+
| 'escape'
|
|
61
|
+
| 'programmatic';
|
|
62
|
+
|
|
63
|
+
interface NileContextMenuOpenContext {
|
|
64
|
+
x: number;
|
|
65
|
+
y: number;
|
|
66
|
+
target: Element | null;
|
|
67
|
+
originalEvent: Event | null;
|
|
68
|
+
}
|
|
69
|
+
|
|
70
|
+
let proxyIdSeq = 0;
|
|
71
|
+
|
|
72
|
+
/**
|
|
73
|
+
* Nile context-menu component.
|
|
74
|
+
*
|
|
75
|
+
* @tag nile-context-menu
|
|
76
|
+
*
|
|
77
|
+
* @slot Default menu content.
|
|
78
|
+
*
|
|
79
|
+
* @event nile-change Fired on menu lifecycle and selection. `detail.type` is one of
|
|
80
|
+
* `'open'`, `'click'`, or `'close'`; remaining fields depend on the type.
|
|
81
|
+
*/
|
|
82
|
+
@customElement('nile-context-menu')
|
|
83
|
+
export class NileContextMenu extends NileElement {
|
|
84
|
+
public static get styles(): CSSResultArray {
|
|
85
|
+
return [styles];
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
|
|
89
|
+
@property({ attribute: true, type: String, reflect: true }) for = '';
|
|
90
|
+
|
|
91
|
+
@property({ attribute: true, type: String, reflect: true }) trigger: NileContextMenuTrigger = 'right';
|
|
92
|
+
|
|
93
|
+
@property({ attribute: true, type: String, reflect: true }) skipOn = '';
|
|
94
|
+
|
|
95
|
+
@property({ attribute: true, type: Number, reflect: true }) zIndex = 9999;
|
|
96
|
+
|
|
97
|
+
@property({ attribute: false }) public items: NileContextMenuData[] = [];
|
|
98
|
+
|
|
99
|
+
private get _isDataMode(): boolean {
|
|
100
|
+
return Array.isArray(this.items) && this.items.length > 0;
|
|
101
|
+
}
|
|
102
|
+
@query('nile-floating-panel') private _floatingPanel?: HTMLElement;
|
|
103
|
+
|
|
104
|
+
@state() private _items: NileContextMenuItem[] = [];
|
|
105
|
+
|
|
106
|
+
@state() private _open = false;
|
|
107
|
+
|
|
108
|
+
protected _openContext: NileContextMenuOpenContext | null = null;
|
|
109
|
+
|
|
110
|
+
protected _targetEl: Element | null = null;
|
|
111
|
+
|
|
112
|
+
private _previouslyFocused: HTMLElement | null = null;
|
|
113
|
+
|
|
114
|
+
private static _openInstances = new Set<NileContextMenu>();
|
|
115
|
+
|
|
116
|
+
private _detachTriggers?: () => void;
|
|
117
|
+
private _lightObserver?: MutationObserver;
|
|
118
|
+
private _menuObserver?: MutationObserver;
|
|
119
|
+
|
|
120
|
+
private _proxyEl?: HTMLDivElement;
|
|
121
|
+
private readonly _proxyId = `nile-context-menu-anchor-${++proxyIdSeq}`;
|
|
122
|
+
|
|
123
|
+
private _wasDataMode = false;
|
|
124
|
+
|
|
125
|
+
public override connectedCallback(): void {
|
|
126
|
+
super.connectedCallback();
|
|
127
|
+
this._ensureProxy();
|
|
128
|
+
this._resolveTarget();
|
|
129
|
+
this._attachTriggers();
|
|
130
|
+
this._lightObserver = new MutationObserver(() => {
|
|
131
|
+
if (this._isDataMode) return;
|
|
132
|
+
this._relocateLightChildren();
|
|
133
|
+
});
|
|
134
|
+
this._lightObserver.observe(this, { childList: true });
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
protected override updated(changed: PropertyValues): void {
|
|
138
|
+
super.updated(changed);
|
|
139
|
+
if ((changed.has('for') || changed.has('trigger') || changed.has('skipOn')) && this._open) {
|
|
140
|
+
this.close('programmatic');
|
|
141
|
+
}
|
|
142
|
+
if (changed.has('for')) {
|
|
143
|
+
this._resolveTarget();
|
|
144
|
+
this._attachTriggers();
|
|
145
|
+
} else if (changed.has('trigger') || changed.has('skipOn')) {
|
|
146
|
+
this._attachTriggers();
|
|
147
|
+
}
|
|
148
|
+
if (changed.has('items') && this._menuContainerRef) {
|
|
149
|
+
if (this._isDataMode) {
|
|
150
|
+
this._renderDataItems();
|
|
151
|
+
this._wasDataMode = true;
|
|
152
|
+
} else if (this._wasDataMode) {
|
|
153
|
+
this._renderDataItems();
|
|
154
|
+
this._wasDataMode = false;
|
|
155
|
+
}
|
|
156
|
+
}
|
|
157
|
+
}
|
|
158
|
+
|
|
159
|
+
private _resolveTarget(): void {
|
|
160
|
+
const value = this.for?.trim();
|
|
161
|
+
if (!value) {
|
|
162
|
+
this._targetEl = null;
|
|
163
|
+
return;
|
|
164
|
+
}
|
|
165
|
+
const root = (this.getRootNode() as Document | ShadowRoot) ?? document;
|
|
166
|
+
if (/^[#.\[:]/.test(value)) {
|
|
167
|
+
this._targetEl = (root as ParentNode).querySelector(value);
|
|
168
|
+
} else if ('getElementById' in root) {
|
|
169
|
+
this._targetEl = (root as Document | ShadowRoot).getElementById(value);
|
|
170
|
+
} else {
|
|
171
|
+
this._targetEl = document.getElementById(value);
|
|
172
|
+
}
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
public get targetElement(): Element | null {
|
|
176
|
+
return this._targetEl;
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
private _attachTriggers(): void {
|
|
180
|
+
this._detachTriggers?.();
|
|
181
|
+
this._detachTriggers = undefined;
|
|
182
|
+
|
|
183
|
+
const trigger = this.trigger;
|
|
184
|
+
if (trigger === 'manual') return;
|
|
185
|
+
|
|
186
|
+
const cleanups: Array<() => void> = [];
|
|
187
|
+
|
|
188
|
+
if (trigger === 'global') {
|
|
189
|
+
const skipSelector = this.skipOn?.trim() ?? '';
|
|
190
|
+
const isInteractive = (el: Element | null): boolean => {
|
|
191
|
+
if (!el || !skipSelector) return false;
|
|
192
|
+
return !!el.closest(skipSelector);
|
|
193
|
+
};
|
|
194
|
+
const handler = (e: Event) => {
|
|
195
|
+
const me = e as MouseEvent;
|
|
196
|
+
if (isInteractive(me.target as Element | null)) return;
|
|
197
|
+
me.preventDefault();
|
|
198
|
+
if (this._open) return;
|
|
199
|
+
this.open({
|
|
200
|
+
x: me.clientX,
|
|
201
|
+
y: me.clientY,
|
|
202
|
+
target: (me.target as Element | null) ?? undefined,
|
|
203
|
+
originalEvent: me,
|
|
204
|
+
});
|
|
205
|
+
};
|
|
206
|
+
window.addEventListener('contextmenu', handler, true);
|
|
207
|
+
cleanups.push(() => window.removeEventListener('contextmenu', handler, true));
|
|
208
|
+
this._detachTriggers = () => cleanups.forEach(fn => fn());
|
|
209
|
+
return;
|
|
210
|
+
}
|
|
211
|
+
|
|
212
|
+
const target = this._targetEl;
|
|
213
|
+
if (!target) return;
|
|
214
|
+
|
|
215
|
+
const wantsContext = trigger === 'right' || trigger === 'both';
|
|
216
|
+
const wantsClick = trigger === 'left' || trigger === 'both';
|
|
217
|
+
|
|
218
|
+
if (wantsContext) {
|
|
219
|
+
const handler = (e: Event) => {
|
|
220
|
+
const me = e as MouseEvent;
|
|
221
|
+
me.preventDefault();
|
|
222
|
+
this.open({
|
|
223
|
+
x: me.clientX,
|
|
224
|
+
y: me.clientY,
|
|
225
|
+
target,
|
|
226
|
+
originalEvent: me,
|
|
227
|
+
});
|
|
228
|
+
};
|
|
229
|
+
target.addEventListener('contextmenu', handler);
|
|
230
|
+
cleanups.push(() => target.removeEventListener('contextmenu', handler));
|
|
231
|
+
}
|
|
232
|
+
|
|
233
|
+
if (wantsClick) {
|
|
234
|
+
const handler = (e: Event) => {
|
|
235
|
+
const me = e as MouseEvent;
|
|
236
|
+
this.open({
|
|
237
|
+
x: me.clientX,
|
|
238
|
+
y: me.clientY,
|
|
239
|
+
target,
|
|
240
|
+
originalEvent: me,
|
|
241
|
+
});
|
|
242
|
+
};
|
|
243
|
+
target.addEventListener('click', handler);
|
|
244
|
+
cleanups.push(() => target.removeEventListener('click', handler));
|
|
245
|
+
}
|
|
246
|
+
|
|
247
|
+
if (cleanups.length === 0) return;
|
|
248
|
+
this._detachTriggers = () => cleanups.forEach(fn => fn());
|
|
249
|
+
}
|
|
250
|
+
|
|
251
|
+
public override disconnectedCallback(): void {
|
|
252
|
+
super.disconnectedCallback();
|
|
253
|
+
NileContextMenu._openInstances.delete(this);
|
|
254
|
+
this._open = false;
|
|
255
|
+
this._detachTriggers?.();
|
|
256
|
+
this._detachTriggers = undefined;
|
|
257
|
+
this._lightObserver?.disconnect();
|
|
258
|
+
this._lightObserver = undefined;
|
|
259
|
+
this._menuObserver?.disconnect();
|
|
260
|
+
this._menuObserver = undefined;
|
|
261
|
+
this._menuContainerRef?.removeEventListener('click', this._onMenuClick);
|
|
262
|
+
this._menuContainerRef?.removeEventListener('mouseover', this._onMenuMouseOver);
|
|
263
|
+
this._menuContainerRef?.removeEventListener(
|
|
264
|
+
'nile-change',
|
|
265
|
+
this._onDescendantSelect as EventListener,
|
|
266
|
+
);
|
|
267
|
+
document.removeEventListener('pointerdown', this._onOutsidePointer, true);
|
|
268
|
+
document.removeEventListener('keydown', this._onKeydown, true);
|
|
269
|
+
window.removeEventListener('scroll', this._onScroll, true);
|
|
270
|
+
this._proxyEl?.remove();
|
|
271
|
+
this._proxyEl = undefined;
|
|
272
|
+
}
|
|
273
|
+
|
|
274
|
+
private _menuContainerRef: HTMLDivElement | null = null;
|
|
275
|
+
|
|
276
|
+
protected override firstUpdated(): void {
|
|
277
|
+
this._menuContainerRef = this.renderRoot.querySelector('.nile-context-menu__menu') as HTMLDivElement | null;
|
|
278
|
+
if (this._isDataMode) {
|
|
279
|
+
this._renderDataItems();
|
|
280
|
+
this._wasDataMode = true;
|
|
281
|
+
} else {
|
|
282
|
+
this._relocateLightChildren();
|
|
283
|
+
}
|
|
284
|
+
if (this._menuContainerRef) {
|
|
285
|
+
this._menuObserver = new MutationObserver(() => this._parseChildren());
|
|
286
|
+
this._menuObserver.observe(this._menuContainerRef, {
|
|
287
|
+
childList: true,
|
|
288
|
+
subtree: true,
|
|
289
|
+
attributes: true,
|
|
290
|
+
attributeFilter: OBSERVED_ITEM_ATTRS,
|
|
291
|
+
});
|
|
292
|
+
this._menuContainerRef.addEventListener('click', this._onMenuClick);
|
|
293
|
+
this._menuContainerRef.addEventListener('mouseover', this._onMenuMouseOver);
|
|
294
|
+
this._menuContainerRef.addEventListener(
|
|
295
|
+
'nile-change',
|
|
296
|
+
this._onDescendantSelect as EventListener,
|
|
297
|
+
);
|
|
298
|
+
}
|
|
299
|
+
this._parseChildren();
|
|
300
|
+
}
|
|
301
|
+
|
|
302
|
+
private _ensureProxy(): void {
|
|
303
|
+
if (this._proxyEl) return;
|
|
304
|
+
const el = document.createElement('div');
|
|
305
|
+
el.id = this._proxyId;
|
|
306
|
+
el.setAttribute('aria-hidden', 'true');
|
|
307
|
+
el.style.cssText =
|
|
308
|
+
'position: fixed; top: 0; left: 0; width: 0; height: 0; pointer-events: none;';
|
|
309
|
+
document.body.appendChild(el);
|
|
310
|
+
this._proxyEl = el;
|
|
311
|
+
}
|
|
312
|
+
|
|
313
|
+
private _positionProxyAt(x: number, y: number): void {
|
|
314
|
+
if (!this._proxyEl) return;
|
|
315
|
+
this._proxyEl.style.left = `${x}px`;
|
|
316
|
+
this._proxyEl.style.top = `${y}px`;
|
|
317
|
+
}
|
|
318
|
+
|
|
319
|
+
private _relocateLightChildren(): void {
|
|
320
|
+
if (!this._menuContainerRef) return;
|
|
321
|
+
const kids = Array.from(this.children);
|
|
322
|
+
if (kids.length === 0) return;
|
|
323
|
+
for (const kid of kids) {
|
|
324
|
+
this._menuContainerRef.appendChild(kid);
|
|
325
|
+
}
|
|
326
|
+
}
|
|
327
|
+
|
|
328
|
+
private _renderDataItems(): void {
|
|
329
|
+
const container = this._menuContainerRef;
|
|
330
|
+
if (!container) return;
|
|
331
|
+
while (container.firstChild) container.removeChild(container.firstChild);
|
|
332
|
+
for (const entry of this.items) {
|
|
333
|
+
const node = this._createDataNode(entry);
|
|
334
|
+
if (node) container.appendChild(node);
|
|
335
|
+
}
|
|
336
|
+
}
|
|
337
|
+
|
|
338
|
+
private _createDataNode(entry: NileContextMenuData): HTMLElement | null {
|
|
339
|
+
if (entry.type === 'group') {
|
|
340
|
+
const group = document.createElement('nile-context-menu-group') as NileContextMenuGroup;
|
|
341
|
+
if (entry.name) group.label = entry.name;
|
|
342
|
+
for (const item of entry.data ?? []) {
|
|
343
|
+
group.appendChild(this._createItemNode(item));
|
|
344
|
+
}
|
|
345
|
+
return group;
|
|
346
|
+
}
|
|
347
|
+
return null;
|
|
348
|
+
}
|
|
349
|
+
|
|
350
|
+
private _createItemNode(data: NileContextMenuItemData): NileContextMenuItem {
|
|
351
|
+
const item = document.createElement('nile-context-menu-item') as NileContextMenuItem;
|
|
352
|
+
if (data.id) item.id = data.id;
|
|
353
|
+
item.value = data.value ?? data.id ?? '';
|
|
354
|
+
if (data.disabled) item.disabled = true;
|
|
355
|
+
if (data.icon) {
|
|
356
|
+
const glyph = document.createElement('nile-glyph');
|
|
357
|
+
glyph.setAttribute('slot', 'icon');
|
|
358
|
+
glyph.setAttribute('name', data.icon);
|
|
359
|
+
if (data.iconSet) glyph.setAttribute('set', data.iconSet);
|
|
360
|
+
if (data.iconSize) glyph.setAttribute('size', data.iconSize);
|
|
361
|
+
if (data.iconMethod) glyph.setAttribute('method', data.iconMethod);
|
|
362
|
+
if (data.iconColor) glyph.setAttribute('color', data.iconColor);
|
|
363
|
+
item.appendChild(glyph);
|
|
364
|
+
}
|
|
365
|
+
item.appendChild(document.createTextNode(data.label));
|
|
366
|
+
if (data.submenu && data.submenu.length > 0) {
|
|
367
|
+
const submenu = document.createElement('nile-context-submenu');
|
|
368
|
+
for (const entry of data.submenu) {
|
|
369
|
+
const node = this._createDataNode(entry);
|
|
370
|
+
if (node) submenu.appendChild(node);
|
|
371
|
+
}
|
|
372
|
+
item.appendChild(submenu);
|
|
373
|
+
}
|
|
374
|
+
return item;
|
|
375
|
+
}
|
|
376
|
+
|
|
377
|
+
private _parseChildren(): void {
|
|
378
|
+
const root: Element = this._menuContainerRef ?? this;
|
|
379
|
+
const all = Array.from(root.querySelectorAll(ITEM_TAG)) as NileContextMenuItem[];
|
|
380
|
+
this._items = all.filter(item => !this._isInsideDescendantSubmenu(item, root));
|
|
381
|
+
}
|
|
382
|
+
|
|
383
|
+
private _isInsideDescendantSubmenu(item: Element, container: Element): boolean {
|
|
384
|
+
let cur: Element | null = item.parentElement;
|
|
385
|
+
while (cur && cur !== container) {
|
|
386
|
+
if (cur.tagName.toLowerCase() === SUBMENU_TAG) return true;
|
|
387
|
+
cur = cur.parentElement;
|
|
388
|
+
}
|
|
389
|
+
return false;
|
|
390
|
+
}
|
|
391
|
+
|
|
392
|
+
public get menuItems(): readonly NileContextMenuItem[] {
|
|
393
|
+
return this._items;
|
|
394
|
+
}
|
|
395
|
+
|
|
396
|
+
public getItemByValue(value: string): NileContextMenuItem | undefined {
|
|
397
|
+
return this._items.find(item => item.value === value);
|
|
398
|
+
}
|
|
399
|
+
|
|
400
|
+
public get isOpen(): boolean {
|
|
401
|
+
return this._open;
|
|
402
|
+
}
|
|
403
|
+
|
|
404
|
+
public open(options: NileContextMenuOpenOptions): void {
|
|
405
|
+
if (this._open) return;
|
|
406
|
+
for (const other of NileContextMenu._openInstances) {
|
|
407
|
+
if (other !== this) other.close('programmatic');
|
|
408
|
+
}
|
|
409
|
+
this._previouslyFocused = document.activeElement as HTMLElement | null;
|
|
410
|
+
this._openContext = {
|
|
411
|
+
x: options.x,
|
|
412
|
+
y: options.y,
|
|
413
|
+
target: options.target ?? null,
|
|
414
|
+
originalEvent: options.originalEvent ?? null,
|
|
415
|
+
};
|
|
416
|
+
this._ensureProxy();
|
|
417
|
+
this._positionProxyAt(options.x, options.y);
|
|
418
|
+
this._open = true;
|
|
419
|
+
NileContextMenu._openInstances.add(this);
|
|
420
|
+
document.addEventListener('pointerdown', this._onOutsidePointer, true);
|
|
421
|
+
document.addEventListener('keydown', this._onKeydown, true);
|
|
422
|
+
requestAnimationFrame(() => {
|
|
423
|
+
if (this._open) window.addEventListener('scroll', this._onScroll, true);
|
|
424
|
+
});
|
|
425
|
+
this.emit('nile-change', { type: 'open', ...this._openContext });
|
|
426
|
+
}
|
|
427
|
+
|
|
428
|
+
public close(reason: NileContextMenuCloseReason = 'programmatic'): void {
|
|
429
|
+
if (!this._open) return;
|
|
430
|
+
this._open = false;
|
|
431
|
+
this._openContext = null;
|
|
432
|
+
NileContextMenu._openInstances.delete(this);
|
|
433
|
+
const subs = this._menuContainerRef?.querySelectorAll(SUBMENU_TAG);
|
|
434
|
+
subs?.forEach(s => (s as unknown as { closeSubmenu?: () => void }).closeSubmenu?.());
|
|
435
|
+
document.removeEventListener('pointerdown', this._onOutsidePointer, true);
|
|
436
|
+
document.removeEventListener('keydown', this._onKeydown, true);
|
|
437
|
+
window.removeEventListener('scroll', this._onScroll, true);
|
|
438
|
+
const active = document.activeElement;
|
|
439
|
+
const focusInMenu =
|
|
440
|
+
active === this ||
|
|
441
|
+
this._items.some(item => item === active || item.shadowRoot?.contains(active as Node));
|
|
442
|
+
if (focusInMenu) {
|
|
443
|
+
this._previouslyFocused?.focus?.();
|
|
444
|
+
}
|
|
445
|
+
this._previouslyFocused = null;
|
|
446
|
+
this.emit('nile-change', { type: 'close', reason });
|
|
447
|
+
}
|
|
448
|
+
|
|
449
|
+
private _enabledItems(): NileContextMenuItem[] {
|
|
450
|
+
return this._items.filter(item => !item.disabled);
|
|
451
|
+
}
|
|
452
|
+
|
|
453
|
+
private _getFocusedItem(): NileContextMenuItem | null {
|
|
454
|
+
const active = document.activeElement;
|
|
455
|
+
for (const item of this._items) {
|
|
456
|
+
if (item === active || item.shadowRoot?.contains(active as Node)) return item;
|
|
457
|
+
}
|
|
458
|
+
return null;
|
|
459
|
+
}
|
|
460
|
+
|
|
461
|
+
private _getFocusedItemAnyLevel(): {
|
|
462
|
+
item: NileContextMenuItem | null;
|
|
463
|
+
container: Element | null;
|
|
464
|
+
} {
|
|
465
|
+
let node: Element | null = document.activeElement as Element | null;
|
|
466
|
+
while (node && node.tagName?.toLowerCase() !== ITEM_TAG) {
|
|
467
|
+
const root = node.getRootNode();
|
|
468
|
+
node = root instanceof ShadowRoot ? root.host : node.parentElement;
|
|
469
|
+
}
|
|
470
|
+
if (!node) return { item: null, container: null };
|
|
471
|
+
const container = node.closest(`.${MENU_CONTAINER_CLASS}`);
|
|
472
|
+
return { item: node as NileContextMenuItem, container };
|
|
473
|
+
}
|
|
474
|
+
|
|
475
|
+
private _enabledItemsIn(container: Element): NileContextMenuItem[] {
|
|
476
|
+
const all = Array.from(container.querySelectorAll(ITEM_TAG)) as NileContextMenuItem[];
|
|
477
|
+
return all.filter(item => {
|
|
478
|
+
if (item.disabled) return false;
|
|
479
|
+
return !this._isInsideDescendantSubmenu(item, container);
|
|
480
|
+
});
|
|
481
|
+
}
|
|
482
|
+
|
|
483
|
+
private _focusFirstEnabled(): void {
|
|
484
|
+
const enabled = this._enabledItems();
|
|
485
|
+
enabled[0]?.focus();
|
|
486
|
+
}
|
|
487
|
+
|
|
488
|
+
private _moveFocus(delta: number): void {
|
|
489
|
+
const { item: focused, container } = this._getFocusedItemAnyLevel();
|
|
490
|
+
const root: Element = container ?? this._menuContainerRef ?? this;
|
|
491
|
+
const enabled = this._enabledItemsIn(root);
|
|
492
|
+
if (enabled.length === 0) return;
|
|
493
|
+
const current = focused ? enabled.indexOf(focused) : -1;
|
|
494
|
+
const next = current + delta;
|
|
495
|
+
const wrapped = ((next % enabled.length) + enabled.length) % enabled.length;
|
|
496
|
+
enabled[wrapped].focus();
|
|
497
|
+
}
|
|
498
|
+
|
|
499
|
+
private _onKeydown = (e: KeyboardEvent): void => {
|
|
500
|
+
if (!this._open) return;
|
|
501
|
+
switch (e.key) {
|
|
502
|
+
case 'Escape':
|
|
503
|
+
e.preventDefault();
|
|
504
|
+
this.close('escape');
|
|
505
|
+
return;
|
|
506
|
+
case 'Tab':
|
|
507
|
+
this.close('programmatic');
|
|
508
|
+
return;
|
|
509
|
+
case 'ArrowDown':
|
|
510
|
+
e.preventDefault();
|
|
511
|
+
this._moveFocus(1);
|
|
512
|
+
return;
|
|
513
|
+
case 'ArrowUp':
|
|
514
|
+
e.preventDefault();
|
|
515
|
+
this._moveFocus(-1);
|
|
516
|
+
return;
|
|
517
|
+
case 'ArrowRight': {
|
|
518
|
+
const { item } = this._getFocusedItemAnyLevel();
|
|
519
|
+
const sub = item?.querySelector(`:scope > ${SUBMENU_TAG}`) as
|
|
520
|
+
| (HTMLElement & { openSubmenu?: () => void })
|
|
521
|
+
| null;
|
|
522
|
+
if (!sub) return;
|
|
523
|
+
e.preventDefault();
|
|
524
|
+
sub.openSubmenu?.();
|
|
525
|
+
requestAnimationFrame(() => {
|
|
526
|
+
(sub as unknown as { focusFirstItem?: () => void }).focusFirstItem?.();
|
|
527
|
+
});
|
|
528
|
+
return;
|
|
529
|
+
}
|
|
530
|
+
case 'ArrowLeft': {
|
|
531
|
+
const { item, container } = this._getFocusedItemAnyLevel();
|
|
532
|
+
if (!item || !container) return;
|
|
533
|
+
const sub = this._findSubmenuOwning(container);
|
|
534
|
+
if (!sub) return;
|
|
535
|
+
e.preventDefault();
|
|
536
|
+
sub.closeSubmenu?.();
|
|
537
|
+
sub.parentItem?.focus();
|
|
538
|
+
return;
|
|
539
|
+
}
|
|
540
|
+
case 'Enter':
|
|
541
|
+
case ' ': {
|
|
542
|
+
e.preventDefault();
|
|
543
|
+
const { item } = this._getFocusedItemAnyLevel();
|
|
544
|
+
if (!item || item.disabled) return;
|
|
545
|
+
const sub = item.querySelector(`:scope > ${SUBMENU_TAG}`) as
|
|
546
|
+
| (HTMLElement & { openSubmenu?: () => void })
|
|
547
|
+
| null;
|
|
548
|
+
if (sub) {
|
|
549
|
+
sub.openSubmenu?.();
|
|
550
|
+
requestAnimationFrame(() => {
|
|
551
|
+
const panel = sub.shadowRoot?.querySelector(`.${MENU_CONTAINER_CLASS}`);
|
|
552
|
+
const first = panel
|
|
553
|
+
? (this._enabledItemsIn(panel)[0] as NileContextMenuItem | undefined)
|
|
554
|
+
: undefined;
|
|
555
|
+
first?.focus();
|
|
556
|
+
});
|
|
557
|
+
return;
|
|
558
|
+
}
|
|
559
|
+
this._selectItem(item);
|
|
560
|
+
return;
|
|
561
|
+
}
|
|
562
|
+
}
|
|
563
|
+
};
|
|
564
|
+
|
|
565
|
+
private _findSubmenuOwning(container: Element):
|
|
566
|
+
| (HTMLElement & { closeSubmenu?: () => void; parentItem?: NileContextMenuItem | null })
|
|
567
|
+
| null {
|
|
568
|
+
type SubCtor = {
|
|
569
|
+
findByContainer?: (c: Element) => HTMLElement & {
|
|
570
|
+
closeSubmenu?: () => void;
|
|
571
|
+
parentItem?: NileContextMenuItem | null;
|
|
572
|
+
} | null;
|
|
573
|
+
};
|
|
574
|
+
const customElements = window.customElements;
|
|
575
|
+
const ctor = customElements.get(SUBMENU_TAG) as unknown as SubCtor | undefined;
|
|
576
|
+
return ctor?.findByContainer?.(container) ?? null;
|
|
577
|
+
}
|
|
578
|
+
|
|
579
|
+
private _onScroll = (e: Event): void => {
|
|
580
|
+
if (!this._open) return;
|
|
581
|
+
const path = e.composedPath();
|
|
582
|
+
// Scroll inside the root menu container?
|
|
583
|
+
if (this._menuContainerRef && path.includes(this._menuContainerRef)) return;
|
|
584
|
+
// Scroll inside any descendant submenu's portaled panel?
|
|
585
|
+
for (const node of path) {
|
|
586
|
+
if (node instanceof HTMLElement && node.classList?.contains(MENU_CONTAINER_CLASS)) {
|
|
587
|
+
return;
|
|
588
|
+
}
|
|
589
|
+
}
|
|
590
|
+
this.close('programmatic');
|
|
591
|
+
};
|
|
592
|
+
|
|
593
|
+
private _onOutsidePointer = (e: Event): void => {
|
|
594
|
+
if (!this._open) return;
|
|
595
|
+
const path = e.composedPath();
|
|
596
|
+
if (this._menuContainerRef && path.includes(this._menuContainerRef)) return;
|
|
597
|
+
for (const node of path) {
|
|
598
|
+
if (
|
|
599
|
+
node instanceof HTMLElement &&
|
|
600
|
+
node.classList?.contains(MENU_CONTAINER_CLASS)
|
|
601
|
+
) {
|
|
602
|
+
return;
|
|
603
|
+
}
|
|
604
|
+
}
|
|
605
|
+
this.close('outside-click');
|
|
606
|
+
};
|
|
607
|
+
|
|
608
|
+
private _onDescendantSelect = (e: CustomEvent): void => {
|
|
609
|
+
if (e.detail?.type !== 'click') return;
|
|
610
|
+
this.close('select');
|
|
611
|
+
};
|
|
612
|
+
private _stop(e: Event): void {
|
|
613
|
+
e.stopPropagation();
|
|
614
|
+
}
|
|
615
|
+
|
|
616
|
+
private _onPanelShown = (e: Event): void => {
|
|
617
|
+
e.stopPropagation();
|
|
618
|
+
};
|
|
619
|
+
|
|
620
|
+
private _onMenuMouseOver = (e: MouseEvent): void => {
|
|
621
|
+
const item = e
|
|
622
|
+
.composedPath()
|
|
623
|
+
.find(
|
|
624
|
+
node =>
|
|
625
|
+
node instanceof HTMLElement &&
|
|
626
|
+
node.tagName.toLowerCase() === ITEM_TAG
|
|
627
|
+
) as NileContextMenuItem | undefined;
|
|
628
|
+
if (!item || item.disabled) return;
|
|
629
|
+
item.focus();
|
|
630
|
+
};
|
|
631
|
+
|
|
632
|
+
private _onMenuClick = (e: MouseEvent): void => {
|
|
633
|
+
const item = e
|
|
634
|
+
.composedPath()
|
|
635
|
+
.find(
|
|
636
|
+
node =>
|
|
637
|
+
node instanceof HTMLElement &&
|
|
638
|
+
node.tagName.toLowerCase() === ITEM_TAG
|
|
639
|
+
) as NileContextMenuItem | undefined;
|
|
640
|
+
|
|
641
|
+
if (!item || item.disabled) return;
|
|
642
|
+
if (item.querySelector(`:scope > ${SUBMENU_TAG}`)) return;
|
|
643
|
+
this._selectItem(item);
|
|
644
|
+
};
|
|
645
|
+
|
|
646
|
+
private _selectItem(item: NileContextMenuItem): void {
|
|
647
|
+
const detail = {
|
|
648
|
+
id: item.id,
|
|
649
|
+
value: item.value,
|
|
650
|
+
name: (item.textContent ?? '').trim(),
|
|
651
|
+
target: this._openContext?.target ?? null,
|
|
652
|
+
originalEvent: this._openContext?.originalEvent ?? null,
|
|
653
|
+
};
|
|
654
|
+
try {
|
|
655
|
+
item.onSelect?.(detail);
|
|
656
|
+
} catch (err) {
|
|
657
|
+
console.error('[nile-context-menu] onSelect callback threw:', err);
|
|
658
|
+
}
|
|
659
|
+
this.emit('nile-change', { type: 'click', ...detail });
|
|
660
|
+
this.close('select');
|
|
661
|
+
}
|
|
662
|
+
|
|
663
|
+
public render(): TemplateResult {
|
|
664
|
+
return html`
|
|
665
|
+
<nile-floating-panel
|
|
666
|
+
for=${this._proxyId}
|
|
667
|
+
trigger="manual"
|
|
668
|
+
placement="bottom-start"
|
|
669
|
+
panelClass="nile-context-menu-panel"
|
|
670
|
+
.zIndex=${this.zIndex}
|
|
671
|
+
?open=${this._open}
|
|
672
|
+
.interactive=${true}
|
|
673
|
+
.hideOnClick=${false}
|
|
674
|
+
.closeOnEscape=${false}
|
|
675
|
+
.arrow=${'none'}
|
|
676
|
+
.distance=${0}
|
|
677
|
+
@nile-init=${this._stop}
|
|
678
|
+
@nile-destroy=${this._stop}
|
|
679
|
+
@nile-show=${this._stop}
|
|
680
|
+
@nile-hide=${this._stop}
|
|
681
|
+
@nile-after-show=${this._onPanelShown}
|
|
682
|
+
@nile-after-hide=${this._stop}
|
|
683
|
+
@nile-toggle=${this._stop}
|
|
684
|
+
@nile-visibility-change=${this._stop}
|
|
685
|
+
>
|
|
686
|
+
<div class="nile-context-menu__menu" role="menu"></div>
|
|
687
|
+
</nile-floating-panel>
|
|
688
|
+
`;
|
|
689
|
+
}
|
|
690
|
+
}
|
|
691
|
+
|
|
692
|
+
export default NileContextMenu;
|
|
693
|
+
|
|
694
|
+
declare global {
|
|
695
|
+
interface HTMLElementTagNameMap {
|
|
696
|
+
'nile-context-menu': NileContextMenu;
|
|
697
|
+
}
|
|
698
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export { NileContextMenuGroup } from './nile-context-menu-group';
|