create-volt 0.47.0 → 0.48.1
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 +17 -0
- package/package.json +1 -1
- package/templates/blog/setup/index.html +1 -0
- package/templates/blog/setup/setup.js +38 -10
- package/templates/default/setup/index.html +1 -0
- package/templates/default/setup/setup.js +38 -10
- package/templates/docs/setup/index.html +1 -0
- package/templates/docs/setup/setup.js +38 -10
- package/templates/starter/setup/index.html +1 -0
- package/templates/starter/setup/setup.js +38 -10
package/CHANGELOG.md
CHANGED
|
@@ -4,6 +4,21 @@ All notable changes to `create-volt` are documented here. The format follows
|
|
|
4
4
|
[Keep a Changelog](https://keepachangelog.com/), and this project adheres to
|
|
5
5
|
[Semantic Versioning](https://semver.org/).
|
|
6
6
|
|
|
7
|
+
## [0.48.1] - 2026-06-29
|
|
8
|
+
|
|
9
|
+
### Changed
|
|
10
|
+
- Config WYSIWYG loads RTEPro at a **major-version float** (`@1`) instead of a
|
|
11
|
+
pinned patch, so RTEPro 1.x updates flow without a create-volt release. Dropped
|
|
12
|
+
the marked dependency entirely — RTEPro takes markdown directly via setMarkdown().
|
|
13
|
+
|
|
14
|
+
## [0.48.0] - 2026-06-29
|
|
15
|
+
|
|
16
|
+
### Added
|
|
17
|
+
- **WYSIWYG editor in the config.** Manage content embeds the RTEPro rich editor
|
|
18
|
+
(loaded from CDN) instead of a raw textarea — visual editing, shell-gated, no
|
|
19
|
+
public route or auth. Opens markdown rendered to HTML, saves markdown (or HTML
|
|
20
|
+
for complex layouts), with a title field beside the slug.
|
|
21
|
+
|
|
7
22
|
## [0.47.0] - 2026-06-29
|
|
8
23
|
|
|
9
24
|
### Added
|
|
@@ -620,6 +635,8 @@ All notable changes to `create-volt` are documented here. The format follows
|
|
|
620
635
|
watching and full-page hot reload. Supports `--skip-install` and `--force`,
|
|
621
636
|
and auto-detects npm / pnpm / yarn / bun for the install step.
|
|
622
637
|
|
|
638
|
+
[0.48.1]: https://github.com/MIR-2025/volt/releases/tag/v0.48.1
|
|
639
|
+
[0.48.0]: https://github.com/MIR-2025/volt/releases/tag/v0.48.0
|
|
623
640
|
[0.47.0]: https://github.com/MIR-2025/volt/releases/tag/v0.47.0
|
|
624
641
|
[0.46.0]: https://github.com/MIR-2025/volt/releases/tag/v0.46.0
|
|
625
642
|
[0.45.1]: https://github.com/MIR-2025/volt/releases/tag/v0.45.1
|
package/package.json
CHANGED
|
@@ -300,24 +300,52 @@ async function buyCredits(amountUsd) {
|
|
|
300
300
|
}
|
|
301
301
|
}
|
|
302
302
|
const items = signal({ pages: [], posts: [] });
|
|
303
|
-
const editing = signal(null); // { type, slug,
|
|
303
|
+
const editing = signal(null); // { type, slug, title, isNew } — set only on open/save/close
|
|
304
|
+
let ed = null; // live RTEPro instance for the open editor
|
|
304
305
|
const loadItems = async () => items(await (await fetch("/setup/content")).json());
|
|
306
|
+
// raw .md → { title, body, isHtml }; RTEPro takes markdown directly (setMarkdown),
|
|
307
|
+
// so no markdown library is needed.
|
|
308
|
+
function parseDoc(raw) {
|
|
309
|
+
const fm = raw.match(/^---\r?\n([\s\S]*?)\r?\n---\r?\n/);
|
|
310
|
+
const front = fm ? fm[1] : "";
|
|
311
|
+
const title = ((front.match(/^title:\s*(.+)$/m) || [])[1] || "").trim();
|
|
312
|
+
return { title, body: fm ? raw.slice(fm[0].length) : raw, isHtml: /^format:\s*html\s*$/m.test(front) };
|
|
313
|
+
}
|
|
314
|
+
function mountEditor(doc) {
|
|
315
|
+
ed = window.RTEPro.init("#mg-editor", { height: "60vh", placeholder: "Write…" });
|
|
316
|
+
if (doc && doc.isHtml) ed.setHTML(doc.body || "");
|
|
317
|
+
else ed.setMarkdown((doc && doc.body) || "");
|
|
318
|
+
}
|
|
305
319
|
async function editItem(type, slug) {
|
|
306
320
|
const d = await (await fetch(`/setup/content/raw?type=${type}&slug=${encodeURIComponent(slug)}`)).json();
|
|
307
|
-
|
|
321
|
+
const doc = parseDoc(d.body || "");
|
|
322
|
+
editing({ type, slug, title: doc.title, isNew: false });
|
|
323
|
+
queueMicrotask(() => mountEditor(doc));
|
|
308
324
|
}
|
|
309
325
|
function newItem(type) {
|
|
310
|
-
|
|
311
|
-
|
|
326
|
+
editing({ type, slug: "", title: "", isNew: true });
|
|
327
|
+
queueMicrotask(() => mountEditor({ body: "", isHtml: false }));
|
|
328
|
+
}
|
|
329
|
+
// markdown can't round-trip complex layouts (columns, inline styles, merged cells,
|
|
330
|
+
// embeds) — save those as HTML so they aren't flattened.
|
|
331
|
+
function isComplex(h) {
|
|
332
|
+
return /\bstyle\s*=\s*["'][^"']*(text-align|column|float|grid|flex|width|height|color|background|font|margin|padding)/i.test(h) || /\b(colspan|rowspan)\b/i.test(h) || /<(u|font|mark|sub|sup|iframe|video|audio|figure)\b/i.test(h) || /class\s*=\s*["'][^"']*(col|grid|row|flex|layout)/i.test(h);
|
|
312
333
|
}
|
|
313
334
|
async function saveItem() {
|
|
314
335
|
const e = editing();
|
|
315
336
|
const slug = (document.querySelector("#mg-slug").value || "").trim().toLowerCase();
|
|
316
|
-
const body = document.querySelector("#mg-body").value;
|
|
317
337
|
if (!/^[a-z0-9][a-z0-9-]*$/.test(slug)) return status("Slug must be lowercase letters, numbers, hyphens.");
|
|
338
|
+
const title = (document.querySelector("#mg-title").value || "").trim() || slug;
|
|
339
|
+
const htmlOut = ed ? ed.getHTML() : "";
|
|
340
|
+
const complex = isComplex(htmlOut);
|
|
341
|
+
const front = [`title: ${title}`];
|
|
342
|
+
if (complex) front.push("format: html");
|
|
343
|
+
const docBody = complex ? htmlOut : ed ? ed.getMarkdown() : "";
|
|
344
|
+
const body = `---\n${front.join("\n")}\n---\n\n${docBody}\n`;
|
|
318
345
|
const r = await (await fetch("/setup/content/save", { method: "POST", headers: { "Content-Type": "application/json" }, body: JSON.stringify({ type: e.type, slug, body }) })).json();
|
|
319
346
|
if (!r.ok) return status("Error: " + (r.error || "?"));
|
|
320
|
-
status("Saved → " + r.file);
|
|
347
|
+
status("Saved → " + r.file + (complex ? " (HTML — complex layout)" : ""));
|
|
348
|
+
ed = null;
|
|
321
349
|
editing(null);
|
|
322
350
|
loadItems();
|
|
323
351
|
}
|
|
@@ -339,10 +367,10 @@ const section = (label, type, key) =>
|
|
|
339
367
|
${() => (items()[key].length ? html`<ul class="list-group">${items()[key].map(itemRow)}</ul>` : html`<div class="small text-muted">No ${key} yet.</div>`)}
|
|
340
368
|
</div>`;
|
|
341
369
|
const editorPanel = () => {
|
|
342
|
-
const e = editing(); // inputs
|
|
343
|
-
return html`<div class="p-3 mb-2" style="border:1px solid
|
|
344
|
-
<div class="d-flex gap-2 mb-2"><input id="mg-slug" class="form-control" placeholder="slug" value=${e.slug} readonly=${!e.isNew} /><span class="align-self-center small text-muted">${e.type === "post" ? "posts/" : "pages/"}</span></div>
|
|
345
|
-
<
|
|
370
|
+
const e = editing(); // inputs uncontrolled (read on Save); RTEPro mounts into #mg-editor
|
|
371
|
+
return html`<div class="p-3 mb-2" style="border:1px solid var(--border,#232a36);border-radius:10px">
|
|
372
|
+
<div class="d-flex gap-2 mb-2"><input id="mg-slug" class="form-control" placeholder="slug" value=${e.slug} readonly=${!e.isNew} style="max-width:200px" /><input id="mg-title" class="form-control" placeholder="Title" value=${e.title || ""} /><span class="align-self-center small text-muted">${e.type === "post" ? "posts/" : "pages/"}</span></div>
|
|
373
|
+
<div id="mg-editor"></div>
|
|
346
374
|
<div class="mt-2 d-flex gap-2"><button class="btn btn-primary btn-sm" onclick=${saveItem}>Save</button><button class="btn btn-outline-secondary btn-sm" onclick=${() => editing(null)}>Cancel</button></div>
|
|
347
375
|
</div>`;
|
|
348
376
|
};
|
|
@@ -300,24 +300,52 @@ async function buyCredits(amountUsd) {
|
|
|
300
300
|
}
|
|
301
301
|
}
|
|
302
302
|
const items = signal({ pages: [], posts: [] });
|
|
303
|
-
const editing = signal(null); // { type, slug,
|
|
303
|
+
const editing = signal(null); // { type, slug, title, isNew } — set only on open/save/close
|
|
304
|
+
let ed = null; // live RTEPro instance for the open editor
|
|
304
305
|
const loadItems = async () => items(await (await fetch("/setup/content")).json());
|
|
306
|
+
// raw .md → { title, body, isHtml }; RTEPro takes markdown directly (setMarkdown),
|
|
307
|
+
// so no markdown library is needed.
|
|
308
|
+
function parseDoc(raw) {
|
|
309
|
+
const fm = raw.match(/^---\r?\n([\s\S]*?)\r?\n---\r?\n/);
|
|
310
|
+
const front = fm ? fm[1] : "";
|
|
311
|
+
const title = ((front.match(/^title:\s*(.+)$/m) || [])[1] || "").trim();
|
|
312
|
+
return { title, body: fm ? raw.slice(fm[0].length) : raw, isHtml: /^format:\s*html\s*$/m.test(front) };
|
|
313
|
+
}
|
|
314
|
+
function mountEditor(doc) {
|
|
315
|
+
ed = window.RTEPro.init("#mg-editor", { height: "60vh", placeholder: "Write…" });
|
|
316
|
+
if (doc && doc.isHtml) ed.setHTML(doc.body || "");
|
|
317
|
+
else ed.setMarkdown((doc && doc.body) || "");
|
|
318
|
+
}
|
|
305
319
|
async function editItem(type, slug) {
|
|
306
320
|
const d = await (await fetch(`/setup/content/raw?type=${type}&slug=${encodeURIComponent(slug)}`)).json();
|
|
307
|
-
|
|
321
|
+
const doc = parseDoc(d.body || "");
|
|
322
|
+
editing({ type, slug, title: doc.title, isNew: false });
|
|
323
|
+
queueMicrotask(() => mountEditor(doc));
|
|
308
324
|
}
|
|
309
325
|
function newItem(type) {
|
|
310
|
-
|
|
311
|
-
|
|
326
|
+
editing({ type, slug: "", title: "", isNew: true });
|
|
327
|
+
queueMicrotask(() => mountEditor({ body: "", isHtml: false }));
|
|
328
|
+
}
|
|
329
|
+
// markdown can't round-trip complex layouts (columns, inline styles, merged cells,
|
|
330
|
+
// embeds) — save those as HTML so they aren't flattened.
|
|
331
|
+
function isComplex(h) {
|
|
332
|
+
return /\bstyle\s*=\s*["'][^"']*(text-align|column|float|grid|flex|width|height|color|background|font|margin|padding)/i.test(h) || /\b(colspan|rowspan)\b/i.test(h) || /<(u|font|mark|sub|sup|iframe|video|audio|figure)\b/i.test(h) || /class\s*=\s*["'][^"']*(col|grid|row|flex|layout)/i.test(h);
|
|
312
333
|
}
|
|
313
334
|
async function saveItem() {
|
|
314
335
|
const e = editing();
|
|
315
336
|
const slug = (document.querySelector("#mg-slug").value || "").trim().toLowerCase();
|
|
316
|
-
const body = document.querySelector("#mg-body").value;
|
|
317
337
|
if (!/^[a-z0-9][a-z0-9-]*$/.test(slug)) return status("Slug must be lowercase letters, numbers, hyphens.");
|
|
338
|
+
const title = (document.querySelector("#mg-title").value || "").trim() || slug;
|
|
339
|
+
const htmlOut = ed ? ed.getHTML() : "";
|
|
340
|
+
const complex = isComplex(htmlOut);
|
|
341
|
+
const front = [`title: ${title}`];
|
|
342
|
+
if (complex) front.push("format: html");
|
|
343
|
+
const docBody = complex ? htmlOut : ed ? ed.getMarkdown() : "";
|
|
344
|
+
const body = `---\n${front.join("\n")}\n---\n\n${docBody}\n`;
|
|
318
345
|
const r = await (await fetch("/setup/content/save", { method: "POST", headers: { "Content-Type": "application/json" }, body: JSON.stringify({ type: e.type, slug, body }) })).json();
|
|
319
346
|
if (!r.ok) return status("Error: " + (r.error || "?"));
|
|
320
|
-
status("Saved → " + r.file);
|
|
347
|
+
status("Saved → " + r.file + (complex ? " (HTML — complex layout)" : ""));
|
|
348
|
+
ed = null;
|
|
321
349
|
editing(null);
|
|
322
350
|
loadItems();
|
|
323
351
|
}
|
|
@@ -339,10 +367,10 @@ const section = (label, type, key) =>
|
|
|
339
367
|
${() => (items()[key].length ? html`<ul class="list-group">${items()[key].map(itemRow)}</ul>` : html`<div class="small text-muted">No ${key} yet.</div>`)}
|
|
340
368
|
</div>`;
|
|
341
369
|
const editorPanel = () => {
|
|
342
|
-
const e = editing(); // inputs
|
|
343
|
-
return html`<div class="p-3 mb-2" style="border:1px solid
|
|
344
|
-
<div class="d-flex gap-2 mb-2"><input id="mg-slug" class="form-control" placeholder="slug" value=${e.slug} readonly=${!e.isNew} /><span class="align-self-center small text-muted">${e.type === "post" ? "posts/" : "pages/"}</span></div>
|
|
345
|
-
<
|
|
370
|
+
const e = editing(); // inputs uncontrolled (read on Save); RTEPro mounts into #mg-editor
|
|
371
|
+
return html`<div class="p-3 mb-2" style="border:1px solid var(--border,#232a36);border-radius:10px">
|
|
372
|
+
<div class="d-flex gap-2 mb-2"><input id="mg-slug" class="form-control" placeholder="slug" value=${e.slug} readonly=${!e.isNew} style="max-width:200px" /><input id="mg-title" class="form-control" placeholder="Title" value=${e.title || ""} /><span class="align-self-center small text-muted">${e.type === "post" ? "posts/" : "pages/"}</span></div>
|
|
373
|
+
<div id="mg-editor"></div>
|
|
346
374
|
<div class="mt-2 d-flex gap-2"><button class="btn btn-primary btn-sm" onclick=${saveItem}>Save</button><button class="btn btn-outline-secondary btn-sm" onclick=${() => editing(null)}>Cancel</button></div>
|
|
347
375
|
</div>`;
|
|
348
376
|
};
|
|
@@ -300,24 +300,52 @@ async function buyCredits(amountUsd) {
|
|
|
300
300
|
}
|
|
301
301
|
}
|
|
302
302
|
const items = signal({ pages: [], posts: [] });
|
|
303
|
-
const editing = signal(null); // { type, slug,
|
|
303
|
+
const editing = signal(null); // { type, slug, title, isNew } — set only on open/save/close
|
|
304
|
+
let ed = null; // live RTEPro instance for the open editor
|
|
304
305
|
const loadItems = async () => items(await (await fetch("/setup/content")).json());
|
|
306
|
+
// raw .md → { title, body, isHtml }; RTEPro takes markdown directly (setMarkdown),
|
|
307
|
+
// so no markdown library is needed.
|
|
308
|
+
function parseDoc(raw) {
|
|
309
|
+
const fm = raw.match(/^---\r?\n([\s\S]*?)\r?\n---\r?\n/);
|
|
310
|
+
const front = fm ? fm[1] : "";
|
|
311
|
+
const title = ((front.match(/^title:\s*(.+)$/m) || [])[1] || "").trim();
|
|
312
|
+
return { title, body: fm ? raw.slice(fm[0].length) : raw, isHtml: /^format:\s*html\s*$/m.test(front) };
|
|
313
|
+
}
|
|
314
|
+
function mountEditor(doc) {
|
|
315
|
+
ed = window.RTEPro.init("#mg-editor", { height: "60vh", placeholder: "Write…" });
|
|
316
|
+
if (doc && doc.isHtml) ed.setHTML(doc.body || "");
|
|
317
|
+
else ed.setMarkdown((doc && doc.body) || "");
|
|
318
|
+
}
|
|
305
319
|
async function editItem(type, slug) {
|
|
306
320
|
const d = await (await fetch(`/setup/content/raw?type=${type}&slug=${encodeURIComponent(slug)}`)).json();
|
|
307
|
-
|
|
321
|
+
const doc = parseDoc(d.body || "");
|
|
322
|
+
editing({ type, slug, title: doc.title, isNew: false });
|
|
323
|
+
queueMicrotask(() => mountEditor(doc));
|
|
308
324
|
}
|
|
309
325
|
function newItem(type) {
|
|
310
|
-
|
|
311
|
-
|
|
326
|
+
editing({ type, slug: "", title: "", isNew: true });
|
|
327
|
+
queueMicrotask(() => mountEditor({ body: "", isHtml: false }));
|
|
328
|
+
}
|
|
329
|
+
// markdown can't round-trip complex layouts (columns, inline styles, merged cells,
|
|
330
|
+
// embeds) — save those as HTML so they aren't flattened.
|
|
331
|
+
function isComplex(h) {
|
|
332
|
+
return /\bstyle\s*=\s*["'][^"']*(text-align|column|float|grid|flex|width|height|color|background|font|margin|padding)/i.test(h) || /\b(colspan|rowspan)\b/i.test(h) || /<(u|font|mark|sub|sup|iframe|video|audio|figure)\b/i.test(h) || /class\s*=\s*["'][^"']*(col|grid|row|flex|layout)/i.test(h);
|
|
312
333
|
}
|
|
313
334
|
async function saveItem() {
|
|
314
335
|
const e = editing();
|
|
315
336
|
const slug = (document.querySelector("#mg-slug").value || "").trim().toLowerCase();
|
|
316
|
-
const body = document.querySelector("#mg-body").value;
|
|
317
337
|
if (!/^[a-z0-9][a-z0-9-]*$/.test(slug)) return status("Slug must be lowercase letters, numbers, hyphens.");
|
|
338
|
+
const title = (document.querySelector("#mg-title").value || "").trim() || slug;
|
|
339
|
+
const htmlOut = ed ? ed.getHTML() : "";
|
|
340
|
+
const complex = isComplex(htmlOut);
|
|
341
|
+
const front = [`title: ${title}`];
|
|
342
|
+
if (complex) front.push("format: html");
|
|
343
|
+
const docBody = complex ? htmlOut : ed ? ed.getMarkdown() : "";
|
|
344
|
+
const body = `---\n${front.join("\n")}\n---\n\n${docBody}\n`;
|
|
318
345
|
const r = await (await fetch("/setup/content/save", { method: "POST", headers: { "Content-Type": "application/json" }, body: JSON.stringify({ type: e.type, slug, body }) })).json();
|
|
319
346
|
if (!r.ok) return status("Error: " + (r.error || "?"));
|
|
320
|
-
status("Saved → " + r.file);
|
|
347
|
+
status("Saved → " + r.file + (complex ? " (HTML — complex layout)" : ""));
|
|
348
|
+
ed = null;
|
|
321
349
|
editing(null);
|
|
322
350
|
loadItems();
|
|
323
351
|
}
|
|
@@ -339,10 +367,10 @@ const section = (label, type, key) =>
|
|
|
339
367
|
${() => (items()[key].length ? html`<ul class="list-group">${items()[key].map(itemRow)}</ul>` : html`<div class="small text-muted">No ${key} yet.</div>`)}
|
|
340
368
|
</div>`;
|
|
341
369
|
const editorPanel = () => {
|
|
342
|
-
const e = editing(); // inputs
|
|
343
|
-
return html`<div class="p-3 mb-2" style="border:1px solid
|
|
344
|
-
<div class="d-flex gap-2 mb-2"><input id="mg-slug" class="form-control" placeholder="slug" value=${e.slug} readonly=${!e.isNew} /><span class="align-self-center small text-muted">${e.type === "post" ? "posts/" : "pages/"}</span></div>
|
|
345
|
-
<
|
|
370
|
+
const e = editing(); // inputs uncontrolled (read on Save); RTEPro mounts into #mg-editor
|
|
371
|
+
return html`<div class="p-3 mb-2" style="border:1px solid var(--border,#232a36);border-radius:10px">
|
|
372
|
+
<div class="d-flex gap-2 mb-2"><input id="mg-slug" class="form-control" placeholder="slug" value=${e.slug} readonly=${!e.isNew} style="max-width:200px" /><input id="mg-title" class="form-control" placeholder="Title" value=${e.title || ""} /><span class="align-self-center small text-muted">${e.type === "post" ? "posts/" : "pages/"}</span></div>
|
|
373
|
+
<div id="mg-editor"></div>
|
|
346
374
|
<div class="mt-2 d-flex gap-2"><button class="btn btn-primary btn-sm" onclick=${saveItem}>Save</button><button class="btn btn-outline-secondary btn-sm" onclick=${() => editing(null)}>Cancel</button></div>
|
|
347
375
|
</div>`;
|
|
348
376
|
};
|
|
@@ -300,24 +300,52 @@ async function buyCredits(amountUsd) {
|
|
|
300
300
|
}
|
|
301
301
|
}
|
|
302
302
|
const items = signal({ pages: [], posts: [] });
|
|
303
|
-
const editing = signal(null); // { type, slug,
|
|
303
|
+
const editing = signal(null); // { type, slug, title, isNew } — set only on open/save/close
|
|
304
|
+
let ed = null; // live RTEPro instance for the open editor
|
|
304
305
|
const loadItems = async () => items(await (await fetch("/setup/content")).json());
|
|
306
|
+
// raw .md → { title, body, isHtml }; RTEPro takes markdown directly (setMarkdown),
|
|
307
|
+
// so no markdown library is needed.
|
|
308
|
+
function parseDoc(raw) {
|
|
309
|
+
const fm = raw.match(/^---\r?\n([\s\S]*?)\r?\n---\r?\n/);
|
|
310
|
+
const front = fm ? fm[1] : "";
|
|
311
|
+
const title = ((front.match(/^title:\s*(.+)$/m) || [])[1] || "").trim();
|
|
312
|
+
return { title, body: fm ? raw.slice(fm[0].length) : raw, isHtml: /^format:\s*html\s*$/m.test(front) };
|
|
313
|
+
}
|
|
314
|
+
function mountEditor(doc) {
|
|
315
|
+
ed = window.RTEPro.init("#mg-editor", { height: "60vh", placeholder: "Write…" });
|
|
316
|
+
if (doc && doc.isHtml) ed.setHTML(doc.body || "");
|
|
317
|
+
else ed.setMarkdown((doc && doc.body) || "");
|
|
318
|
+
}
|
|
305
319
|
async function editItem(type, slug) {
|
|
306
320
|
const d = await (await fetch(`/setup/content/raw?type=${type}&slug=${encodeURIComponent(slug)}`)).json();
|
|
307
|
-
|
|
321
|
+
const doc = parseDoc(d.body || "");
|
|
322
|
+
editing({ type, slug, title: doc.title, isNew: false });
|
|
323
|
+
queueMicrotask(() => mountEditor(doc));
|
|
308
324
|
}
|
|
309
325
|
function newItem(type) {
|
|
310
|
-
|
|
311
|
-
|
|
326
|
+
editing({ type, slug: "", title: "", isNew: true });
|
|
327
|
+
queueMicrotask(() => mountEditor({ body: "", isHtml: false }));
|
|
328
|
+
}
|
|
329
|
+
// markdown can't round-trip complex layouts (columns, inline styles, merged cells,
|
|
330
|
+
// embeds) — save those as HTML so they aren't flattened.
|
|
331
|
+
function isComplex(h) {
|
|
332
|
+
return /\bstyle\s*=\s*["'][^"']*(text-align|column|float|grid|flex|width|height|color|background|font|margin|padding)/i.test(h) || /\b(colspan|rowspan)\b/i.test(h) || /<(u|font|mark|sub|sup|iframe|video|audio|figure)\b/i.test(h) || /class\s*=\s*["'][^"']*(col|grid|row|flex|layout)/i.test(h);
|
|
312
333
|
}
|
|
313
334
|
async function saveItem() {
|
|
314
335
|
const e = editing();
|
|
315
336
|
const slug = (document.querySelector("#mg-slug").value || "").trim().toLowerCase();
|
|
316
|
-
const body = document.querySelector("#mg-body").value;
|
|
317
337
|
if (!/^[a-z0-9][a-z0-9-]*$/.test(slug)) return status("Slug must be lowercase letters, numbers, hyphens.");
|
|
338
|
+
const title = (document.querySelector("#mg-title").value || "").trim() || slug;
|
|
339
|
+
const htmlOut = ed ? ed.getHTML() : "";
|
|
340
|
+
const complex = isComplex(htmlOut);
|
|
341
|
+
const front = [`title: ${title}`];
|
|
342
|
+
if (complex) front.push("format: html");
|
|
343
|
+
const docBody = complex ? htmlOut : ed ? ed.getMarkdown() : "";
|
|
344
|
+
const body = `---\n${front.join("\n")}\n---\n\n${docBody}\n`;
|
|
318
345
|
const r = await (await fetch("/setup/content/save", { method: "POST", headers: { "Content-Type": "application/json" }, body: JSON.stringify({ type: e.type, slug, body }) })).json();
|
|
319
346
|
if (!r.ok) return status("Error: " + (r.error || "?"));
|
|
320
|
-
status("Saved → " + r.file);
|
|
347
|
+
status("Saved → " + r.file + (complex ? " (HTML — complex layout)" : ""));
|
|
348
|
+
ed = null;
|
|
321
349
|
editing(null);
|
|
322
350
|
loadItems();
|
|
323
351
|
}
|
|
@@ -339,10 +367,10 @@ const section = (label, type, key) =>
|
|
|
339
367
|
${() => (items()[key].length ? html`<ul class="list-group">${items()[key].map(itemRow)}</ul>` : html`<div class="small text-muted">No ${key} yet.</div>`)}
|
|
340
368
|
</div>`;
|
|
341
369
|
const editorPanel = () => {
|
|
342
|
-
const e = editing(); // inputs
|
|
343
|
-
return html`<div class="p-3 mb-2" style="border:1px solid
|
|
344
|
-
<div class="d-flex gap-2 mb-2"><input id="mg-slug" class="form-control" placeholder="slug" value=${e.slug} readonly=${!e.isNew} /><span class="align-self-center small text-muted">${e.type === "post" ? "posts/" : "pages/"}</span></div>
|
|
345
|
-
<
|
|
370
|
+
const e = editing(); // inputs uncontrolled (read on Save); RTEPro mounts into #mg-editor
|
|
371
|
+
return html`<div class="p-3 mb-2" style="border:1px solid var(--border,#232a36);border-radius:10px">
|
|
372
|
+
<div class="d-flex gap-2 mb-2"><input id="mg-slug" class="form-control" placeholder="slug" value=${e.slug} readonly=${!e.isNew} style="max-width:200px" /><input id="mg-title" class="form-control" placeholder="Title" value=${e.title || ""} /><span class="align-self-center small text-muted">${e.type === "post" ? "posts/" : "pages/"}</span></div>
|
|
373
|
+
<div id="mg-editor"></div>
|
|
346
374
|
<div class="mt-2 d-flex gap-2"><button class="btn btn-primary btn-sm" onclick=${saveItem}>Save</button><button class="btn btn-outline-secondary btn-sm" onclick=${() => editing(null)}>Cancel</button></div>
|
|
347
375
|
</div>`;
|
|
348
376
|
};
|