PSA/docs/ui/datatables.md
Hermes 284313f908
Some checks are pending
Bidi Control Character Guard / bidi-control-guard (push) Waiting to run
Circular Dependency Check / Check for new circular dependencies (push) Waiting to run
Citus Migration Smoke / Combined migrations on single-node Citus (push) Waiting to run
E2E Fresh Install Tests / fresh-install-e2e (push) Waiting to run
ext-v2 guardrails / Run ext-v2 guard and ESLint (push) Waiting to run
Integration Tests / Check for relevant changes (push) Waiting to run
Integration Tests / ${{ (github.event_name == 'schedule' || github.event.inputs.suite == 'full') && 'Full integration suite' || 'Tier-1 integration subset' }} (push) Blocked by required conditions
Mobile checks / Mobile lint + typecheck (push) Waiting to run
Mobile checks / Mobile unit tests (push) Waiting to run
Mobile checks / Mobile dependency audit (report) (push) Waiting to run
Mobile checks / Mobile reproducibility checks (push) Waiting to run
Secrets guard (env backups) / Ensure no tracked env backup files (push) Waiting to run
Temporal Readiness / fast-readiness (push) Waiting to run
Temporal Readiness / docker-parity (push) Waiting to run
TypeScript Type Check / Nx affected typecheck (push) Waiting to run
Unit Tests / Skipped-test budget (push) Waiting to run
Unit Tests / Nx affected unit tests (push) Waiting to run
Unit Tests / Server unit coverage (informational) (push) Waiting to run
Validate Tenant Management Schema / Check for relevant changes (push) Waiting to run
Validate Tenant Management Schema / Validate Tenant Management Schema (push) Blocked by required conditions
EE Workflows Build Guard / ee-workflows-build-guard (push) Waiting to run
Initial import of AlgaPSA codebase from PSA server
Excluded: .git, node_modules, secrets/, compose.env, assemblyscript tgz

Source: /opt/alga-psa on psa.joliet.tech
2026-06-22 16:12:17 -05:00

922 lines
22 KiB
Markdown

# 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 (
<DataTable
data={data}
columns={columns}
pagination={true}
currentPage={currentPage}
onPageChange={setCurrentPage}
pageSize={pageSize}
onItemsPerPageChange={handlePageSizeChange}
/>
);
}
```
---
## 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<Client[]>([]);
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 (
<DataTable
data={clients} // Pass ALL data
columns={columns}
pagination={true}
currentPage={currentPage}
onPageChange={setCurrentPage}
pageSize={pageSize}
onItemsPerPageChange={handlePageSizeChange}
// ❌ Do NOT pass totalItems
/>
);
}
```
#### 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<Activity[]>([]);
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 (
<DataTable
data={activities} // Current page items only
columns={columns}
pagination={true}
currentPage={currentPage}
onPageChange={handlePageChange}
pageSize={pageSize}
onItemsPerPageChange={handlePageSizeChange}
totalItems={totalItems} // ✅ Required for server-side
/>
);
}
```
#### Backend Response Format
```typescript
interface PagedResponse<T> {
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<number>(PAGE_SIZE_KEY, {
defaultValue: 10,
localStorageKey: PAGE_SIZE_KEY,
debounceMs: 300
});
const handlePageSizeChange = (newPageSize: number) => {
setPageSize(newPageSize); // Saves to localStorage + server
setCurrentPage(1);
};
return (
<DataTable
data={data}
columns={columns}
pagination={true}
currentPage={currentPage}
onPageChange={setCurrentPage}
pageSize={pageSize}
onItemsPerPageChange={handlePageSizeChange}
/>
);
}
```
#### 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
<DataTable
data={items}
columns={columnDefinitions}
/>
```
### Defining Columns
Columns are defined using the `ColumnDefinition` interface:
```tsx
import { ColumnDefinition } from '@alga-psa/types';
const columns: ColumnDefinition<Contact>[] = [
{
title: 'Name',
dataIndex: 'full_name',
sortable: true,
},
{
title: 'Email',
dataIndex: 'email',
},
{
title: 'Status',
dataIndex: 'status',
render: (value, record) => (
<Badge variant={value === 'active' ? 'success' : 'default'}>
{value}
</Badge>
),
},
{
title: 'Actions',
dataIndex: 'actions',
render: (_, record) => (
<button onClick={() => handleEdit(record)}>Edit</button>
),
},
];
```
#### 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
<DataTable
data={items}
columns={columns}
onRowClick={(record) => {
navigate(`/clients/${record.id}`);
}}
/>
```
---
## Props Interface
### DataTableProps
```typescript
interface DataTableProps<T> {
// Data
data: T[];
columns: ColumnDefinition<T>[];
// 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<T> {
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<T, V> extends BaseColumnDefinition<T> {
render: (value: V, record: T, index: number) => ReactNode;
}
// Variant without render (simple text display)
interface SimpleColumnDefinition<T> extends BaseColumnDefinition<T> {
render?: never;
}
type ColumnDefinition<T> = RenderColumnDefinition<T, any> | SimpleColumnDefinition<T>;
```
---
## 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<IContact[]>([]);
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) => (
<div className="flex items-center gap-2">
<img
className="h-8 w-8 rounded-full"
src={`https://ui-avatars.com/api/?name=${encodeURIComponent(text)}`}
alt=""
/>
<span>{text}</span>
</div>
),
},
{
title: 'Email',
dataIndex: 'email',
sortable: true,
},
{
title: 'Phone',
dataIndex: 'phone_number',
},
{
title: 'Status',
dataIndex: 'is_active',
render: (value: boolean) => (
<Badge variant={value ? 'success' : 'default'}>
{value ? 'Active' : 'Inactive'}
</Badge>
),
},
];
return (
<DataTable
data={contacts}
columns={columns}
pagination={true}
currentPage={currentPage}
onPageChange={setCurrentPage}
pageSize={pageSize}
onItemsPerPageChange={handlePageSizeChange}
onRowClick={(contact) => 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<Activity[]>([]);
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) => (
<Badge>{value}</Badge>
),
},
{
title: 'Due Date',
dataIndex: 'dueDate',
render: (value: string) => formatDate(value),
},
];
return (
<DataTable
data={activities}
columns={columns}
pagination={true}
currentPage={currentPage}
onPageChange={handlePageChange}
pageSize={pageSize}
onItemsPerPageChange={handlePageSizeChange}
totalItems={totalItems}
/>
);
}
```
### 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<Client[]>([]);
const [currentPage, setCurrentPage] = useState(1);
// User preference automatically persists
const {
value: pageSize,
setValue: setPageSize
} = useUserPreference<number>(CLIENTS_PAGE_SIZE_KEY, {
defaultValue: 10,
localStorageKey: CLIENTS_PAGE_SIZE_KEY,
});
const handlePageSizeChange = (newPageSize: number) => {
setPageSize(newPageSize); // Saves automatically
setCurrentPage(1);
};
return (
<DataTable
data={clients}
columns={columns}
pagination={true}
currentPage={currentPage}
onPageChange={setCurrentPage}
pageSize={pageSize}
onItemsPerPageChange={handlePageSizeChange}
/>
);
}
```
---
## Common Mistakes
### ❌ Mistake 1: Missing `onItemsPerPageChange`
**Problem**: Page size dropdown doesn't appear or doesn't work.
```tsx
// WRONG
<DataTable
data={items}
pagination={true}
currentPage={currentPage}
onPageChange={setCurrentPage}
pageSize={pageSize}
// ❌ Missing onItemsPerPageChange
/>
```
**Fix**:
```tsx
const handlePageSizeChange = (newPageSize: number) => {
setPageSize(newPageSize);
setCurrentPage(1);
};
<DataTable
// ...
onItemsPerPageChange={handlePageSizeChange}
/>
```
---
### ❌ 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
<DataTable
data={allClients}
totalItems={allClients.length} // ❌ Triggers manual pagination
/>
```
**Fix**:
```tsx
// Correct - Client-side pagination
<DataTable
data={allClients}
// ✅ No totalItems prop
/>
```
---
### ❌ Mistake 4: Passing All Data for Server-Side
**Problem**: Loads unnecessary data, defeats purpose of server-side pagination.
```tsx
// WRONG
<DataTable
data={all10000Activities} // ❌ All data loaded
totalItems={10000}
/>
```
**Fix**:
```tsx
// Fetch only current page
const result = await fetchPage(currentPage, pageSize);
<DataTable
data={result.items} // ✅ Only 10-100 items
totalItems={result.total}
/>
```
---
## 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<T>(() => {
// 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
<DataTable data={data} columns={columns} pagination={false} />
```
---
### Can I use custom components in cells?
Yes, use the `render` function:
```tsx
{
title: 'Actions',
dataIndex: 'actions',
render: (_, record) => (
<ActionMenu record={record} />
),
}
```
---
### How do I change default page size options?
Pass custom options:
```tsx
<DataTable
// ...
itemsPerPageOptions={[
{ value: '5', label: '5 per page' },
{ value: '15', label: '15 per page' },
]}
/>
```
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
<DataTable
data={items}
columns={columns}
onRowClick={(record) => {
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.