212 lines
7.8 KiB
React
212 lines
7.8 KiB
React
// Dashy — main app (DB-backed via SQLite/WASM)
|
|
|
|
const TWEAK_DEFAULTS = /*EDITMODE-BEGIN*/{
|
|
"theme": "light",
|
|
"accent": "#2A6FDB",
|
|
"density": "cozy",
|
|
"showTags": true
|
|
}/*EDITMODE-END*/;
|
|
|
|
const ACCENTS = ['#2A6FDB', '#1F8A5B', '#D97757', '#7A5AF8'];
|
|
|
|
function useDashyDB() {
|
|
const [ready, setReady] = React.useState(DashyDB.isReady);
|
|
const [, force] = React.useReducer(x => x + 1, 0);
|
|
React.useEffect(() => {
|
|
if (!DashyDB.isReady) DashyDB.init().then(() => setReady(true));
|
|
const off = DashyDB.subscribe(() => force());
|
|
return off;
|
|
}, []);
|
|
return ready;
|
|
}
|
|
|
|
function App() {
|
|
const [t, setTweak] = useTweaks(TWEAK_DEFAULTS);
|
|
const ready = useDashyDB();
|
|
const [authed, setAuthed] = React.useState(false);
|
|
const [meId, setMeId] = React.useState('rod');
|
|
const [tab, setTab] = React.useState('overview');
|
|
const [adding, setAdding] = React.useState(null);
|
|
const [openTaskId, setOpenTaskId] = React.useState(null);
|
|
const [showLogs, setShowLogs] = React.useState(false);
|
|
const [showSettings, setShowSettings] = React.useState(false);
|
|
const [showDB, setShowDB] = React.useState(false);
|
|
const [headsUp, setHeadsUp] = React.useState([
|
|
{ id: 'h1', kind: 'unsuccessful', taskId: 't7',
|
|
title: 'WO #2188 auto-marked Unsuccessful',
|
|
sub: 'Two missed bookings — assigned to Kirra for review' },
|
|
{ id: 'h2', kind: 'billing', taskId: 't4',
|
|
title: 'Form response switched to Billing',
|
|
sub: 'K. Wynne · originally Service Booking' },
|
|
]);
|
|
|
|
if (!ready) return <BootSplash />;
|
|
|
|
const tasks = DashyDB.listTasks();
|
|
const audit = DashyDB.listAudit();
|
|
const dbUsers = DashyDB.listUsers();
|
|
const userMap = Object.fromEntries(dbUsers.map(u => [u.id, u]));
|
|
const merge = (id) => {
|
|
const base = findUser(id); if (!base && !userMap[id]) return null;
|
|
const live = userMap[id] || {};
|
|
if (!base) {
|
|
// user added at runtime
|
|
return { id, name: live.name, role: live.role, hue: live.hue, initials: live.initials,
|
|
photo: live.photo || null, account_type: live.account_type || 'standard' };
|
|
}
|
|
return { ...base, name: live.name || base.name, role: live.role || base.role,
|
|
photo: live.photo || null, account_type: live.account_type || 'standard' };
|
|
};
|
|
const me = merge(meId);
|
|
const isAdmin = me && me.account_type === 'admin';
|
|
const openTask = tasks.find(x => x.id === openTaskId);
|
|
|
|
const handleLogin = (id) => {
|
|
setMeId(id); setAuthed(true);
|
|
DashyDB.addAudit({
|
|
actor: id, action: 'login',
|
|
summary: (merge(id) || {}).name + ' signed in',
|
|
});
|
|
};
|
|
|
|
const addTask = ({ title, description, assignee, priority }) => {
|
|
const id = 't_' + Date.now().toString(36);
|
|
const at = new Date('2026-05-08T10:30:00').toISOString();
|
|
DashyDB.createTask({
|
|
id, title, description, assignee, priority,
|
|
addedBy: meId, source: 'manual', status: 'open', addedAt: at, tags: []
|
|
});
|
|
DashyDB.addAudit({
|
|
at, actor: meId, action: 'task_created',
|
|
summary: 'Created task "' + title + '" for ' + (merge(assignee)||{}).name,
|
|
target: id
|
|
});
|
|
setAdding(null);
|
|
};
|
|
|
|
const moveTask = (taskId, toUserId) => DashyDB.moveTask(taskId, toUserId, meId);
|
|
const setPriority = (taskId, p) => DashyDB.setPriority(taskId, p);
|
|
const dismissHU = (id) => setHeadsUp(h => h.filter(x => x.id !== id));
|
|
const openTaskFromAnywhere = (id) => { setOpenTaskId(id); setShowLogs(false); };
|
|
|
|
if (!authed) return <LoginScreen onLogin={handleLogin} />;
|
|
|
|
return (
|
|
<div className="app">
|
|
<TopBar
|
|
me={me}
|
|
isAdmin={isAdmin}
|
|
tab={tab}
|
|
setTab={setTab}
|
|
onAdd={() => setAdding(meId)}
|
|
onLogs={() => setShowLogs(true)}
|
|
onProfile={() => setShowSettings(true)}
|
|
onDB={() => setShowDB(true)}
|
|
/>
|
|
|
|
<HeadsUp items={headsUp} onDismiss={dismissHU} onOpenTask={openTaskFromAnywhere} />
|
|
|
|
<main className="main">
|
|
{tab === 'overview' && (
|
|
<OverviewScreen
|
|
tasks={tasks} density={t.density}
|
|
onOpen={(task) => setOpenTaskId(task.id)}
|
|
onAddFor={(uid) => setAdding(uid)}
|
|
onMoveTask={moveTask}
|
|
/>
|
|
)}
|
|
{tab !== 'overview' && (
|
|
<UserScreen
|
|
user={merge(tab)} tasks={tasks} density={t.density}
|
|
onOpen={(task) => setOpenTaskId(task.id)}
|
|
onAddFor={(uid) => setAdding(uid)}
|
|
/>
|
|
)}
|
|
</main>
|
|
|
|
<AddTaskModal open={!!adding} onClose={() => setAdding(null)} onSubmit={addTask} defaultAssignee={adding} me={me} />
|
|
{openTask && (
|
|
<TaskDetail task={openTask} onClose={() => setOpenTaskId(null)} onMove={moveTask} onPriority={setPriority} />
|
|
)}
|
|
{showLogs && (
|
|
<Modal title="Audit log" onClose={() => setShowLogs(false)} wide>
|
|
<AuditScreen entries={audit} onOpen={openTaskFromAnywhere} />
|
|
</Modal>
|
|
)}
|
|
{showSettings && (
|
|
<SettingsScreen
|
|
user={me}
|
|
isAdmin={isAdmin}
|
|
onClose={() => setShowSettings(false)}
|
|
onSave={(edits) => {
|
|
DashyDB.updateUser(meId, edits);
|
|
DashyDB.addAudit({ actor: meId, action: 'profile_updated', summary: 'Updated profile details' });
|
|
}}
|
|
onLogout={() => { setShowSettings(false); setAuthed(false); }}
|
|
onSwitchUser={(id) => { setMeId(id); setShowSettings(false); }}
|
|
onCreateUser={(u) => {
|
|
const id = DashyDB.createUser(u);
|
|
DashyDB.addAudit({ actor: meId, action: 'user_created', summary: 'Added ' + u.name + ' (' + (u.account_type||'standard') + ')', target: id });
|
|
}}
|
|
onDeleteUser={(id) => {
|
|
const u = userMap[id];
|
|
DashyDB.deleteUser(id);
|
|
DashyDB.addAudit({ actor: meId, action: 'user_deleted', summary: 'Removed ' + (u?u.name:id), target: null });
|
|
}}
|
|
onUpdateUserRole={(id, edits) => {
|
|
DashyDB.updateUser(id, edits);
|
|
DashyDB.addAudit({ actor: meId, action: 'user_updated', summary: 'Updated ' + (userMap[id]?userMap[id].name:id) + ' permissions', target: null });
|
|
}}
|
|
/>
|
|
)}
|
|
{showDB && isAdmin && <DatabaseInspector onClose={() => setShowDB(false)} />}
|
|
|
|
<DashyTweaks t={t} setTweak={setTweak} />
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function BootSplash() {
|
|
return (
|
|
<div className="boot">
|
|
<div className="boot__pulse" />
|
|
<div className="boot__label mono">opening dashy.db…</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function DashyTweaks({ t, setTweak }) {
|
|
return (
|
|
<TweaksPanel title="Tweaks">
|
|
<TweakSection title="Appearance">
|
|
<TweakRadio label="Theme" value={t.theme}
|
|
options={[{value:'light',label:'Light'},{value:'dark',label:'Dark'}]}
|
|
onChange={v => setTweak('theme', v)} />
|
|
<TweakColor label="Accent" value={t.accent} options={ACCENTS} onChange={v => setTweak('accent', v)} />
|
|
<TweakRadio label="Density" value={t.density}
|
|
options={[{value:'compact',label:'Compact'},{value:'cozy',label:'Cozy'}]}
|
|
onChange={v => setTweak('density', v)} />
|
|
<TweakToggle label="Show tags on cards" value={t.showTags} onChange={v => setTweak('showTags', v)} />
|
|
</TweakSection>
|
|
<TweakSection title="Database">
|
|
<TweakButton label="Reset SQLite" onClick={() => { if (confirm('Wipe and reseed dashy.db?')) DashyDB.reset(); }}>
|
|
Reset
|
|
</TweakButton>
|
|
</TweakSection>
|
|
</TweaksPanel>
|
|
);
|
|
}
|
|
|
|
function ThemeBridge() {
|
|
const [t] = useTweaks(TWEAK_DEFAULTS);
|
|
React.useEffect(() => {
|
|
document.documentElement.dataset.theme = t.theme;
|
|
document.documentElement.style.setProperty('--accent', t.accent);
|
|
document.documentElement.dataset.showTags = t.showTags ? 'on' : 'off';
|
|
}, [t.theme, t.accent, t.showTags]);
|
|
return null;
|
|
}
|
|
|
|
const root = ReactDOM.createRoot(document.getElementById('root'));
|
|
root.render(<><ThemeBridge /><App /></>);
|