open-design/apps/web/tests/comments.test.ts
Martin Atrin 79fcaef129
Add Tweaks mode for HTML previews with picker, pod selection, and batched chat attachments (#513)
* Add tweaks mode for HTML preview comments

* Fix tweaks geometry and restore critique migration

* Harden tweaks mode reload sync

* Guard tweaks batch sends during active runs

---------

Co-authored-by: puma <puma@pumas-MacBook-Air.local>
2026-05-05 21:09:20 +08:00

256 lines
8.7 KiB
TypeScript

import { describe, expect, it } from 'vitest';
import {
buildBoardCommentAttachments,
commentsToAttachments,
historyWithCommentAttachmentContext,
liveSnapshotForComment,
mergeAttachedComments,
messageContentWithCommentAttachments,
overlayBoundsFromSnapshot,
removeAttachedComment,
targetFromSnapshot,
} from '../src/comments';
import type { ChatMessage, PreviewComment } from '../src/types';
describe('preview comment attachment helpers', () => {
it('builds compact target context from an iframe snapshot', () => {
const target = targetFromSnapshot({
filePath: 'index.html',
elementId: 'hero-title',
selector: '[data-od-id="hero-title"]',
label: 'h1.hero-title',
text: ` ${'Title '.repeat(80)} `,
htmlHint: `<h1 class="hero-title" data-od-id="hero-title">${'x'.repeat(240)}</h1>`,
position: { x: 10.4, y: 20.5, width: 300.2, height: 88.8 },
});
expect(target.text.length).toBeLessThanOrEqual(160);
expect(target.htmlHint.length).toBeLessThanOrEqual(180);
expect(target.position).toEqual({ x: 10, y: 21, width: 300, height: 89 });
});
it('creates ordered compact send payloads from attached comments', () => {
const attachments = commentsToAttachments([
comment({ id: 'c1', elementId: 'hero-title', note: 'Shorten this title' }),
comment({ id: 'c2', elementId: 'chart', note: 'Make it feel real' }),
]);
expect(attachments).toMatchObject([
{ id: 'c1', order: 1, elementId: 'hero-title', comment: 'Shorten this title' },
{ id: 'c2', order: 2, elementId: 'chart', comment: 'Make it feel real' },
]);
});
it('builds grouped board payloads for pod selections', () => {
const attachments = buildBoardCommentAttachments({
target: {
filePath: 'atlas.html',
elementId: 'pod-1',
selector: '[data-od-id="hero"], [data-od-id="chart"]',
label: 'Hero and chart',
text: 'Hero title Chart value',
position: { x: 10, y: 20, width: 300, height: 200 },
htmlHint: '<section data-od-id="hero">',
selectionKind: 'pod',
memberCount: 2,
podMembers: [
{
elementId: 'hero',
selector: '[data-od-id="hero"]',
label: 'section.hero',
text: 'Hero title',
position: { x: 10, y: 20, width: 200, height: 100 },
htmlHint: '<section data-od-id="hero">',
},
{
elementId: 'chart',
selector: '[data-od-id="chart"]',
label: 'section.chart',
text: 'Chart value',
position: { x: 120, y: 80, width: 190, height: 120 },
htmlHint: '<section data-od-id="chart">',
},
],
},
notes: ['Tighten the hierarchy', 'Make the chart feel premium'],
});
expect(attachments).toHaveLength(2);
expect(attachments[0]).toMatchObject({
selectionKind: 'pod',
memberCount: 2,
source: 'board-batch',
comment: 'Tighten the hierarchy',
});
expect(messageContentWithCommentAttachments('', attachments)).toContain('memberCount: 2');
});
it('keeps large queued board-note batches ordered in one send payload', () => {
const notes = Array.from({ length: 8 }, (_, index) => `Note ${index + 1}`);
const attachments = buildBoardCommentAttachments({
target: {
filePath: 'atlas.html',
elementId: 'pod-2',
selector: '[data-od-id="card"]',
label: 'Card pod',
text: 'Heading Body CTA',
position: { x: 20, y: 30, width: 240, height: 160 },
htmlHint: '<section data-od-id="card">',
selectionKind: 'pod',
memberCount: 3,
podMembers: [
{
elementId: 'card-heading',
selector: '[data-od-id="card-heading"]',
label: 'h2.card-heading',
text: 'Heading',
position: { x: 24, y: 34, width: 100, height: 32 },
htmlHint: '<h2 data-od-id="card-heading">',
},
{
elementId: 'card-body',
selector: '[data-od-id="card-body"]',
label: 'p.card-body',
text: 'Body',
position: { x: 24, y: 72, width: 180, height: 48 },
htmlHint: '<p data-od-id="card-body">',
},
{
elementId: 'card-cta',
selector: '[data-od-id="card-cta"]',
label: 'button.card-cta',
text: 'CTA',
position: { x: 24, y: 128, width: 96, height: 32 },
htmlHint: '<button data-od-id="card-cta">',
},
],
},
notes,
});
expect(attachments).toHaveLength(8);
expect(attachments.map((attachment) => attachment.order)).toEqual([1, 2, 3, 4, 5, 6, 7, 8]);
expect(attachments.map((attachment) => attachment.comment)).toEqual(notes);
expect(messageContentWithCommentAttachments('', attachments)).toContain('8. pod-2');
});
it('updates and removes attached comments by saved comment id', () => {
const first = comment({ id: 'c1', elementId: 'hero-title', note: 'Original' });
const updated = comment({ id: 'c1', elementId: 'hero-title', note: 'Updated' });
const chart = comment({ id: 'c2', elementId: 'chart', note: 'Fix chart' });
const merged = mergeAttachedComments([first, chart], updated);
expect(merged).toHaveLength(2);
expect(merged[0]?.note).toBe('Updated');
const remaining = removeAttachedComment(merged, 'c1');
expect(commentsToAttachments(remaining)).toEqual([
expect.objectContaining({ id: 'c2', elementId: 'chart' }),
]);
});
it('converts iframe snapshot bounds into scaled overlay bounds', () => {
expect(overlayBoundsFromSnapshot({
filePath: 'index.html',
elementId: 'hero-title',
selector: '[data-od-id="hero-title"]',
label: 'h1.hero-title',
text: '',
htmlHint: '',
position: { x: 10, y: 20, width: 120, height: 40 },
}, 1.25)).toEqual({
left: 12.5,
top: 25,
width: 150,
height: 50,
});
});
it('only resolves saved markers from live snapshots for the same file', () => {
const saved = comment({ filePath: 'index.html', elementId: 'hero-title' });
const snapshots = new Map([
['hero-title', {
filePath: 'index.html',
elementId: 'hero-title',
selector: '[data-od-id="hero-title"]',
label: 'h1.hero-title',
text: '',
htmlHint: '',
position: { x: 1, y: 2, width: 3, height: 4 },
}],
]);
expect(liveSnapshotForComment(saved, snapshots)?.elementId).toBe('hero-title');
expect(liveSnapshotForComment(comment({ filePath: 'other.html' }), snapshots)).toBeNull();
});
it('serializes selected comments into API-mode prompt context without visible input', () => {
const attachments = commentsToAttachments([
comment({ id: 'c1', elementId: 'hero-title', note: 'Only shorten this title' }),
]);
const content = messageContentWithCommentAttachments('', attachments);
expect(content).toContain('(No extra typed instruction.)');
expect(content).toContain('<attached-preview-comments>');
expect(content).toContain('selector: [data-od-id="hero-title"]');
expect(content).toContain('comment: Only shorten this title');
});
it('adds hidden comment context only to the current user message sent to API providers', () => {
const attachments = commentsToAttachments([
comment({ id: 'c1', elementId: 'hero-title', note: 'Make it bolder' }),
]);
const history: ChatMessage[] = [
{
id: 'old',
role: 'user',
content: 'Previous request',
createdAt: 0,
commentAttachments: attachments,
},
{
id: 'u1',
role: 'user',
content: '',
createdAt: 1,
commentAttachments: attachments,
},
{
id: 'a1',
role: 'assistant',
content: 'Ready',
createdAt: 2,
commentAttachments: attachments,
},
];
const next = historyWithCommentAttachmentContext(history, 'u1');
expect(next[0]?.content).toBe('Previous request');
expect(next[1]?.content).toContain('<attached-preview-comments>');
expect(next[1]?.content).toContain('comment: Make it bolder');
expect(next[2]?.content).toBe('Ready');
expect(history[1]?.content).toBe('');
});
});
function comment(patch: Partial<PreviewComment>): PreviewComment {
return {
id: 'c1',
projectId: 'project-1',
conversationId: 'conversation-1',
filePath: 'index.html',
elementId: 'hero-title',
selector: '[data-od-id="hero-title"]',
label: 'h1.hero-title',
text: 'Current title',
position: { x: 1, y: 2, width: 3, height: 4 },
htmlHint: '<h1 data-od-id="hero-title">',
note: 'Comment',
status: 'open',
createdAt: 1,
updatedAt: 1,
...patch,
};
}