add rescan button
This commit is contained in:
@@ -47,6 +47,22 @@ async def get_scan_status(library_id: int):
|
|||||||
return {"scanning": scanner.is_scanning(library_id)}
|
return {"scanning": scanner.is_scanning(library_id)}
|
||||||
|
|
||||||
|
|
||||||
|
@router.post("/{library_id}/rescan")
|
||||||
|
async def rescan_library(
|
||||||
|
library_id: int,
|
||||||
|
background_tasks: BackgroundTasks,
|
||||||
|
db: AsyncSession = Depends(get_db),
|
||||||
|
):
|
||||||
|
result = await db.execute(select(Library).where(Library.id == library_id))
|
||||||
|
lib = result.scalars().first()
|
||||||
|
if not lib:
|
||||||
|
raise HTTPException(404, "Library not found")
|
||||||
|
if scanner.is_scanning(library_id):
|
||||||
|
raise HTTPException(409, "Scan already in progress")
|
||||||
|
background_tasks.add_task(scanner.scan_library_background, lib.id, lib.path)
|
||||||
|
return {"scanning": True}
|
||||||
|
|
||||||
|
|
||||||
@router.delete("/{library_id}", status_code=204)
|
@router.delete("/{library_id}", status_code=204)
|
||||||
async def delete_library(library_id: int, db: AsyncSession = Depends(get_db)):
|
async def delete_library(library_id: int, db: AsyncSession = Depends(get_db)):
|
||||||
result = await db.execute(select(Library).where(Library.id == library_id))
|
result = await db.execute(select(Library).where(Library.id == library_id))
|
||||||
@@ -88,7 +104,7 @@ async def browse_library(library_id: int, path: str = "", db: AsyncSession = Dep
|
|||||||
|
|
||||||
entries: list[BrowseEntry] = []
|
entries: list[BrowseEntry] = []
|
||||||
|
|
||||||
for entry in sorted(target.iterdir(), key=lambda e: (e.is_file(), e.name.lower())):
|
for entry in sorted((e for e in target.iterdir() if not e.name.startswith(".")), key=lambda e: (e.is_file(), e.name.lower())):
|
||||||
rel_entry = str(entry.relative_to(root))
|
rel_entry = str(entry.relative_to(root))
|
||||||
if entry.is_dir():
|
if entry.is_dir():
|
||||||
entries.append(BrowseEntry(name=entry.name, type="dir", rel_path=rel_entry))
|
entries.append(BrowseEntry(name=entry.name, type="dir", rel_path=rel_entry))
|
||||||
|
|||||||
@@ -69,6 +69,8 @@ export const api = {
|
|||||||
request<BrowseResult>(`/libraries/${id}/browse?path=${encodeURIComponent(path)}`),
|
request<BrowseResult>(`/libraries/${id}/browse?path=${encodeURIComponent(path)}`),
|
||||||
scanStatus: (id: number) =>
|
scanStatus: (id: number) =>
|
||||||
request<{ scanning: boolean }>(`/libraries/${id}/scan-status`),
|
request<{ scanning: boolean }>(`/libraries/${id}/scan-status`),
|
||||||
|
rescan: (id: number) =>
|
||||||
|
request<{ scanning: boolean }>(`/libraries/${id}/rescan`, { method: "POST" }),
|
||||||
},
|
},
|
||||||
|
|
||||||
media: {
|
media: {
|
||||||
|
|||||||
@@ -8,9 +8,11 @@ interface Props {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export default function FileBrowser({ libraryId }: Props) {
|
export default function FileBrowser({ libraryId }: Props) {
|
||||||
const [currentPath, setCurrentPath] = useState("");
|
const [libraryPaths, setLibraryPaths] = useState<Record<number, string>>({});
|
||||||
const [viewingId, setViewingId] = useState<number | null>(null);
|
const [viewingId, setViewingId] = useState<number | null>(null);
|
||||||
|
|
||||||
|
const currentPath = libraryPaths[libraryId] ?? "";
|
||||||
|
|
||||||
const { data, isLoading } = useQuery<BrowseResult>({
|
const { data, isLoading } = useQuery<BrowseResult>({
|
||||||
queryKey: ["browse", libraryId, currentPath],
|
queryKey: ["browse", libraryId, currentPath],
|
||||||
queryFn: () => api.libraries.browse(libraryId, currentPath),
|
queryFn: () => api.libraries.browse(libraryId, currentPath),
|
||||||
@@ -19,7 +21,7 @@ export default function FileBrowser({ libraryId }: Props) {
|
|||||||
const pathParts = currentPath ? currentPath.split("/").filter(Boolean) : [];
|
const pathParts = currentPath ? currentPath.split("/").filter(Boolean) : [];
|
||||||
|
|
||||||
function navigate(relPath: string) {
|
function navigate(relPath: string) {
|
||||||
setCurrentPath(relPath);
|
setLibraryPaths((prev) => ({ ...prev, [libraryId]: relPath }));
|
||||||
setViewingId(null);
|
setViewingId(null);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -17,6 +17,9 @@ export default function TagPanel({ item }: Props) {
|
|||||||
|
|
||||||
const [newName, setNewName] = useState("");
|
const [newName, setNewName] = useState("");
|
||||||
const [newCategory, setNewCategory] = useState("");
|
const [newCategory, setNewCategory] = useState("");
|
||||||
|
const [quickAddCategory, setQuickAddCategory] = useState<string | null>(null);
|
||||||
|
const [quickAddName, setQuickAddName] = useState("");
|
||||||
|
const [categorySearch, setCategorySearch] = useState<Record<string, string>>({});
|
||||||
|
|
||||||
const setTagsMutation = useMutation({
|
const setTagsMutation = useMutation({
|
||||||
mutationFn: (ids: number[]) => api.media.setTags(item.id, ids),
|
mutationFn: (ids: number[]) => api.media.setTags(item.id, ids),
|
||||||
@@ -24,12 +27,15 @@ export default function TagPanel({ item }: Props) {
|
|||||||
});
|
});
|
||||||
|
|
||||||
const createTagMutation = useMutation({
|
const createTagMutation = useMutation({
|
||||||
mutationFn: () => api.tags.create(newName.trim(), newCategory.trim()),
|
mutationFn: ({ name, category }: { name: string; category: string }) =>
|
||||||
|
api.tags.create(name.trim(), category.trim()),
|
||||||
onSuccess: (tag: Tag) => {
|
onSuccess: (tag: Tag) => {
|
||||||
qc.invalidateQueries({ queryKey: ["tags"] });
|
qc.invalidateQueries({ queryKey: ["tags"] });
|
||||||
setTagsMutation.mutate([...assignedIds, tag.id]);
|
setTagsMutation.mutate([...assignedIds, tag.id]);
|
||||||
setNewName("");
|
setNewName("");
|
||||||
setNewCategory("");
|
setNewCategory("");
|
||||||
|
setQuickAddCategory(null);
|
||||||
|
setQuickAddName("");
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -40,53 +46,130 @@ export default function TagPanel({ item }: Props) {
|
|||||||
setTagsMutation.mutate(next);
|
setTagsMutation.mutate(next);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function toggleQuickAdd(category: string) {
|
||||||
|
if (quickAddCategory === category) {
|
||||||
|
setQuickAddCategory(null);
|
||||||
|
setQuickAddName("");
|
||||||
|
} else {
|
||||||
|
setQuickAddCategory(category);
|
||||||
|
setQuickAddName("");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div style={{ minWidth: 220 }}>
|
<div style={{ minWidth: 220 }}>
|
||||||
<h3 style={{ margin: "0 0 12px", color: "var(--text)" }}>Tags</h3>
|
<h3 style={{ margin: "0 0 12px", color: "var(--text)" }}>Tags</h3>
|
||||||
|
|
||||||
{grouped.map((group) => (
|
{grouped.map((group) => {
|
||||||
<div key={group.category} style={{ marginBottom: 12 }}>
|
const search = categorySearch[group.category] ?? "";
|
||||||
<div style={{ fontSize: 11, fontWeight: 700, textTransform: "uppercase", color: "var(--text-muted)", marginBottom: 4 }}>
|
const visibleTags = search
|
||||||
{group.category}
|
? group.tags.filter((t) => t.name.toLowerCase().includes(search.toLowerCase()))
|
||||||
</div>
|
: group.tags;
|
||||||
<div style={{ display: "flex", flexWrap: "wrap", gap: 6 }}>
|
|
||||||
{group.tags.map((tag) => (
|
return (
|
||||||
|
<div key={group.category} style={{ marginBottom: 16 }}>
|
||||||
|
{/* Category header row */}
|
||||||
|
<div style={{ display: "flex", justifyContent: "space-between", alignItems: "center", marginBottom: 6 }}>
|
||||||
|
<span style={{ fontSize: 11, fontWeight: 700, textTransform: "uppercase", color: "var(--text-muted)" }}>
|
||||||
|
{group.category}
|
||||||
|
</span>
|
||||||
<button
|
<button
|
||||||
key={tag.id}
|
onClick={() => toggleQuickAdd(group.category)}
|
||||||
onClick={() => toggle(tag.id)}
|
title={`Add tag to ${group.category}`}
|
||||||
style={{
|
style={{
|
||||||
padding: "3px 10px",
|
background: quickAddCategory === group.category ? "var(--accent)" : "transparent",
|
||||||
borderRadius: 12,
|
color: quickAddCategory === group.category ? "#fff" : "var(--text-muted)",
|
||||||
border: "1px solid",
|
border: "1px solid var(--border)",
|
||||||
|
borderRadius: 4,
|
||||||
|
padding: "1px 7px",
|
||||||
|
fontSize: 16,
|
||||||
|
lineHeight: 1,
|
||||||
cursor: "pointer",
|
cursor: "pointer",
|
||||||
background: assignedIds.has(tag.id) ? "var(--accent)" : "transparent",
|
|
||||||
color: assignedIds.has(tag.id) ? "#fff" : "var(--text)",
|
|
||||||
borderColor: assignedIds.has(tag.id) ? "var(--accent)" : "var(--border)",
|
|
||||||
fontSize: 13,
|
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
{tag.name}
|
+
|
||||||
</button>
|
</button>
|
||||||
))}
|
</div>
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
))}
|
|
||||||
|
|
||||||
<details style={{ marginTop: 16 }}>
|
{/* Per-category search */}
|
||||||
|
<input
|
||||||
|
placeholder={`Search ${group.category.toLowerCase()}…`}
|
||||||
|
value={search}
|
||||||
|
onChange={(e) =>
|
||||||
|
setCategorySearch((prev) => ({ ...prev, [group.category]: e.target.value }))
|
||||||
|
}
|
||||||
|
style={{ marginBottom: 6, fontSize: 12, padding: "3px 8px" }}
|
||||||
|
/>
|
||||||
|
|
||||||
|
{/* Tag chips */}
|
||||||
|
<div style={{ display: "flex", flexWrap: "wrap", gap: 6 }}>
|
||||||
|
{visibleTags.map((tag) => (
|
||||||
|
<button
|
||||||
|
key={tag.id}
|
||||||
|
onClick={() => toggle(tag.id)}
|
||||||
|
style={{
|
||||||
|
padding: "3px 10px",
|
||||||
|
borderRadius: 12,
|
||||||
|
border: "1px solid",
|
||||||
|
cursor: "pointer",
|
||||||
|
background: assignedIds.has(tag.id) ? "var(--accent)" : "transparent",
|
||||||
|
color: assignedIds.has(tag.id) ? "#fff" : "var(--text)",
|
||||||
|
borderColor: assignedIds.has(tag.id) ? "var(--accent)" : "var(--border)",
|
||||||
|
fontSize: 13,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{tag.name}
|
||||||
|
</button>
|
||||||
|
))}
|
||||||
|
{search && visibleTags.length === 0 && (
|
||||||
|
<span style={{ fontSize: 12, color: "var(--text-muted)" }}>No matches</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Inline quick-add form */}
|
||||||
|
{quickAddCategory === group.category && (
|
||||||
|
<div style={{ display: "flex", gap: 6, marginTop: 8 }}>
|
||||||
|
<input
|
||||||
|
placeholder="Tag name"
|
||||||
|
value={quickAddName}
|
||||||
|
onChange={(e) => setQuickAddName(e.target.value)}
|
||||||
|
onKeyDown={(e) => {
|
||||||
|
if (e.key === "Enter" && quickAddName.trim()) {
|
||||||
|
createTagMutation.mutate({ name: quickAddName, category: group.category });
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
autoFocus
|
||||||
|
style={{ flex: 1, fontSize: 12, padding: "3px 8px" }}
|
||||||
|
/>
|
||||||
|
<button
|
||||||
|
onClick={() => createTagMutation.mutate({ name: quickAddName, category: group.category })}
|
||||||
|
disabled={!quickAddName.trim() || createTagMutation.isPending}
|
||||||
|
style={{ background: "var(--accent)", color: "#fff", border: "none", padding: "3px 10px", borderRadius: 4, fontSize: 12 }}
|
||||||
|
>
|
||||||
|
Add
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
|
||||||
|
{/* Global new-tag form */}
|
||||||
|
<details style={{ marginTop: 8 }}>
|
||||||
<summary style={{ cursor: "pointer", fontSize: 13, color: "var(--accent-text)" }}>+ New tag</summary>
|
<summary style={{ cursor: "pointer", fontSize: 13, color: "var(--accent-text)" }}>+ New tag</summary>
|
||||||
<div style={{ display: "flex", flexDirection: "column", gap: 6, marginTop: 8 }}>
|
<div style={{ display: "flex", flexDirection: "column", gap: 6, marginTop: 8 }}>
|
||||||
<input
|
|
||||||
placeholder="Tag name"
|
|
||||||
value={newName}
|
|
||||||
onChange={(e) => setNewName(e.target.value)}
|
|
||||||
/>
|
|
||||||
<input
|
<input
|
||||||
placeholder="Category"
|
placeholder="Category"
|
||||||
value={newCategory}
|
value={newCategory}
|
||||||
onChange={(e) => setNewCategory(e.target.value)}
|
onChange={(e) => setNewCategory(e.target.value)}
|
||||||
/>
|
/>
|
||||||
|
<input
|
||||||
|
placeholder="Tag name"
|
||||||
|
value={newName}
|
||||||
|
onChange={(e) => setNewName(e.target.value)}
|
||||||
|
/>
|
||||||
<button
|
<button
|
||||||
onClick={() => createTagMutation.mutate()}
|
onClick={() => createTagMutation.mutate({ name: newName, category: newCategory })}
|
||||||
disabled={!newName.trim() || !newCategory.trim() || createTagMutation.isPending}
|
disabled={!newName.trim() || !newCategory.trim() || createTagMutation.isPending}
|
||||||
>
|
>
|
||||||
Create & assign
|
Create & assign
|
||||||
|
|||||||
@@ -3,6 +3,7 @@ import { useQuery, useMutation, useQueryClient } from "@tanstack/react-query";
|
|||||||
import { api, type Library } from "../api/client";
|
import { api, type Library } from "../api/client";
|
||||||
|
|
||||||
function LibraryRow({ lib, onRemove }: { lib: Library; onRemove: (id: number) => void }) {
|
function LibraryRow({ lib, onRemove }: { lib: Library; onRemove: (id: number) => void }) {
|
||||||
|
const qc = useQueryClient();
|
||||||
const { data } = useQuery({
|
const { data } = useQuery({
|
||||||
queryKey: ["scan-status", lib.id],
|
queryKey: ["scan-status", lib.id],
|
||||||
queryFn: () => api.libraries.scanStatus(lib.id),
|
queryFn: () => api.libraries.scanStatus(lib.id),
|
||||||
@@ -11,6 +12,11 @@ function LibraryRow({ lib, onRemove }: { lib: Library; onRemove: (id: number) =>
|
|||||||
|
|
||||||
const scanning = data?.scanning ?? false;
|
const scanning = data?.scanning ?? false;
|
||||||
|
|
||||||
|
const rescanMutation = useMutation({
|
||||||
|
mutationFn: () => api.libraries.rescan(lib.id),
|
||||||
|
onSuccess: () => qc.invalidateQueries({ queryKey: ["scan-status", lib.id] }),
|
||||||
|
});
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<li style={{ display: "flex", justifyContent: "space-between", alignItems: "center", padding: "8px 0", borderBottom: "1px solid var(--border-subtle)" }}>
|
<li style={{ display: "flex", justifyContent: "space-between", alignItems: "center", padding: "8px 0", borderBottom: "1px solid var(--border-subtle)" }}>
|
||||||
<div>
|
<div>
|
||||||
@@ -22,13 +28,22 @@ function LibraryRow({ lib, onRemove }: { lib: Library; onRemove: (id: number) =>
|
|||||||
)}
|
)}
|
||||||
<div style={{ fontSize: 12, color: "var(--text-secondary)" }}>{lib.path}</div>
|
<div style={{ fontSize: 12, color: "var(--text-secondary)" }}>{lib.path}</div>
|
||||||
</div>
|
</div>
|
||||||
<button
|
<div style={{ display: "flex", gap: 8 }}>
|
||||||
onClick={() => onRemove(lib.id)}
|
<button
|
||||||
disabled={scanning}
|
onClick={() => rescanMutation.mutate()}
|
||||||
style={{ color: scanning ? "var(--text-muted)" : "var(--danger)", background: "transparent", border: "none" }}
|
disabled={scanning}
|
||||||
>
|
style={{ color: scanning ? "var(--text-muted)" : "var(--accent)", background: "transparent", border: "none" }}
|
||||||
Remove
|
>
|
||||||
</button>
|
Rescan
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
onClick={() => onRemove(lib.id)}
|
||||||
|
disabled={scanning}
|
||||||
|
style={{ color: scanning ? "var(--text-muted)" : "var(--danger)", background: "transparent", border: "none" }}
|
||||||
|
>
|
||||||
|
Remove
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
</li>
|
</li>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user