Files
rose-ash/lib/host/playwright/block-editor.spec.js
giles f804a71726 host: block editor live-swap — :sx-post (not sx-disable) + a Playwright check
The block-editor move/remove controls used :sx-disable "true" (the OLD relate-picker pattern
= plain POST → 303 → full reload). Switched to :sx-post + :sx-target #block-editor + :sx-swap
outerHTML (the current pattern): the click is a text/sx form round-trip through the WASM
engine, the handler returns the re-rendered #block-editor, and it swaps IN PLACE — no reload.

Added lib/host/playwright/{block-editor.spec.js, run-block-check.sh} (the run-picker-check
harness pattern: ephemeral host server + one editable post + the main worktree's chromium).
Verifies the irreducibly-browser behaviour the SX conformance can't see: adding, reordering
(↑), and removing blocks re-render #block-editor live, and the controls RE-BIND on the
content each swap brings in. PASSES (1/1, 16s). blog conformance still 165/165.

Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
2026-07-01 05:20:52 +00:00

71 lines
3.5 KiB
JavaScript

// Browser check for the BLOCK EDITOR (lib/host/blog.sx, composition step 6). Runs against
// an ephemeral host server seeded with one editable host post by run-block-check.sh, which
// copies this spec into the Playwright env and sets SX_TEST_URL.
//
// What needs a real boosted-SPA browser (the SX conformance tests cover the model ops +
// server routes; this covers the live SX-htmx swap the engine drives): adding, reordering,
// and removing blocks re-renders #block-editor IN PLACE (sx-post → outerHTML swap), and the
// controls RE-BIND on the content brought in by each swap (the case an inline script fails).
const { test, expect } = require('playwright/test');
const USER = process.env.SX_ADMIN_USER || 'admin';
const PASS = process.env.SX_ADMIN_PASSWORD || 'letmein';
const HOST = 'block-host'; // the post whose edit page we drive
const BE = '#block-editor';
const ROWS = `${BE} > ul > li`; // block rows (exclude the add form)
async function waitReady(page) {
await expect(page.locator('html[data-sx-ready="true"]')).toHaveCount(1, { timeout: 45000 });
}
async function loginTo(page, path) {
await page.goto(path);
await page.waitForURL(/\/login/);
await page.fill('input[name="username"]', USER);
await page.fill('input[name="password"]', PASS);
await page.click('button[type="submit"]');
await page.waitForURL((u) => !u.pathname.startsWith('/login'));
}
// add a block via the add-block form (select a card type, type text, submit).
async function addBlock(page, ctype, text) {
await page.selectOption(`${BE} select[name="ctype"]`, ctype);
await page.fill(`${BE} input[name="text"]`, text);
await page.click(`${BE} form[sx-post$="/blocks/add"] button`);
}
test.describe('block editor (browser-only, live SX-htmx swap)', () => {
test('add, reorder, and remove blocks re-render #block-editor in place', async ({ page }) => {
test.setTimeout(90000);
await loginTo(page, `/${HOST}/edit`);
await waitReady(page);
await page.evaluate(() => { window.__noReload = true; });
// a fresh post has no :body -> no blocks yet
await expect(page.locator(ROWS)).toHaveCount(0);
// ADD #1 (text) -> one row appears live, showing its preview
await addBlock(page, 'card-text', 'First block');
await expect.poll(() => page.locator(ROWS).count(), { timeout: 15000 }).toBe(1);
await expect(page.locator(BE)).toContainText('First block');
// ADD #2 (heading) -> a second row on the swapped-in editor (controls re-bound)
await addBlock(page, 'card-heading', 'A Heading');
await expect.poll(() => page.locator(ROWS).count(), { timeout: 15000 }).toBe(2);
// order is add-order: block 0 = First block, block 1 = A Heading
await expect(page.locator(`${ROWS}`).first()).toContainText('First block');
// REORDER: move the 2nd block (A Heading) UP -> it becomes the first row
await page.locator(`${ROWS}`).nth(1).locator('button', { hasText: '↑' }).click();
await expect.poll(
() => page.locator(`${ROWS}`).first().innerText(), { timeout: 15000 }
).toContain('A Heading');
await expect(page.locator(ROWS)).toHaveCount(2);
// REMOVE the first row (A Heading) -> one row remains (First block)
await page.locator(`${ROWS}`).first().locator('button', { hasText: 'remove' }).click();
await expect.poll(() => page.locator(ROWS).count(), { timeout: 15000 }).toBe(1);
await expect(page.locator(BE)).toContainText('First block');
await expect(page.locator(BE)).not.toContainText('A Heading');
});
});