Compare commits
	
		
			No commits in common. "8686744c7eaa4978799f680d98c7938501f499b9" and "a035d19a5ad1f2c6792cb0b254eb8082d4559160" have entirely different histories.
		
	
	
		
			8686744c7e
			...
			a035d19a5a
		
	
		
| @ -2,8 +2,6 @@ import { useEffect, useState } from 'react'; | ||||
| import AuthGate from './components/AuthGate'; | ||||
| import EditorShell from './components/EditorShell'; | ||||
| import PostsList from './components/PostsList'; | ||||
| import AdminTopBar from './components/layout/AdminTopBar'; | ||||
| import { Box } from '@mui/material'; | ||||
| import './App.css'; | ||||
| 
 | ||||
| function App() { | ||||
| @ -17,44 +15,15 @@ function App() { | ||||
| 
 | ||||
|   const handleLogout = () => { | ||||
|     localStorage.removeItem('voxblog_authed'); | ||||
|     localStorage.removeItem('voxblog_post_id'); | ||||
|     localStorage.removeItem('voxblog_draft_id'); | ||||
|     setAuthenticated(false); | ||||
|     setSelectedPostId(null); | ||||
|   }; | ||||
| 
 | ||||
|   const createNewPost = async () => { | ||||
|     try { | ||||
|       const res = await fetch('/api/posts', { | ||||
|         method: 'POST', | ||||
|         headers: { 'Content-Type': 'application/json' }, | ||||
|         body: JSON.stringify({ title: 'Untitled', contentHtml: '<p></p>', status: 'editing' }), | ||||
|       }); | ||||
|       const data = await res.json(); | ||||
|       if (res.ok && data.id) { | ||||
|         setSelectedPostId(data.id); | ||||
|         localStorage.setItem('voxblog_post_id', data.id); | ||||
|       } else { | ||||
|         alert('Failed to create post'); | ||||
|       } | ||||
|     } catch (e: any) { | ||||
|       alert('Failed to create post: ' + (e?.message || 'unknown error')); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   return ( | ||||
|     <div className="app"> | ||||
|       {authenticated ? ( | ||||
|         <Box sx={{ display: 'grid', gridTemplateRows: 'auto 1fr', minHeight: '100dvh' }}> | ||||
|           <AdminTopBar | ||||
|             userName={undefined} | ||||
|             onGoPosts={() => setSelectedPostId(null)} | ||||
|             onNewPost={createNewPost} | ||||
|             onSettings={undefined} | ||||
|             onLogout={handleLogout} | ||||
|           /> | ||||
|           <Box sx={{ minHeight: 0, overflow: 'hidden' }}> | ||||
|             {selectedPostId ? ( | ||||
|         selectedPostId ? ( | ||||
|           <EditorShell | ||||
|             onLogout={handleLogout} | ||||
|             initialPostId={selectedPostId} | ||||
| @ -64,13 +33,28 @@ function App() { | ||||
|           <PostsList | ||||
|             onSelect={(id) => { | ||||
|               setSelectedPostId(id); | ||||
|                   localStorage.setItem('voxblog_post_id', id); | ||||
|               localStorage.setItem('voxblog_draft_id', id); | ||||
|             }} | ||||
|             onNew={async () => { | ||||
|               try { | ||||
|                 const res = await fetch('/api/posts', { | ||||
|                   method: 'POST', | ||||
|                   headers: { 'Content-Type': 'application/json' }, | ||||
|                   body: JSON.stringify({ title: 'Untitled', contentHtml: '<p></p>', status: 'editing' }), | ||||
|                 }); | ||||
|                 const data = await res.json(); | ||||
|                 if (res.ok && data.id) { | ||||
|                   setSelectedPostId(data.id); | ||||
|                   localStorage.setItem('voxblog_draft_id', data.id); | ||||
|                 } else { | ||||
|                   alert('Failed to create post'); | ||||
|                 } | ||||
|               } catch (e: any) { | ||||
|                 alert('Failed to create post: ' + (e?.message || 'unknown error')); | ||||
|               } | ||||
|             }} | ||||
|                 onNew={createNewPost} | ||||
|           /> | ||||
|             )} | ||||
|           </Box> | ||||
|         </Box> | ||||
|         ) | ||||
|       ) : ( | ||||
|         <AuthGate onAuth={() => { | ||||
|           localStorage.setItem('voxblog_authed', '1'); | ||||
|  | ||||
| @ -1,4 +1,5 @@ | ||||
| import { Box, Snackbar, Alert, Stepper, Step, StepLabel, StepButton, Stack, Button } from '@mui/material'; | ||||
| import { Box, Button, Stack, Typography, TextField, MenuItem, Snackbar, Alert, Stepper, Step, StepLabel, StepButton } from '@mui/material'; | ||||
| import AdminLayout from '../layout/AdminLayout'; | ||||
| import type { RichEditorHandle } from './RichEditor'; | ||||
| import StepAssets from './steps/StepAssets'; | ||||
| import StepAiPrompt from './steps/StepAiPrompt'; | ||||
| @ -9,11 +10,8 @@ import StepPublish from './steps/StepPublish'; | ||||
| import StepContainer from './steps/StepContainer'; | ||||
| import { usePostEditor } from '../hooks/usePostEditor'; | ||||
| import { useRef } from 'react'; | ||||
| import PageContainer from './layout/PageContainer'; | ||||
| import PostSidebar from './layout/PostSidebar'; | ||||
| import StepNavigation from './layout/StepNavigation'; | ||||
| 
 | ||||
| export default function EditorShell({ onLogout: _onLogout, initialPostId, onBack }: { onLogout?: () => void; initialPostId?: string | null; onBack?: () => void }) { | ||||
| export default function EditorShell({ onLogout, initialPostId, onBack }: { onLogout?: () => void; initialPostId?: string | null; onBack?: () => void }) { | ||||
|   const editorRef = useRef<RichEditorHandle | null>(null); | ||||
|   const { | ||||
|     // state
 | ||||
| @ -49,7 +47,7 @@ export default function EditorShell({ onLogout: _onLogout, initialPostId, onBack | ||||
|   // No inline post switching here; selection happens on Posts page
 | ||||
| 
 | ||||
|   return ( | ||||
|     <Box sx={{ minHeight: '100dvh', display: 'flex', flexDirection: 'column', p: { xs: 2, md: 3 }, overflow: 'hidden' }}> | ||||
|     <AdminLayout title="VoxBlog Admin" onLogout={onLogout}> | ||||
|       <Snackbar | ||||
|         open={!!toast?.open} | ||||
|         autoHideDuration={2500} | ||||
| @ -60,15 +58,29 @@ export default function EditorShell({ onLogout: _onLogout, initialPostId, onBack | ||||
|           {toast?.message || ''} | ||||
|         </Alert> | ||||
|       </Snackbar> | ||||
|       <PageContainer | ||||
|         left={ | ||||
|           <PostSidebar | ||||
|             meta={meta} | ||||
|             postStatus={postStatus} | ||||
|             onChangeMeta={setMeta} | ||||
|             onChangeStatus={setPostStatus} | ||||
|             onSave={() => { void savePost(); }} | ||||
|             onDelete={async () => { | ||||
|       <Box sx={{ display: { md: 'grid' }, gridTemplateColumns: { md: '300px 1fr' }, gap: 2 }}> | ||||
|         {/* Left sticky sidebar: Post controls */} | ||||
|         <Box sx={{ position: 'sticky', top: 12, alignSelf: 'start', border: '1px solid', borderColor: 'divider', borderRadius: 1, p: 2 }}> | ||||
|           <Typography variant="subtitle1" sx={{ mb: 1 }}>Post</Typography> | ||||
|           <Stack spacing={1}> | ||||
|             <TextField | ||||
|               size="small" | ||||
|               label="Title" | ||||
|               value={meta.title} | ||||
|               onChange={(e) => setMeta((m) => ({ ...m, title: e.target.value }))} | ||||
|               fullWidth | ||||
|             /> | ||||
|             <TextField size="small" select label="Status" value={postStatus} onChange={(e) => setPostStatus(e.target.value as any)} fullWidth> | ||||
|               <MenuItem value="inbox">inbox</MenuItem> | ||||
|               <MenuItem value="editing">editing</MenuItem> | ||||
|               <MenuItem value="ready_for_publish">ready_for_publish</MenuItem> | ||||
|               <MenuItem value="published">published</MenuItem> | ||||
|               <MenuItem value="archived">archived</MenuItem> | ||||
|             </TextField> | ||||
|             <Stack direction="row" spacing={1}> | ||||
|               <Button variant="contained" onClick={() => { void savePost(); }} fullWidth>Save</Button> | ||||
|               {postId && ( | ||||
|                 <Button color="error" variant="outlined" onClick={async () => { | ||||
|                   if (!postId) return; | ||||
|                   if (!confirm('Delete this post? This will remove its audio clips too.')) return; | ||||
|                   try { | ||||
| @ -77,14 +89,15 @@ export default function EditorShell({ onLogout: _onLogout, initialPostId, onBack | ||||
|                   } catch (e: any) { | ||||
|                     alert('Delete failed: ' + (e?.message || 'unknown error')); | ||||
|                   } | ||||
|             }} | ||||
|             onBack={onBack} | ||||
|             showDelete={!!postId} | ||||
|           /> | ||||
|         } | ||||
|       > | ||||
|                 }}>Delete</Button> | ||||
|               )} | ||||
|             </Stack> | ||||
|             {onBack && <Button variant="text" onClick={onBack}>Back to Posts</Button>} | ||||
|           </Stack> | ||||
|         </Box> | ||||
| 
 | ||||
|         {/* Right content: Stepper and step panels */} | ||||
|         <Box sx={{ display: 'grid', gridTemplateRows: 'auto 1fr auto', height: '100%', minHeight: 0, overflow: 'hidden' }}> | ||||
|         <Box sx={{ display: 'grid', gridTemplateRows: 'auto 1fr auto', height: '85vh', minHeight: 500, overflow: 'hidden' }}> | ||||
|           <Stepper nonLinear activeStep={activeStep} sx={{ mb: 2 }}> | ||||
|             {[ 'Assets', 'AI Prompt', 'Generate', 'Edit', 'Metadata', 'Publish' ].map((label, idx) => ( | ||||
|               <Step key={label} completed={false}> | ||||
| @ -163,14 +176,17 @@ export default function EditorShell({ onLogout: _onLogout, initialPostId, onBack | ||||
|           )} | ||||
|           </Box> | ||||
| 
 | ||||
|           {/* Step navigation (not page navigation) */} | ||||
|           <StepNavigation | ||||
|             disableBack={activeStep === 0} | ||||
|             onBack={() => setActiveStep((s) => Math.max(0, s - 1))} | ||||
|             onNext={() => setActiveStep((s) => Math.min(5, s + 1))} | ||||
|           /> | ||||
|           {/* Bottom nav fixed to panel bottom via grid */} | ||||
|           <Box sx={{ bgcolor: 'background.paper', py: 1, borderTop: '1px solid', borderColor: 'divider' }}> | ||||
|             <Stack direction="row" spacing={1} justifyContent="space-between"> | ||||
|               <Button disabled={activeStep === 0} onClick={() => setActiveStep((s) => Math.max(0, s - 1))}>Back</Button> | ||||
|               <Stack direction="row" spacing={1}> | ||||
|                 <Button variant="outlined" onClick={() => setActiveStep((s) => Math.min(5, s + 1))}>Next</Button> | ||||
|               </Stack> | ||||
|             </Stack> | ||||
|           </Box> | ||||
|       </PageContainer> | ||||
|         </Box> | ||||
|       </Box> | ||||
|     </AdminLayout> | ||||
|   ); | ||||
| } | ||||
|  | ||||
| @ -1,35 +0,0 @@ | ||||
| import { AppBar, Box, Button, Toolbar, Typography } from '@mui/material'; | ||||
| 
 | ||||
| export default function AdminTopBar({ | ||||
|   userName, | ||||
|   onGoPosts, | ||||
|   onNewPost, | ||||
|   onSettings, | ||||
|   onLogout, | ||||
| }: { | ||||
|   userName?: string; | ||||
|   onGoPosts?: () => void; | ||||
|   onNewPost?: () => void; | ||||
|   onSettings?: () => void; | ||||
|   onLogout?: () => void; | ||||
| }) { | ||||
|   return ( | ||||
|     <AppBar position="sticky" elevation={0} sx={{ | ||||
|       backgroundColor: 'rgba(255,255,255,0.8)', | ||||
|       backdropFilter: 'blur(8px)', | ||||
|       borderBottom: '1px solid rgba(0,0,0,0.06)' | ||||
|     }}> | ||||
|       <Toolbar> | ||||
|         <Typography variant="h6" sx={{ flexGrow: 1 }}> | ||||
|           {userName ? `Hi, ${userName}` : 'VoxBlog Admin'} | ||||
|         </Typography> | ||||
|         <Box sx={{ display: 'flex', gap: 1 }}> | ||||
|           {onGoPosts && <Button color="inherit" onClick={onGoPosts}>Posts</Button>} | ||||
|           {onNewPost && <Button color="inherit" onClick={onNewPost}>New Post</Button>} | ||||
|           {onSettings && <Button color="inherit" onClick={onSettings}>Settings</Button>} | ||||
|           {onLogout && <Button color="inherit" onClick={onLogout}>Logout</Button>} | ||||
|         </Box> | ||||
|       </Toolbar> | ||||
|     </AppBar> | ||||
|   ); | ||||
| } | ||||
| @ -1,17 +0,0 @@ | ||||
| import { Box } from '@mui/material'; | ||||
| import type { ReactNode } from 'react'; | ||||
| 
 | ||||
| export default function PageContainer({ left, children, leftWidth = 300 }: { left: ReactNode; children: ReactNode; leftWidth?: number }) { | ||||
|   return ( | ||||
|     <Box sx={{ minHeight: '100dvh', display: 'flex', flexDirection: 'column', p: { xs: 2, md: 3 }, overflow: 'hidden' }}> | ||||
|       <Box sx={{ display: { md: 'grid' }, gridTemplateColumns: { md: `${leftWidth}px 1fr` }, gap: 2, height: '100%', minHeight: 0, overflow: 'hidden' }}> | ||||
|         <Box sx={{ position: 'sticky', top: 12, alignSelf: 'start', border: '1px solid', borderColor: 'divider', borderRadius: 1, p: 2, maxHeight: '100%', overflowY: 'auto' }}> | ||||
|           {left} | ||||
|         </Box> | ||||
|         <Box sx={{ height: '100%', minHeight: 0, overflow: 'hidden' }}> | ||||
|           {children} | ||||
|         </Box> | ||||
|       </Box> | ||||
|     </Box> | ||||
|   ); | ||||
| } | ||||
| @ -1,51 +0,0 @@ | ||||
| import { Box, Button, MenuItem, Stack, TextField, Typography } from '@mui/material'; | ||||
| import type { Metadata } from '../MetadataPanel'; | ||||
| 
 | ||||
| export default function PostSidebar({ | ||||
|   meta, | ||||
|   postStatus, | ||||
|   onChangeMeta, | ||||
|   onChangeStatus, | ||||
|   onSave, | ||||
|   onDelete, | ||||
|   onBack, | ||||
|   showDelete, | ||||
| }: { | ||||
|   meta: Metadata; | ||||
|   postStatus: 'inbox' | 'editing' | 'ready_for_publish' | 'published' | 'archived'; | ||||
|   onChangeMeta: (m: Metadata) => void; | ||||
|   onChangeStatus: (s: 'inbox' | 'editing' | 'ready_for_publish' | 'published' | 'archived') => void; | ||||
|   onSave: () => void | Promise<void>; | ||||
|   onDelete: () => void | Promise<void>; | ||||
|   onBack?: () => void; | ||||
|   showDelete?: boolean; | ||||
| }) { | ||||
|   return ( | ||||
|     <Box> | ||||
|       <Typography variant="subtitle1" sx={{ mb: 1 }}>Post</Typography> | ||||
|       <Stack spacing={1}> | ||||
|         <TextField | ||||
|           size="small" | ||||
|           label="Title" | ||||
|           value={meta.title} | ||||
|           onChange={(e) => onChangeMeta({ ...meta, title: e.target.value })} | ||||
|           fullWidth | ||||
|         /> | ||||
|         <TextField size="small" select label="Status" value={postStatus} onChange={(e) => onChangeStatus(e.target.value as any)} fullWidth> | ||||
|           <MenuItem value="inbox">inbox</MenuItem> | ||||
|           <MenuItem value="editing">editing</MenuItem> | ||||
|           <MenuItem value="ready_for_publish">ready_for_publish</MenuItem> | ||||
|           <MenuItem value="published">published</MenuItem> | ||||
|           <MenuItem value="archived">archived</MenuItem> | ||||
|         </TextField> | ||||
|         <Stack direction="row" spacing={1}> | ||||
|           <Button variant="contained" onClick={onSave} fullWidth>Save</Button> | ||||
|           {showDelete && ( | ||||
|             <Button color="error" variant="outlined" onClick={onDelete}>Delete</Button> | ||||
|           )} | ||||
|         </Stack> | ||||
|         {onBack && <Button variant="text" onClick={onBack}>Back to Posts</Button>} | ||||
|       </Stack> | ||||
|     </Box> | ||||
|   ); | ||||
| } | ||||
| @ -1,25 +0,0 @@ | ||||
| import { Box, Button, Stack } from '@mui/material'; | ||||
| 
 | ||||
| export default function StepNavigation({ | ||||
|   disableBack, | ||||
|   onBack, | ||||
|   onNext, | ||||
|   right, | ||||
| }: { | ||||
|   disableBack?: boolean; | ||||
|   onBack: () => void; | ||||
|   onNext: () => void; | ||||
|   right?: React.ReactNode; | ||||
| }) { | ||||
|   return ( | ||||
|     <Box sx={{ bgcolor: 'background.paper', py: 1, borderTop: '1px solid', borderColor: 'divider' }}> | ||||
|       <Stack direction="row" spacing={1} justifyContent="space-between"> | ||||
|         <Button disabled={!!disableBack} onClick={onBack}>Back</Button> | ||||
|         <Stack direction="row" spacing={1} alignItems="center"> | ||||
|           {right} | ||||
|           <Button variant="outlined" onClick={onNext}>Next</Button> | ||||
|         </Stack> | ||||
|       </Stack> | ||||
|     </Box> | ||||
|   ); | ||||
| } | ||||
| @ -18,7 +18,7 @@ export default function AdminLayout({ title, onLogout, children }: { title?: str | ||||
|           )} | ||||
|         </Toolbar> | ||||
|       </AppBar> | ||||
|       <Container maxWidth="lg" sx={{ py: 3, flex: 1, display: 'flex', flexDirection: 'column', minHeight: 0, overflow: 'hidden' }}> | ||||
|       <Container maxWidth="lg" sx={{ py: 3, flex: 1 }}> | ||||
|         {children} | ||||
|       </Container> | ||||
|     </Box> | ||||
|  | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user