2026-01-31 13:04:57 +01:00
|
|
|
|
import React, { useCallback, useEffect, useLayoutEffect, useMemo, useRef, useState } from "react";
|
2026-01-31 00:37:25 +01:00
|
|
|
|
import { open, save } from "@tauri-apps/plugin-dialog";
|
|
|
|
|
|
import { readTextFile, writeTextFile } from "@tauri-apps/plugin-fs";
|
|
|
|
|
|
import { getCurrentWindow } from "@tauri-apps/api/window";
|
|
|
|
|
|
import { appDataDir } from "@tauri-apps/api/path";
|
2026-01-31 00:41:12 +01:00
|
|
|
|
import { listen } from "@tauri-apps/api/event";
|
|
|
|
|
|
import { invoke } from "@tauri-apps/api/core";
|
2026-01-30 23:14:06 +01:00
|
|
|
|
import historyIcon from "./assets/history.png";
|
2026-01-31 12:45:26 +01:00
|
|
|
|
import historyIconBright from "./assets/history_b.png";
|
2026-01-30 23:14:06 +01:00
|
|
|
|
import {
|
|
|
|
|
|
createText,
|
|
|
|
|
|
deleteText,
|
|
|
|
|
|
deleteTextVersion,
|
|
|
|
|
|
discardDraft,
|
|
|
|
|
|
getDraft,
|
|
|
|
|
|
getLatestManualVersion,
|
|
|
|
|
|
getText,
|
|
|
|
|
|
listTexts,
|
|
|
|
|
|
listVersions,
|
|
|
|
|
|
searchTexts,
|
|
|
|
|
|
saveManualVersion,
|
|
|
|
|
|
updateTextTitle,
|
|
|
|
|
|
upsertDraft,
|
|
|
|
|
|
type Text,
|
|
|
|
|
|
} from "./lib/db";
|
|
|
|
|
|
|
|
|
|
|
|
const formatDate = (timestamp: number) => {
|
|
|
|
|
|
if (!timestamp) return "";
|
|
|
|
|
|
return new Date(timestamp).toLocaleString();
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
type HistorySnapshot = {
|
|
|
|
|
|
body: string;
|
|
|
|
|
|
lastPersistedBody: string;
|
|
|
|
|
|
lastPersistedTitle: string;
|
|
|
|
|
|
hasDraft: boolean;
|
|
|
|
|
|
restoredDraft: boolean;
|
|
|
|
|
|
draftBaseVersionId: string | null;
|
|
|
|
|
|
latestManualVersionId: string | null;
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
type ConfirmState = {
|
|
|
|
|
|
title: string;
|
|
|
|
|
|
message: string;
|
|
|
|
|
|
actionLabel?: string;
|
|
|
|
|
|
onConfirm: () => Promise<void> | void;
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
type HistoryEntry = {
|
|
|
|
|
|
id: string;
|
|
|
|
|
|
created_at: number;
|
|
|
|
|
|
kind: "manual" | "draft";
|
|
|
|
|
|
body: string;
|
|
|
|
|
|
baseVersionId?: string | null;
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const DEFAULT_TITLE = "Untitled Text";
|
|
|
|
|
|
|
|
|
|
|
|
export default function App() {
|
|
|
|
|
|
const [texts, setTexts] = useState<Text[]>([]);
|
|
|
|
|
|
const [selectedTextId, setSelectedTextId] = useState<string | null>(null);
|
|
|
|
|
|
const [search, setSearch] = useState("");
|
|
|
|
|
|
const [loadingTexts, setLoadingTexts] = useState(true);
|
|
|
|
|
|
|
|
|
|
|
|
const [title, setTitle] = useState("");
|
|
|
|
|
|
const [lastPersistedTitle, setLastPersistedTitle] = useState("");
|
|
|
|
|
|
const [body, setBody] = useState("");
|
|
|
|
|
|
const [lastPersistedBody, setLastPersistedBody] = useState("");
|
|
|
|
|
|
const [hasDraft, setHasDraft] = useState(false);
|
|
|
|
|
|
const [restoredDraft, setRestoredDraft] = useState(false);
|
|
|
|
|
|
const [latestManualVersionId, setLatestManualVersionId] = useState<string | null>(null);
|
|
|
|
|
|
const [draftBaseVersionId, setDraftBaseVersionId] = useState<string | null>(null);
|
|
|
|
|
|
|
|
|
|
|
|
const [historyOpen, setHistoryOpen] = useState(false);
|
|
|
|
|
|
const [historyItems, setHistoryItems] = useState<HistoryEntry[]>([]);
|
|
|
|
|
|
const [viewingVersion, setViewingVersion] = useState<HistoryEntry | null>(null);
|
|
|
|
|
|
const [selectedHistoryId, setSelectedHistoryId] = useState<string | null>(null);
|
|
|
|
|
|
const [confirmState, setConfirmState] = useState<ConfirmState | null>(null);
|
2026-01-31 12:39:25 +01:00
|
|
|
|
const [settingsOpen, setSettingsOpen] = useState(false);
|
2026-01-31 12:47:55 +01:00
|
|
|
|
const [theme, setTheme] = useState<"default" | "light">(() => {
|
|
|
|
|
|
const storedTheme = localStorage.getItem("textdb.theme");
|
|
|
|
|
|
return storedTheme === "light" ? "light" : "default";
|
|
|
|
|
|
});
|
|
|
|
|
|
const [textSize, setTextSize] = useState(() => {
|
|
|
|
|
|
const storedSize = Number(localStorage.getItem("textdb.textSize"));
|
|
|
|
|
|
if (!Number.isNaN(storedSize) && storedSize >= 12 && storedSize <= 18) {
|
|
|
|
|
|
return storedSize;
|
|
|
|
|
|
}
|
|
|
|
|
|
return 16;
|
|
|
|
|
|
});
|
2026-01-31 12:58:29 +01:00
|
|
|
|
const [showLineNumbers, setShowLineNumbers] = useState(() => {
|
|
|
|
|
|
return localStorage.getItem("textdb.lineNumbers") === "true";
|
|
|
|
|
|
});
|
2026-01-31 13:04:57 +01:00
|
|
|
|
const [lineHeights, setLineHeights] = useState<number[]>([]);
|
|
|
|
|
|
const [measureTick, setMeasureTick] = useState(0);
|
2026-01-31 12:51:04 +01:00
|
|
|
|
const [sidebarCollapsed, setSidebarCollapsed] = useState(() => {
|
|
|
|
|
|
return localStorage.getItem("textdb.sidebarCollapsed") === "true";
|
|
|
|
|
|
});
|
2026-01-30 23:14:06 +01:00
|
|
|
|
|
|
|
|
|
|
const bodyRef = useRef(body);
|
2026-01-31 12:58:29 +01:00
|
|
|
|
const textareaRef = useRef<HTMLTextAreaElement | null>(null);
|
|
|
|
|
|
const lineNumbersRef = useRef<HTMLDivElement | null>(null);
|
2026-01-31 13:04:57 +01:00
|
|
|
|
const measureRef = useRef<HTMLDivElement | null>(null);
|
2026-01-30 23:14:06 +01:00
|
|
|
|
const historySnapshotRef = useRef<HistorySnapshot | null>(null);
|
2026-01-31 01:04:56 +01:00
|
|
|
|
const recentOpenRef = useRef(new Map<string, number>());
|
2026-01-30 23:14:06 +01:00
|
|
|
|
|
2026-01-31 00:37:32 +01:00
|
|
|
|
|
2026-01-30 23:14:06 +01:00
|
|
|
|
useEffect(() => {
|
|
|
|
|
|
bodyRef.current = body;
|
|
|
|
|
|
}, [body]);
|
|
|
|
|
|
|
2026-01-31 12:45:26 +01:00
|
|
|
|
|
2026-01-31 12:39:25 +01:00
|
|
|
|
useEffect(() => {
|
|
|
|
|
|
document.body.dataset.theme = theme;
|
2026-01-31 12:45:26 +01:00
|
|
|
|
localStorage.setItem("textdb.theme", theme);
|
2026-01-31 12:39:25 +01:00
|
|
|
|
}, [theme]);
|
|
|
|
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
|
document.documentElement.style.setProperty("--base-font-size", `${textSize}px`);
|
2026-01-31 12:45:26 +01:00
|
|
|
|
localStorage.setItem("textdb.textSize", String(textSize));
|
2026-01-31 12:39:25 +01:00
|
|
|
|
}, [textSize]);
|
|
|
|
|
|
|
2026-01-31 12:51:04 +01:00
|
|
|
|
useEffect(() => {
|
|
|
|
|
|
localStorage.setItem("textdb.sidebarCollapsed", String(sidebarCollapsed));
|
|
|
|
|
|
}, [sidebarCollapsed]);
|
|
|
|
|
|
|
2026-01-31 13:00:56 +01:00
|
|
|
|
useEffect(() => {
|
|
|
|
|
|
localStorage.setItem("textdb.lineNumbers", String(showLineNumbers));
|
|
|
|
|
|
}, [showLineNumbers]);
|
|
|
|
|
|
|
2026-01-31 12:51:04 +01:00
|
|
|
|
useEffect(() => {
|
|
|
|
|
|
if (selectedTextId) {
|
|
|
|
|
|
localStorage.setItem("textdb.selectedTextId", selectedTextId);
|
|
|
|
|
|
}
|
|
|
|
|
|
}, [selectedTextId]);
|
|
|
|
|
|
|
2026-01-30 23:14:06 +01:00
|
|
|
|
const isViewingHistory = viewingVersion !== null;
|
|
|
|
|
|
const isDirty = !isViewingHistory && body !== lastPersistedBody;
|
|
|
|
|
|
const hasText = body.trim().length > 0;
|
|
|
|
|
|
|
|
|
|
|
|
const statusKey = useMemo(() => {
|
|
|
|
|
|
if (isViewingHistory) return "history";
|
|
|
|
|
|
if (isDirty) return "unsaved";
|
|
|
|
|
|
if (hasDraft) return "draft";
|
|
|
|
|
|
return "saved";
|
|
|
|
|
|
}, [hasDraft, isDirty, isViewingHistory]);
|
|
|
|
|
|
|
|
|
|
|
|
const canSave = !isViewingHistory && (isDirty || hasDraft);
|
|
|
|
|
|
|
|
|
|
|
|
const statusLabel = useMemo(() => {
|
|
|
|
|
|
switch (statusKey) {
|
|
|
|
|
|
case "history":
|
|
|
|
|
|
return "Viewing history";
|
|
|
|
|
|
case "unsaved":
|
|
|
|
|
|
return "Unsaved";
|
|
|
|
|
|
case "draft":
|
|
|
|
|
|
return "Draft autosaved";
|
|
|
|
|
|
default:
|
|
|
|
|
|
return "Saved";
|
|
|
|
|
|
}
|
|
|
|
|
|
}, [statusKey]);
|
|
|
|
|
|
|
2026-01-31 12:45:26 +01:00
|
|
|
|
const historyIconSrc = theme === "light" ? historyIconBright : historyIcon;
|
|
|
|
|
|
|
2026-01-31 13:05:32 +01:00
|
|
|
|
const lines = useMemo(() => body.split("
|
|
|
|
|
|
"), [body]);
|
|
|
|
|
|
const lineNumbers = useMemo(() => lines.map((_, index) => index + 1), [lines]);
|
2026-01-31 13:00:02 +01:00
|
|
|
|
|
|
|
|
|
|
const handleTextareaScroll = useCallback((event: React.UIEvent<HTMLTextAreaElement>) => {
|
|
|
|
|
|
if (!showLineNumbers) return;
|
|
|
|
|
|
if (lineNumbersRef.current) {
|
|
|
|
|
|
lineNumbersRef.current.scrollTop = event.currentTarget.scrollTop;
|
|
|
|
|
|
}
|
|
|
|
|
|
}, [showLineNumbers]);
|
|
|
|
|
|
|
2026-01-31 13:04:57 +01:00
|
|
|
|
useEffect(() => {
|
|
|
|
|
|
if (!showLineNumbers) return;
|
|
|
|
|
|
const textarea = textareaRef.current;
|
|
|
|
|
|
if (!textarea || typeof ResizeObserver === "undefined") return;
|
|
|
|
|
|
const observer = new ResizeObserver(() => {
|
|
|
|
|
|
setMeasureTick((tick) => tick + 1);
|
|
|
|
|
|
});
|
|
|
|
|
|
observer.observe(textarea);
|
|
|
|
|
|
return () => observer.disconnect();
|
|
|
|
|
|
}, [showLineNumbers]);
|
|
|
|
|
|
|
|
|
|
|
|
useLayoutEffect(() => {
|
|
|
|
|
|
if (!showLineNumbers) return;
|
|
|
|
|
|
const textarea = textareaRef.current;
|
|
|
|
|
|
const measure = measureRef.current;
|
|
|
|
|
|
if (!textarea || !measure) return;
|
|
|
|
|
|
measure.style.width = `${textarea.clientWidth}px`;
|
|
|
|
|
|
const heights = Array.from(measure.children).map((child) =>
|
|
|
|
|
|
Math.ceil((child as HTMLElement).getBoundingClientRect().height)
|
|
|
|
|
|
);
|
|
|
|
|
|
setLineHeights(heights);
|
|
|
|
|
|
if (lineNumbersRef.current) {
|
|
|
|
|
|
lineNumbersRef.current.scrollTop = textarea.scrollTop;
|
|
|
|
|
|
}
|
|
|
|
|
|
}, [lines, showLineNumbers, textSize, measureTick]);
|
|
|
|
|
|
|
2026-01-31 13:00:02 +01:00
|
|
|
|
useEffect(() => {
|
|
|
|
|
|
if (showLineNumbers && textareaRef.current && lineNumbersRef.current) {
|
|
|
|
|
|
lineNumbersRef.current.scrollTop = textareaRef.current.scrollTop;
|
|
|
|
|
|
}
|
|
|
|
|
|
}, [showLineNumbers, body]);
|
|
|
|
|
|
|
2026-01-31 12:45:26 +01:00
|
|
|
|
|
2026-01-30 23:14:06 +01:00
|
|
|
|
const refreshTexts = useCallback(async () => {
|
|
|
|
|
|
setLoadingTexts(true);
|
|
|
|
|
|
try {
|
|
|
|
|
|
const trimmed = search.trim();
|
|
|
|
|
|
const rows = trimmed ? await searchTexts(trimmed) : await listTexts();
|
|
|
|
|
|
setTexts(rows);
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
setLoadingTexts(false);
|
|
|
|
|
|
}
|
|
|
|
|
|
}, [search]);
|
|
|
|
|
|
|
|
|
|
|
|
const refreshVersions = useCallback(async () => {
|
|
|
|
|
|
if (!selectedTextId || !historyOpen) return;
|
|
|
|
|
|
const [manualRows, draft] = await Promise.all([
|
|
|
|
|
|
listVersions(selectedTextId),
|
|
|
|
|
|
getDraft(selectedTextId)
|
|
|
|
|
|
]);
|
|
|
|
|
|
const manualItems: HistoryEntry[] = manualRows.map((row) => ({
|
|
|
|
|
|
id: row.id,
|
|
|
|
|
|
created_at: row.created_at,
|
|
|
|
|
|
kind: "manual",
|
|
|
|
|
|
body: row.body
|
|
|
|
|
|
}));
|
|
|
|
|
|
const draftItem: HistoryEntry[] = draft
|
|
|
|
|
|
? [
|
|
|
|
|
|
{
|
|
|
|
|
|
id: `draft:${selectedTextId}`,
|
|
|
|
|
|
created_at: draft.updated_at,
|
|
|
|
|
|
kind: "draft",
|
|
|
|
|
|
body: draft.body,
|
|
|
|
|
|
baseVersionId: draft.base_version_id ?? null
|
|
|
|
|
|
}
|
|
|
|
|
|
]
|
|
|
|
|
|
: [];
|
|
|
|
|
|
const combined = [...draftItem, ...manualItems].sort(
|
|
|
|
|
|
(a, b) => b.created_at - a.created_at
|
|
|
|
|
|
);
|
|
|
|
|
|
setHistoryItems(combined);
|
|
|
|
|
|
}, [historyOpen, selectedTextId]);
|
|
|
|
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
|
refreshTexts().catch((error) => {
|
|
|
|
|
|
console.error("Failed to load texts", error);
|
|
|
|
|
|
});
|
|
|
|
|
|
}, [refreshTexts]);
|
|
|
|
|
|
|
|
|
|
|
|
useEffect(() => {
|
2026-01-31 12:51:04 +01:00
|
|
|
|
if (selectedTextId || texts.length === 0) return;
|
|
|
|
|
|
const storedId = localStorage.getItem("textdb.selectedTextId");
|
|
|
|
|
|
const fallback = texts[0].id;
|
|
|
|
|
|
const resolved = storedId && texts.some((text) => text.id === storedId)
|
|
|
|
|
|
? storedId
|
|
|
|
|
|
: fallback;
|
|
|
|
|
|
setSelectedTextId(resolved);
|
2026-01-30 23:14:06 +01:00
|
|
|
|
}, [selectedTextId, texts]);
|
|
|
|
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
|
if (!historyOpen) {
|
|
|
|
|
|
setHistoryItems([]);
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
refreshVersions().catch((error) => {
|
|
|
|
|
|
console.error("Failed to load versions", error);
|
|
|
|
|
|
});
|
|
|
|
|
|
}, [historyOpen, refreshVersions]);
|
|
|
|
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
|
if (!selectedTextId) {
|
|
|
|
|
|
setTitle("");
|
|
|
|
|
|
setLastPersistedTitle("");
|
|
|
|
|
|
setBody("");
|
|
|
|
|
|
setLastPersistedBody("");
|
|
|
|
|
|
setHasDraft(false);
|
|
|
|
|
|
setRestoredDraft(false);
|
|
|
|
|
|
setLatestManualVersionId(null);
|
|
|
|
|
|
setDraftBaseVersionId(null);
|
|
|
|
|
|
setViewingVersion(null);
|
|
|
|
|
|
setSelectedHistoryId(null);
|
|
|
|
|
|
historySnapshotRef.current = null;
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
let cancelled = false;
|
|
|
|
|
|
const loadText = async () => {
|
|
|
|
|
|
const [text, manualVersion, draft] = await Promise.all([
|
|
|
|
|
|
getText(selectedTextId),
|
|
|
|
|
|
getLatestManualVersion(selectedTextId),
|
|
|
|
|
|
getDraft(selectedTextId)
|
|
|
|
|
|
]);
|
|
|
|
|
|
|
|
|
|
|
|
if (cancelled) return;
|
|
|
|
|
|
|
|
|
|
|
|
if (!text) {
|
|
|
|
|
|
setSelectedTextId(null);
|
2026-01-31 12:51:04 +01:00
|
|
|
|
localStorage.removeItem("textdb.selectedTextId");
|
2026-01-30 23:14:06 +01:00
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const resolvedTitle = text.title || DEFAULT_TITLE;
|
|
|
|
|
|
const resolvedBody = draft?.body ?? manualVersion?.body ?? "";
|
|
|
|
|
|
const baseVersionId = draft?.base_version_id ?? manualVersion?.id ?? null;
|
|
|
|
|
|
|
|
|
|
|
|
setTitle(resolvedTitle);
|
|
|
|
|
|
setLastPersistedTitle(resolvedTitle);
|
|
|
|
|
|
setBody(resolvedBody);
|
|
|
|
|
|
setLastPersistedBody(resolvedBody);
|
|
|
|
|
|
setHasDraft(Boolean(draft));
|
|
|
|
|
|
setRestoredDraft(Boolean(draft));
|
|
|
|
|
|
setLatestManualVersionId(manualVersion?.id ?? null);
|
|
|
|
|
|
setDraftBaseVersionId(baseVersionId);
|
|
|
|
|
|
setViewingVersion(null);
|
|
|
|
|
|
setSelectedHistoryId(
|
|
|
|
|
|
draft ? `draft:${selectedTextId}` : manualVersion?.id ?? null
|
|
|
|
|
|
);
|
|
|
|
|
|
historySnapshotRef.current = null;
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
loadText().catch((error) => {
|
|
|
|
|
|
console.error("Failed to load text", error);
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
return () => {
|
|
|
|
|
|
cancelled = true;
|
|
|
|
|
|
};
|
|
|
|
|
|
}, [selectedTextId]);
|
|
|
|
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
|
if (!selectedTextId || !isDirty || isViewingHistory) return;
|
|
|
|
|
|
|
|
|
|
|
|
const currentBody = body;
|
|
|
|
|
|
const handle = window.setTimeout(() => {
|
|
|
|
|
|
upsertDraft(selectedTextId, currentBody, draftBaseVersionId)
|
|
|
|
|
|
.then(() => {
|
|
|
|
|
|
if (bodyRef.current !== currentBody) return;
|
|
|
|
|
|
setHasDraft(true);
|
|
|
|
|
|
setLastPersistedBody(currentBody);
|
|
|
|
|
|
setSelectedHistoryId(`draft:${selectedTextId}`);
|
|
|
|
|
|
if (historyOpen) {
|
|
|
|
|
|
refreshVersions().catch((error) => {
|
|
|
|
|
|
console.error("Failed to refresh history", error);
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
})
|
|
|
|
|
|
.catch((error) => {
|
|
|
|
|
|
console.error("Failed to autosave draft", error);
|
|
|
|
|
|
});
|
|
|
|
|
|
}, 600);
|
|
|
|
|
|
|
|
|
|
|
|
return () => window.clearTimeout(handle);
|
|
|
|
|
|
}, [
|
|
|
|
|
|
body,
|
|
|
|
|
|
draftBaseVersionId,
|
|
|
|
|
|
historyOpen,
|
|
|
|
|
|
isDirty,
|
|
|
|
|
|
isViewingHistory,
|
|
|
|
|
|
refreshVersions,
|
|
|
|
|
|
selectedTextId
|
|
|
|
|
|
]);
|
|
|
|
|
|
|
|
|
|
|
|
const handleNewText = useCallback(async () => {
|
|
|
|
|
|
const { textId } = await createText(DEFAULT_TITLE, "");
|
|
|
|
|
|
await refreshTexts();
|
|
|
|
|
|
setSelectedTextId(textId);
|
|
|
|
|
|
}, [refreshTexts]);
|
|
|
|
|
|
|
2026-01-31 00:37:25 +01:00
|
|
|
|
const createTextFromFile = useCallback(
|
|
|
|
|
|
async (filePath: string) => {
|
2026-01-31 00:42:19 +01:00
|
|
|
|
try {
|
|
|
|
|
|
const filename = filePath.split(/[\/]/).pop() || DEFAULT_TITLE;
|
|
|
|
|
|
const title = filename.replace(/\.txt$/i, "") || DEFAULT_TITLE;
|
|
|
|
|
|
const contents = await readTextFile(filePath);
|
|
|
|
|
|
const { textId } = await createText(title, contents);
|
|
|
|
|
|
await refreshTexts();
|
|
|
|
|
|
setSelectedTextId(textId);
|
|
|
|
|
|
} catch (error) {
|
|
|
|
|
|
console.error("Failed to open text file", error);
|
|
|
|
|
|
}
|
2026-01-31 00:37:25 +01:00
|
|
|
|
},
|
|
|
|
|
|
[refreshTexts]
|
|
|
|
|
|
);
|
|
|
|
|
|
|
2026-01-31 00:42:19 +01:00
|
|
|
|
const handleFilePaths = useCallback(
|
|
|
|
|
|
async (paths: string[]) => {
|
2026-01-31 01:04:56 +01:00
|
|
|
|
const now = Date.now();
|
2026-01-31 00:42:19 +01:00
|
|
|
|
const txtPaths = paths.filter((path) => path.toLowerCase().endsWith(".txt"));
|
2026-01-31 01:04:56 +01:00
|
|
|
|
const recent = recentOpenRef.current;
|
2026-01-31 00:42:19 +01:00
|
|
|
|
for (const path of txtPaths) {
|
2026-01-31 01:04:56 +01:00
|
|
|
|
const key = path.toLowerCase();
|
|
|
|
|
|
const last = recent.get(key);
|
|
|
|
|
|
if (last && now - last < 1000) continue;
|
|
|
|
|
|
recent.set(key, now);
|
2026-01-31 00:42:19 +01:00
|
|
|
|
await createTextFromFile(path);
|
|
|
|
|
|
}
|
2026-01-31 01:04:56 +01:00
|
|
|
|
for (const [key, timestamp] of recent.entries()) {
|
|
|
|
|
|
if (now - timestamp > 2000) recent.delete(key);
|
|
|
|
|
|
}
|
2026-01-31 00:42:19 +01:00
|
|
|
|
},
|
|
|
|
|
|
[createTextFromFile]
|
|
|
|
|
|
);
|
|
|
|
|
|
|
2026-01-31 00:44:08 +01:00
|
|
|
|
useEffect(() => {
|
|
|
|
|
|
let unlisten: (() => void) | null = null;
|
|
|
|
|
|
getCurrentWindow()
|
|
|
|
|
|
.onDragDropEvent(async (event) => {
|
|
|
|
|
|
if (event.payload.type !== "drop") return;
|
|
|
|
|
|
await handleFilePaths(event.payload.paths ?? []);
|
|
|
|
|
|
})
|
|
|
|
|
|
.then((cleanup) => {
|
|
|
|
|
|
unlisten = cleanup;
|
|
|
|
|
|
})
|
|
|
|
|
|
.catch((error) => {
|
|
|
|
|
|
console.error("Failed to register drag/drop handler", error);
|
|
|
|
|
|
});
|
|
|
|
|
|
return () => {
|
|
|
|
|
|
if (unlisten) unlisten();
|
|
|
|
|
|
};
|
|
|
|
|
|
}, [handleFilePaths]);
|
|
|
|
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
|
let unlisten: (() => void) | null = null;
|
|
|
|
|
|
listen<string[]>("file-opened", async (event) => {
|
|
|
|
|
|
await handleFilePaths(event.payload ?? []);
|
|
|
|
|
|
})
|
|
|
|
|
|
.then((cleanup) => {
|
|
|
|
|
|
unlisten = cleanup;
|
|
|
|
|
|
})
|
|
|
|
|
|
.catch((error) => {
|
|
|
|
|
|
console.error("Failed to register file-open listener", error);
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
invoke<string[]>("take_pending_opens")
|
|
|
|
|
|
.then((paths) => handleFilePaths(paths ?? []))
|
|
|
|
|
|
.catch((error) => {
|
|
|
|
|
|
console.error("Failed to load pending file opens", error);
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
return () => {
|
|
|
|
|
|
if (unlisten) unlisten();
|
|
|
|
|
|
};
|
|
|
|
|
|
}, [handleFilePaths]);
|
|
|
|
|
|
|
2026-01-31 00:37:25 +01:00
|
|
|
|
const handleOpenText = useCallback(async () => {
|
|
|
|
|
|
const baseDir = await appDataDir();
|
|
|
|
|
|
const path = await open({
|
|
|
|
|
|
multiple: false,
|
|
|
|
|
|
directory: false,
|
|
|
|
|
|
filters: [{ name: "Text", extensions: ["txt"] }],
|
|
|
|
|
|
defaultPath: baseDir
|
|
|
|
|
|
});
|
|
|
|
|
|
if (!path || Array.isArray(path)) return;
|
|
|
|
|
|
await createTextFromFile(path);
|
|
|
|
|
|
}, [createTextFromFile]);
|
|
|
|
|
|
|
2026-01-30 23:14:06 +01:00
|
|
|
|
const handleDeleteText = useCallback(
|
|
|
|
|
|
async (promptId: string) => {
|
|
|
|
|
|
await deleteText(promptId);
|
|
|
|
|
|
await refreshTexts();
|
|
|
|
|
|
if (selectedTextId === promptId) {
|
|
|
|
|
|
setSelectedTextId(null);
|
2026-01-31 12:51:04 +01:00
|
|
|
|
localStorage.removeItem("textdb.selectedTextId");
|
2026-01-30 23:14:06 +01:00
|
|
|
|
}
|
|
|
|
|
|
},
|
|
|
|
|
|
[refreshTexts, selectedTextId]
|
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
const handleSaveVersion = useCallback(async () => {
|
|
|
|
|
|
if (!selectedTextId || !canSave) return;
|
|
|
|
|
|
const normalizedTitle = title.trim() || DEFAULT_TITLE;
|
|
|
|
|
|
if (normalizedTitle !== title) {
|
|
|
|
|
|
setTitle(normalizedTitle);
|
|
|
|
|
|
}
|
|
|
|
|
|
const result = await saveManualVersion(selectedTextId, normalizedTitle, body);
|
|
|
|
|
|
setLastPersistedBody(body);
|
|
|
|
|
|
setLastPersistedTitle(normalizedTitle);
|
|
|
|
|
|
setHasDraft(false);
|
|
|
|
|
|
setRestoredDraft(false);
|
|
|
|
|
|
setLatestManualVersionId(result.versionId);
|
|
|
|
|
|
setDraftBaseVersionId(result.versionId);
|
|
|
|
|
|
setSelectedHistoryId(result.versionId);
|
|
|
|
|
|
await refreshTexts();
|
|
|
|
|
|
await refreshVersions();
|
|
|
|
|
|
}, [body, canSave, refreshTexts, refreshVersions, selectedTextId, title]);
|
|
|
|
|
|
|
|
|
|
|
|
const applyTitleUpdate = useCallback((promptId: string, nextTitle: string) => {
|
|
|
|
|
|
const now = Date.now();
|
|
|
|
|
|
setTexts((prev) => {
|
|
|
|
|
|
let updated: Text | null = null;
|
|
|
|
|
|
const remaining: Text[] = [];
|
|
|
|
|
|
for (const text of prev) {
|
|
|
|
|
|
if (text.id === promptId) {
|
|
|
|
|
|
updated = { ...text, title: nextTitle, updated_at: now };
|
|
|
|
|
|
} else {
|
|
|
|
|
|
remaining.push(text);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
if (!updated) return prev;
|
|
|
|
|
|
return [updated, ...remaining];
|
|
|
|
|
|
});
|
|
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
|
|
|
|
const handleTitleChange = useCallback(
|
|
|
|
|
|
(event: React.ChangeEvent<HTMLInputElement>) => {
|
|
|
|
|
|
const nextTitle = event.target.value;
|
|
|
|
|
|
setTitle(nextTitle);
|
|
|
|
|
|
if (!selectedTextId || isViewingHistory) return;
|
|
|
|
|
|
applyTitleUpdate(selectedTextId, nextTitle);
|
|
|
|
|
|
setLastPersistedTitle(nextTitle);
|
|
|
|
|
|
updateTextTitle(selectedTextId, nextTitle).catch((error) => {
|
|
|
|
|
|
console.error("Failed to update title", error);
|
|
|
|
|
|
});
|
|
|
|
|
|
},
|
|
|
|
|
|
[applyTitleUpdate, isViewingHistory, selectedTextId]
|
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
const handleDiscardDraft = useCallback(async () => {
|
|
|
|
|
|
if (!selectedTextId || isViewingHistory) return;
|
|
|
|
|
|
await discardDraft(selectedTextId);
|
|
|
|
|
|
const manualVersion = await getLatestManualVersion(selectedTextId);
|
|
|
|
|
|
const resolvedBody = manualVersion?.body ?? "";
|
|
|
|
|
|
setBody(resolvedBody);
|
|
|
|
|
|
setLastPersistedBody(resolvedBody);
|
|
|
|
|
|
setHasDraft(false);
|
|
|
|
|
|
setRestoredDraft(false);
|
|
|
|
|
|
setLatestManualVersionId(manualVersion?.id ?? null);
|
|
|
|
|
|
setDraftBaseVersionId(manualVersion?.id ?? null);
|
|
|
|
|
|
setSelectedHistoryId(manualVersion?.id ?? null);
|
|
|
|
|
|
await refreshVersions();
|
|
|
|
|
|
}, [isViewingHistory, refreshVersions, selectedTextId]);
|
|
|
|
|
|
|
|
|
|
|
|
const handleExportText = useCallback(async () => {
|
|
|
|
|
|
if (!hasText) return;
|
|
|
|
|
|
const filename = `${title.trim() || DEFAULT_TITLE}.txt`;
|
|
|
|
|
|
const path = await save({
|
|
|
|
|
|
defaultPath: filename,
|
|
|
|
|
|
filters: [{ name: "Text", extensions: ["txt"] }]
|
|
|
|
|
|
});
|
|
|
|
|
|
if (!path) return;
|
|
|
|
|
|
const finalPath = path.endsWith(".txt") ? path : `${path}.txt`;
|
|
|
|
|
|
await writeTextFile(finalPath, body);
|
|
|
|
|
|
}, [body, hasText, title]);
|
|
|
|
|
|
|
|
|
|
|
|
const handleExitHistory = useCallback(() => {
|
|
|
|
|
|
const snapshot = historySnapshotRef.current;
|
|
|
|
|
|
if (snapshot) {
|
|
|
|
|
|
setBody(snapshot.body);
|
|
|
|
|
|
setLastPersistedBody(snapshot.lastPersistedBody);
|
|
|
|
|
|
setLastPersistedTitle(snapshot.lastPersistedTitle);
|
|
|
|
|
|
setHasDraft(snapshot.hasDraft);
|
|
|
|
|
|
setRestoredDraft(snapshot.restoredDraft);
|
|
|
|
|
|
setDraftBaseVersionId(snapshot.draftBaseVersionId);
|
|
|
|
|
|
setLatestManualVersionId(snapshot.latestManualVersionId);
|
|
|
|
|
|
}
|
|
|
|
|
|
setViewingVersion(null);
|
|
|
|
|
|
setSelectedHistoryId(null);
|
|
|
|
|
|
historySnapshotRef.current = null;
|
|
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
|
|
|
|
const applyVersionAsCurrent = useCallback(
|
|
|
|
|
|
(version: HistoryEntry) => {
|
|
|
|
|
|
const snapshot = historySnapshotRef.current;
|
|
|
|
|
|
if (snapshot) {
|
|
|
|
|
|
setLastPersistedBody(snapshot.lastPersistedBody);
|
|
|
|
|
|
setLastPersistedTitle(snapshot.lastPersistedTitle);
|
|
|
|
|
|
setHasDraft(snapshot.hasDraft);
|
|
|
|
|
|
setRestoredDraft(snapshot.restoredDraft);
|
|
|
|
|
|
setLatestManualVersionId(snapshot.latestManualVersionId);
|
|
|
|
|
|
}
|
|
|
|
|
|
setDraftBaseVersionId(
|
|
|
|
|
|
version.kind === "manual" ? version.id : version.baseVersionId ?? null
|
|
|
|
|
|
);
|
|
|
|
|
|
setBody(version.body);
|
|
|
|
|
|
setLastPersistedBody(version.body);
|
|
|
|
|
|
setViewingVersion(null);
|
|
|
|
|
|
setSelectedHistoryId(version.id);
|
|
|
|
|
|
historySnapshotRef.current = null;
|
|
|
|
|
|
},
|
|
|
|
|
|
[]
|
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
const handleDeleteVersion = useCallback(
|
|
|
|
|
|
async (version: HistoryEntry) => {
|
|
|
|
|
|
if (!selectedTextId) return;
|
|
|
|
|
|
|
|
|
|
|
|
const currentItems = historyItems;
|
|
|
|
|
|
const index = currentItems.findIndex((item) => item.id === version.id);
|
|
|
|
|
|
const olderCandidate =
|
|
|
|
|
|
index >= 0 ? currentItems[index + 1] ?? null : null;
|
|
|
|
|
|
const fallbackCandidate =
|
|
|
|
|
|
index > 0 ? currentItems[index - 1] ?? null : null;
|
|
|
|
|
|
const shouldAutoSelect =
|
|
|
|
|
|
historyOpen && (!viewingVersion || viewingVersion.id === version.id);
|
|
|
|
|
|
|
|
|
|
|
|
if (version.kind === "draft") {
|
|
|
|
|
|
await discardDraft(selectedTextId);
|
|
|
|
|
|
setHasDraft(false);
|
|
|
|
|
|
setRestoredDraft(false);
|
|
|
|
|
|
setDraftBaseVersionId(latestManualVersionId ?? null);
|
|
|
|
|
|
if (historySnapshotRef.current) {
|
|
|
|
|
|
historySnapshotRef.current.hasDraft = false;
|
|
|
|
|
|
historySnapshotRef.current.restoredDraft = false;
|
|
|
|
|
|
historySnapshotRef.current.draftBaseVersionId =
|
|
|
|
|
|
latestManualVersionId ?? null;
|
|
|
|
|
|
}
|
|
|
|
|
|
} else {
|
|
|
|
|
|
await deleteTextVersion(selectedTextId, version.id);
|
|
|
|
|
|
}
|
|
|
|
|
|
await refreshVersions();
|
|
|
|
|
|
|
|
|
|
|
|
if (version.kind === "manual" && latestManualVersionId === version.id) {
|
|
|
|
|
|
const manualVersion = await getLatestManualVersion(selectedTextId);
|
|
|
|
|
|
const resolvedBody = manualVersion?.body ?? "";
|
|
|
|
|
|
const nextManualId = manualVersion?.id ?? null;
|
|
|
|
|
|
setLatestManualVersionId(nextManualId);
|
|
|
|
|
|
setDraftBaseVersionId(nextManualId);
|
|
|
|
|
|
if (historySnapshotRef.current) {
|
|
|
|
|
|
historySnapshotRef.current.latestManualVersionId = nextManualId;
|
|
|
|
|
|
historySnapshotRef.current.draftBaseVersionId = nextManualId;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (!hasDraft && !isViewingHistory) {
|
|
|
|
|
|
setBody(resolvedBody);
|
|
|
|
|
|
setLastPersistedBody(resolvedBody);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (shouldAutoSelect) {
|
|
|
|
|
|
if (olderCandidate) {
|
|
|
|
|
|
applyVersionAsCurrent(olderCandidate);
|
|
|
|
|
|
} else if (fallbackCandidate) {
|
|
|
|
|
|
applyVersionAsCurrent(fallbackCandidate);
|
|
|
|
|
|
} else {
|
|
|
|
|
|
handleExitHistory();
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
|
|
|
|
|
[
|
|
|
|
|
|
applyVersionAsCurrent,
|
|
|
|
|
|
handleExitHistory,
|
|
|
|
|
|
hasDraft,
|
|
|
|
|
|
isViewingHistory,
|
|
|
|
|
|
historyOpen,
|
|
|
|
|
|
latestManualVersionId,
|
|
|
|
|
|
refreshVersions,
|
|
|
|
|
|
selectedTextId,
|
|
|
|
|
|
historyItems,
|
|
|
|
|
|
viewingVersion
|
|
|
|
|
|
]
|
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
|
|
const handleConfirm = useCallback(async () => {
|
|
|
|
|
|
if (!confirmState) return;
|
|
|
|
|
|
const action = confirmState.onConfirm;
|
|
|
|
|
|
setConfirmState(null);
|
|
|
|
|
|
try {
|
|
|
|
|
|
await action();
|
|
|
|
|
|
} catch (error) {
|
|
|
|
|
|
console.error("Delete failed", error);
|
|
|
|
|
|
}
|
|
|
|
|
|
}, [confirmState]);
|
|
|
|
|
|
|
|
|
|
|
|
const handleTitleBlur = useCallback(async () => {
|
|
|
|
|
|
if (!selectedTextId || isViewingHistory) return;
|
|
|
|
|
|
const normalizedTitle = title.trim() || DEFAULT_TITLE;
|
|
|
|
|
|
if (normalizedTitle !== title) {
|
|
|
|
|
|
setTitle(normalizedTitle);
|
|
|
|
|
|
applyTitleUpdate(selectedTextId, normalizedTitle);
|
|
|
|
|
|
}
|
|
|
|
|
|
if (normalizedTitle === lastPersistedTitle) return;
|
|
|
|
|
|
await updateTextTitle(selectedTextId, normalizedTitle);
|
|
|
|
|
|
setLastPersistedTitle(normalizedTitle);
|
|
|
|
|
|
await refreshTexts();
|
|
|
|
|
|
}, [
|
|
|
|
|
|
applyTitleUpdate,
|
|
|
|
|
|
isViewingHistory,
|
|
|
|
|
|
lastPersistedTitle,
|
|
|
|
|
|
refreshTexts,
|
|
|
|
|
|
selectedTextId,
|
|
|
|
|
|
title
|
|
|
|
|
|
]);
|
|
|
|
|
|
|
|
|
|
|
|
const handleToggleHistory = useCallback(() => {
|
|
|
|
|
|
if (historyOpen && viewingVersion) {
|
|
|
|
|
|
const snapshot = historySnapshotRef.current;
|
|
|
|
|
|
if (snapshot) {
|
|
|
|
|
|
setBody(snapshot.body);
|
|
|
|
|
|
setLastPersistedBody(snapshot.lastPersistedBody);
|
|
|
|
|
|
setLastPersistedTitle(snapshot.lastPersistedTitle);
|
|
|
|
|
|
setHasDraft(snapshot.hasDraft);
|
|
|
|
|
|
setRestoredDraft(snapshot.restoredDraft);
|
|
|
|
|
|
setDraftBaseVersionId(snapshot.draftBaseVersionId);
|
|
|
|
|
|
setLatestManualVersionId(snapshot.latestManualVersionId);
|
|
|
|
|
|
}
|
|
|
|
|
|
setViewingVersion(null);
|
|
|
|
|
|
historySnapshotRef.current = null;
|
|
|
|
|
|
}
|
|
|
|
|
|
setHistoryOpen((prev) => !prev);
|
|
|
|
|
|
}, [historyOpen, viewingVersion]);
|
|
|
|
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
|
const handleKeyDown = (event: KeyboardEvent) => {
|
|
|
|
|
|
const isSave =
|
|
|
|
|
|
(event.metaKey || event.ctrlKey) && event.key.toLowerCase() === "s";
|
|
|
|
|
|
if (!isSave) return;
|
|
|
|
|
|
event.preventDefault();
|
|
|
|
|
|
handleSaveVersion().catch((error) => {
|
|
|
|
|
|
console.error("Failed to save version", error);
|
|
|
|
|
|
});
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
window.addEventListener("keydown", handleKeyDown);
|
|
|
|
|
|
return () => window.removeEventListener("keydown", handleKeyDown);
|
|
|
|
|
|
}, [handleSaveVersion]);
|
|
|
|
|
|
|
|
|
|
|
|
return (
|
2026-01-31 12:39:25 +01:00
|
|
|
|
<div className={`app app--theme-${theme}${sidebarCollapsed ? " app--sidebar-collapsed" : ""}`}>
|
2026-01-31 12:26:10 +01:00
|
|
|
|
{!sidebarCollapsed ? (
|
2026-01-30 23:14:06 +01:00
|
|
|
|
<aside className="sidebar">
|
|
|
|
|
|
<div className="sidebar__header">
|
2026-01-31 12:26:10 +01:00
|
|
|
|
<div className="sidebar__title-row">
|
|
|
|
|
|
<div className="app-title">TextDB</div>
|
2026-01-31 12:39:25 +01:00
|
|
|
|
<div className="sidebar__actions">
|
|
|
|
|
|
<button
|
|
|
|
|
|
className="icon-button icon-button--ghost"
|
|
|
|
|
|
onClick={() => setSettingsOpen(true)}
|
|
|
|
|
|
aria-label="Open settings"
|
|
|
|
|
|
title="Settings"
|
|
|
|
|
|
type="button"
|
|
|
|
|
|
>
|
2026-01-31 12:45:26 +01:00
|
|
|
|
<span className="icon-button__glyph icon-button__glyph--large" aria-hidden="true">⚙</span>
|
2026-01-31 12:39:25 +01:00
|
|
|
|
</button>
|
|
|
|
|
|
<button
|
|
|
|
|
|
className="icon-button icon-button--ghost"
|
|
|
|
|
|
onClick={() => setSidebarCollapsed(true)}
|
|
|
|
|
|
aria-label="Collapse sidebar"
|
|
|
|
|
|
title="Collapse sidebar"
|
|
|
|
|
|
type="button"
|
|
|
|
|
|
>
|
|
|
|
|
|
<span aria-hidden="true"><</span>
|
|
|
|
|
|
</button>
|
|
|
|
|
|
</div>
|
2026-01-31 12:26:10 +01:00
|
|
|
|
</div>
|
2026-01-30 23:14:06 +01:00
|
|
|
|
<input
|
|
|
|
|
|
className="search"
|
|
|
|
|
|
placeholder="Search texts"
|
|
|
|
|
|
value={search}
|
|
|
|
|
|
onChange={(event) => setSearch(event.target.value)}
|
|
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div className="prompt-list">
|
|
|
|
|
|
<div className="prompt-list__inner">
|
|
|
|
|
|
{loadingTexts ? (
|
|
|
|
|
|
<div className="empty">Loading texts…</div>
|
|
|
|
|
|
) : texts.length === 0 ? (
|
|
|
|
|
|
<div className="empty">No texts yet.</div>
|
|
|
|
|
|
) : (
|
|
|
|
|
|
texts.map((text) => (
|
|
|
|
|
|
<div
|
|
|
|
|
|
key={text.id}
|
|
|
|
|
|
className={`prompt-item${
|
|
|
|
|
|
text.id === selectedTextId ? " is-active" : ""
|
|
|
|
|
|
}`}
|
|
|
|
|
|
onClick={() => setSelectedTextId(text.id)}
|
|
|
|
|
|
role="button"
|
|
|
|
|
|
tabIndex={0}
|
|
|
|
|
|
onKeyDown={(event) => {
|
|
|
|
|
|
if (event.key === "Enter" || event.key === " ") {
|
|
|
|
|
|
event.preventDefault();
|
|
|
|
|
|
setSelectedTextId(text.id);
|
|
|
|
|
|
}
|
|
|
|
|
|
}}
|
|
|
|
|
|
>
|
|
|
|
|
|
<div className="prompt-item__content">
|
|
|
|
|
|
<div className="prompt-item__title">{text.title}</div>
|
|
|
|
|
|
<div className="prompt-item__meta">
|
|
|
|
|
|
Updated {formatDate(text.updated_at)}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<button
|
|
|
|
|
|
className="prompt-item__delete"
|
|
|
|
|
|
onClick={(event) => {
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
setConfirmState({
|
|
|
|
|
|
title: "Delete text",
|
|
|
|
|
|
message: `Delete \"${text.title}\"? This removes all versions and drafts.`,
|
|
|
|
|
|
actionLabel: "Delete text",
|
|
|
|
|
|
onConfirm: () => handleDeleteText(text.id)
|
|
|
|
|
|
});
|
|
|
|
|
|
}}
|
|
|
|
|
|
aria-label="Delete text"
|
|
|
|
|
|
title="Delete text"
|
|
|
|
|
|
>
|
|
|
|
|
|
×
|
|
|
|
|
|
</button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
))
|
|
|
|
|
|
)}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div className="sidebar__footer">
|
|
|
|
|
|
<button className="button button--primary" onClick={handleNewText}>
|
|
|
|
|
|
New Text
|
|
|
|
|
|
</button>
|
2026-01-31 00:37:47 +01:00
|
|
|
|
<button className="button" onClick={handleOpenText}>
|
|
|
|
|
|
Open Text
|
|
|
|
|
|
</button>
|
2026-01-30 23:14:06 +01:00
|
|
|
|
</div>
|
|
|
|
|
|
</aside>
|
2026-01-31 12:26:10 +01:00
|
|
|
|
) : null}
|
2026-01-30 23:14:06 +01:00
|
|
|
|
|
|
|
|
|
|
<main className="workspace">
|
2026-01-31 12:26:10 +01:00
|
|
|
|
{sidebarCollapsed ? (
|
|
|
|
|
|
<div className="workspace__topbar">
|
|
|
|
|
|
<button
|
|
|
|
|
|
className="icon-button icon-button--ghost"
|
|
|
|
|
|
onClick={() => setSidebarCollapsed(false)}
|
|
|
|
|
|
aria-label="Expand sidebar"
|
|
|
|
|
|
title="Expand sidebar"
|
|
|
|
|
|
type="button"
|
|
|
|
|
|
>
|
|
|
|
|
|
<span aria-hidden="true">></span>
|
|
|
|
|
|
</button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
) : null}
|
2026-01-30 23:14:06 +01:00
|
|
|
|
{!selectedTextId ? (
|
|
|
|
|
|
<div className="empty-state">
|
|
|
|
|
|
<div className="empty-state__title">Create your first text</div>
|
|
|
|
|
|
<div className="empty-state__subtitle">
|
|
|
|
|
|
Everything stays offline in a single SQLite database.
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<button className="button button--primary" onClick={handleNewText}>
|
|
|
|
|
|
New Text
|
|
|
|
|
|
</button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
) : (
|
|
|
|
|
|
<div
|
|
|
|
|
|
className={`workspace__content${
|
|
|
|
|
|
historyOpen ? " workspace__content--history" : ""
|
|
|
|
|
|
}`}
|
|
|
|
|
|
>
|
|
|
|
|
|
<section className="editor">
|
|
|
|
|
|
<div className="editor__header">
|
|
|
|
|
|
<input
|
|
|
|
|
|
className="title-input"
|
|
|
|
|
|
value={title}
|
|
|
|
|
|
onChange={handleTitleChange}
|
|
|
|
|
|
onBlur={handleTitleBlur}
|
|
|
|
|
|
placeholder="Text title"
|
|
|
|
|
|
disabled={isViewingHistory}
|
|
|
|
|
|
/>
|
|
|
|
|
|
<div className="editor__status-row">
|
|
|
|
|
|
<div className="status-line">
|
|
|
|
|
|
<span className={`status status--${statusKey}`}></span>
|
|
|
|
|
|
{statusLabel}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<button
|
|
|
|
|
|
className={`icon-button${historyOpen ? " is-active" : ""}`}
|
|
|
|
|
|
onClick={handleToggleHistory}
|
|
|
|
|
|
aria-label={historyOpen ? "Close history" : "Open history"}
|
|
|
|
|
|
title={historyOpen ? "Close history" : "Open history"}
|
|
|
|
|
|
type="button"
|
|
|
|
|
|
>
|
2026-01-31 12:45:26 +01:00
|
|
|
|
<img src={historyIconSrc} alt="" className="icon-button__img" />
|
2026-01-30 23:14:06 +01:00
|
|
|
|
</button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
2026-01-31 12:58:29 +01:00
|
|
|
|
<div className="editor__textarea-wrap">
|
2026-01-31 13:04:57 +01:00
|
|
|
|
{showLineNumbers ? (
|
|
|
|
|
|
<div className="line-measure" ref={measureRef} aria-hidden="true">
|
|
|
|
|
|
{lines.map((line, index) => (
|
|
|
|
|
|
<div key={index} className="line-measure__line">
|
|
|
|
|
|
{line.length > 0 ? line : " "}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
))}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
) : null}
|
2026-01-31 12:58:29 +01:00
|
|
|
|
{showLineNumbers ? (
|
|
|
|
|
|
<div className="line-numbers" ref={lineNumbersRef}>
|
2026-01-31 13:04:57 +01:00
|
|
|
|
{lineNumbers.map((line, index) => (
|
|
|
|
|
|
<div
|
|
|
|
|
|
key={line}
|
|
|
|
|
|
className="line-numbers__line"
|
|
|
|
|
|
style={{ height: lineHeights[index] ? `${lineHeights[index]}px` : undefined }}
|
|
|
|
|
|
>
|
2026-01-31 12:58:29 +01:00
|
|
|
|
{line}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
))}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
) : null}
|
|
|
|
|
|
<textarea
|
|
|
|
|
|
ref={textareaRef}
|
|
|
|
|
|
className="editor__textarea"
|
|
|
|
|
|
value={body}
|
|
|
|
|
|
onChange={(event) => setBody(event.target.value)}
|
|
|
|
|
|
onScroll={handleTextareaScroll}
|
|
|
|
|
|
placeholder="Write your text here…"
|
|
|
|
|
|
readOnly={isViewingHistory}
|
|
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
2026-01-30 23:14:06 +01:00
|
|
|
|
|
|
|
|
|
|
<div className="editor__footer">
|
|
|
|
|
|
{hasText ? (
|
|
|
|
|
|
<button className="button" onClick={handleExportText}>
|
|
|
|
|
|
Export Text
|
|
|
|
|
|
</button>
|
|
|
|
|
|
) : null}
|
|
|
|
|
|
{hasDraft && !isViewingHistory ? (
|
|
|
|
|
|
<button
|
|
|
|
|
|
className="button"
|
|
|
|
|
|
onClick={() =>
|
|
|
|
|
|
setConfirmState({
|
|
|
|
|
|
title: "Discard draft",
|
|
|
|
|
|
message: "Discard this draft? This cannot be undone.",
|
|
|
|
|
|
actionLabel: "Discard draft",
|
|
|
|
|
|
onConfirm: handleDiscardDraft
|
|
|
|
|
|
})
|
|
|
|
|
|
}
|
|
|
|
|
|
>
|
|
|
|
|
|
Discard Draft
|
|
|
|
|
|
</button>
|
|
|
|
|
|
) : null}
|
|
|
|
|
|
<button
|
|
|
|
|
|
className="button button--primary button--save"
|
|
|
|
|
|
onClick={handleSaveVersion}
|
|
|
|
|
|
disabled={!canSave}
|
|
|
|
|
|
>
|
|
|
|
|
|
Save Version (⌘S)
|
|
|
|
|
|
</button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</section>
|
|
|
|
|
|
|
|
|
|
|
|
{historyOpen ? (
|
|
|
|
|
|
<aside className="history">
|
|
|
|
|
|
<div className="history__header">
|
|
|
|
|
|
<span>History</span>
|
|
|
|
|
|
<button
|
|
|
|
|
|
className="history__close"
|
|
|
|
|
|
onClick={handleToggleHistory}
|
|
|
|
|
|
aria-label="Close history"
|
|
|
|
|
|
title="Close history"
|
|
|
|
|
|
type="button"
|
|
|
|
|
|
>
|
|
|
|
|
|
×
|
|
|
|
|
|
</button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div className="history__list">
|
|
|
|
|
|
{historyItems.length === 0 ? (
|
|
|
|
|
|
<div className="empty">No versions yet.</div>
|
|
|
|
|
|
) : (
|
|
|
|
|
|
historyItems.map((version) => (
|
|
|
|
|
|
<div
|
|
|
|
|
|
key={version.id}
|
|
|
|
|
|
className={`history__item${
|
|
|
|
|
|
selectedHistoryId === version.id ? " is-active" : ""
|
|
|
|
|
|
}`}
|
|
|
|
|
|
onClick={() => applyVersionAsCurrent(version)}
|
|
|
|
|
|
role="button"
|
|
|
|
|
|
tabIndex={0}
|
|
|
|
|
|
onKeyDown={(event) => {
|
|
|
|
|
|
if (event.key === "Enter" || event.key === " ") {
|
|
|
|
|
|
event.preventDefault();
|
|
|
|
|
|
applyVersionAsCurrent(version);
|
|
|
|
|
|
}
|
|
|
|
|
|
}}
|
|
|
|
|
|
>
|
|
|
|
|
|
<div className="history__item-content">
|
|
|
|
|
|
<div className="history__item-title">
|
|
|
|
|
|
{formatDate(version.created_at)}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div className="history__item-meta">
|
|
|
|
|
|
{version.kind === "draft" ? "Draft" : "Manual save"}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<button
|
|
|
|
|
|
className="history__item-delete"
|
|
|
|
|
|
onClick={(event) => {
|
|
|
|
|
|
event.stopPropagation();
|
|
|
|
|
|
setConfirmState({
|
|
|
|
|
|
title:
|
|
|
|
|
|
version.kind === "draft"
|
|
|
|
|
|
? "Discard draft"
|
|
|
|
|
|
: "Delete version",
|
|
|
|
|
|
message:
|
|
|
|
|
|
version.kind === "draft"
|
|
|
|
|
|
? "Discard this draft? This cannot be undone."
|
|
|
|
|
|
: "Delete this version? This cannot be undone.",
|
|
|
|
|
|
actionLabel:
|
|
|
|
|
|
version.kind === "draft"
|
|
|
|
|
|
? "Discard draft"
|
|
|
|
|
|
: "Delete version",
|
|
|
|
|
|
onConfirm: () => handleDeleteVersion(version)
|
|
|
|
|
|
});
|
|
|
|
|
|
}}
|
|
|
|
|
|
aria-label={
|
|
|
|
|
|
version.kind === "draft"
|
|
|
|
|
|
? "Discard draft"
|
|
|
|
|
|
: "Delete version"
|
|
|
|
|
|
}
|
|
|
|
|
|
title={
|
|
|
|
|
|
version.kind === "draft"
|
|
|
|
|
|
? "Discard draft"
|
|
|
|
|
|
: "Delete version"
|
|
|
|
|
|
}
|
|
|
|
|
|
>
|
|
|
|
|
|
×
|
|
|
|
|
|
</button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
))
|
|
|
|
|
|
)}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</aside>
|
|
|
|
|
|
) : null}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
)}
|
|
|
|
|
|
</main>
|
|
|
|
|
|
|
2026-01-31 12:39:25 +01:00
|
|
|
|
{settingsOpen ? (
|
|
|
|
|
|
<div className="settings-overlay">
|
|
|
|
|
|
<div
|
|
|
|
|
|
className="settings-overlay__backdrop"
|
|
|
|
|
|
onClick={() => setSettingsOpen(false)}
|
|
|
|
|
|
/>
|
|
|
|
|
|
<div className="settings-panel" role="dialog" aria-modal="true">
|
|
|
|
|
|
<div className="settings-panel__header">
|
|
|
|
|
|
<div className="settings-panel__title">Settings</div>
|
|
|
|
|
|
<button
|
|
|
|
|
|
className="icon-button icon-button--ghost"
|
|
|
|
|
|
onClick={() => setSettingsOpen(false)}
|
|
|
|
|
|
aria-label="Close settings"
|
|
|
|
|
|
title="Close settings"
|
|
|
|
|
|
type="button"
|
|
|
|
|
|
>
|
|
|
|
|
|
<span aria-hidden="true">×</span>
|
|
|
|
|
|
</button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div className="settings-panel__section">
|
|
|
|
|
|
<label className="settings-panel__label" htmlFor="theme-select">
|
|
|
|
|
|
Theme
|
|
|
|
|
|
</label>
|
|
|
|
|
|
<select
|
|
|
|
|
|
id="theme-select"
|
|
|
|
|
|
className="settings-panel__select"
|
|
|
|
|
|
value={theme}
|
|
|
|
|
|
onChange={(event) =>
|
|
|
|
|
|
setTheme(event.target.value as "default" | "light")
|
|
|
|
|
|
}
|
|
|
|
|
|
>
|
|
|
|
|
|
<option value="default">Default</option>
|
|
|
|
|
|
<option value="light">Bright</option>
|
|
|
|
|
|
</select>
|
|
|
|
|
|
</div>
|
2026-01-31 12:58:29 +01:00
|
|
|
|
<div className="settings-panel__section settings-panel__section--row">
|
|
|
|
|
|
<label className="settings-panel__label" htmlFor="line-numbers-toggle">
|
|
|
|
|
|
Line numbers
|
|
|
|
|
|
</label>
|
|
|
|
|
|
<input
|
|
|
|
|
|
id="line-numbers-toggle"
|
|
|
|
|
|
type="checkbox"
|
|
|
|
|
|
checked={showLineNumbers}
|
|
|
|
|
|
onChange={(event) => setShowLineNumbers(event.target.checked)}
|
|
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
2026-01-31 12:39:25 +01:00
|
|
|
|
<div className="settings-panel__section">
|
|
|
|
|
|
<label className="settings-panel__label" htmlFor="text-size">
|
|
|
|
|
|
Text size
|
|
|
|
|
|
</label>
|
|
|
|
|
|
<div className="settings-panel__slider-row">
|
|
|
|
|
|
<input
|
|
|
|
|
|
id="text-size"
|
|
|
|
|
|
className="settings-panel__range"
|
|
|
|
|
|
type="range"
|
|
|
|
|
|
min={12}
|
|
|
|
|
|
max={18}
|
|
|
|
|
|
step={1}
|
|
|
|
|
|
value={textSize}
|
|
|
|
|
|
onChange={(event) => setTextSize(Number(event.target.value))}
|
|
|
|
|
|
/>
|
|
|
|
|
|
<div className="settings-panel__value">{textSize}px</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
) : null}
|
|
|
|
|
|
|
2026-01-30 23:14:06 +01:00
|
|
|
|
{confirmState ? (
|
|
|
|
|
|
<div className="modal">
|
|
|
|
|
|
<div className="modal__overlay" onClick={() => setConfirmState(null)} />
|
|
|
|
|
|
<div className="modal__card" role="dialog" aria-modal="true">
|
|
|
|
|
|
<div className="modal__title">{confirmState.title}</div>
|
|
|
|
|
|
<div className="modal__message">{confirmState.message}</div>
|
|
|
|
|
|
<div className="modal__actions">
|
|
|
|
|
|
<button className="button" onClick={() => setConfirmState(null)}>
|
|
|
|
|
|
Cancel
|
|
|
|
|
|
</button>
|
|
|
|
|
|
<button className="button button--danger" onClick={handleConfirm}>
|
|
|
|
|
|
{confirmState.actionLabel ?? "Delete"}
|
|
|
|
|
|
</button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
) : null}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|