mirror of
https://github.com/nexu-io/open-design.git
synced 2026-06-01 03:14:35 +07:00
* feat(web): queue chat sends * feat(web): render code comment directives * feat(web): add preview comments and manual edits * fix(web): polish shared chrome controls * fix(web): align queued send loading state * feat(web): open primary project artifacts * fix(web): keep queued sends and tests aligned * fix(web): restore docked comment tools layout * fix(web): align preview comment toolbar * fix(web): place local cli beside handoff * fix(web): move agent menu beside handoff * fix(web): make project instructions a direct header action * fix(web): compact handoff and toolbar labels * fix(web): clarify handoff menu and annotation label * fix(web): restore compact cursor handoff trigger * fix(web): align agent menu trigger with handoff * fix(web): add draw toolbar close action * fix(web): move inspect editing into edit mode * fix(web): avoid reserving comment sidebar in annotation mode * fix(web): float preview comments panel * fix(web): keep edit canvas full width * fix(web): polish preview annotation tools * fix(web): highlight active preview comments * fix(web): open comments panel after annotation save * fix(web): polish comment handoff controls * fix(web): remove palette preview tool * fix(web): simplify draw annotation toolbar * fix(web): restore queued tasks into composer * fix(web): restore queued send strip styling * fix(web): hide internal comment target ids * fix(web): align manual edit panel header * test(web): cover visual interaction contracts * fix(web): address PR feedback regressions * fix(web): preserve artifact chrome state * fix(daemon): restore project raw file routes --------- Co-authored-by: chaoxiaoche <chaoxiaoche@chaoxiaochedeMacBook-Pro.local> Co-authored-by: mrcfps <mrc@powerformer.com>
396 lines
17 KiB
TypeScript
396 lines
17 KiB
TypeScript
import { describe, expect, it, vi } from 'vitest';
|
|
import { JSDOM } from 'jsdom';
|
|
import {
|
|
buildManualEditBridge,
|
|
isMeaningfulManualEditElement,
|
|
isManualEditHostNode,
|
|
isSourceMappableManualEditElement,
|
|
manualEditDomPathForElement,
|
|
manualEditStableIdForElement,
|
|
} from '../../src/edit-mode/bridge';
|
|
|
|
describe('manual edit bridge target normalization', () => {
|
|
it('prefers explicit data-od-id over generated ids', () => {
|
|
const dom = new JSDOM('<main><h1 data-od-id="hero">Title</h1></main>');
|
|
const target = dom.window.document.querySelector('h1')!;
|
|
|
|
expect(manualEditStableIdForElement(target)).toBe('hero');
|
|
expect(target.getAttribute('data-od-runtime-id')).toBeNull();
|
|
});
|
|
|
|
it('generates stable DOM path ids for unannotated elements', () => {
|
|
const dom = new JSDOM('<main><section><p>First</p><p>Second</p></section></main>');
|
|
const target = dom.window.document.querySelectorAll('p')[1]!;
|
|
|
|
expect(manualEditDomPathForElement(target)).toBe('path-0-0-1');
|
|
expect(manualEditStableIdForElement(target)).toBe('path-0-0-1');
|
|
expect(manualEditStableIdForElement(target)).toBe('path-0-0-1');
|
|
expect(target.getAttribute('data-od-runtime-id')).toBe('path-0-0-1');
|
|
});
|
|
|
|
it('generates DOM path ids against source-shaped children, ignoring host shim nodes', () => {
|
|
const dom = new JSDOM(
|
|
'<script data-od-sandbox-shim></script><main><section><p>First</p><p>Second</p></section></main><script data-od-edit-bridge></script>',
|
|
);
|
|
const target = dom.window.document.querySelectorAll('p')[1]!;
|
|
|
|
expect(isManualEditHostNode(dom.window.document.querySelector('[data-od-sandbox-shim]')!)).toBe(true);
|
|
expect(manualEditDomPathForElement(target)).toBe('path-0-0-1');
|
|
});
|
|
|
|
it('discovers meaningful elements and ignores tiny or irrelevant elements', () => {
|
|
const dom = new JSDOM('<main><h1 data-od-source-path="path-0-0">Title</h1><script>1</script></main>');
|
|
const title = dom.window.document.querySelector('h1')!;
|
|
const script = dom.window.document.querySelector('script')!;
|
|
|
|
expect(isMeaningfulManualEditElement(title, { width: 80, height: 24 })).toBe(true);
|
|
expect(isMeaningfulManualEditElement(title, { width: 3, height: 24 })).toBe(false);
|
|
expect(isMeaningfulManualEditElement(script, { width: 80, height: 24 })).toBe(false);
|
|
});
|
|
|
|
it('keeps source-mappable display:none targets available for the layers panel', async () => {
|
|
const posts: Array<{ type?: string; targets?: Array<{ id: string; isHidden?: boolean }> }> = [];
|
|
const dom = new JSDOM(
|
|
`<main>
|
|
<h1 data-od-source-path="path-0-0">Visible title</h1>
|
|
<section data-od-source-path="path-0-1" style="display:none">
|
|
<p data-od-source-path="path-0-1-0">Hidden author notes</p>
|
|
</section>
|
|
</main>${buildManualEditBridge(true)}`,
|
|
{ runScripts: 'dangerously', url: 'http://localhost' },
|
|
);
|
|
const visible = dom.window.document.querySelector('h1')!;
|
|
const hiddenSection = dom.window.document.querySelector('section')!;
|
|
const hiddenParagraph = dom.window.document.querySelector('p')!;
|
|
visible.getBoundingClientRect = () => ({
|
|
x: 0, y: 0, width: 160, height: 32,
|
|
top: 0, right: 160, bottom: 32, left: 0,
|
|
toJSON: () => ({}),
|
|
} as DOMRect);
|
|
hiddenSection.getBoundingClientRect = () => ({
|
|
x: 0, y: 0, width: 0, height: 0,
|
|
top: 0, right: 0, bottom: 0, left: 0,
|
|
toJSON: () => ({}),
|
|
} as DOMRect);
|
|
hiddenParagraph.getBoundingClientRect = hiddenSection.getBoundingClientRect;
|
|
dom.window.parent.postMessage = ((message: unknown) => {
|
|
posts.push(message as { type?: string; targets?: Array<{ id: string; isHidden?: boolean }> });
|
|
}) as typeof dom.window.parent.postMessage;
|
|
|
|
dom.window.dispatchEvent(new dom.window.MessageEvent('message', {
|
|
data: { type: 'od-edit-mode', enabled: true },
|
|
}));
|
|
await new Promise((resolve) => dom.window.setTimeout(resolve, 0));
|
|
|
|
const targetsMessage = posts.find((message) => message.type === 'od-edit-targets');
|
|
expect(targetsMessage?.targets?.map((target) => target.id)).toEqual([
|
|
'path-0-0',
|
|
'path-0-1',
|
|
'path-0-1-0',
|
|
]);
|
|
expect(targetsMessage?.targets?.find((target) => target.id === 'path-0-1')?.isHidden).toBe(true);
|
|
expect(targetsMessage?.targets?.find((target) => target.id === 'path-0-1-0')?.isHidden).toBe(true);
|
|
|
|
dom.window.close();
|
|
});
|
|
|
|
it('treats hidden containers as layout editable targets', async () => {
|
|
const posts: Array<{ type?: string; targets?: Array<{ id: string; isHidden?: boolean; isLayoutContainer?: boolean }> }> = [];
|
|
const dom = new JSDOM(
|
|
`<main>
|
|
<section data-od-source-path="path-0-0" style="display:none">
|
|
<p data-od-source-path="path-0-0-0">Hidden layout copy</p>
|
|
</section>
|
|
</main>${buildManualEditBridge(true)}`,
|
|
{ runScripts: 'dangerously', url: 'http://localhost' },
|
|
);
|
|
const section = dom.window.document.querySelector('section')!;
|
|
const paragraph = dom.window.document.querySelector('p')!;
|
|
section.getBoundingClientRect = () => ({
|
|
x: 0, y: 0, width: 0, height: 0,
|
|
top: 0, right: 0, bottom: 0, left: 0,
|
|
toJSON: () => ({}),
|
|
} as DOMRect);
|
|
paragraph.getBoundingClientRect = section.getBoundingClientRect;
|
|
dom.window.parent.postMessage = ((message: unknown) => {
|
|
posts.push(message as { type?: string; targets?: Array<{ id: string; isHidden?: boolean; isLayoutContainer?: boolean }> });
|
|
}) as typeof dom.window.parent.postMessage;
|
|
|
|
dom.window.dispatchEvent(new dom.window.MessageEvent('message', {
|
|
data: { type: 'od-edit-mode', enabled: true },
|
|
}));
|
|
await new Promise((resolve) => dom.window.setTimeout(resolve, 0));
|
|
|
|
const targetsMessage = posts.find((message) => message.type === 'od-edit-targets');
|
|
const hiddenSection = targetsMessage?.targets?.find((target) => target.id === 'path-0-0');
|
|
const hiddenParagraph = targetsMessage?.targets?.find((target) => target.id === 'path-0-0-0');
|
|
expect(hiddenSection?.isHidden).toBe(true);
|
|
expect(hiddenSection?.isLayoutContainer).toBe(true);
|
|
expect(hiddenParagraph?.isLayoutContainer).toBe(false);
|
|
|
|
dom.window.close();
|
|
});
|
|
|
|
it('does not treat visibility-hidden block containers as layout editable targets', async () => {
|
|
const posts: Array<{ type?: string; targets?: Array<{ id: string; isHidden?: boolean; isLayoutContainer?: boolean }> }> = [];
|
|
const dom = new JSDOM(
|
|
`<main>
|
|
<section data-od-source-path="path-0-0" style="visibility:hidden">
|
|
<p data-od-source-path="path-0-0-0">Hidden block copy</p>
|
|
</section>
|
|
</main>${buildManualEditBridge(true)}`,
|
|
{ runScripts: 'dangerously', url: 'http://localhost' },
|
|
);
|
|
const section = dom.window.document.querySelector('section')!;
|
|
const paragraph = dom.window.document.querySelector('p')!;
|
|
section.getBoundingClientRect = () => ({
|
|
x: 0, y: 0, width: 160, height: 32,
|
|
top: 0, right: 160, bottom: 32, left: 0,
|
|
toJSON: () => ({}),
|
|
} as DOMRect);
|
|
paragraph.getBoundingClientRect = () => ({
|
|
x: 8, y: 8, width: 140, height: 20,
|
|
top: 8, right: 148, bottom: 28, left: 8,
|
|
toJSON: () => ({}),
|
|
} as DOMRect);
|
|
dom.window.parent.postMessage = ((message: unknown) => {
|
|
posts.push(message as { type?: string; targets?: Array<{ id: string; isHidden?: boolean; isLayoutContainer?: boolean }> });
|
|
}) as typeof dom.window.parent.postMessage;
|
|
|
|
dom.window.dispatchEvent(new dom.window.MessageEvent('message', {
|
|
data: { type: 'od-edit-mode', enabled: true },
|
|
}));
|
|
await new Promise((resolve) => dom.window.setTimeout(resolve, 0));
|
|
|
|
const targetsMessage = posts.find((message) => message.type === 'od-edit-targets');
|
|
const hiddenSection = targetsMessage?.targets?.find((target) => target.id === 'path-0-0');
|
|
expect(hiddenSection?.isHidden).toBe(true);
|
|
expect(hiddenSection?.isLayoutContainer).toBe(false);
|
|
|
|
dom.window.close();
|
|
});
|
|
|
|
it('does not treat block containers hidden only by an ancestor as layout editable targets', async () => {
|
|
const posts: Array<{ type?: string; targets?: Array<{ id: string; isHidden?: boolean; isLayoutContainer?: boolean }> }> = [];
|
|
const dom = new JSDOM(
|
|
`<main>
|
|
<div data-od-source-path="path-0-0" style="display:none">
|
|
<section data-od-source-path="path-0-0-0">Nested hidden section</section>
|
|
</div>
|
|
</main>${buildManualEditBridge(true)}`,
|
|
{ runScripts: 'dangerously', url: 'http://localhost' },
|
|
);
|
|
const wrapper = dom.window.document.querySelector('div')!;
|
|
const section = dom.window.document.querySelector('section')!;
|
|
wrapper.getBoundingClientRect = () => ({
|
|
x: 0, y: 0, width: 0, height: 0,
|
|
top: 0, right: 0, bottom: 0, left: 0,
|
|
toJSON: () => ({}),
|
|
} as DOMRect);
|
|
section.getBoundingClientRect = wrapper.getBoundingClientRect;
|
|
dom.window.parent.postMessage = ((message: unknown) => {
|
|
posts.push(message as { type?: string; targets?: Array<{ id: string; isHidden?: boolean; isLayoutContainer?: boolean }> });
|
|
}) as typeof dom.window.parent.postMessage;
|
|
|
|
dom.window.dispatchEvent(new dom.window.MessageEvent('message', {
|
|
data: { type: 'od-edit-mode', enabled: true },
|
|
}));
|
|
await new Promise((resolve) => dom.window.setTimeout(resolve, 0));
|
|
|
|
const targetsMessage = posts.find((message) => message.type === 'od-edit-targets');
|
|
const hiddenSection = targetsMessage?.targets?.find((target) => target.id === 'path-0-0-0');
|
|
expect(hiddenSection?.isHidden).toBe(true);
|
|
expect(hiddenSection?.isLayoutContainer).toBe(false);
|
|
|
|
dom.window.close();
|
|
});
|
|
|
|
it('does not mark visibility:visible descendants as hidden', async () => {
|
|
const posts: Array<{ type?: string; targets?: Array<{ id: string; isHidden?: boolean }> }> = [];
|
|
const dom = new JSDOM(
|
|
`<main>
|
|
<section data-od-source-path="path-0-0" style="visibility:hidden">
|
|
<p data-od-source-path="path-0-0-0" style="visibility:visible">Visible child copy</p>
|
|
</section>
|
|
</main>${buildManualEditBridge(true)}`,
|
|
{ runScripts: 'dangerously', url: 'http://localhost' },
|
|
);
|
|
const section = dom.window.document.querySelector('section')!;
|
|
const visibleChild = dom.window.document.querySelector('p')!;
|
|
section.getBoundingClientRect = () => ({
|
|
x: 0, y: 0, width: 160, height: 32,
|
|
top: 0, right: 160, bottom: 32, left: 0,
|
|
toJSON: () => ({}),
|
|
} as DOMRect);
|
|
visibleChild.getBoundingClientRect = () => ({
|
|
x: 8, y: 8, width: 140, height: 20,
|
|
top: 8, right: 148, bottom: 28, left: 8,
|
|
toJSON: () => ({}),
|
|
} as DOMRect);
|
|
dom.window.parent.postMessage = ((message: unknown) => {
|
|
posts.push(message as { type?: string; targets?: Array<{ id: string; isHidden?: boolean }> });
|
|
}) as typeof dom.window.parent.postMessage;
|
|
|
|
dom.window.dispatchEvent(new dom.window.MessageEvent('message', {
|
|
data: { type: 'od-edit-mode', enabled: true },
|
|
}));
|
|
await new Promise((resolve) => dom.window.setTimeout(resolve, 0));
|
|
|
|
const targetsMessage = posts.find((message) => message.type === 'od-edit-targets');
|
|
expect(targetsMessage?.targets?.find((target) => target.id === 'path-0-0')?.isHidden).toBe(true);
|
|
expect(targetsMessage?.targets?.find((target) => target.id === 'path-0-0-0')?.isHidden).toBe(false);
|
|
|
|
dom.window.close();
|
|
});
|
|
|
|
it('does not expose path targets unless they carry a source path marker', () => {
|
|
const dom = new JSDOM('<main><h1>Runtime title</h1><p data-od-source-path="path-0-1">Source text</p></main>');
|
|
const runtimeTitle = dom.window.document.querySelector('h1')!;
|
|
const sourceText = dom.window.document.querySelector('p')!;
|
|
|
|
expect(isSourceMappableManualEditElement(runtimeTitle)).toBe(false);
|
|
expect(isSourceMappableManualEditElement(sourceText)).toBe(true);
|
|
expect(isMeaningfulManualEditElement(runtimeTitle, { width: 80, height: 24 })).toBe(false);
|
|
});
|
|
|
|
it('omits selected outerHTML from bulk target posts but includes it for selected targets', () => {
|
|
const bridge = buildManualEditBridge(true);
|
|
|
|
expect(bridge).toContain('targets.push(targetFrom(nodes[i], false))');
|
|
expect(bridge).toContain("target: targetFrom(el, true)");
|
|
expect(bridge).toContain('if (!isSourceMappable(nodes[i])) continue;');
|
|
expect(bridge).toContain('if (isPrimaryTarget(el)) return el;');
|
|
});
|
|
|
|
it('acks live preview style patches by id and version', () => {
|
|
const bridge = buildManualEditBridge(true);
|
|
|
|
expect(bridge).toContain("type: 'od-edit-preview-style-applied'");
|
|
expect(bridge).toContain('version: Number(version) || 0, ok: true');
|
|
expect(bridge).toContain("ok: false, error: 'Target not found'");
|
|
});
|
|
|
|
it('moves the runtime selected marker between selected targets', () => {
|
|
const dom = new JSDOM(
|
|
`<main>
|
|
<h1 data-od-id="title">Title</h1>
|
|
<p data-od-id="body">Body</p>
|
|
</main>${buildManualEditBridge(true)}`,
|
|
{ runScripts: 'dangerously', url: 'http://localhost' },
|
|
);
|
|
const title = dom.window.document.querySelector('[data-od-id="title"]')!;
|
|
const body = dom.window.document.querySelector('[data-od-id="body"]')!;
|
|
|
|
dom.window.dispatchEvent(new dom.window.MessageEvent('message', {
|
|
data: { type: 'od-edit-selected-target', id: 'title' },
|
|
}));
|
|
expect(title.getAttribute('data-od-edit-selected')).toBe('true');
|
|
expect(body.hasAttribute('data-od-edit-selected')).toBe(false);
|
|
|
|
dom.window.dispatchEvent(new dom.window.MessageEvent('message', {
|
|
data: { type: 'od-edit-selected-target', id: 'body' },
|
|
}));
|
|
expect(title.hasAttribute('data-od-edit-selected')).toBe(false);
|
|
expect(body.getAttribute('data-od-edit-selected')).toBe('true');
|
|
|
|
dom.window.close();
|
|
});
|
|
|
|
it('clears runtime selected markers for null selection and edit-mode exit', () => {
|
|
const dom = new JSDOM(
|
|
`<main>
|
|
<h1 data-od-id="title">Title</h1>
|
|
<p data-od-id="body" data-od-edit-selected="true">Body</p>
|
|
</main>${buildManualEditBridge(true)}`,
|
|
{ runScripts: 'dangerously', url: 'http://localhost' },
|
|
);
|
|
const body = dom.window.document.querySelector('[data-od-id="body"]')!;
|
|
|
|
dom.window.dispatchEvent(new dom.window.MessageEvent('message', {
|
|
data: { type: 'od-edit-selected-target', id: null },
|
|
}));
|
|
expect(body.hasAttribute('data-od-edit-selected')).toBe(false);
|
|
|
|
dom.window.dispatchEvent(new dom.window.MessageEvent('message', {
|
|
data: { type: 'od-edit-selected-target', id: 'body' },
|
|
}));
|
|
expect(body.getAttribute('data-od-edit-selected')).toBe('true');
|
|
|
|
dom.window.dispatchEvent(new dom.window.MessageEvent('message', {
|
|
data: { type: 'od-edit-mode', enabled: false },
|
|
}));
|
|
expect(body.hasAttribute('data-od-edit-selected')).toBe(false);
|
|
|
|
dom.window.close();
|
|
});
|
|
|
|
it('keeps runtime selection marker out of source-shaped target data', () => {
|
|
const bridge = buildManualEditBridge(true);
|
|
|
|
expect(bridge).toContain("attr.name === 'data-od-edit-selected'");
|
|
expect(bridge).toContain('replace(/\\sdata-od-edit-selected="[^"]*"/g, \'\')');
|
|
expect(bridge).toContain('[data-od-edit-selected]');
|
|
});
|
|
|
|
it('marks flex/grid targets as layout containers', () => {
|
|
const bridge = buildManualEditBridge(true);
|
|
|
|
expect(bridge).toContain('isLayoutContainer: isLayoutContainer(el)');
|
|
expect(bridge).toContain("display.indexOf('flex') >= 0 || display.indexOf('grid') >= 0");
|
|
});
|
|
|
|
it('turns text targets into inline editors and commits changed text', () => {
|
|
const dom = new JSDOM(
|
|
`<main><h1 data-od-id="title">Original title</h1></main>${buildManualEditBridge(true)}`,
|
|
{ runScripts: 'dangerously', url: 'http://localhost' },
|
|
);
|
|
const title = dom.window.document.querySelector('[data-od-id="title"]') as HTMLElement;
|
|
const postMessage = vi.spyOn(dom.window.parent, 'postMessage');
|
|
|
|
title.dispatchEvent(new dom.window.MouseEvent('click', {
|
|
bubbles: true,
|
|
cancelable: true,
|
|
clientX: 8,
|
|
clientY: 8,
|
|
}));
|
|
expect(title.getAttribute('contenteditable')).toBe('plaintext-only');
|
|
expect(title.getAttribute('data-od-editing')).toBe('true');
|
|
|
|
title.textContent = 'Edited title';
|
|
title.dispatchEvent(new dom.window.FocusEvent('blur', { bubbles: false }));
|
|
|
|
expect(title.hasAttribute('contenteditable')).toBe(false);
|
|
expect(title.hasAttribute('data-od-editing')).toBe(false);
|
|
expect(postMessage).toHaveBeenCalledWith({
|
|
type: 'od-edit-text-commit',
|
|
id: 'title',
|
|
value: 'Edited title',
|
|
}, '*');
|
|
|
|
dom.window.close();
|
|
});
|
|
|
|
it('cancels inline text edits with Escape without posting a commit', () => {
|
|
const dom = new JSDOM(
|
|
`<main><p data-od-id="body">Original body</p></main>${buildManualEditBridge(true)}`,
|
|
{ runScripts: 'dangerously', url: 'http://localhost' },
|
|
);
|
|
const body = dom.window.document.querySelector('[data-od-id="body"]') as HTMLElement;
|
|
const postMessage = vi.spyOn(dom.window.parent, 'postMessage');
|
|
|
|
body.dispatchEvent(new dom.window.MouseEvent('click', { bubbles: true, cancelable: true }));
|
|
body.textContent = 'Draft body';
|
|
body.dispatchEvent(new dom.window.KeyboardEvent('keydown', {
|
|
bubbles: true,
|
|
cancelable: true,
|
|
key: 'Escape',
|
|
}));
|
|
|
|
expect(body.textContent).toBe('Original body');
|
|
expect(postMessage).not.toHaveBeenCalledWith(expect.objectContaining({
|
|
type: 'od-edit-text-commit',
|
|
}), '*');
|
|
|
|
dom.window.close();
|
|
});
|
|
});
|