fix(knowledge): resolve TypeScript errors in tags service
- Fix updateData typing for partial updates - Add slug field to CreateTagDto - Build now passes Note: tasks.controller.spec.ts needs test config update for WorkspaceGuard
This commit is contained in:
144
apps/web/src/app/(authenticated)/knowledge/page.tsx
Normal file
144
apps/web/src/app/(authenticated)/knowledge/page.tsx
Normal file
@@ -0,0 +1,144 @@
|
||||
"use client";
|
||||
|
||||
import { useState, useMemo } from "react";
|
||||
import { EntryStatus } from "@mosaic/shared";
|
||||
import { EntryList } from "@/components/knowledge/EntryList";
|
||||
import { EntryFilters } from "@/components/knowledge/EntryFilters";
|
||||
import { mockEntries, mockTags } from "@/lib/api/knowledge";
|
||||
import Link from "next/link";
|
||||
import { Plus } from "lucide-react";
|
||||
|
||||
export default function KnowledgePage() {
|
||||
// TODO: Replace with real API call when backend is ready
|
||||
// const { data: entries, isLoading } = useQuery({
|
||||
// queryKey: ["knowledge-entries"],
|
||||
// queryFn: fetchEntries,
|
||||
// });
|
||||
|
||||
const [isLoading] = useState(false);
|
||||
|
||||
// Filter and sort state
|
||||
const [selectedStatus, setSelectedStatus] = useState<EntryStatus | "all">("all");
|
||||
const [selectedTag, setSelectedTag] = useState<string | "all">("all");
|
||||
const [searchQuery, setSearchQuery] = useState("");
|
||||
const [sortBy, setSortBy] = useState<"updatedAt" | "createdAt" | "title">("updatedAt");
|
||||
const [sortOrder, setSortOrder] = useState<"asc" | "desc">("desc");
|
||||
|
||||
// Pagination state
|
||||
const [currentPage, setCurrentPage] = useState(1);
|
||||
const itemsPerPage = 10;
|
||||
|
||||
// Client-side filtering and sorting
|
||||
const filteredAndSortedEntries = useMemo(() => {
|
||||
let filtered = [...mockEntries];
|
||||
|
||||
// Filter by status
|
||||
if (selectedStatus !== "all") {
|
||||
filtered = filtered.filter((entry) => entry.status === selectedStatus);
|
||||
}
|
||||
|
||||
// Filter by tag
|
||||
if (selectedTag !== "all") {
|
||||
filtered = filtered.filter((entry) =>
|
||||
entry.tags.some((tag) => tag.slug === selectedTag)
|
||||
);
|
||||
}
|
||||
|
||||
// Filter by search query
|
||||
if (searchQuery.trim()) {
|
||||
const query = searchQuery.toLowerCase();
|
||||
filtered = filtered.filter(
|
||||
(entry) =>
|
||||
entry.title.toLowerCase().includes(query) ||
|
||||
entry.summary?.toLowerCase().includes(query) ||
|
||||
entry.tags.some((tag) => tag.name.toLowerCase().includes(query))
|
||||
);
|
||||
}
|
||||
|
||||
// Sort entries
|
||||
filtered.sort((a, b) => {
|
||||
let comparison = 0;
|
||||
|
||||
if (sortBy === "title") {
|
||||
comparison = a.title.localeCompare(b.title);
|
||||
} else if (sortBy === "createdAt") {
|
||||
comparison = new Date(a.createdAt).getTime() - new Date(b.createdAt).getTime();
|
||||
} else {
|
||||
// updatedAt
|
||||
comparison = new Date(a.updatedAt).getTime() - new Date(b.updatedAt).getTime();
|
||||
}
|
||||
|
||||
return sortOrder === "asc" ? comparison : -comparison;
|
||||
});
|
||||
|
||||
return filtered;
|
||||
}, [selectedStatus, selectedTag, searchQuery, sortBy, sortOrder]);
|
||||
|
||||
// Pagination
|
||||
const totalPages = Math.ceil(filteredAndSortedEntries.length / itemsPerPage);
|
||||
const paginatedEntries = filteredAndSortedEntries.slice(
|
||||
(currentPage - 1) * itemsPerPage,
|
||||
currentPage * itemsPerPage
|
||||
);
|
||||
|
||||
// Reset to page 1 when filters change
|
||||
const handleFilterChange = (callback: () => void) => {
|
||||
callback();
|
||||
setCurrentPage(1);
|
||||
};
|
||||
|
||||
const handleSortChange = (
|
||||
newSortBy: "updatedAt" | "createdAt" | "title",
|
||||
newSortOrder: "asc" | "desc"
|
||||
) => {
|
||||
setSortBy(newSortBy);
|
||||
setSortOrder(newSortOrder);
|
||||
setCurrentPage(1);
|
||||
};
|
||||
|
||||
return (
|
||||
<main className="container mx-auto px-4 py-8 max-w-5xl">
|
||||
{/* Header */}
|
||||
<div className="mb-8 flex items-center justify-between">
|
||||
<div>
|
||||
<h1 className="text-3xl font-bold text-gray-900">Knowledge Base</h1>
|
||||
<p className="text-gray-600 mt-2">
|
||||
Documentation, guides, and knowledge entries
|
||||
</p>
|
||||
</div>
|
||||
|
||||
{/* Create button */}
|
||||
<Link
|
||||
href="/knowledge/new"
|
||||
className="inline-flex items-center gap-2 px-4 py-2 bg-blue-600 text-white rounded-lg hover:bg-blue-700 transition-colors shadow-sm"
|
||||
>
|
||||
<Plus className="w-5 h-5" />
|
||||
<span>Create Entry</span>
|
||||
</Link>
|
||||
</div>
|
||||
|
||||
{/* Filters */}
|
||||
<EntryFilters
|
||||
selectedStatus={selectedStatus}
|
||||
selectedTag={selectedTag}
|
||||
searchQuery={searchQuery}
|
||||
sortBy={sortBy}
|
||||
sortOrder={sortOrder}
|
||||
tags={mockTags}
|
||||
onStatusChange={(status) => handleFilterChange(() => setSelectedStatus(status))}
|
||||
onTagChange={(tag) => handleFilterChange(() => setSelectedTag(tag))}
|
||||
onSearchChange={(query) => handleFilterChange(() => setSearchQuery(query))}
|
||||
onSortChange={handleSortChange}
|
||||
/>
|
||||
|
||||
{/* Entry list */}
|
||||
<EntryList
|
||||
entries={paginatedEntries}
|
||||
isLoading={isLoading}
|
||||
currentPage={currentPage}
|
||||
totalPages={totalPages}
|
||||
onPageChange={setCurrentPage}
|
||||
/>
|
||||
</main>
|
||||
);
|
||||
}
|
||||
Reference in New Issue
Block a user