@deskwork/studio 0.12.1 → 0.14.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 (131) hide show
  1. package/dist/components/scrapbook-item.d.ts +9 -1
  2. package/dist/components/scrapbook-item.d.ts.map +1 -1
  3. package/dist/components/scrapbook-item.js +11 -2
  4. package/dist/components/scrapbook-item.js.map +1 -1
  5. package/dist/data/glossary.json +62 -0
  6. package/dist/lib/glossary-helper.d.ts +16 -0
  7. package/dist/lib/glossary-helper.d.ts.map +1 -0
  8. package/dist/lib/glossary-helper.js +26 -0
  9. package/dist/lib/glossary-helper.js.map +1 -0
  10. package/dist/pages/chrome.d.ts +24 -13
  11. package/dist/pages/chrome.d.ts.map +1 -1
  12. package/dist/pages/chrome.js +25 -24
  13. package/dist/pages/chrome.js.map +1 -1
  14. package/dist/pages/chrome.ts +38 -27
  15. package/dist/pages/content-detail.d.ts.map +1 -1
  16. package/dist/pages/content-detail.js +6 -5
  17. package/dist/pages/content-detail.js.map +1 -1
  18. package/dist/pages/content-detail.ts +6 -5
  19. package/dist/pages/content.d.ts.map +1 -1
  20. package/dist/pages/content.js +4 -6
  21. package/dist/pages/content.js.map +1 -1
  22. package/dist/pages/content.ts +4 -6
  23. package/dist/pages/dashboard/affordances.d.ts +6 -1
  24. package/dist/pages/dashboard/affordances.d.ts.map +1 -1
  25. package/dist/pages/dashboard/affordances.js +17 -2
  26. package/dist/pages/dashboard/affordances.js.map +1 -1
  27. package/dist/pages/dashboard/affordances.ts +18 -2
  28. package/dist/pages/dashboard/header.js +2 -2
  29. package/dist/pages/dashboard/header.ts +2 -2
  30. package/dist/pages/dashboard/section.d.ts +2 -2
  31. package/dist/pages/dashboard/section.d.ts.map +1 -1
  32. package/dist/pages/dashboard/section.js +5 -5
  33. package/dist/pages/dashboard/section.js.map +1 -1
  34. package/dist/pages/dashboard/section.ts +9 -5
  35. package/dist/pages/dashboard.d.ts.map +1 -1
  36. package/dist/pages/dashboard.js +3 -2
  37. package/dist/pages/dashboard.js.map +1 -1
  38. package/dist/pages/dashboard.ts +3 -2
  39. package/dist/pages/entry-review/data.d.ts +54 -0
  40. package/dist/pages/entry-review/data.d.ts.map +1 -0
  41. package/dist/pages/entry-review/data.js +116 -0
  42. package/dist/pages/entry-review/data.js.map +1 -0
  43. package/dist/pages/entry-review/decision-strip.d.ts +37 -0
  44. package/dist/pages/entry-review/decision-strip.d.ts.map +1 -0
  45. package/dist/pages/entry-review/decision-strip.js +137 -0
  46. package/dist/pages/entry-review/decision-strip.js.map +1 -0
  47. package/dist/pages/entry-review/edit-panes.d.ts +12 -0
  48. package/dist/pages/entry-review/edit-panes.d.ts.map +1 -0
  49. package/dist/pages/entry-review/edit-panes.js +28 -0
  50. package/dist/pages/entry-review/edit-panes.js.map +1 -0
  51. package/dist/pages/entry-review/edit-toolbar.d.ts +20 -0
  52. package/dist/pages/entry-review/edit-toolbar.d.ts.map +1 -0
  53. package/dist/pages/entry-review/edit-toolbar.js +46 -0
  54. package/dist/pages/entry-review/edit-toolbar.js.map +1 -0
  55. package/dist/pages/entry-review/index.d.ts +50 -0
  56. package/dist/pages/entry-review/index.d.ts.map +1 -0
  57. package/dist/pages/entry-review/index.js +219 -0
  58. package/dist/pages/entry-review/index.js.map +1 -0
  59. package/dist/pages/entry-review/marginalia.d.ts +28 -0
  60. package/dist/pages/entry-review/marginalia.d.ts.map +1 -0
  61. package/dist/pages/entry-review/marginalia.js +67 -0
  62. package/dist/pages/entry-review/marginalia.js.map +1 -0
  63. package/dist/pages/entry-review/not-found.d.ts +9 -0
  64. package/dist/pages/entry-review/not-found.d.ts.map +1 -0
  65. package/dist/pages/entry-review/not-found.js +34 -0
  66. package/dist/pages/entry-review/not-found.js.map +1 -0
  67. package/dist/pages/entry-review/outline-drawer.d.ts +12 -0
  68. package/dist/pages/entry-review/outline-drawer.d.ts.map +1 -0
  69. package/dist/pages/entry-review/outline-drawer.js +28 -0
  70. package/dist/pages/entry-review/outline-drawer.js.map +1 -0
  71. package/dist/pages/entry-review/shortcuts.d.ts +14 -0
  72. package/dist/pages/entry-review/shortcuts.d.ts.map +1 -0
  73. package/dist/pages/entry-review/shortcuts.js +36 -0
  74. package/dist/pages/entry-review/shortcuts.js.map +1 -0
  75. package/dist/pages/entry-review/version-strip.d.ts +33 -0
  76. package/dist/pages/entry-review/version-strip.d.ts.map +1 -0
  77. package/dist/pages/entry-review/version-strip.js +81 -0
  78. package/dist/pages/entry-review/version-strip.js.map +1 -0
  79. package/dist/pages/entry-review.d.ts +6 -21
  80. package/dist/pages/entry-review.d.ts.map +1 -1
  81. package/dist/pages/entry-review.js +6 -144
  82. package/dist/pages/entry-review.js.map +1 -1
  83. package/dist/pages/entry-review.ts +11 -181
  84. package/dist/pages/html.d.ts +2 -0
  85. package/dist/pages/html.d.ts.map +1 -1
  86. package/dist/pages/html.js +2 -0
  87. package/dist/pages/html.js.map +1 -1
  88. package/dist/pages/html.ts +4 -0
  89. package/dist/pages/index.d.ts.map +1 -1
  90. package/dist/pages/index.js +6 -10
  91. package/dist/pages/index.js.map +1 -1
  92. package/dist/pages/index.ts +6 -10
  93. package/dist/pages/layout.d.ts.map +1 -1
  94. package/dist/pages/layout.js +6 -0
  95. package/dist/pages/layout.js.map +1 -1
  96. package/dist/pages/layout.ts +7 -0
  97. package/dist/pages/review-scrapbook-drawer.d.ts +7 -0
  98. package/dist/pages/review-scrapbook-drawer.d.ts.map +1 -1
  99. package/dist/pages/review-scrapbook-drawer.js +45 -6
  100. package/dist/pages/review-scrapbook-drawer.js.map +1 -1
  101. package/dist/pages/review-scrapbook-drawer.ts +50 -6
  102. package/dist/pages/review.d.ts.map +1 -1
  103. package/dist/pages/review.js +168 -41
  104. package/dist/pages/review.js.map +1 -1
  105. package/dist/pages/review.ts +192 -41
  106. package/dist/pages/scrapbook.d.ts +7 -14
  107. package/dist/pages/scrapbook.d.ts.map +1 -1
  108. package/dist/pages/scrapbook.js +548 -192
  109. package/dist/pages/scrapbook.js.map +1 -1
  110. package/dist/pages/scrapbook.ts +588 -221
  111. package/dist/pages/shortform-review.d.ts +32 -0
  112. package/dist/pages/shortform-review.d.ts.map +1 -0
  113. package/dist/pages/shortform-review.js +270 -0
  114. package/dist/pages/shortform-review.js.map +1 -0
  115. package/dist/pages/shortform-review.ts +342 -0
  116. package/dist/pages/shortform.js +1 -1
  117. package/dist/pages/shortform.js.map +1 -1
  118. package/dist/pages/shortform.ts +1 -1
  119. package/dist/routes/api.d.ts +8 -0
  120. package/dist/routes/api.d.ts.map +1 -1
  121. package/dist/routes/api.js +115 -0
  122. package/dist/routes/api.js.map +1 -1
  123. package/dist/routes/entry-annotation-body.d.ts +26 -0
  124. package/dist/routes/entry-annotation-body.d.ts.map +1 -0
  125. package/dist/routes/entry-annotation-body.js +152 -0
  126. package/dist/routes/entry-annotation-body.js.map +1 -0
  127. package/dist/server.d.ts +1 -1
  128. package/dist/server.d.ts.map +1 -1
  129. package/dist/server.js +56 -196
  130. package/dist/server.js.map +1 -1
  131. package/package.json +4 -4
@@ -1,206 +1,550 @@
1
1
  /**
2
- * Scrapbook viewer — `/dev/scrapbook/:site/<path>` (path may include `/`).
2
+ * Scrapbook viewer — `/dev/scrapbook/:site/<path>`.
3
3
  *
4
- * Reads the scrapbook directory at the given path and lists every
5
- * file with type chips + relative timestamps, plus secret items
6
- * (inside `scrapbook/secret/`) in a quiet second section. Empty
7
- * scrapbooks render an empty state with quick-add affordances.
4
+ * Issue #161 redesign: aside-left folder card with numbered item list,
5
+ * vertical card grid with per-kind colored ribbons + always-visible foot
6
+ * toolbar + per-kind preview rendering, drop zone, secret section,
7
+ * single-expanded card invariant, aside cross-linking.
8
8
  *
9
- * The `path` argument is the hierarchical address of the scrapbook
10
- * any slash-separated kebab-case identifier under the site's
11
- * contentDir. It does not need to correspond to a calendar entry;
12
- * organizational nodes (e.g. `the-outbound/characters` with no
13
- * own README) can host their own scrapbooks too.
14
- *
15
- * Port of `pages/dev/scrapbook/[site]/[slug].astro`. Layout swap
16
- * (Astro `<Layout>` → studio shell) and CSS link added; structurally
17
- * similar otherwise.
9
+ * Mockup: docs/superpowers/frontend-design/2026-05-02-review-redesign/scrapbook-redesign.html
10
+ * Spec: docs/superpowers/specs/2026-05-02-scrapbook-redesign-impl-spec.md
18
11
  */
19
- import { formatRelativeTime, formatSize, listScrapbook, } from '@deskwork/core/scrapbook';
12
+ import { existsSync, readFileSync } from 'node:fs';
13
+ import { readCalendar } from '@deskwork/core/calendar';
14
+ import { findEntry } from '@deskwork/core/calendar-mutations';
15
+ import { resolveCalendarPath } from '@deskwork/core/paths';
16
+ import { formatRelativeTime, formatSize, listScrapbook, scrapbookFilePath, } from '@deskwork/core/scrapbook';
20
17
  import { html, unsafe } from "./html.js";
21
18
  import { layout } from "./layout.js";
22
19
  import { renderEditorialFolio } from "./chrome.js";
23
- function renderItemRow(item, index, opts = {}) {
24
- const { secret = false, withTools = true } = opts;
25
- const editBtn = withTools && item.kind === 'md'
26
- ? unsafe(html `<button type="button" class="scrapbook-tool" data-action="edit">edit</button>`)
27
- : '';
28
- const seq = String(index + 1).padStart(2, '0');
29
- const kindLabel = item.kind === 'other' ? '·' : item.kind.toUpperCase();
30
- const idPrefix = secret ? 'secret-' : '';
31
- const dataSecret = secret ? ' data-secret="true"' : '';
32
- // The "mark secret/public" toggle is the cross-section rename
33
- // affordance. The button label flips with the source section.
34
- const sectionToggleLabel = secret ? 'mark public' : 'mark secret';
35
- const toolbar = withTools
36
- ? unsafe(html `<div class="scrapbook-toolbar" data-toolbar>
37
- ${editBtn}
38
- <button type="button" class="scrapbook-tool" data-action="rename">rename</button>
39
- <button type="button" class="scrapbook-tool" data-action="toggle-secret">${sectionToggleLabel}</button>
40
- <button type="button" class="scrapbook-tool scrapbook-tool--delete" data-action="delete">delete</button>
41
- </div>`)
42
- : '';
43
- return unsafe(html `
44
- <li class="scrapbook-item${secret ? ' scrapbook-item--secret' : ''}" data-state="closed" data-open="false"
45
- data-filename="${item.name}" data-kind="${item.kind}"
46
- data-size="${item.size}" data-mtime="${item.mtime}"${unsafe(dataSecret)}
47
- id="${idPrefix}item-${encodeURIComponent(item.name)}">
48
- <button type="button" class="scrapbook-item-header" aria-expanded="false">
49
- <span class="scrapbook-seq" aria-hidden="true">§ ${seq}</span>
50
- <span class="scrapbook-kind scrapbook-kind--${item.kind}" aria-hidden="true">${kindLabel}</span>
51
- <span class="scrapbook-filename" data-filename-cell>${item.name}</span>
52
- <time class="scrapbook-mtime" datetime="${item.mtime}">${formatRelativeTime(item.mtime)}</time>
53
- <span class="scrapbook-disclosure" aria-hidden="true">▸</span>
54
- </button>
55
- ${toolbar}
56
- <div class="scrapbook-perforation" aria-hidden="true"></div>
57
- <div class="scrapbook-item-body" data-body>
58
- <div data-body-content></div>
59
- </div>
60
- </li>`);
20
+ const KIND_LABEL = {
21
+ md: 'MD',
22
+ img: 'IMG',
23
+ json: 'JSON',
24
+ js: 'JS',
25
+ txt: 'TXT',
26
+ other: '·',
27
+ };
28
+ function escapeHtml(s) {
29
+ return s
30
+ .replace(/&/g, '&amp;')
31
+ .replace(/</g, '&lt;')
32
+ .replace(/>/g, '&gt;')
33
+ .replace(/"/g, '&quot;')
34
+ .replace(/'/g, '&#39;');
61
35
  }
62
36
  /**
63
- * Build a hierarchical breadcrumb from the path. Each segment links to
64
- * the scrapbook view for its prefix. The root segment (site) just
65
- * goes back to the editorial dashboard.
37
+ * Strip a YAML frontmatter block from the top of an md file. Only strips
38
+ * the leading `---\n...\n---\n` block; body-level `---` separators (Setext
39
+ * H2 underline, thematic break) are preserved because the function only
40
+ * looks at the first 4 chars for the opener.
66
41
  */
67
- function renderBreadcrumb(site, path) {
68
- const segments = path.split('/');
69
- const links = [];
70
- for (let i = 0; i < segments.length; i++) {
71
- const prefix = segments.slice(0, i + 1).join('/');
72
- const isLast = i === segments.length - 1;
73
- if (isLast) {
74
- links.push(html `<span class="scrapbook-breadcrumb-current">${segments[i]}</span>`);
42
+ function stripFrontmatter(text) {
43
+ if (!text.startsWith('---\n'))
44
+ return text;
45
+ const closeIdx = text.indexOf('\n---\n', 4);
46
+ if (closeIdx < 0)
47
+ return text;
48
+ return text.slice(closeIdx + 5).replace(/^\n+/, '');
49
+ }
50
+ /**
51
+ * Build the closed-state preview excerpt for md/json/txt. Returns null
52
+ * when there's nothing useful to render — empty file, frontmatter-only
53
+ * file, or binary masquerading as text — so the caller can omit the
54
+ * preview block entirely (matches "other" kind treatment, avoids the
55
+ * 6rem min-height void).
56
+ *
57
+ * For json: pretty-print via JSON.parse + JSON.stringify(_, null, 2) so
58
+ * minified single-line files still render multi-line. Falls back to raw
59
+ * content on parse error (bad JSON is still readable as text).
60
+ *
61
+ * Binary detection: NUL byte presence after UTF-8 decode. Real text
62
+ * almost never has NUL; binary files have it within the first KB.
63
+ */
64
+ function previewExcerpt(buf, kind) {
65
+ let text = buf.subarray(0, Math.min(buf.byteLength, 2400)).toString('utf-8');
66
+ if (text.indexOf('\0') >= 0)
67
+ return null;
68
+ if (kind === 'md')
69
+ text = stripFrontmatter(text);
70
+ if (kind === 'json') {
71
+ try {
72
+ const fullText = buf.toString('utf-8');
73
+ text = JSON.stringify(JSON.parse(fullText), null, 2);
74
+ }
75
+ catch {
76
+ // Invalid JSON — fall through to the raw-text excerpt below.
77
+ }
78
+ }
79
+ const excerpt = text.split('\n').slice(0, 8).join('\n').slice(0, 600);
80
+ if (excerpt.trim() === '')
81
+ return null;
82
+ return excerpt;
83
+ }
84
+ /**
85
+ * Count lines in a text file: number of `\n` bytes plus 1 if the last
86
+ * byte isn't `\n` (so a 3-line file whether or not it has a trailing
87
+ * newline reports 3).
88
+ */
89
+ function countLines(buf) {
90
+ let count = 0;
91
+ for (const b of buf)
92
+ if (b === 0x0a)
93
+ count++;
94
+ if (buf.length > 0 && buf[buf.length - 1] !== 0x0a)
95
+ count++;
96
+ return count;
97
+ }
98
+ /**
99
+ * Count top-level keys in a JSON object. Returns null if the file is not
100
+ * valid JSON or its root is not a plain object (arrays, primitives →
101
+ * null; caller renders no extra meta).
102
+ */
103
+ function countJsonKeys(buf) {
104
+ try {
105
+ const obj = JSON.parse(buf.toString('utf-8'));
106
+ if (obj !== null && typeof obj === 'object' && !Array.isArray(obj)) {
107
+ return Object.keys(obj).length;
108
+ }
109
+ return null;
110
+ }
111
+ catch {
112
+ return null;
113
+ }
114
+ }
115
+ /**
116
+ * Read width × height from a buffer. Recognizes PNG, JPEG, WebP, and
117
+ * GIF; returns null for unrecognized signatures or truncated/malformed
118
+ * buffers. Used by the scrapbook card meta to render `{W} × {H}` next
119
+ * to the image kind chip.
120
+ *
121
+ * Each format is parsed from its file-header structure (no external
122
+ * dependency); each branch returns null on any unexpected byte rather
123
+ * than throwing, so a corrupt image still renders with empty meta.
124
+ */
125
+ function readImageDimensions(buf) {
126
+ if (buf.length < 12)
127
+ return null;
128
+ // PNG: 89 50 4E 47 0D 0A 1A 0A then IHDR (chunk-length 4, "IHDR" 4,
129
+ // width 4, height 4 — width@16, height@20).
130
+ if (buf[0] === 0x89 && buf[1] === 0x50 && buf[2] === 0x4e && buf[3] === 0x47) {
131
+ if (buf.length < 24)
132
+ return null;
133
+ return { width: buf.readUInt32BE(16), height: buf.readUInt32BE(20) };
134
+ }
135
+ // GIF: "GIF87a" or "GIF89a" + logical screen descriptor (width LE @6,
136
+ // height LE @8).
137
+ if (buf[0] === 0x47 && buf[1] === 0x49 && buf[2] === 0x46 &&
138
+ buf[3] === 0x38 && (buf[4] === 0x37 || buf[4] === 0x39) && buf[5] === 0x61) {
139
+ if (buf.length < 10)
140
+ return null;
141
+ return { width: buf.readUInt16LE(6), height: buf.readUInt16LE(8) };
142
+ }
143
+ // JPEG: starts FF D8. Width/height live in a Start-Of-Frame marker
144
+ // (FF C0–CF, excluding DHT C4 / JPG C8 / DAC CC). Walk markers
145
+ // skipping their payloads until the SOF is found.
146
+ if (buf[0] === 0xff && buf[1] === 0xd8) {
147
+ return readJpegDimensions(buf);
148
+ }
149
+ // WebP: "RIFF" {size} "WEBP" then a chunk (VP8 / VP8L / VP8X).
150
+ if (buf[0] === 0x52 && buf[1] === 0x49 && buf[2] === 0x46 && buf[3] === 0x46 &&
151
+ buf[8] === 0x57 && buf[9] === 0x45 && buf[10] === 0x42 && buf[11] === 0x50) {
152
+ return readWebpDimensions(buf);
153
+ }
154
+ return null;
155
+ }
156
+ function readJpegDimensions(buf) {
157
+ // Marker walk: skip FF D8 (SOI), then each marker is FF Xn followed
158
+ // by a 2-byte big-endian segment length (which includes its own 2
159
+ // bytes). The SOF segment's payload is: 1 byte precision, 2 bytes
160
+ // height, 2 bytes width (the rest is component info).
161
+ let i = 2;
162
+ while (i + 4 <= buf.length) {
163
+ if (buf[i] !== 0xff)
164
+ return null;
165
+ let marker = buf[i + 1] ?? 0;
166
+ // Skip fill bytes (0xff padding before the actual marker byte).
167
+ while (marker === 0xff && i + 2 < buf.length) {
168
+ i++;
169
+ marker = buf[i + 1] ?? 0;
170
+ }
171
+ i += 2;
172
+ // Standalone markers (no length): RST0–7 (D0–D7) and SOI/EOI/TEM.
173
+ if (marker === 0xd9 || marker === 0xd8 || marker === 0x01)
174
+ return null;
175
+ if (marker >= 0xd0 && marker <= 0xd7)
176
+ continue;
177
+ if (i + 2 > buf.length)
178
+ return null;
179
+ const segLen = buf.readUInt16BE(i);
180
+ // SOF markers carry the dimensions. Exclusions per JPEG spec:
181
+ // C4 (DHT — Huffman tables), C8 (JPG reserved), CC (DAC — arithmetic
182
+ // coding conditioning).
183
+ const isSof = marker >= 0xc0 && marker <= 0xcf &&
184
+ marker !== 0xc4 && marker !== 0xc8 && marker !== 0xcc;
185
+ if (isSof) {
186
+ if (i + 7 > buf.length)
187
+ return null;
188
+ return {
189
+ width: buf.readUInt16BE(i + 5),
190
+ height: buf.readUInt16BE(i + 3),
191
+ };
192
+ }
193
+ i += segLen;
194
+ }
195
+ return null;
196
+ }
197
+ function readWebpDimensions(buf) {
198
+ // Three sub-formats (per RFC 6386 / VP8L spec / WebP container spec):
199
+ // - "VP8 " (lossy): header @20-22: signature 9D 01 2A, then
200
+ // 14-bit width LE @23, 14-bit height LE @25
201
+ // (each masked to 14 bits — top 2 bits are
202
+ // horizontal/vertical scale).
203
+ // - "VP8L" (lossless): header @20: signature 0x2F, then 32 bits
204
+ // packed: width-1 (14 bits) + height-1 (14
205
+ // bits) + alpha-flag + version (4 bits).
206
+ // - "VP8X" (extended): flags @20, reserved @21-23, width-1 (24
207
+ // bits LE) @24, height-1 (24 bits LE) @27.
208
+ // Need at least the RIFF header + chunk fourcc; per-variant length
209
+ // checks below cover the variant-specific payload sizes.
210
+ if (buf.length < 16)
211
+ return null;
212
+ const fourcc = buf.subarray(12, 16).toString('ascii');
213
+ if (fourcc === 'VP8 ') {
214
+ if (buf.length < 30)
215
+ return null;
216
+ if (buf[23] !== 0x9d || buf[24] !== 0x01 || buf[25] !== 0x2a)
217
+ return null;
218
+ const w = buf.readUInt16LE(26) & 0x3fff;
219
+ const h = buf.readUInt16LE(28) & 0x3fff;
220
+ return { width: w, height: h };
221
+ }
222
+ if (fourcc === 'VP8L') {
223
+ if (buf.length < 25)
224
+ return null;
225
+ if (buf[20] !== 0x2f)
226
+ return null;
227
+ // Read 4 bytes LE at offset 21, then unpack.
228
+ const packed = buf.readUInt32LE(21);
229
+ const widthMinus1 = packed & 0x3fff;
230
+ const heightMinus1 = (packed >>> 14) & 0x3fff;
231
+ return { width: widthMinus1 + 1, height: heightMinus1 + 1 };
232
+ }
233
+ if (fourcc === 'VP8X') {
234
+ if (buf.length < 30)
235
+ return null;
236
+ // 24-bit little-endian: low byte + (mid << 8) + (high << 16).
237
+ const w = ((buf[24] ?? 0) | ((buf[25] ?? 0) << 8) | ((buf[26] ?? 0) << 16)) + 1;
238
+ const h = ((buf[27] ?? 0) | ((buf[28] ?? 0) << 8) | ((buf[29] ?? 0) << 16)) + 1;
239
+ return { width: w, height: h };
240
+ }
241
+ return null;
242
+ }
243
+ /**
244
+ * Compute the per-kind extra meta string shown after the kind chip + size:
245
+ * md / txt → "{N} lines"
246
+ * json → "{N} keys" (root must be a plain object; otherwise empty)
247
+ * img → "{W} × {H}" (PNG only; other formats → empty)
248
+ * other → empty
249
+ *
250
+ * ENOENT (race-window with delete) returns empty so the card still
251
+ * renders; other errors propagate to the page renderer.
252
+ */
253
+ function computeKindMeta(ctx, site, path, item) {
254
+ if (item.kind !== 'md' && item.kind !== 'txt' && item.kind !== 'json' && item.kind !== 'img') {
255
+ return '';
256
+ }
257
+ let buf;
258
+ try {
259
+ const fullPath = scrapbookFilePath(ctx.projectRoot, ctx.config, site, path, item.name);
260
+ buf = readFileSync(fullPath);
261
+ }
262
+ catch (e) {
263
+ if (e instanceof Error && 'code' in e && e.code === 'ENOENT')
264
+ return '';
265
+ throw e;
266
+ }
267
+ if (item.kind === 'md' || item.kind === 'txt')
268
+ return `${countLines(buf)} lines`;
269
+ if (item.kind === 'json') {
270
+ const keys = countJsonKeys(buf);
271
+ return keys !== null ? `${keys} keys` : '';
272
+ }
273
+ // img
274
+ const dims = readImageDimensions(buf);
275
+ return dims ? `${dims.width} × ${dims.height}` : '';
276
+ }
277
+ /**
278
+ * Server-side preview for the closed-state card. Img → bg-frame URL;
279
+ * md → italic Newsreader excerpt with frontmatter stripped; json → mono
280
+ * pre with parse-then-stringify pretty-print; txt → mono pre raw excerpt.
281
+ * Other / empty / binary-as-text → no preview block.
282
+ */
283
+ function renderPreview(ctx, site, path, item, opts = {}) {
284
+ const { secret = false } = opts;
285
+ if (item.kind === 'img') {
286
+ const params = new URLSearchParams({ site, path, name: item.name });
287
+ if (secret)
288
+ params.set('secret', '1');
289
+ const url = `/api/dev/scrapbook-file?${params.toString()}`;
290
+ return unsafe(html `
291
+ <div class="scrap-preview scrap-preview--img" aria-hidden="true">
292
+ <div class="scrap-preview--img-frame" style="background-image: url(&quot;${url}&quot;);"></div>
293
+ </div>`);
294
+ }
295
+ if (item.kind !== 'md' && item.kind !== 'txt' && item.kind !== 'json') {
296
+ return unsafe('');
297
+ }
298
+ try {
299
+ const fullPath = scrapbookFilePath(ctx.projectRoot, ctx.config, site, path, item.name, secret ? { secret: true } : {});
300
+ const buf = readFileSync(fullPath);
301
+ const excerpt = previewExcerpt(buf, item.kind);
302
+ if (excerpt === null)
303
+ return unsafe('');
304
+ const safe = escapeHtml(excerpt);
305
+ if (item.kind === 'json' || item.kind === 'txt') {
306
+ return unsafe(html `
307
+ <pre class="scrap-preview scrap-preview--mono" aria-hidden="true">${unsafe(safe)}</pre>`);
75
308
  }
76
- else {
77
- links.push(html `<a class="scrapbook-breadcrumb-link" href="/dev/scrapbook/${site}/${prefix}">${segments[i]}</a>`);
309
+ return unsafe(html `
310
+ <div class="scrap-preview scrap-preview-md" aria-hidden="true"><p>${unsafe(safe)}</p></div>`);
311
+ }
312
+ catch (e) {
313
+ // ENOENT = file disappeared between listScrapbook and this read (race
314
+ // window with delete); rendering an empty preview is the right call.
315
+ // Anything else (EACCES, EISDIR, encoding bugs) propagates so the
316
+ // operator sees a real error instead of a silently-broken page.
317
+ if (e instanceof Error && 'code' in e && e.code === 'ENOENT') {
318
+ return unsafe('');
78
319
  }
320
+ throw e;
79
321
  }
80
- const sep = '<span class="scrapbook-breadcrumb-sep" aria-hidden="true">›</span>';
81
- const joined = links.join(`\n${sep}\n`);
322
+ }
323
+ function countByKind(items) {
324
+ const counts = {
325
+ all: items.length,
326
+ md: 0,
327
+ img: 0,
328
+ json: 0,
329
+ js: 0,
330
+ txt: 0,
331
+ other: 0,
332
+ };
333
+ for (const i of items)
334
+ counts[i.kind]++;
335
+ return counts;
336
+ }
337
+ function renderFilterChips(counts) {
338
+ const chip = (kind, label, isAll = false) => unsafe(html `
339
+ <button class="scrap-filter" type="button" data-filter="${kind}"
340
+ aria-pressed="${isAll ? 'true' : 'false'}">${label} · ${counts[kind]}</button>`);
341
+ return unsafe(html `
342
+ <div class="scrap-filters" role="toolbar" aria-label="filter by kind">
343
+ ${chip('all', 'all', true)}
344
+ ${chip('md', 'md')}
345
+ ${chip('img', 'img')}
346
+ ${chip('json', 'json')}
347
+ ${chip('txt', 'txt')}
348
+ ${chip('other', 'other')}
349
+ </div>`);
350
+ }
351
+ function renderSearch() {
352
+ return unsafe(html `
353
+ <div class="scrap-search">
354
+ <input type="search" placeholder="filter by name or content" aria-label="filter scrapbook" data-scrap-search />
355
+ <span class="scrap-search-kbd">/</span>
356
+ </div>`);
357
+ }
358
+ function renderBreadcrumb(site, path) {
359
+ const segments = path.split('/').filter(Boolean);
360
+ const last = segments[segments.length - 1] ?? path;
82
361
  return unsafe(html `
83
- <nav class="scrapbook-breadcrumb" aria-label="scrapbook hierarchy">
84
- <a class="scrapbook-breadcrumb-link" href="/dev/editorial-studio">${site}</a>
85
- <span class="scrapbook-breadcrumb-sep" aria-hidden="true">›</span>
86
- ${unsafe(joined)}
362
+ <nav class="scrap-breadcrumb" aria-label="hierarchy">
363
+ <a href="/dev/content/${site}">${site}</a><span class="sep">›</span>
364
+ <b>${last}</b>
87
365
  </nav>`);
88
366
  }
89
- function renderIndexSidebar(items, site, path) {
90
- const totalBytes = items.reduce((acc, item) => acc + item.size, 0);
91
- const lastModified = items.length > 0
92
- ? items.reduce((a, b) => (a.mtime > b.mtime ? a : b)).mtime
93
- : null;
367
+ function renderAside(site, path, items, totalSize, lastModified, secretCount, reviewLink) {
368
+ const lastModifiedLabel = lastModified ? formatRelativeTime(lastModified) : '—';
369
+ const publicCount = items.length;
370
+ const sizeLabel = formatSize(totalSize);
371
+ const folderLabel = path.split('/').filter(Boolean).pop() ?? path;
372
+ const fullPath = `${site}/${path}/scrapbook/`;
373
+ // #168 Phase 34 ship-pass — when this scrapbook belongs to a tracked
374
+ // calendar entry, expose a "← back to review" link so the operator
375
+ // who arrived from the entry-review surface (or via the dashboard's
376
+ // scrapbook chip) has an obvious path back. Pre-fix the only nav
377
+ // affordance was the breadcrumb's site link, which lands on the
378
+ // content tree, not the entry-review.
379
+ const backLink = reviewLink !== null
380
+ ? unsafe(html `<p class="scrap-aside-back"><a href="${reviewLink}">← back to review</a></p><hr />`)
381
+ : unsafe('');
94
382
  return unsafe(html `
95
- <aside class="scrapbook-index">
96
- <p class="scrapbook-index-kicker">
97
- <span aria-hidden="true">§</span> The folder
383
+ <aside class="scrap-aside">
384
+ ${backLink}
385
+ <p class="scrap-aside-kicker"><em>§</em> The folder</p>
386
+ <h1 class="scrap-aside-title">${folderLabel}</h1>
387
+ <p class="scrap-aside-meta">${site}</p>
388
+ <hr />
389
+ <p class="scrap-aside-totals">
390
+ <strong>${publicCount}</strong> public ·
391
+ <strong>${secretCount}</strong> secret ·
392
+ <em>${sizeLabel}</em>
98
393
  </p>
99
- <p class="scrapbook-index-meta">${path}</p>
100
- <p class="scrapbook-index-meta">${site}</p>
394
+ <p class="scrap-aside-meta">last modified ${lastModifiedLabel}</p>
101
395
  <hr />
102
- <ol class="scrapbook-index-list" data-scrapbook-index>
103
- ${items.map((item, i) => unsafe(html `<li data-index-for="${item.name}">
104
- <span class="scrapbook-index-num">No. ${String(i + 1).padStart(2, '0')}</span>
105
- <a href="#item-${encodeURIComponent(item.name)}">${item.name}</a>
106
- </li>`))}
396
+ <ol class="scrap-aside-list" data-scrap-aside-list>
397
+ ${items.map((item, i) => {
398
+ const seq = String(i + 1).padStart(2, '0');
399
+ return unsafe(html `<li><span class="num">${seq}</span><a href="#item-${i + 1}" data-scrap-aside-link>${item.name}</a></li>`);
400
+ })}
107
401
  </ol>
108
402
  <hr />
109
- <p class="scrapbook-index-totals">${items.length} ${items.length === 1 ? 'item' : 'items'} · ${formatSize(totalBytes)}</p>
110
- ${lastModified
111
- ? unsafe(html `<p class="scrapbook-index-subtotal">last modified ${formatRelativeTime(lastModified)}</p>`)
112
- : ''}
113
- <hr />
114
- <div class="scrapbook-index-actions">
115
- <button type="button" class="scrapbook-index-btn" data-action="new-note">+ new note</button>
116
- <button type="button" class="scrapbook-index-btn" data-action="upload">+ upload file</button>
403
+ <div class="scrap-aside-actions">
404
+ <button class="scrap-aside-btn scrap-aside-btn--primary" type="button" data-action="new-note">+ new note</button>
405
+ <button class="scrap-aside-btn" type="button" data-action="upload">+ upload file</button>
117
406
  </div>
118
407
  <hr />
119
- <p class="scrapbook-index-path">${site}/${path}/scrapbook/</p>
408
+ <p class="scrap-aside-path">${fullPath}</p>
120
409
  </aside>`);
121
410
  }
122
- function renderEmpty() {
411
+ function renderCard(ctx, site, path, item, index, opts = {}) {
412
+ const { secret = false } = opts;
413
+ const seq = String(index + 1).padStart(2, '0');
414
+ const kindLabel = KIND_LABEL[item.kind];
415
+ const kindClass = item.kind === 'other' ? '' : `scrap-kind--${item.kind}`;
416
+ const time = item.mtime
417
+ ? html `<time class="scrap-time" datetime="${item.mtime}">${formatRelativeTime(item.mtime)}</time>`
418
+ : '';
419
+ const preview = renderPreview(ctx, site, path, item, { secret });
420
+ const kindMeta = computeKindMeta(ctx, site, path, item);
421
+ const kindMetaHtml = kindMeta
422
+ ? unsafe(html `<span>·</span><span>${kindMeta}</span>`)
423
+ : unsafe('');
424
+ const editBtn = item.kind === 'img'
425
+ ? unsafe('')
426
+ : unsafe(html `<button class="scrap-tool" type="button" data-action="edit">edit</button>`);
427
+ // Secret cards get id="secret-item-N" to disambiguate from public ids in
428
+ // restoreFromHash + aside cross-link lookups (F4 contract); the
429
+ // mark-secret action toggle reads "mark public" since clicking it moves
430
+ // the card OUT of the secret section.
431
+ const id = secret ? `secret-item-${index + 1}` : `item-${index + 1}`;
432
+ const markSecretLabel = secret ? 'mark public' : 'mark secret';
433
+ const dataSecretAttr = secret ? ' data-secret="true"' : '';
434
+ // #164 Phase 34b — small ⚿ glyph next to .scrap-name on secret
435
+ // cards. Provides visual continuity for the secret marker once a
436
+ // card is expanded (where it grows outside the grouped section's
437
+ // visual scope).
438
+ const secretGlyph = secret
439
+ ? unsafe(html `<span class="scrap-name-secret-mark" aria-label="secret" title="secret — never published">⚿</span>`)
440
+ : unsafe('');
123
441
  return unsafe(html `
124
- <section class="scrapbook-empty">
125
- <p>
126
- This scrapbook is empty. Write the first note, or drop a file
127
- anywhere on this page.
128
- </p>
129
- <div class="scrapbook-empty-actions">
130
- <button type="button" class="scrapbook-index-btn" data-action="new-note">+ new note</button>
131
- <button type="button" class="scrapbook-index-btn" data-action="upload">+ upload file</button>
442
+ <li class="scrap-card" data-kind="${item.kind}" data-state="closed" id="${id}"${unsafe(dataSecretAttr)}>
443
+ <div class="scrap-card-head">
444
+ <span class="scrap-seq">N° ${seq}</span>
445
+ ${secretGlyph}
446
+ <span class="scrap-name" data-action="open">${item.name}</span>
447
+ ${unsafe(time)}
132
448
  </div>
133
- </section>`);
134
- }
135
- function renderReadingPanel(items) {
136
- return unsafe(html `
137
- <section class="scrapbook-reading">
138
- <form class="scrapbook-composer" data-scrapbook-composer hidden>
139
- <div class="scrapbook-composer-header">
140
- <span class="scrapbook-composer-seq" aria-hidden="true">✎</span>
141
- <span class="scrapbook-composer-kind">NEW</span>
142
- <input type="text" class="scrapbook-composer-filename" data-composer-filename
143
- placeholder="note-name.md" aria-label="new note filename" />
144
- <div class="scrapbook-editor-footer scrapbook-composer-actions">
145
- <label class="scrapbook-secret-toggle" title="save under scrapbook/secret/ — never published">
146
- <input type="checkbox" data-composer-secret />
147
- <span>secret</span>
148
- </label>
149
- <button type="button" class="scrapbook-tool" data-action="composer-cancel">cancel</button>
150
- <button type="submit" class="scrapbook-tool scrapbook-tool--primary" data-action="composer-save">save →</button>
151
- </div>
152
- </div>
153
- <div class="scrapbook-composer-body">
154
- <textarea data-composer-body
155
- placeholder="Write the note in markdown. Cmd/Ctrl+S saves."
156
- aria-label="new note body"></textarea>
157
- </div>
158
- </form>
159
- <ol class="scrapbook-items" data-scrapbook-items>
160
- ${items.map((item, i) => renderItemRow(item, i))}
161
- </ol>
162
- <div class="scrapbook-drop" data-scrapbook-drop role="button" tabindex="0"
163
- aria-label="upload a file to the scrapbook">
164
- <span class="scrapbook-drop-label">── drop a file here, or pick one ──</span>
165
- <input type="file" data-scrapbook-file-input
166
- accept="image/*,application/json,text/plain,text/markdown,.md,.json,.txt" />
167
- <label class="scrapbook-secret-toggle scrapbook-secret-toggle--upload"
168
- title="save the upload under scrapbook/secret/ — never published">
169
- <input type="checkbox" data-upload-secret />
170
- <span>upload as secret</span>
171
- </label>
449
+ <div class="scrap-card-meta">
450
+ <span class="scrap-kind ${kindClass}">${kindLabel}</span>
451
+ <span class="scrap-size">${formatSize(item.size)}</span>
452
+ ${kindMetaHtml}
172
453
  </div>
173
- </section>`);
454
+ ${preview}
455
+ <div class="scrap-card-foot">
456
+ <button class="scrap-tool scrap-tool--primary" type="button" data-action="open">open</button>
457
+ ${editBtn}
458
+ <button class="scrap-tool" type="button" data-action="rename">rename</button>
459
+ <button class="scrap-tool" type="button" data-action="mark-secret">${markSecretLabel}</button>
460
+ <span class="spacer"></span>
461
+ <button class="scrap-tool scrap-tool--delete" type="button" data-action="delete">delete</button>
462
+ </div>
463
+ </li>`);
174
464
  }
175
465
  /**
176
- * Quiet second section listing items inside `scrapbook/secret/`. Read-
177
- * only in v1 — operators populate the directory by hand or via the
178
- * core API; the studio surface just shows what's there. The "private"
179
- * badge gives unmistakable visual differentiation from the public
180
- * items above.
466
+ * Inline new-note composer (Phase 34b #166).
467
+ *
468
+ * Mirrors the pre-F1 inline composer (`44094ee^:scrapbook.ts:274-294`),
469
+ * adapted to the F1 `.scrap-*` design vocabulary. Hidden by default;
470
+ * the aside's `+ new note` button reveals it via the client wire-up.
471
+ *
472
+ * Per `.claude/rules/affordance-placement.md`: component-attached to
473
+ * the page (not a generic toolbar), placed where the resulting note
474
+ * will appear in sorted position. Direct manipulation: in-page form,
475
+ * filename + body + secret toggle visible inline, Cmd/Ctrl+S saves,
476
+ * Esc cancels. Replaces the F1 `window.prompt()` regression (#166).
181
477
  */
182
- function renderSecretSection(items) {
478
+ function renderComposer() {
183
479
  return unsafe(html `
184
- <section class="scrapbook-secret" data-scrapbook-secret>
185
- <header class="scrapbook-secret-header">
186
- <span class="scrapbook-secret-mark" aria-hidden="true">⚿</span>
187
- <h2 class="scrapbook-secret-title">Secret</h2>
188
- <span class="scrapbook-secret-badge" aria-label="private — never published">
189
- private
190
- </span>
191
- <span class="scrapbook-secret-count">
192
- ${items.length} ${items.length === 1 ? 'item' : 'items'}
193
- </span>
480
+ <form class="scrap-composer" data-scrap-composer hidden>
481
+ <header class="scrap-composer-head">
482
+ <span class="scrap-composer-glyph" aria-hidden="true">✎</span>
483
+ <span class="scrap-composer-kicker">NEW NOTE</span>
484
+ <input type="text" class="scrap-composer-filename" data-composer-filename
485
+ placeholder="note-name.md" aria-label="new note filename" />
486
+ <label class="scrap-composer-secret" title="save under scrapbook/secret/ — never published">
487
+ <input type="checkbox" data-composer-secret />
488
+ <span>secret</span>
489
+ </label>
490
+ <button class="scrap-tool" type="button" data-action="composer-cancel">cancel</button>
491
+ <button class="scrap-tool scrap-tool--primary" type="submit" data-action="composer-save">save →</button>
194
492
  </header>
195
- <p class="scrapbook-secret-help">
196
- Items inside <code>scrapbook/secret/</code>. Excluded from the
197
- public site by the host's content-collection patterns.
198
- </p>
199
- <ol class="scrapbook-items scrapbook-items--secret">
200
- ${items.map((item, i) => renderItemRow(item, i, { secret: true, withTools: true }))}
493
+ <textarea class="scrap-composer-body" data-composer-body
494
+ placeholder="Write the note in markdown. Cmd/Ctrl+S saves, Esc cancels."
495
+ aria-label="new note body" rows="8"></textarea>
496
+ </form>`);
497
+ }
498
+ function renderDropZone() {
499
+ return unsafe(html `
500
+ <div class="scrap-drop" role="button" tabindex="0" data-action="upload"
501
+ aria-label="Drop a file here, or press Enter to pick one">
502
+ ── drop a file here, or pick one ──
503
+ </div>`);
504
+ }
505
+ function renderSecretSection(ctx, site, path, secretItems) {
506
+ if (secretItems.length === 0)
507
+ return unsafe('');
508
+ const cards = secretItems.map((item, i) => renderCard(ctx, site, path, item, i, { secret: true }));
509
+ return unsafe(html `
510
+ <section class="scrap-secret" aria-label="secret items">
511
+ <header class="scrap-secret-head">
512
+ <span class="scrap-secret-mark" aria-hidden="true">⚿</span>
513
+ <h2 class="scrap-secret-title">Secret</h2>
514
+ <span class="scrap-secret-badge">private — never published</span>
515
+ </header>
516
+ <ol class="scrap-cards">
517
+ ${cards}
201
518
  </ol>
202
519
  </section>`);
203
520
  }
521
+ /**
522
+ * #168 Phase 34 ship-pass — when the scrapbook path matches a tracked
523
+ * calendar entry with a stamped UUID, return the entry-keyed review
524
+ * URL so the aside can render a "← back to review" link. Returns null
525
+ * when no entry matches (organizational subdirs, ad-hoc paths, or
526
+ * pre-doctor entries lacking an id) — the link is then omitted.
527
+ *
528
+ * Failures (calendar absent, parse error) fall through to null so a
529
+ * transient calendar issue never blocks the scrapbook render.
530
+ */
531
+ function lookupEntryReviewLink(ctx, site, path) {
532
+ if (!(site in ctx.config.sites))
533
+ return null;
534
+ try {
535
+ const calendarPath = resolveCalendarPath(ctx.projectRoot, ctx.config, site);
536
+ if (!existsSync(calendarPath))
537
+ return null;
538
+ const cal = readCalendar(calendarPath);
539
+ const entry = findEntry(cal, path);
540
+ if (!entry || !entry.id)
541
+ return null;
542
+ return `/dev/editorial-review/entry/${entry.id}`;
543
+ }
544
+ catch {
545
+ return null;
546
+ }
547
+ }
204
548
  export function renderScrapbookPage(ctx, site, path) {
205
549
  // Validate site against the project's configured site list. Without
206
550
  // this check, an unknown site key reaches the path resolver and
@@ -208,41 +552,53 @@ export function renderScrapbookPage(ctx, site, path) {
208
552
  if (!(site in ctx.config.sites)) {
209
553
  throw new Error(`unknown site: ${site}`);
210
554
  }
211
- const summary = listScrapbook(ctx.projectRoot, ctx.config, site, path);
212
- const items = summary.items;
213
- const secretItems = summary.secretItems;
214
- const publicBlock = items.length === 0
215
- ? renderEmpty().__raw
216
- : renderReadingPanel(items).__raw + renderIndexSidebar(items, site, path).__raw;
217
- const secretBlock = secretItems.length > 0 ? renderSecretSection(secretItems).__raw : '';
555
+ // listScrapbook returns { exists: false, items: [] } for missing dirs
556
+ // (packages/core/src/scrapbook.ts:337-339), so an empty scrapbook is not
557
+ // an error path. Real errors (slug validation, scrapbookDir resolution
558
+ // failures, FS permission issues) propagate to the studio's error handler.
559
+ const result = listScrapbook(ctx.projectRoot, ctx.config, site, path);
560
+ const items = result.items;
561
+ const secretItems = result.secretItems;
562
+ const totalSize = items.reduce((s, i) => s + i.size, 0);
563
+ const lastModified = items.reduce((acc, i) => {
564
+ if (!i.mtime)
565
+ return acc;
566
+ if (!acc || i.mtime > acc)
567
+ return i.mtime;
568
+ return acc;
569
+ }, null);
570
+ const counts = countByKind(items);
571
+ const folderLabel = path.split('/').filter(Boolean).pop() ?? path;
572
+ const cards = items.map((item, i) => renderCard(ctx, site, path, item, i));
573
+ const cardsHtml = cards.map((c) => c.__raw).join('');
574
+ const reviewLink = lookupEntryReviewLink(ctx, site, path);
218
575
  const body = html `
219
576
  ${renderEditorialFolio('content', `scrapbook · ${site}/${path}`)}
220
- <main class="scrapbook-page" data-site="${site}" data-slug="${path}" data-scrapbook-root>
221
- <header class="er-pagehead er-pagehead--compact scrapbook-header">
222
- ${renderBreadcrumb(site, path)}
223
- <p class="er-pagehead__kicker scrapbook-kicker">
224
- <span class="scrapbook-kicker-mark" aria-hidden="true">§</span>
225
- Scrapbook
226
- </p>
227
- <h1 class="er-pagehead__title scrapbook-title">${path}</h1>
228
- <a class="scrapbook-back" href="/dev/editorial-studio">← back to the desk</a>
229
- </header>
230
- <div class="scrapbook-status" data-scrapbook-status hidden></div>
231
- ${unsafe(publicBlock)}
232
- ${unsafe(secretBlock)}
233
- <div class="scrapbook-drop-overlay" data-scrapbook-overlay aria-hidden="true">
234
- <span class="scrapbook-drop-overlay-text">drop to add to the scrapbook ◇</span>
235
- </div>
577
+ <main class="scrap-page" data-site="${site}" data-path="${path}">
578
+ ${renderAside(site, path, items, totalSize, lastModified, secretItems.length, reviewLink)}
579
+ <section class="scrap-main">
580
+ <header class="scrap-main-header">
581
+ ${renderBreadcrumb(site, path)}
582
+ ${renderSearch()}
583
+ </header>
584
+ ${renderFilterChips(counts)}
585
+ ${renderComposer()}
586
+ <ol class="scrap-cards" id="cards" data-scrap-cards>
587
+ ${unsafe(cardsHtml)}
588
+ </ol>
589
+ ${renderDropZone()}
590
+ ${renderSecretSection(ctx, site, path, secretItems)}
591
+ </section>
236
592
  </main>`;
237
593
  return layout({
238
- title: `scrapbook · ${path} — dev`,
594
+ title: `scrapbook · ${folderLabel} — dev`,
239
595
  cssHrefs: [
240
596
  '/static/css/editorial-review.css',
241
597
  '/static/css/editorial-nav.css',
242
598
  '/static/css/scrapbook.css',
243
599
  '/static/css/blog-figure.css',
244
600
  ],
245
- bodyAttrs: 'data-review-ui="studio"',
601
+ bodyAttrs: 'data-review-ui="scrapbook"',
246
602
  bodyHtml: body,
247
603
  scriptModules: ['scrapbook-client'],
248
604
  });