All checks were successful
Release / build-and-release (push) Successful in 30m13s
Add per-section subtotals (expenses, income, transfers) to budget table and budget vs actual report. Fix category detail panel visibility when scrolling through long category lists. Closes #11, closes #12 Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
152 lines
5.7 KiB
TypeScript
152 lines
5.7 KiB
TypeScript
import { useState } from "react";
|
|
import { useTranslation } from "react-i18next";
|
|
import { Plus, RotateCcw, List, AlertTriangle } from "lucide-react";
|
|
import { PageHelp } from "../components/shared/PageHelp";
|
|
import { useCategories } from "../hooks/useCategories";
|
|
import CategoryTree from "../components/categories/CategoryTree";
|
|
import CategoryDetailPanel from "../components/categories/CategoryDetailPanel";
|
|
import AllKeywordsPanel from "../components/categories/AllKeywordsPanel";
|
|
|
|
export default function CategoriesPage() {
|
|
const { t } = useTranslation();
|
|
const [showAllKeywords, setShowAllKeywords] = useState(false);
|
|
const [showReinitConfirm, setShowReinitConfirm] = useState(false);
|
|
const {
|
|
state,
|
|
selectCategory,
|
|
startCreating,
|
|
startEditing,
|
|
cancelEditing,
|
|
saveCategory,
|
|
deleteCategory,
|
|
addKeyword,
|
|
editKeyword,
|
|
removeKeyword,
|
|
reinitializeCategories,
|
|
moveCategory,
|
|
} = useCategories();
|
|
|
|
const handleReinitialize = async () => {
|
|
setShowReinitConfirm(false);
|
|
await reinitializeCategories();
|
|
};
|
|
|
|
const selectedCategory =
|
|
state.selectedCategoryId !== null
|
|
? state.categories.find((c) => c.id === state.selectedCategoryId) ?? null
|
|
: null;
|
|
|
|
return (
|
|
<div>
|
|
<div className="relative flex items-center justify-between mb-6">
|
|
<div className="flex items-center gap-3">
|
|
<h1 className="text-2xl font-bold">{t("categories.title")}</h1>
|
|
<PageHelp helpKey="categories" />
|
|
</div>
|
|
<div className="flex items-center gap-2">
|
|
<button
|
|
onClick={() => setShowAllKeywords((v) => !v)}
|
|
className={`flex items-center gap-2 px-3 py-2 rounded-lg border text-sm font-medium transition-colors ${
|
|
showAllKeywords
|
|
? "border-[var(--primary)] bg-[var(--primary)]/10 text-[var(--primary)]"
|
|
: "border-[var(--border)] hover:bg-[var(--muted)]"
|
|
}`}
|
|
>
|
|
<List size={16} />
|
|
{t("categories.allKeywords")}
|
|
</button>
|
|
<button
|
|
onClick={() => setShowReinitConfirm(true)}
|
|
disabled={state.isSaving}
|
|
className="flex items-center gap-2 px-3 py-2 rounded-lg border border-[var(--border)] text-sm font-medium hover:bg-[var(--muted)] transition-colors disabled:opacity-50"
|
|
>
|
|
<RotateCcw size={16} />
|
|
{t("categories.reinitialize")}
|
|
</button>
|
|
<button
|
|
onClick={startCreating}
|
|
className="flex items-center gap-2 px-4 py-2 rounded-lg bg-[var(--primary)] text-white text-sm font-medium hover:opacity-90"
|
|
>
|
|
<Plus size={16} />
|
|
{t("categories.addCategory")}
|
|
</button>
|
|
</div>
|
|
</div>
|
|
|
|
{state.error && (
|
|
<div className="mb-4 p-3 rounded-lg bg-[var(--negative)]/10 text-[var(--negative)] text-sm">
|
|
{state.error}
|
|
</div>
|
|
)}
|
|
|
|
{state.isLoading ? (
|
|
<p className="text-[var(--muted-foreground)]">{t("common.loading")}</p>
|
|
) : showAllKeywords ? (
|
|
<AllKeywordsPanel
|
|
onSelectCategory={(id) => {
|
|
setShowAllKeywords(false);
|
|
selectCategory(id);
|
|
}}
|
|
onRemove={removeKeyword}
|
|
/>
|
|
) : (
|
|
<div className="flex gap-6" style={{ height: "calc(100vh - 180px)" }}>
|
|
<div className="w-1/3 bg-[var(--card)] rounded-xl border border-[var(--border)] p-3 overflow-y-auto">
|
|
<CategoryTree
|
|
tree={state.tree}
|
|
selectedId={state.selectedCategoryId}
|
|
onSelect={selectCategory}
|
|
onMoveCategory={moveCategory}
|
|
/>
|
|
</div>
|
|
<CategoryDetailPanel
|
|
categories={state.categories}
|
|
selectedCategory={selectedCategory}
|
|
keywords={state.keywords}
|
|
editingCategory={state.editingCategory}
|
|
isCreating={state.isCreating}
|
|
isSaving={state.isSaving}
|
|
onStartEditing={startEditing}
|
|
onCancelEditing={cancelEditing}
|
|
onSave={saveCategory}
|
|
onDelete={deleteCategory}
|
|
onAddKeyword={addKeyword}
|
|
onUpdateKeyword={editKeyword}
|
|
onRemoveKeyword={removeKeyword}
|
|
/>
|
|
</div>
|
|
)}
|
|
|
|
{/* Reinitialize confirmation modal */}
|
|
{showReinitConfirm && (
|
|
<div className="fixed inset-0 z-50 flex items-center justify-center bg-black/50">
|
|
<div className="bg-[var(--card)] rounded-xl border border-[var(--border)] shadow-lg p-6 max-w-md w-full mx-4">
|
|
<div className="flex items-center gap-3 mb-4">
|
|
<div className="p-2 rounded-full bg-[var(--negative)]/10">
|
|
<AlertTriangle size={20} className="text-[var(--negative)]" />
|
|
</div>
|
|
<h2 className="text-lg font-semibold">{t("categories.reinitialize")}</h2>
|
|
</div>
|
|
<p className="text-sm text-[var(--muted-foreground)] mb-6">
|
|
{t("categories.reinitializeConfirm")}
|
|
</p>
|
|
<div className="flex justify-end gap-3">
|
|
<button
|
|
onClick={() => setShowReinitConfirm(false)}
|
|
className="px-4 py-2 text-sm rounded-lg border border-[var(--border)] hover:bg-[var(--muted)] transition-colors"
|
|
>
|
|
{t("common.cancel")}
|
|
</button>
|
|
<button
|
|
onClick={handleReinitialize}
|
|
className="px-4 py-2 text-sm rounded-lg bg-[var(--negative)] text-white hover:opacity-90 transition-opacity"
|
|
>
|
|
{t("common.confirm")}
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|