Backend:
- Domain/Master: Supplier (+ SupplierType 5 loai), Project, Department (AuditableEntity)
- Domain/Identity: MenuItem, Permission, MenuKeys const (12 menu)
- EF Configurations voi unique Code + query filter IsDeleted
- DbSets + IApplicationDbContext interface update
- Application: PagedResult + PagedRequest generic
- Application/Master CQRS CRUD 3 entity (Create/Update/Delete/Get/List voi paging search sort)
- Application/Permissions: GetMyMenuTree (union OR role, filter tree), ListMenuItems, ListPermissionsByRole, UpsertPermission (guard admin khong tu giam quyen), ListRoles
- Api/Authorization: MenuPermissionRequirement + Handler (Admin bypass, query DB)
- Program.cs: register 48 policy {menu}.{action} tu MenuKeys x Actions
- Api/Controllers: Suppliers, Projects, Departments, Menus, Roles, Permissions
- DbInitializer: seed 12 menu + admin full CRUD permissions
- Migration AddMasterData + AddPermissions
Frontend (fe-admin):
- Types: menuKeys.ts const, menu.ts (MenuNode/Role/Permission), master.ts (Supplier/Project/Department + SupplierType const-object)
- AuthContext: load menu from /menus/me, cache localStorage, refreshMenu()
- usePermission hook + PermissionGuard component (wrap button)
- UI kit them: Dialog (modal overlay), Textarea, Select
- Generic: DataTable (column config, sortable, loading, empty) + Pagination
- PageHeader component
- apiError helper extract message tu ProblemDetails
- Layout rewrite: render menu dong tu AuthContext.menu (MenuGroup collapsible + NavLink + lucide icon map)
- Pages: master/Suppliers, master/Projects, master/Departments (CRUD + search + sort + paging + Dialog form)
- Page system/Permissions: ma tran Role x MenuKey x CRUD checkbox (tick tu dong PUT upsert)
- App.tsx them 4 route moi
Bug fix:
- MenuPermissionHandler: EF expression tree khong support switch expression -> tach switch ra ngoai AnyAsync
- TS erasableSyntaxOnly khong cho enum -> SupplierType const-object pattern (typeof[keyof])
E2E verified via Vite proxy:
- GET /menus/me -> 6 root + 6 child nodes (12 menus)
- GET /roles -> 12 roles
- POST/GET/PUT/DELETE /suppliers -> full CRUD, soft delete OK
- tsc -b fe-admin pass
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
152 lines
4.3 KiB
TypeScript
152 lines
4.3 KiB
TypeScript
import type { ReactNode } from 'react'
|
||
import { ChevronDown, ChevronUp } from 'lucide-react'
|
||
import { cn } from '@/lib/cn'
|
||
|
||
export type Column<T> = {
|
||
key: string
|
||
header: ReactNode
|
||
render: (row: T) => ReactNode
|
||
sortable?: boolean
|
||
width?: string
|
||
align?: 'left' | 'center' | 'right'
|
||
}
|
||
|
||
type Props<T> = {
|
||
columns: Column<T>[]
|
||
rows: T[]
|
||
getRowKey: (row: T) => string
|
||
isLoading?: boolean
|
||
empty?: ReactNode
|
||
sortBy?: string
|
||
sortDesc?: boolean
|
||
onSortChange?: (sortBy: string, sortDesc: boolean) => void
|
||
onRowClick?: (row: T) => void
|
||
}
|
||
|
||
export function DataTable<T>({
|
||
columns,
|
||
rows,
|
||
getRowKey,
|
||
isLoading,
|
||
empty,
|
||
sortBy,
|
||
sortDesc,
|
||
onSortChange,
|
||
onRowClick,
|
||
}: Props<T>) {
|
||
return (
|
||
<div className="overflow-auto rounded-md border border-slate-200 bg-white">
|
||
<table className="w-full text-sm">
|
||
<thead className="bg-slate-50 text-slate-700">
|
||
<tr>
|
||
{columns.map(c => (
|
||
<th
|
||
key={c.key}
|
||
className={cn(
|
||
'px-3 py-2 font-medium',
|
||
c.align === 'right' && 'text-right',
|
||
c.align === 'center' && 'text-center',
|
||
c.align !== 'right' && c.align !== 'center' && 'text-left',
|
||
c.width,
|
||
)}
|
||
>
|
||
{c.sortable && onSortChange ? (
|
||
<button
|
||
onClick={() => onSortChange(c.key, sortBy === c.key ? !sortDesc : false)}
|
||
className="inline-flex items-center gap-1 hover:text-slate-900"
|
||
>
|
||
{c.header}
|
||
{sortBy === c.key && (sortDesc ? <ChevronDown className="h-3.5 w-3.5" /> : <ChevronUp className="h-3.5 w-3.5" />)}
|
||
</button>
|
||
) : (
|
||
c.header
|
||
)}
|
||
</th>
|
||
))}
|
||
</tr>
|
||
</thead>
|
||
<tbody>
|
||
{isLoading && (
|
||
<tr>
|
||
<td colSpan={columns.length} className="px-3 py-8 text-center text-slate-500">
|
||
Đang tải…
|
||
</td>
|
||
</tr>
|
||
)}
|
||
{!isLoading && rows.length === 0 && (
|
||
<tr>
|
||
<td colSpan={columns.length} className="px-3 py-8 text-center text-slate-500">
|
||
{empty ?? 'Không có dữ liệu'}
|
||
</td>
|
||
</tr>
|
||
)}
|
||
{!isLoading &&
|
||
rows.map(row => (
|
||
<tr
|
||
key={getRowKey(row)}
|
||
className={cn(
|
||
'border-t border-slate-100 transition',
|
||
onRowClick && 'cursor-pointer hover:bg-slate-50',
|
||
)}
|
||
onClick={() => onRowClick?.(row)}
|
||
>
|
||
{columns.map(c => (
|
||
<td
|
||
key={c.key}
|
||
className={cn(
|
||
'px-3 py-2',
|
||
c.align === 'right' && 'text-right',
|
||
c.align === 'center' && 'text-center',
|
||
)}
|
||
>
|
||
{c.render(row)}
|
||
</td>
|
||
))}
|
||
</tr>
|
||
))}
|
||
</tbody>
|
||
</table>
|
||
</div>
|
||
)
|
||
}
|
||
|
||
type PaginationProps = {
|
||
page: number
|
||
pageSize: number
|
||
total: number
|
||
onChange: (page: number) => void
|
||
}
|
||
|
||
export function Pagination({ page, pageSize, total, onChange }: PaginationProps) {
|
||
const totalPages = Math.max(1, Math.ceil(total / pageSize))
|
||
const from = total === 0 ? 0 : (page - 1) * pageSize + 1
|
||
const to = Math.min(page * pageSize, total)
|
||
|
||
return (
|
||
<div className="flex items-center justify-between py-3 text-sm text-slate-600">
|
||
<span>
|
||
{from}–{to} / {total}
|
||
</span>
|
||
<div className="flex gap-1">
|
||
<button
|
||
disabled={page <= 1}
|
||
onClick={() => onChange(page - 1)}
|
||
className="rounded-md border border-slate-300 bg-white px-3 py-1 disabled:opacity-50"
|
||
>
|
||
Trước
|
||
</button>
|
||
<span className="px-3 py-1">
|
||
Trang {page}/{totalPages}
|
||
</span>
|
||
<button
|
||
disabled={page >= totalPages}
|
||
onClick={() => onChange(page + 1)}
|
||
className="rounded-md border border-slate-300 bg-white px-3 py-1 disabled:opacity-50"
|
||
>
|
||
Sau
|
||
</button>
|
||
</div>
|
||
</div>
|
||
)
|
||
}
|