*,*:before,*:after{box-sizing:border-box;margin:0;padding:0}:root{--green-dark: #1c3a24;--green-mid: #2a5436;--green-border: #3a5e42;--green-muted: #7a9e82;--gold: #c9a84c;--gold-light: #f5ecd0;--parchment: #edeade;--parchment-lt: #f5f2e8;--parchment-dk: #e4e0d4;--ink: #2a3a2e;--ink-mid: #5a5448;--ink-light: #8a8478;--ink-muted: #b0a898;--border: #d8d4c8;--border-lt: #e8e4d8;--font-serif: "Georgia", serif;--font-mono: "Courier New", monospace;--t1777-bg: #f5ecd0;--t1777-text: #8a6010;--t1777-border: #e0cc90;--t2026-bg: #dceaf5;--t2026-text: #2a5a7a;--t2026-border: #a0c4e0}body{background:var(--parchment);font-family:var(--font-serif);color:var(--ink)}.signin-screen{min-height:100vh;display:flex;flex-direction:column;align-items:center;justify-content:center;background:var(--parchment);gap:1.25rem}.signin-wordmark{font-family:var(--font-serif);font-style:italic;font-size:52px;color:var(--green-dark);text-align:center}.signin-tagline{font-family:var(--font-mono);font-size:11px;letter-spacing:.15em;text-transform:uppercase;color:var(--ink-light);text-align:center}.signin-divider{width:40px;height:1px;background:var(--border);margin:.5rem auto}.signin-btn{font-family:var(--font-mono);font-size:11px;letter-spacing:.1em;text-transform:uppercase;background:var(--green-dark);color:#f0ece0;border:none;padding:13px 30px;border-radius:5px;cursor:pointer;margin-top:.5rem}.signin-btn:hover{background:var(--green-mid)}.signin-btn:disabled{opacity:.6;cursor:not-allowed}.signin-error{font-family:var(--font-mono);font-size:11px;color:#a03020;text-align:center}.app-wrapper{min-height:100vh;background:var(--parchment)}.global-header{background:var(--green-dark);padding:10px 1.5rem;display:flex;align-items:center;justify-content:space-between;border-bottom:1px solid rgba(0,0,0,.2)}.wordmark{font-family:var(--font-serif);font-style:italic;font-size:20px;color:var(--gold);line-height:1.1}.subtitle{font-family:var(--font-mono);font-size:9px;letter-spacing:.15em;text-transform:uppercase;color:var(--green-muted);margin-top:2px}.back-btn{font-family:var(--font-mono);font-size:10px;letter-spacing:.06em;color:#c8d8c4;border:1px solid var(--green-border);padding:5px 12px;border-radius:4px;background:transparent;cursor:pointer}.back-btn:hover{background:#ffffff0f}.shelf-hero{background:var(--green-dark);padding:2rem 1.5rem 1.75rem}.shelf-hero .tagline{font-family:var(--font-mono);font-size:10px;letter-spacing:.15em;text-transform:uppercase;color:var(--green-muted);display:block;margin-bottom:.5rem}.shelf-hero h1{font-family:var(--font-serif);font-style:italic;font-size:36px;font-weight:400;color:#f0ece0;margin-bottom:.3rem}.shelf-hero p{font-family:var(--font-serif);font-style:italic;font-size:15px;color:var(--green-muted)}.shelf-container{padding:1.5rem}.shelf-grid{display:grid;grid-template-columns:repeat(auto-fill,minmax(280px,1fr));gap:14px;margin-top:1rem}.project-card{background:var(--parchment-lt);border:1px solid var(--border);border-radius:8px;padding:1.1rem 1.25rem;cursor:pointer;position:relative;transition:border-color .15s}.project-card:hover{border-color:#b0a898}.project-card h2{font-family:var(--font-serif);font-style:italic;font-size:20px;font-weight:400;color:var(--ink);margin-bottom:.3rem}.project-meta{font-family:var(--font-mono);font-size:10px;color:var(--ink-muted);letter-spacing:.04em}.project-card-delete{background:none;border:none;color:var(--ink-muted);cursor:pointer;font-size:13px;padding:2px 5px}.project-card-delete:hover{color:var(--ink)}.project-header{background:var(--green-dark);padding:12px 1.5rem 0}.project-name-row{display:flex;align-items:baseline;gap:12px;margin-bottom:10px}.project-name-row h1{font-family:var(--font-serif);font-style:italic;font-size:26px;font-weight:400;color:#f0ece0}.project-type-label{font-family:var(--font-mono);font-size:10px;letter-spacing:.12em;text-transform:uppercase;color:var(--green-muted)}.tab-nav{display:flex;gap:0}.tab-btn{padding:8px 18px 10px;font-family:var(--font-serif);font-style:italic;font-size:15px;color:var(--green-muted);cursor:pointer;border:none;background:transparent;border-bottom:3px solid transparent;position:relative;top:1px;white-space:nowrap}.tab-btn.active{color:var(--gold);border-bottom-color:var(--gold);background:var(--parchment);border-radius:6px 6px 0 0}.tab-btn:hover:not(.active){color:#c8d8c4}.tab-content{padding:1.5rem}.tab-header{display:flex;align-items:flex-start;justify-content:space-between;margin-bottom:1.25rem}.tab-header-left h2{font-family:var(--font-serif);font-style:italic;font-size:26px;font-weight:400;color:var(--ink)}.tab-header-left p{font-family:var(--font-serif);font-style:italic;font-size:14px;color:var(--ink-light);margin-top:3px}.btn-primary{font-family:var(--font-mono);font-size:10px;letter-spacing:.08em;text-transform:uppercase;background:var(--green-dark);color:#f0ece0;border:none;padding:8px 16px;border-radius:4px;cursor:pointer;white-space:nowrap}.btn-primary:hover{background:var(--green-mid)}.btn-secondary{font-family:var(--font-mono);font-size:10px;letter-spacing:.08em;text-transform:uppercase;background:transparent;color:var(--ink-light);border:1px solid var(--border);padding:8px 16px;border-radius:4px;cursor:pointer}.btn-secondary:hover{border-color:#b0a898;color:var(--ink)}.btn-ghost{font-family:var(--font-mono);font-size:10px;letter-spacing:.06em;text-transform:uppercase;background:transparent;color:var(--ink-light);border:1px solid var(--border);padding:6px 12px;border-radius:4px;cursor:pointer;white-space:nowrap}.btn-ghost:hover{border-color:#b0a898;color:var(--ink)}.stats-row{display:grid;grid-template-columns:repeat(4,1fr);gap:10px;margin-bottom:1.25rem}.stat-card{background:var(--parchment-lt);border:1px solid var(--border);border-radius:6px;padding:12px 14px}.stat-label{font-family:var(--font-mono);font-size:9px;letter-spacing:.12em;text-transform:uppercase;color:var(--ink-muted);margin-bottom:5px;display:flex;align-items:center;gap:5px}.stat-value{font-family:var(--font-serif);font-size:22px;color:var(--ink)}.stat-sub{font-family:var(--font-mono);font-size:9px;color:var(--ink-muted);margin-top:2px}.table-toolbar{display:flex;align-items:center;justify-content:space-between;margin-bottom:1rem}.section-label{font-family:var(--font-mono);font-size:10px;letter-spacing:.12em;text-transform:uppercase;color:var(--ink-light)}.chapter-table{background:var(--parchment-lt);border:1px solid var(--border);border-radius:8px;overflow:hidden}.table-header-row{display:grid;grid-template-columns:28px 1fr 84px 72px 96px 74px 28px;padding:8px 14px;border-bottom:1px solid var(--border);background:var(--parchment-dk)}.th{font-family:var(--font-mono);font-size:9px;letter-spacing:.12em;text-transform:uppercase;color:var(--ink-muted)}.chapter-row{display:grid;grid-template-columns:28px 1fr 84px 72px 96px 74px 28px;padding:10px 14px;border-bottom:1px solid var(--border-lt);align-items:center;cursor:pointer}.chapter-row:last-child{border-bottom:none}.chapter-row:hover{background:var(--parchment-dk)}.ch-num{font-family:var(--font-mono);font-size:11px;color:var(--ink-muted)}.ch-title{font-family:var(--font-serif);font-style:italic;font-size:14px;color:var(--ink)}.pov-text{font-family:var(--font-mono);font-size:11px;color:var(--ink-light)}.wc-text{font-family:var(--font-mono);font-size:11px;color:var(--ink-muted);text-align:right}.thread-dot{width:6px;height:6px;border-radius:50%;display:inline-block;flex-shrink:0}.dot-1777{background:#b8860b}.dot-2026{background:#4a7fa0}.thread-pill{display:inline-flex;align-items:center;gap:4px;font-family:var(--font-mono);font-size:9px;letter-spacing:.06em;text-transform:uppercase;padding:3px 7px;border-radius:3px}.thread-1777{background:var(--t1777-bg);color:var(--t1777-text);border:1px solid var(--t1777-border)}.thread-2026{background:var(--t2026-bg);color:var(--t2026-text);border:1px solid var(--t2026-border)}.badge{display:inline-block;font-family:var(--font-mono);font-size:9px;letter-spacing:.06em;text-transform:uppercase;padding:3px 7px;border-radius:3px}.badge-locked{background:#d8eede;color:#2a6a3a;border:1px solid #90c8a0}.badge-drafting{background:var(--t2026-bg);color:var(--t2026-text);border:1px solid var(--t2026-border)}.badge-needs{background:var(--t1777-bg);color:var(--t1777-text);border:1px solid var(--t1777-border)}.badge-notstarted{background:var(--parchment-dk);color:var(--ink-muted);border:1px solid var(--border)}.cards-grid{display:grid;grid-template-columns:repeat(auto-fill,minmax(280px,1fr));gap:14px}.card{background:var(--parchment-lt);border:1px solid var(--border);border-radius:8px;padding:1rem 1.1rem;position:relative}.card-top{display:flex;align-items:flex-start;justify-content:space-between;margin-bottom:8px}.card-label{font-family:var(--font-mono);font-size:9px;letter-spacing:.12em;text-transform:uppercase;color:var(--ink-muted);margin-bottom:4px}.card-title{font-family:var(--font-serif);font-style:italic;font-size:15px;color:var(--ink)}.card-meta{display:flex;align-items:center;gap:6px;flex-wrap:wrap;margin-bottom:8px}.card-body{font-family:var(--font-serif);font-style:italic;font-size:13px;color:var(--ink-mid);line-height:1.55}.card-note,.card-example{font-family:var(--font-mono);font-size:10px;color:var(--ink-light);margin-top:8px;padding-top:8px;border-top:1px solid var(--border-lt);line-height:1.5}.card-actions{display:flex;gap:3px;flex-shrink:0;margin-left:8px}.icon-btn{background:none;border:none;color:var(--ink-muted);cursor:pointer;font-size:13px;padding:2px 5px;line-height:1}.icon-btn:hover{color:var(--ink)}.icon-btn.danger:hover{color:#a03020}.chapter-number{font-family:var(--font-mono);font-size:18px;color:var(--ink-muted);min-width:28px}.word-count{font-family:var(--font-mono);font-size:10px;color:var(--ink-muted)}.chapter-link{font-family:var(--font-mono);font-size:10px;color:var(--green-dark);text-decoration:none;letter-spacing:.04em}.cat-pills{display:flex;gap:8px;flex-wrap:wrap;margin-bottom:1.25rem}.cat-pill{font-family:var(--font-mono);font-size:10px;letter-spacing:.1em;text-transform:uppercase;padding:6px 14px;border-radius:99px;border:1px solid var(--border);background:transparent;color:var(--ink-light);cursor:pointer;display:flex;align-items:center;gap:6px}.cat-pill:hover{border-color:#b0a898;color:var(--ink)}.cat-pill.active{background:var(--green-dark);color:#f0ece0;border-color:var(--green-dark)}.tuning-list{display:flex;flex-direction:column;gap:10px}.tuning-card{background:var(--parchment-lt);border:1px solid var(--border);border-left:3px solid var(--gold);border-radius:0 8px 8px 0;padding:1rem 1.1rem;position:relative}.tuning-num{font-family:var(--font-mono);font-size:9px;letter-spacing:.1em;text-transform:uppercase;color:var(--gold);margin-bottom:5px}.tuning-body{font-family:var(--font-serif);font-style:italic;font-size:14px;color:var(--ink);line-height:1.6}.empty-state{text-align:center;padding:3rem 1rem;color:var(--ink-light)}.empty-icon{font-size:32px;margin-bottom:1rem}.empty-state p{font-family:var(--font-serif);font-style:italic;font-size:15px}.modal-overlay{position:fixed;top:0;right:0;bottom:0;left:0;background:#1c3a248c;display:flex;align-items:center;justify-content:center;z-index:100}.modal{background:var(--parchment-lt);border:1px solid var(--border);border-radius:10px;padding:1.5rem;width:540px;max-width:94vw;max-height:90vh;overflow-y:auto}.modal h3{font-family:var(--font-serif);font-style:italic;font-size:20px;font-weight:400;color:var(--ink);margin-bottom:1.25rem;padding-bottom:.75rem;border-bottom:1px solid var(--border)}.form-group{margin-bottom:1rem}.form-group label{display:block;font-family:var(--font-mono);font-size:10px;letter-spacing:.1em;text-transform:uppercase;color:var(--ink-light);margin-bottom:5px}.form-group input,.form-group textarea,.form-group select{width:100%;font-family:var(--font-serif);font-size:14px;color:var(--ink);background:#fff;border:1px solid var(--border);border-radius:4px;padding:8px 10px;outline:none}.form-group input:focus,.form-group textarea:focus,.form-group select:focus{border-color:var(--green-dark)}.form-group textarea{resize:vertical;line-height:1.55}.modal-actions{display:flex;justify-content:flex-end;gap:8px;margin-top:1.25rem;padding-top:1rem;border-top:1px solid var(--border)}.placeholder-tab{display:flex;flex-direction:column;align-items:center;justify-content:center;padding:5rem 1rem;text-align:center}.placeholder-tab h2{font-family:var(--font-serif);font-style:italic;font-size:30px;font-weight:400;color:var(--ink);margin-bottom:.75rem}.placeholder-tab p{font-family:var(--font-serif);font-style:italic;font-size:15px;color:var(--ink-light);max-width:420px;line-height:1.65}.chapter-row[draggable=true]{cursor:grab}.chapter-row[draggable=true]:active{cursor:grabbing}// api/docs-reorder.js // Rewrites a Google Doc with chapters in a new order // Preserves Heading 1,Heading 2,Normal text styles and paragraph spacing export default async function handler(req,res){res.setHeader("Access-Control-Allow-Origin","https://thehearthside.app");res.setHeader("Access-Control-Allow-Methods","POST, OPTIONS");res.setHeader("Access-Control-Allow-Headers","Content-Type");if (req.method === "OPTIONS") return res.status(200).end();if (req.method !== "POST") return res.status(405).json({error: "Method not allowed"});const{accessToken,docId,chapters}= req.body;if (!accessToken || !docId || !chapters){return res.status(400).json({error: "Missing accessToken, docId, or chapters"})}try{const readResponse = await fetch(`https://docs.googleapis.com/v1/documents/${docId}`,{headers: {Authorization: `Bearer ${accessToken}`}});if (!readResponse.ok){const err = await readResponse.json();return res.status(readResponse.status).json({error: err.error?.message || "Failed to read document"})}const doc = await readResponse.json();// Extract chapter blocks preserving paragraph styles const chapterBlocks = extractChapterBlocks(doc);if (Object.keys(chapterBlocks).length === 0){return res.status(400).json({error: "No chapters found. Make sure chapters use Heading 1 style."})}// Build ordered list of paragraphs with their styles // Each entry:{text:string,style: "HEADING_1" | "HEADING_2" | "NORMAL_TEXT"}const orderedParagraphs = [];for (let i = 0; i < chapters.length; i++){const chapter = chapters[i];let blockKey = Object.keys(chapterBlocks).find(k => k.toLowerCase() === chapter.title.toLowerCase());if (!blockKey){blockKey = Object.keys(chapterBlocks).find(k => {const bare = k.replace(/^chapter \d+s*[-—–-]+s*/i,"").trim().toLowerCase(); return bare === chapter.title.toLowerCase() || k.toLowerCase().includes(chapter.title.toLowerCase()) || chapter.title.toLowerCase().includes(bare);})}if (!blockKey) continue;const block = chapterBlocks[blockKey];// Renumber the heading — strip any existing "Chapter N —" prefix const bareTitle = block.headingText.replace(/^chapter \d+s*[-—–-]+s*/i,"").trim();const newHeading = `Chapter ${chapter.number}u2014 ${bareTitle}`;orderedParagraphs.push({text: newHeading,style: "HEADING_1"});for (const para of block.paragraphs){orderedParagraphs.push(para)}// Add spacing paragraph between chapters orderedParagraphs.push({text: "",style: "NORMAL_TEXT"})}if (orderedParagraphs.length === 0){return res.status(400).json({error: "Could not match any chapters to document content."})}// Build full text and track style ranges let fullText = "";const styleRanges = [];//{startIndex,endIndex,style}for (const para of orderedParagraphs){const startIndex = fullText.length + 1;// 1-based const text = para.text + "n";fullText += text;const endIndex = fullText.length;// exclusive if (para.style !== "NORMAL_TEXT"){styleRanges.push({startIndex,endIndex,style: para.style})}}// Get current doc length const lastContent = doc.body.content[doc.body.content.length - 1];const docLength = lastContent?.endIndex || 1;// Build requests const requests = [];// Delete existing content if (docLength > 2){requests.push({deleteContentRange: {range: {startIndex: 1,endIndex: docLength - 1}}})}// Insert all content requests.push({insertText: {location: {index: 1},text: fullText,}});// Apply Normal text to everything first requests.push({updateParagraphStyle: {range: {startIndex: 1,endIndex: fullText.length + 1},paragraphStyle: {namedStyleType: "NORMAL_TEXT"},fields: "namedStyleType",}});// Apply HEADING_1 styles for (const range of styleRanges.filter(r => r.style === "HEADING_1")){requests.push({updateParagraphStyle: {range: {startIndex: range.startIndex,endIndex: range.endIndex},paragraphStyle: {namedStyleType: "HEADING_1"},fields: "namedStyleType",}})}// Apply HEADING_2 style and CENTER alignment to scene breaks for (const range of styleRanges.filter(r => r.style === "HEADING_2")){requests.push({updateParagraphStyle: {range: {startIndex: range.startIndex,endIndex: range.endIndex},paragraphStyle: {namedStyleType: "HEADING_2",alignment: "CENTER"},fields: "namedStyleType,alignment",}})}const updateResponse = await fetch(`https://docs.googleapis.com/v1/documents/${docId}:batchUpdate`,{method: "POST",headers: {Authorization: `Bearer ${accessToken}`,"Content-Type": "application/json",},body: JSON.stringify({requests}),});if (!updateResponse.ok){const err = await updateResponse.json();return res.status(updateResponse.status).json({error: err.error?.message || "Failed to update document"})}return res.status(200).json({success: true,chaptersWritten: chapters.length})}catch (err){console.error("Docs reorder error:",err);return res.status(500).json({error: "Failed to reorder: " + err.message})}}function extractChapterBlocks(doc){const blocks ={}let currentKey = null;let currentHeadingText = null;let currentParagraphs = [];for (const element of doc.body.content || []){if (!element.paragraph) continue;const style = element.paragraph.paragraphStyle?.namedStyleType || "NORMAL_TEXT";const text = element.paragraph.elements ?.map(e => e.textRun?.content || "") .join("") .replace(/n$/,"") || "";if (style === "HEADING_1"){if (currentKey){blocks[currentKey] ={headingText:currentHeadingText,paragraphs: currentParagraphs,}}currentKey = text.trim();currentHeadingText = text.trim();currentParagraphs = []}else if (currentKey){currentParagraphs.push({text,style})}}if (currentKey){blocks[currentKey] ={headingText:currentHeadingText,paragraphs: currentParagraphs,}}return blocks}
