# DataTable Component Developer Documentation ## Overview The `DataTable` component is a reusable, flexible, and customizable table component designed to display data in a tabular format. It uses TanStack Table (formerly React Table) to provide functionalities such as pagination, sorting, row selection, and custom rendering. This component aims to standardize the way lists of items are displayed across the application, promoting code reusability and consistency. **Location**: `packages/ui/src/components/DataTable.tsx` **Last Updated**: January 2025 (Mass pagination fixes + user preference improvements) --- ## Table of Contents - [Features](#features) - [Installation](#installation) - [Quick Start](#quick-start) - [Pagination Guide](#pagination-guide) - [Pagination Modes](#pagination-modes) - [Client-Side Pagination](#client-side-pagination) - [Server-Side Pagination](#server-side-pagination) - [User Preferences](#user-preferences) - [Usage](#usage) - [Basic Usage](#basic-usage) - [Defining Columns](#defining-columns) - [Sorting](#sorting) - [Row Click Handlers](#row-click-handlers) - [Props Interface](#props-interface) - [DataTableProps](#datatableprops) - [ColumnDefinition](#columndefinition) - [Examples](#examples) - [Client-Side Pagination Example](#client-side-pagination-example) - [Server-Side Pagination Example](#server-side-pagination-example) - [With User Preferences Example](#with-user-preferences-example) - [Common Mistakes](#common-mistakes) - [Troubleshooting](#troubleshooting) - [Migration History](#migration-history) - [Frequently Asked Questions](#frequently-asked-questions) --- ## Features - **Data Display**: Render data in a customizable table format with TanStack Table - **Pagination**: Client-side and server-side pagination with customizable page sizes - **Sorting**: Multi-column sorting with asc/desc toggle - **Custom Rendering**: Full control over cell and header rendering - **Row Click Handlers**: Navigate or open drawers on row click - **User Preferences**: Persist page size preferences via `useUserPreference` hook - **Responsive Design**: Adapts to different screen sizes - **Row Styling**: Custom row class names via `rowClassName` callback - **Editable Cells**: Inline editing support via `editableConfig` --- ## Installation The DataTable component is built-in and ready to use: ```tsx import { DataTable } from '@alga-psa/ui'; import { ColumnDefinition, DataTableProps } from '@alga-psa/types'; ``` **Dependencies** (already installed): - `@tanstack/react-table` - Core table functionality - `lucide-react` - Icons for sorting and pagination --- ## Quick Start Here's a minimal working example: ```tsx import { DataTable } from '@alga-psa/ui'; import { useState } from 'react'; function MyComponent() { const [currentPage, setCurrentPage] = useState(1); const [pageSize, setPageSize] = useState(10); const handlePageSizeChange = (newSize: number) => { setPageSize(newSize); setCurrentPage(1); }; const data = [ { id: 1, name: 'John Doe', email: 'john@example.com' }, { id: 2, name: 'Jane Smith', email: 'jane@example.com' }, // ... more data ]; const columns = [ { title: 'Name', dataIndex: 'name' }, { title: 'Email', dataIndex: 'email' }, ]; return ( ); } ``` --- ## Pagination Guide ### Pagination Modes The DataTable supports **two pagination modes**: | Mode | Use When | Data Loading | totalItems Prop | |------|----------|--------------|-----------------| | **Client-Side** | All data can be loaded at once (< 1000 items) | Load all data upfront | ❌ Do NOT pass | | **Server-Side** | Large datasets or expensive queries | Load one page at a time | ✅ Required | **How it decides**: The presence of `totalItems` prop determines the mode. - **No `totalItems`** → Client-side pagination (DataTable handles slicing) - **Has `totalItems`** → Server-side pagination (you handle slicing) --- ### Client-Side Pagination All data is loaded into memory, and DataTable automatically slices it into pages. #### Implementation ```tsx import { useState } from 'react'; import { DataTable } from '@alga-psa/ui'; function ClientsList() { const [clients, setClients] = useState([]); const [currentPage, setCurrentPage] = useState(1); const [pageSize, setPageSize] = useState(10); const handlePageSizeChange = (newPageSize: number) => { setPageSize(newPageSize); setCurrentPage(1); // ⚠️ ALWAYS reset to page 1 }; // Load all data once useEffect(() => { fetchAllClients().then(setClients); }, []); return ( ); } ``` #### Key Points - ✅ Pass **entire dataset** to `data` prop - ✅ DataTable calculates `totalPages` from data length - ✅ DataTable slices data internally - ❌ **Never** pass `totalItems` prop --- ### Server-Side Pagination Only one page of data is fetched from the server at a time. #### Implementation ```tsx import { useState, useEffect } from 'react'; import { DataTable } from '@alga-psa/ui'; function ActivitiesList() { const [activities, setActivities] = useState([]); const [currentPage, setCurrentPage] = useState(1); const [pageSize, setPageSize] = useState(10); const [totalItems, setTotalItems] = useState(0); const handlePageChange = (newPage: number) => { setCurrentPage(newPage); }; const handlePageSizeChange = (newPageSize: number) => { setPageSize(newPageSize); setCurrentPage(1); // ⚠️ ALWAYS reset to page 1 }; // Fetch only current page useEffect(() => { fetchActivitiesPage(currentPage, pageSize).then(result => { setActivities(result.items); // Current page only setTotalItems(result.totalCount); // Total across all pages }); }, [currentPage, pageSize]); return ( ); } ``` #### Backend Response Format ```typescript interface PagedResponse { items: T[]; // Items for current page totalCount: number; // Total count across all pages page: number; // Current page number pageSize: number; // Items per page } ``` #### Key Points - ✅ Pass **only current page items** to `data` prop - ✅ Pass `totalItems` with total count from server - ✅ Re-fetch data when `currentPage` or `pageSize` changes - ✅ Backend must handle LIMIT/OFFSET pagination --- ### User Preferences Use the `useUserPreference` hook to persist the user's page size preference: ```tsx import { useUserPreference } from '@alga-psa/users/hooks'; const PAGE_SIZE_KEY = 'my_page_size_preference'; function MyComponent() { const [currentPage, setCurrentPage] = useState(1); const { value: pageSize, setValue: setPageSize } = useUserPreference(PAGE_SIZE_KEY, { defaultValue: 10, localStorageKey: PAGE_SIZE_KEY, debounceMs: 300 }); const handlePageSizeChange = (newPageSize: number) => { setPageSize(newPageSize); // Saves to localStorage + server setCurrentPage(1); }; return ( ); } ``` #### How it Works 1. **Initial Render**: Reads from localStorage immediately (no flash) 2. **After Mount**: Syncs with server preference 3. **On Change**: Saves to localStorage immediately, debounces server save 4. **Result**: User sees their preference instantly on page load --- ## Usage ### Basic Usage Minimum required props: ```tsx ``` ### Defining Columns Columns are defined using the `ColumnDefinition` interface: ```tsx import { ColumnDefinition } from '@alga-psa/types'; const columns: ColumnDefinition[] = [ { title: 'Name', dataIndex: 'full_name', sortable: true, }, { title: 'Email', dataIndex: 'email', }, { title: 'Status', dataIndex: 'status', render: (value, record) => ( {value} ), }, { title: 'Actions', dataIndex: 'actions', render: (_, record) => ( ), }, ]; ``` #### Column Properties - `title` (required): Column header text or ReactNode - `dataIndex` (required): Key in data object (string or string array for nested paths) - `sortable` (optional): Enable sorting for this column (default: true) - `render` (optional): Custom cell renderer function - `width` (optional): Column width (CSS string value) - `headerClassName` (optional): CSS class for the header cell - `cellClassName` (optional): CSS class for body cells ### Sorting Sorting is enabled by default on all columns. Use the `sortable` property to disable it: ```tsx const columns = [ { title: 'Name', dataIndex: 'name', sortable: true, // ✅ Enable sorting }, { title: 'Created', dataIndex: 'created_at', sortable: true, }, ]; ``` Users can click column headers to toggle between ascending, descending, and no sort. ### Row Click Handlers Handle row clicks for navigation or opening drawers: ```tsx { navigate(`/clients/${record.id}`); }} /> ``` --- ## Props Interface ### DataTableProps ```typescript interface DataTableProps { // Data data: T[]; columns: ColumnDefinition[]; // Pagination pagination?: boolean; currentPage?: number; onPageChange?: (page: number) => void; pageSize?: number; onItemsPerPageChange?: (itemsPerPage: number) => void; totalItems?: number; // Enables server-side pagination itemsPerPageOptions?: Array<{ value: string; label: string }>; // Row interaction onRowClick?: (record: T) => void; rowClassName?: (record: T) => string; onVisibleRowsChange?: (rows: T[]) => void; // Sorting initialSorting?: { id: string; desc: boolean }[]; manualSorting?: boolean; sortBy?: string; sortDirection?: 'asc' | 'desc'; onSortChange?: (sortBy: string, sortDirection: 'asc' | 'desc') => void; // Editable support editableConfig?: EditableConfig; } ``` ### ColumnDefinition The `ColumnDefinition` type is a union of two variants: ```typescript interface BaseColumnDefinition { title: string | ReactNode; // Column header text or ReactNode dataIndex: string | string[]; // Key in data object (or nested path) width?: string; // Column width (CSS string) headerClassName?: string; // CSS class for the header cell cellClassName?: string; // CSS class for body cells sortable?: boolean; // Enable sorting (default: true) } // Variant with custom render function interface RenderColumnDefinition extends BaseColumnDefinition { render: (value: V, record: T, index: number) => ReactNode; } // Variant without render (simple text display) interface SimpleColumnDefinition extends BaseColumnDefinition { render?: never; } type ColumnDefinition = RenderColumnDefinition | SimpleColumnDefinition; ``` --- ## Examples ### Client-Side Pagination Example ```tsx import { useState, useEffect } from 'react'; import { DataTable } from '@alga-psa/ui'; import { IContact } from 'server/src/interfaces/contact.interfaces'; function ContactsList() { const [contacts, setContacts] = useState([]); const [currentPage, setCurrentPage] = useState(1); const [pageSize, setPageSize] = useState(10); const handlePageSizeChange = (newPageSize: number) => { setPageSize(newPageSize); setCurrentPage(1); }; // Load all contacts once useEffect(() => { fetchAllContacts().then(setContacts); }, []); const columns = [ { title: 'Name', dataIndex: 'full_name', sortable: true, render: (text: string, record: IContact) => (
{text}
), }, { title: 'Email', dataIndex: 'email', sortable: true, }, { title: 'Phone', dataIndex: 'phone_number', }, { title: 'Status', dataIndex: 'is_active', render: (value: boolean) => ( {value ? 'Active' : 'Inactive'} ), }, ]; return ( navigate(`/contacts/${contact.contact_name_id}`)} /> ); } ``` ### Server-Side Pagination Example ```tsx import { useState, useEffect } from 'react'; import { DataTable } from '@alga-psa/ui'; import { Activity } from 'server/src/interfaces/activity.interfaces'; function ActivitiesList() { const [activities, setActivities] = useState([]); const [currentPage, setCurrentPage] = useState(1); const [pageSize, setPageSize] = useState(10); const [totalItems, setTotalItems] = useState(0); const handlePageChange = (newPage: number) => { setCurrentPage(newPage); }; const handlePageSizeChange = (newPageSize: number) => { setPageSize(newPageSize); setCurrentPage(1); }; // Fetch current page only useEffect(() => { fetchActivitiesPage(currentPage, pageSize) .then(result => { setActivities(result.items); setTotalItems(result.totalCount); }); }, [currentPage, pageSize]); const columns = [ { title: 'Title', dataIndex: 'title', sortable: true, }, { title: 'Status', dataIndex: 'status', render: (value: string) => ( {value} ), }, { title: 'Due Date', dataIndex: 'dueDate', render: (value: string) => formatDate(value), }, ]; return ( ); } ``` ### With User Preferences Example ```tsx import { useState } from 'react'; import { useUserPreference } from '@alga-psa/users/hooks'; import { DataTable } from '@alga-psa/ui'; const CLIENTS_PAGE_SIZE_KEY = 'clients_list_page_size'; function ClientsList() { const [clients, setClients] = useState([]); const [currentPage, setCurrentPage] = useState(1); // User preference automatically persists const { value: pageSize, setValue: setPageSize } = useUserPreference(CLIENTS_PAGE_SIZE_KEY, { defaultValue: 10, localStorageKey: CLIENTS_PAGE_SIZE_KEY, }); const handlePageSizeChange = (newPageSize: number) => { setPageSize(newPageSize); // Saves automatically setCurrentPage(1); }; return ( ); } ``` --- ## Common Mistakes ### ❌ Mistake 1: Missing `onItemsPerPageChange` **Problem**: Page size dropdown doesn't appear or doesn't work. ```tsx // WRONG ``` **Fix**: ```tsx const handlePageSizeChange = (newPageSize: number) => { setPageSize(newPageSize); setCurrentPage(1); }; ``` --- ### ❌ Mistake 2: Not Resetting to Page 1 **Problem**: User sees empty page after changing page size. ```tsx // WRONG const handlePageSizeChange = (newPageSize: number) => { setPageSize(newPageSize); // ❌ User is still on page 5, which might not exist }; ``` **Fix**: ```tsx const handlePageSizeChange = (newPageSize: number) => { setPageSize(newPageSize); setCurrentPage(1); // ✅ Always reset }; ``` --- ### ❌ Mistake 3: Using `totalItems` for Client-Side **Problem**: Only first page displays despite having all data. ```tsx // WRONG - Triggers server-side mode ``` **Fix**: ```tsx // Correct - Client-side pagination ``` --- ### ❌ Mistake 4: Passing All Data for Server-Side **Problem**: Loads unnecessary data, defeats purpose of server-side pagination. ```tsx // WRONG ``` **Fix**: ```tsx // Fetch only current page const result = await fetchPage(currentPage, pageSize); ``` --- ## Troubleshooting ### Pagination Controls Don't Appear **Possible Causes**: 1. Missing `onItemsPerPageChange` prop 2. Less than 1 page of data 3. `pagination={false}` **Solution**: Ensure all 5 pagination props are provided. --- ### Page Size Dropdown Doesn't Work **Debug**: ```tsx const handlePageSizeChange = (newPageSize: number) => { console.log('Page size changing to:', newPageSize); setPageSize(newPageSize); setCurrentPage(1); }; ``` **Common Issues**: - Handler not passed to DataTable - pageSize state not updating - Not resetting currentPage --- ### Empty Page After Size Change **Cause**: Not resetting to page 1. **Example**: User on page 5 with 10/page → changes to 100/page → page 5 doesn't exist. **Fix**: Always `setCurrentPage(1)` in `handlePageSizeChange`. --- ### Visual Flash on Load **If you see a flash**: User preference may not be using lazy initialization. **Check**: `useUserPreference.ts` should initialize with: ```typescript const [value, setValueState] = useState(() => { // Read localStorage BEFORE first render if (typeof window !== 'undefined' && localStorageKey) { const stored = localStorage.getItem(localStorageKey); if (stored !== null) return JSON.parse(stored); } return defaultValue; }); ``` --- ## Migration History ### January 2025: Mass Pagination Fix **Problem**: ~45-50 DataTable instances missing pagination props after commit `84b5ee258` changed visibility logic. **Solution**: - Deployed parallel agents to fix 38+ DataTable instances - Added `handlePageSizeChange` handlers across the application - Standardized page size options (10/25/50/100 for lists) **Files Modified**: Multiple DataTable instances across the application **Pattern Established**: 1. Add pagination state (`currentPage`, `pageSize`) 2. Add `handlePageSizeChange` that resets to page 1 3. Pass 5 required props to DataTable 4. Use `useUserPreference` for persistence --- ### January 2025: useUserPreference Race Condition Fix **Problem**: Visual flash showing default value (10 items) before loading saved preference (50 items). **Root Cause**: `requestAnimationFrame` delayed localStorage loading until after first paint. **Solution**: Changed to lazy initialization - read localStorage **before** first render. **Files Modified**: `packages/users/src/hooks/useUserPreference.ts` --- ## Frequently Asked Questions ### How do I disable pagination? Set `pagination={false}`: ```tsx ``` --- ### Can I use custom components in cells? Yes, use the `render` function: ```tsx { title: 'Actions', dataIndex: 'actions', render: (_, record) => ( ), } ``` --- ### How do I change default page size options? Pass custom options: ```tsx ``` Defaults are 10/25/50/100 for lists, 9/18/27/36 for grids. --- ### What's the difference between the two DataTable components? 1. **Main DataTable** (`packages/ui/src/components/DataTable.tsx`): Full-featured with pagination, used by main app 2. **UI Kit DataTable** (`packages/ui-kit/src/components/DataTable.tsx`): Minimal version for extensions, **no pagination** Use the server DataTable unless building a standalone extension. --- ### How do I handle row clicks? Use `onRowClick` prop: ```tsx { navigate(`/details/${record.id}`); }} /> ``` --- ### Can I persist sorting preferences? Currently, sorting state is not persisted. Page size preferences are persisted via `useUserPreference`. --- ## Additional Resources - **DataTable Implementation**: `packages/ui/src/components/DataTable.tsx` - **DataTable Types**: `@alga-psa/types` (ColumnDefinition, DataTableProps) - **Pagination Component**: `packages/ui/src/components/Pagination.tsx` - **useUserPreference Hook**: `packages/users/src/hooks/useUserPreference.ts` --- **Last Updated**: January 2025 **Note**: This documentation reflects the current state after the mass pagination fixes. Keep it updated with any changes to the DataTable component.