bridgerte 0.9.12 → 0.9.13

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/dom.cjs CHANGED
@@ -1,2 +1,2 @@
1
- "use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const e=require("./index-BBzHVrSi.cjs"),t=require("./index-DIJtCo91.cjs");exports.createFloatingLayer=e.createFloatingLayer;exports.createRichTextEditor=e.createRichTextEditor;exports.createWebViewBridgeRuntime=e.createWebViewBridgeRuntime;exports.createRichTextToolbar=t.createRichTextToolbar;exports.hasMeaningfulHtmlContent=t.hasMeaningfulHtmlContent;
1
+ "use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const e=require("./index-BBzHVrSi.cjs"),t=require("./index-DZnoOnKI.cjs");exports.createFloatingLayer=e.createFloatingLayer;exports.createRichTextEditor=e.createRichTextEditor;exports.createWebViewBridgeRuntime=e.createWebViewBridgeRuntime;exports.createRichTextToolbar=t.createRichTextToolbar;exports.hasMeaningfulHtmlContent=t.hasMeaningfulHtmlContent;
2
2
  //# sourceMappingURL=dom.cjs.map
package/dist/dom.js CHANGED
@@ -1,5 +1,5 @@
1
1
  import { c as t, a as r, b as o } from "./index-DDfgjstx.js";
2
- import { c as i, h as n } from "./index-CQie9v_I.js";
2
+ import { c as i, h as n } from "./index-CXVSZZ93.js";
3
3
  export {
4
4
  t as createFloatingLayer,
5
5
  r as createRichTextEditor,
@@ -0,0 +1,337 @@
1
+ import { defaultMenuSchema as et, resolveToolbarMenu as ot } from "./native-spec.js";
2
+ import { g as H, d as N, e as K, r as nt, f as q, h as rt, c as at } from "./index-DDfgjstx.js";
3
+ const X = {
4
+ toolbarGroup: `
5
+ <svg
6
+ aria-hidden="true"
7
+ class="lucide lucide-ellipsis"
8
+ xmlns="http://www.w3.org/2000/svg"
9
+ width="18"
10
+ height="18"
11
+ viewBox="0 0 24 24"
12
+ fill="none"
13
+ stroke="currentColor"
14
+ stroke-width="2"
15
+ stroke-linecap="round"
16
+ stroke-linejoin="round"
17
+ >
18
+ <circle cx="12" cy="12" r="1" />
19
+ <circle cx="19" cy="12" r="1" />
20
+ <circle cx="5" cy="12" r="1" />
21
+ </svg>
22
+ `,
23
+ chevronDown: `
24
+ <svg
25
+ aria-hidden="true"
26
+ class="lucide lucide-chevron-down"
27
+ xmlns="http://www.w3.org/2000/svg"
28
+ width="12"
29
+ height="12"
30
+ viewBox="0 0 24 24"
31
+ fill="none"
32
+ stroke="currentColor"
33
+ stroke-width="2"
34
+ stroke-linecap="round"
35
+ stroke-linejoin="round"
36
+ >
37
+ <path d="m6 9 6 6 6-6" />
38
+ </svg>
39
+ `
40
+ }, z = (t) => Array.from(
41
+ t.querySelectorAll(".bridgerte__toolbar-group-menu-item")
42
+ ).filter((e) => !e.disabled), Y = (t, e) => {
43
+ const c = z(t)[e];
44
+ c && c.focus();
45
+ }, st = (t, e, i, c, d) => {
46
+ var s, u;
47
+ const n = z(e), r = document.activeElement instanceof HTMLButtonElement ? n.indexOf(document.activeElement) : -1, m = (g) => {
48
+ var o;
49
+ if (n.length === 0) return;
50
+ const v = ((r >= 0 ? r : 0) + g + n.length) % n.length;
51
+ (o = n[v]) == null || o.focus();
52
+ };
53
+ switch (t.key) {
54
+ case "ArrowDown":
55
+ t.preventDefault(), m(1);
56
+ break;
57
+ case "ArrowUp":
58
+ t.preventDefault(), m(-1);
59
+ break;
60
+ case "Home":
61
+ t.preventDefault(), (s = n[0]) == null || s.focus();
62
+ break;
63
+ case "End":
64
+ t.preventDefault(), (u = n.at(-1)) == null || u.focus();
65
+ break;
66
+ case "Enter":
67
+ case " ":
68
+ document.activeElement instanceof HTMLButtonElement && (t.preventDefault(), c(document.activeElement), d(), i.focus());
69
+ break;
70
+ case "Escape":
71
+ t.preventDefault(), i.focus(), d();
72
+ break;
73
+ }
74
+ }, L = "button[data-bridgerte-toolbar-group-id]", it = "button[data-bridgerte-toolbar-item-id]", I = [
75
+ it,
76
+ ".bridgerte__toolbar-group-menu-item"
77
+ ].join(","), lt = (t, e) => {
78
+ const i = t.map((c) => H(c, e));
79
+ return {
80
+ active: i.some((c) => c.active),
81
+ disabled: i.length > 0 && i.every((c) => c.disabled)
82
+ };
83
+ }, ct = (t, e, i, c, d) => {
84
+ const n = H(e, i), r = document.createElement("button"), m = c[e.icon] ?? N[e.icon];
85
+ r.type = "button", r.className = "bridgerte__toolbar-button", r.disabled = n.disabled, r.dataset.active = String(n.active), r.dataset.bridgerteToolbarItemId = e.id, r.setAttribute("aria-label", e.label), r.setAttribute("aria-pressed", String(n.active)), d && (r.dataset.tooltip = e.label), K(r, m, e.label), t.append(r);
86
+ }, dt = (t, e, i, c, d) => {
87
+ const n = document.createElement("button"), r = lt(e.items, i), m = e.icon ? c[e.icon] ?? N[e.icon] : X.toolbarGroup, s = document.createElement("span");
88
+ n.type = "button", n.className = "bridgerte__toolbar-button bridgerte__toolbar-group-button", n.disabled = r.disabled, n.dataset.active = String(r.active), n.dataset.bridgerteToolbarGroupId = e.key, n.dataset.open = "false", n.setAttribute("aria-label", e.title), n.setAttribute("aria-haspopup", "menu"), n.setAttribute("aria-expanded", "false"), n.setAttribute("aria-pressed", String(r.active)), d && (n.dataset.tooltip = e.title), K(n, m, e.title), s.className = "bridgerte__toolbar-group-indicator", s.setAttribute("aria-hidden", "true"), s.innerHTML = X.chevronDown, n.append(s), t.append(n);
89
+ }, ut = (t, e, i, c, d) => {
90
+ t.textContent = "", e.forEach((n) => {
91
+ if (n.type === "separator") {
92
+ const m = document.createElement("span");
93
+ m.className = "bridgerte__toolbar-separator", m.dataset.separatorId = n.key, m.setAttribute("aria-hidden", "true"), t.append(m);
94
+ return;
95
+ }
96
+ if (n.type === "button") {
97
+ ct(
98
+ t,
99
+ n.item,
100
+ i,
101
+ c,
102
+ d
103
+ );
104
+ return;
105
+ }
106
+ const r = document.createElement("div");
107
+ r.className = "bridgerte__toolbar-group", r.dataset.group = n.key, r.setAttribute("aria-label", n.title), t.append(r), dt(
108
+ r,
109
+ n,
110
+ i,
111
+ c,
112
+ d
113
+ );
114
+ });
115
+ }, G = (t, e, i, c) => {
116
+ if (t.textContent = "", !e) {
117
+ t.dataset.visible = "false";
118
+ return;
119
+ }
120
+ e.items.forEach((d) => {
121
+ const n = H(d, i), r = document.createElement("button"), m = c[d.icon] ?? N[d.icon], s = document.createElement("span");
122
+ r.type = "button", r.className = "bridgerte__menu-item bridgerte__toolbar-group-menu-item", r.disabled = n.disabled, r.dataset.active = String(n.active), r.dataset.bridgerteToolbarItemId = d.id, r.setAttribute("role", "menuitem"), r.setAttribute("aria-label", d.label), r.setAttribute("aria-pressed", String(n.active)), K(r, m, d.label), s.className = "bridgerte__toolbar-group-menu-label", s.textContent = d.label, r.append(s), t.append(r);
123
+ }), t.dataset.visible = "true", t.style.minWidth = `${e.button.offsetWidth}px`;
124
+ }, D = (t, e) => {
125
+ t.querySelectorAll(L).forEach((i) => {
126
+ const c = (e == null ? void 0 : e.groupKey) === i.dataset.bridgerteToolbarGroupId;
127
+ i.dataset.open = String(c), i.setAttribute("aria-expanded", String(c));
128
+ });
129
+ }, U = (t, e) => t.find((i) => i.type === "group" && i.key === e), $ = 8, S = (t, e) => e instanceof Node && t.contains(e), _ = (t, e) => t instanceof Element ? t.closest(e) : null, bt = (t, e, i, c, d) => {
130
+ let n = null, r = null;
131
+ const m = (o) => _(o, L) ?? (S(e, o) ? _(o, I) : null), s = (o) => {
132
+ m(o.target) && o.preventDefault();
133
+ }, u = (o) => {
134
+ const p = o.touches[0], h = _(o.target, L), T = S(e, o.target) ? _(o.target, I) : null, y = h ?? T;
135
+ !y || !p || (o.preventDefault(), n = {
136
+ button: y,
137
+ kind: h ? "group" : "menu-item",
138
+ startX: p.clientX,
139
+ startY: p.clientY
140
+ });
141
+ }, g = () => {
142
+ n = null;
143
+ }, f = (o) => {
144
+ if (!n) return;
145
+ const p = n, h = o.changedTouches[0], T = m(o.target);
146
+ if (n = null, !h || T !== p.button) return;
147
+ const y = Math.abs(h.clientX - p.startX), w = Math.abs(h.clientY - p.startY);
148
+ if (!(y > $ || w > $)) {
149
+ if (o.preventDefault(), o.stopPropagation(), r = p.button, p.kind === "group") {
150
+ i(p.button);
151
+ return;
152
+ }
153
+ c(p.button), d();
154
+ }
155
+ }, v = (o) => {
156
+ S(t, o.target) || S(e, o.target) || d();
157
+ };
158
+ return t.addEventListener("pointerdown", s), t.addEventListener("mousedown", s), t.addEventListener("touchstart", u, { passive: !1 }), t.addEventListener("touchend", f), t.addEventListener("touchcancel", g), e.addEventListener("pointerdown", s), e.addEventListener("mousedown", s), e.addEventListener("touchstart", u, { passive: !1 }), e.addEventListener("touchend", f), e.addEventListener("touchcancel", g), document.addEventListener("pointerdown", v, !0), {
159
+ consumeSuppressedClick(o) {
160
+ return r !== o ? !1 : (r = null, !0);
161
+ },
162
+ destroy() {
163
+ t.removeEventListener("pointerdown", s), t.removeEventListener("mousedown", s), t.removeEventListener("touchstart", u), t.removeEventListener("touchend", f), t.removeEventListener("touchcancel", g), e.removeEventListener("pointerdown", s), e.removeEventListener("mousedown", s), e.removeEventListener("touchstart", u), e.removeEventListener("touchend", f), e.removeEventListener("touchcancel", g), document.removeEventListener("pointerdown", v, !0);
164
+ }
165
+ };
166
+ }, pt = 8, mt = 6, ft = () => {
167
+ var t;
168
+ return typeof window < "u" && ((t = window.matchMedia) == null ? void 0 : t.call(window, "(hover: hover) and (pointer: fine)").matches) === !0;
169
+ }, P = (t) => {
170
+ const e = t instanceof Element ? t.closest(I) : null;
171
+ return e instanceof HTMLButtonElement ? e : null;
172
+ }, V = (t) => {
173
+ const e = t instanceof Element ? t.closest(L) : null;
174
+ return e instanceof HTMLButtonElement ? e : null;
175
+ };
176
+ function Et(t, e) {
177
+ const i = e.placement ?? "top", c = nt(e.menuSchema ?? et, {
178
+ menuLabels: e.menuLabels,
179
+ payloadPanelConfig: e.payloadPanelConfig
180
+ }), d = ot(e.toolbarConfig, c), n = d.flatMap((a) => a.type === "button" ? [a.item] : a.type === "group" ? a.items : []), r = e.icons ?? {}, m = ft(), s = document.createElement("div"), u = document.createElement("div"), g = t.closest(".bridgerte") ?? t;
181
+ let f = !1, v = e.editor.getCommandStates(), o = null, p = null;
182
+ const h = q(t, {
183
+ targetSelector: [
184
+ "button[data-bridgerte-toolbar-item-id]",
185
+ "button[data-bridgerte-toolbar-group-id]"
186
+ ].join(",")
187
+ }), T = q(u, {
188
+ targetSelector: ".bridgerte__toolbar-group-menu-item"
189
+ }), y = () => {
190
+ g.append(s), g.append(u);
191
+ }, w = () => {
192
+ p == null || p.setOpen(!1), p == null || p.destroy(), p = null;
193
+ }, M = () => {
194
+ o && (w(), p = at(o.button, u, {
195
+ placement: i === "bottom" ? "top-start" : "bottom-start",
196
+ offset: mt,
197
+ strategy: "fixed"
198
+ }), p.setOpen(!0));
199
+ }, E = () => {
200
+ w(), o = null, G(u, o, v, r), D(t, o);
201
+ }, R = (a) => {
202
+ if (!f && (v = a, o && w(), ut(t, d, a, r, m), y(), o)) {
203
+ const l = Array.from(
204
+ t.querySelectorAll(L)
205
+ ).find((tt) => tt.dataset.bridgerteToolbarGroupId === (o == null ? void 0 : o.groupKey)), b = U(d, o.groupKey);
206
+ o = l && b ? {
207
+ groupKey: o.groupKey,
208
+ button: l,
209
+ items: b.items
210
+ } : null, G(u, o, v, r), D(t, o), o && M();
211
+ }
212
+ }, J = () => {
213
+ f || R(e.editor.getCommandStates());
214
+ }, x = () => {
215
+ s.dataset.visible = "false", s.textContent = "";
216
+ }, Q = (a) => {
217
+ const l = a.dataset.tooltip;
218
+ if (!m || !l) return;
219
+ const b = a.getBoundingClientRect();
220
+ s.textContent = l, s.dataset.visible = "true", s.style.left = `${b.left + b.width / 2}px`, s.style.top = `${b.top - pt}px`;
221
+ }, F = (a) => {
222
+ const l = P(a.target);
223
+ l && Q(l);
224
+ }, O = (a) => {
225
+ const l = a.relatedTarget, b = P(a.target);
226
+ b && l instanceof Node && b.contains(l) || x();
227
+ }, B = (a) => {
228
+ if (!(a instanceof HTMLButtonElement) || a.disabled) return;
229
+ const l = n.find((b) => b.id === a.dataset.bridgerteToolbarItemId);
230
+ if (l) {
231
+ if (l.payloadPanel) {
232
+ const b = a.getBoundingClientRect();
233
+ e.editor.requestPayloadPanel({
234
+ menuId: l.id,
235
+ command: l.command,
236
+ panel: l.payloadPanel,
237
+ currentValues: rt(l, e.editor.getCommandStates()),
238
+ anchorRect: {
239
+ x: b.left,
240
+ y: b.top,
241
+ width: b.width,
242
+ height: b.height
243
+ }
244
+ });
245
+ return;
246
+ }
247
+ e.editor.executeCommand(l.command);
248
+ }
249
+ }, C = (a, l = !1) => {
250
+ const b = U(
251
+ d,
252
+ a.dataset.bridgerteToolbarGroupId
253
+ );
254
+ if (!(!b || a.disabled)) {
255
+ if ((o == null ? void 0 : o.groupKey) === b.key) {
256
+ if (l) {
257
+ Y(u, 0);
258
+ return;
259
+ }
260
+ E();
261
+ return;
262
+ }
263
+ o = {
264
+ groupKey: b.key,
265
+ button: a,
266
+ items: b.items
267
+ }, x(), G(u, o, v, r), D(t, o), M(), l && Y(u, 0);
268
+ }
269
+ }, k = (a) => {
270
+ const l = V(a.target);
271
+ if (l) {
272
+ if (a.preventDefault(), a.stopPropagation(), A.consumeSuppressedClick(l)) return;
273
+ C(l);
274
+ return;
275
+ }
276
+ const b = P(a.target);
277
+ b && (a.preventDefault(), a.stopPropagation(), !A.consumeSuppressedClick(b) && (B(b), E()));
278
+ }, j = (a) => {
279
+ if (o && u.contains(document.activeElement)) {
280
+ st(
281
+ a,
282
+ u,
283
+ o.button,
284
+ B,
285
+ E
286
+ );
287
+ return;
288
+ }
289
+ const l = V(a.target);
290
+ if (l && (a.key === "Enter" || a.key === " " || a.key === "ArrowDown")) {
291
+ a.preventDefault(), a.stopPropagation(), C(l, !0);
292
+ return;
293
+ }
294
+ a.key === "Escape" && E();
295
+ };
296
+ t.classList.add("bridgerte__toolbar"), s.className = "bridgerte__toolbar-tooltip", s.dataset.visible = "false", u.className = "bridgerte__floating-menu bridgerte__toolbar-group-menu", u.dataset.visible = "false", u.setAttribute("role", "menu"), t.dataset.placement = i, t.setAttribute("role", "toolbar"), t.setAttribute(
297
+ "aria-label",
298
+ i === "bottom" ? "BridgeRTE tabbar" : "BridgeRTE toolbar"
299
+ ), t.addEventListener("click", k), u.addEventListener("click", k), document.addEventListener("keydown", j), m && (t.addEventListener("mouseover", F), t.addEventListener("mouseout", O)), t.addEventListener("focusout", x), y();
300
+ const A = bt(
301
+ t,
302
+ u,
303
+ C,
304
+ B,
305
+ E
306
+ ), Z = e.editor.subscribeCommandStateChange(R);
307
+ return {
308
+ update: J,
309
+ destroy() {
310
+ f || (f = !0, Z(), E(), t.removeEventListener("click", k), u.removeEventListener("click", k), document.removeEventListener("keydown", j), A.destroy(), m && (t.removeEventListener("mouseover", F), t.removeEventListener("mouseout", O)), h(), T(), t.removeEventListener("focusout", x), s.remove(), u.remove(), t.classList.remove("bridgerte__toolbar"), delete t.dataset.placement, t.textContent = "", t.removeAttribute("role"), t.removeAttribute("aria-label"));
311
+ }
312
+ };
313
+ }
314
+ const gt = /[\u200B-\u200D\uFEFF]/g, vt = [
315
+ "img",
316
+ "video",
317
+ "audio",
318
+ "iframe",
319
+ "table",
320
+ "pre",
321
+ "code",
322
+ "hr",
323
+ "figure",
324
+ "canvas",
325
+ "svg",
326
+ "math",
327
+ '[data-type="mention"]'
328
+ ].join(","), W = (t) => (t == null ? void 0 : t.replace(gt, "").trim()) ?? "", Tt = (t) => {
329
+ if (!W(t)) return !1;
330
+ const e = document.createElement("template");
331
+ return e.innerHTML = t, !!(W(e.content.textContent) || e.content.querySelector(vt));
332
+ };
333
+ export {
334
+ Et as c,
335
+ Tt as h
336
+ };
337
+ //# sourceMappingURL=index-CXVSZZ93.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index-CXVSZZ93.js","sources":["../../dom/src/menuIcon/uiIcons.ts","../../dom/src/richTextToolbar/groupMenuKeyboard.ts","../../dom/src/richTextToolbar/render.ts","../../dom/src/richTextToolbar/groupMenuInteraction.ts","../../dom/src/richTextToolbar/index.ts","../../dom/src/htmlContent/index.ts"],"sourcesContent":["/**\r\n * DOM UI chrome icon 表。\r\n *\r\n * 这里放 toolbar 收纳入口、下拉箭头这类控件自身图标。它们不属于 `MenuItem.icon`\r\n * 跨端 schema,也不应该进入 `defaultMenuIcons` 的 schema 对齐检查。\r\n */\r\nexport const defaultMenuUiIcons = {\r\n toolbarGroup: `\r\n <svg\r\n aria-hidden=\"true\"\r\n class=\"lucide lucide-ellipsis\"\r\n xmlns=\"http://www.w3.org/2000/svg\"\r\n width=\"18\"\r\n height=\"18\"\r\n viewBox=\"0 0 24 24\"\r\n fill=\"none\"\r\n stroke=\"currentColor\"\r\n stroke-width=\"2\"\r\n stroke-linecap=\"round\"\r\n stroke-linejoin=\"round\"\r\n >\r\n <circle cx=\"12\" cy=\"12\" r=\"1\" />\r\n <circle cx=\"19\" cy=\"12\" r=\"1\" />\r\n <circle cx=\"5\" cy=\"12\" r=\"1\" />\r\n </svg>\r\n `,\r\n chevronDown: `\r\n <svg\r\n aria-hidden=\"true\"\r\n class=\"lucide lucide-chevron-down\"\r\n xmlns=\"http://www.w3.org/2000/svg\"\r\n width=\"12\"\r\n height=\"12\"\r\n viewBox=\"0 0 24 24\"\r\n fill=\"none\"\r\n stroke=\"currentColor\"\r\n stroke-width=\"2\"\r\n stroke-linecap=\"round\"\r\n stroke-linejoin=\"round\"\r\n >\r\n <path d=\"m6 9 6 6 6-6\" />\r\n </svg>\r\n `\r\n} as const;\r\n","export const getEnabledGroupMenuButtons = (menuElement: HTMLElement) => (\r\n Array.from(\r\n menuElement.querySelectorAll<HTMLButtonElement>('.bridgerte__toolbar-group-menu-item')\r\n ).filter((button) => !button.disabled)\r\n);\r\n\r\nexport const focusToolbarGroupMenuItem = (menuElement: HTMLElement, index: number) => {\r\n const buttons = getEnabledGroupMenuButtons(menuElement);\r\n const nextButton = buttons[index];\r\n\r\n if (nextButton) nextButton.focus();\r\n};\r\n\r\nexport const handleToolbarGroupMenuKeyDown = (\n event: KeyboardEvent,\r\n menuElement: HTMLElement,\r\n returnButton: HTMLButtonElement,\r\n executeToolbarButton: (button: HTMLButtonElement) => void,\r\n closeGroupMenu: () => void\r\n) => {\r\n const buttons = getEnabledGroupMenuButtons(menuElement);\r\n const activeIndex = document.activeElement instanceof HTMLButtonElement\r\n ? buttons.indexOf(document.activeElement)\r\n : -1;\r\n const focusByOffset = (offset: number) => {\r\n if (buttons.length === 0) return;\r\n\r\n const baseIndex = activeIndex >= 0 ? activeIndex : 0;\r\n const nextIndex = (baseIndex + offset + buttons.length) % buttons.length;\r\n\r\n buttons[nextIndex]?.focus();\r\n };\r\n\r\n /*\r\n * ARIA menu 语义要求支持 roving focus。这里不改 toolbar 的横向键盘模型,\r\n * 只在已打开的纵向收纳菜单内处理上下方向、首尾跳转和执行/关闭。\r\n */\r\n switch (event.key) {\r\n case 'ArrowDown':\r\n event.preventDefault();\r\n focusByOffset(1);\r\n break;\r\n case 'ArrowUp':\r\n event.preventDefault();\r\n focusByOffset(-1);\r\n break;\r\n case 'Home':\r\n event.preventDefault();\r\n buttons[0]?.focus();\r\n break;\r\n case 'End':\r\n event.preventDefault();\r\n buttons.at(-1)?.focus();\r\n break;\r\n case 'Enter':\r\n case ' ':\r\n if (document.activeElement instanceof HTMLButtonElement) {\r\n event.preventDefault();\r\n executeToolbarButton(document.activeElement);\r\n closeGroupMenu();\r\n returnButton.focus();\r\n }\r\n break;\r\n case 'Escape':\r\n event.preventDefault();\r\n returnButton.focus();\r\n closeGroupMenu();\r\n break;\r\n default:\r\n break;\r\n }\r\n};\r\n","import type { CommandState } from '@bridgerte/core';\r\nimport type { MenuItem, ResolvedToolbarItem } from '@bridgerte/native-spec';\r\nimport {\r\n appendMenuIcon,\r\n defaultMenuIcons,\r\n defaultMenuUiIcons\r\n} from './icons';\r\nimport { getMenuStateForItem } from '../menuRuntime';\r\nimport type {\r\n RichTextToolbarIcons,\r\n ToolbarGroupMenuState\r\n} from './type';\r\n\r\nexport const toolbarGroupButtonSelector = 'button[data-bridgerte-toolbar-group-id]';\r\nexport const toolbarButtonSelector = 'button[data-bridgerte-toolbar-item-id]';\r\n// 可执行按钮同时覆盖 toolbar 主按钮和 group menu 子项,避免浮层子菜单绕过统一命令入口。\r\nexport const toolbarExecutableButtonSelector = [\n toolbarButtonSelector,\n '.bridgerte__toolbar-group-menu-item'\n].join(',');\n\r\nconst getGroupMenuState = (items: MenuItem[], commandStates: CommandState[]) => {\r\n const itemStates = items.map((item) => getMenuStateForItem(item, commandStates));\r\n\r\n return {\r\n active: itemStates.some((state) => state.active),\r\n disabled: itemStates.length > 0 && itemStates.every((state) => state.disabled)\r\n };\r\n};\r\n\r\nconst renderToolbarButton = (\r\n groupElement: HTMLElement,\r\n item: MenuItem,\r\n commandStates: CommandState[],\r\n icons: RichTextToolbarIcons,\r\n enableTooltip: boolean\r\n) => {\r\n const state = getMenuStateForItem(item, commandStates);\r\n const button = document.createElement('button');\r\n // icon 兜底顺序固定为:业务覆盖 > DOM 默认 SVG > label 文本。\r\n const iconSvg = icons[item.icon] ?? defaultMenuIcons[item.icon];\r\n\r\n button.type = 'button';\r\n button.className = 'bridgerte__toolbar-button';\r\n button.disabled = state.disabled;\r\n button.dataset.active = String(state.active);\r\n button.dataset.bridgerteToolbarItemId = item.id;\r\n button.setAttribute('aria-label', item.label);\r\n button.setAttribute('aria-pressed', String(state.active));\r\n if (enableTooltip) button.dataset.tooltip = item.label;\r\n\r\n appendMenuIcon(button, iconSvg, item.label);\r\n\r\n groupElement.append(button);\r\n};\r\n\r\nconst renderToolbarGroupButton = (\r\n groupElement: HTMLElement,\r\n toolbarItem: Extract<ResolvedToolbarItem, { type: 'group' }>,\r\n commandStates: CommandState[],\r\n icons: RichTextToolbarIcons,\r\n enableTooltip: boolean\r\n) => {\r\n const button = document.createElement('button');\r\n const groupState = getGroupMenuState(toolbarItem.items, commandStates);\r\n const iconSvg = toolbarItem.icon\r\n ? icons[toolbarItem.icon] ?? defaultMenuIcons[toolbarItem.icon]\r\n : defaultMenuUiIcons.toolbarGroup;\r\n const indicator = document.createElement('span');\r\n\r\n button.type = 'button';\r\n button.className = 'bridgerte__toolbar-button bridgerte__toolbar-group-button';\r\n button.disabled = groupState.disabled;\r\n button.dataset.active = String(groupState.active);\r\n button.dataset.bridgerteToolbarGroupId = toolbarItem.key;\r\n button.dataset.open = 'false';\r\n button.setAttribute('aria-label', toolbarItem.title);\r\n button.setAttribute('aria-haspopup', 'menu');\r\n button.setAttribute('aria-expanded', 'false');\r\n button.setAttribute('aria-pressed', String(groupState.active));\r\n if (enableTooltip) button.dataset.tooltip = toolbarItem.title;\r\n\r\n appendMenuIcon(button, iconSvg, toolbarItem.title);\r\n\r\n indicator.className = 'bridgerte__toolbar-group-indicator';\r\n indicator.setAttribute('aria-hidden', 'true');\r\n indicator.innerHTML = defaultMenuUiIcons.chevronDown;\r\n button.append(indicator);\r\n groupElement.append(button);\r\n};\r\n\r\n/**\r\n * 渲染 toolbar 横向主入口。\n *\n * 字符串菜单直接渲染为按钮,`|` 渲染分割线。\n * 只有用户显式声明的 group 配置才生成收纳入口,避免 schema 隐式改变 DOM 结构。\n */\r\nexport const renderToolbar = (\r\n toolbarElement: HTMLElement,\r\n toolbarItems: ResolvedToolbarItem[],\r\n commandStates: CommandState[],\r\n icons: RichTextToolbarIcons,\r\n enableTooltip: boolean\r\n) => {\r\n toolbarElement.textContent = '';\r\n\r\n toolbarItems.forEach((toolbarItem) => {\n if (toolbarItem.type === 'separator') {\n const separatorElement = document.createElement('span');\n\n separatorElement.className = 'bridgerte__toolbar-separator';\n separatorElement.dataset.separatorId = toolbarItem.key;\r\n separatorElement.setAttribute('aria-hidden', 'true');\r\n toolbarElement.append(separatorElement);\r\n return;\n }\n\n if (toolbarItem.type === 'button') {\n renderToolbarButton(\n toolbarElement,\n toolbarItem.item,\n commandStates,\n icons,\n enableTooltip\n );\n return;\n }\n\n /*\n * 收纳菜单是用户在 toolbarConfig 里显式声明的结构;普通菜单不会自动生成 DOM 包裹,\n * 显示分组完全交给用户用 group 配置或 `|` 控制。\n */\n const groupElement = document.createElement('div');\n\n groupElement.className = 'bridgerte__toolbar-group';\n groupElement.dataset.group = toolbarItem.key;\n groupElement.setAttribute('aria-label', toolbarItem.title);\n toolbarElement.append(groupElement);\n\n renderToolbarGroupButton(\n groupElement,\n toolbarItem,\n commandStates,\n icons,\n enableTooltip\r\n );\r\n });\r\n};\r\n\r\n/**\r\n * 渲染 group button 打开的纵向收纳菜单。\r\n *\r\n * 浮层只展示 MenuItem 子项并复用现有 item id,点击执行仍由 index.ts 统一走 EditorAPI。\r\n */\r\nexport const renderToolbarGroupMenu = (\r\n menuElement: HTMLElement,\r\n groupMenuState: ToolbarGroupMenuState | null,\r\n commandStates: CommandState[],\r\n icons: RichTextToolbarIcons\r\n) => {\r\n menuElement.textContent = '';\r\n\r\n if (!groupMenuState) {\r\n menuElement.dataset.visible = 'false';\r\n return;\r\n }\r\n\r\n groupMenuState.items.forEach((item) => {\r\n const state = getMenuStateForItem(item, commandStates);\r\n const button = document.createElement('button');\r\n const iconSvg = icons[item.icon] ?? defaultMenuIcons[item.icon];\r\n const labelElement = document.createElement('span');\n\n button.type = 'button';\n button.className = 'bridgerte__menu-item bridgerte__toolbar-group-menu-item';\n button.disabled = state.disabled;\r\n button.dataset.active = String(state.active);\r\n button.dataset.bridgerteToolbarItemId = item.id;\r\n button.setAttribute('role', 'menuitem');\r\n button.setAttribute('aria-label', item.label);\r\n button.setAttribute('aria-pressed', String(state.active));\r\n\r\n appendMenuIcon(button, iconSvg, item.label);\r\n labelElement.className = 'bridgerte__toolbar-group-menu-label';\r\n labelElement.textContent = item.label;\r\n button.append(labelElement);\r\n menuElement.append(button);\r\n });\r\n\r\n menuElement.dataset.visible = 'true';\r\n menuElement.style.minWidth = `${groupMenuState.button.offsetWidth}px`;\r\n};\r\n\r\nexport const syncToolbarGroupButtonState = (\r\n container: HTMLElement,\r\n groupMenuState: ToolbarGroupMenuState | null\r\n) => {\r\n container.querySelectorAll<HTMLButtonElement>(toolbarGroupButtonSelector).forEach((button) => {\r\n const open = groupMenuState?.groupKey === button.dataset.bridgerteToolbarGroupId;\r\n\r\n button.dataset.open = String(open);\r\n button.setAttribute('aria-expanded', String(open));\r\n });\r\n};\r\n\r\nexport const findToolbarGroupMenuItem = (\r\n toolbarItems: ResolvedToolbarItem[],\r\n groupKey: string | undefined\r\n) => toolbarItems.find((item): item is Extract<ResolvedToolbarItem, { type: 'group' }> => (\r\n item.type === 'group' && item.key === groupKey\r\n));\r\n","import {\n toolbarExecutableButtonSelector,\n toolbarGroupButtonSelector\n} from './render';\n\nconst toolbarGroupTouchMoveTolerancePx = 8;\n\nconst isInsideNode = (container: HTMLElement, target: EventTarget | null) => (\n target instanceof Node && container.contains(target)\n);\n\nconst getClosestButton = (\n target: EventTarget | null,\n selector: string\n) => (\n target instanceof Element ? target.closest<HTMLButtonElement>(selector) : null\n);\n\n/**\n * 创建 toolbar 收纳菜单交互控制器。\n *\n * 这个控制器只处理 group menu overlay 的交互边界:收纳入口和菜单项在按下阶段不能抢走\n * 编辑器焦点,touch tap 需要在 touchend 主动完成,外部 pointerdown 才关闭菜单。\n */\nexport const createToolbarGroupMenuInteraction = (\n container: HTMLElement,\n groupMenuElement: HTMLElement,\n toggleGroupMenu: (button: HTMLButtonElement) => void,\n executeToolbarButton: (button: HTMLButtonElement) => void,\n closeGroupMenu: () => void\n) => {\n let pendingTouch: {\n button: HTMLButtonElement;\n kind: 'group' | 'menu-item';\n startX: number;\n startY: number;\n } | null = null;\n let suppressNextClickButton: HTMLButtonElement | null = null;\n\n const getInteractiveButtonFromTarget = (target: EventTarget | null) => (\n getClosestButton(target, toolbarGroupButtonSelector)\n ?? (\n isInsideNode(groupMenuElement, target)\n ? getClosestButton(target, toolbarExecutableButtonSelector)\n : null\n )\n );\n\n const handlePressStart = (event: PointerEvent | MouseEvent) => {\n const button = getInteractiveButtonFromTarget(event.target);\n\n if (!button) return;\n\n /*\n * 收纳入口和收纳菜单项都属于编辑器工具区。按下阶段阻止默认 focus 转移,\n * 避免 H5 contenteditable blur 后软键盘收起,宿主随即隐藏 toolbar 和 overlay。\n */\n event.preventDefault();\n };\n\n const handleTouchStart = (event: TouchEvent) => {\n const touch = event.touches[0];\n const groupButton = getClosestButton(event.target, toolbarGroupButtonSelector);\n const menuItemButton = isInsideNode(groupMenuElement, event.target)\n ? getClosestButton(event.target, toolbarExecutableButtonSelector)\n : null;\n const button = groupButton ?? menuItemButton;\n\n if (!button || !touch) return;\n\n event.preventDefault();\n pendingTouch = {\n button,\n kind: groupButton ? 'group' : 'menu-item',\n startX: touch.clientX,\n startY: touch.clientY\n };\n };\n\n const clearPendingTouch = () => {\n pendingTouch = null;\n };\n\n const handleTouchEnd = (event: TouchEvent) => {\n if (!pendingTouch) return;\n\n const touchState = pendingTouch;\n const touch = event.changedTouches[0];\n const endButton = getInteractiveButtonFromTarget(event.target);\n\n pendingTouch = null;\n if (!touch || endButton !== touchState.button) return;\n\n const deltaX = Math.abs(touch.clientX - touchState.startX);\n const deltaY = Math.abs(touch.clientY - touchState.startY);\n\n if (\n deltaX > toolbarGroupTouchMoveTolerancePx\n || deltaY > toolbarGroupTouchMoveTolerancePx\n ) return;\n\n event.preventDefault();\n event.stopPropagation();\n suppressNextClickButton = touchState.button;\n\n if (touchState.kind === 'group') {\n toggleGroupMenu(touchState.button);\n return;\n }\n\n executeToolbarButton(touchState.button);\n closeGroupMenu();\n };\n\n const handleDocumentPointerDown = (event: PointerEvent) => {\n if (\n isInsideNode(container, event.target)\n || isInsideNode(groupMenuElement, event.target)\n ) return;\n\n closeGroupMenu();\n };\n\n container.addEventListener('pointerdown', handlePressStart);\n container.addEventListener('mousedown', handlePressStart);\n container.addEventListener('touchstart', handleTouchStart, { passive: false });\n container.addEventListener('touchend', handleTouchEnd);\n container.addEventListener('touchcancel', clearPendingTouch);\n groupMenuElement.addEventListener('pointerdown', handlePressStart);\n groupMenuElement.addEventListener('mousedown', handlePressStart);\n groupMenuElement.addEventListener('touchstart', handleTouchStart, { passive: false });\n groupMenuElement.addEventListener('touchend', handleTouchEnd);\n groupMenuElement.addEventListener('touchcancel', clearPendingTouch);\n document.addEventListener('pointerdown', handleDocumentPointerDown, true);\n\n return {\n consumeSuppressedClick(button: HTMLButtonElement) {\n if (suppressNextClickButton !== button) return false;\n\n suppressNextClickButton = null;\n return true;\n },\n destroy() {\n container.removeEventListener('pointerdown', handlePressStart);\n container.removeEventListener('mousedown', handlePressStart);\n container.removeEventListener('touchstart', handleTouchStart);\n container.removeEventListener('touchend', handleTouchEnd);\n container.removeEventListener('touchcancel', clearPendingTouch);\n groupMenuElement.removeEventListener('pointerdown', handlePressStart);\n groupMenuElement.removeEventListener('mousedown', handlePressStart);\n groupMenuElement.removeEventListener('touchstart', handleTouchStart);\n groupMenuElement.removeEventListener('touchend', handleTouchEnd);\n groupMenuElement.removeEventListener('touchcancel', clearPendingTouch);\n document.removeEventListener('pointerdown', handleDocumentPointerDown, true);\n }\n };\n};\n","import type { CommandState } from '@bridgerte/core';\r\nimport {\r\n defaultMenuSchema,\r\n resolveToolbarMenu\r\n} from '@bridgerte/native-spec';\r\nimport { createFloatingLayer, type RichTextFloatingLayer } from '../floatingLayer';\r\nimport { bindTouchPressedState } from '../interactionState';\r\nimport {\r\n getPayloadPanelCurrentValues,\r\n resolveMenuSchemaForDom\r\n} from '../menuRuntime';\r\nimport {\n focusToolbarGroupMenuItem,\n handleToolbarGroupMenuKeyDown\n} from './groupMenuKeyboard';\nimport { createToolbarGroupMenuInteraction } from './groupMenuInteraction';\nimport {\r\n findToolbarGroupMenuItem,\r\n renderToolbar,\r\n renderToolbarGroupMenu,\r\n syncToolbarGroupButtonState,\r\n toolbarExecutableButtonSelector,\r\n toolbarGroupButtonSelector\r\n} from './render';\r\nimport type {\r\n RichTextToolbarAPI,\r\n ToolbarGroupMenuState,\r\n RichTextToolbarOptions\r\n} from './type';\r\n\r\nconst toolbarTooltipOffsetPx = 8;\r\nconst toolbarGroupMenuOffsetPx = 6;\r\n\r\nexport type * from './type';\r\n\r\nconst canUseHoverTooltip = () => (\r\n typeof window !== 'undefined'\r\n && window.matchMedia?.('(hover: hover) and (pointer: fine)').matches === true\r\n);\r\n\r\nconst getToolbarButtonFromTarget = (target: EventTarget | null) => {\r\n /*\r\n * 图标覆盖常用 SVG,H5 pointer 事件可能落在 svg/path 上。\r\n * 这里用 Element 而不是 HTMLElement,保证触屏兜底和 click 都能向上命中真实按钮。\r\n */\r\n const button = target instanceof Element\r\n ? target.closest<HTMLButtonElement>(toolbarExecutableButtonSelector)\r\n : null;\r\n\r\n return button instanceof HTMLButtonElement ? button : null;\r\n};\r\n\r\nconst getToolbarGroupButtonFromTarget = (target: EventTarget | null) => {\r\n const button = target instanceof Element\r\n ? target.closest<HTMLButtonElement>(toolbarGroupButtonSelector)\r\n : null;\r\n\r\n return button instanceof HTMLButtonElement ? button : null;\r\n};\r\n\r\n/**\r\n * 创建独立菜单实例。\r\n *\r\n * toolbar/tabbar 只订阅 EditorAPI 状态并派发命令,不接触编辑器内部 DOM 或 Lexical 实例。\r\n */\r\nexport function createRichTextToolbar(\r\n container: HTMLElement,\r\n options: RichTextToolbarOptions\r\n): RichTextToolbarAPI {\r\n const placement = options.placement ?? 'top';\r\n const menuSchema = resolveMenuSchemaForDom(options.menuSchema ?? defaultMenuSchema, {\r\n menuLabels: options.menuLabels,\r\n payloadPanelConfig: options.payloadPanelConfig\r\n });\r\n const toolbarItems = resolveToolbarMenu(options.toolbarConfig, menuSchema);\r\n /*\r\n * click 事件只需要能执行命令的菜单项。分割线和 group 容器是渲染结构,\r\n * 先在这里摊平,后续点击时就不用理解 toolbarConfig 的展示层级。\r\n */\r\n const executableMenuItems = toolbarItems.flatMap((toolbarItem) => (\r\n toolbarItem.type === 'button' ? [toolbarItem.item]\r\n : toolbarItem.type === 'group' ? toolbarItem.items\r\n : []\r\n ));\r\n const icons = options.icons ?? {};\r\n /*\r\n * tooltip 只属于 PC 精细指针体验。H5 上即使没有原生 title,部分浏览器也会把 hover/mouseover\r\n * 模拟成首触摸状态,导致第一次点像是只唤醒提示;所以触屏端不写 tooltip 数据也不挂监听。\r\n */\r\n const enableTooltip = canUseHoverTooltip();\r\n const tooltipElement = document.createElement('div');\r\n const groupMenuElement = document.createElement('div');\r\n const overlayHost = container.closest('.bridgerte') ?? container;\r\n let destroyed = false;\r\n let latestCommandStates = options.editor.getCommandStates();\r\n let groupMenuState: ToolbarGroupMenuState | null = null;\r\n let groupMenuFloatingLayer: RichTextFloatingLayer | null = null;\r\n const clearToolbarPressedState = bindTouchPressedState(container, {\r\n targetSelector: [\r\n 'button[data-bridgerte-toolbar-item-id]',\r\n 'button[data-bridgerte-toolbar-group-id]'\r\n ].join(',')\r\n });\r\n const clearGroupMenuPressedState = bindTouchPressedState(groupMenuElement, {\r\n targetSelector: '.bridgerte__toolbar-group-menu-item'\r\n });\r\n\r\n const mountToolbarOverlays = () => {\r\n // 独立 toolbar 可能不在 `.bridgerte` 内,渲染按钮会清空 container,需要把浮层重新挂回去。\r\n overlayHost.append(tooltipElement);\r\n overlayHost.append(groupMenuElement);\r\n };\r\n\r\n const closeGroupMenuFloatingLayer = () => {\r\n groupMenuFloatingLayer?.setOpen(false);\r\n groupMenuFloatingLayer?.destroy();\r\n groupMenuFloatingLayer = null;\r\n };\r\n\r\n const openGroupMenuFloatingLayer = () => {\r\n if (!groupMenuState) return;\r\n\r\n closeGroupMenuFloatingLayer();\r\n /*\r\n * group menu 和 hoverbar/mention/slash 一样属于轻浮层,必须走 floatingLayer。\r\n * 这里按 toolbar placement 给出首选方向,真正的翻转、键盘可视区和左右夹紧交给\r\n * createFloatingLayer 内部的 visualViewport + flip + shift + clamp 统一处理。\r\n */\r\n groupMenuFloatingLayer = createFloatingLayer(groupMenuState.button, groupMenuElement, {\r\n placement: placement === 'bottom' ? 'top-start' : 'bottom-start',\r\n offset: toolbarGroupMenuOffsetPx,\r\n strategy: 'fixed'\r\n });\r\n groupMenuFloatingLayer.setOpen(true);\r\n };\r\n\r\n const closeGroupMenu = () => {\r\n closeGroupMenuFloatingLayer();\r\n groupMenuState = null;\r\n renderToolbarGroupMenu(groupMenuElement, groupMenuState, latestCommandStates, icons);\r\n syncToolbarGroupButtonState(container, groupMenuState);\r\n };\r\n\r\n const renderStates = (states: CommandState[]) => {\r\n if (destroyed) return;\r\n\r\n latestCommandStates = states;\r\n if (groupMenuState) closeGroupMenuFloatingLayer();\r\n renderToolbar(container, toolbarItems, states, icons, enableTooltip);\r\n mountToolbarOverlays();\r\n if (groupMenuState) {\r\n const nextButton = Array.from(\r\n container.querySelectorAll<HTMLButtonElement>(toolbarGroupButtonSelector)\r\n ).find((button) => button.dataset.bridgerteToolbarGroupId === groupMenuState?.groupKey);\r\n const nextGroupItem = findToolbarGroupMenuItem(toolbarItems, groupMenuState.groupKey);\r\n\r\n groupMenuState = nextButton && nextGroupItem\r\n ? {\r\n groupKey: groupMenuState.groupKey,\r\n button: nextButton,\r\n items: nextGroupItem.items\r\n }\r\n : null;\r\n renderToolbarGroupMenu(groupMenuElement, groupMenuState, latestCommandStates, icons);\r\n syncToolbarGroupButtonState(container, groupMenuState);\r\n if (groupMenuState) openGroupMenuFloatingLayer();\r\n }\r\n };\r\n\r\n const update = () => {\r\n if (destroyed) return;\r\n\r\n renderStates(options.editor.getCommandStates());\r\n };\r\n\r\n const hideTooltip = () => {\r\n tooltipElement.dataset.visible = 'false';\r\n tooltipElement.textContent = '';\r\n };\r\n\r\n const showTooltip = (button: HTMLButtonElement) => {\r\n const tooltipText = button.dataset.tooltip;\r\n\r\n if (!enableTooltip || !tooltipText) return;\r\n\r\n const buttonRect = button.getBoundingClientRect();\r\n\r\n tooltipElement.textContent = tooltipText;\r\n tooltipElement.dataset.visible = 'true';\r\n tooltipElement.style.left = `${buttonRect.left + buttonRect.width / 2}px`;\r\n tooltipElement.style.top = `${buttonRect.top - toolbarTooltipOffsetPx}px`;\r\n };\r\n\r\n const handleTooltipTarget = (event: Event) => {\r\n const button = getToolbarButtonFromTarget(event.target);\r\n\r\n if (button) {\r\n showTooltip(button);\r\n }\r\n };\r\n\r\n const handleTooltipLeave = (event: MouseEvent) => {\r\n const relatedTarget = event.relatedTarget;\r\n const button = getToolbarButtonFromTarget(event.target);\r\n\r\n if (\r\n button\r\n && relatedTarget instanceof Node\r\n && button.contains(relatedTarget)\r\n ) return;\r\n\r\n hideTooltip();\r\n };\r\n\r\n const executeToolbarButton = (button: HTMLButtonElement) => {\r\n if (!(button instanceof HTMLButtonElement) || button.disabled) return;\r\n\r\n const menuItem = executableMenuItems.find((item) => (\r\n item.id === button.dataset.bridgerteToolbarItemId\r\n ));\r\n\r\n if (!menuItem) return;\r\n\r\n if (menuItem.payloadPanel) {\r\n const buttonRect = button.getBoundingClientRect();\r\n\r\n /*\r\n * 带 payloadPanel 的菜单不直接执行基础 command,而是发起参数请求。\r\n * DOM 内置面板和业务/RN/Flutter 自绘都走同一个 request,避免后续颜色、\r\n * 字体、链接等参数菜单各自发明一套协议。\r\n */\r\n options.editor.requestPayloadPanel({\r\n menuId: menuItem.id,\r\n command: menuItem.command,\r\n panel: menuItem.payloadPanel,\r\n currentValues: getPayloadPanelCurrentValues(menuItem, options.editor.getCommandStates()),\r\n anchorRect: {\r\n x: buttonRect.left,\r\n y: buttonRect.top,\r\n width: buttonRect.width,\r\n height: buttonRect.height\r\n }\r\n });\r\n return;\r\n }\r\n\r\n options.editor.executeCommand(menuItem.command);\r\n };\r\n\r\n const toggleGroupMenu = (button: HTMLButtonElement, shouldFocusMenu = false) => {\r\n const groupItem = findToolbarGroupMenuItem(\r\n toolbarItems,\r\n button.dataset.bridgerteToolbarGroupId\r\n );\r\n\r\n if (!groupItem || button.disabled) return;\r\n\r\n if (groupMenuState?.groupKey === groupItem.key) {\r\n if (shouldFocusMenu) {\r\n focusToolbarGroupMenuItem(groupMenuElement, 0);\r\n return;\r\n }\r\n closeGroupMenu();\r\n return;\r\n }\r\n\r\n groupMenuState = {\r\n groupKey: groupItem.key,\r\n button,\r\n items: groupItem.items\r\n };\r\n hideTooltip();\r\n renderToolbarGroupMenu(groupMenuElement, groupMenuState, latestCommandStates, icons);\r\n syncToolbarGroupButtonState(container, groupMenuState);\r\n openGroupMenuFloatingLayer();\r\n if (shouldFocusMenu) focusToolbarGroupMenuItem(groupMenuElement, 0);\r\n };\r\n\r\n const handleClick = (event: MouseEvent) => {\n const groupButton = getToolbarGroupButtonFromTarget(event.target);\n if (groupButton) {\n event.preventDefault();\n event.stopPropagation();\n if (groupMenuInteraction.consumeSuppressedClick(groupButton)) return;\n\n toggleGroupMenu(groupButton);\n return;\n }\n\n const button = getToolbarButtonFromTarget(event.target);\n if (!button) return;\r\n\r\n event.preventDefault();\n event.stopPropagation();\n if (groupMenuInteraction.consumeSuppressedClick(button)) return;\n\n executeToolbarButton(button);\n closeGroupMenu();\n };\n\r\n const handleKeyDown = (event: KeyboardEvent) => {\r\n if (groupMenuState && groupMenuElement.contains(document.activeElement)) {\r\n handleToolbarGroupMenuKeyDown(\r\n event,\r\n groupMenuElement,\r\n groupMenuState.button,\r\n executeToolbarButton,\r\n closeGroupMenu\r\n );\r\n return;\r\n }\r\n\r\n const groupButton = getToolbarGroupButtonFromTarget(event.target);\r\n if (\r\n groupButton\r\n && (event.key === 'Enter' || event.key === ' ' || event.key === 'ArrowDown')\r\n ) {\r\n /*\r\n * 指针点击收纳菜单不能抢编辑器焦点;键盘从入口按钮打开时则进入菜单项,\r\n * 保留 PC 可访问性和方向键 roving focus。\r\n */\r\n event.preventDefault();\r\n event.stopPropagation();\r\n toggleGroupMenu(groupButton, true);\r\n return;\r\n }\r\n\r\n if (event.key !== 'Escape') return;\r\n\r\n closeGroupMenu();\r\n };\r\n\r\n container.classList.add('bridgerte__toolbar');\r\n tooltipElement.className = 'bridgerte__toolbar-tooltip';\r\n tooltipElement.dataset.visible = 'false';\r\n groupMenuElement.className = 'bridgerte__floating-menu bridgerte__toolbar-group-menu';\r\n groupMenuElement.dataset.visible = 'false';\r\n groupMenuElement.setAttribute('role', 'menu');\r\n container.dataset.placement = placement;\r\n container.setAttribute('role', 'toolbar');\r\n container.setAttribute(\r\n 'aria-label',\r\n placement === 'bottom' ? 'BridgeRTE tabbar' : 'BridgeRTE toolbar'\r\n );\r\n container.addEventListener('click', handleClick);\n groupMenuElement.addEventListener('click', handleClick);\n document.addEventListener('keydown', handleKeyDown);\n if (enableTooltip) {\r\n container.addEventListener('mouseover', handleTooltipTarget);\r\n container.addEventListener('mouseout', handleTooltipLeave);\r\n }\r\n container.addEventListener('focusout', hideTooltip);\r\n // 浮层挂在最近的编辑器根容器下,既能继承变量,也不会操作编辑内容 DOM。\n mountToolbarOverlays();\n const groupMenuInteraction = createToolbarGroupMenuInteraction(\n container,\n groupMenuElement,\n toggleGroupMenu,\n executeToolbarButton,\n closeGroupMenu\n );\n\r\n // 独立 toolbar 只订阅 public API 状态,不依赖 DOM 编辑器内部实现。\r\n const unsubscribe = options.editor.subscribeCommandStateChange(renderStates);\r\n\r\n return {\r\n update,\r\n destroy() {\r\n if (destroyed) return;\r\n\r\n destroyed = true;\r\n unsubscribe();\r\n closeGroupMenu();\n container.removeEventListener('click', handleClick);\n groupMenuElement.removeEventListener('click', handleClick);\n document.removeEventListener('keydown', handleKeyDown);\n groupMenuInteraction.destroy();\n if (enableTooltip) {\r\n container.removeEventListener('mouseover', handleTooltipTarget);\r\n container.removeEventListener('mouseout', handleTooltipLeave);\r\n }\r\n clearToolbarPressedState();\r\n clearGroupMenuPressedState();\r\n container.removeEventListener('focusout', hideTooltip);\r\n tooltipElement.remove();\r\n groupMenuElement.remove();\r\n container.classList.remove('bridgerte__toolbar');\r\n delete container.dataset.placement;\r\n container.textContent = '';\r\n container.removeAttribute('role');\r\n container.removeAttribute('aria-label');\r\n }\r\n };\r\n}\r\n","const invisibleTextPattern = /[\\u200B-\\u200D\\uFEFF]/g;\n\n/*\n * HTML 片段里这些元素没有 textContent 也应算有效内容。\n * 纯排版容器不放进来,避免 `<p><br></p>`、空列表项这类编辑器占位被误判为非空。\n */\nconst meaningfulHtmlContentSelector = [\n 'img',\n 'video',\n 'audio',\n 'iframe',\n 'table',\n 'pre',\n 'code',\n 'hr',\n 'figure',\n 'canvas',\n 'svg',\n 'math',\n '[data-type=\"mention\"]'\n].join(',');\n\nconst normalizeHtmlText = (text: string | null | undefined): string => (\n text?.replace(invisibleTextPattern, '').trim() ?? ''\n);\n\n/**\n * 判断一段 HTML 片段是否包含可保存的富文本内容。\n *\n * 这个工具会使用浏览器 template 解析 HTML,适合只拿到 HTML 字符串的业务表单;如果已经持有\n * `EditorContent`,优先使用 core 的 `isEditorContentEmpty()`,避免额外 DOM parse。\n */\nexport const hasMeaningfulHtmlContent = (html: string): boolean => {\n if (!normalizeHtmlText(html)) return false;\n\n const template = document.createElement('template');\n\n template.innerHTML = html;\n\n return Boolean(\n normalizeHtmlText(template.content.textContent)\n || template.content.querySelector(meaningfulHtmlContentSelector)\n );\n};\n"],"names":["defaultMenuUiIcons","getEnabledGroupMenuButtons","menuElement","button","focusToolbarGroupMenuItem","index","nextButton","handleToolbarGroupMenuKeyDown","event","returnButton","executeToolbarButton","closeGroupMenu","buttons","activeIndex","focusByOffset","offset","nextIndex","_a","_b","toolbarGroupButtonSelector","toolbarButtonSelector","toolbarExecutableButtonSelector","getGroupMenuState","items","commandStates","itemStates","item","getMenuStateForItem","state","renderToolbarButton","groupElement","icons","enableTooltip","iconSvg","defaultMenuIcons","appendMenuIcon","renderToolbarGroupButton","toolbarItem","groupState","indicator","renderToolbar","toolbarElement","toolbarItems","separatorElement","renderToolbarGroupMenu","groupMenuState","labelElement","syncToolbarGroupButtonState","container","open","findToolbarGroupMenuItem","groupKey","toolbarGroupTouchMoveTolerancePx","isInsideNode","target","getClosestButton","selector","createToolbarGroupMenuInteraction","groupMenuElement","toggleGroupMenu","pendingTouch","suppressNextClickButton","getInteractiveButtonFromTarget","handlePressStart","handleTouchStart","touch","groupButton","menuItemButton","clearPendingTouch","handleTouchEnd","touchState","endButton","deltaX","deltaY","handleDocumentPointerDown","toolbarTooltipOffsetPx","toolbarGroupMenuOffsetPx","canUseHoverTooltip","getToolbarButtonFromTarget","getToolbarGroupButtonFromTarget","createRichTextToolbar","options","placement","menuSchema","resolveMenuSchemaForDom","defaultMenuSchema","resolveToolbarMenu","executableMenuItems","tooltipElement","overlayHost","destroyed","latestCommandStates","groupMenuFloatingLayer","clearToolbarPressedState","bindTouchPressedState","clearGroupMenuPressedState","mountToolbarOverlays","closeGroupMenuFloatingLayer","openGroupMenuFloatingLayer","createFloatingLayer","renderStates","states","nextGroupItem","update","hideTooltip","showTooltip","tooltipText","buttonRect","handleTooltipTarget","handleTooltipLeave","relatedTarget","menuItem","getPayloadPanelCurrentValues","shouldFocusMenu","groupItem","handleClick","groupMenuInteraction","handleKeyDown","unsubscribe","invisibleTextPattern","meaningfulHtmlContentSelector","normalizeHtmlText","text","hasMeaningfulHtmlContent","html","template"],"mappings":"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"}
@@ -0,0 +1,36 @@
1
+ "use strict";const q=require("./native-spec.cjs"),f=require("./index-BBzHVrSi.cjs"),O={toolbarGroup:`
2
+ <svg
3
+ aria-hidden="true"
4
+ class="lucide lucide-ellipsis"
5
+ xmlns="http://www.w3.org/2000/svg"
6
+ width="18"
7
+ height="18"
8
+ viewBox="0 0 24 24"
9
+ fill="none"
10
+ stroke="currentColor"
11
+ stroke-width="2"
12
+ stroke-linecap="round"
13
+ stroke-linejoin="round"
14
+ >
15
+ <circle cx="12" cy="12" r="1" />
16
+ <circle cx="19" cy="12" r="1" />
17
+ <circle cx="5" cy="12" r="1" />
18
+ </svg>
19
+ `,chevronDown:`
20
+ <svg
21
+ aria-hidden="true"
22
+ class="lucide lucide-chevron-down"
23
+ xmlns="http://www.w3.org/2000/svg"
24
+ width="12"
25
+ height="12"
26
+ viewBox="0 0 24 24"
27
+ fill="none"
28
+ stroke="currentColor"
29
+ stroke-width="2"
30
+ stroke-linecap="round"
31
+ stroke-linejoin="round"
32
+ >
33
+ <path d="m6 9 6 6 6-6" />
34
+ </svg>
35
+ `},V=t=>Array.from(t.querySelectorAll(".bridgerte__toolbar-group-menu-item")).filter(e=>!e.disabled),j=(t,e)=>{const l=V(t)[e];l&&l.focus()},Z=(t,e,i,l,u)=>{var s,d;const n=V(e),r=document.activeElement instanceof HTMLButtonElement?n.indexOf(document.activeElement):-1,m=v=>{var o;if(n.length===0)return;const h=((r>=0?r:0)+v+n.length)%n.length;(o=n[h])==null||o.focus()};switch(t.key){case"ArrowDown":t.preventDefault(),m(1);break;case"ArrowUp":t.preventDefault(),m(-1);break;case"Home":t.preventDefault(),(s=n[0])==null||s.focus();break;case"End":t.preventDefault(),(d=n.at(-1))==null||d.focus();break;case"Enter":case" ":document.activeElement instanceof HTMLButtonElement&&(t.preventDefault(),l(document.activeElement),u(),i.focus());break;case"Escape":t.preventDefault(),i.focus(),u();break}},x="button[data-bridgerte-toolbar-group-id]",tt="button[data-bridgerte-toolbar-item-id]",M=[tt,".bridgerte__toolbar-group-menu-item"].join(","),et=(t,e)=>{const i=t.map(l=>f.getMenuStateForItem(l,e));return{active:i.some(l=>l.active),disabled:i.length>0&&i.every(l=>l.disabled)}},ot=(t,e,i,l,u)=>{const n=f.getMenuStateForItem(e,i),r=document.createElement("button"),m=l[e.icon]??f.defaultMenuIcons[e.icon];r.type="button",r.className="bridgerte__toolbar-button",r.disabled=n.disabled,r.dataset.active=String(n.active),r.dataset.bridgerteToolbarItemId=e.id,r.setAttribute("aria-label",e.label),r.setAttribute("aria-pressed",String(n.active)),u&&(r.dataset.tooltip=e.label),f.appendMenuIcon(r,m,e.label),t.append(r)},nt=(t,e,i,l,u)=>{const n=document.createElement("button"),r=et(e.items,i),m=e.icon?l[e.icon]??f.defaultMenuIcons[e.icon]:O.toolbarGroup,s=document.createElement("span");n.type="button",n.className="bridgerte__toolbar-button bridgerte__toolbar-group-button",n.disabled=r.disabled,n.dataset.active=String(r.active),n.dataset.bridgerteToolbarGroupId=e.key,n.dataset.open="false",n.setAttribute("aria-label",e.title),n.setAttribute("aria-haspopup","menu"),n.setAttribute("aria-expanded","false"),n.setAttribute("aria-pressed",String(r.active)),u&&(n.dataset.tooltip=e.title),f.appendMenuIcon(n,m,e.title),s.className="bridgerte__toolbar-group-indicator",s.setAttribute("aria-hidden","true"),s.innerHTML=O.chevronDown,n.append(s),t.append(n)},rt=(t,e,i,l,u)=>{t.textContent="",e.forEach(n=>{if(n.type==="separator"){const m=document.createElement("span");m.className="bridgerte__toolbar-separator",m.dataset.separatorId=n.key,m.setAttribute("aria-hidden","true"),t.append(m);return}if(n.type==="button"){ot(t,n.item,i,l,u);return}const r=document.createElement("div");r.className="bridgerte__toolbar-group",r.dataset.group=n.key,r.setAttribute("aria-label",n.title),t.append(r),nt(r,n,i,l,u)})},P=(t,e,i,l)=>{if(t.textContent="",!e){t.dataset.visible="false";return}e.items.forEach(u=>{const n=f.getMenuStateForItem(u,i),r=document.createElement("button"),m=l[u.icon]??f.defaultMenuIcons[u.icon],s=document.createElement("span");r.type="button",r.className="bridgerte__menu-item bridgerte__toolbar-group-menu-item",r.disabled=n.disabled,r.dataset.active=String(n.active),r.dataset.bridgerteToolbarItemId=u.id,r.setAttribute("role","menuitem"),r.setAttribute("aria-label",u.label),r.setAttribute("aria-pressed",String(n.active)),f.appendMenuIcon(r,m,u.label),s.className="bridgerte__toolbar-group-menu-label",s.textContent=u.label,r.append(s),t.append(r)}),t.dataset.visible="true",t.style.minWidth=`${e.button.offsetWidth}px`},D=(t,e)=>{t.querySelectorAll(x).forEach(i=>{const l=(e==null?void 0:e.groupKey)===i.dataset.bridgerteToolbarGroupId;i.dataset.open=String(l),i.setAttribute("aria-expanded",String(l))})},X=(t,e)=>t.find(i=>i.type==="group"&&i.key===e),Y=8,_=(t,e)=>e instanceof Node&&t.contains(e),B=(t,e)=>t instanceof Element?t.closest(e):null,at=(t,e,i,l,u)=>{let n=null,r=null;const m=o=>B(o,x)??(_(e,o)?B(o,M):null),s=o=>{m(o.target)&&o.preventDefault()},d=o=>{const p=o.touches[0],y=B(o.target,x),w=_(e,o.target)?B(o.target,M):null,T=y??w;!T||!p||(o.preventDefault(),n={button:T,kind:y?"group":"menu-item",startX:p.clientX,startY:p.clientY})},v=()=>{n=null},g=o=>{if(!n)return;const p=n,y=o.changedTouches[0],w=m(o.target);if(n=null,!y||w!==p.button)return;const T=Math.abs(y.clientX-p.startX),L=Math.abs(y.clientY-p.startY);if(!(T>Y||L>Y)){if(o.preventDefault(),o.stopPropagation(),r=p.button,p.kind==="group"){i(p.button);return}l(p.button),u()}},h=o=>{_(t,o.target)||_(e,o.target)||u()};return t.addEventListener("pointerdown",s),t.addEventListener("mousedown",s),t.addEventListener("touchstart",d,{passive:!1}),t.addEventListener("touchend",g),t.addEventListener("touchcancel",v),e.addEventListener("pointerdown",s),e.addEventListener("mousedown",s),e.addEventListener("touchstart",d,{passive:!1}),e.addEventListener("touchend",g),e.addEventListener("touchcancel",v),document.addEventListener("pointerdown",h,!0),{consumeSuppressedClick(o){return r!==o?!1:(r=null,!0)},destroy(){t.removeEventListener("pointerdown",s),t.removeEventListener("mousedown",s),t.removeEventListener("touchstart",d),t.removeEventListener("touchend",g),t.removeEventListener("touchcancel",v),e.removeEventListener("pointerdown",s),e.removeEventListener("mousedown",s),e.removeEventListener("touchstart",d),e.removeEventListener("touchend",g),e.removeEventListener("touchcancel",v),document.removeEventListener("pointerdown",h,!0)}}},st=8,it=6,ct=()=>{var t;return typeof window<"u"&&((t=window.matchMedia)==null?void 0:t.call(window,"(hover: hover) and (pointer: fine)").matches)===!0},I=t=>{const e=t instanceof Element?t.closest(M):null;return e instanceof HTMLButtonElement?e:null},U=t=>{const e=t instanceof Element?t.closest(x):null;return e instanceof HTMLButtonElement?e:null};function lt(t,e){const i=e.placement??"top",l=f.resolveMenuSchemaForDom(e.menuSchema??q.defaultMenuSchema,{menuLabels:e.menuLabels,payloadPanelConfig:e.payloadPanelConfig}),u=q.resolveToolbarMenu(e.toolbarConfig,l),n=u.flatMap(a=>a.type==="button"?[a.item]:a.type==="group"?a.items:[]),r=e.icons??{},m=ct(),s=document.createElement("div"),d=document.createElement("div"),v=t.closest(".bridgerte")??t;let g=!1,h=e.editor.getCommandStates(),o=null,p=null;const y=f.bindTouchPressedState(t,{targetSelector:["button[data-bridgerte-toolbar-item-id]","button[data-bridgerte-toolbar-group-id]"].join(",")}),w=f.bindTouchPressedState(d,{targetSelector:".bridgerte__toolbar-group-menu-item"}),T=()=>{v.append(s),v.append(d)},L=()=>{p==null||p.setOpen(!1),p==null||p.destroy(),p=null},H=()=>{o&&(L(),p=f.createFloatingLayer(o.button,d,{placement:i==="bottom"?"top-start":"bottom-start",offset:it,strategy:"fixed"}),p.setOpen(!0))},E=()=>{L(),o=null,P(d,o,h,r),D(t,o)},N=a=>{if(!g&&(h=a,o&&L(),rt(t,u,a,r,m),T(),o)){const c=Array.from(t.querySelectorAll(x)).find(Q=>Q.dataset.bridgerteToolbarGroupId===(o==null?void 0:o.groupKey)),b=X(u,o.groupKey);o=c&&b?{groupKey:o.groupKey,button:c,items:b.items}:null,P(d,o,h,r),D(t,o),o&&H()}},W=()=>{g||N(e.editor.getCommandStates())},S=()=>{s.dataset.visible="false",s.textContent=""},z=a=>{const c=a.dataset.tooltip;if(!m||!c)return;const b=a.getBoundingClientRect();s.textContent=c,s.dataset.visible="true",s.style.left=`${b.left+b.width/2}px`,s.style.top=`${b.top-st}px`},F=a=>{const c=I(a.target);c&&z(c)},K=a=>{const c=a.relatedTarget,b=I(a.target);b&&c instanceof Node&&b.contains(c)||S()},C=a=>{if(!(a instanceof HTMLButtonElement)||a.disabled)return;const c=n.find(b=>b.id===a.dataset.bridgerteToolbarItemId);if(c){if(c.payloadPanel){const b=a.getBoundingClientRect();e.editor.requestPayloadPanel({menuId:c.id,command:c.command,panel:c.payloadPanel,currentValues:f.getPayloadPanelCurrentValues(c,e.editor.getCommandStates()),anchorRect:{x:b.left,y:b.top,width:b.width,height:b.height}});return}e.editor.executeCommand(c.command)}},A=(a,c=!1)=>{const b=X(u,a.dataset.bridgerteToolbarGroupId);if(!(!b||a.disabled)){if((o==null?void 0:o.groupKey)===b.key){if(c){j(d,0);return}E();return}o={groupKey:b.key,button:a,items:b.items},S(),P(d,o,h,r),D(t,o),H(),c&&j(d,0)}},k=a=>{const c=U(a.target);if(c){if(a.preventDefault(),a.stopPropagation(),G.consumeSuppressedClick(c))return;A(c);return}const b=I(a.target);b&&(a.preventDefault(),a.stopPropagation(),!G.consumeSuppressedClick(b)&&(C(b),E()))},R=a=>{if(o&&d.contains(document.activeElement)){Z(a,d,o.button,C,E);return}const c=U(a.target);if(c&&(a.key==="Enter"||a.key===" "||a.key==="ArrowDown")){a.preventDefault(),a.stopPropagation(),A(c,!0);return}a.key==="Escape"&&E()};t.classList.add("bridgerte__toolbar"),s.className="bridgerte__toolbar-tooltip",s.dataset.visible="false",d.className="bridgerte__floating-menu bridgerte__toolbar-group-menu",d.dataset.visible="false",d.setAttribute("role","menu"),t.dataset.placement=i,t.setAttribute("role","toolbar"),t.setAttribute("aria-label",i==="bottom"?"BridgeRTE tabbar":"BridgeRTE toolbar"),t.addEventListener("click",k),d.addEventListener("click",k),document.addEventListener("keydown",R),m&&(t.addEventListener("mouseover",F),t.addEventListener("mouseout",K)),t.addEventListener("focusout",S),T();const G=at(t,d,A,C,E),J=e.editor.subscribeCommandStateChange(N);return{update:W,destroy(){g||(g=!0,J(),E(),t.removeEventListener("click",k),d.removeEventListener("click",k),document.removeEventListener("keydown",R),G.destroy(),m&&(t.removeEventListener("mouseover",F),t.removeEventListener("mouseout",K)),y(),w(),t.removeEventListener("focusout",S),s.remove(),d.remove(),t.classList.remove("bridgerte__toolbar"),delete t.dataset.placement,t.textContent="",t.removeAttribute("role"),t.removeAttribute("aria-label"))}}}const ut=/[\u200B-\u200D\uFEFF]/g,dt=["img","video","audio","iframe","table","pre","code","hr","figure","canvas","svg","math",'[data-type="mention"]'].join(","),$=t=>(t==null?void 0:t.replace(ut,"").trim())??"",bt=t=>{if(!$(t))return!1;const e=document.createElement("template");return e.innerHTML=t,!!($(e.content.textContent)||e.content.querySelector(dt))};exports.createRichTextToolbar=lt;exports.hasMeaningfulHtmlContent=bt;
36
+ //# sourceMappingURL=index-DZnoOnKI.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index-DZnoOnKI.cjs","sources":["../../dom/src/menuIcon/uiIcons.ts","../../dom/src/richTextToolbar/groupMenuKeyboard.ts","../../dom/src/richTextToolbar/render.ts","../../dom/src/richTextToolbar/groupMenuInteraction.ts","../../dom/src/richTextToolbar/index.ts","../../dom/src/htmlContent/index.ts"],"sourcesContent":["/**\r\n * DOM UI chrome icon 表。\r\n *\r\n * 这里放 toolbar 收纳入口、下拉箭头这类控件自身图标。它们不属于 `MenuItem.icon`\r\n * 跨端 schema,也不应该进入 `defaultMenuIcons` 的 schema 对齐检查。\r\n */\r\nexport const defaultMenuUiIcons = {\r\n toolbarGroup: `\r\n <svg\r\n aria-hidden=\"true\"\r\n class=\"lucide lucide-ellipsis\"\r\n xmlns=\"http://www.w3.org/2000/svg\"\r\n width=\"18\"\r\n height=\"18\"\r\n viewBox=\"0 0 24 24\"\r\n fill=\"none\"\r\n stroke=\"currentColor\"\r\n stroke-width=\"2\"\r\n stroke-linecap=\"round\"\r\n stroke-linejoin=\"round\"\r\n >\r\n <circle cx=\"12\" cy=\"12\" r=\"1\" />\r\n <circle cx=\"19\" cy=\"12\" r=\"1\" />\r\n <circle cx=\"5\" cy=\"12\" r=\"1\" />\r\n </svg>\r\n `,\r\n chevronDown: `\r\n <svg\r\n aria-hidden=\"true\"\r\n class=\"lucide lucide-chevron-down\"\r\n xmlns=\"http://www.w3.org/2000/svg\"\r\n width=\"12\"\r\n height=\"12\"\r\n viewBox=\"0 0 24 24\"\r\n fill=\"none\"\r\n stroke=\"currentColor\"\r\n stroke-width=\"2\"\r\n stroke-linecap=\"round\"\r\n stroke-linejoin=\"round\"\r\n >\r\n <path d=\"m6 9 6 6 6-6\" />\r\n </svg>\r\n `\r\n} as const;\r\n","export const getEnabledGroupMenuButtons = (menuElement: HTMLElement) => (\r\n Array.from(\r\n menuElement.querySelectorAll<HTMLButtonElement>('.bridgerte__toolbar-group-menu-item')\r\n ).filter((button) => !button.disabled)\r\n);\r\n\r\nexport const focusToolbarGroupMenuItem = (menuElement: HTMLElement, index: number) => {\r\n const buttons = getEnabledGroupMenuButtons(menuElement);\r\n const nextButton = buttons[index];\r\n\r\n if (nextButton) nextButton.focus();\r\n};\r\n\r\nexport const handleToolbarGroupMenuKeyDown = (\n event: KeyboardEvent,\r\n menuElement: HTMLElement,\r\n returnButton: HTMLButtonElement,\r\n executeToolbarButton: (button: HTMLButtonElement) => void,\r\n closeGroupMenu: () => void\r\n) => {\r\n const buttons = getEnabledGroupMenuButtons(menuElement);\r\n const activeIndex = document.activeElement instanceof HTMLButtonElement\r\n ? buttons.indexOf(document.activeElement)\r\n : -1;\r\n const focusByOffset = (offset: number) => {\r\n if (buttons.length === 0) return;\r\n\r\n const baseIndex = activeIndex >= 0 ? activeIndex : 0;\r\n const nextIndex = (baseIndex + offset + buttons.length) % buttons.length;\r\n\r\n buttons[nextIndex]?.focus();\r\n };\r\n\r\n /*\r\n * ARIA menu 语义要求支持 roving focus。这里不改 toolbar 的横向键盘模型,\r\n * 只在已打开的纵向收纳菜单内处理上下方向、首尾跳转和执行/关闭。\r\n */\r\n switch (event.key) {\r\n case 'ArrowDown':\r\n event.preventDefault();\r\n focusByOffset(1);\r\n break;\r\n case 'ArrowUp':\r\n event.preventDefault();\r\n focusByOffset(-1);\r\n break;\r\n case 'Home':\r\n event.preventDefault();\r\n buttons[0]?.focus();\r\n break;\r\n case 'End':\r\n event.preventDefault();\r\n buttons.at(-1)?.focus();\r\n break;\r\n case 'Enter':\r\n case ' ':\r\n if (document.activeElement instanceof HTMLButtonElement) {\r\n event.preventDefault();\r\n executeToolbarButton(document.activeElement);\r\n closeGroupMenu();\r\n returnButton.focus();\r\n }\r\n break;\r\n case 'Escape':\r\n event.preventDefault();\r\n returnButton.focus();\r\n closeGroupMenu();\r\n break;\r\n default:\r\n break;\r\n }\r\n};\r\n","import type { CommandState } from '@bridgerte/core';\r\nimport type { MenuItem, ResolvedToolbarItem } from '@bridgerte/native-spec';\r\nimport {\r\n appendMenuIcon,\r\n defaultMenuIcons,\r\n defaultMenuUiIcons\r\n} from './icons';\r\nimport { getMenuStateForItem } from '../menuRuntime';\r\nimport type {\r\n RichTextToolbarIcons,\r\n ToolbarGroupMenuState\r\n} from './type';\r\n\r\nexport const toolbarGroupButtonSelector = 'button[data-bridgerte-toolbar-group-id]';\r\nexport const toolbarButtonSelector = 'button[data-bridgerte-toolbar-item-id]';\r\n// 可执行按钮同时覆盖 toolbar 主按钮和 group menu 子项,避免浮层子菜单绕过统一命令入口。\r\nexport const toolbarExecutableButtonSelector = [\n toolbarButtonSelector,\n '.bridgerte__toolbar-group-menu-item'\n].join(',');\n\r\nconst getGroupMenuState = (items: MenuItem[], commandStates: CommandState[]) => {\r\n const itemStates = items.map((item) => getMenuStateForItem(item, commandStates));\r\n\r\n return {\r\n active: itemStates.some((state) => state.active),\r\n disabled: itemStates.length > 0 && itemStates.every((state) => state.disabled)\r\n };\r\n};\r\n\r\nconst renderToolbarButton = (\r\n groupElement: HTMLElement,\r\n item: MenuItem,\r\n commandStates: CommandState[],\r\n icons: RichTextToolbarIcons,\r\n enableTooltip: boolean\r\n) => {\r\n const state = getMenuStateForItem(item, commandStates);\r\n const button = document.createElement('button');\r\n // icon 兜底顺序固定为:业务覆盖 > DOM 默认 SVG > label 文本。\r\n const iconSvg = icons[item.icon] ?? defaultMenuIcons[item.icon];\r\n\r\n button.type = 'button';\r\n button.className = 'bridgerte__toolbar-button';\r\n button.disabled = state.disabled;\r\n button.dataset.active = String(state.active);\r\n button.dataset.bridgerteToolbarItemId = item.id;\r\n button.setAttribute('aria-label', item.label);\r\n button.setAttribute('aria-pressed', String(state.active));\r\n if (enableTooltip) button.dataset.tooltip = item.label;\r\n\r\n appendMenuIcon(button, iconSvg, item.label);\r\n\r\n groupElement.append(button);\r\n};\r\n\r\nconst renderToolbarGroupButton = (\r\n groupElement: HTMLElement,\r\n toolbarItem: Extract<ResolvedToolbarItem, { type: 'group' }>,\r\n commandStates: CommandState[],\r\n icons: RichTextToolbarIcons,\r\n enableTooltip: boolean\r\n) => {\r\n const button = document.createElement('button');\r\n const groupState = getGroupMenuState(toolbarItem.items, commandStates);\r\n const iconSvg = toolbarItem.icon\r\n ? icons[toolbarItem.icon] ?? defaultMenuIcons[toolbarItem.icon]\r\n : defaultMenuUiIcons.toolbarGroup;\r\n const indicator = document.createElement('span');\r\n\r\n button.type = 'button';\r\n button.className = 'bridgerte__toolbar-button bridgerte__toolbar-group-button';\r\n button.disabled = groupState.disabled;\r\n button.dataset.active = String(groupState.active);\r\n button.dataset.bridgerteToolbarGroupId = toolbarItem.key;\r\n button.dataset.open = 'false';\r\n button.setAttribute('aria-label', toolbarItem.title);\r\n button.setAttribute('aria-haspopup', 'menu');\r\n button.setAttribute('aria-expanded', 'false');\r\n button.setAttribute('aria-pressed', String(groupState.active));\r\n if (enableTooltip) button.dataset.tooltip = toolbarItem.title;\r\n\r\n appendMenuIcon(button, iconSvg, toolbarItem.title);\r\n\r\n indicator.className = 'bridgerte__toolbar-group-indicator';\r\n indicator.setAttribute('aria-hidden', 'true');\r\n indicator.innerHTML = defaultMenuUiIcons.chevronDown;\r\n button.append(indicator);\r\n groupElement.append(button);\r\n};\r\n\r\n/**\r\n * 渲染 toolbar 横向主入口。\n *\n * 字符串菜单直接渲染为按钮,`|` 渲染分割线。\n * 只有用户显式声明的 group 配置才生成收纳入口,避免 schema 隐式改变 DOM 结构。\n */\r\nexport const renderToolbar = (\r\n toolbarElement: HTMLElement,\r\n toolbarItems: ResolvedToolbarItem[],\r\n commandStates: CommandState[],\r\n icons: RichTextToolbarIcons,\r\n enableTooltip: boolean\r\n) => {\r\n toolbarElement.textContent = '';\r\n\r\n toolbarItems.forEach((toolbarItem) => {\n if (toolbarItem.type === 'separator') {\n const separatorElement = document.createElement('span');\n\n separatorElement.className = 'bridgerte__toolbar-separator';\n separatorElement.dataset.separatorId = toolbarItem.key;\r\n separatorElement.setAttribute('aria-hidden', 'true');\r\n toolbarElement.append(separatorElement);\r\n return;\n }\n\n if (toolbarItem.type === 'button') {\n renderToolbarButton(\n toolbarElement,\n toolbarItem.item,\n commandStates,\n icons,\n enableTooltip\n );\n return;\n }\n\n /*\n * 收纳菜单是用户在 toolbarConfig 里显式声明的结构;普通菜单不会自动生成 DOM 包裹,\n * 显示分组完全交给用户用 group 配置或 `|` 控制。\n */\n const groupElement = document.createElement('div');\n\n groupElement.className = 'bridgerte__toolbar-group';\n groupElement.dataset.group = toolbarItem.key;\n groupElement.setAttribute('aria-label', toolbarItem.title);\n toolbarElement.append(groupElement);\n\n renderToolbarGroupButton(\n groupElement,\n toolbarItem,\n commandStates,\n icons,\n enableTooltip\r\n );\r\n });\r\n};\r\n\r\n/**\r\n * 渲染 group button 打开的纵向收纳菜单。\r\n *\r\n * 浮层只展示 MenuItem 子项并复用现有 item id,点击执行仍由 index.ts 统一走 EditorAPI。\r\n */\r\nexport const renderToolbarGroupMenu = (\r\n menuElement: HTMLElement,\r\n groupMenuState: ToolbarGroupMenuState | null,\r\n commandStates: CommandState[],\r\n icons: RichTextToolbarIcons\r\n) => {\r\n menuElement.textContent = '';\r\n\r\n if (!groupMenuState) {\r\n menuElement.dataset.visible = 'false';\r\n return;\r\n }\r\n\r\n groupMenuState.items.forEach((item) => {\r\n const state = getMenuStateForItem(item, commandStates);\r\n const button = document.createElement('button');\r\n const iconSvg = icons[item.icon] ?? defaultMenuIcons[item.icon];\r\n const labelElement = document.createElement('span');\n\n button.type = 'button';\n button.className = 'bridgerte__menu-item bridgerte__toolbar-group-menu-item';\n button.disabled = state.disabled;\r\n button.dataset.active = String(state.active);\r\n button.dataset.bridgerteToolbarItemId = item.id;\r\n button.setAttribute('role', 'menuitem');\r\n button.setAttribute('aria-label', item.label);\r\n button.setAttribute('aria-pressed', String(state.active));\r\n\r\n appendMenuIcon(button, iconSvg, item.label);\r\n labelElement.className = 'bridgerte__toolbar-group-menu-label';\r\n labelElement.textContent = item.label;\r\n button.append(labelElement);\r\n menuElement.append(button);\r\n });\r\n\r\n menuElement.dataset.visible = 'true';\r\n menuElement.style.minWidth = `${groupMenuState.button.offsetWidth}px`;\r\n};\r\n\r\nexport const syncToolbarGroupButtonState = (\r\n container: HTMLElement,\r\n groupMenuState: ToolbarGroupMenuState | null\r\n) => {\r\n container.querySelectorAll<HTMLButtonElement>(toolbarGroupButtonSelector).forEach((button) => {\r\n const open = groupMenuState?.groupKey === button.dataset.bridgerteToolbarGroupId;\r\n\r\n button.dataset.open = String(open);\r\n button.setAttribute('aria-expanded', String(open));\r\n });\r\n};\r\n\r\nexport const findToolbarGroupMenuItem = (\r\n toolbarItems: ResolvedToolbarItem[],\r\n groupKey: string | undefined\r\n) => toolbarItems.find((item): item is Extract<ResolvedToolbarItem, { type: 'group' }> => (\r\n item.type === 'group' && item.key === groupKey\r\n));\r\n","import {\n toolbarExecutableButtonSelector,\n toolbarGroupButtonSelector\n} from './render';\n\nconst toolbarGroupTouchMoveTolerancePx = 8;\n\nconst isInsideNode = (container: HTMLElement, target: EventTarget | null) => (\n target instanceof Node && container.contains(target)\n);\n\nconst getClosestButton = (\n target: EventTarget | null,\n selector: string\n) => (\n target instanceof Element ? target.closest<HTMLButtonElement>(selector) : null\n);\n\n/**\n * 创建 toolbar 收纳菜单交互控制器。\n *\n * 这个控制器只处理 group menu overlay 的交互边界:收纳入口和菜单项在按下阶段不能抢走\n * 编辑器焦点,touch tap 需要在 touchend 主动完成,外部 pointerdown 才关闭菜单。\n */\nexport const createToolbarGroupMenuInteraction = (\n container: HTMLElement,\n groupMenuElement: HTMLElement,\n toggleGroupMenu: (button: HTMLButtonElement) => void,\n executeToolbarButton: (button: HTMLButtonElement) => void,\n closeGroupMenu: () => void\n) => {\n let pendingTouch: {\n button: HTMLButtonElement;\n kind: 'group' | 'menu-item';\n startX: number;\n startY: number;\n } | null = null;\n let suppressNextClickButton: HTMLButtonElement | null = null;\n\n const getInteractiveButtonFromTarget = (target: EventTarget | null) => (\n getClosestButton(target, toolbarGroupButtonSelector)\n ?? (\n isInsideNode(groupMenuElement, target)\n ? getClosestButton(target, toolbarExecutableButtonSelector)\n : null\n )\n );\n\n const handlePressStart = (event: PointerEvent | MouseEvent) => {\n const button = getInteractiveButtonFromTarget(event.target);\n\n if (!button) return;\n\n /*\n * 收纳入口和收纳菜单项都属于编辑器工具区。按下阶段阻止默认 focus 转移,\n * 避免 H5 contenteditable blur 后软键盘收起,宿主随即隐藏 toolbar 和 overlay。\n */\n event.preventDefault();\n };\n\n const handleTouchStart = (event: TouchEvent) => {\n const touch = event.touches[0];\n const groupButton = getClosestButton(event.target, toolbarGroupButtonSelector);\n const menuItemButton = isInsideNode(groupMenuElement, event.target)\n ? getClosestButton(event.target, toolbarExecutableButtonSelector)\n : null;\n const button = groupButton ?? menuItemButton;\n\n if (!button || !touch) return;\n\n event.preventDefault();\n pendingTouch = {\n button,\n kind: groupButton ? 'group' : 'menu-item',\n startX: touch.clientX,\n startY: touch.clientY\n };\n };\n\n const clearPendingTouch = () => {\n pendingTouch = null;\n };\n\n const handleTouchEnd = (event: TouchEvent) => {\n if (!pendingTouch) return;\n\n const touchState = pendingTouch;\n const touch = event.changedTouches[0];\n const endButton = getInteractiveButtonFromTarget(event.target);\n\n pendingTouch = null;\n if (!touch || endButton !== touchState.button) return;\n\n const deltaX = Math.abs(touch.clientX - touchState.startX);\n const deltaY = Math.abs(touch.clientY - touchState.startY);\n\n if (\n deltaX > toolbarGroupTouchMoveTolerancePx\n || deltaY > toolbarGroupTouchMoveTolerancePx\n ) return;\n\n event.preventDefault();\n event.stopPropagation();\n suppressNextClickButton = touchState.button;\n\n if (touchState.kind === 'group') {\n toggleGroupMenu(touchState.button);\n return;\n }\n\n executeToolbarButton(touchState.button);\n closeGroupMenu();\n };\n\n const handleDocumentPointerDown = (event: PointerEvent) => {\n if (\n isInsideNode(container, event.target)\n || isInsideNode(groupMenuElement, event.target)\n ) return;\n\n closeGroupMenu();\n };\n\n container.addEventListener('pointerdown', handlePressStart);\n container.addEventListener('mousedown', handlePressStart);\n container.addEventListener('touchstart', handleTouchStart, { passive: false });\n container.addEventListener('touchend', handleTouchEnd);\n container.addEventListener('touchcancel', clearPendingTouch);\n groupMenuElement.addEventListener('pointerdown', handlePressStart);\n groupMenuElement.addEventListener('mousedown', handlePressStart);\n groupMenuElement.addEventListener('touchstart', handleTouchStart, { passive: false });\n groupMenuElement.addEventListener('touchend', handleTouchEnd);\n groupMenuElement.addEventListener('touchcancel', clearPendingTouch);\n document.addEventListener('pointerdown', handleDocumentPointerDown, true);\n\n return {\n consumeSuppressedClick(button: HTMLButtonElement) {\n if (suppressNextClickButton !== button) return false;\n\n suppressNextClickButton = null;\n return true;\n },\n destroy() {\n container.removeEventListener('pointerdown', handlePressStart);\n container.removeEventListener('mousedown', handlePressStart);\n container.removeEventListener('touchstart', handleTouchStart);\n container.removeEventListener('touchend', handleTouchEnd);\n container.removeEventListener('touchcancel', clearPendingTouch);\n groupMenuElement.removeEventListener('pointerdown', handlePressStart);\n groupMenuElement.removeEventListener('mousedown', handlePressStart);\n groupMenuElement.removeEventListener('touchstart', handleTouchStart);\n groupMenuElement.removeEventListener('touchend', handleTouchEnd);\n groupMenuElement.removeEventListener('touchcancel', clearPendingTouch);\n document.removeEventListener('pointerdown', handleDocumentPointerDown, true);\n }\n };\n};\n","import type { CommandState } from '@bridgerte/core';\r\nimport {\r\n defaultMenuSchema,\r\n resolveToolbarMenu\r\n} from '@bridgerte/native-spec';\r\nimport { createFloatingLayer, type RichTextFloatingLayer } from '../floatingLayer';\r\nimport { bindTouchPressedState } from '../interactionState';\r\nimport {\r\n getPayloadPanelCurrentValues,\r\n resolveMenuSchemaForDom\r\n} from '../menuRuntime';\r\nimport {\n focusToolbarGroupMenuItem,\n handleToolbarGroupMenuKeyDown\n} from './groupMenuKeyboard';\nimport { createToolbarGroupMenuInteraction } from './groupMenuInteraction';\nimport {\r\n findToolbarGroupMenuItem,\r\n renderToolbar,\r\n renderToolbarGroupMenu,\r\n syncToolbarGroupButtonState,\r\n toolbarExecutableButtonSelector,\r\n toolbarGroupButtonSelector\r\n} from './render';\r\nimport type {\r\n RichTextToolbarAPI,\r\n ToolbarGroupMenuState,\r\n RichTextToolbarOptions\r\n} from './type';\r\n\r\nconst toolbarTooltipOffsetPx = 8;\r\nconst toolbarGroupMenuOffsetPx = 6;\r\n\r\nexport type * from './type';\r\n\r\nconst canUseHoverTooltip = () => (\r\n typeof window !== 'undefined'\r\n && window.matchMedia?.('(hover: hover) and (pointer: fine)').matches === true\r\n);\r\n\r\nconst getToolbarButtonFromTarget = (target: EventTarget | null) => {\r\n /*\r\n * 图标覆盖常用 SVG,H5 pointer 事件可能落在 svg/path 上。\r\n * 这里用 Element 而不是 HTMLElement,保证触屏兜底和 click 都能向上命中真实按钮。\r\n */\r\n const button = target instanceof Element\r\n ? target.closest<HTMLButtonElement>(toolbarExecutableButtonSelector)\r\n : null;\r\n\r\n return button instanceof HTMLButtonElement ? button : null;\r\n};\r\n\r\nconst getToolbarGroupButtonFromTarget = (target: EventTarget | null) => {\r\n const button = target instanceof Element\r\n ? target.closest<HTMLButtonElement>(toolbarGroupButtonSelector)\r\n : null;\r\n\r\n return button instanceof HTMLButtonElement ? button : null;\r\n};\r\n\r\n/**\r\n * 创建独立菜单实例。\r\n *\r\n * toolbar/tabbar 只订阅 EditorAPI 状态并派发命令,不接触编辑器内部 DOM 或 Lexical 实例。\r\n */\r\nexport function createRichTextToolbar(\r\n container: HTMLElement,\r\n options: RichTextToolbarOptions\r\n): RichTextToolbarAPI {\r\n const placement = options.placement ?? 'top';\r\n const menuSchema = resolveMenuSchemaForDom(options.menuSchema ?? defaultMenuSchema, {\r\n menuLabels: options.menuLabels,\r\n payloadPanelConfig: options.payloadPanelConfig\r\n });\r\n const toolbarItems = resolveToolbarMenu(options.toolbarConfig, menuSchema);\r\n /*\r\n * click 事件只需要能执行命令的菜单项。分割线和 group 容器是渲染结构,\r\n * 先在这里摊平,后续点击时就不用理解 toolbarConfig 的展示层级。\r\n */\r\n const executableMenuItems = toolbarItems.flatMap((toolbarItem) => (\r\n toolbarItem.type === 'button' ? [toolbarItem.item]\r\n : toolbarItem.type === 'group' ? toolbarItem.items\r\n : []\r\n ));\r\n const icons = options.icons ?? {};\r\n /*\r\n * tooltip 只属于 PC 精细指针体验。H5 上即使没有原生 title,部分浏览器也会把 hover/mouseover\r\n * 模拟成首触摸状态,导致第一次点像是只唤醒提示;所以触屏端不写 tooltip 数据也不挂监听。\r\n */\r\n const enableTooltip = canUseHoverTooltip();\r\n const tooltipElement = document.createElement('div');\r\n const groupMenuElement = document.createElement('div');\r\n const overlayHost = container.closest('.bridgerte') ?? container;\r\n let destroyed = false;\r\n let latestCommandStates = options.editor.getCommandStates();\r\n let groupMenuState: ToolbarGroupMenuState | null = null;\r\n let groupMenuFloatingLayer: RichTextFloatingLayer | null = null;\r\n const clearToolbarPressedState = bindTouchPressedState(container, {\r\n targetSelector: [\r\n 'button[data-bridgerte-toolbar-item-id]',\r\n 'button[data-bridgerte-toolbar-group-id]'\r\n ].join(',')\r\n });\r\n const clearGroupMenuPressedState = bindTouchPressedState(groupMenuElement, {\r\n targetSelector: '.bridgerte__toolbar-group-menu-item'\r\n });\r\n\r\n const mountToolbarOverlays = () => {\r\n // 独立 toolbar 可能不在 `.bridgerte` 内,渲染按钮会清空 container,需要把浮层重新挂回去。\r\n overlayHost.append(tooltipElement);\r\n overlayHost.append(groupMenuElement);\r\n };\r\n\r\n const closeGroupMenuFloatingLayer = () => {\r\n groupMenuFloatingLayer?.setOpen(false);\r\n groupMenuFloatingLayer?.destroy();\r\n groupMenuFloatingLayer = null;\r\n };\r\n\r\n const openGroupMenuFloatingLayer = () => {\r\n if (!groupMenuState) return;\r\n\r\n closeGroupMenuFloatingLayer();\r\n /*\r\n * group menu 和 hoverbar/mention/slash 一样属于轻浮层,必须走 floatingLayer。\r\n * 这里按 toolbar placement 给出首选方向,真正的翻转、键盘可视区和左右夹紧交给\r\n * createFloatingLayer 内部的 visualViewport + flip + shift + clamp 统一处理。\r\n */\r\n groupMenuFloatingLayer = createFloatingLayer(groupMenuState.button, groupMenuElement, {\r\n placement: placement === 'bottom' ? 'top-start' : 'bottom-start',\r\n offset: toolbarGroupMenuOffsetPx,\r\n strategy: 'fixed'\r\n });\r\n groupMenuFloatingLayer.setOpen(true);\r\n };\r\n\r\n const closeGroupMenu = () => {\r\n closeGroupMenuFloatingLayer();\r\n groupMenuState = null;\r\n renderToolbarGroupMenu(groupMenuElement, groupMenuState, latestCommandStates, icons);\r\n syncToolbarGroupButtonState(container, groupMenuState);\r\n };\r\n\r\n const renderStates = (states: CommandState[]) => {\r\n if (destroyed) return;\r\n\r\n latestCommandStates = states;\r\n if (groupMenuState) closeGroupMenuFloatingLayer();\r\n renderToolbar(container, toolbarItems, states, icons, enableTooltip);\r\n mountToolbarOverlays();\r\n if (groupMenuState) {\r\n const nextButton = Array.from(\r\n container.querySelectorAll<HTMLButtonElement>(toolbarGroupButtonSelector)\r\n ).find((button) => button.dataset.bridgerteToolbarGroupId === groupMenuState?.groupKey);\r\n const nextGroupItem = findToolbarGroupMenuItem(toolbarItems, groupMenuState.groupKey);\r\n\r\n groupMenuState = nextButton && nextGroupItem\r\n ? {\r\n groupKey: groupMenuState.groupKey,\r\n button: nextButton,\r\n items: nextGroupItem.items\r\n }\r\n : null;\r\n renderToolbarGroupMenu(groupMenuElement, groupMenuState, latestCommandStates, icons);\r\n syncToolbarGroupButtonState(container, groupMenuState);\r\n if (groupMenuState) openGroupMenuFloatingLayer();\r\n }\r\n };\r\n\r\n const update = () => {\r\n if (destroyed) return;\r\n\r\n renderStates(options.editor.getCommandStates());\r\n };\r\n\r\n const hideTooltip = () => {\r\n tooltipElement.dataset.visible = 'false';\r\n tooltipElement.textContent = '';\r\n };\r\n\r\n const showTooltip = (button: HTMLButtonElement) => {\r\n const tooltipText = button.dataset.tooltip;\r\n\r\n if (!enableTooltip || !tooltipText) return;\r\n\r\n const buttonRect = button.getBoundingClientRect();\r\n\r\n tooltipElement.textContent = tooltipText;\r\n tooltipElement.dataset.visible = 'true';\r\n tooltipElement.style.left = `${buttonRect.left + buttonRect.width / 2}px`;\r\n tooltipElement.style.top = `${buttonRect.top - toolbarTooltipOffsetPx}px`;\r\n };\r\n\r\n const handleTooltipTarget = (event: Event) => {\r\n const button = getToolbarButtonFromTarget(event.target);\r\n\r\n if (button) {\r\n showTooltip(button);\r\n }\r\n };\r\n\r\n const handleTooltipLeave = (event: MouseEvent) => {\r\n const relatedTarget = event.relatedTarget;\r\n const button = getToolbarButtonFromTarget(event.target);\r\n\r\n if (\r\n button\r\n && relatedTarget instanceof Node\r\n && button.contains(relatedTarget)\r\n ) return;\r\n\r\n hideTooltip();\r\n };\r\n\r\n const executeToolbarButton = (button: HTMLButtonElement) => {\r\n if (!(button instanceof HTMLButtonElement) || button.disabled) return;\r\n\r\n const menuItem = executableMenuItems.find((item) => (\r\n item.id === button.dataset.bridgerteToolbarItemId\r\n ));\r\n\r\n if (!menuItem) return;\r\n\r\n if (menuItem.payloadPanel) {\r\n const buttonRect = button.getBoundingClientRect();\r\n\r\n /*\r\n * 带 payloadPanel 的菜单不直接执行基础 command,而是发起参数请求。\r\n * DOM 内置面板和业务/RN/Flutter 自绘都走同一个 request,避免后续颜色、\r\n * 字体、链接等参数菜单各自发明一套协议。\r\n */\r\n options.editor.requestPayloadPanel({\r\n menuId: menuItem.id,\r\n command: menuItem.command,\r\n panel: menuItem.payloadPanel,\r\n currentValues: getPayloadPanelCurrentValues(menuItem, options.editor.getCommandStates()),\r\n anchorRect: {\r\n x: buttonRect.left,\r\n y: buttonRect.top,\r\n width: buttonRect.width,\r\n height: buttonRect.height\r\n }\r\n });\r\n return;\r\n }\r\n\r\n options.editor.executeCommand(menuItem.command);\r\n };\r\n\r\n const toggleGroupMenu = (button: HTMLButtonElement, shouldFocusMenu = false) => {\r\n const groupItem = findToolbarGroupMenuItem(\r\n toolbarItems,\r\n button.dataset.bridgerteToolbarGroupId\r\n );\r\n\r\n if (!groupItem || button.disabled) return;\r\n\r\n if (groupMenuState?.groupKey === groupItem.key) {\r\n if (shouldFocusMenu) {\r\n focusToolbarGroupMenuItem(groupMenuElement, 0);\r\n return;\r\n }\r\n closeGroupMenu();\r\n return;\r\n }\r\n\r\n groupMenuState = {\r\n groupKey: groupItem.key,\r\n button,\r\n items: groupItem.items\r\n };\r\n hideTooltip();\r\n renderToolbarGroupMenu(groupMenuElement, groupMenuState, latestCommandStates, icons);\r\n syncToolbarGroupButtonState(container, groupMenuState);\r\n openGroupMenuFloatingLayer();\r\n if (shouldFocusMenu) focusToolbarGroupMenuItem(groupMenuElement, 0);\r\n };\r\n\r\n const handleClick = (event: MouseEvent) => {\n const groupButton = getToolbarGroupButtonFromTarget(event.target);\n if (groupButton) {\n event.preventDefault();\n event.stopPropagation();\n if (groupMenuInteraction.consumeSuppressedClick(groupButton)) return;\n\n toggleGroupMenu(groupButton);\n return;\n }\n\n const button = getToolbarButtonFromTarget(event.target);\n if (!button) return;\r\n\r\n event.preventDefault();\n event.stopPropagation();\n if (groupMenuInteraction.consumeSuppressedClick(button)) return;\n\n executeToolbarButton(button);\n closeGroupMenu();\n };\n\r\n const handleKeyDown = (event: KeyboardEvent) => {\r\n if (groupMenuState && groupMenuElement.contains(document.activeElement)) {\r\n handleToolbarGroupMenuKeyDown(\r\n event,\r\n groupMenuElement,\r\n groupMenuState.button,\r\n executeToolbarButton,\r\n closeGroupMenu\r\n );\r\n return;\r\n }\r\n\r\n const groupButton = getToolbarGroupButtonFromTarget(event.target);\r\n if (\r\n groupButton\r\n && (event.key === 'Enter' || event.key === ' ' || event.key === 'ArrowDown')\r\n ) {\r\n /*\r\n * 指针点击收纳菜单不能抢编辑器焦点;键盘从入口按钮打开时则进入菜单项,\r\n * 保留 PC 可访问性和方向键 roving focus。\r\n */\r\n event.preventDefault();\r\n event.stopPropagation();\r\n toggleGroupMenu(groupButton, true);\r\n return;\r\n }\r\n\r\n if (event.key !== 'Escape') return;\r\n\r\n closeGroupMenu();\r\n };\r\n\r\n container.classList.add('bridgerte__toolbar');\r\n tooltipElement.className = 'bridgerte__toolbar-tooltip';\r\n tooltipElement.dataset.visible = 'false';\r\n groupMenuElement.className = 'bridgerte__floating-menu bridgerte__toolbar-group-menu';\r\n groupMenuElement.dataset.visible = 'false';\r\n groupMenuElement.setAttribute('role', 'menu');\r\n container.dataset.placement = placement;\r\n container.setAttribute('role', 'toolbar');\r\n container.setAttribute(\r\n 'aria-label',\r\n placement === 'bottom' ? 'BridgeRTE tabbar' : 'BridgeRTE toolbar'\r\n );\r\n container.addEventListener('click', handleClick);\n groupMenuElement.addEventListener('click', handleClick);\n document.addEventListener('keydown', handleKeyDown);\n if (enableTooltip) {\r\n container.addEventListener('mouseover', handleTooltipTarget);\r\n container.addEventListener('mouseout', handleTooltipLeave);\r\n }\r\n container.addEventListener('focusout', hideTooltip);\r\n // 浮层挂在最近的编辑器根容器下,既能继承变量,也不会操作编辑内容 DOM。\n mountToolbarOverlays();\n const groupMenuInteraction = createToolbarGroupMenuInteraction(\n container,\n groupMenuElement,\n toggleGroupMenu,\n executeToolbarButton,\n closeGroupMenu\n );\n\r\n // 独立 toolbar 只订阅 public API 状态,不依赖 DOM 编辑器内部实现。\r\n const unsubscribe = options.editor.subscribeCommandStateChange(renderStates);\r\n\r\n return {\r\n update,\r\n destroy() {\r\n if (destroyed) return;\r\n\r\n destroyed = true;\r\n unsubscribe();\r\n closeGroupMenu();\n container.removeEventListener('click', handleClick);\n groupMenuElement.removeEventListener('click', handleClick);\n document.removeEventListener('keydown', handleKeyDown);\n groupMenuInteraction.destroy();\n if (enableTooltip) {\r\n container.removeEventListener('mouseover', handleTooltipTarget);\r\n container.removeEventListener('mouseout', handleTooltipLeave);\r\n }\r\n clearToolbarPressedState();\r\n clearGroupMenuPressedState();\r\n container.removeEventListener('focusout', hideTooltip);\r\n tooltipElement.remove();\r\n groupMenuElement.remove();\r\n container.classList.remove('bridgerte__toolbar');\r\n delete container.dataset.placement;\r\n container.textContent = '';\r\n container.removeAttribute('role');\r\n container.removeAttribute('aria-label');\r\n }\r\n };\r\n}\r\n","const invisibleTextPattern = /[\\u200B-\\u200D\\uFEFF]/g;\n\n/*\n * HTML 片段里这些元素没有 textContent 也应算有效内容。\n * 纯排版容器不放进来,避免 `<p><br></p>`、空列表项这类编辑器占位被误判为非空。\n */\nconst meaningfulHtmlContentSelector = [\n 'img',\n 'video',\n 'audio',\n 'iframe',\n 'table',\n 'pre',\n 'code',\n 'hr',\n 'figure',\n 'canvas',\n 'svg',\n 'math',\n '[data-type=\"mention\"]'\n].join(',');\n\nconst normalizeHtmlText = (text: string | null | undefined): string => (\n text?.replace(invisibleTextPattern, '').trim() ?? ''\n);\n\n/**\n * 判断一段 HTML 片段是否包含可保存的富文本内容。\n *\n * 这个工具会使用浏览器 template 解析 HTML,适合只拿到 HTML 字符串的业务表单;如果已经持有\n * `EditorContent`,优先使用 core 的 `isEditorContentEmpty()`,避免额外 DOM parse。\n */\nexport const hasMeaningfulHtmlContent = (html: string): boolean => {\n if (!normalizeHtmlText(html)) return false;\n\n const template = document.createElement('template');\n\n template.innerHTML = html;\n\n return Boolean(\n normalizeHtmlText(template.content.textContent)\n || template.content.querySelector(meaningfulHtmlContentSelector)\n );\n};\n"],"names":["defaultMenuUiIcons","getEnabledGroupMenuButtons","menuElement","button","focusToolbarGroupMenuItem","index","nextButton","handleToolbarGroupMenuKeyDown","event","returnButton","executeToolbarButton","closeGroupMenu","buttons","activeIndex","focusByOffset","offset","nextIndex","_a","_b","toolbarGroupButtonSelector","toolbarButtonSelector","toolbarExecutableButtonSelector","getGroupMenuState","items","commandStates","itemStates","item","getMenuStateForItem","state","renderToolbarButton","groupElement","icons","enableTooltip","iconSvg","defaultMenuIcons","appendMenuIcon","renderToolbarGroupButton","toolbarItem","groupState","indicator","renderToolbar","toolbarElement","toolbarItems","separatorElement","renderToolbarGroupMenu","groupMenuState","labelElement","syncToolbarGroupButtonState","container","open","findToolbarGroupMenuItem","groupKey","toolbarGroupTouchMoveTolerancePx","isInsideNode","target","getClosestButton","selector","createToolbarGroupMenuInteraction","groupMenuElement","toggleGroupMenu","pendingTouch","suppressNextClickButton","getInteractiveButtonFromTarget","handlePressStart","handleTouchStart","touch","groupButton","menuItemButton","clearPendingTouch","handleTouchEnd","touchState","endButton","deltaX","deltaY","handleDocumentPointerDown","toolbarTooltipOffsetPx","toolbarGroupMenuOffsetPx","canUseHoverTooltip","getToolbarButtonFromTarget","getToolbarGroupButtonFromTarget","createRichTextToolbar","options","placement","menuSchema","resolveMenuSchemaForDom","defaultMenuSchema","resolveToolbarMenu","executableMenuItems","tooltipElement","overlayHost","destroyed","latestCommandStates","groupMenuFloatingLayer","clearToolbarPressedState","bindTouchPressedState","clearGroupMenuPressedState","mountToolbarOverlays","closeGroupMenuFloatingLayer","openGroupMenuFloatingLayer","createFloatingLayer","renderStates","states","nextGroupItem","update","hideTooltip","showTooltip","tooltipText","buttonRect","handleTooltipTarget","handleTooltipLeave","relatedTarget","menuItem","getPayloadPanelCurrentValues","shouldFocusMenu","groupItem","handleClick","groupMenuInteraction","handleKeyDown","unsubscribe","invisibleTextPattern","meaningfulHtmlContentSelector","normalizeHtmlText","text","hasMeaningfulHtmlContent","html","template"],"mappings":"oFAMaA,EAAqB,CAChC,aAAc;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,IAmBd,YAAa;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,GAiBf,EC3CaC,EAA8BC,GACzC,MAAM,KACJA,EAAY,iBAAoC,qCAAqC,CACvF,EAAE,OAAQC,GAAW,CAACA,EAAO,QAAQ,EAG1BC,EAA4B,CAACF,EAA0BG,IAAkB,CAEpF,MAAMC,EADUL,EAA2BC,CAAW,EAC3BG,CAAK,EAE5BC,KAAuB,MAAA,CAC7B,EAEaC,EAAgC,CAC3CC,EACAN,EACAO,EACAC,EACAC,IACG,SACH,MAAMC,EAAUX,EAA2BC,CAAW,EAChDW,EAAc,SAAS,yBAAyB,kBAClDD,EAAQ,QAAQ,SAAS,aAAa,EACtC,GACEE,EAAiBC,GAAmB,OACxC,GAAIH,EAAQ,SAAW,EAAG,OAG1B,MAAMI,IADYH,GAAe,EAAIA,EAAc,GACpBE,EAASH,EAAQ,QAAUA,EAAQ,QAElEK,EAAAL,EAAQI,CAAS,IAAjB,MAAAC,EAAoB,OACtB,EAMA,OAAQT,EAAM,IAAA,CACZ,IAAK,YACHA,EAAM,eAAA,EACNM,EAAc,CAAC,EACf,MACF,IAAK,UACHN,EAAM,eAAA,EACNM,EAAc,EAAE,EAChB,MACF,IAAK,OACHN,EAAM,eAAA,GACNS,EAAAL,EAAQ,CAAC,IAAT,MAAAK,EAAY,QACZ,MACF,IAAK,MACHT,EAAM,eAAA,GACNU,EAAAN,EAAQ,GAAG,EAAE,IAAb,MAAAM,EAAgB,QAChB,MACF,IAAK,QACL,IAAK,IACC,SAAS,yBAAyB,oBACpCV,EAAM,eAAA,EACNE,EAAqB,SAAS,aAAa,EAC3CC,EAAA,EACAF,EAAa,MAAA,GAEf,MACF,IAAK,SACHD,EAAM,eAAA,EACNC,EAAa,MAAA,EACbE,EAAA,EACA,KAEA,CAEN,EC1DaQ,EAA6B,0CAC7BC,GAAwB,yCAExBC,EAAkC,CAC7CD,GACA,qCACF,EAAE,KAAK,GAAG,EAEJE,GAAoB,CAACC,EAAmBC,IAAkC,CAC9E,MAAMC,EAAaF,EAAM,IAAKG,GAASC,sBAAoBD,EAAMF,CAAa,CAAC,EAE/E,MAAO,CACL,OAAQC,EAAW,KAAMG,GAAUA,EAAM,MAAM,EAC/C,SAAUH,EAAW,OAAS,GAAKA,EAAW,MAAOG,GAAUA,EAAM,QAAQ,CAAA,CAEjF,EAEMC,GAAsB,CAC1BC,EACAJ,EACAF,EACAO,EACAC,IACG,CACH,MAAMJ,EAAQD,EAAAA,oBAAoBD,EAAMF,CAAa,EAC/CrB,EAAS,SAAS,cAAc,QAAQ,EAExC8B,EAAUF,EAAML,EAAK,IAAI,GAAKQ,EAAAA,iBAAiBR,EAAK,IAAI,EAE9DvB,EAAO,KAAO,SACdA,EAAO,UAAY,4BACnBA,EAAO,SAAWyB,EAAM,SACxBzB,EAAO,QAAQ,OAAS,OAAOyB,EAAM,MAAM,EAC3CzB,EAAO,QAAQ,uBAAyBuB,EAAK,GAC7CvB,EAAO,aAAa,aAAcuB,EAAK,KAAK,EAC5CvB,EAAO,aAAa,eAAgB,OAAOyB,EAAM,MAAM,CAAC,EACpDI,IAAe7B,EAAO,QAAQ,QAAUuB,EAAK,OAEjDS,EAAAA,eAAehC,EAAQ8B,EAASP,EAAK,KAAK,EAE1CI,EAAa,OAAO3B,CAAM,CAC5B,EAEMiC,GAA2B,CAC/BN,EACAO,EACAb,EACAO,EACAC,IACG,CACH,MAAM7B,EAAS,SAAS,cAAc,QAAQ,EACxCmC,EAAahB,GAAkBe,EAAY,MAAOb,CAAa,EAC/DS,EAAUI,EAAY,KACxBN,EAAMM,EAAY,IAAI,GAAKH,EAAAA,iBAAiBG,EAAY,IAAI,EAC5DrC,EAAmB,aACjBuC,EAAY,SAAS,cAAc,MAAM,EAE/CpC,EAAO,KAAO,SACdA,EAAO,UAAY,4DACnBA,EAAO,SAAWmC,EAAW,SAC7BnC,EAAO,QAAQ,OAAS,OAAOmC,EAAW,MAAM,EAChDnC,EAAO,QAAQ,wBAA0BkC,EAAY,IACrDlC,EAAO,QAAQ,KAAO,QACtBA,EAAO,aAAa,aAAckC,EAAY,KAAK,EACnDlC,EAAO,aAAa,gBAAiB,MAAM,EAC3CA,EAAO,aAAa,gBAAiB,OAAO,EAC5CA,EAAO,aAAa,eAAgB,OAAOmC,EAAW,MAAM,CAAC,EACzDN,IAAe7B,EAAO,QAAQ,QAAUkC,EAAY,OAExDF,EAAAA,eAAehC,EAAQ8B,EAASI,EAAY,KAAK,EAEjDE,EAAU,UAAY,qCACtBA,EAAU,aAAa,cAAe,MAAM,EAC5CA,EAAU,UAAYvC,EAAmB,YACzCG,EAAO,OAAOoC,CAAS,EACvBT,EAAa,OAAO3B,CAAM,CAC5B,EAQaqC,GAAgB,CAC3BC,EACAC,EACAlB,EACAO,EACAC,IACG,CACHS,EAAe,YAAc,GAE7BC,EAAa,QAASL,GAAgB,CACpC,GAAIA,EAAY,OAAS,YAAa,CACpC,MAAMM,EAAmB,SAAS,cAAc,MAAM,EAEtDA,EAAiB,UAAY,+BAC7BA,EAAiB,QAAQ,YAAcN,EAAY,IACnDM,EAAiB,aAAa,cAAe,MAAM,EACnDF,EAAe,OAAOE,CAAgB,EACtC,MACF,CAEA,GAAIN,EAAY,OAAS,SAAU,CACjCR,GACEY,EACAJ,EAAY,KACZb,EACAO,EACAC,CAAA,EAEF,MACF,CAMA,MAAMF,EAAe,SAAS,cAAc,KAAK,EAEjDA,EAAa,UAAY,2BACzBA,EAAa,QAAQ,MAAQO,EAAY,IACzCP,EAAa,aAAa,aAAcO,EAAY,KAAK,EACzDI,EAAe,OAAOX,CAAY,EAElCM,GACEN,EACAO,EACAb,EACAO,EACAC,CAAA,CAEJ,CAAC,CACH,EAOaY,EAAyB,CACpC1C,EACA2C,EACArB,EACAO,IACG,CAGH,GAFA7B,EAAY,YAAc,GAEtB,CAAC2C,EAAgB,CACnB3C,EAAY,QAAQ,QAAU,QAC9B,MACF,CAEA2C,EAAe,MAAM,QAASnB,GAAS,CACrC,MAAME,EAAQD,EAAAA,oBAAoBD,EAAMF,CAAa,EAC/CrB,EAAS,SAAS,cAAc,QAAQ,EACxC8B,EAAUF,EAAML,EAAK,IAAI,GAAKQ,EAAAA,iBAAiBR,EAAK,IAAI,EACxDoB,EAAe,SAAS,cAAc,MAAM,EAElD3C,EAAO,KAAO,SACdA,EAAO,UAAY,0DACnBA,EAAO,SAAWyB,EAAM,SACxBzB,EAAO,QAAQ,OAAS,OAAOyB,EAAM,MAAM,EAC3CzB,EAAO,QAAQ,uBAAyBuB,EAAK,GAC7CvB,EAAO,aAAa,OAAQ,UAAU,EACtCA,EAAO,aAAa,aAAcuB,EAAK,KAAK,EAC5CvB,EAAO,aAAa,eAAgB,OAAOyB,EAAM,MAAM,CAAC,EAExDO,EAAAA,eAAehC,EAAQ8B,EAASP,EAAK,KAAK,EAC1CoB,EAAa,UAAY,sCACzBA,EAAa,YAAcpB,EAAK,MAChCvB,EAAO,OAAO2C,CAAY,EAC1B5C,EAAY,OAAOC,CAAM,CAC3B,CAAC,EAEDD,EAAY,QAAQ,QAAU,OAC9BA,EAAY,MAAM,SAAW,GAAG2C,EAAe,OAAO,WAAW,IACnE,EAEaE,EAA8B,CACzCC,EACAH,IACG,CACHG,EAAU,iBAAoC7B,CAA0B,EAAE,QAAShB,GAAW,CAC5F,MAAM8C,GAAOJ,GAAA,YAAAA,EAAgB,YAAa1C,EAAO,QAAQ,wBAEzDA,EAAO,QAAQ,KAAO,OAAO8C,CAAI,EACjC9C,EAAO,aAAa,gBAAiB,OAAO8C,CAAI,CAAC,CACnD,CAAC,CACH,EAEaC,EAA2B,CACtCR,EACAS,IACGT,EAAa,KAAMhB,GACtBA,EAAK,OAAS,SAAWA,EAAK,MAAQyB,CACvC,EC7MKC,EAAmC,EAEnCC,EAAe,CAACL,EAAwBM,IAC5CA,aAAkB,MAAQN,EAAU,SAASM,CAAM,EAG/CC,EAAmB,CACvBD,EACAE,IAEAF,aAAkB,QAAUA,EAAO,QAA2BE,CAAQ,EAAI,KAS/DC,GAAoC,CAC/CT,EACAU,EACAC,EACAjD,EACAC,IACG,CACH,IAAIiD,EAKO,KACPC,EAAoD,KAExD,MAAMC,EAAkCR,GACtCC,EAAiBD,EAAQnC,CAA0B,IAE/CkC,EAAaK,EAAkBJ,CAAM,EACjCC,EAAiBD,EAAQjC,CAA+B,EACxD,MAIJ0C,EAAoBvD,GAAqC,CAC9CsD,EAA+BtD,EAAM,MAAM,GAQ1DA,EAAM,eAAA,CACR,EAEMwD,EAAoBxD,GAAsB,CAC9C,MAAMyD,EAAQzD,EAAM,QAAQ,CAAC,EACvB0D,EAAcX,EAAiB/C,EAAM,OAAQW,CAA0B,EACvEgD,EAAiBd,EAAaK,EAAkBlD,EAAM,MAAM,EAC9D+C,EAAiB/C,EAAM,OAAQa,CAA+B,EAC9D,KACElB,EAAS+D,GAAeC,EAE1B,CAAChE,GAAU,CAAC8D,IAEhBzD,EAAM,eAAA,EACNoD,EAAe,CACb,OAAAzD,EACA,KAAM+D,EAAc,QAAU,YAC9B,OAAQD,EAAM,QACd,OAAQA,EAAM,OAAA,EAElB,EAEMG,EAAoB,IAAM,CAC9BR,EAAe,IACjB,EAEMS,EAAkB7D,GAAsB,CAC5C,GAAI,CAACoD,EAAc,OAEnB,MAAMU,EAAaV,EACbK,EAAQzD,EAAM,eAAe,CAAC,EAC9B+D,EAAYT,EAA+BtD,EAAM,MAAM,EAG7D,GADAoD,EAAe,KACX,CAACK,GAASM,IAAcD,EAAW,OAAQ,OAE/C,MAAME,EAAS,KAAK,IAAIP,EAAM,QAAUK,EAAW,MAAM,EACnDG,EAAS,KAAK,IAAIR,EAAM,QAAUK,EAAW,MAAM,EAEzD,GACE,EAAAE,EAASpB,GACJqB,EAASrB,GAOhB,IAJA5C,EAAM,eAAA,EACNA,EAAM,gBAAA,EACNqD,EAA0BS,EAAW,OAEjCA,EAAW,OAAS,QAAS,CAC/BX,EAAgBW,EAAW,MAAM,EACjC,MACF,CAEA5D,EAAqB4D,EAAW,MAAM,EACtC3D,EAAA,EACF,EAEM+D,EAA6BlE,GAAwB,CAEvD6C,EAAaL,EAAWxC,EAAM,MAAM,GAC/B6C,EAAaK,EAAkBlD,EAAM,MAAM,GAGlDG,EAAA,CACF,EAEA,OAAAqC,EAAU,iBAAiB,cAAee,CAAgB,EAC1Df,EAAU,iBAAiB,YAAae,CAAgB,EACxDf,EAAU,iBAAiB,aAAcgB,EAAkB,CAAE,QAAS,GAAO,EAC7EhB,EAAU,iBAAiB,WAAYqB,CAAc,EACrDrB,EAAU,iBAAiB,cAAeoB,CAAiB,EAC3DV,EAAiB,iBAAiB,cAAeK,CAAgB,EACjEL,EAAiB,iBAAiB,YAAaK,CAAgB,EAC/DL,EAAiB,iBAAiB,aAAcM,EAAkB,CAAE,QAAS,GAAO,EACpFN,EAAiB,iBAAiB,WAAYW,CAAc,EAC5DX,EAAiB,iBAAiB,cAAeU,CAAiB,EAClE,SAAS,iBAAiB,cAAeM,EAA2B,EAAI,EAEjE,CACL,uBAAuBvE,EAA2B,CAChD,OAAI0D,IAA4B1D,EAAe,IAE/C0D,EAA0B,KACnB,GACT,EACA,SAAU,CACRb,EAAU,oBAAoB,cAAee,CAAgB,EAC7Df,EAAU,oBAAoB,YAAae,CAAgB,EAC3Df,EAAU,oBAAoB,aAAcgB,CAAgB,EAC5DhB,EAAU,oBAAoB,WAAYqB,CAAc,EACxDrB,EAAU,oBAAoB,cAAeoB,CAAiB,EAC9DV,EAAiB,oBAAoB,cAAeK,CAAgB,EACpEL,EAAiB,oBAAoB,YAAaK,CAAgB,EAClEL,EAAiB,oBAAoB,aAAcM,CAAgB,EACnEN,EAAiB,oBAAoB,WAAYW,CAAc,EAC/DX,EAAiB,oBAAoB,cAAeU,CAAiB,EACrE,SAAS,oBAAoB,cAAeM,EAA2B,EAAI,CAC7E,CAAA,CAEJ,EC9HMC,GAAyB,EACzBC,GAA2B,EAI3BC,GAAqB,IAAA,OACzB,cAAO,OAAW,OACb5D,EAAA,OAAO,aAAP,YAAAA,EAAA,YAAoB,sCAAsC,WAAY,IAGvE6D,EAA8BxB,GAA+B,CAKjE,MAAMnD,EAASmD,aAAkB,QAC7BA,EAAO,QAA2BjC,CAA+B,EACjE,KAEJ,OAAOlB,aAAkB,kBAAoBA,EAAS,IACxD,EAEM4E,EAAmCzB,GAA+B,CACtE,MAAMnD,EAASmD,aAAkB,QAC7BA,EAAO,QAA2BnC,CAA0B,EAC5D,KAEJ,OAAOhB,aAAkB,kBAAoBA,EAAS,IACxD,EAOO,SAAS6E,GACdhC,EACAiC,EACoB,CACpB,MAAMC,EAAYD,EAAQ,WAAa,MACjCE,EAAaC,EAAAA,wBAAwBH,EAAQ,YAAcI,EAAAA,kBAAmB,CAClF,WAAYJ,EAAQ,WACpB,mBAAoBA,EAAQ,kBAAA,CAC7B,EACKvC,EAAe4C,EAAAA,mBAAmBL,EAAQ,cAAeE,CAAU,EAKnEI,EAAsB7C,EAAa,QAASL,GAChDA,EAAY,OAAS,SAAW,CAACA,EAAY,IAAI,EAC7CA,EAAY,OAAS,QAAUA,EAAY,MACzC,EACP,EACKN,EAAQkD,EAAQ,OAAS,CAAA,EAKzBjD,EAAgB6C,GAAA,EAChBW,EAAiB,SAAS,cAAc,KAAK,EAC7C9B,EAAmB,SAAS,cAAc,KAAK,EAC/C+B,EAAczC,EAAU,QAAQ,YAAY,GAAKA,EACvD,IAAI0C,EAAY,GACZC,EAAsBV,EAAQ,OAAO,iBAAA,EACrCpC,EAA+C,KAC/C+C,EAAuD,KAC3D,MAAMC,EAA2BC,EAAAA,sBAAsB9C,EAAW,CAChE,eAAgB,CACd,yCACA,yCAAA,EACA,KAAK,GAAG,CAAA,CACX,EACK+C,EAA6BD,EAAAA,sBAAsBpC,EAAkB,CACzE,eAAgB,qCAAA,CACjB,EAEKsC,EAAuB,IAAM,CAEjCP,EAAY,OAAOD,CAAc,EACjCC,EAAY,OAAO/B,CAAgB,CACrC,EAEMuC,EAA8B,IAAM,CACxCL,GAAA,MAAAA,EAAwB,QAAQ,IAChCA,GAAA,MAAAA,EAAwB,UACxBA,EAAyB,IAC3B,EAEMM,EAA6B,IAAM,CAClCrD,IAELoD,EAAA,EAMAL,EAAyBO,EAAAA,oBAAoBtD,EAAe,OAAQa,EAAkB,CACpF,UAAWwB,IAAc,SAAW,YAAc,eAClD,OAAQN,GACR,SAAU,OAAA,CACX,EACDgB,EAAuB,QAAQ,EAAI,EACrC,EAEMjF,EAAiB,IAAM,CAC3BsF,EAAA,EACApD,EAAiB,KACjBD,EAAuBc,EAAkBb,EAAgB8C,EAAqB5D,CAAK,EACnFgB,EAA4BC,EAAWH,CAAc,CACvD,EAEMuD,EAAgBC,GAA2B,CAC/C,GAAI,CAAAX,IAEJC,EAAsBU,EAClBxD,GAAgBoD,EAAA,EACpBzD,GAAcQ,EAAWN,EAAc2D,EAAQtE,EAAOC,CAAa,EACnEgE,EAAA,EACInD,GAAgB,CAClB,MAAMvC,EAAa,MAAM,KACvB0C,EAAU,iBAAoC7B,CAA0B,CAAA,EACxE,KAAMhB,GAAWA,EAAO,QAAQ,2BAA4B0C,GAAA,YAAAA,EAAgB,SAAQ,EAChFyD,EAAgBpD,EAAyBR,EAAcG,EAAe,QAAQ,EAEpFA,EAAiBvC,GAAcgG,EAC3B,CACA,SAAUzD,EAAe,SACzB,OAAQvC,EACR,MAAOgG,EAAc,KAAA,EAErB,KACJ1D,EAAuBc,EAAkBb,EAAgB8C,EAAqB5D,CAAK,EACnFgB,EAA4BC,EAAWH,CAAc,EACjDA,GAAgBqD,EAAA,CACtB,CACF,EAEMK,EAAS,IAAM,CACfb,GAEJU,EAAanB,EAAQ,OAAO,kBAAkB,CAChD,EAEMuB,EAAc,IAAM,CACxBhB,EAAe,QAAQ,QAAU,QACjCA,EAAe,YAAc,EAC/B,EAEMiB,EAAetG,GAA8B,CACjD,MAAMuG,EAAcvG,EAAO,QAAQ,QAEnC,GAAI,CAAC6B,GAAiB,CAAC0E,EAAa,OAEpC,MAAMC,EAAaxG,EAAO,sBAAA,EAE1BqF,EAAe,YAAckB,EAC7BlB,EAAe,QAAQ,QAAU,OACjCA,EAAe,MAAM,KAAO,GAAGmB,EAAW,KAAOA,EAAW,MAAQ,CAAC,KACrEnB,EAAe,MAAM,IAAM,GAAGmB,EAAW,IAAMhC,EAAsB,IACvE,EAEMiC,EAAuBpG,GAAiB,CAC5C,MAAML,EAAS2E,EAA2BtE,EAAM,MAAM,EAElDL,GACFsG,EAAYtG,CAAM,CAEtB,EAEM0G,EAAsBrG,GAAsB,CAChD,MAAMsG,EAAgBtG,EAAM,cACtBL,EAAS2E,EAA2BtE,EAAM,MAAM,EAGpDL,GACK2G,aAAyB,MACzB3G,EAAO,SAAS2G,CAAa,GAGpCN,EAAA,CACF,EAEM9F,EAAwBP,GAA8B,CAC1D,GAAI,EAAEA,aAAkB,oBAAsBA,EAAO,SAAU,OAE/D,MAAM4G,EAAWxB,EAAoB,KAAM7D,GACzCA,EAAK,KAAOvB,EAAO,QAAQ,sBAC5B,EAED,GAAK4G,EAEL,IAAIA,EAAS,aAAc,CACzB,MAAMJ,EAAaxG,EAAO,sBAAA,EAO1B8E,EAAQ,OAAO,oBAAoB,CACjC,OAAQ8B,EAAS,GACjB,QAASA,EAAS,QAClB,MAAOA,EAAS,aAChB,cAAeC,EAAAA,6BAA6BD,EAAU9B,EAAQ,OAAO,kBAAkB,EACvF,WAAY,CACV,EAAG0B,EAAW,KACd,EAAGA,EAAW,IACd,MAAOA,EAAW,MAClB,OAAQA,EAAW,MAAA,CACrB,CACD,EACD,MACF,CAEA1B,EAAQ,OAAO,eAAe8B,EAAS,OAAO,EAChD,EAEMpD,EAAkB,CAACxD,EAA2B8G,EAAkB,KAAU,CAC9E,MAAMC,EAAYhE,EAChBR,EACAvC,EAAO,QAAQ,uBAAA,EAGjB,GAAI,GAAC+G,GAAa/G,EAAO,UAEzB,KAAI0C,GAAA,YAAAA,EAAgB,YAAaqE,EAAU,IAAK,CAC9C,GAAID,EAAiB,CACnB7G,EAA0BsD,EAAkB,CAAC,EAC7C,MACF,CACA/C,EAAA,EACA,MACF,CAEAkC,EAAiB,CACf,SAAUqE,EAAU,IACpB,OAAA/G,EACA,MAAO+G,EAAU,KAAA,EAEnBV,EAAA,EACA5D,EAAuBc,EAAkBb,EAAgB8C,EAAqB5D,CAAK,EACnFgB,EAA4BC,EAAWH,CAAc,EACrDqD,EAAA,EACIe,GAAiB7G,EAA0BsD,EAAkB,CAAC,EACpE,EAEMyD,EAAe3G,GAAsB,CACzC,MAAM0D,EAAca,EAAgCvE,EAAM,MAAM,EAChE,GAAI0D,EAAa,CAGf,GAFA1D,EAAM,eAAA,EACNA,EAAM,gBAAA,EACF4G,EAAqB,uBAAuBlD,CAAW,EAAG,OAE9DP,EAAgBO,CAAW,EAC3B,MACF,CAEA,MAAM/D,EAAS2E,EAA2BtE,EAAM,MAAM,EACjDL,IAELK,EAAM,eAAA,EACNA,EAAM,gBAAA,EACF,CAAA4G,EAAqB,uBAAuBjH,CAAM,IAEtDO,EAAqBP,CAAM,EAC3BQ,EAAA,GACF,EAEM0G,EAAiB7G,GAAyB,CAC9C,GAAIqC,GAAkBa,EAAiB,SAAS,SAAS,aAAa,EAAG,CACvEnD,EACEC,EACAkD,EACAb,EAAe,OACfnC,EACAC,CAAA,EAEF,MACF,CAEA,MAAMuD,EAAca,EAAgCvE,EAAM,MAAM,EAChE,GACE0D,IACM1D,EAAM,MAAQ,SAAWA,EAAM,MAAQ,KAAOA,EAAM,MAAQ,aAClE,CAKAA,EAAM,eAAA,EACNA,EAAM,gBAAA,EACNmD,EAAgBO,EAAa,EAAI,EACjC,MACF,CAEI1D,EAAM,MAAQ,UAElBG,EAAA,CACF,EAEAqC,EAAU,UAAU,IAAI,oBAAoB,EAC5CwC,EAAe,UAAY,6BAC3BA,EAAe,QAAQ,QAAU,QACjC9B,EAAiB,UAAY,yDAC7BA,EAAiB,QAAQ,QAAU,QACnCA,EAAiB,aAAa,OAAQ,MAAM,EAC5CV,EAAU,QAAQ,UAAYkC,EAC9BlC,EAAU,aAAa,OAAQ,SAAS,EACxCA,EAAU,aACR,aACAkC,IAAc,SAAW,mBAAqB,mBAAA,EAEhDlC,EAAU,iBAAiB,QAASmE,CAAW,EAC/CzD,EAAiB,iBAAiB,QAASyD,CAAW,EACtD,SAAS,iBAAiB,UAAWE,CAAa,EAC9CrF,IACFgB,EAAU,iBAAiB,YAAa4D,CAAmB,EAC3D5D,EAAU,iBAAiB,WAAY6D,CAAkB,GAE3D7D,EAAU,iBAAiB,WAAYwD,CAAW,EAElDR,EAAA,EACA,MAAMoB,EAAuB3D,GAC3BT,EACAU,EACAC,EACAjD,EACAC,CAAA,EAII2G,EAAcrC,EAAQ,OAAO,4BAA4BmB,CAAY,EAE3E,MAAO,CACL,OAAAG,EACA,SAAU,CACJb,IAEJA,EAAY,GACZ4B,EAAA,EACA3G,EAAA,EACAqC,EAAU,oBAAoB,QAASmE,CAAW,EAClDzD,EAAiB,oBAAoB,QAASyD,CAAW,EACzD,SAAS,oBAAoB,UAAWE,CAAa,EACrDD,EAAqB,QAAA,EACjBpF,IACFgB,EAAU,oBAAoB,YAAa4D,CAAmB,EAC9D5D,EAAU,oBAAoB,WAAY6D,CAAkB,GAE9DhB,EAAA,EACAE,EAAA,EACA/C,EAAU,oBAAoB,WAAYwD,CAAW,EACrDhB,EAAe,OAAA,EACf9B,EAAiB,OAAA,EACjBV,EAAU,UAAU,OAAO,oBAAoB,EAC/C,OAAOA,EAAU,QAAQ,UACzBA,EAAU,YAAc,GACxBA,EAAU,gBAAgB,MAAM,EAChCA,EAAU,gBAAgB,YAAY,EACxC,CAAA,CAEJ,CCzYA,MAAMuE,GAAuB,yBAMvBC,GAAgC,CACpC,MACA,QACA,QACA,SACA,QACA,MACA,OACA,KACA,SACA,SACA,MACA,OACA,uBACF,EAAE,KAAK,GAAG,EAEJC,EAAqBC,IACzBA,GAAA,YAAAA,EAAM,QAAQH,GAAsB,IAAI,SAAU,GASvCI,GAA4BC,GAA0B,CACjE,GAAI,CAACH,EAAkBG,CAAI,EAAG,MAAO,GAErC,MAAMC,EAAW,SAAS,cAAc,UAAU,EAElD,OAAAA,EAAS,UAAYD,EAEd,GACLH,EAAkBI,EAAS,QAAQ,WAAW,GAC3CA,EAAS,QAAQ,cAAcL,EAA6B,EAEnE"}
package/dist/index.cjs CHANGED
@@ -1,2 +1,2 @@
1
- "use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const t=require("./index-GaS65GL0.cjs"),a=require("./bridge.cjs"),e=require("./native-spec.cjs"),n=require("./index-BBzHVrSi.cjs"),o=require("./index-DIJtCo91.cjs"),r=require("./core.cjs"),i=require("./index-DF8OhKI4.cjs");exports.BRIDGERTE_CONTENT_VERSION=t.BRIDGERTE_CONTENT_VERSION;exports.BRIDGERTE_TABLE_INSERT_MAX_COLS=t.BRIDGERTE_TABLE_INSERT_MAX_COLS;exports.BRIDGERTE_TABLE_INSERT_MAX_ROWS=t.BRIDGERTE_TABLE_INSERT_MAX_ROWS;exports.BRIDGE_CONTENT_CHANGE_DEBOUNCE_MS=a.BRIDGE_CONTENT_CHANGE_DEBOUNCE_MS;exports.BRIDGE_HEIGHT_CHANGE_THROTTLE_MS=a.BRIDGE_HEIGHT_CHANGE_THROTTLE_MS;exports.defaultBridgeEventTiming=a.defaultBridgeEventTiming;exports.isBridgeMessage=a.isBridgeMessage;exports.codeBlockLanguagePanel=e.codeBlockLanguagePanel;exports.codeBlockLanguagePayloadPanel=e.codeBlockLanguagePayloadPanel;exports.createDisabledCommandStates=e.createDisabledCommandStates;exports.defaultMenuSchema=e.defaultMenuSchema;exports.defaultToolbarConfig=e.defaultToolbarConfig;exports.getCommandStateMatchValue=e.getCommandStateMatchValue;exports.isCommandStateForCommand=e.isCommandStateForCommand;exports.isMenuItemCommandState=e.isMenuItemCommandState;exports.resolveToolbarMenu=e.resolveToolbarMenu;exports.tableHeaderMenuItems=e.tableHeaderMenuItems;exports.createFloatingLayer=n.createFloatingLayer;exports.createRichTextEditor=n.createRichTextEditor;exports.createWebViewBridgeRuntime=n.createWebViewBridgeRuntime;exports.createRichTextToolbar=o.createRichTextToolbar;exports.hasMeaningfulHtmlContent=o.hasMeaningfulHtmlContent;exports.isEditorContentEmpty=r.isEditorContentEmpty;exports.resolvePayloadPanelSchema=i.resolvePayloadPanelSchema;
1
+ "use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const t=require("./index-GaS65GL0.cjs"),a=require("./bridge.cjs"),e=require("./native-spec.cjs"),n=require("./index-BBzHVrSi.cjs"),o=require("./index-DZnoOnKI.cjs"),r=require("./core.cjs"),i=require("./index-DF8OhKI4.cjs");exports.BRIDGERTE_CONTENT_VERSION=t.BRIDGERTE_CONTENT_VERSION;exports.BRIDGERTE_TABLE_INSERT_MAX_COLS=t.BRIDGERTE_TABLE_INSERT_MAX_COLS;exports.BRIDGERTE_TABLE_INSERT_MAX_ROWS=t.BRIDGERTE_TABLE_INSERT_MAX_ROWS;exports.BRIDGE_CONTENT_CHANGE_DEBOUNCE_MS=a.BRIDGE_CONTENT_CHANGE_DEBOUNCE_MS;exports.BRIDGE_HEIGHT_CHANGE_THROTTLE_MS=a.BRIDGE_HEIGHT_CHANGE_THROTTLE_MS;exports.defaultBridgeEventTiming=a.defaultBridgeEventTiming;exports.isBridgeMessage=a.isBridgeMessage;exports.codeBlockLanguagePanel=e.codeBlockLanguagePanel;exports.codeBlockLanguagePayloadPanel=e.codeBlockLanguagePayloadPanel;exports.createDisabledCommandStates=e.createDisabledCommandStates;exports.defaultMenuSchema=e.defaultMenuSchema;exports.defaultToolbarConfig=e.defaultToolbarConfig;exports.getCommandStateMatchValue=e.getCommandStateMatchValue;exports.isCommandStateForCommand=e.isCommandStateForCommand;exports.isMenuItemCommandState=e.isMenuItemCommandState;exports.resolveToolbarMenu=e.resolveToolbarMenu;exports.tableHeaderMenuItems=e.tableHeaderMenuItems;exports.createFloatingLayer=n.createFloatingLayer;exports.createRichTextEditor=n.createRichTextEditor;exports.createWebViewBridgeRuntime=n.createWebViewBridgeRuntime;exports.createRichTextToolbar=o.createRichTextToolbar;exports.hasMeaningfulHtmlContent=o.hasMeaningfulHtmlContent;exports.isEditorContentEmpty=r.isEditorContentEmpty;exports.resolvePayloadPanelSchema=i.resolvePayloadPanelSchema;
2
2
  //# sourceMappingURL=index.cjs.map
package/dist/index.js CHANGED
@@ -2,7 +2,7 @@ import { B as t, a as o, b as r } from "./index-CuNKUHed.js";
2
2
  import { BRIDGE_CONTENT_CHANGE_DEBOUNCE_MS as n, BRIDGE_HEIGHT_CHANGE_THROTTLE_MS as E, defaultBridgeEventTiming as l, isBridgeMessage as s } from "./bridge.js";
3
3
  import { codeBlockLanguagePanel as d, codeBlockLanguagePayloadPanel as i, createDisabledCommandStates as _, defaultMenuSchema as c, defaultToolbarConfig as R, getCommandStateMatchValue as B, isCommandStateForCommand as C, isMenuItemCommandState as g, resolveToolbarMenu as S, tableHeaderMenuItems as f } from "./native-spec.js";
4
4
  import { c as I, a as M, b as N } from "./index-DDfgjstx.js";
5
- import { c as b, h as p } from "./index-CQie9v_I.js";
5
+ import { c as b, h as p } from "./index-CXVSZZ93.js";
6
6
  import { isEditorContentEmpty as h } from "./core.js";
7
7
  import { r as H } from "./index-sbZNOcCB.js";
8
8
  export {
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "bridgerte",
3
- "version": "0.9.12",
3
+ "version": "0.9.13",
4
4
  "description": "Cross-platform rich text editor with WebView bridge support.",
5
5
  "type": "module",
6
6
  "main": "./dist/index.cjs",
@@ -1,296 +0,0 @@
1
- import { defaultMenuSchema as X, resolveToolbarMenu as Y } from "./native-spec.js";
2
- import { g as _, d as S, e as B, r as Z, f as K, h as ee, c as te } from "./index-DDfgjstx.js";
3
- const R = {
4
- toolbarGroup: `
5
- <svg
6
- aria-hidden="true"
7
- class="lucide lucide-ellipsis"
8
- xmlns="http://www.w3.org/2000/svg"
9
- width="18"
10
- height="18"
11
- viewBox="0 0 24 24"
12
- fill="none"
13
- stroke="currentColor"
14
- stroke-width="2"
15
- stroke-linecap="round"
16
- stroke-linejoin="round"
17
- >
18
- <circle cx="12" cy="12" r="1" />
19
- <circle cx="19" cy="12" r="1" />
20
- <circle cx="5" cy="12" r="1" />
21
- </svg>
22
- `,
23
- chevronDown: `
24
- <svg
25
- aria-hidden="true"
26
- class="lucide lucide-chevron-down"
27
- xmlns="http://www.w3.org/2000/svg"
28
- width="12"
29
- height="12"
30
- viewBox="0 0 24 24"
31
- fill="none"
32
- stroke="currentColor"
33
- stroke-width="2"
34
- stroke-linecap="round"
35
- stroke-linejoin="round"
36
- >
37
- <path d="m6 9 6 6 6-6" />
38
- </svg>
39
- `
40
- }, U = (e) => Array.from(
41
- e.querySelectorAll(".bridgerte__toolbar-group-menu-item")
42
- ).filter((t) => !t.disabled), O = (e, t) => {
43
- const i = U(e)[t];
44
- i && i.focus();
45
- }, oe = (e, t, s, i, d) => {
46
- var u, b;
47
- const r = U(t), n = document.activeElement instanceof HTMLButtonElement ? r.indexOf(document.activeElement) : -1, p = (y) => {
48
- var l;
49
- if (r.length === 0) return;
50
- const f = ((n >= 0 ? n : 0) + y + r.length) % r.length;
51
- (l = r[f]) == null || l.focus();
52
- };
53
- switch (e.key) {
54
- case "ArrowDown":
55
- e.preventDefault(), p(1);
56
- break;
57
- case "ArrowUp":
58
- e.preventDefault(), p(-1);
59
- break;
60
- case "Home":
61
- e.preventDefault(), (u = r[0]) == null || u.focus();
62
- break;
63
- case "End":
64
- e.preventDefault(), (b = r.at(-1)) == null || b.focus();
65
- break;
66
- case "Enter":
67
- case " ":
68
- document.activeElement instanceof HTMLButtonElement && (e.preventDefault(), i(document.activeElement), d(), s.focus());
69
- break;
70
- case "Escape":
71
- e.preventDefault(), s.focus(), d();
72
- break;
73
- }
74
- }, A = "button[data-bridgerte-toolbar-group-id]", re = "button[data-bridgerte-toolbar-item-id]", ne = [
75
- re,
76
- ".bridgerte__toolbar-group-menu-item"
77
- ].join(","), ae = (e, t) => {
78
- const s = e.map((i) => _(i, t));
79
- return {
80
- active: s.some((i) => i.active),
81
- disabled: s.length > 0 && s.every((i) => i.disabled)
82
- };
83
- }, se = (e, t, s, i, d) => {
84
- const r = _(t, s), n = document.createElement("button"), p = i[t.icon] ?? S[t.icon];
85
- n.type = "button", n.className = "bridgerte__toolbar-button", n.disabled = r.disabled, n.dataset.active = String(r.active), n.dataset.bridgerteToolbarItemId = t.id, n.setAttribute("aria-label", t.label), n.setAttribute("aria-pressed", String(r.active)), d && (n.dataset.tooltip = t.label), B(n, p, t.label), e.append(n);
86
- }, le = (e, t, s, i, d) => {
87
- const r = document.createElement("button"), n = ae(t.items, s), p = t.icon ? i[t.icon] ?? S[t.icon] : R.toolbarGroup, u = document.createElement("span");
88
- r.type = "button", r.className = "bridgerte__toolbar-button bridgerte__toolbar-group-button", r.disabled = n.disabled, r.dataset.active = String(n.active), r.dataset.bridgerteToolbarGroupId = t.key, r.dataset.open = "false", r.setAttribute("aria-label", t.title), r.setAttribute("aria-haspopup", "menu"), r.setAttribute("aria-expanded", "false"), r.setAttribute("aria-pressed", String(n.active)), d && (r.dataset.tooltip = t.title), B(r, p, t.title), u.className = "bridgerte__toolbar-group-indicator", u.setAttribute("aria-hidden", "true"), u.innerHTML = R.chevronDown, r.append(u), e.append(r);
89
- }, ie = (e, t, s, i, d) => {
90
- e.textContent = "", t.forEach((r) => {
91
- if (r.type === "separator") {
92
- const p = document.createElement("span");
93
- p.className = "bridgerte__toolbar-separator", p.dataset.separatorId = r.key, p.setAttribute("aria-hidden", "true"), e.append(p);
94
- return;
95
- }
96
- if (r.type === "button") {
97
- se(
98
- e,
99
- r.item,
100
- s,
101
- i,
102
- d
103
- );
104
- return;
105
- }
106
- const n = document.createElement("div");
107
- n.className = "bridgerte__toolbar-group", n.dataset.group = r.key, n.setAttribute("aria-label", r.title), e.append(n), le(
108
- n,
109
- r,
110
- s,
111
- i,
112
- d
113
- );
114
- });
115
- }, x = (e, t, s, i) => {
116
- if (e.textContent = "", !t) {
117
- e.dataset.visible = "false";
118
- return;
119
- }
120
- t.items.forEach((d) => {
121
- const r = _(d, s), n = document.createElement("button"), p = i[d.icon] ?? S[d.icon], u = document.createElement("span");
122
- n.type = "button", n.className = "bridgerte__menu-item bridgerte__toolbar-group-menu-item", n.disabled = r.disabled, n.dataset.active = String(r.active), n.dataset.bridgerteToolbarItemId = d.id, n.setAttribute("role", "menuitem"), n.setAttribute("aria-label", d.label), n.setAttribute("aria-pressed", String(r.active)), B(n, p, d.label), u.className = "bridgerte__toolbar-group-menu-label", u.textContent = d.label, n.append(u), e.append(n);
123
- }), e.dataset.visible = "true", e.style.minWidth = `${t.button.offsetWidth}px`;
124
- }, k = (e, t) => {
125
- e.querySelectorAll(A).forEach((s) => {
126
- const i = (t == null ? void 0 : t.groupKey) === s.dataset.bridgerteToolbarGroupId;
127
- s.dataset.open = String(i), s.setAttribute("aria-expanded", String(i));
128
- });
129
- }, F = (e, t) => e.find((s) => s.type === "group" && s.key === t), ce = 8, de = 6, ue = () => {
130
- var e;
131
- return typeof window < "u" && ((e = window.matchMedia) == null ? void 0 : e.call(window, "(hover: hover) and (pointer: fine)").matches) === !0;
132
- }, w = (e) => {
133
- const t = e instanceof Element ? e.closest(ne) : null;
134
- return t instanceof HTMLButtonElement ? t : null;
135
- }, j = (e) => {
136
- const t = e instanceof Element ? e.closest(A) : null;
137
- return t instanceof HTMLButtonElement ? t : null;
138
- };
139
- function ge(e, t) {
140
- const s = t.placement ?? "top", i = Z(t.menuSchema ?? X, {
141
- menuLabels: t.menuLabels,
142
- payloadPanelConfig: t.payloadPanelConfig
143
- }), d = Y(t.toolbarConfig, i), r = d.flatMap((o) => o.type === "button" ? [o.item] : o.type === "group" ? o.items : []), n = t.icons ?? {}, p = ue(), u = document.createElement("div"), b = document.createElement("div"), y = e.closest(".bridgerte") ?? e;
144
- let v = !1, f = t.editor.getCommandStates(), l = null, m = null;
145
- const $ = K(e, {
146
- targetSelector: [
147
- "button[data-bridgerte-toolbar-item-id]",
148
- "button[data-bridgerte-toolbar-group-id]"
149
- ].join(",")
150
- }), V = K(b, {
151
- targetSelector: ".bridgerte__toolbar-group-menu-item"
152
- }), C = () => {
153
- y.append(u), y.append(b);
154
- }, T = () => {
155
- m == null || m.setOpen(!1), m == null || m.destroy(), m = null;
156
- }, L = () => {
157
- l && (T(), m = te(l.button, b, {
158
- placement: s === "bottom" ? "top-start" : "bottom-start",
159
- offset: de,
160
- strategy: "fixed"
161
- }), m.setOpen(!0));
162
- }, g = () => {
163
- T(), l = null, x(b, l, f, n), k(e, l);
164
- }, G = (o) => {
165
- if (!v && (f = o, l && T(), ie(e, d, o, n, p), C(), l)) {
166
- const a = Array.from(
167
- e.querySelectorAll(A)
168
- ).find((Q) => Q.dataset.bridgerteToolbarGroupId === (l == null ? void 0 : l.groupKey)), c = F(d, l.groupKey);
169
- l = a && c ? {
170
- groupKey: l.groupKey,
171
- button: a,
172
- items: c.items
173
- } : null, x(b, l, f, n), k(e, l), l && L();
174
- }
175
- }, W = () => {
176
- v || G(t.editor.getCommandStates());
177
- }, h = () => {
178
- u.dataset.visible = "false", u.textContent = "";
179
- }, z = (o) => {
180
- const a = o.dataset.tooltip;
181
- if (!p || !a) return;
182
- const c = o.getBoundingClientRect();
183
- u.textContent = a, u.dataset.visible = "true", u.style.left = `${c.left + c.width / 2}px`, u.style.top = `${c.top - ce}px`;
184
- }, D = (o) => {
185
- const a = w(o.target);
186
- a && z(a);
187
- }, M = (o) => {
188
- const a = o.relatedTarget, c = w(o.target);
189
- c && a instanceof Node && c.contains(a) || h();
190
- }, P = (o) => {
191
- if (!(o instanceof HTMLButtonElement) || o.disabled) return;
192
- const a = r.find((c) => c.id === o.dataset.bridgerteToolbarItemId);
193
- if (a) {
194
- if (a.payloadPanel) {
195
- const c = o.getBoundingClientRect();
196
- t.editor.requestPayloadPanel({
197
- menuId: a.id,
198
- command: a.command,
199
- panel: a.payloadPanel,
200
- currentValues: ee(a, t.editor.getCommandStates()),
201
- anchorRect: {
202
- x: c.left,
203
- y: c.top,
204
- width: c.width,
205
- height: c.height
206
- }
207
- });
208
- return;
209
- }
210
- t.editor.executeCommand(a.command);
211
- }
212
- }, H = (o, a = !1) => {
213
- const c = F(
214
- d,
215
- o.dataset.bridgerteToolbarGroupId
216
- );
217
- if (!(!c || o.disabled)) {
218
- if ((l == null ? void 0 : l.groupKey) === c.key) {
219
- if (a) {
220
- O(b, 0);
221
- return;
222
- }
223
- g();
224
- return;
225
- }
226
- l = {
227
- groupKey: c.key,
228
- button: o,
229
- items: c.items
230
- }, h(), x(b, l, f, n), k(e, l), L(), a && O(b, 0);
231
- }
232
- }, E = (o) => {
233
- const a = j(o.target);
234
- if (a) {
235
- o.preventDefault(), o.stopPropagation(), H(a);
236
- return;
237
- }
238
- const c = w(o.target);
239
- c && (o.preventDefault(), o.stopPropagation(), P(c), g());
240
- }, I = (o) => {
241
- const a = o.target;
242
- a instanceof Node && (e.contains(a) || b.contains(a)) || g();
243
- }, N = (o) => {
244
- if (l && b.contains(document.activeElement)) {
245
- oe(
246
- o,
247
- b,
248
- l.button,
249
- P,
250
- g
251
- );
252
- return;
253
- }
254
- const a = j(o.target);
255
- if (a && (o.key === "Enter" || o.key === " " || o.key === "ArrowDown")) {
256
- o.preventDefault(), o.stopPropagation(), H(a, !0);
257
- return;
258
- }
259
- o.key === "Escape" && g();
260
- };
261
- e.classList.add("bridgerte__toolbar"), u.className = "bridgerte__toolbar-tooltip", u.dataset.visible = "false", b.className = "bridgerte__floating-menu bridgerte__toolbar-group-menu", b.dataset.visible = "false", b.setAttribute("role", "menu"), e.dataset.placement = s, e.setAttribute("role", "toolbar"), e.setAttribute(
262
- "aria-label",
263
- s === "bottom" ? "BridgeRTE tabbar" : "BridgeRTE toolbar"
264
- ), e.addEventListener("click", E), b.addEventListener("click", E), document.addEventListener("click", I), document.addEventListener("keydown", N), p && (e.addEventListener("mouseover", D), e.addEventListener("mouseout", M)), e.addEventListener("focusout", h), C();
265
- const J = t.editor.subscribeCommandStateChange(G);
266
- return {
267
- update: W,
268
- destroy() {
269
- v || (v = !0, J(), g(), e.removeEventListener("click", E), b.removeEventListener("click", E), document.removeEventListener("click", I), document.removeEventListener("keydown", N), p && (e.removeEventListener("mouseover", D), e.removeEventListener("mouseout", M)), $(), V(), e.removeEventListener("focusout", h), u.remove(), b.remove(), e.classList.remove("bridgerte__toolbar"), delete e.dataset.placement, e.textContent = "", e.removeAttribute("role"), e.removeAttribute("aria-label"));
270
- }
271
- };
272
- }
273
- const be = /[\u200B-\u200D\uFEFF]/g, pe = [
274
- "img",
275
- "video",
276
- "audio",
277
- "iframe",
278
- "table",
279
- "pre",
280
- "code",
281
- "hr",
282
- "figure",
283
- "canvas",
284
- "svg",
285
- "math",
286
- '[data-type="mention"]'
287
- ].join(","), q = (e) => (e == null ? void 0 : e.replace(be, "").trim()) ?? "", ve = (e) => {
288
- if (!q(e)) return !1;
289
- const t = document.createElement("template");
290
- return t.innerHTML = e, !!(q(t.content.textContent) || t.content.querySelector(pe));
291
- };
292
- export {
293
- ge as c,
294
- ve as h
295
- };
296
- //# sourceMappingURL=index-CQie9v_I.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"index-CQie9v_I.js","sources":["../../dom/src/menuIcon/uiIcons.ts","../../dom/src/richTextToolbar/groupMenuKeyboard.ts","../../dom/src/richTextToolbar/render.ts","../../dom/src/richTextToolbar/index.ts","../../dom/src/htmlContent/index.ts"],"sourcesContent":["/**\r\n * DOM UI chrome icon 表。\r\n *\r\n * 这里放 toolbar 收纳入口、下拉箭头这类控件自身图标。它们不属于 `MenuItem.icon`\r\n * 跨端 schema,也不应该进入 `defaultMenuIcons` 的 schema 对齐检查。\r\n */\r\nexport const defaultMenuUiIcons = {\r\n toolbarGroup: `\r\n <svg\r\n aria-hidden=\"true\"\r\n class=\"lucide lucide-ellipsis\"\r\n xmlns=\"http://www.w3.org/2000/svg\"\r\n width=\"18\"\r\n height=\"18\"\r\n viewBox=\"0 0 24 24\"\r\n fill=\"none\"\r\n stroke=\"currentColor\"\r\n stroke-width=\"2\"\r\n stroke-linecap=\"round\"\r\n stroke-linejoin=\"round\"\r\n >\r\n <circle cx=\"12\" cy=\"12\" r=\"1\" />\r\n <circle cx=\"19\" cy=\"12\" r=\"1\" />\r\n <circle cx=\"5\" cy=\"12\" r=\"1\" />\r\n </svg>\r\n `,\r\n chevronDown: `\r\n <svg\r\n aria-hidden=\"true\"\r\n class=\"lucide lucide-chevron-down\"\r\n xmlns=\"http://www.w3.org/2000/svg\"\r\n width=\"12\"\r\n height=\"12\"\r\n viewBox=\"0 0 24 24\"\r\n fill=\"none\"\r\n stroke=\"currentColor\"\r\n stroke-width=\"2\"\r\n stroke-linecap=\"round\"\r\n stroke-linejoin=\"round\"\r\n >\r\n <path d=\"m6 9 6 6 6-6\" />\r\n </svg>\r\n `\r\n} as const;\r\n","export const getEnabledGroupMenuButtons = (menuElement: HTMLElement) => (\r\n Array.from(\r\n menuElement.querySelectorAll<HTMLButtonElement>('.bridgerte__toolbar-group-menu-item')\r\n ).filter((button) => !button.disabled)\r\n);\r\n\r\nexport const focusToolbarGroupMenuItem = (menuElement: HTMLElement, index: number) => {\r\n const buttons = getEnabledGroupMenuButtons(menuElement);\r\n const nextButton = buttons[index];\r\n\r\n if (nextButton) nextButton.focus();\r\n};\r\n\r\nexport const handleToolbarGroupMenuKeyDown = (\n event: KeyboardEvent,\r\n menuElement: HTMLElement,\r\n returnButton: HTMLButtonElement,\r\n executeToolbarButton: (button: HTMLButtonElement) => void,\r\n closeGroupMenu: () => void\r\n) => {\r\n const buttons = getEnabledGroupMenuButtons(menuElement);\r\n const activeIndex = document.activeElement instanceof HTMLButtonElement\r\n ? buttons.indexOf(document.activeElement)\r\n : -1;\r\n const focusByOffset = (offset: number) => {\r\n if (buttons.length === 0) return;\r\n\r\n const baseIndex = activeIndex >= 0 ? activeIndex : 0;\r\n const nextIndex = (baseIndex + offset + buttons.length) % buttons.length;\r\n\r\n buttons[nextIndex]?.focus();\r\n };\r\n\r\n /*\r\n * ARIA menu 语义要求支持 roving focus。这里不改 toolbar 的横向键盘模型,\r\n * 只在已打开的纵向收纳菜单内处理上下方向、首尾跳转和执行/关闭。\r\n */\r\n switch (event.key) {\r\n case 'ArrowDown':\r\n event.preventDefault();\r\n focusByOffset(1);\r\n break;\r\n case 'ArrowUp':\r\n event.preventDefault();\r\n focusByOffset(-1);\r\n break;\r\n case 'Home':\r\n event.preventDefault();\r\n buttons[0]?.focus();\r\n break;\r\n case 'End':\r\n event.preventDefault();\r\n buttons.at(-1)?.focus();\r\n break;\r\n case 'Enter':\r\n case ' ':\r\n if (document.activeElement instanceof HTMLButtonElement) {\r\n event.preventDefault();\r\n executeToolbarButton(document.activeElement);\r\n closeGroupMenu();\r\n returnButton.focus();\r\n }\r\n break;\r\n case 'Escape':\r\n event.preventDefault();\r\n returnButton.focus();\r\n closeGroupMenu();\r\n break;\r\n default:\r\n break;\r\n }\r\n};\r\n","import type { CommandState } from '@bridgerte/core';\r\nimport type { MenuItem, ResolvedToolbarItem } from '@bridgerte/native-spec';\r\nimport {\r\n appendMenuIcon,\r\n defaultMenuIcons,\r\n defaultMenuUiIcons\r\n} from './icons';\r\nimport { getMenuStateForItem } from '../menuRuntime';\r\nimport type {\r\n RichTextToolbarIcons,\r\n ToolbarGroupMenuState\r\n} from './type';\r\n\r\nexport const toolbarGroupButtonSelector = 'button[data-bridgerte-toolbar-group-id]';\r\nexport const toolbarButtonSelector = 'button[data-bridgerte-toolbar-item-id]';\r\n// 可执行按钮同时覆盖 toolbar 主按钮和 group menu 子项,避免浮层子菜单绕过统一命令入口。\r\nexport const toolbarExecutableButtonSelector = [\n toolbarButtonSelector,\n '.bridgerte__toolbar-group-menu-item'\n].join(',');\n\r\nconst getGroupMenuState = (items: MenuItem[], commandStates: CommandState[]) => {\r\n const itemStates = items.map((item) => getMenuStateForItem(item, commandStates));\r\n\r\n return {\r\n active: itemStates.some((state) => state.active),\r\n disabled: itemStates.length > 0 && itemStates.every((state) => state.disabled)\r\n };\r\n};\r\n\r\nconst renderToolbarButton = (\r\n groupElement: HTMLElement,\r\n item: MenuItem,\r\n commandStates: CommandState[],\r\n icons: RichTextToolbarIcons,\r\n enableTooltip: boolean\r\n) => {\r\n const state = getMenuStateForItem(item, commandStates);\r\n const button = document.createElement('button');\r\n // icon 兜底顺序固定为:业务覆盖 > DOM 默认 SVG > label 文本。\r\n const iconSvg = icons[item.icon] ?? defaultMenuIcons[item.icon];\r\n\r\n button.type = 'button';\r\n button.className = 'bridgerte__toolbar-button';\r\n button.disabled = state.disabled;\r\n button.dataset.active = String(state.active);\r\n button.dataset.bridgerteToolbarItemId = item.id;\r\n button.setAttribute('aria-label', item.label);\r\n button.setAttribute('aria-pressed', String(state.active));\r\n if (enableTooltip) button.dataset.tooltip = item.label;\r\n\r\n appendMenuIcon(button, iconSvg, item.label);\r\n\r\n groupElement.append(button);\r\n};\r\n\r\nconst renderToolbarGroupButton = (\r\n groupElement: HTMLElement,\r\n toolbarItem: Extract<ResolvedToolbarItem, { type: 'group' }>,\r\n commandStates: CommandState[],\r\n icons: RichTextToolbarIcons,\r\n enableTooltip: boolean\r\n) => {\r\n const button = document.createElement('button');\r\n const groupState = getGroupMenuState(toolbarItem.items, commandStates);\r\n const iconSvg = toolbarItem.icon\r\n ? icons[toolbarItem.icon] ?? defaultMenuIcons[toolbarItem.icon]\r\n : defaultMenuUiIcons.toolbarGroup;\r\n const indicator = document.createElement('span');\r\n\r\n button.type = 'button';\r\n button.className = 'bridgerte__toolbar-button bridgerte__toolbar-group-button';\r\n button.disabled = groupState.disabled;\r\n button.dataset.active = String(groupState.active);\r\n button.dataset.bridgerteToolbarGroupId = toolbarItem.key;\r\n button.dataset.open = 'false';\r\n button.setAttribute('aria-label', toolbarItem.title);\r\n button.setAttribute('aria-haspopup', 'menu');\r\n button.setAttribute('aria-expanded', 'false');\r\n button.setAttribute('aria-pressed', String(groupState.active));\r\n if (enableTooltip) button.dataset.tooltip = toolbarItem.title;\r\n\r\n appendMenuIcon(button, iconSvg, toolbarItem.title);\r\n\r\n indicator.className = 'bridgerte__toolbar-group-indicator';\r\n indicator.setAttribute('aria-hidden', 'true');\r\n indicator.innerHTML = defaultMenuUiIcons.chevronDown;\r\n button.append(indicator);\r\n groupElement.append(button);\r\n};\r\n\r\n/**\r\n * 渲染 toolbar 横向主入口。\n *\n * 字符串菜单直接渲染为按钮,`|` 渲染分割线。\n * 只有用户显式声明的 group 配置才生成收纳入口,避免 schema 隐式改变 DOM 结构。\n */\r\nexport const renderToolbar = (\r\n toolbarElement: HTMLElement,\r\n toolbarItems: ResolvedToolbarItem[],\r\n commandStates: CommandState[],\r\n icons: RichTextToolbarIcons,\r\n enableTooltip: boolean\r\n) => {\r\n toolbarElement.textContent = '';\r\n\r\n toolbarItems.forEach((toolbarItem) => {\n if (toolbarItem.type === 'separator') {\n const separatorElement = document.createElement('span');\n\n separatorElement.className = 'bridgerte__toolbar-separator';\n separatorElement.dataset.separatorId = toolbarItem.key;\r\n separatorElement.setAttribute('aria-hidden', 'true');\r\n toolbarElement.append(separatorElement);\r\n return;\n }\n\n if (toolbarItem.type === 'button') {\n renderToolbarButton(\n toolbarElement,\n toolbarItem.item,\n commandStates,\n icons,\n enableTooltip\n );\n return;\n }\n\n /*\n * 收纳菜单是用户在 toolbarConfig 里显式声明的结构;普通菜单不会自动生成 DOM 包裹,\n * 显示分组完全交给用户用 group 配置或 `|` 控制。\n */\n const groupElement = document.createElement('div');\n\n groupElement.className = 'bridgerte__toolbar-group';\n groupElement.dataset.group = toolbarItem.key;\n groupElement.setAttribute('aria-label', toolbarItem.title);\n toolbarElement.append(groupElement);\n\n renderToolbarGroupButton(\n groupElement,\n toolbarItem,\n commandStates,\n icons,\n enableTooltip\r\n );\r\n });\r\n};\r\n\r\n/**\r\n * 渲染 group button 打开的纵向收纳菜单。\r\n *\r\n * 浮层只展示 MenuItem 子项并复用现有 item id,点击执行仍由 index.ts 统一走 EditorAPI。\r\n */\r\nexport const renderToolbarGroupMenu = (\r\n menuElement: HTMLElement,\r\n groupMenuState: ToolbarGroupMenuState | null,\r\n commandStates: CommandState[],\r\n icons: RichTextToolbarIcons\r\n) => {\r\n menuElement.textContent = '';\r\n\r\n if (!groupMenuState) {\r\n menuElement.dataset.visible = 'false';\r\n return;\r\n }\r\n\r\n groupMenuState.items.forEach((item) => {\r\n const state = getMenuStateForItem(item, commandStates);\r\n const button = document.createElement('button');\r\n const iconSvg = icons[item.icon] ?? defaultMenuIcons[item.icon];\r\n const labelElement = document.createElement('span');\n\n button.type = 'button';\n button.className = 'bridgerte__menu-item bridgerte__toolbar-group-menu-item';\n button.disabled = state.disabled;\r\n button.dataset.active = String(state.active);\r\n button.dataset.bridgerteToolbarItemId = item.id;\r\n button.setAttribute('role', 'menuitem');\r\n button.setAttribute('aria-label', item.label);\r\n button.setAttribute('aria-pressed', String(state.active));\r\n\r\n appendMenuIcon(button, iconSvg, item.label);\r\n labelElement.className = 'bridgerte__toolbar-group-menu-label';\r\n labelElement.textContent = item.label;\r\n button.append(labelElement);\r\n menuElement.append(button);\r\n });\r\n\r\n menuElement.dataset.visible = 'true';\r\n menuElement.style.minWidth = `${groupMenuState.button.offsetWidth}px`;\r\n};\r\n\r\nexport const syncToolbarGroupButtonState = (\r\n container: HTMLElement,\r\n groupMenuState: ToolbarGroupMenuState | null\r\n) => {\r\n container.querySelectorAll<HTMLButtonElement>(toolbarGroupButtonSelector).forEach((button) => {\r\n const open = groupMenuState?.groupKey === button.dataset.bridgerteToolbarGroupId;\r\n\r\n button.dataset.open = String(open);\r\n button.setAttribute('aria-expanded', String(open));\r\n });\r\n};\r\n\r\nexport const findToolbarGroupMenuItem = (\r\n toolbarItems: ResolvedToolbarItem[],\r\n groupKey: string | undefined\r\n) => toolbarItems.find((item): item is Extract<ResolvedToolbarItem, { type: 'group' }> => (\r\n item.type === 'group' && item.key === groupKey\r\n));\r\n","import type { CommandState } from '@bridgerte/core';\r\nimport {\r\n defaultMenuSchema,\r\n resolveToolbarMenu\r\n} from '@bridgerte/native-spec';\r\nimport { createFloatingLayer, type RichTextFloatingLayer } from '../floatingLayer';\r\nimport { bindTouchPressedState } from '../interactionState';\r\nimport {\r\n getPayloadPanelCurrentValues,\r\n resolveMenuSchemaForDom\r\n} from '../menuRuntime';\r\nimport {\r\n focusToolbarGroupMenuItem,\r\n handleToolbarGroupMenuKeyDown\r\n} from './groupMenuKeyboard';\r\nimport {\r\n findToolbarGroupMenuItem,\r\n renderToolbar,\r\n renderToolbarGroupMenu,\r\n syncToolbarGroupButtonState,\r\n toolbarExecutableButtonSelector,\r\n toolbarGroupButtonSelector\r\n} from './render';\r\nimport type {\r\n RichTextToolbarAPI,\r\n ToolbarGroupMenuState,\r\n RichTextToolbarOptions\r\n} from './type';\r\n\r\nconst toolbarTooltipOffsetPx = 8;\r\nconst toolbarGroupMenuOffsetPx = 6;\r\n\r\nexport type * from './type';\r\n\r\nconst canUseHoverTooltip = () => (\r\n typeof window !== 'undefined'\r\n && window.matchMedia?.('(hover: hover) and (pointer: fine)').matches === true\r\n);\r\n\r\nconst getToolbarButtonFromTarget = (target: EventTarget | null) => {\r\n /*\r\n * 图标覆盖常用 SVG,H5 pointer 事件可能落在 svg/path 上。\r\n * 这里用 Element 而不是 HTMLElement,保证触屏兜底和 click 都能向上命中真实按钮。\r\n */\r\n const button = target instanceof Element\r\n ? target.closest<HTMLButtonElement>(toolbarExecutableButtonSelector)\r\n : null;\r\n\r\n return button instanceof HTMLButtonElement ? button : null;\r\n};\r\n\r\nconst getToolbarGroupButtonFromTarget = (target: EventTarget | null) => {\r\n const button = target instanceof Element\r\n ? target.closest<HTMLButtonElement>(toolbarGroupButtonSelector)\r\n : null;\r\n\r\n return button instanceof HTMLButtonElement ? button : null;\r\n};\r\n\r\n/**\r\n * 创建独立菜单实例。\r\n *\r\n * toolbar/tabbar 只订阅 EditorAPI 状态并派发命令,不接触编辑器内部 DOM 或 Lexical 实例。\r\n */\r\nexport function createRichTextToolbar(\r\n container: HTMLElement,\r\n options: RichTextToolbarOptions\r\n): RichTextToolbarAPI {\r\n const placement = options.placement ?? 'top';\r\n const menuSchema = resolveMenuSchemaForDom(options.menuSchema ?? defaultMenuSchema, {\r\n menuLabels: options.menuLabels,\r\n payloadPanelConfig: options.payloadPanelConfig\r\n });\r\n const toolbarItems = resolveToolbarMenu(options.toolbarConfig, menuSchema);\r\n /*\r\n * click 事件只需要能执行命令的菜单项。分割线和 group 容器是渲染结构,\r\n * 先在这里摊平,后续点击时就不用理解 toolbarConfig 的展示层级。\r\n */\r\n const executableMenuItems = toolbarItems.flatMap((toolbarItem) => (\r\n toolbarItem.type === 'button' ? [toolbarItem.item]\r\n : toolbarItem.type === 'group' ? toolbarItem.items\r\n : []\r\n ));\r\n const icons = options.icons ?? {};\r\n /*\r\n * tooltip 只属于 PC 精细指针体验。H5 上即使没有原生 title,部分浏览器也会把 hover/mouseover\r\n * 模拟成首触摸状态,导致第一次点像是只唤醒提示;所以触屏端不写 tooltip 数据也不挂监听。\r\n */\r\n const enableTooltip = canUseHoverTooltip();\r\n const tooltipElement = document.createElement('div');\r\n const groupMenuElement = document.createElement('div');\r\n const overlayHost = container.closest('.bridgerte') ?? container;\r\n let destroyed = false;\r\n let latestCommandStates = options.editor.getCommandStates();\r\n let groupMenuState: ToolbarGroupMenuState | null = null;\r\n let groupMenuFloatingLayer: RichTextFloatingLayer | null = null;\r\n const clearToolbarPressedState = bindTouchPressedState(container, {\r\n targetSelector: [\r\n 'button[data-bridgerte-toolbar-item-id]',\r\n 'button[data-bridgerte-toolbar-group-id]'\r\n ].join(',')\r\n });\r\n const clearGroupMenuPressedState = bindTouchPressedState(groupMenuElement, {\r\n targetSelector: '.bridgerte__toolbar-group-menu-item'\r\n });\r\n\r\n const mountToolbarOverlays = () => {\r\n // 独立 toolbar 可能不在 `.bridgerte` 内,渲染按钮会清空 container,需要把浮层重新挂回去。\r\n overlayHost.append(tooltipElement);\r\n overlayHost.append(groupMenuElement);\r\n };\r\n\r\n const closeGroupMenuFloatingLayer = () => {\r\n groupMenuFloatingLayer?.setOpen(false);\r\n groupMenuFloatingLayer?.destroy();\r\n groupMenuFloatingLayer = null;\r\n };\r\n\r\n const openGroupMenuFloatingLayer = () => {\r\n if (!groupMenuState) return;\r\n\r\n closeGroupMenuFloatingLayer();\r\n /*\r\n * group menu 和 hoverbar/mention/slash 一样属于轻浮层,必须走 floatingLayer。\r\n * 这里按 toolbar placement 给出首选方向,真正的翻转、键盘可视区和左右夹紧交给\r\n * createFloatingLayer 内部的 visualViewport + flip + shift + clamp 统一处理。\r\n */\r\n groupMenuFloatingLayer = createFloatingLayer(groupMenuState.button, groupMenuElement, {\r\n placement: placement === 'bottom' ? 'top-start' : 'bottom-start',\r\n offset: toolbarGroupMenuOffsetPx,\r\n strategy: 'fixed'\r\n });\r\n groupMenuFloatingLayer.setOpen(true);\r\n };\r\n\r\n const closeGroupMenu = () => {\r\n closeGroupMenuFloatingLayer();\r\n groupMenuState = null;\r\n renderToolbarGroupMenu(groupMenuElement, groupMenuState, latestCommandStates, icons);\r\n syncToolbarGroupButtonState(container, groupMenuState);\r\n };\r\n\r\n const renderStates = (states: CommandState[]) => {\r\n if (destroyed) return;\r\n\r\n latestCommandStates = states;\r\n if (groupMenuState) closeGroupMenuFloatingLayer();\r\n renderToolbar(container, toolbarItems, states, icons, enableTooltip);\r\n mountToolbarOverlays();\r\n if (groupMenuState) {\r\n const nextButton = Array.from(\r\n container.querySelectorAll<HTMLButtonElement>(toolbarGroupButtonSelector)\r\n ).find((button) => button.dataset.bridgerteToolbarGroupId === groupMenuState?.groupKey);\r\n const nextGroupItem = findToolbarGroupMenuItem(toolbarItems, groupMenuState.groupKey);\r\n\r\n groupMenuState = nextButton && nextGroupItem\r\n ? {\r\n groupKey: groupMenuState.groupKey,\r\n button: nextButton,\r\n items: nextGroupItem.items\r\n }\r\n : null;\r\n renderToolbarGroupMenu(groupMenuElement, groupMenuState, latestCommandStates, icons);\r\n syncToolbarGroupButtonState(container, groupMenuState);\r\n if (groupMenuState) openGroupMenuFloatingLayer();\r\n }\r\n };\r\n\r\n const update = () => {\r\n if (destroyed) return;\r\n\r\n renderStates(options.editor.getCommandStates());\r\n };\r\n\r\n const hideTooltip = () => {\r\n tooltipElement.dataset.visible = 'false';\r\n tooltipElement.textContent = '';\r\n };\r\n\r\n const showTooltip = (button: HTMLButtonElement) => {\r\n const tooltipText = button.dataset.tooltip;\r\n\r\n if (!enableTooltip || !tooltipText) return;\r\n\r\n const buttonRect = button.getBoundingClientRect();\r\n\r\n tooltipElement.textContent = tooltipText;\r\n tooltipElement.dataset.visible = 'true';\r\n tooltipElement.style.left = `${buttonRect.left + buttonRect.width / 2}px`;\r\n tooltipElement.style.top = `${buttonRect.top - toolbarTooltipOffsetPx}px`;\r\n };\r\n\r\n const handleTooltipTarget = (event: Event) => {\r\n const button = getToolbarButtonFromTarget(event.target);\r\n\r\n if (button) {\r\n showTooltip(button);\r\n }\r\n };\r\n\r\n const handleTooltipLeave = (event: MouseEvent) => {\r\n const relatedTarget = event.relatedTarget;\r\n const button = getToolbarButtonFromTarget(event.target);\r\n\r\n if (\r\n button\r\n && relatedTarget instanceof Node\r\n && button.contains(relatedTarget)\r\n ) return;\r\n\r\n hideTooltip();\r\n };\r\n\r\n const executeToolbarButton = (button: HTMLButtonElement) => {\r\n if (!(button instanceof HTMLButtonElement) || button.disabled) return;\r\n\r\n const menuItem = executableMenuItems.find((item) => (\r\n item.id === button.dataset.bridgerteToolbarItemId\r\n ));\r\n\r\n if (!menuItem) return;\r\n\r\n if (menuItem.payloadPanel) {\r\n const buttonRect = button.getBoundingClientRect();\r\n\r\n /*\r\n * 带 payloadPanel 的菜单不直接执行基础 command,而是发起参数请求。\r\n * DOM 内置面板和业务/RN/Flutter 自绘都走同一个 request,避免后续颜色、\r\n * 字体、链接等参数菜单各自发明一套协议。\r\n */\r\n options.editor.requestPayloadPanel({\r\n menuId: menuItem.id,\r\n command: menuItem.command,\r\n panel: menuItem.payloadPanel,\r\n currentValues: getPayloadPanelCurrentValues(menuItem, options.editor.getCommandStates()),\r\n anchorRect: {\r\n x: buttonRect.left,\r\n y: buttonRect.top,\r\n width: buttonRect.width,\r\n height: buttonRect.height\r\n }\r\n });\r\n return;\r\n }\r\n\r\n options.editor.executeCommand(menuItem.command);\r\n };\r\n\r\n const toggleGroupMenu = (button: HTMLButtonElement, shouldFocusMenu = false) => {\r\n const groupItem = findToolbarGroupMenuItem(\r\n toolbarItems,\r\n button.dataset.bridgerteToolbarGroupId\r\n );\r\n\r\n if (!groupItem || button.disabled) return;\r\n\r\n if (groupMenuState?.groupKey === groupItem.key) {\r\n if (shouldFocusMenu) {\r\n focusToolbarGroupMenuItem(groupMenuElement, 0);\r\n return;\r\n }\r\n closeGroupMenu();\r\n return;\r\n }\r\n\r\n groupMenuState = {\r\n groupKey: groupItem.key,\r\n button,\r\n items: groupItem.items\r\n };\r\n hideTooltip();\r\n renderToolbarGroupMenu(groupMenuElement, groupMenuState, latestCommandStates, icons);\r\n syncToolbarGroupButtonState(container, groupMenuState);\r\n openGroupMenuFloatingLayer();\r\n if (shouldFocusMenu) focusToolbarGroupMenuItem(groupMenuElement, 0);\r\n };\r\n\r\n const handleClick = (event: MouseEvent) => {\r\n const groupButton = getToolbarGroupButtonFromTarget(event.target);\r\n if (groupButton) {\r\n event.preventDefault();\r\n event.stopPropagation();\r\n toggleGroupMenu(groupButton);\r\n return;\r\n }\r\n\r\n const button = getToolbarButtonFromTarget(event.target);\r\n if (!button) return;\r\n\r\n event.preventDefault();\r\n event.stopPropagation();\r\n executeToolbarButton(button);\r\n closeGroupMenu();\r\n };\r\n\r\n const handleDocumentClick = (event: MouseEvent) => {\r\n const target = event.target;\r\n\r\n if (\r\n target instanceof Node\r\n && (container.contains(target) || groupMenuElement.contains(target))\r\n ) return;\r\n\r\n closeGroupMenu();\r\n };\r\n\r\n const handleKeyDown = (event: KeyboardEvent) => {\r\n if (groupMenuState && groupMenuElement.contains(document.activeElement)) {\r\n handleToolbarGroupMenuKeyDown(\r\n event,\r\n groupMenuElement,\r\n groupMenuState.button,\r\n executeToolbarButton,\r\n closeGroupMenu\r\n );\r\n return;\r\n }\r\n\r\n const groupButton = getToolbarGroupButtonFromTarget(event.target);\r\n if (\r\n groupButton\r\n && (event.key === 'Enter' || event.key === ' ' || event.key === 'ArrowDown')\r\n ) {\r\n /*\r\n * 指针点击收纳菜单不能抢编辑器焦点;键盘从入口按钮打开时则进入菜单项,\r\n * 保留 PC 可访问性和方向键 roving focus。\r\n */\r\n event.preventDefault();\r\n event.stopPropagation();\r\n toggleGroupMenu(groupButton, true);\r\n return;\r\n }\r\n\r\n if (event.key !== 'Escape') return;\r\n\r\n closeGroupMenu();\r\n };\r\n\r\n container.classList.add('bridgerte__toolbar');\r\n tooltipElement.className = 'bridgerte__toolbar-tooltip';\r\n tooltipElement.dataset.visible = 'false';\r\n groupMenuElement.className = 'bridgerte__floating-menu bridgerte__toolbar-group-menu';\r\n groupMenuElement.dataset.visible = 'false';\r\n groupMenuElement.setAttribute('role', 'menu');\r\n container.dataset.placement = placement;\r\n container.setAttribute('role', 'toolbar');\r\n container.setAttribute(\r\n 'aria-label',\r\n placement === 'bottom' ? 'BridgeRTE tabbar' : 'BridgeRTE toolbar'\r\n );\r\n container.addEventListener('click', handleClick);\r\n groupMenuElement.addEventListener('click', handleClick);\r\n document.addEventListener('click', handleDocumentClick);\r\n document.addEventListener('keydown', handleKeyDown);\r\n if (enableTooltip) {\r\n container.addEventListener('mouseover', handleTooltipTarget);\r\n container.addEventListener('mouseout', handleTooltipLeave);\r\n }\r\n container.addEventListener('focusout', hideTooltip);\r\n // 浮层挂在最近的编辑器根容器下,既能继承变量,也不会操作编辑内容 DOM。\r\n mountToolbarOverlays();\r\n\r\n // 独立 toolbar 只订阅 public API 状态,不依赖 DOM 编辑器内部实现。\r\n const unsubscribe = options.editor.subscribeCommandStateChange(renderStates);\r\n\r\n return {\r\n update,\r\n destroy() {\r\n if (destroyed) return;\r\n\r\n destroyed = true;\r\n unsubscribe();\r\n closeGroupMenu();\r\n container.removeEventListener('click', handleClick);\r\n groupMenuElement.removeEventListener('click', handleClick);\r\n document.removeEventListener('click', handleDocumentClick);\r\n document.removeEventListener('keydown', handleKeyDown);\r\n if (enableTooltip) {\r\n container.removeEventListener('mouseover', handleTooltipTarget);\r\n container.removeEventListener('mouseout', handleTooltipLeave);\r\n }\r\n clearToolbarPressedState();\r\n clearGroupMenuPressedState();\r\n container.removeEventListener('focusout', hideTooltip);\r\n tooltipElement.remove();\r\n groupMenuElement.remove();\r\n container.classList.remove('bridgerte__toolbar');\r\n delete container.dataset.placement;\r\n container.textContent = '';\r\n container.removeAttribute('role');\r\n container.removeAttribute('aria-label');\r\n }\r\n };\r\n}\r\n","const invisibleTextPattern = /[\\u200B-\\u200D\\uFEFF]/g;\n\n/*\n * HTML 片段里这些元素没有 textContent 也应算有效内容。\n * 纯排版容器不放进来,避免 `<p><br></p>`、空列表项这类编辑器占位被误判为非空。\n */\nconst meaningfulHtmlContentSelector = [\n 'img',\n 'video',\n 'audio',\n 'iframe',\n 'table',\n 'pre',\n 'code',\n 'hr',\n 'figure',\n 'canvas',\n 'svg',\n 'math',\n '[data-type=\"mention\"]'\n].join(',');\n\nconst normalizeHtmlText = (text: string | null | undefined): string => (\n text?.replace(invisibleTextPattern, '').trim() ?? ''\n);\n\n/**\n * 判断一段 HTML 片段是否包含可保存的富文本内容。\n *\n * 这个工具会使用浏览器 template 解析 HTML,适合只拿到 HTML 字符串的业务表单;如果已经持有\n * `EditorContent`,优先使用 core 的 `isEditorContentEmpty()`,避免额外 DOM parse。\n */\nexport const hasMeaningfulHtmlContent = (html: string): boolean => {\n if (!normalizeHtmlText(html)) return false;\n\n const template = document.createElement('template');\n\n template.innerHTML = html;\n\n return Boolean(\n normalizeHtmlText(template.content.textContent)\n || template.content.querySelector(meaningfulHtmlContentSelector)\n );\n};\n"],"names":["defaultMenuUiIcons","getEnabledGroupMenuButtons","menuElement","button","focusToolbarGroupMenuItem","index","nextButton","handleToolbarGroupMenuKeyDown","event","returnButton","executeToolbarButton","closeGroupMenu","buttons","activeIndex","focusByOffset","offset","nextIndex","_a","_b","toolbarGroupButtonSelector","toolbarButtonSelector","toolbarExecutableButtonSelector","getGroupMenuState","items","commandStates","itemStates","item","getMenuStateForItem","state","renderToolbarButton","groupElement","icons","enableTooltip","iconSvg","defaultMenuIcons","appendMenuIcon","renderToolbarGroupButton","toolbarItem","groupState","indicator","renderToolbar","toolbarElement","toolbarItems","separatorElement","renderToolbarGroupMenu","groupMenuState","labelElement","syncToolbarGroupButtonState","container","open","findToolbarGroupMenuItem","groupKey","toolbarTooltipOffsetPx","toolbarGroupMenuOffsetPx","canUseHoverTooltip","getToolbarButtonFromTarget","target","getToolbarGroupButtonFromTarget","createRichTextToolbar","options","placement","menuSchema","resolveMenuSchemaForDom","defaultMenuSchema","resolveToolbarMenu","executableMenuItems","tooltipElement","groupMenuElement","overlayHost","destroyed","latestCommandStates","groupMenuFloatingLayer","clearToolbarPressedState","bindTouchPressedState","clearGroupMenuPressedState","mountToolbarOverlays","closeGroupMenuFloatingLayer","openGroupMenuFloatingLayer","createFloatingLayer","renderStates","states","nextGroupItem","update","hideTooltip","showTooltip","tooltipText","buttonRect","handleTooltipTarget","handleTooltipLeave","relatedTarget","menuItem","getPayloadPanelCurrentValues","toggleGroupMenu","shouldFocusMenu","groupItem","handleClick","groupButton","handleDocumentClick","handleKeyDown","unsubscribe","invisibleTextPattern","meaningfulHtmlContentSelector","normalizeHtmlText","text","hasMeaningfulHtmlContent","html","template"],"mappings":"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"}
@@ -1,36 +0,0 @@
1
- "use strict";const H=require("./native-spec.cjs"),m=require("./index-BBzHVrSi.cjs"),N={toolbarGroup:`
2
- <svg
3
- aria-hidden="true"
4
- class="lucide lucide-ellipsis"
5
- xmlns="http://www.w3.org/2000/svg"
6
- width="18"
7
- height="18"
8
- viewBox="0 0 24 24"
9
- fill="none"
10
- stroke="currentColor"
11
- stroke-width="2"
12
- stroke-linecap="round"
13
- stroke-linejoin="round"
14
- >
15
- <circle cx="12" cy="12" r="1" />
16
- <circle cx="19" cy="12" r="1" />
17
- <circle cx="5" cy="12" r="1" />
18
- </svg>
19
- `,chevronDown:`
20
- <svg
21
- aria-hidden="true"
22
- class="lucide lucide-chevron-down"
23
- xmlns="http://www.w3.org/2000/svg"
24
- width="12"
25
- height="12"
26
- viewBox="0 0 24 24"
27
- fill="none"
28
- stroke="currentColor"
29
- stroke-width="2"
30
- stroke-linecap="round"
31
- stroke-linejoin="round"
32
- >
33
- <path d="m6 9 6 6 6-6" />
34
- </svg>
35
- `},O=e=>Array.from(e.querySelectorAll(".bridgerte__toolbar-group-menu-item")).filter(t=>!t.disabled),K=(e,t)=>{const l=O(e)[t];l&&l.focus()},J=(e,t,s,l,u)=>{var d,b;const n=O(t),r=document.activeElement instanceof HTMLButtonElement?n.indexOf(document.activeElement):-1,p=y=>{var i;if(n.length===0)return;const g=((r>=0?r:0)+y+n.length)%n.length;(i=n[g])==null||i.focus()};switch(e.key){case"ArrowDown":e.preventDefault(),p(1);break;case"ArrowUp":e.preventDefault(),p(-1);break;case"Home":e.preventDefault(),(d=n[0])==null||d.focus();break;case"End":e.preventDefault(),(b=n.at(-1))==null||b.focus();break;case"Enter":case" ":document.activeElement instanceof HTMLButtonElement&&(e.preventDefault(),l(document.activeElement),u(),s.focus());break;case"Escape":e.preventDefault(),s.focus(),u();break}},S="button[data-bridgerte-toolbar-group-id]",Q="button[data-bridgerte-toolbar-item-id]",X=[Q,".bridgerte__toolbar-group-menu-item"].join(","),Y=(e,t)=>{const s=e.map(l=>m.getMenuStateForItem(l,t));return{active:s.some(l=>l.active),disabled:s.length>0&&s.every(l=>l.disabled)}},Z=(e,t,s,l,u)=>{const n=m.getMenuStateForItem(t,s),r=document.createElement("button"),p=l[t.icon]??m.defaultMenuIcons[t.icon];r.type="button",r.className="bridgerte__toolbar-button",r.disabled=n.disabled,r.dataset.active=String(n.active),r.dataset.bridgerteToolbarItemId=t.id,r.setAttribute("aria-label",t.label),r.setAttribute("aria-pressed",String(n.active)),u&&(r.dataset.tooltip=t.label),m.appendMenuIcon(r,p,t.label),e.append(r)},ee=(e,t,s,l,u)=>{const n=document.createElement("button"),r=Y(t.items,s),p=t.icon?l[t.icon]??m.defaultMenuIcons[t.icon]:N.toolbarGroup,d=document.createElement("span");n.type="button",n.className="bridgerte__toolbar-button bridgerte__toolbar-group-button",n.disabled=r.disabled,n.dataset.active=String(r.active),n.dataset.bridgerteToolbarGroupId=t.key,n.dataset.open="false",n.setAttribute("aria-label",t.title),n.setAttribute("aria-haspopup","menu"),n.setAttribute("aria-expanded","false"),n.setAttribute("aria-pressed",String(r.active)),u&&(n.dataset.tooltip=t.title),m.appendMenuIcon(n,p,t.title),d.className="bridgerte__toolbar-group-indicator",d.setAttribute("aria-hidden","true"),d.innerHTML=N.chevronDown,n.append(d),e.append(n)},te=(e,t,s,l,u)=>{e.textContent="",t.forEach(n=>{if(n.type==="separator"){const p=document.createElement("span");p.className="bridgerte__toolbar-separator",p.dataset.separatorId=n.key,p.setAttribute("aria-hidden","true"),e.append(p);return}if(n.type==="button"){Z(e,n.item,s,l,u);return}const r=document.createElement("div");r.className="bridgerte__toolbar-group",r.dataset.group=n.key,r.setAttribute("aria-label",n.title),e.append(r),ee(r,n,s,l,u)})},k=(e,t,s,l)=>{if(e.textContent="",!t){e.dataset.visible="false";return}t.items.forEach(u=>{const n=m.getMenuStateForItem(u,s),r=document.createElement("button"),p=l[u.icon]??m.defaultMenuIcons[u.icon],d=document.createElement("span");r.type="button",r.className="bridgerte__menu-item bridgerte__toolbar-group-menu-item",r.disabled=n.disabled,r.dataset.active=String(n.active),r.dataset.bridgerteToolbarItemId=u.id,r.setAttribute("role","menuitem"),r.setAttribute("aria-label",u.label),r.setAttribute("aria-pressed",String(n.active)),m.appendMenuIcon(r,p,u.label),d.className="bridgerte__toolbar-group-menu-label",d.textContent=u.label,r.append(d),e.append(r)}),e.dataset.visible="true",e.style.minWidth=`${t.button.offsetWidth}px`},w=(e,t)=>{e.querySelectorAll(S).forEach(s=>{const l=(t==null?void 0:t.groupKey)===s.dataset.bridgerteToolbarGroupId;s.dataset.open=String(l),s.setAttribute("aria-expanded",String(l))})},R=(e,t)=>e.find(s=>s.type==="group"&&s.key===t),oe=8,ne=6,re=()=>{var e;return typeof window<"u"&&((e=window.matchMedia)==null?void 0:e.call(window,"(hover: hover) and (pointer: fine)").matches)===!0},_=e=>{const t=e instanceof Element?e.closest(X):null;return t instanceof HTMLButtonElement?t:null},F=e=>{const t=e instanceof Element?e.closest(S):null;return t instanceof HTMLButtonElement?t:null};function ae(e,t){const s=t.placement??"top",l=m.resolveMenuSchemaForDom(t.menuSchema??H.defaultMenuSchema,{menuLabels:t.menuLabels,payloadPanelConfig:t.payloadPanelConfig}),u=H.resolveToolbarMenu(t.toolbarConfig,l),n=u.flatMap(o=>o.type==="button"?[o.item]:o.type==="group"?o.items:[]),r=t.icons??{},p=re(),d=document.createElement("div"),b=document.createElement("div"),y=e.closest(".bridgerte")??e;let h=!1,g=t.editor.getCommandStates(),i=null,f=null;const j=m.bindTouchPressedState(e,{targetSelector:["button[data-bridgerte-toolbar-item-id]","button[data-bridgerte-toolbar-group-id]"].join(",")}),U=m.bindTouchPressedState(b,{targetSelector:".bridgerte__toolbar-group-menu-item"}),B=()=>{y.append(d),y.append(b)},x=()=>{f==null||f.setOpen(!1),f==null||f.destroy(),f=null},C=()=>{i&&(x(),f=m.createFloatingLayer(i.button,b,{placement:s==="bottom"?"top-start":"bottom-start",offset:ne,strategy:"fixed"}),f.setOpen(!0))},v=()=>{x(),i=null,k(b,i,g,r),w(e,i)},A=o=>{if(!h&&(g=o,i&&x(),te(e,u,o,r,p),B(),i)){const a=Array.from(e.querySelectorAll(S)).find(z=>z.dataset.bridgerteToolbarGroupId===(i==null?void 0:i.groupKey)),c=R(u,i.groupKey);i=a&&c?{groupKey:i.groupKey,button:a,items:c.items}:null,k(b,i,g,r),w(e,i),i&&C()}},$=()=>{h||A(t.editor.getCommandStates())},T=()=>{d.dataset.visible="false",d.textContent=""},V=o=>{const a=o.dataset.tooltip;if(!p||!a)return;const c=o.getBoundingClientRect();d.textContent=a,d.dataset.visible="true",d.style.left=`${c.left+c.width/2}px`,d.style.top=`${c.top-oe}px`},L=o=>{const a=_(o.target);a&&V(a)},M=o=>{const a=o.relatedTarget,c=_(o.target);c&&a instanceof Node&&c.contains(a)||T()},G=o=>{if(!(o instanceof HTMLButtonElement)||o.disabled)return;const a=n.find(c=>c.id===o.dataset.bridgerteToolbarItemId);if(a){if(a.payloadPanel){const c=o.getBoundingClientRect();t.editor.requestPayloadPanel({menuId:a.id,command:a.command,panel:a.payloadPanel,currentValues:m.getPayloadPanelCurrentValues(a,t.editor.getCommandStates()),anchorRect:{x:c.left,y:c.top,width:c.width,height:c.height}});return}t.editor.executeCommand(a.command)}},D=(o,a=!1)=>{const c=R(u,o.dataset.bridgerteToolbarGroupId);if(!(!c||o.disabled)){if((i==null?void 0:i.groupKey)===c.key){if(a){K(b,0);return}v();return}i={groupKey:c.key,button:o,items:c.items},T(),k(b,i,g,r),w(e,i),C(),a&&K(b,0)}},E=o=>{const a=F(o.target);if(a){o.preventDefault(),o.stopPropagation(),D(a);return}const c=_(o.target);c&&(o.preventDefault(),o.stopPropagation(),G(c),v())},P=o=>{const a=o.target;a instanceof Node&&(e.contains(a)||b.contains(a))||v()},I=o=>{if(i&&b.contains(document.activeElement)){J(o,b,i.button,G,v);return}const a=F(o.target);if(a&&(o.key==="Enter"||o.key===" "||o.key==="ArrowDown")){o.preventDefault(),o.stopPropagation(),D(a,!0);return}o.key==="Escape"&&v()};e.classList.add("bridgerte__toolbar"),d.className="bridgerte__toolbar-tooltip",d.dataset.visible="false",b.className="bridgerte__floating-menu bridgerte__toolbar-group-menu",b.dataset.visible="false",b.setAttribute("role","menu"),e.dataset.placement=s,e.setAttribute("role","toolbar"),e.setAttribute("aria-label",s==="bottom"?"BridgeRTE tabbar":"BridgeRTE toolbar"),e.addEventListener("click",E),b.addEventListener("click",E),document.addEventListener("click",P),document.addEventListener("keydown",I),p&&(e.addEventListener("mouseover",L),e.addEventListener("mouseout",M)),e.addEventListener("focusout",T),B();const W=t.editor.subscribeCommandStateChange(A);return{update:$,destroy(){h||(h=!0,W(),v(),e.removeEventListener("click",E),b.removeEventListener("click",E),document.removeEventListener("click",P),document.removeEventListener("keydown",I),p&&(e.removeEventListener("mouseover",L),e.removeEventListener("mouseout",M)),j(),U(),e.removeEventListener("focusout",T),d.remove(),b.remove(),e.classList.remove("bridgerte__toolbar"),delete e.dataset.placement,e.textContent="",e.removeAttribute("role"),e.removeAttribute("aria-label"))}}}const se=/[\u200B-\u200D\uFEFF]/g,ie=["img","video","audio","iframe","table","pre","code","hr","figure","canvas","svg","math",'[data-type="mention"]'].join(","),q=e=>(e==null?void 0:e.replace(se,"").trim())??"",le=e=>{if(!q(e))return!1;const t=document.createElement("template");return t.innerHTML=e,!!(q(t.content.textContent)||t.content.querySelector(ie))};exports.createRichTextToolbar=ae;exports.hasMeaningfulHtmlContent=le;
36
- //# sourceMappingURL=index-DIJtCo91.cjs.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"index-DIJtCo91.cjs","sources":["../../dom/src/menuIcon/uiIcons.ts","../../dom/src/richTextToolbar/groupMenuKeyboard.ts","../../dom/src/richTextToolbar/render.ts","../../dom/src/richTextToolbar/index.ts","../../dom/src/htmlContent/index.ts"],"sourcesContent":["/**\r\n * DOM UI chrome icon 表。\r\n *\r\n * 这里放 toolbar 收纳入口、下拉箭头这类控件自身图标。它们不属于 `MenuItem.icon`\r\n * 跨端 schema,也不应该进入 `defaultMenuIcons` 的 schema 对齐检查。\r\n */\r\nexport const defaultMenuUiIcons = {\r\n toolbarGroup: `\r\n <svg\r\n aria-hidden=\"true\"\r\n class=\"lucide lucide-ellipsis\"\r\n xmlns=\"http://www.w3.org/2000/svg\"\r\n width=\"18\"\r\n height=\"18\"\r\n viewBox=\"0 0 24 24\"\r\n fill=\"none\"\r\n stroke=\"currentColor\"\r\n stroke-width=\"2\"\r\n stroke-linecap=\"round\"\r\n stroke-linejoin=\"round\"\r\n >\r\n <circle cx=\"12\" cy=\"12\" r=\"1\" />\r\n <circle cx=\"19\" cy=\"12\" r=\"1\" />\r\n <circle cx=\"5\" cy=\"12\" r=\"1\" />\r\n </svg>\r\n `,\r\n chevronDown: `\r\n <svg\r\n aria-hidden=\"true\"\r\n class=\"lucide lucide-chevron-down\"\r\n xmlns=\"http://www.w3.org/2000/svg\"\r\n width=\"12\"\r\n height=\"12\"\r\n viewBox=\"0 0 24 24\"\r\n fill=\"none\"\r\n stroke=\"currentColor\"\r\n stroke-width=\"2\"\r\n stroke-linecap=\"round\"\r\n stroke-linejoin=\"round\"\r\n >\r\n <path d=\"m6 9 6 6 6-6\" />\r\n </svg>\r\n `\r\n} as const;\r\n","export const getEnabledGroupMenuButtons = (menuElement: HTMLElement) => (\r\n Array.from(\r\n menuElement.querySelectorAll<HTMLButtonElement>('.bridgerte__toolbar-group-menu-item')\r\n ).filter((button) => !button.disabled)\r\n);\r\n\r\nexport const focusToolbarGroupMenuItem = (menuElement: HTMLElement, index: number) => {\r\n const buttons = getEnabledGroupMenuButtons(menuElement);\r\n const nextButton = buttons[index];\r\n\r\n if (nextButton) nextButton.focus();\r\n};\r\n\r\nexport const handleToolbarGroupMenuKeyDown = (\n event: KeyboardEvent,\r\n menuElement: HTMLElement,\r\n returnButton: HTMLButtonElement,\r\n executeToolbarButton: (button: HTMLButtonElement) => void,\r\n closeGroupMenu: () => void\r\n) => {\r\n const buttons = getEnabledGroupMenuButtons(menuElement);\r\n const activeIndex = document.activeElement instanceof HTMLButtonElement\r\n ? buttons.indexOf(document.activeElement)\r\n : -1;\r\n const focusByOffset = (offset: number) => {\r\n if (buttons.length === 0) return;\r\n\r\n const baseIndex = activeIndex >= 0 ? activeIndex : 0;\r\n const nextIndex = (baseIndex + offset + buttons.length) % buttons.length;\r\n\r\n buttons[nextIndex]?.focus();\r\n };\r\n\r\n /*\r\n * ARIA menu 语义要求支持 roving focus。这里不改 toolbar 的横向键盘模型,\r\n * 只在已打开的纵向收纳菜单内处理上下方向、首尾跳转和执行/关闭。\r\n */\r\n switch (event.key) {\r\n case 'ArrowDown':\r\n event.preventDefault();\r\n focusByOffset(1);\r\n break;\r\n case 'ArrowUp':\r\n event.preventDefault();\r\n focusByOffset(-1);\r\n break;\r\n case 'Home':\r\n event.preventDefault();\r\n buttons[0]?.focus();\r\n break;\r\n case 'End':\r\n event.preventDefault();\r\n buttons.at(-1)?.focus();\r\n break;\r\n case 'Enter':\r\n case ' ':\r\n if (document.activeElement instanceof HTMLButtonElement) {\r\n event.preventDefault();\r\n executeToolbarButton(document.activeElement);\r\n closeGroupMenu();\r\n returnButton.focus();\r\n }\r\n break;\r\n case 'Escape':\r\n event.preventDefault();\r\n returnButton.focus();\r\n closeGroupMenu();\r\n break;\r\n default:\r\n break;\r\n }\r\n};\r\n","import type { CommandState } from '@bridgerte/core';\r\nimport type { MenuItem, ResolvedToolbarItem } from '@bridgerte/native-spec';\r\nimport {\r\n appendMenuIcon,\r\n defaultMenuIcons,\r\n defaultMenuUiIcons\r\n} from './icons';\r\nimport { getMenuStateForItem } from '../menuRuntime';\r\nimport type {\r\n RichTextToolbarIcons,\r\n ToolbarGroupMenuState\r\n} from './type';\r\n\r\nexport const toolbarGroupButtonSelector = 'button[data-bridgerte-toolbar-group-id]';\r\nexport const toolbarButtonSelector = 'button[data-bridgerte-toolbar-item-id]';\r\n// 可执行按钮同时覆盖 toolbar 主按钮和 group menu 子项,避免浮层子菜单绕过统一命令入口。\r\nexport const toolbarExecutableButtonSelector = [\n toolbarButtonSelector,\n '.bridgerte__toolbar-group-menu-item'\n].join(',');\n\r\nconst getGroupMenuState = (items: MenuItem[], commandStates: CommandState[]) => {\r\n const itemStates = items.map((item) => getMenuStateForItem(item, commandStates));\r\n\r\n return {\r\n active: itemStates.some((state) => state.active),\r\n disabled: itemStates.length > 0 && itemStates.every((state) => state.disabled)\r\n };\r\n};\r\n\r\nconst renderToolbarButton = (\r\n groupElement: HTMLElement,\r\n item: MenuItem,\r\n commandStates: CommandState[],\r\n icons: RichTextToolbarIcons,\r\n enableTooltip: boolean\r\n) => {\r\n const state = getMenuStateForItem(item, commandStates);\r\n const button = document.createElement('button');\r\n // icon 兜底顺序固定为:业务覆盖 > DOM 默认 SVG > label 文本。\r\n const iconSvg = icons[item.icon] ?? defaultMenuIcons[item.icon];\r\n\r\n button.type = 'button';\r\n button.className = 'bridgerte__toolbar-button';\r\n button.disabled = state.disabled;\r\n button.dataset.active = String(state.active);\r\n button.dataset.bridgerteToolbarItemId = item.id;\r\n button.setAttribute('aria-label', item.label);\r\n button.setAttribute('aria-pressed', String(state.active));\r\n if (enableTooltip) button.dataset.tooltip = item.label;\r\n\r\n appendMenuIcon(button, iconSvg, item.label);\r\n\r\n groupElement.append(button);\r\n};\r\n\r\nconst renderToolbarGroupButton = (\r\n groupElement: HTMLElement,\r\n toolbarItem: Extract<ResolvedToolbarItem, { type: 'group' }>,\r\n commandStates: CommandState[],\r\n icons: RichTextToolbarIcons,\r\n enableTooltip: boolean\r\n) => {\r\n const button = document.createElement('button');\r\n const groupState = getGroupMenuState(toolbarItem.items, commandStates);\r\n const iconSvg = toolbarItem.icon\r\n ? icons[toolbarItem.icon] ?? defaultMenuIcons[toolbarItem.icon]\r\n : defaultMenuUiIcons.toolbarGroup;\r\n const indicator = document.createElement('span');\r\n\r\n button.type = 'button';\r\n button.className = 'bridgerte__toolbar-button bridgerte__toolbar-group-button';\r\n button.disabled = groupState.disabled;\r\n button.dataset.active = String(groupState.active);\r\n button.dataset.bridgerteToolbarGroupId = toolbarItem.key;\r\n button.dataset.open = 'false';\r\n button.setAttribute('aria-label', toolbarItem.title);\r\n button.setAttribute('aria-haspopup', 'menu');\r\n button.setAttribute('aria-expanded', 'false');\r\n button.setAttribute('aria-pressed', String(groupState.active));\r\n if (enableTooltip) button.dataset.tooltip = toolbarItem.title;\r\n\r\n appendMenuIcon(button, iconSvg, toolbarItem.title);\r\n\r\n indicator.className = 'bridgerte__toolbar-group-indicator';\r\n indicator.setAttribute('aria-hidden', 'true');\r\n indicator.innerHTML = defaultMenuUiIcons.chevronDown;\r\n button.append(indicator);\r\n groupElement.append(button);\r\n};\r\n\r\n/**\r\n * 渲染 toolbar 横向主入口。\n *\n * 字符串菜单直接渲染为按钮,`|` 渲染分割线。\n * 只有用户显式声明的 group 配置才生成收纳入口,避免 schema 隐式改变 DOM 结构。\n */\r\nexport const renderToolbar = (\r\n toolbarElement: HTMLElement,\r\n toolbarItems: ResolvedToolbarItem[],\r\n commandStates: CommandState[],\r\n icons: RichTextToolbarIcons,\r\n enableTooltip: boolean\r\n) => {\r\n toolbarElement.textContent = '';\r\n\r\n toolbarItems.forEach((toolbarItem) => {\n if (toolbarItem.type === 'separator') {\n const separatorElement = document.createElement('span');\n\n separatorElement.className = 'bridgerte__toolbar-separator';\n separatorElement.dataset.separatorId = toolbarItem.key;\r\n separatorElement.setAttribute('aria-hidden', 'true');\r\n toolbarElement.append(separatorElement);\r\n return;\n }\n\n if (toolbarItem.type === 'button') {\n renderToolbarButton(\n toolbarElement,\n toolbarItem.item,\n commandStates,\n icons,\n enableTooltip\n );\n return;\n }\n\n /*\n * 收纳菜单是用户在 toolbarConfig 里显式声明的结构;普通菜单不会自动生成 DOM 包裹,\n * 显示分组完全交给用户用 group 配置或 `|` 控制。\n */\n const groupElement = document.createElement('div');\n\n groupElement.className = 'bridgerte__toolbar-group';\n groupElement.dataset.group = toolbarItem.key;\n groupElement.setAttribute('aria-label', toolbarItem.title);\n toolbarElement.append(groupElement);\n\n renderToolbarGroupButton(\n groupElement,\n toolbarItem,\n commandStates,\n icons,\n enableTooltip\r\n );\r\n });\r\n};\r\n\r\n/**\r\n * 渲染 group button 打开的纵向收纳菜单。\r\n *\r\n * 浮层只展示 MenuItem 子项并复用现有 item id,点击执行仍由 index.ts 统一走 EditorAPI。\r\n */\r\nexport const renderToolbarGroupMenu = (\r\n menuElement: HTMLElement,\r\n groupMenuState: ToolbarGroupMenuState | null,\r\n commandStates: CommandState[],\r\n icons: RichTextToolbarIcons\r\n) => {\r\n menuElement.textContent = '';\r\n\r\n if (!groupMenuState) {\r\n menuElement.dataset.visible = 'false';\r\n return;\r\n }\r\n\r\n groupMenuState.items.forEach((item) => {\r\n const state = getMenuStateForItem(item, commandStates);\r\n const button = document.createElement('button');\r\n const iconSvg = icons[item.icon] ?? defaultMenuIcons[item.icon];\r\n const labelElement = document.createElement('span');\n\n button.type = 'button';\n button.className = 'bridgerte__menu-item bridgerte__toolbar-group-menu-item';\n button.disabled = state.disabled;\r\n button.dataset.active = String(state.active);\r\n button.dataset.bridgerteToolbarItemId = item.id;\r\n button.setAttribute('role', 'menuitem');\r\n button.setAttribute('aria-label', item.label);\r\n button.setAttribute('aria-pressed', String(state.active));\r\n\r\n appendMenuIcon(button, iconSvg, item.label);\r\n labelElement.className = 'bridgerte__toolbar-group-menu-label';\r\n labelElement.textContent = item.label;\r\n button.append(labelElement);\r\n menuElement.append(button);\r\n });\r\n\r\n menuElement.dataset.visible = 'true';\r\n menuElement.style.minWidth = `${groupMenuState.button.offsetWidth}px`;\r\n};\r\n\r\nexport const syncToolbarGroupButtonState = (\r\n container: HTMLElement,\r\n groupMenuState: ToolbarGroupMenuState | null\r\n) => {\r\n container.querySelectorAll<HTMLButtonElement>(toolbarGroupButtonSelector).forEach((button) => {\r\n const open = groupMenuState?.groupKey === button.dataset.bridgerteToolbarGroupId;\r\n\r\n button.dataset.open = String(open);\r\n button.setAttribute('aria-expanded', String(open));\r\n });\r\n};\r\n\r\nexport const findToolbarGroupMenuItem = (\r\n toolbarItems: ResolvedToolbarItem[],\r\n groupKey: string | undefined\r\n) => toolbarItems.find((item): item is Extract<ResolvedToolbarItem, { type: 'group' }> => (\r\n item.type === 'group' && item.key === groupKey\r\n));\r\n","import type { CommandState } from '@bridgerte/core';\r\nimport {\r\n defaultMenuSchema,\r\n resolveToolbarMenu\r\n} from '@bridgerte/native-spec';\r\nimport { createFloatingLayer, type RichTextFloatingLayer } from '../floatingLayer';\r\nimport { bindTouchPressedState } from '../interactionState';\r\nimport {\r\n getPayloadPanelCurrentValues,\r\n resolveMenuSchemaForDom\r\n} from '../menuRuntime';\r\nimport {\r\n focusToolbarGroupMenuItem,\r\n handleToolbarGroupMenuKeyDown\r\n} from './groupMenuKeyboard';\r\nimport {\r\n findToolbarGroupMenuItem,\r\n renderToolbar,\r\n renderToolbarGroupMenu,\r\n syncToolbarGroupButtonState,\r\n toolbarExecutableButtonSelector,\r\n toolbarGroupButtonSelector\r\n} from './render';\r\nimport type {\r\n RichTextToolbarAPI,\r\n ToolbarGroupMenuState,\r\n RichTextToolbarOptions\r\n} from './type';\r\n\r\nconst toolbarTooltipOffsetPx = 8;\r\nconst toolbarGroupMenuOffsetPx = 6;\r\n\r\nexport type * from './type';\r\n\r\nconst canUseHoverTooltip = () => (\r\n typeof window !== 'undefined'\r\n && window.matchMedia?.('(hover: hover) and (pointer: fine)').matches === true\r\n);\r\n\r\nconst getToolbarButtonFromTarget = (target: EventTarget | null) => {\r\n /*\r\n * 图标覆盖常用 SVG,H5 pointer 事件可能落在 svg/path 上。\r\n * 这里用 Element 而不是 HTMLElement,保证触屏兜底和 click 都能向上命中真实按钮。\r\n */\r\n const button = target instanceof Element\r\n ? target.closest<HTMLButtonElement>(toolbarExecutableButtonSelector)\r\n : null;\r\n\r\n return button instanceof HTMLButtonElement ? button : null;\r\n};\r\n\r\nconst getToolbarGroupButtonFromTarget = (target: EventTarget | null) => {\r\n const button = target instanceof Element\r\n ? target.closest<HTMLButtonElement>(toolbarGroupButtonSelector)\r\n : null;\r\n\r\n return button instanceof HTMLButtonElement ? button : null;\r\n};\r\n\r\n/**\r\n * 创建独立菜单实例。\r\n *\r\n * toolbar/tabbar 只订阅 EditorAPI 状态并派发命令,不接触编辑器内部 DOM 或 Lexical 实例。\r\n */\r\nexport function createRichTextToolbar(\r\n container: HTMLElement,\r\n options: RichTextToolbarOptions\r\n): RichTextToolbarAPI {\r\n const placement = options.placement ?? 'top';\r\n const menuSchema = resolveMenuSchemaForDom(options.menuSchema ?? defaultMenuSchema, {\r\n menuLabels: options.menuLabels,\r\n payloadPanelConfig: options.payloadPanelConfig\r\n });\r\n const toolbarItems = resolveToolbarMenu(options.toolbarConfig, menuSchema);\r\n /*\r\n * click 事件只需要能执行命令的菜单项。分割线和 group 容器是渲染结构,\r\n * 先在这里摊平,后续点击时就不用理解 toolbarConfig 的展示层级。\r\n */\r\n const executableMenuItems = toolbarItems.flatMap((toolbarItem) => (\r\n toolbarItem.type === 'button' ? [toolbarItem.item]\r\n : toolbarItem.type === 'group' ? toolbarItem.items\r\n : []\r\n ));\r\n const icons = options.icons ?? {};\r\n /*\r\n * tooltip 只属于 PC 精细指针体验。H5 上即使没有原生 title,部分浏览器也会把 hover/mouseover\r\n * 模拟成首触摸状态,导致第一次点像是只唤醒提示;所以触屏端不写 tooltip 数据也不挂监听。\r\n */\r\n const enableTooltip = canUseHoverTooltip();\r\n const tooltipElement = document.createElement('div');\r\n const groupMenuElement = document.createElement('div');\r\n const overlayHost = container.closest('.bridgerte') ?? container;\r\n let destroyed = false;\r\n let latestCommandStates = options.editor.getCommandStates();\r\n let groupMenuState: ToolbarGroupMenuState | null = null;\r\n let groupMenuFloatingLayer: RichTextFloatingLayer | null = null;\r\n const clearToolbarPressedState = bindTouchPressedState(container, {\r\n targetSelector: [\r\n 'button[data-bridgerte-toolbar-item-id]',\r\n 'button[data-bridgerte-toolbar-group-id]'\r\n ].join(',')\r\n });\r\n const clearGroupMenuPressedState = bindTouchPressedState(groupMenuElement, {\r\n targetSelector: '.bridgerte__toolbar-group-menu-item'\r\n });\r\n\r\n const mountToolbarOverlays = () => {\r\n // 独立 toolbar 可能不在 `.bridgerte` 内,渲染按钮会清空 container,需要把浮层重新挂回去。\r\n overlayHost.append(tooltipElement);\r\n overlayHost.append(groupMenuElement);\r\n };\r\n\r\n const closeGroupMenuFloatingLayer = () => {\r\n groupMenuFloatingLayer?.setOpen(false);\r\n groupMenuFloatingLayer?.destroy();\r\n groupMenuFloatingLayer = null;\r\n };\r\n\r\n const openGroupMenuFloatingLayer = () => {\r\n if (!groupMenuState) return;\r\n\r\n closeGroupMenuFloatingLayer();\r\n /*\r\n * group menu 和 hoverbar/mention/slash 一样属于轻浮层,必须走 floatingLayer。\r\n * 这里按 toolbar placement 给出首选方向,真正的翻转、键盘可视区和左右夹紧交给\r\n * createFloatingLayer 内部的 visualViewport + flip + shift + clamp 统一处理。\r\n */\r\n groupMenuFloatingLayer = createFloatingLayer(groupMenuState.button, groupMenuElement, {\r\n placement: placement === 'bottom' ? 'top-start' : 'bottom-start',\r\n offset: toolbarGroupMenuOffsetPx,\r\n strategy: 'fixed'\r\n });\r\n groupMenuFloatingLayer.setOpen(true);\r\n };\r\n\r\n const closeGroupMenu = () => {\r\n closeGroupMenuFloatingLayer();\r\n groupMenuState = null;\r\n renderToolbarGroupMenu(groupMenuElement, groupMenuState, latestCommandStates, icons);\r\n syncToolbarGroupButtonState(container, groupMenuState);\r\n };\r\n\r\n const renderStates = (states: CommandState[]) => {\r\n if (destroyed) return;\r\n\r\n latestCommandStates = states;\r\n if (groupMenuState) closeGroupMenuFloatingLayer();\r\n renderToolbar(container, toolbarItems, states, icons, enableTooltip);\r\n mountToolbarOverlays();\r\n if (groupMenuState) {\r\n const nextButton = Array.from(\r\n container.querySelectorAll<HTMLButtonElement>(toolbarGroupButtonSelector)\r\n ).find((button) => button.dataset.bridgerteToolbarGroupId === groupMenuState?.groupKey);\r\n const nextGroupItem = findToolbarGroupMenuItem(toolbarItems, groupMenuState.groupKey);\r\n\r\n groupMenuState = nextButton && nextGroupItem\r\n ? {\r\n groupKey: groupMenuState.groupKey,\r\n button: nextButton,\r\n items: nextGroupItem.items\r\n }\r\n : null;\r\n renderToolbarGroupMenu(groupMenuElement, groupMenuState, latestCommandStates, icons);\r\n syncToolbarGroupButtonState(container, groupMenuState);\r\n if (groupMenuState) openGroupMenuFloatingLayer();\r\n }\r\n };\r\n\r\n const update = () => {\r\n if (destroyed) return;\r\n\r\n renderStates(options.editor.getCommandStates());\r\n };\r\n\r\n const hideTooltip = () => {\r\n tooltipElement.dataset.visible = 'false';\r\n tooltipElement.textContent = '';\r\n };\r\n\r\n const showTooltip = (button: HTMLButtonElement) => {\r\n const tooltipText = button.dataset.tooltip;\r\n\r\n if (!enableTooltip || !tooltipText) return;\r\n\r\n const buttonRect = button.getBoundingClientRect();\r\n\r\n tooltipElement.textContent = tooltipText;\r\n tooltipElement.dataset.visible = 'true';\r\n tooltipElement.style.left = `${buttonRect.left + buttonRect.width / 2}px`;\r\n tooltipElement.style.top = `${buttonRect.top - toolbarTooltipOffsetPx}px`;\r\n };\r\n\r\n const handleTooltipTarget = (event: Event) => {\r\n const button = getToolbarButtonFromTarget(event.target);\r\n\r\n if (button) {\r\n showTooltip(button);\r\n }\r\n };\r\n\r\n const handleTooltipLeave = (event: MouseEvent) => {\r\n const relatedTarget = event.relatedTarget;\r\n const button = getToolbarButtonFromTarget(event.target);\r\n\r\n if (\r\n button\r\n && relatedTarget instanceof Node\r\n && button.contains(relatedTarget)\r\n ) return;\r\n\r\n hideTooltip();\r\n };\r\n\r\n const executeToolbarButton = (button: HTMLButtonElement) => {\r\n if (!(button instanceof HTMLButtonElement) || button.disabled) return;\r\n\r\n const menuItem = executableMenuItems.find((item) => (\r\n item.id === button.dataset.bridgerteToolbarItemId\r\n ));\r\n\r\n if (!menuItem) return;\r\n\r\n if (menuItem.payloadPanel) {\r\n const buttonRect = button.getBoundingClientRect();\r\n\r\n /*\r\n * 带 payloadPanel 的菜单不直接执行基础 command,而是发起参数请求。\r\n * DOM 内置面板和业务/RN/Flutter 自绘都走同一个 request,避免后续颜色、\r\n * 字体、链接等参数菜单各自发明一套协议。\r\n */\r\n options.editor.requestPayloadPanel({\r\n menuId: menuItem.id,\r\n command: menuItem.command,\r\n panel: menuItem.payloadPanel,\r\n currentValues: getPayloadPanelCurrentValues(menuItem, options.editor.getCommandStates()),\r\n anchorRect: {\r\n x: buttonRect.left,\r\n y: buttonRect.top,\r\n width: buttonRect.width,\r\n height: buttonRect.height\r\n }\r\n });\r\n return;\r\n }\r\n\r\n options.editor.executeCommand(menuItem.command);\r\n };\r\n\r\n const toggleGroupMenu = (button: HTMLButtonElement, shouldFocusMenu = false) => {\r\n const groupItem = findToolbarGroupMenuItem(\r\n toolbarItems,\r\n button.dataset.bridgerteToolbarGroupId\r\n );\r\n\r\n if (!groupItem || button.disabled) return;\r\n\r\n if (groupMenuState?.groupKey === groupItem.key) {\r\n if (shouldFocusMenu) {\r\n focusToolbarGroupMenuItem(groupMenuElement, 0);\r\n return;\r\n }\r\n closeGroupMenu();\r\n return;\r\n }\r\n\r\n groupMenuState = {\r\n groupKey: groupItem.key,\r\n button,\r\n items: groupItem.items\r\n };\r\n hideTooltip();\r\n renderToolbarGroupMenu(groupMenuElement, groupMenuState, latestCommandStates, icons);\r\n syncToolbarGroupButtonState(container, groupMenuState);\r\n openGroupMenuFloatingLayer();\r\n if (shouldFocusMenu) focusToolbarGroupMenuItem(groupMenuElement, 0);\r\n };\r\n\r\n const handleClick = (event: MouseEvent) => {\r\n const groupButton = getToolbarGroupButtonFromTarget(event.target);\r\n if (groupButton) {\r\n event.preventDefault();\r\n event.stopPropagation();\r\n toggleGroupMenu(groupButton);\r\n return;\r\n }\r\n\r\n const button = getToolbarButtonFromTarget(event.target);\r\n if (!button) return;\r\n\r\n event.preventDefault();\r\n event.stopPropagation();\r\n executeToolbarButton(button);\r\n closeGroupMenu();\r\n };\r\n\r\n const handleDocumentClick = (event: MouseEvent) => {\r\n const target = event.target;\r\n\r\n if (\r\n target instanceof Node\r\n && (container.contains(target) || groupMenuElement.contains(target))\r\n ) return;\r\n\r\n closeGroupMenu();\r\n };\r\n\r\n const handleKeyDown = (event: KeyboardEvent) => {\r\n if (groupMenuState && groupMenuElement.contains(document.activeElement)) {\r\n handleToolbarGroupMenuKeyDown(\r\n event,\r\n groupMenuElement,\r\n groupMenuState.button,\r\n executeToolbarButton,\r\n closeGroupMenu\r\n );\r\n return;\r\n }\r\n\r\n const groupButton = getToolbarGroupButtonFromTarget(event.target);\r\n if (\r\n groupButton\r\n && (event.key === 'Enter' || event.key === ' ' || event.key === 'ArrowDown')\r\n ) {\r\n /*\r\n * 指针点击收纳菜单不能抢编辑器焦点;键盘从入口按钮打开时则进入菜单项,\r\n * 保留 PC 可访问性和方向键 roving focus。\r\n */\r\n event.preventDefault();\r\n event.stopPropagation();\r\n toggleGroupMenu(groupButton, true);\r\n return;\r\n }\r\n\r\n if (event.key !== 'Escape') return;\r\n\r\n closeGroupMenu();\r\n };\r\n\r\n container.classList.add('bridgerte__toolbar');\r\n tooltipElement.className = 'bridgerte__toolbar-tooltip';\r\n tooltipElement.dataset.visible = 'false';\r\n groupMenuElement.className = 'bridgerte__floating-menu bridgerte__toolbar-group-menu';\r\n groupMenuElement.dataset.visible = 'false';\r\n groupMenuElement.setAttribute('role', 'menu');\r\n container.dataset.placement = placement;\r\n container.setAttribute('role', 'toolbar');\r\n container.setAttribute(\r\n 'aria-label',\r\n placement === 'bottom' ? 'BridgeRTE tabbar' : 'BridgeRTE toolbar'\r\n );\r\n container.addEventListener('click', handleClick);\r\n groupMenuElement.addEventListener('click', handleClick);\r\n document.addEventListener('click', handleDocumentClick);\r\n document.addEventListener('keydown', handleKeyDown);\r\n if (enableTooltip) {\r\n container.addEventListener('mouseover', handleTooltipTarget);\r\n container.addEventListener('mouseout', handleTooltipLeave);\r\n }\r\n container.addEventListener('focusout', hideTooltip);\r\n // 浮层挂在最近的编辑器根容器下,既能继承变量,也不会操作编辑内容 DOM。\r\n mountToolbarOverlays();\r\n\r\n // 独立 toolbar 只订阅 public API 状态,不依赖 DOM 编辑器内部实现。\r\n const unsubscribe = options.editor.subscribeCommandStateChange(renderStates);\r\n\r\n return {\r\n update,\r\n destroy() {\r\n if (destroyed) return;\r\n\r\n destroyed = true;\r\n unsubscribe();\r\n closeGroupMenu();\r\n container.removeEventListener('click', handleClick);\r\n groupMenuElement.removeEventListener('click', handleClick);\r\n document.removeEventListener('click', handleDocumentClick);\r\n document.removeEventListener('keydown', handleKeyDown);\r\n if (enableTooltip) {\r\n container.removeEventListener('mouseover', handleTooltipTarget);\r\n container.removeEventListener('mouseout', handleTooltipLeave);\r\n }\r\n clearToolbarPressedState();\r\n clearGroupMenuPressedState();\r\n container.removeEventListener('focusout', hideTooltip);\r\n tooltipElement.remove();\r\n groupMenuElement.remove();\r\n container.classList.remove('bridgerte__toolbar');\r\n delete container.dataset.placement;\r\n container.textContent = '';\r\n container.removeAttribute('role');\r\n container.removeAttribute('aria-label');\r\n }\r\n };\r\n}\r\n","const invisibleTextPattern = /[\\u200B-\\u200D\\uFEFF]/g;\n\n/*\n * HTML 片段里这些元素没有 textContent 也应算有效内容。\n * 纯排版容器不放进来,避免 `<p><br></p>`、空列表项这类编辑器占位被误判为非空。\n */\nconst meaningfulHtmlContentSelector = [\n 'img',\n 'video',\n 'audio',\n 'iframe',\n 'table',\n 'pre',\n 'code',\n 'hr',\n 'figure',\n 'canvas',\n 'svg',\n 'math',\n '[data-type=\"mention\"]'\n].join(',');\n\nconst normalizeHtmlText = (text: string | null | undefined): string => (\n text?.replace(invisibleTextPattern, '').trim() ?? ''\n);\n\n/**\n * 判断一段 HTML 片段是否包含可保存的富文本内容。\n *\n * 这个工具会使用浏览器 template 解析 HTML,适合只拿到 HTML 字符串的业务表单;如果已经持有\n * `EditorContent`,优先使用 core 的 `isEditorContentEmpty()`,避免额外 DOM parse。\n */\nexport const hasMeaningfulHtmlContent = (html: string): boolean => {\n if (!normalizeHtmlText(html)) return false;\n\n const template = document.createElement('template');\n\n template.innerHTML = html;\n\n return Boolean(\n normalizeHtmlText(template.content.textContent)\n || template.content.querySelector(meaningfulHtmlContentSelector)\n );\n};\n"],"names":["defaultMenuUiIcons","getEnabledGroupMenuButtons","menuElement","button","focusToolbarGroupMenuItem","index","nextButton","handleToolbarGroupMenuKeyDown","event","returnButton","executeToolbarButton","closeGroupMenu","buttons","activeIndex","focusByOffset","offset","nextIndex","_a","_b","toolbarGroupButtonSelector","toolbarButtonSelector","toolbarExecutableButtonSelector","getGroupMenuState","items","commandStates","itemStates","item","getMenuStateForItem","state","renderToolbarButton","groupElement","icons","enableTooltip","iconSvg","defaultMenuIcons","appendMenuIcon","renderToolbarGroupButton","toolbarItem","groupState","indicator","renderToolbar","toolbarElement","toolbarItems","separatorElement","renderToolbarGroupMenu","groupMenuState","labelElement","syncToolbarGroupButtonState","container","open","findToolbarGroupMenuItem","groupKey","toolbarTooltipOffsetPx","toolbarGroupMenuOffsetPx","canUseHoverTooltip","getToolbarButtonFromTarget","target","getToolbarGroupButtonFromTarget","createRichTextToolbar","options","placement","menuSchema","resolveMenuSchemaForDom","defaultMenuSchema","resolveToolbarMenu","executableMenuItems","tooltipElement","groupMenuElement","overlayHost","destroyed","latestCommandStates","groupMenuFloatingLayer","clearToolbarPressedState","bindTouchPressedState","clearGroupMenuPressedState","mountToolbarOverlays","closeGroupMenuFloatingLayer","openGroupMenuFloatingLayer","createFloatingLayer","renderStates","states","nextGroupItem","update","hideTooltip","showTooltip","tooltipText","buttonRect","handleTooltipTarget","handleTooltipLeave","relatedTarget","menuItem","getPayloadPanelCurrentValues","toggleGroupMenu","shouldFocusMenu","groupItem","handleClick","groupButton","handleDocumentClick","handleKeyDown","unsubscribe","invisibleTextPattern","meaningfulHtmlContentSelector","normalizeHtmlText","text","hasMeaningfulHtmlContent","html","template"],"mappings":"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"}