@bastani/atomic 0.8.20 → 0.8.21-0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +6 -0
- package/dist/builtin/intercom/package.json +1 -1
- package/dist/builtin/mcp/package.json +1 -1
- package/dist/builtin/subagents/agents/code-simplifier.md +78 -22
- package/dist/builtin/subagents/agents/debugger.md +4 -3
- package/dist/builtin/subagents/package.json +1 -1
- package/dist/builtin/web-access/package.json +1 -1
- package/dist/builtin/workflows/CHANGELOG.md +19 -0
- package/dist/builtin/workflows/package.json +1 -1
- package/dist/builtin/workflows/skills/create-spec/SKILL.md +169 -125
- package/dist/builtin/workflows/skills/impeccable/SKILL.md +89 -80
- package/dist/builtin/workflows/skills/impeccable/agents/impeccable_asset_producer.toml +92 -0
- package/dist/builtin/workflows/skills/impeccable/agents/impeccable_manual_edit_applier.toml +95 -0
- package/dist/builtin/workflows/skills/impeccable/agents/openai.yaml +4 -0
- package/dist/builtin/workflows/skills/impeccable/reference/adapt.md +122 -1
- package/dist/builtin/workflows/skills/impeccable/reference/animate.md +38 -12
- package/dist/builtin/workflows/skills/impeccable/reference/audit.md +5 -5
- package/dist/builtin/workflows/skills/impeccable/reference/bolder.md +7 -7
- package/dist/builtin/workflows/skills/impeccable/reference/brand.md +4 -14
- package/dist/builtin/workflows/skills/impeccable/reference/clarify.md +115 -1
- package/dist/builtin/workflows/skills/impeccable/reference/codex.md +3 -3
- package/dist/builtin/workflows/skills/impeccable/reference/colorize.md +109 -6
- package/dist/builtin/workflows/skills/impeccable/reference/craft.md +7 -7
- package/dist/builtin/workflows/skills/impeccable/reference/critique.md +623 -94
- package/dist/builtin/workflows/skills/impeccable/reference/delight.md +2 -2
- package/dist/builtin/workflows/skills/impeccable/reference/distill.md +2 -2
- package/dist/builtin/workflows/skills/impeccable/reference/document.md +16 -14
- package/dist/builtin/workflows/skills/impeccable/reference/extract.md +1 -1
- package/dist/builtin/workflows/skills/impeccable/reference/harden.md +1 -1
- package/dist/builtin/workflows/skills/impeccable/reference/init.md +172 -0
- package/dist/builtin/workflows/skills/impeccable/reference/interaction-design.md +0 -6
- package/dist/builtin/workflows/skills/impeccable/reference/layout.md +33 -13
- package/dist/builtin/workflows/skills/impeccable/reference/live.md +96 -19
- package/dist/builtin/workflows/skills/impeccable/reference/onboard.md +1 -1
- package/dist/builtin/workflows/skills/impeccable/reference/optimize.md +1 -1
- package/dist/builtin/workflows/skills/impeccable/reference/overdrive.md +1 -1
- package/dist/builtin/workflows/skills/impeccable/reference/polish.md +3 -4
- package/dist/builtin/workflows/skills/impeccable/reference/product.md +1 -3
- package/dist/builtin/workflows/skills/impeccable/reference/quieter.md +2 -2
- package/dist/builtin/workflows/skills/impeccable/reference/shape.md +5 -5
- package/dist/builtin/workflows/skills/impeccable/reference/typeset.md +158 -3
- package/dist/builtin/workflows/skills/impeccable/scripts/cleanup-deprecated.mjs +1 -1
- package/dist/builtin/workflows/skills/impeccable/scripts/command-metadata.json +2 -2
- package/dist/builtin/workflows/skills/impeccable/scripts/context-signals.mjs +225 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/context.mjs +266 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/critique-storage.mjs +17 -1
- package/dist/builtin/workflows/skills/impeccable/scripts/design-parser.mjs +16 -1
- package/dist/builtin/workflows/skills/impeccable/scripts/detect.mjs +21 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/detector/browser/injected/index.mjs +1725 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/detector/cli/main.mjs +244 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/detector/detect-antipatterns-browser.js +4543 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/detector/detect-antipatterns.mjs +43 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/detector/engines/browser/detect-url.mjs +252 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/detector/engines/regex/detect-text.mjs +535 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/detector/engines/static-html/css-cascade.mjs +986 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/detector/engines/static-html/detect-html.mjs +208 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/detector/engines/visual/screenshot-contrast.mjs +189 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/detector/findings.mjs +12 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/detector/node/file-system.mjs +198 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/detector/profile/profiler.mjs +166 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/detector/registry/antipatterns.mjs +419 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/detector/rules/checks.mjs +2316 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/detector/shared/color.mjs +124 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/detector/shared/constants.mjs +101 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/detector/shared/page.mjs +7 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/impeccable-paths.mjs +17 -1
- package/dist/builtin/workflows/skills/impeccable/scripts/is-generated.mjs +2 -2
- package/dist/builtin/workflows/skills/impeccable/scripts/live-accept.mjs +139 -96
- package/dist/builtin/workflows/skills/impeccable/scripts/live-browser.js +4491 -526
- package/dist/builtin/workflows/skills/impeccable/scripts/live-commit-manual-edits.mjs +1241 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/live-copy-edit-agent.mjs +683 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/live-discard-manual-edits.mjs +51 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/live-event-validation.mjs +136 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/live-inject.mjs +22 -9
- package/dist/builtin/workflows/skills/impeccable/scripts/live-insert-ui.mjs +458 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/live-insert.mjs +232 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/live-manual-edit-evidence.mjs +363 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/live-manual-edits-buffer.mjs +152 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/live-poll.mjs +288 -110
- package/dist/builtin/workflows/skills/impeccable/scripts/live-resume.mjs +47 -1
- package/dist/builtin/workflows/skills/impeccable/scripts/live-server.mjs +1443 -100
- package/dist/builtin/workflows/skills/impeccable/scripts/live-session-store.mjs +17 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/live-status.mjs +17 -3
- package/dist/builtin/workflows/skills/impeccable/scripts/live-wrap.mjs +216 -6
- package/dist/builtin/workflows/skills/impeccable/scripts/live.mjs +2 -3
- package/dist/builtin/workflows/skills/impeccable/scripts/palette.mjs +633 -0
- package/dist/builtin/workflows/skills/impeccable/scripts/pin.mjs +1 -1
- package/dist/builtin/workflows/src/extension/index.ts +67 -3
- package/dist/builtin/workflows/src/extension/render-result.ts +26 -1
- package/dist/builtin/workflows/src/runs/foreground/executor.ts +227 -3
- package/dist/builtin/workflows/src/runs/foreground/stage-runner.ts +94 -7
- package/dist/builtin/workflows/src/shared/stage-prompt.ts +326 -0
- package/dist/builtin/workflows/src/shared/stage-ui-broker.ts +62 -7
- package/dist/builtin/workflows/src/shared/store-types.ts +43 -0
- package/dist/builtin/workflows/src/shared/store.ts +37 -0
- package/dist/builtin/workflows/src/tui/chat-surface-message.ts +22 -4
- package/dist/builtin/workflows/src/tui/graph-view.ts +47 -0
- package/dist/builtin/workflows/src/tui/overlay-adapter.ts +43 -1
- package/dist/builtin/workflows/src/tui/run-detail.ts +10 -4
- package/dist/builtin/workflows/src/tui/stage-chat-view.ts +117 -15
- package/dist/builtin/workflows/src/tui/workflow-attach-pane.ts +9 -0
- package/dist/core/skills.d.ts.map +1 -1
- package/dist/core/skills.js +2 -5
- package/dist/core/skills.js.map +1 -1
- package/dist/core/system-prompt.d.ts.map +1 -1
- package/dist/core/system-prompt.js +11 -29
- package/dist/core/system-prompt.js.map +1 -1
- package/dist/index.d.ts +1 -0
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +3 -0
- package/dist/index.js.map +1 -1
- package/docs/quickstart.md +1 -2
- package/package.json +4 -4
- package/dist/builtin/workflows/skills/impeccable/reference/cognitive-load.md +0 -106
- package/dist/builtin/workflows/skills/impeccable/reference/color-and-contrast.md +0 -105
- package/dist/builtin/workflows/skills/impeccable/reference/heuristics-scoring.md +0 -234
- package/dist/builtin/workflows/skills/impeccable/reference/motion-design.md +0 -109
- package/dist/builtin/workflows/skills/impeccable/reference/personas.md +0 -179
- package/dist/builtin/workflows/skills/impeccable/reference/responsive-design.md +0 -114
- package/dist/builtin/workflows/skills/impeccable/reference/spatial-design.md +0 -100
- package/dist/builtin/workflows/skills/impeccable/reference/teach.md +0 -156
- package/dist/builtin/workflows/skills/impeccable/reference/typography.md +0 -159
- package/dist/builtin/workflows/skills/impeccable/reference/ux-writing.md +0 -107
- package/dist/builtin/workflows/skills/impeccable/scripts/load-context.mjs +0 -141
|
@@ -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
|
+
}
|
|
@@ -0,0 +1,232 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* CLI helper: find an anchor element in source and splice an insert-variant
|
|
3
|
+
* wrapper before or after it (no original variant — net-new content).
|
|
4
|
+
*
|
|
5
|
+
* Usage:
|
|
6
|
+
* node live-insert.mjs --id SESSION_ID --count N --position after \
|
|
7
|
+
* --classes "hero" --tag section [--file path]
|
|
8
|
+
*/
|
|
9
|
+
|
|
10
|
+
import fs from 'node:fs';
|
|
11
|
+
import path from 'node:path';
|
|
12
|
+
import { isGeneratedFile } from './is-generated.mjs';
|
|
13
|
+
import {
|
|
14
|
+
buildSearchQueries,
|
|
15
|
+
findElement,
|
|
16
|
+
findAllElements,
|
|
17
|
+
filterByText,
|
|
18
|
+
findFileWithQuery,
|
|
19
|
+
detectCommentSyntax,
|
|
20
|
+
detectStyleMode,
|
|
21
|
+
buildCssAuthoring,
|
|
22
|
+
buildCssSelectorPrefixExamples,
|
|
23
|
+
} from './live-wrap.mjs';
|
|
24
|
+
|
|
25
|
+
const INSERT_POSITIONS = new Set(['before', 'after']);
|
|
26
|
+
|
|
27
|
+
export function isInsertPosition(value) {
|
|
28
|
+
return INSERT_POSITIONS.has(value);
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
export function computeInsertLine(startLine, endLine, position) {
|
|
32
|
+
return position === 'before' ? startLine : endLine + 1;
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
export function buildInsertWrapperLines({ id, count, indent, commentSyntax, isJsx }) {
|
|
36
|
+
const styleContents = isJsx ? 'style={{ display: "contents" }}' : 'style="display: contents"';
|
|
37
|
+
const attrs =
|
|
38
|
+
'data-impeccable-variants="' + id + '" ' +
|
|
39
|
+
'data-impeccable-mode="insert" ' +
|
|
40
|
+
'data-impeccable-variant-count="' + count + '" ' +
|
|
41
|
+
styleContents;
|
|
42
|
+
|
|
43
|
+
if (isJsx) {
|
|
44
|
+
return [
|
|
45
|
+
indent + '<div ' + attrs + '>',
|
|
46
|
+
indent + ' ' + commentSyntax.open + ' impeccable-variants-start ' + id + ' ' + commentSyntax.close,
|
|
47
|
+
indent + ' ' + commentSyntax.open + ' Variants: insert below this line ' + commentSyntax.close,
|
|
48
|
+
indent + ' ' + commentSyntax.open + ' impeccable-variants-end ' + id + ' ' + commentSyntax.close,
|
|
49
|
+
indent + '</div>',
|
|
50
|
+
];
|
|
51
|
+
}
|
|
52
|
+
|
|
53
|
+
return [
|
|
54
|
+
indent + commentSyntax.open + ' impeccable-variants-start ' + id + ' ' + commentSyntax.close,
|
|
55
|
+
indent + '<div ' + attrs + '>',
|
|
56
|
+
indent + ' ' + commentSyntax.open + ' Variants: insert below this line ' + commentSyntax.close,
|
|
57
|
+
indent + '</div>',
|
|
58
|
+
indent + commentSyntax.open + ' impeccable-variants-end ' + id + ' ' + commentSyntax.close,
|
|
59
|
+
];
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
function argVal(args, flag) {
|
|
63
|
+
const idx = args.indexOf(flag);
|
|
64
|
+
return idx !== -1 && idx + 1 < args.length ? args[idx + 1] : null;
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
function resolveElementMatch({ lines, queries, tag, text }) {
|
|
68
|
+
if (text) {
|
|
69
|
+
const candidates = [];
|
|
70
|
+
for (const q of queries) {
|
|
71
|
+
const all = findAllElements(lines, q, tag);
|
|
72
|
+
for (const c of all) {
|
|
73
|
+
if (!candidates.some((x) => x.startLine === c.startLine)) candidates.push(c);
|
|
74
|
+
}
|
|
75
|
+
if (candidates.length === 1) break;
|
|
76
|
+
}
|
|
77
|
+
if (candidates.length === 0) return { error: 'element_not_found' };
|
|
78
|
+
if (candidates.length === 1) return { match: candidates[0] };
|
|
79
|
+
const filtered = filterByText(candidates, lines, text);
|
|
80
|
+
if (filtered.length === 1) return { match: filtered[0] };
|
|
81
|
+
if (filtered.length === 0) return { match: candidates[0] };
|
|
82
|
+
return { error: 'element_ambiguous', candidates: filtered };
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
for (const q of queries) {
|
|
86
|
+
const match = findElement(lines, q, tag);
|
|
87
|
+
if (match) return { match };
|
|
88
|
+
}
|
|
89
|
+
return { error: 'element_not_found' };
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
export async function insertCli() {
|
|
93
|
+
const args = process.argv.slice(2);
|
|
94
|
+
|
|
95
|
+
if (args.includes('--help') || args.includes('-h')) {
|
|
96
|
+
console.log(`Usage: node live-insert.mjs [options]
|
|
97
|
+
|
|
98
|
+
Find an anchor element in source and splice an insert-variant wrapper.
|
|
99
|
+
|
|
100
|
+
Required:
|
|
101
|
+
--id ID Session ID for the variant wrapper
|
|
102
|
+
--count N Number of expected variants (1-8)
|
|
103
|
+
--position POS before | after (relative to the anchor element)
|
|
104
|
+
|
|
105
|
+
Element identification (at least one required):
|
|
106
|
+
--element-id ID HTML id attribute of the anchor element
|
|
107
|
+
--classes A,B,C Comma-separated CSS class names
|
|
108
|
+
--tag TAG Tag name (div, section, etc.)
|
|
109
|
+
--query TEXT Fallback: raw text to search for
|
|
110
|
+
|
|
111
|
+
Optional:
|
|
112
|
+
--file PATH Source file to search in (skips auto-detection)
|
|
113
|
+
--text TEXT Anchor textContent for disambiguation (~80 chars)
|
|
114
|
+
|
|
115
|
+
Output (JSON):
|
|
116
|
+
{ mode: "insert", file, position, insertLine, commentSyntax, styleMode, styleTag, cssAuthoring }`);
|
|
117
|
+
process.exit(0);
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
const id = argVal(args, '--id');
|
|
121
|
+
const count = parseInt(argVal(args, '--count') || '3', 10);
|
|
122
|
+
const position = argVal(args, '--position');
|
|
123
|
+
const elementId = argVal(args, '--element-id');
|
|
124
|
+
const classes = argVal(args, '--classes');
|
|
125
|
+
const tag = argVal(args, '--tag');
|
|
126
|
+
const query = argVal(args, '--query');
|
|
127
|
+
const filePath = argVal(args, '--file');
|
|
128
|
+
const text = argVal(args, '--text');
|
|
129
|
+
|
|
130
|
+
if (!id) { console.error('Missing --id'); process.exit(1); }
|
|
131
|
+
if (!position) { console.error('Missing --position (before | after)'); process.exit(1); }
|
|
132
|
+
if (!isInsertPosition(position)) { console.error('Invalid --position: ' + position); process.exit(1); }
|
|
133
|
+
if (!elementId && !classes && !query) {
|
|
134
|
+
console.error('Need at least one of: --element-id, --classes, --query');
|
|
135
|
+
process.exit(1);
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
const queries = buildSearchQueries(elementId, classes, tag, query);
|
|
139
|
+
const genOpts = { cwd: process.cwd() };
|
|
140
|
+
|
|
141
|
+
let targetFile = filePath;
|
|
142
|
+
if (!targetFile) {
|
|
143
|
+
for (const q of queries) {
|
|
144
|
+
targetFile = findFileWithQuery(q, process.cwd(), genOpts);
|
|
145
|
+
if (targetFile) break;
|
|
146
|
+
}
|
|
147
|
+
if (!targetFile) {
|
|
148
|
+
let generatedHit = null;
|
|
149
|
+
for (const q of queries) {
|
|
150
|
+
generatedHit = findFileWithQuery(q, process.cwd(), { ...genOpts, includeGenerated: true });
|
|
151
|
+
if (generatedHit) break;
|
|
152
|
+
}
|
|
153
|
+
console.error(JSON.stringify({
|
|
154
|
+
error: generatedHit ? 'element_not_in_source' : 'element_not_found',
|
|
155
|
+
fallback: 'agent-driven',
|
|
156
|
+
hint: 'See "Handle fallback" in live.md.',
|
|
157
|
+
}));
|
|
158
|
+
process.exit(1);
|
|
159
|
+
}
|
|
160
|
+
} else if (isGeneratedFile(targetFile, genOpts)) {
|
|
161
|
+
console.error(JSON.stringify({
|
|
162
|
+
error: 'file_is_generated',
|
|
163
|
+
fallback: 'agent-driven',
|
|
164
|
+
file: path.relative(process.cwd(), path.resolve(process.cwd(), targetFile)),
|
|
165
|
+
}));
|
|
166
|
+
process.exit(1);
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
const content = fs.readFileSync(targetFile, 'utf-8');
|
|
170
|
+
const lines = content.split('\n');
|
|
171
|
+
const resolved = resolveElementMatch({ lines, queries, tag, text });
|
|
172
|
+
|
|
173
|
+
if (resolved.error === 'element_ambiguous') {
|
|
174
|
+
console.error(JSON.stringify({
|
|
175
|
+
error: 'element_ambiguous',
|
|
176
|
+
fallback: 'agent-driven',
|
|
177
|
+
file: path.relative(process.cwd(), targetFile),
|
|
178
|
+
candidates: resolved.candidates.map((c) => ({
|
|
179
|
+
startLine: c.startLine + 1,
|
|
180
|
+
endLine: c.endLine + 1,
|
|
181
|
+
})),
|
|
182
|
+
}));
|
|
183
|
+
process.exit(1);
|
|
184
|
+
}
|
|
185
|
+
if (!resolved.match) {
|
|
186
|
+
console.error(JSON.stringify({ error: 'element_not_found', fallback: 'agent-driven' }));
|
|
187
|
+
process.exit(1);
|
|
188
|
+
}
|
|
189
|
+
|
|
190
|
+
const { startLine, endLine } = resolved.match;
|
|
191
|
+
const commentSyntax = detectCommentSyntax(targetFile);
|
|
192
|
+
const styleMode = detectStyleMode(targetFile);
|
|
193
|
+
const isJsx = commentSyntax.open === '{/*';
|
|
194
|
+
const spliceIndex = computeInsertLine(startLine, endLine, position);
|
|
195
|
+
const indent = lines[spliceIndex]?.match(/^(\s*)/)?.[1]
|
|
196
|
+
?? lines[startLine]?.match(/^(\s*)/)?.[1]
|
|
197
|
+
?? '';
|
|
198
|
+
|
|
199
|
+
const wrapperLines = buildInsertWrapperLines({
|
|
200
|
+
id,
|
|
201
|
+
count,
|
|
202
|
+
indent,
|
|
203
|
+
commentSyntax,
|
|
204
|
+
isJsx,
|
|
205
|
+
});
|
|
206
|
+
|
|
207
|
+
const newLines = [
|
|
208
|
+
...lines.slice(0, spliceIndex),
|
|
209
|
+
...wrapperLines,
|
|
210
|
+
...lines.slice(spliceIndex),
|
|
211
|
+
];
|
|
212
|
+
fs.writeFileSync(targetFile, newLines.join('\n'), 'utf-8');
|
|
213
|
+
|
|
214
|
+
const insertLine = spliceIndex + 3;
|
|
215
|
+
|
|
216
|
+
console.log(JSON.stringify({
|
|
217
|
+
mode: 'insert',
|
|
218
|
+
position,
|
|
219
|
+
file: path.relative(process.cwd(), targetFile),
|
|
220
|
+
insertLine: insertLine + 1,
|
|
221
|
+
commentSyntax,
|
|
222
|
+
styleMode: styleMode.mode,
|
|
223
|
+
styleTag: styleMode.styleTag,
|
|
224
|
+
cssSelectorPrefixExamples: buildCssSelectorPrefixExamples(styleMode.mode, count),
|
|
225
|
+
cssAuthoring: buildCssAuthoring(styleMode, count),
|
|
226
|
+
}));
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
const _running = process.argv[1];
|
|
230
|
+
if (_running?.endsWith('live-insert.mjs') || _running?.endsWith('live-insert.mjs/')) {
|
|
231
|
+
insertCli();
|
|
232
|
+
}
|