claudecode-omc 5.9.1 → 5.11.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (103) hide show
  1. package/.local/settings/settings.json +8 -0
  2. package/.omc-curation/governance.json +3 -0
  3. package/.omc-curation/sources.lock.json +5 -0
  4. package/README.md +10 -1
  5. package/bundled/manifest.json +2 -1
  6. package/bundled/upstream/impeccable/.omc-source/bundle.json +20 -0
  7. package/bundled/upstream/impeccable/.omc-source/provenance.json +105 -0
  8. package/bundled/upstream/impeccable/agents/impeccable-manual-edit-applier.md +97 -0
  9. package/bundled/upstream/impeccable/skills/impeccable/SKILL.md +168 -0
  10. package/bundled/upstream/impeccable/skills/impeccable/reference/adapt.md +311 -0
  11. package/bundled/upstream/impeccable/skills/impeccable/reference/animate.md +201 -0
  12. package/bundled/upstream/impeccable/skills/impeccable/reference/audit.md +133 -0
  13. package/bundled/upstream/impeccable/skills/impeccable/reference/bolder.md +113 -0
  14. package/bundled/upstream/impeccable/skills/impeccable/reference/brand.md +108 -0
  15. package/bundled/upstream/impeccable/skills/impeccable/reference/clarify.md +288 -0
  16. package/bundled/upstream/impeccable/skills/impeccable/reference/codex.md +105 -0
  17. package/bundled/upstream/impeccable/skills/impeccable/reference/colorize.md +257 -0
  18. package/bundled/upstream/impeccable/skills/impeccable/reference/craft.md +123 -0
  19. package/bundled/upstream/impeccable/skills/impeccable/reference/critique.md +767 -0
  20. package/bundled/upstream/impeccable/skills/impeccable/reference/delight.md +302 -0
  21. package/bundled/upstream/impeccable/skills/impeccable/reference/distill.md +111 -0
  22. package/bundled/upstream/impeccable/skills/impeccable/reference/document.md +429 -0
  23. package/bundled/upstream/impeccable/skills/impeccable/reference/extract.md +69 -0
  24. package/bundled/upstream/impeccable/skills/impeccable/reference/harden.md +347 -0
  25. package/bundled/upstream/impeccable/skills/impeccable/reference/hooks.md +88 -0
  26. package/bundled/upstream/impeccable/skills/impeccable/reference/init.md +172 -0
  27. package/bundled/upstream/impeccable/skills/impeccable/reference/interaction-design.md +189 -0
  28. package/bundled/upstream/impeccable/skills/impeccable/reference/layout.md +161 -0
  29. package/bundled/upstream/impeccable/skills/impeccable/reference/live.md +718 -0
  30. package/bundled/upstream/impeccable/skills/impeccable/reference/onboard.md +234 -0
  31. package/bundled/upstream/impeccable/skills/impeccable/reference/optimize.md +258 -0
  32. package/bundled/upstream/impeccable/skills/impeccable/reference/overdrive.md +130 -0
  33. package/bundled/upstream/impeccable/skills/impeccable/reference/polish.md +241 -0
  34. package/bundled/upstream/impeccable/skills/impeccable/reference/product.md +60 -0
  35. package/bundled/upstream/impeccable/skills/impeccable/reference/quieter.md +99 -0
  36. package/bundled/upstream/impeccable/skills/impeccable/reference/shape.md +165 -0
  37. package/bundled/upstream/impeccable/skills/impeccable/reference/typeset.md +279 -0
  38. package/bundled/upstream/impeccable/skills/impeccable/scripts/command-metadata.json +94 -0
  39. package/bundled/upstream/impeccable/skills/impeccable/scripts/context-signals.mjs +225 -0
  40. package/bundled/upstream/impeccable/skills/impeccable/scripts/context.mjs +280 -0
  41. package/bundled/upstream/impeccable/skills/impeccable/scripts/critique-storage.mjs +242 -0
  42. package/bundled/upstream/impeccable/skills/impeccable/scripts/detect-csp.mjs +198 -0
  43. package/bundled/upstream/impeccable/skills/impeccable/scripts/detect.mjs +21 -0
  44. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/browser/injected/index.mjs +1735 -0
  45. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/cli/main.mjs +244 -0
  46. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/detect-antipatterns-browser.js +4907 -0
  47. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/detect-antipatterns.mjs +43 -0
  48. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/engines/browser/detect-url.mjs +252 -0
  49. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/engines/regex/detect-text.mjs +552 -0
  50. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/engines/static-html/css-cascade.mjs +1013 -0
  51. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/engines/static-html/detect-html.mjs +208 -0
  52. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/engines/visual/screenshot-contrast.mjs +189 -0
  53. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/findings.mjs +12 -0
  54. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/node/file-system.mjs +198 -0
  55. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/profile/profiler.mjs +166 -0
  56. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/registry/antipatterns.mjs +419 -0
  57. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/rules/checks.mjs +2671 -0
  58. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/shared/color.mjs +124 -0
  59. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/shared/constants.mjs +101 -0
  60. package/bundled/upstream/impeccable/skills/impeccable/scripts/detector/shared/page.mjs +7 -0
  61. package/bundled/upstream/impeccable/skills/impeccable/scripts/hook-admin.mjs +574 -0
  62. package/bundled/upstream/impeccable/skills/impeccable/scripts/hook-before-edit.mjs +473 -0
  63. package/bundled/upstream/impeccable/skills/impeccable/scripts/hook-lib.mjs +1286 -0
  64. package/bundled/upstream/impeccable/skills/impeccable/scripts/hook.mjs +61 -0
  65. package/bundled/upstream/impeccable/skills/impeccable/scripts/lib/design-parser.mjs +835 -0
  66. package/bundled/upstream/impeccable/skills/impeccable/scripts/lib/impeccable-paths.mjs +126 -0
  67. package/bundled/upstream/impeccable/skills/impeccable/scripts/lib/is-generated.mjs +69 -0
  68. package/bundled/upstream/impeccable/skills/impeccable/scripts/live/browser-script-parts.mjs +49 -0
  69. package/bundled/upstream/impeccable/skills/impeccable/scripts/live/completion.mjs +19 -0
  70. package/bundled/upstream/impeccable/skills/impeccable/scripts/live/event-validation.mjs +137 -0
  71. package/bundled/upstream/impeccable/skills/impeccable/scripts/live/insert-ui.mjs +458 -0
  72. package/bundled/upstream/impeccable/skills/impeccable/scripts/live/manual-apply.mjs +939 -0
  73. package/bundled/upstream/impeccable/skills/impeccable/scripts/live/manual-edit-routes.mjs +357 -0
  74. package/bundled/upstream/impeccable/skills/impeccable/scripts/live/manual-edits-buffer.mjs +152 -0
  75. package/bundled/upstream/impeccable/skills/impeccable/scripts/live/session-store.mjs +289 -0
  76. package/bundled/upstream/impeccable/skills/impeccable/scripts/live/svelte-component.mjs +826 -0
  77. package/bundled/upstream/impeccable/skills/impeccable/scripts/live/sveltekit-adapter.mjs +274 -0
  78. package/bundled/upstream/impeccable/skills/impeccable/scripts/live/ui-core.mjs +180 -0
  79. package/bundled/upstream/impeccable/skills/impeccable/scripts/live/vocabulary.mjs +36 -0
  80. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-accept.mjs +812 -0
  81. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-browser-dom.js +146 -0
  82. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-browser-session.js +123 -0
  83. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-browser.js +11086 -0
  84. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-commit-manual-edits.mjs +1241 -0
  85. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-complete.mjs +75 -0
  86. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-copy-edit-agent.mjs +683 -0
  87. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-discard-manual-edits.mjs +51 -0
  88. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-inject.mjs +583 -0
  89. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-insert.mjs +272 -0
  90. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-manual-edit-evidence.mjs +363 -0
  91. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-poll.mjs +379 -0
  92. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-resume.mjs +94 -0
  93. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-server.mjs +1134 -0
  94. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-status.mjs +61 -0
  95. package/bundled/upstream/impeccable/skills/impeccable/scripts/live-wrap.mjs +894 -0
  96. package/bundled/upstream/impeccable/skills/impeccable/scripts/live.mjs +246 -0
  97. package/bundled/upstream/impeccable/skills/impeccable/scripts/modern-screenshot.umd.js +14 -0
  98. package/bundled/upstream/impeccable/skills/impeccable/scripts/palette.mjs +633 -0
  99. package/bundled/upstream/impeccable/skills/impeccable/scripts/pin.mjs +214 -0
  100. package/package.json +1 -1
  101. package/src/cli/source.js +6 -0
  102. package/src/config/sources.js +15 -0
  103. package/src/merge/content-patch.js +4 -0
@@ -0,0 +1,458 @@
1
+ /**
2
+ * Pure helpers for live-mode insert UI (browser + tests).
3
+ * Kept separate from live-browser.js so insert logic is unit-testable.
4
+ */
5
+
6
+ export const PLACEHOLDER_DEFAULT_HEIGHT = 80;
7
+ export const PLACEHOLDER_MIN_HEIGHT = 48;
8
+ export const PLACEHOLDER_MIN_WIDTH = 120;
9
+
10
+ /** @typedef {'before' | 'after'} InsertPosition */
11
+ /** @typedef {'row' | 'column'} InsertAxis */
12
+
13
+ /**
14
+ * Infer sibling flow axis from a container's computed layout styles.
15
+ * @param {{ display?: string, flexDirection?: string, gridTemplateColumns?: string, gridAutoFlow?: string }} style
16
+ * @returns {InsertAxis}
17
+ */
18
+ export function detectInsertAxisFromStyle(style) {
19
+ const display = style?.display || 'block';
20
+ if (display.includes('flex')) {
21
+ const dir = style.flexDirection || 'row';
22
+ return dir.startsWith('row') ? 'row' : 'column';
23
+ }
24
+ if (display === 'grid' || display === 'inline-grid') {
25
+ const flow = style.gridAutoFlow || 'row';
26
+ if (flow.includes('column')) return 'column';
27
+ const cols = (style.gridTemplateColumns || '').trim();
28
+ if (cols && cols !== 'none') {
29
+ const colCount = cols.split(/\s+/).filter(Boolean).length;
30
+ if (colCount > 1) return 'row';
31
+ }
32
+ return 'row';
33
+ }
34
+ return 'column';
35
+ }
36
+
37
+ /**
38
+ * Pick insertion side from pointer position against an anchor element box.
39
+ * @param {number} clientX
40
+ * @param {number} clientY
41
+ * @param {{ top: number, left: number, width: number, height: number, bottom?: number, right?: number }} rect
42
+ * @param {InsertAxis} [axis]
43
+ * @returns {InsertPosition}
44
+ */
45
+ export function computeInsertPosition(clientX, clientY, rect, axis = 'column') {
46
+ if (!rect) return 'after';
47
+ if (axis === 'row') {
48
+ if (!Number.isFinite(rect.left) || !Number.isFinite(rect.width) || rect.width <= 0) return 'after';
49
+ const mid = rect.left + rect.width / 2;
50
+ return clientX < mid ? 'before' : 'after';
51
+ }
52
+ if (!Number.isFinite(rect.top) || !Number.isFinite(rect.height) || rect.height <= 0) return 'after';
53
+ const mid = rect.top + rect.height / 2;
54
+ return clientY < mid ? 'before' : 'after';
55
+ }
56
+
57
+ /**
58
+ * Whether Create is allowed for an insert session.
59
+ * Requires a non-empty prompt OR at least one annotation.
60
+ */
61
+ export function canCreateInsert({ prompt, comments, strokes }) {
62
+ const hasPrompt = typeof prompt === 'string' && prompt.trim().length > 0;
63
+ const hasComments = Array.isArray(comments) && comments.length > 0;
64
+ const hasStrokes = Array.isArray(strokes) && strokes.some(
65
+ (s) => Array.isArray(s?.points) && s.points.length >= 2,
66
+ );
67
+ return hasPrompt || hasComments || hasStrokes;
68
+ }
69
+
70
+ /** Tooltip/title when Create is disabled. */
71
+ export function insertCreateDisabledReason({ prompt, comments, strokes }) {
72
+ if (canCreateInsert({ prompt, comments, strokes })) return null;
73
+ return 'Add a prompt or annotate the placeholder to create';
74
+ }
75
+
76
+ /**
77
+ * Fixed-position insert line coordinates (viewport px).
78
+ * @param {{ top: number, left: number, width: number, height: number, bottom?: number, right?: number }} rect
79
+ * @param {InsertPosition} position
80
+ * @param {InsertAxis} [axis]
81
+ */
82
+ export function insertLineCoords(rect, position, axis = 'column') {
83
+ if (axis === 'row') {
84
+ const right = rect.right ?? rect.left + rect.width;
85
+ const x = position === 'before' ? rect.left - 2 : right + 2;
86
+ return { axis: 'row', top: rect.top, left: x, width: 0, height: rect.height };
87
+ }
88
+ const bottom = rect.bottom ?? rect.top + rect.height;
89
+ const y = position === 'before' ? rect.top - 2 : bottom + 2;
90
+ return { axis: 'column', top: y, left: rect.left, width: rect.width, height: 0 };
91
+ }
92
+
93
+ /** Cursor while hovering an insert boundary. */
94
+ export function cursorForInsertAxis(axis) {
95
+ return axis === 'row' ? 'ew-resize' : 'ns-resize';
96
+ }
97
+
98
+ function groupSiblingRows(siblings, rowThreshold = 8) {
99
+ const sorted = [...siblings].sort((a, b) => a.rect.top - b.rect.top || a.rect.left - b.rect.left);
100
+ const rows = [];
101
+ for (const entry of sorted) {
102
+ let placed = false;
103
+ for (const row of rows) {
104
+ if (Math.abs(entry.rect.top - row[0].rect.top) <= rowThreshold) {
105
+ row.push(entry);
106
+ placed = true;
107
+ break;
108
+ }
109
+ }
110
+ if (!placed) rows.push([entry]);
111
+ }
112
+ return rows;
113
+ }
114
+
115
+ function horizontalOverlap(a, b) {
116
+ const left = Math.max(a.left, b.left);
117
+ const right = Math.min(a.right ?? a.left + a.width, b.right ?? b.left + b.width);
118
+ return Math.max(0, right - left);
119
+ }
120
+
121
+ /**
122
+ * Hit-test the gap between adjacent siblings (flex rows, grid columns, stacked blocks).
123
+ * @param {number} clientX
124
+ * @param {number} clientY
125
+ * @param {Array<{ el: unknown, rect: { top: number, left: number, width: number, height: number, bottom?: number, right?: number } }>} siblings
126
+ * @param {{ slop?: number, minOverlap?: number }} [opts]
127
+ */
128
+ export function hitSiblingInsertGap(clientX, clientY, siblings, opts = {}) {
129
+ if (!Array.isArray(siblings) || siblings.length < 2) return null;
130
+ const slop = opts.slop ?? 12;
131
+ const minOverlap = opts.minOverlap ?? 0.25;
132
+
133
+ for (const row of groupSiblingRows(siblings)) {
134
+ if (row.length < 2) continue;
135
+ const sorted = [...row].sort((a, b) => a.rect.left - b.rect.left);
136
+ for (let i = 0; i < sorted.length - 1; i++) {
137
+ const a = sorted[i];
138
+ const b = sorted[i + 1];
139
+ const aRight = a.rect.right ?? a.rect.left + a.rect.width;
140
+ const bLeft = b.rect.left;
141
+ if (bLeft <= aRight) continue;
142
+ const top = Math.max(a.rect.top, b.rect.top);
143
+ const aBottom = a.rect.bottom ?? a.rect.top + a.rect.height;
144
+ const bBottom = b.rect.bottom ?? b.rect.top + b.rect.height;
145
+ const bottom = Math.min(aBottom, bBottom);
146
+ const span = bottom - top;
147
+ const minH = Math.min(a.rect.height, b.rect.height);
148
+ if (span < minH * minOverlap) continue;
149
+
150
+ const inX = clientX >= aRight - slop && clientX <= bLeft + slop;
151
+ const inY = clientY >= top - slop && clientY <= bottom + slop;
152
+ if (!inX || !inY) continue;
153
+
154
+ const midX = (aRight + bLeft) / 2;
155
+ return {
156
+ anchor: b.el,
157
+ position: 'before',
158
+ axis: 'row',
159
+ line: { axis: 'row', left: midX, top, width: 0, height: span },
160
+ };
161
+ }
162
+ }
163
+
164
+ const sortedCol = [...siblings].sort((a, b) => a.rect.top - b.rect.top || a.rect.left - b.rect.left);
165
+ for (let i = 0; i < sortedCol.length - 1; i++) {
166
+ const a = sortedCol[i];
167
+ const b = sortedCol[i + 1];
168
+ const overlap = horizontalOverlap(a.rect, b.rect);
169
+ const minW = Math.min(a.rect.width, b.rect.width);
170
+ if (overlap < minW * minOverlap) continue;
171
+
172
+ const aBottom = a.rect.bottom ?? a.rect.top + a.rect.height;
173
+ const gapTop = aBottom;
174
+ const gapBottom = b.rect.top;
175
+ if (gapBottom <= gapTop) continue;
176
+
177
+ const overlapLeft = Math.max(a.rect.left, b.rect.left);
178
+ const overlapRight = Math.min(
179
+ a.rect.right ?? a.rect.left + a.rect.width,
180
+ b.rect.right ?? b.rect.left + b.rect.width,
181
+ );
182
+ const inY = clientY >= gapTop - slop && clientY <= gapBottom + slop;
183
+ const inX = clientX >= overlapLeft - slop && clientX <= overlapRight + slop;
184
+ if (!inY || !inX) continue;
185
+
186
+ const midY = (gapTop + gapBottom) / 2;
187
+ return {
188
+ anchor: b.el,
189
+ position: 'before',
190
+ axis: 'column',
191
+ line: { axis: 'column', top: midY, left: overlapLeft, width: overlap, height: 0 },
192
+ };
193
+ }
194
+
195
+ return null;
196
+ }
197
+
198
+ /**
199
+ * Resolve insert hover target, side, axis, and indicator line for the pointer.
200
+ */
201
+ export function resolveInsertHover({ clientX, clientY, target, rect, axis, siblings }) {
202
+ const gap = hitSiblingInsertGap(clientX, clientY, siblings);
203
+ if (gap) return gap;
204
+
205
+ const position = computeInsertPosition(clientX, clientY, rect, axis);
206
+ const line = insertLineCoords(rect, position, axis);
207
+ return { anchor: target, position, axis, line };
208
+ }
209
+
210
+ /**
211
+ * How the in-flow placeholder should participate in layout.
212
+ * Prefer implicit sizing (flex / %) so row inserts don't inherit the full parent width in px.
213
+ * @returns {{ kind: 'flex', flex: string, minWidth: number } | { kind: 'percent' } | { kind: 'auto' } | { kind: 'explicit', width: number }}
214
+ */
215
+ export function placeholderSizing({ axis, parentDisplay, parentWidth, anchorFlex }) {
216
+ const display = parentDisplay || 'block';
217
+ const w = Number.isFinite(parentWidth) ? parentWidth : 0;
218
+
219
+ if (axis === 'row') {
220
+ if (display.includes('flex')) {
221
+ const flex = anchorFlex && anchorFlex !== 'none' && anchorFlex !== '0 1 auto'
222
+ ? anchorFlex
223
+ : '1 1 0';
224
+ return { kind: 'flex', flex, minWidth: 0 };
225
+ }
226
+ if (display === 'grid' || display === 'inline-grid') {
227
+ return { kind: 'auto' };
228
+ }
229
+ }
230
+
231
+ if (w >= PLACEHOLDER_MIN_WIDTH) {
232
+ return { kind: 'percent' };
233
+ }
234
+
235
+ return {
236
+ kind: 'explicit',
237
+ width: Math.max(PLACEHOLDER_MIN_WIDTH, w || PLACEHOLDER_MIN_WIDTH),
238
+ };
239
+ }
240
+
241
+ /** Width kinds that need materializing to px before edge-resize. */
242
+ export function placeholderWidthIsImplicit(kind) {
243
+ return kind === 'flex' || kind === 'percent' || kind === 'auto';
244
+ }
245
+
246
+ /**
247
+ * Clamp user-resized placeholder dimensions.
248
+ */
249
+ export function clampPlaceholderSize(width, height, parentWidth, opts = {}) {
250
+ const minW = opts.minWidth ?? PLACEHOLDER_MIN_WIDTH;
251
+ const minH = opts.minHeight ?? PLACEHOLDER_MIN_HEIGHT;
252
+ const maxW = opts.maxWidth ?? Math.max(minW, parentWidth || minW);
253
+ return {
254
+ width: Math.min(maxW, Math.max(minW, Math.round(width))),
255
+ height: Math.max(minH, Math.round(height)),
256
+ };
257
+ }
258
+
259
+ /** CSS cursor for a placeholder edge resize handle. */
260
+ export function cursorForPlaceholderEdge(edge) {
261
+ if (edge === 'n' || edge === 's') return 'ns-resize';
262
+ if (edge === 'e' || edge === 'w') return 'ew-resize';
263
+ return 'default';
264
+ }
265
+
266
+ /**
267
+ * Compute placeholder box after dragging one edge (in-flow margins shift for n/w).
268
+ * @param {{ width: number, height: number, marginLeft?: number, marginTop?: number }} start
269
+ * @param {'n'|'e'|'s'|'w'} edge
270
+ * @param {number} dx pointer delta X since drag start
271
+ * @param {number} dy pointer delta Y since drag start
272
+ * @param {number} parentWidth
273
+ */
274
+ export function resizePlaceholderFromEdge(start, edge, dx, dy, parentWidth, opts = {}) {
275
+ const base = {
276
+ width: start.width,
277
+ height: start.height,
278
+ marginLeft: start.marginLeft ?? 0,
279
+ marginTop: start.marginTop ?? 0,
280
+ };
281
+ if (edge === 'e') base.width = start.width + dx;
282
+ else if (edge === 'w') {
283
+ base.width = start.width - dx;
284
+ base.marginLeft = start.marginLeft + dx;
285
+ } else if (edge === 's') base.height = start.height + dy;
286
+ else if (edge === 'n') {
287
+ base.height = start.height - dy;
288
+ base.marginTop = start.marginTop + dy;
289
+ }
290
+
291
+ const clamped = clampPlaceholderSize(base.width, base.height, parentWidth, opts);
292
+ if (edge === 'w') {
293
+ base.marginLeft = start.marginLeft + start.width - clamped.width;
294
+ } else if (edge === 'n') {
295
+ base.marginTop = start.marginTop + start.height - clamped.height;
296
+ }
297
+
298
+ return {
299
+ width: clamped.width,
300
+ height: clamped.height,
301
+ marginLeft: Math.round(base.marginLeft),
302
+ marginTop: Math.round(base.marginTop),
303
+ };
304
+ }
305
+
306
+ /** Pick and insert toggles are independent but turning one ON turns the other OFF. */
307
+ export function applyPickToggle(pickActive, insertActive) {
308
+ const nextPick = !pickActive;
309
+ return {
310
+ pickActive: nextPick,
311
+ insertActive: nextPick ? false : insertActive,
312
+ };
313
+ }
314
+
315
+ export function applyInsertToggle(pickActive, insertActive) {
316
+ const nextInsert = !insertActive;
317
+ return {
318
+ pickActive: nextInsert ? false : pickActive,
319
+ insertActive: nextInsert,
320
+ };
321
+ }
322
+
323
+ /**
324
+ * Build the browser generate payload for insert mode.
325
+ */
326
+ export function buildInsertGeneratePayload({
327
+ id,
328
+ count,
329
+ pageUrl,
330
+ anchorContext,
331
+ position,
332
+ placeholder,
333
+ freeformPrompt,
334
+ comments,
335
+ strokes,
336
+ screenshotPath,
337
+ }) {
338
+ const payload = {
339
+ type: 'generate',
340
+ mode: 'insert',
341
+ id,
342
+ count,
343
+ pageUrl,
344
+ insert: {
345
+ position,
346
+ anchor: anchorContext,
347
+ },
348
+ placeholder,
349
+ freeformPrompt: freeformPrompt?.trim() || undefined,
350
+ };
351
+ if (comments?.length) payload.comments = comments;
352
+ if (strokes?.length) payload.strokes = strokes;
353
+ if (screenshotPath) payload.screenshotPath = screenshotPath;
354
+ return payload;
355
+ }
356
+
357
+ /**
358
+ * Whether a variant wrapper is currently shown (handles `hidden` and display:none).
359
+ * @param {{ hidden?: boolean, style?: { display?: string } } | null | undefined} el
360
+ */
361
+ export function isVariantShown(el) {
362
+ if (!el) return false;
363
+ if (el.hidden) return false;
364
+ if (el.style?.display === 'none') return false;
365
+ return true;
366
+ }
367
+
368
+ /**
369
+ * Show or hide a variant wrapper for cycling.
370
+ * @param {{ hidden?: boolean, style?: { display?: string }, removeAttribute?: (name: string) => void, setAttribute?: (name: string, value?: string) => void } | null | undefined} el
371
+ * @param {boolean} shown
372
+ */
373
+ export function setVariantShown(el, shown) {
374
+ if (!el) return;
375
+ if (shown) {
376
+ el.removeAttribute?.('hidden');
377
+ if (el.style) el.style.display = '';
378
+ } else {
379
+ el.setAttribute?.('hidden', '');
380
+ if (el.style) el.style.display = 'none';
381
+ }
382
+ }
383
+
384
+ /**
385
+ * Pick the best live anchor during an insert session (placeholder until variants land).
386
+ * @param {{
387
+ * wrapper?: unknown,
388
+ * variantCount?: number,
389
+ * visibleVariant?: number,
390
+ * placeholder?: unknown,
391
+ * insertAnchor?: unknown,
392
+ * pickVariantContent?: (wrapper: unknown, index: number) => unknown,
393
+ * }} opts
394
+ */
395
+ export function resolveInsertSessionAnchor(opts) {
396
+ const {
397
+ wrapper,
398
+ variantCount = 0,
399
+ visibleVariant = 0,
400
+ placeholder,
401
+ insertAnchor,
402
+ pickVariantContent,
403
+ } = opts || {};
404
+ if (wrapper && variantCount > 0 && visibleVariant > 0 && pickVariantContent) {
405
+ const vis = pickVariantContent(wrapper, visibleVariant);
406
+ if (vis) return vis;
407
+ }
408
+ return placeholder || insertAnchor || null;
409
+ }
410
+
411
+ /**
412
+ * Snapshot placeholder geometry + anchor fingerprint so HMR can recreate the box.
413
+ * @param {{
414
+ * tagName?: string,
415
+ * className?: string,
416
+ * textContent?: string,
417
+ * }} anchor
418
+ * @param {{
419
+ * offsetWidth?: number,
420
+ * offsetHeight?: number,
421
+ * style?: { marginLeft?: string, marginTop?: string },
422
+ * }} placeholder
423
+ * @param {{ position: 'before' | 'after', layoutAxis?: 'row' | 'column' }} meta
424
+ */
425
+ export function buildInsertPlaceholderSnapshot(anchor, placeholder, { position, layoutAxis }) {
426
+ return {
427
+ width: Math.round(placeholder.offsetWidth || 0),
428
+ height: Math.round(placeholder.offsetHeight || PLACEHOLDER_DEFAULT_HEIGHT),
429
+ marginLeft: parseFloat(placeholder.style?.marginLeft || '') || 0,
430
+ marginTop: parseFloat(placeholder.style?.marginTop || '') || 0,
431
+ position,
432
+ layoutAxis: layoutAxis || 'column',
433
+ anchorTag: anchor.tagName || 'DIV',
434
+ anchorClasses: anchor.className || '',
435
+ anchorText: (anchor.textContent || '').trim().slice(0, 120),
436
+ };
437
+ }
438
+
439
+ /**
440
+ * Re-find an insert anchor after framework HMR replaced the live DOM node.
441
+ * @param {Pick<Document, 'body' | 'querySelectorAll'>} doc
442
+ * @param {ReturnType<typeof buildInsertPlaceholderSnapshot> | null | undefined} snapshot
443
+ * @param {Element | null | undefined} liveAnchor
444
+ */
445
+ export function findInsertAnchorInDom(doc, snapshot, liveAnchor = null) {
446
+ if (liveAnchor && doc.body.contains(liveAnchor)) return liveAnchor;
447
+ if (!snapshot) return null;
448
+ const tag = (snapshot.anchorTag || 'div').toLowerCase();
449
+ const cls = (snapshot.anchorClasses || '').split(/\s+/).filter(Boolean)[0];
450
+ const needle = snapshot.anchorText || '';
451
+ const sel = cls ? `${tag}.${cls}` : tag;
452
+ const candidates = doc.querySelectorAll(sel);
453
+ for (const candidate of candidates) {
454
+ if (needle && !(candidate.textContent || '').includes(needle.slice(0, 40))) continue;
455
+ return candidate;
456
+ }
457
+ return null;
458
+ }