periplo-ui 3.21.0 → 3.21.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/components/Button/Button.d.ts +0 -10
- package/dist/components/Button/Button.js +3 -5
- package/dist/components/Button/Button.js.map +1 -1
- package/dist/components/Carousel/Carousel.d.ts +0 -2
- package/dist/components/DataTable/DataTable.d.ts +2 -90
- package/dist/components/DataTable/DataTable.js +101 -52
- package/dist/components/DataTable/DataTable.js.map +1 -1
- package/dist/components/DataTable/components/DataTableBody.d.ts +5 -5
- package/dist/components/DataTable/components/DataTableBody.js +20 -31
- package/dist/components/DataTable/components/DataTableBody.js.map +1 -1
- package/dist/components/DataTable/{DataTablePagination.d.ts → components/DataTablePagination.d.ts} +2 -2
- package/dist/components/DataTable/{DataTablePagination.js → components/DataTablePagination.js} +2 -2
- package/dist/components/DataTable/components/DataTablePagination.js.map +1 -0
- package/dist/components/DataTable/components/DataTableRowSelector.d.ts +9 -0
- package/dist/components/DataTable/components/DataTableRowSelector.js +21 -0
- package/dist/components/DataTable/components/DataTableRowSelector.js.map +1 -0
- package/dist/components/DataTable/components/DataTableSelectHeader.d.ts +9 -0
- package/dist/components/DataTable/components/DataTableSelectHeader.js +26 -0
- package/dist/components/DataTable/components/DataTableSelectHeader.js.map +1 -0
- package/dist/components/DataTable/components/DataTableToolbar.d.ts +13 -1
- package/dist/components/DataTable/components/DataTableToolbar.js +36 -12
- package/dist/components/DataTable/components/DataTableToolbar.js.map +1 -1
- package/dist/components/DataTable/data.d.ts +8 -0
- package/dist/components/DataTable/data.js +506 -0
- package/dist/components/DataTable/data.js.map +1 -0
- package/dist/components/DataTable/hooks/useSelection.d.ts +15 -0
- package/dist/components/DataTable/hooks/useSelection.js +37 -0
- package/dist/components/DataTable/hooks/useSelection.js.map +1 -0
- package/dist/components/DataTable/types.d.js +2 -0
- package/dist/components/DataTable/types.d.js.map +1 -0
- package/dist/components/DataTable/types.d.ts +149 -0
- package/dist/components/Sidebar/Sidebar.d.ts +0 -1
- package/package.json +1 -1
- package/dist/components/DataTable/DataTablePagination.js.map +0 -1
- package/dist/components/DataTable/hooks/useTableSelection.d.ts +0 -14
- package/dist/components/DataTable/hooks/useTableSelection.js +0 -55
- package/dist/components/DataTable/hooks/useTableSelection.js.map +0 -1
|
@@ -44,11 +44,6 @@ export type ButtonProps = {
|
|
|
44
44
|
* Not displayed when iconOnly is true.
|
|
45
45
|
*/
|
|
46
46
|
children?: React.ReactNode;
|
|
47
|
-
/**
|
|
48
|
-
* When true, renders the button as a child component using Radix Slot.
|
|
49
|
-
* Useful for custom wrappers or links.
|
|
50
|
-
*/
|
|
51
|
-
asChild?: boolean;
|
|
52
47
|
/**
|
|
53
48
|
* Custom props to pass to the icons (StartIcon, EndIcon, SpinnerGap).
|
|
54
49
|
* Allows overriding size, weight, color, etc.
|
|
@@ -110,11 +105,6 @@ declare const Button: React.ForwardRefExoticComponent<{
|
|
|
110
105
|
* Not displayed when iconOnly is true.
|
|
111
106
|
*/
|
|
112
107
|
children?: React.ReactNode;
|
|
113
|
-
/**
|
|
114
|
-
* When true, renders the button as a child component using Radix Slot.
|
|
115
|
-
* Useful for custom wrappers or links.
|
|
116
|
-
*/
|
|
117
|
-
asChild?: boolean;
|
|
118
108
|
/**
|
|
119
109
|
* Custom props to pass to the icons (StartIcon, EndIcon, SpinnerGap).
|
|
120
110
|
* Allows overriding size, weight, color, etc.
|
|
@@ -4,7 +4,6 @@ import { cva } from 'class-variance-authority';
|
|
|
4
4
|
import * as React from 'react';
|
|
5
5
|
import { cn } from '../../lib/utils.js';
|
|
6
6
|
import { typographyVariants } from '../Typography/Typography.js';
|
|
7
|
-
import { Slot } from '@radix-ui/react-slot';
|
|
8
7
|
|
|
9
8
|
const buttonVariants = cva(
|
|
10
9
|
"inline-flex items-center justify-center gap-2 rounded-md font-medium transition-colors focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:pointer-events-none disabled:opacity-50",
|
|
@@ -46,7 +45,7 @@ const buttonVariants = cva(
|
|
|
46
45
|
}
|
|
47
46
|
);
|
|
48
47
|
const Button = React.forwardRef(
|
|
49
|
-
({ className, variant = "primary", size = "md", StartIcon, EndIcon, loading, iconOnly, iconProps = {},
|
|
48
|
+
({ className, variant = "primary", size = "md", StartIcon, EndIcon, loading, iconOnly, iconProps = {}, ...props }, ref) => {
|
|
50
49
|
const defaultIconProps = () => {
|
|
51
50
|
switch (size) {
|
|
52
51
|
case "sm":
|
|
@@ -68,9 +67,8 @@ const Button = React.forwardRef(
|
|
|
68
67
|
return "h-10 w-10";
|
|
69
68
|
}
|
|
70
69
|
};
|
|
71
|
-
const Comp = asChild ? Slot : "button";
|
|
72
70
|
return /* @__PURE__ */ jsxs(
|
|
73
|
-
|
|
71
|
+
"button",
|
|
74
72
|
{
|
|
75
73
|
className: cn(
|
|
76
74
|
buttonVariants({ variant, size, className }),
|
|
@@ -79,7 +77,7 @@ const Button = React.forwardRef(
|
|
|
79
77
|
className
|
|
80
78
|
),
|
|
81
79
|
ref,
|
|
82
|
-
disabled: loading
|
|
80
|
+
disabled: loading,
|
|
83
81
|
...props,
|
|
84
82
|
children: [
|
|
85
83
|
loading && /* @__PURE__ */ jsx(SpinnerGap, { ...defaultIconProps(), className: "animate-spin", "aria-label": "Loading" }),
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"Button.js","sources":["../../../src/components/Button/Button.tsx"],"sourcesContent":["import { Icon, IconWeight, IconProps as PhosphorIconProps } from '@phosphor-icons/react'\nimport { SpinnerGap } from '@phosphor-icons/react/dist/ssr'\nimport { cva, type VariantProps } from 'class-variance-authority'\nimport * as React from 'react'\n\nimport { cn } from '../../lib/utils'\nimport { typographyVariants } from '../Typography'\
|
|
1
|
+
{"version":3,"file":"Button.js","sources":["../../../src/components/Button/Button.tsx"],"sourcesContent":["import { Icon, IconWeight, IconProps as PhosphorIconProps } from '@phosphor-icons/react'\nimport { SpinnerGap } from '@phosphor-icons/react/dist/ssr'\nimport { cva, type VariantProps } from 'class-variance-authority'\nimport * as React from 'react'\n\nimport { cn } from '../../lib/utils'\nimport { typographyVariants } from '../Typography'\n\nconst buttonVariants = cva(\n 'inline-flex items-center justify-center gap-2 rounded-md font-medium transition-colors focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:pointer-events-none disabled:opacity-50',\n {\n variants: {\n /**\n * Defines the visual style of the button\n */\n variant: {\n primary:\n 'bg-primary-500 text-primary-foreground hover:bg-primary-700 active:bg-primary-900 disabled:bg-background-200 dark:disabled:bg-background-200 dark:disabled:text-neutral',\n secondary:\n 'bg-neutral-950 text-neutral-foreground hover:bg-neutral-800 active:bg-neutral-700 disabled:bg-neutral-200 dark:bg-neutral-600 dark:hover:bg-neutral-500 dark:disable:bg-neutral-900 disabled:text-neutral-600 dark:disabled:text-neutral-300',\n ghost:\n 'border border-neutral-500 text-primary-foreground hover:bg-neutral-100/50 active:bg-neutral-200/50 dark:border-neutral dark:text-neutral-50 dark:active:bg-neutral-600/50 disabled:border-neutral-500 dark:hover:bg-neutral/50 disabled:text-neutral-500 dark:disabled:border-neutral-300 dark:disabled:text-neutral-300',\n 'inverse-ghost':\n 'border border-neutral-50 text-neutral-50 hover:bg-neutral/50 hover:border-neutral-50 active:bg-neutral-600/50 dark:border-neutral-500 dark:text-primary-foreground dark:hover:bg-neutral-100/50 dark:active:bg-neutral-200/50 disabled:border-neutral-300 disabled:text-neutral-300 dark:disabled:border-neutral-500 dark:disabled:text-neutral-500',\n text: 'hover:bg-neutral-50 active:bg-neutral-100 text-primary-foreground disabled:text-neutral-500 dark:disable:text-neutral-300 dark:text-neutral-50 dark:hover:bg-neutral dark:active:bg-neutral-600',\n error:\n 'bg-error text-white hover:bg-error-800 active:bg-error-900 disabled:bg-error-200 disabled:text-neutral-600',\n success:\n 'bg-success text-white hover:bg-success-800 active:bg-success-900 disabled:bg-success-200 disabled:text-neutral-600',\n input:\n 'border border-neutral-200 bg-white text-neutral-950 hover:bg-neutral-50 active:bg-neutral-100 focus-visible:border-neutral-950 disabled:bg-neutral-50 dark:border-neutral-500 dark:bg-neutral-950 dark:text-neutral-50 dark:hover:bg-neutral-800 dark:active:bg-neutral-700 dark:focus-visible:border-neutral-500 focus-visible:ring-0',\n },\n /**\n * Controls the size and padding of the button\n */\n size: {\n sm: 'h-8 px-3 text-sm',\n md: 'h-10 px-4 text-sm',\n lg: 'h-12 px-6 text-base',\n },\n },\n compoundVariants: [\n {\n variant: 'input',\n size: ['sm', 'md', 'lg'],\n className: typographyVariants({ variant: 'body-lg' }),\n },\n ],\n defaultVariants: {\n variant: 'primary',\n size: 'md',\n },\n },\n)\n\nexport type ButtonProps = {\n /**\n * The visual style of the button.\n * @default primary\n */\n variant?: 'primary' | 'secondary' | 'ghost' | 'inverse-ghost' | 'text' | 'error' | 'success'\n\n /**\n * The size of the button, affecting height, padding, and font size.\n * @default md\n */\n size?: 'sm' | 'md' | 'lg'\n\n /**\n * Icon component to display at the start of the button content.\n * Use Phosphor icons for consistent styling.\n * @example <Button StartIcon={House}>Home</Button>\n */\n StartIcon?: Icon\n\n /**\n * Icon component to display at the end of the button content.\n * Use Phosphor icons for consistent styling.\n * @example <Button EndIcon={ArrowRight}>Next</Button>\n */\n EndIcon?: Icon\n\n /**\n * When true, displays a loading spinner and disables the button.\n * The spinner replaces the StartIcon if present.\n * @default false\n */\n loading?: boolean\n\n /**\n * When true, creates a circular button with only the StartIcon visible.\n * @default false\n */\n iconOnly?: boolean\n\n /**\n * The content to display inside the button.\n * Not displayed when iconOnly is true.\n */\n children?: React.ReactNode\n\n /**\n * Custom props to pass to the icons (StartIcon, EndIcon, SpinnerGap).\n * Allows overriding size, weight, color, etc.\n */\n iconProps?: Partial<PhosphorIconProps>\n} & React.ButtonHTMLAttributes<HTMLButtonElement> &\n VariantProps<typeof buttonVariants>\n\n/**\n * A versatile button component that supports multiple variants, sizes, and icon placements.\n *\n * @example\n * // Basic usage\n * <Button>Click me</Button>\n *\n * // With icon\n * <Button StartIcon={House}>Home</Button>\n *\n * // Icon only button\n * <Button StartIcon={Plus} iconOnly aria-label=\"Add item\" />\n *\n * // Loading state\n * <Button loading>Processing...</Button>\n */\nconst Button = React.forwardRef<HTMLButtonElement, ButtonProps>(\n (\n { className, variant = 'primary', size = 'md', StartIcon, EndIcon, loading, iconOnly, iconProps = {}, ...props },\n ref,\n ) => {\n const defaultIconProps = (): { size: number; weight: IconWeight } => {\n switch (size) {\n case 'sm':\n return { size: 16, weight: 'regular' }\n case 'lg':\n return { size: 20, weight: 'regular' }\n default:\n return { size: 18, weight: 'regular' }\n }\n }\n\n const mergedIconProps = { ...defaultIconProps(), ...iconProps }\n\n const getIconOnlyClasses = () => {\n switch (size) {\n case 'sm':\n return 'h-8 w-8'\n case 'lg':\n return 'h-12 w-12'\n default:\n return 'h-10 w-10'\n }\n }\n\n return (\n <button\n className={cn(\n buttonVariants({ variant, size, className }),\n iconOnly && 'rounded-full p-0',\n iconOnly && getIconOnlyClasses(),\n className,\n )}\n ref={ref}\n disabled={loading}\n {...props}\n >\n {loading && <SpinnerGap {...defaultIconProps()} className=\"animate-spin\" aria-label=\"Loading\" />}\n {StartIcon && !loading && <StartIcon {...mergedIconProps} />}\n {iconOnly ? <span className=\"sr-only\">{props.children}</span> : props.children}\n {EndIcon && !iconOnly && <EndIcon {...mergedIconProps} />}\n </button>\n )\n },\n)\nButton.displayName = 'Button'\n\nexport { Button, buttonVariants }\n"],"names":[],"mappings":"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"}
|
|
@@ -30,7 +30,6 @@ declare const CarouselPrevious: React.ForwardRefExoticComponent<Omit<{
|
|
|
30
30
|
loading?: boolean;
|
|
31
31
|
iconOnly?: boolean;
|
|
32
32
|
children?: React.ReactNode;
|
|
33
|
-
asChild?: boolean;
|
|
34
33
|
iconProps?: Partial<import('@phosphor-icons/react').IconProps>;
|
|
35
34
|
} & React.ButtonHTMLAttributes<HTMLButtonElement> & import('class-variance-authority').VariantProps<(props?: ({
|
|
36
35
|
variant?: "primary" | "secondary" | "success" | "error" | "text" | "input" | "ghost" | "inverse-ghost" | null | undefined;
|
|
@@ -44,7 +43,6 @@ declare const CarouselNext: React.ForwardRefExoticComponent<Omit<{
|
|
|
44
43
|
loading?: boolean;
|
|
45
44
|
iconOnly?: boolean;
|
|
46
45
|
children?: React.ReactNode;
|
|
47
|
-
asChild?: boolean;
|
|
48
46
|
iconProps?: Partial<import('@phosphor-icons/react').IconProps>;
|
|
49
47
|
} & React.ButtonHTMLAttributes<HTMLButtonElement> & import('class-variance-authority').VariantProps<(props?: ({
|
|
50
48
|
variant?: "primary" | "secondary" | "success" | "error" | "text" | "input" | "ghost" | "inverse-ghost" | null | undefined;
|
|
@@ -1,94 +1,6 @@
|
|
|
1
|
-
import {
|
|
2
|
-
import * as React from 'react';
|
|
3
|
-
type BasePaginationProps = {
|
|
4
|
-
/** Number of rows per page */
|
|
5
|
-
readonly pageSize: number;
|
|
6
|
-
/** Whether the pagination is in a loading state */
|
|
7
|
-
readonly isLoading?: boolean;
|
|
8
|
-
/** Text customization for pagination */
|
|
9
|
-
readonly labels?: {
|
|
10
|
-
/** Text shown before the page size number (default: "Showing") */
|
|
11
|
-
showing?: string;
|
|
12
|
-
/** Text shown before the total number (default: "of") */
|
|
13
|
-
of?: string;
|
|
14
|
-
/** Text shown after the total number (default: "results") */
|
|
15
|
-
results?: string;
|
|
16
|
-
/** Aria label for previous page button (default: "Previous page") */
|
|
17
|
-
previousPage?: string;
|
|
18
|
-
/** Aria label for next page button (default: "Next page") */
|
|
19
|
-
nextPage?: string;
|
|
20
|
-
/** Aria label for page number (default: "Page {number}") */
|
|
21
|
-
pageLabel?: string;
|
|
22
|
-
};
|
|
23
|
-
};
|
|
24
|
-
type BackendPaginationProps = BasePaginationProps & {
|
|
25
|
-
/** Current page */
|
|
26
|
-
readonly currentPage: number;
|
|
27
|
-
/** Total number of items */
|
|
28
|
-
readonly total: number;
|
|
29
|
-
/** Callback when page changes */
|
|
30
|
-
readonly onPageChange: (page: number) => void;
|
|
31
|
-
};
|
|
32
|
-
/**
|
|
33
|
-
* Type helper to check if a type has an 'id' property
|
|
34
|
-
*/
|
|
35
|
-
type HasId<T> = T extends {
|
|
36
|
-
id: string | number;
|
|
37
|
-
} ? true : false;
|
|
38
|
-
/**
|
|
39
|
-
* Props for the DataTable component
|
|
40
|
-
* @template TData The type of data being displayed in the table
|
|
41
|
-
*/
|
|
42
|
-
export type DataTableProps<TData> = {
|
|
43
|
-
/** Array of column definitions that describe the table structure */
|
|
44
|
-
readonly columns: Array<ColumnDef<TData>>;
|
|
45
|
-
/** Array of data items to be displayed in the table */
|
|
46
|
-
readonly data: Array<TData>;
|
|
47
|
-
/** Whether to show the column visibility toggle menu */
|
|
48
|
-
readonly showColumnVisibilityControls?: boolean;
|
|
49
|
-
/** Whether the table is in a loading state */
|
|
50
|
-
readonly isLoading?: boolean;
|
|
51
|
-
/** Pagination configuration. If not provided, pagination is disabled */
|
|
52
|
-
readonly pagination?: BasePaginationProps | BackendPaginationProps;
|
|
53
|
-
/** Primary filters that appear directly above the table */
|
|
54
|
-
readonly primaryFilters?: React.ReactNode;
|
|
55
|
-
/** Secondary filters that appear in the filters dropdown */
|
|
56
|
-
readonly secondaryFilters?: React.ReactNode;
|
|
57
|
-
/** Number of active primary filters */
|
|
58
|
-
readonly activePrimaryFiltersCount?: number;
|
|
59
|
-
/** Number of active secondary filters */
|
|
60
|
-
readonly activeSecondaryFiltersCount?: number;
|
|
61
|
-
/** Text customization for filters */
|
|
62
|
-
readonly labels?: {
|
|
63
|
-
/** Text for the column visibility button (default: "Hide columns") */
|
|
64
|
-
columnVisibilityButton?: string;
|
|
65
|
-
/** Text for the filters button when only secondary filters are present (default: "Filters") */
|
|
66
|
-
filters?: string;
|
|
67
|
-
/** Text for the more filters button when both primary and secondary filters are present (default: "More filters") */
|
|
68
|
-
moreFilters?: string;
|
|
69
|
-
};
|
|
70
|
-
/** Callback when all rows are selected */
|
|
71
|
-
readonly onSelectAll?: (selected: boolean) => void;
|
|
72
|
-
/** Callback when a row is selected */
|
|
73
|
-
readonly onSelect?: (selected: boolean, row: TData) => void;
|
|
74
|
-
/** Optional className for the table container */
|
|
75
|
-
readonly className?: string;
|
|
76
|
-
/** Optional className for the table */
|
|
77
|
-
readonly tableClassName?: string;
|
|
78
|
-
} & (HasId<TData> extends true ? {
|
|
79
|
-
/** Function to get unique identifier from a row. Not needed when data has 'id' property */
|
|
80
|
-
readonly getRowId?: never;
|
|
81
|
-
} : {
|
|
82
|
-
/** Function to get unique identifier from a row. Required when data doesn't have 'id' property */
|
|
83
|
-
readonly getRowId: RowIdentifierFn<TData>;
|
|
84
|
-
});
|
|
85
|
-
/**
|
|
86
|
-
* Function to get a unique identifier from a row
|
|
87
|
-
*/
|
|
88
|
-
type RowIdentifierFn<T> = (row: T) => string;
|
|
1
|
+
import { DataTableProps } from './types';
|
|
89
2
|
/**
|
|
90
3
|
* A feature-rich data table component built on top of TanStack Table.
|
|
91
4
|
* Provides sorting, filtering, pagination, and column visibility controls.
|
|
92
5
|
*/
|
|
93
|
-
export declare function DataTable<TData extends object>({ columns: userColumns, data, getRowId, showColumnVisibilityControls, isLoading, pagination, primaryFilters, secondaryFilters, activePrimaryFiltersCount, activeSecondaryFiltersCount,
|
|
94
|
-
export {};
|
|
6
|
+
export declare function DataTable<TData extends object>({ columns: userColumns, data, getRowId, showColumnVisibilityControls, isLoading, pagination, primaryFilters, secondaryFilters, activePrimaryFiltersCount, activeSecondaryFiltersCount, selection, labels, className, tableClassName, }: DataTableProps<TData>): import("react/jsx-runtime").JSX.Element;
|
|
@@ -1,14 +1,15 @@
|
|
|
1
1
|
"use client";
|
|
2
|
-
import {
|
|
3
|
-
import { useReactTable, getCoreRowModel,
|
|
2
|
+
import { jsx, jsxs } from 'react/jsx-runtime';
|
|
3
|
+
import { getPaginationRowModel, useReactTable, getCoreRowModel, flexRender } from '@tanstack/react-table';
|
|
4
4
|
import * as React from 'react';
|
|
5
|
-
import { Checkbox } from '../Checkbox/Checkbox.js';
|
|
6
5
|
import { Table, TableHeader, TableRow, TableHead } from '../Table/Table.js';
|
|
7
6
|
import { Typography } from '../Typography/Typography.js';
|
|
8
7
|
import { DataTableBody } from './components/DataTableBody.js';
|
|
8
|
+
import { DataTablePagination } from './components/DataTablePagination.js';
|
|
9
|
+
import { DataTableRowSelector } from './components/DataTableRowSelector.js';
|
|
10
|
+
import { DataTableSelectHeader } from './components/DataTableSelectHeader.js';
|
|
9
11
|
import { DataTableToolbar } from './components/DataTableToolbar.js';
|
|
10
|
-
import {
|
|
11
|
-
import { useTableSelection } from './hooks/useTableSelection.js';
|
|
12
|
+
import { useSelection } from './hooks/useSelection.js';
|
|
12
13
|
import { cn } from '../../lib/utils.js';
|
|
13
14
|
|
|
14
15
|
function DataTable({
|
|
@@ -22,51 +23,110 @@ function DataTable({
|
|
|
22
23
|
secondaryFilters,
|
|
23
24
|
activePrimaryFiltersCount = 0,
|
|
24
25
|
activeSecondaryFiltersCount = 0,
|
|
26
|
+
selection,
|
|
25
27
|
labels = {
|
|
26
28
|
columnVisibilityButton: "Hide columns",
|
|
27
29
|
filters: "Filters",
|
|
28
30
|
moreFilters: "More filters"
|
|
29
31
|
},
|
|
30
|
-
onSelectAll,
|
|
31
|
-
onSelect,
|
|
32
32
|
className,
|
|
33
33
|
tableClassName
|
|
34
34
|
}) {
|
|
35
35
|
const [columnVisibility, setColumnVisibility] = React.useState({});
|
|
36
36
|
const [pageIndex, setPageIndex] = React.useState(0);
|
|
37
|
-
const isBackendPagination = pagination && "onPageChange" in pagination;
|
|
37
|
+
const isBackendPagination = pagination !== undefined && "onPageChange" in pagination;
|
|
38
38
|
const total = isBackendPagination ? pagination.total : data.length;
|
|
39
|
-
const pageSize = pagination?.pageSize ?? data.length;
|
|
39
|
+
const pageSize = pagination?.pageSize ?? (isLoading ? 10 : data.length);
|
|
40
40
|
const totalPages = Math.ceil(total / pageSize);
|
|
41
|
-
const {
|
|
41
|
+
const { selectedIds, handleRowSelect, handleSelectAll } = useSelection({
|
|
42
42
|
data,
|
|
43
43
|
getRowId,
|
|
44
|
-
|
|
45
|
-
onSelect
|
|
44
|
+
selection
|
|
46
45
|
});
|
|
46
|
+
const rowSelection = React.useMemo(
|
|
47
|
+
() => Object.fromEntries(Array.from(selectedIds).map((id) => [id, true])),
|
|
48
|
+
[selectedIds]
|
|
49
|
+
);
|
|
50
|
+
const SelectColumnHeader = React.useCallback(
|
|
51
|
+
() => /* @__PURE__ */ jsx(
|
|
52
|
+
DataTableSelectHeader,
|
|
53
|
+
{
|
|
54
|
+
isBackendPagination,
|
|
55
|
+
selectedIds,
|
|
56
|
+
data,
|
|
57
|
+
handleSelectAll,
|
|
58
|
+
isLoading
|
|
59
|
+
}
|
|
60
|
+
),
|
|
61
|
+
[isBackendPagination, selectedIds, data, handleSelectAll, isLoading]
|
|
62
|
+
);
|
|
63
|
+
const SelectColumnCell = React.useCallback(
|
|
64
|
+
({ row }) => /* @__PURE__ */ jsx(DataTableRowSelector, { row, selectedIds, getRowId, handleRowSelect }),
|
|
65
|
+
[selectedIds, getRowId, handleRowSelect]
|
|
66
|
+
);
|
|
67
|
+
const tableColumns = React.useMemo(() => {
|
|
68
|
+
const baseColumns = [...userColumns];
|
|
69
|
+
if (selection !== undefined) {
|
|
70
|
+
const selectColumn = {
|
|
71
|
+
id: "select",
|
|
72
|
+
size: 50,
|
|
73
|
+
cell: SelectColumnCell,
|
|
74
|
+
header: SelectColumnHeader
|
|
75
|
+
};
|
|
76
|
+
baseColumns.unshift(selectColumn);
|
|
77
|
+
}
|
|
78
|
+
return baseColumns;
|
|
79
|
+
}, [userColumns, selection, SelectColumnCell, SelectColumnHeader]);
|
|
80
|
+
const toolbarSelectionProps = React.useMemo(() => {
|
|
81
|
+
if (isBackendPagination && selection) {
|
|
82
|
+
return {
|
|
83
|
+
isSelectionStarted: selection.isSelectionStarted,
|
|
84
|
+
onSelectionStart: selection.onSelectionStart,
|
|
85
|
+
onSelectionCancel: selection.onSelectionCancel,
|
|
86
|
+
selectionSummary: selection.labels?.selectionSummary,
|
|
87
|
+
labels: selection.labels
|
|
88
|
+
};
|
|
89
|
+
}
|
|
90
|
+
return undefined;
|
|
91
|
+
}, [isBackendPagination, selection]);
|
|
92
|
+
const currentPage = isBackendPagination ? pagination.currentPage : pageIndex + 1;
|
|
93
|
+
const handlePageChange = isBackendPagination ? pagination.onPageChange : (page) => setPageIndex(page - 1);
|
|
94
|
+
const tablePaginationState = React.useMemo(() => {
|
|
95
|
+
if (!pagination) {
|
|
96
|
+
return undefined;
|
|
97
|
+
}
|
|
98
|
+
return {
|
|
99
|
+
pageIndex: isBackendPagination ? pagination.currentPage - 1 : pageIndex,
|
|
100
|
+
pageSize
|
|
101
|
+
};
|
|
102
|
+
}, [pagination, isBackendPagination, pageIndex, pageSize]);
|
|
103
|
+
const tableGetPaginationRowModel = React.useMemo(() => {
|
|
104
|
+
return pagination && !isBackendPagination ? getPaginationRowModel() : undefined;
|
|
105
|
+
}, [pagination, isBackendPagination]);
|
|
106
|
+
const tableOnPaginationChange = React.useCallback(
|
|
107
|
+
(updater) => {
|
|
108
|
+
if (typeof updater === "function") {
|
|
109
|
+
const newState = updater({ pageIndex, pageSize });
|
|
110
|
+
setPageIndex(newState.pageIndex);
|
|
111
|
+
}
|
|
112
|
+
},
|
|
113
|
+
[pageIndex, pageSize, setPageIndex]
|
|
114
|
+
);
|
|
47
115
|
const table = useReactTable({
|
|
48
116
|
data,
|
|
49
|
-
columns:
|
|
117
|
+
columns: tableColumns,
|
|
50
118
|
getCoreRowModel: getCoreRowModel(),
|
|
51
|
-
getPaginationRowModel:
|
|
119
|
+
getPaginationRowModel: tableGetPaginationRowModel,
|
|
52
120
|
onColumnVisibilityChange: setColumnVisibility,
|
|
53
|
-
enableRowSelection:
|
|
121
|
+
enableRowSelection: !!selection,
|
|
54
122
|
getRowId,
|
|
55
123
|
state: {
|
|
56
124
|
columnVisibility,
|
|
57
125
|
rowSelection,
|
|
58
|
-
pagination:
|
|
59
|
-
pageIndex: isBackendPagination ? pagination.currentPage - 1 : pageIndex,
|
|
60
|
-
pageSize
|
|
61
|
-
} : undefined
|
|
126
|
+
pagination: tablePaginationState
|
|
62
127
|
},
|
|
63
128
|
manualPagination: isBackendPagination,
|
|
64
|
-
onPaginationChange: isBackendPagination ? undefined :
|
|
65
|
-
if (typeof updater === "function") {
|
|
66
|
-
const newState = updater({ pageIndex, pageSize });
|
|
67
|
-
setPageIndex(newState.pageIndex);
|
|
68
|
-
}
|
|
69
|
-
}
|
|
129
|
+
onPaginationChange: isBackendPagination ? undefined : tableOnPaginationChange
|
|
70
130
|
});
|
|
71
131
|
return /* @__PURE__ */ jsxs("div", { className: cn("flex h-full min-h-0 w-full flex-1 flex-col gap-2 overflow-hidden", className), children: [
|
|
72
132
|
/* @__PURE__ */ jsx(
|
|
@@ -78,41 +138,30 @@ function DataTable({
|
|
|
78
138
|
secondaryFilters,
|
|
79
139
|
activePrimaryFiltersCount,
|
|
80
140
|
activeSecondaryFiltersCount,
|
|
81
|
-
labels
|
|
141
|
+
labels,
|
|
142
|
+
selection: toolbarSelectionProps
|
|
82
143
|
}
|
|
83
144
|
),
|
|
84
145
|
/* @__PURE__ */ jsxs("div", { className: "flex min-h-0 flex-1 flex-col rounded-md border bg-white", children: [
|
|
85
|
-
/* @__PURE__ */ jsx("div", { className: "min-h-0 flex-1 overflow-auto", children: /* @__PURE__ */ jsx("div", { className: "h-full overflow-auto", children: /* @__PURE__ */ jsxs(Table, { className: "w-full", tableClassName: cn("table-fixed", tableClassName), children: [
|
|
86
|
-
/* @__PURE__ */ jsx(TableHeader, { className: "sticky top-0 z-10 bg-neutral-50", children: table.getHeaderGroups().map((headerGroup) => /* @__PURE__ */
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
) }),
|
|
96
|
-
headerGroup.headers.map((header) => /* @__PURE__ */ jsx(
|
|
97
|
-
TableHead,
|
|
98
|
-
{
|
|
99
|
-
className: "whitespace-normal",
|
|
100
|
-
style: { width: header.column.columnDef.size },
|
|
101
|
-
children: /* @__PURE__ */ jsx(Typography, { weight: "medium", children: header.isPlaceholder ? null : flexRender(header.column.columnDef.header, header.getContext()) })
|
|
102
|
-
},
|
|
103
|
-
header.id
|
|
104
|
-
))
|
|
105
|
-
] }, headerGroup.id)) }),
|
|
146
|
+
/* @__PURE__ */ jsx("div", { className: "min-h-0 flex-1 overflow-auto", children: /* @__PURE__ */ jsx("div", { className: "h-full overflow-auto", children: /* @__PURE__ */ jsxs(Table, { className: "w-full overscroll-contain", tableClassName: cn("table-fixed ", tableClassName), children: [
|
|
147
|
+
/* @__PURE__ */ jsx(TableHeader, { className: "sticky top-0 z-10 bg-neutral-50", children: table.getHeaderGroups().map((headerGroup) => /* @__PURE__ */ jsx(TableRow, { children: headerGroup.headers.map((header) => /* @__PURE__ */ jsx(
|
|
148
|
+
TableHead,
|
|
149
|
+
{
|
|
150
|
+
className: "whitespace-normal",
|
|
151
|
+
style: { width: header.column.columnDef.size },
|
|
152
|
+
children: /* @__PURE__ */ jsx(Typography, { weight: "medium", children: header.isPlaceholder ? null : flexRender(header.column.columnDef.header, header.getContext()) })
|
|
153
|
+
},
|
|
154
|
+
header.id
|
|
155
|
+
)) }, headerGroup.id)) }),
|
|
106
156
|
/* @__PURE__ */ jsx(
|
|
107
157
|
DataTableBody,
|
|
108
158
|
{
|
|
109
159
|
table,
|
|
110
160
|
isLoading,
|
|
111
|
-
isSelectable,
|
|
161
|
+
isSelectable: selection !== undefined,
|
|
112
162
|
pageSize,
|
|
113
163
|
getRowId,
|
|
114
|
-
rowSelection
|
|
115
|
-
handleRowSelect
|
|
164
|
+
rowSelection
|
|
116
165
|
}
|
|
117
166
|
)
|
|
118
167
|
] }) }) }),
|
|
@@ -122,9 +171,9 @@ function DataTable({
|
|
|
122
171
|
table,
|
|
123
172
|
total,
|
|
124
173
|
pageSize,
|
|
125
|
-
currentPage
|
|
174
|
+
currentPage,
|
|
126
175
|
totalPages,
|
|
127
|
-
onPageChange:
|
|
176
|
+
onPageChange: handlePageChange,
|
|
128
177
|
isLoading: pagination.isLoading,
|
|
129
178
|
labels: pagination.labels
|
|
130
179
|
}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"DataTable.js","sources":["../../../src/components/DataTable/DataTable.tsx"],"sourcesContent":["/* eslint-disable @typescript-eslint/prefer-nullish-coalescing */\n/* eslint-disable @typescript-eslint/no-dynamic-delete */\n'use client'\n\nimport {\n ColumnDef,\n flexRender,\n getCoreRowModel,\n getPaginationRowModel,\n useReactTable,\n VisibilityState,\n} from '@tanstack/react-table'\nimport * as React from 'react'\n\nimport { Checkbox } from '../Checkbox'\nimport { Table as TableComponent, TableHeader, TableHead, TableRow } from '../Table'\nimport { Typography } from '../Typography'\n\nimport { DataTableBody } from './components/DataTableBody'\nimport { DataTableToolbar } from './components/DataTableToolbar'\nimport { DataTablePagination } from './DataTablePagination'\nimport { useTableSelection } from './hooks/useTableSelection'\n\nimport { cn } from '@/lib/utils'\n\ntype BasePaginationProps = {\n /** Number of rows per page */\n readonly pageSize: number\n /** Whether the pagination is in a loading state */\n readonly isLoading?: boolean\n /** Text customization for pagination */\n readonly labels?: {\n /** Text shown before the page size number (default: \"Showing\") */\n showing?: string\n /** Text shown before the total number (default: \"of\") */\n of?: string\n /** Text shown after the total number (default: \"results\") */\n results?: string\n /** Aria label for previous page button (default: \"Previous page\") */\n previousPage?: string\n /** Aria label for next page button (default: \"Next page\") */\n nextPage?: string\n /** Aria label for page number (default: \"Page {number}\") */\n pageLabel?: string\n }\n}\n\ntype BackendPaginationProps = BasePaginationProps & {\n /** Current page */\n readonly currentPage: number\n /** Total number of items */\n readonly total: number\n /** Callback when page changes */\n readonly onPageChange: (page: number) => void\n}\n\n/**\n * Type helper to check if a type has an 'id' property\n */\ntype HasId<T> = T extends { id: string | number } ? true : false\n\n/**\n * Props for the DataTable component\n * @template TData The type of data being displayed in the table\n */\nexport type DataTableProps<TData> = {\n /** Array of column definitions that describe the table structure */\n readonly columns: Array<ColumnDef<TData>>\n /** Array of data items to be displayed in the table */\n readonly data: Array<TData>\n /** Whether to show the column visibility toggle menu */\n readonly showColumnVisibilityControls?: boolean\n /** Whether the table is in a loading state */\n readonly isLoading?: boolean\n /** Pagination configuration. If not provided, pagination is disabled */\n readonly pagination?: BasePaginationProps | BackendPaginationProps\n /** Primary filters that appear directly above the table */\n readonly primaryFilters?: React.ReactNode\n /** Secondary filters that appear in the filters dropdown */\n readonly secondaryFilters?: React.ReactNode\n /** Number of active primary filters */\n readonly activePrimaryFiltersCount?: number\n /** Number of active secondary filters */\n readonly activeSecondaryFiltersCount?: number\n /** Text customization for filters */\n readonly labels?: {\n /** Text for the column visibility button (default: \"Hide columns\") */\n columnVisibilityButton?: string\n /** Text for the filters button when only secondary filters are present (default: \"Filters\") */\n filters?: string\n /** Text for the more filters button when both primary and secondary filters are present (default: \"More filters\") */\n moreFilters?: string\n }\n /** Callback when all rows are selected */\n readonly onSelectAll?: (selected: boolean) => void\n /** Callback when a row is selected */\n readonly onSelect?: (selected: boolean, row: TData) => void\n /** Optional className for the table container */\n readonly className?: string\n /** Optional className for the table */\n readonly tableClassName?: string\n} & (HasId<TData> extends true\n ? {\n /** Function to get unique identifier from a row. Not needed when data has 'id' property */\n readonly getRowId?: never\n }\n : {\n /** Function to get unique identifier from a row. Required when data doesn't have 'id' property */\n readonly getRowId: RowIdentifierFn<TData>\n })\n\n/**\n * Function to get a unique identifier from a row\n */\ntype RowIdentifierFn<T> = (row: T) => string\n\n/**\n * A feature-rich data table component built on top of TanStack Table.\n * Provides sorting, filtering, pagination, and column visibility controls.\n */\nexport function DataTable<TData extends object>({\n columns: userColumns,\n data,\n getRowId = (row: TData) => (row as { id: string }).id,\n showColumnVisibilityControls = true,\n isLoading = false,\n pagination,\n primaryFilters,\n secondaryFilters,\n activePrimaryFiltersCount = 0,\n activeSecondaryFiltersCount = 0,\n labels = {\n columnVisibilityButton: 'Hide columns',\n filters: 'Filters',\n moreFilters: 'More filters',\n },\n onSelectAll,\n onSelect,\n className,\n tableClassName,\n}: DataTableProps<TData>) {\n const [columnVisibility, setColumnVisibility] = React.useState<VisibilityState>({})\n const [pageIndex, setPageIndex] = React.useState(0)\n\n const isBackendPagination = pagination && 'onPageChange' in pagination\n const total = isBackendPagination ? pagination.total : data.length\n const pageSize = pagination?.pageSize ?? data.length\n const totalPages = Math.ceil(total / pageSize)\n\n const { isSelectable, isAllRowsSelected, rowSelection, handleSelectAll, handleRowSelect } = useTableSelection({\n data,\n getRowId,\n onSelectAll,\n onSelect,\n })\n\n const table = useReactTable({\n data,\n columns: userColumns,\n getCoreRowModel: getCoreRowModel(),\n getPaginationRowModel: pagination && !isBackendPagination ? getPaginationRowModel() : undefined,\n onColumnVisibilityChange: setColumnVisibility,\n enableRowSelection: isSelectable,\n getRowId: getRowId,\n state: {\n columnVisibility,\n rowSelection,\n pagination: pagination\n ? {\n pageIndex: isBackendPagination ? pagination.currentPage - 1 : pageIndex,\n pageSize,\n }\n : undefined,\n },\n manualPagination: isBackendPagination,\n onPaginationChange: isBackendPagination\n ? undefined\n : (updater) => {\n if (typeof updater === 'function') {\n const newState = updater({ pageIndex, pageSize })\n setPageIndex(newState.pageIndex)\n }\n },\n })\n\n return (\n <div className={cn('flex h-full min-h-0 w-full flex-1 flex-col gap-2 overflow-hidden', className)}>\n <DataTableToolbar\n table={table}\n showColumnVisibilityControls={showColumnVisibilityControls}\n primaryFilters={primaryFilters}\n secondaryFilters={secondaryFilters}\n activePrimaryFiltersCount={activePrimaryFiltersCount}\n activeSecondaryFiltersCount={activeSecondaryFiltersCount}\n labels={labels}\n />\n\n <div className=\"flex min-h-0 flex-1 flex-col rounded-md border bg-white\">\n <div className=\"min-h-0 flex-1 overflow-auto\">\n <div className=\"h-full overflow-auto\">\n <TableComponent className=\"w-full\" tableClassName={cn('table-fixed', tableClassName)}>\n <TableHeader className=\"sticky top-0 z-10 bg-neutral-50\">\n {table.getHeaderGroups().map((headerGroup) => (\n <TableRow key={headerGroup.id}>\n {isSelectable && (\n <TableHead className=\"w-[50px]\">\n <Checkbox\n checked={isAllRowsSelected}\n onCheckedChange={handleSelectAll}\n disabled={isLoading || !data.length}\n aria-label=\"Select all rows\"\n />\n </TableHead>\n )}\n {headerGroup.headers.map((header) => (\n <TableHead\n key={header.id}\n className=\"whitespace-normal\"\n style={{ width: header.column.columnDef.size }}\n >\n <Typography weight=\"medium\">\n {header.isPlaceholder\n ? null\n : flexRender(header.column.columnDef.header, header.getContext())}\n </Typography>\n </TableHead>\n ))}\n </TableRow>\n ))}\n </TableHeader>\n <DataTableBody\n table={table}\n isLoading={isLoading}\n isSelectable={isSelectable}\n pageSize={pageSize}\n getRowId={getRowId}\n rowSelection={rowSelection}\n handleRowSelect={handleRowSelect}\n />\n </TableComponent>\n </div>\n </div>\n {!!pagination && (\n <div className=\"border-t px-4 py-2\">\n <DataTablePagination\n table={table}\n total={total}\n pageSize={pageSize}\n currentPage={\n isBackendPagination ? pagination.currentPage : table.getState().pagination.pageIndex + 1 || 1\n }\n totalPages={totalPages}\n onPageChange={isBackendPagination ? pagination.onPageChange : undefined}\n isLoading={pagination.isLoading}\n labels={pagination.labels}\n />\n </div>\n )}\n </div>\n </div>\n )\n}\n"],"names":[],"mappings":";;;;;;;;;;;;;AAwHO;AAAyC;AACrC;AACT;AACmD;AACpB;AACnB;AACZ;AACA;AACA;AAC4B;AACE;AACrB;AACiB;AACf;AACI;AACf;AACA;AACA;AACA;AAEF;AACE;AACA;AAEA;AACA;AACA;AACA;AAEA;AAA8G;AAC5G;AACA;AACA;AACA;AAGF;AAA4B;AAC1B;AACS;AACwB;AACqD;AAC5D;AACN;AACpB;AACO;AACL;AACA;AAEI;AACgE;AAC9D;AAEF;AACN;AACkB;AAIZ;AACE;AACA;AAA+B;AACjC;AACF;AAGN;AAEI;AAAA;AAAC;AAAA;AACC;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACF;AAGE;AAGM;AAGO;AAEG;AAAC;AAAA;AACU;AACQ;AACY;AAClB;AAAA;AAEf;AAGA;AAAC;AAAA;AAEW;AACmC;AAM7C;AAAA;AARY;AAUf;AAGP;AACA;AAAC;AAAA;AACC;AACA;AACA;AACA;AACA;AACA;AACA;AAAA;AACF;AAGN;AAGI;AAAC;AAAA;AACC;AACA;AACA;AAE8F;AAE9F;AAC8D;AACxC;AACH;AAAA;AAEvB;AAEJ;AAGN;;"}
|
|
1
|
+
{"version":3,"file":"DataTable.js","sources":["../../../src/components/DataTable/DataTable.tsx"],"sourcesContent":["/* eslint-disable @typescript-eslint/prefer-nullish-coalescing */\n/* eslint-disable @typescript-eslint/no-dynamic-delete */\n'use client'\n\nimport {\n flexRender,\n getCoreRowModel,\n getPaginationRowModel,\n useReactTable,\n VisibilityState,\n Row,\n PaginationState,\n Updater,\n} from '@tanstack/react-table'\nimport * as React from 'react'\n\nimport { Table as TableComponent, TableHeader, TableHead, TableRow } from '../Table'\nimport { Typography } from '../Typography'\n\nimport { DataTableBody } from './components/DataTableBody'\nimport { DataTablePagination } from './components/DataTablePagination'\nimport { DataTableRowSelector } from './components/DataTableRowSelector'\nimport { DataTableSelectHeader } from './components/DataTableSelectHeader'\nimport { DataTableToolbar } from './components/DataTableToolbar'\nimport { useSelection } from './hooks/useSelection'\nimport { DataTableProps } from './types'\n\nimport { cn } from '@/lib/utils'\n\n/**\n * A feature-rich data table component built on top of TanStack Table.\n * Provides sorting, filtering, pagination, and column visibility controls.\n */\nexport function DataTable<TData extends object>({\n columns: userColumns,\n data,\n getRowId = (row: TData) => (row as { id: string }).id,\n showColumnVisibilityControls = true,\n isLoading = false,\n pagination,\n primaryFilters,\n secondaryFilters,\n activePrimaryFiltersCount = 0,\n activeSecondaryFiltersCount = 0,\n selection,\n labels = {\n columnVisibilityButton: 'Hide columns',\n filters: 'Filters',\n moreFilters: 'More filters',\n },\n className,\n tableClassName,\n}: DataTableProps<TData>) {\n const [columnVisibility, setColumnVisibility] = React.useState<VisibilityState>({})\n const [pageIndex, setPageIndex] = React.useState(0)\n\n const isBackendPagination = pagination !== undefined && 'onPageChange' in pagination\n\n const total = isBackendPagination ? pagination.total : data.length\n const pageSize = pagination?.pageSize ?? (isLoading ? 10 : data.length)\n const totalPages = Math.ceil(total / pageSize)\n\n const { selectedIds, handleRowSelect, handleSelectAll } = useSelection({\n data,\n getRowId,\n selection,\n })\n\n const rowSelection = React.useMemo(\n () => Object.fromEntries(Array.from(selectedIds).map((id) => [id, true])),\n [selectedIds],\n )\n\n const SelectColumnHeader = React.useCallback(\n () => (\n <DataTableSelectHeader\n isBackendPagination={isBackendPagination}\n selectedIds={selectedIds}\n data={data}\n handleSelectAll={handleSelectAll}\n isLoading={isLoading}\n />\n ),\n [isBackendPagination, selectedIds, data, handleSelectAll, isLoading],\n )\n\n const SelectColumnCell = React.useCallback(\n ({ row }: { row: Row<TData> }) => (\n <DataTableRowSelector row={row} selectedIds={selectedIds} getRowId={getRowId} handleRowSelect={handleRowSelect} />\n ),\n [selectedIds, getRowId, handleRowSelect],\n )\n\n const tableColumns = React.useMemo(() => {\n const baseColumns = [...userColumns]\n if (selection !== undefined) {\n const selectColumn = {\n id: 'select',\n size: 50,\n cell: SelectColumnCell,\n header: SelectColumnHeader,\n }\n baseColumns.unshift(selectColumn)\n }\n return baseColumns\n }, [userColumns, selection, SelectColumnCell, SelectColumnHeader])\n\n const toolbarSelectionProps = React.useMemo(() => {\n if (isBackendPagination && selection) {\n return {\n isSelectionStarted: selection.isSelectionStarted,\n onSelectionStart: selection.onSelectionStart,\n onSelectionCancel: selection.onSelectionCancel,\n selectionSummary: selection.labels?.selectionSummary,\n labels: selection.labels,\n }\n }\n return undefined\n }, [isBackendPagination, selection])\n\n const currentPage = isBackendPagination ? pagination.currentPage : pageIndex + 1\n const handlePageChange = isBackendPagination ? pagination.onPageChange : (page: number) => setPageIndex(page - 1)\n\n const tablePaginationState = React.useMemo(() => {\n if (!pagination) {\n return undefined\n }\n return {\n pageIndex: isBackendPagination ? pagination.currentPage - 1 : pageIndex,\n pageSize,\n }\n }, [pagination, isBackendPagination, pageIndex, pageSize])\n\n const tableGetPaginationRowModel = React.useMemo(() => {\n return pagination && !isBackendPagination ? getPaginationRowModel() : undefined\n }, [pagination, isBackendPagination])\n\n const tableOnPaginationChange = React.useCallback(\n (updater: Updater<PaginationState>) => {\n if (typeof updater === 'function') {\n const newState = updater({ pageIndex, pageSize })\n setPageIndex(newState.pageIndex)\n }\n },\n [pageIndex, pageSize, setPageIndex],\n )\n\n const table = useReactTable({\n data,\n columns: tableColumns,\n getCoreRowModel: getCoreRowModel(),\n getPaginationRowModel: tableGetPaginationRowModel,\n onColumnVisibilityChange: setColumnVisibility,\n enableRowSelection: !!selection,\n getRowId: getRowId,\n state: {\n columnVisibility,\n rowSelection,\n pagination: tablePaginationState,\n },\n manualPagination: isBackendPagination,\n onPaginationChange: isBackendPagination ? undefined : tableOnPaginationChange,\n })\n\n return (\n <div className={cn('flex h-full min-h-0 w-full flex-1 flex-col gap-2 overflow-hidden', className)}>\n <DataTableToolbar\n table={table}\n showColumnVisibilityControls={showColumnVisibilityControls}\n primaryFilters={primaryFilters}\n secondaryFilters={secondaryFilters}\n activePrimaryFiltersCount={activePrimaryFiltersCount}\n activeSecondaryFiltersCount={activeSecondaryFiltersCount}\n labels={labels}\n selection={toolbarSelectionProps}\n />\n\n <div className=\"flex min-h-0 flex-1 flex-col rounded-md border bg-white\">\n <div className=\"min-h-0 flex-1 overflow-auto\">\n <div className=\"h-full overflow-auto\">\n <TableComponent className=\"w-full overscroll-contain\" tableClassName={cn('table-fixed ', tableClassName)}>\n <TableHeader className=\"sticky top-0 z-10 bg-neutral-50\">\n {table.getHeaderGroups().map((headerGroup) => (\n <TableRow key={headerGroup.id}>\n {headerGroup.headers.map((header) => (\n <TableHead\n key={header.id}\n className=\"whitespace-normal\"\n style={{ width: header.column.columnDef.size }}\n >\n <Typography weight=\"medium\">\n {header.isPlaceholder\n ? null\n : flexRender(header.column.columnDef.header, header.getContext())}\n </Typography>\n </TableHead>\n ))}\n </TableRow>\n ))}\n </TableHeader>\n <DataTableBody\n table={table}\n isLoading={isLoading}\n isSelectable={selection !== undefined}\n pageSize={pageSize}\n getRowId={getRowId}\n rowSelection={rowSelection}\n />\n </TableComponent>\n </div>\n </div>\n {!!pagination && (\n <div className=\"border-t px-4 py-2\">\n <DataTablePagination\n table={table}\n total={total}\n pageSize={pageSize}\n currentPage={currentPage}\n totalPages={totalPages}\n onPageChange={handlePageChange}\n isLoading={pagination.isLoading}\n labels={pagination.labels}\n />\n </div>\n )}\n </div>\n </div>\n )\n}\n"],"names":[],"mappings":";;;;;;;;;;;;;;AAiCO;AAAyC;AACrC;AACT;AACmD;AACpB;AACnB;AACZ;AACA;AACA;AAC4B;AACE;AAC9B;AACS;AACiB;AACf;AACI;AACf;AACA;AAEF;AACE;AACA;AAEA;AAEA;AACA;AACA;AAEA;AAAuE;AACrE;AACA;AACA;AAGF;AAA2B;AAC+C;AAC5D;AAGd;AAAiC;AAE7B;AAAC;AAAA;AACC;AACA;AACA;AACA;AACA;AAAA;AACF;AAEiE;AAGrE;AAA+B;AAEqF;AAE3E;AAGzC;AACE;AACA;AACE;AAAqB;AACf;AACE;AACA;AACE;AAEV;AAAgC;AAElC;AAAO;AAGT;AACE;AACE;AAAO;AACyB;AACF;AACC;AACO;AAClB;AACpB;AAEF;AAAO;AAGT;AACA;AAEA;AACE;AACE;AAAO;AAET;AAAO;AACyD;AAC9D;AACF;AAGF;AACE;AAAsE;AAGxE;AAAsC;AAElC;AACE;AACA;AAA+B;AACjC;AACF;AACkC;AAGpC;AAA4B;AAC1B;AACS;AACwB;AACV;AACG;AACJ;AACtB;AACO;AACL;AACA;AACY;AACd;AACkB;AACoC;AAGxD;AAEI;AAAA;AAAC;AAAA;AACC;AACA;AACA;AACA;AACA;AACA;AACA;AACW;AAAA;AACb;AAGE;AAGM;AAIQ;AAAC;AAAA;AAEW;AACmC;AAM7C;AAAA;AARY;AAatB;AACA;AAAC;AAAA;AACC;AACA;AAC4B;AAC5B;AACA;AACA;AAAA;AACF;AAGN;AAGI;AAAC;AAAA;AACC;AACA;AACA;AACA;AACA;AACc;AACQ;AACH;AAAA;AAEvB;AAEJ;AAGN;;"}
|
|
@@ -1,12 +1,12 @@
|
|
|
1
1
|
import { Table } from '@tanstack/react-table';
|
|
2
|
+
import { RowIdentifierFn } from '../types';
|
|
2
3
|
type DataTableBodyProps<TData> = {
|
|
3
4
|
readonly table: Table<TData>;
|
|
4
|
-
readonly isLoading
|
|
5
|
-
readonly isSelectable
|
|
5
|
+
readonly isLoading?: boolean;
|
|
6
|
+
readonly isSelectable?: boolean;
|
|
6
7
|
readonly pageSize: number;
|
|
7
|
-
readonly getRowId:
|
|
8
|
+
readonly getRowId: RowIdentifierFn<TData>;
|
|
8
9
|
readonly rowSelection: Record<string, boolean>;
|
|
9
|
-
readonly handleRowSelect: (checked: boolean, rowData: TData) => void;
|
|
10
10
|
};
|
|
11
|
-
export declare function DataTableBody<TData>({ table, isLoading, isSelectable, pageSize, getRowId, rowSelection,
|
|
11
|
+
export declare function DataTableBody<TData>({ table, isLoading, isSelectable, pageSize, getRowId, rowSelection, }: DataTableBodyProps<TData>): import("react/jsx-runtime").JSX.Element;
|
|
12
12
|
export {};
|
|
@@ -1,34 +1,36 @@
|
|
|
1
1
|
import { jsx, jsxs } from 'react/jsx-runtime';
|
|
2
2
|
import { flexRender } from '@tanstack/react-table';
|
|
3
|
-
import
|
|
3
|
+
import * as React from 'react';
|
|
4
4
|
import { Skeleton } from '../../Skeleton/Skeleton.js';
|
|
5
|
-
import {
|
|
5
|
+
import { TableRow, TableCell, TableBody } from '../../Table/Table.js';
|
|
6
6
|
import { Typography } from '../../Typography/Typography.js';
|
|
7
7
|
import { cn } from '../../../lib/utils.js';
|
|
8
8
|
|
|
9
|
+
const MemoizedTableRow = React.memo(
|
|
10
|
+
function MemoizedTableRow2({
|
|
11
|
+
row,
|
|
12
|
+
isSelected
|
|
13
|
+
}) {
|
|
14
|
+
return /* @__PURE__ */ jsx(TableRow, { "data-selected": isSelected, children: row.getVisibleCells().map((cell) => /* @__PURE__ */ jsx(TableCell, { style: { width: cell.column.columnDef.size }, children: /* @__PURE__ */ jsx(Typography, { children: cell.column.columnDef.cell && flexRender(cell.column.columnDef.cell, cell.getContext()) }) }, cell.id)) });
|
|
15
|
+
},
|
|
16
|
+
(prevProps, nextProps) => prevProps.row === nextProps.row && prevProps.isSelected === nextProps.isSelected
|
|
17
|
+
);
|
|
9
18
|
function DataTableBody({
|
|
10
19
|
table,
|
|
11
20
|
isLoading,
|
|
12
21
|
isSelectable,
|
|
13
22
|
pageSize,
|
|
14
23
|
getRowId,
|
|
15
|
-
rowSelection
|
|
16
|
-
handleRowSelect
|
|
24
|
+
rowSelection
|
|
17
25
|
}) {
|
|
18
26
|
if (isLoading) {
|
|
19
|
-
return /* @__PURE__ */ jsx(TableBody, { children: Array.from({ length: pageSize }).map((_unused,
|
|
20
|
-
isSelectable && /* @__PURE__ */ jsx(TableCell, { className: "w-[50px]", children: /* @__PURE__ */ jsx(
|
|
21
|
-
table.getAllColumns().filter((column) => column.getIsVisible()).map((column) => /* @__PURE__ */ jsx(
|
|
22
|
-
|
|
23
|
-
{
|
|
24
|
-
style: { width: column.columnDef.size },
|
|
25
|
-
children: /* @__PURE__ */ jsx("div", { className: cn("flex items-center justify-center", column.id === "actions" && "justify-end"), children: /* @__PURE__ */ jsx(Skeleton, { className: cn(column.id === "actions" ? "h-8 w-8" : "h-[20px] w-full") }) })
|
|
26
|
-
},
|
|
27
|
-
`skeleton-cell-${rowIndex.toString()}-${column.id}`
|
|
28
|
-
))
|
|
29
|
-
] }, `skeleton-row-${rowIndex.toString()}`)) });
|
|
27
|
+
return /* @__PURE__ */ jsx(TableBody, { children: Array.from({ length: pageSize }).map((_unused, index) => /* @__PURE__ */ jsxs(TableRow, { children: [
|
|
28
|
+
isSelectable && /* @__PURE__ */ jsx(TableCell, { className: "w-[50px]", children: /* @__PURE__ */ jsx(Skeleton, { className: "h-5 w-5" }) }),
|
|
29
|
+
table.getAllColumns().filter((column) => column.getIsVisible()).map((column) => /* @__PURE__ */ jsx(TableCell, { style: { width: column.columnDef.size }, children: /* @__PURE__ */ jsx("div", { className: cn("flex items-center justify-center", column.id === "actions" && "justify-end"), children: /* @__PURE__ */ jsx(Skeleton, { className: cn(column.id === "actions" ? "h-8 w-8" : "h-[20px] w-full") }) }) }, `skeleton-cell-${String(index)}-${column.id}`))
|
|
30
|
+
] }, `skeleton-row-${String(index)}`)) });
|
|
30
31
|
}
|
|
31
|
-
|
|
32
|
+
const rows = table.getRowModel().rows;
|
|
33
|
+
if (!rows.length) {
|
|
32
34
|
return /* @__PURE__ */ jsx(TableBody, { children: /* @__PURE__ */ jsx(TableRow, { children: /* @__PURE__ */ jsx(
|
|
33
35
|
TableCell,
|
|
34
36
|
{
|
|
@@ -38,23 +40,10 @@ function DataTableBody({
|
|
|
38
40
|
}
|
|
39
41
|
) }) });
|
|
40
42
|
}
|
|
41
|
-
return /* @__PURE__ */ jsx(TableBody, { children:
|
|
43
|
+
return /* @__PURE__ */ jsx(TableBody, { children: rows.map((row) => {
|
|
42
44
|
const rowId = getRowId(row.original);
|
|
43
45
|
const isSelected = rowSelection[rowId] ?? false;
|
|
44
|
-
return /* @__PURE__ */
|
|
45
|
-
isSelectable && /* @__PURE__ */ jsx(TableCell, { className: "w-[50px]", children: /* @__PURE__ */ jsx(
|
|
46
|
-
Checkbox,
|
|
47
|
-
{
|
|
48
|
-
checked: isSelected,
|
|
49
|
-
onCheckedChange: (checked) => {
|
|
50
|
-
const isChecked = checked;
|
|
51
|
-
handleRowSelect(isChecked, row.original);
|
|
52
|
-
},
|
|
53
|
-
"aria-label": `Select row ${rowId}`
|
|
54
|
-
}
|
|
55
|
-
) }),
|
|
56
|
-
row.getVisibleCells().map((cell) => /* @__PURE__ */ jsx(TableCell, { style: { width: cell.column.columnDef.size }, children: /* @__PURE__ */ jsx(Typography, { children: cell.column.columnDef.cell && flexRender(cell.column.columnDef.cell, cell.getContext()) }) }, cell.id))
|
|
57
|
-
] }, rowId);
|
|
46
|
+
return /* @__PURE__ */ jsx(MemoizedTableRow, { row, isSelected }, rowId);
|
|
58
47
|
}) });
|
|
59
48
|
}
|
|
60
49
|
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"DataTableBody.js","sources":["../../../../src/components/DataTable/components/DataTableBody.tsx"],"sourcesContent":["import { Table, flexRender } from '@tanstack/react-table'\
|
|
1
|
+
{"version":3,"file":"DataTableBody.js","sources":["../../../../src/components/DataTable/components/DataTableBody.tsx"],"sourcesContent":["import { Table, flexRender, Row } from '@tanstack/react-table'\nimport * as React from 'react'\n\nimport { Skeleton } from '../../Skeleton'\nimport { TableBody, TableCell, TableRow } from '../../Table'\nimport { Typography } from '../../Typography'\nimport { RowIdentifierFn } from '../types'\n\nimport { cn } from '@/lib/utils'\n\ntype DataTableBodyProps<TData> = {\n readonly table: Table<TData>\n readonly isLoading?: boolean\n readonly isSelectable?: boolean\n readonly pageSize: number\n readonly getRowId: RowIdentifierFn<TData>\n readonly rowSelection: Record<string, boolean>\n}\n\ntype MemoizedTableRowProps<TData> = {\n row: Row<TData>\n isSelectable: boolean\n isSelected: boolean\n getRowId: (row: TData) => string\n handleRowSelect: (checked: boolean, row: TData) => void\n}\n\nconst MemoizedTableRow = React.memo(\n function MemoizedTableRow<TData>({\n row,\n isSelected,\n }: Omit<MemoizedTableRowProps<TData>, 'isSelectable' | 'getRowId' | 'handleRowSelect'>) {\n return (\n <TableRow data-selected={isSelected}>\n {row.getVisibleCells().map((cell) => (\n <TableCell key={cell.id} style={{ width: cell.column.columnDef.size }}>\n <Typography>\n {cell.column.columnDef.cell && flexRender(cell.column.columnDef.cell, cell.getContext())}\n </Typography>\n </TableCell>\n ))}\n </TableRow>\n )\n },\n (prevProps, nextProps) => prevProps.row === nextProps.row && prevProps.isSelected === nextProps.isSelected,\n) as <TData>(\n props: Omit<MemoizedTableRowProps<TData>, 'isSelectable' | 'getRowId' | 'handleRowSelect'>,\n) => React.ReactElement\n\nexport function DataTableBody<TData>({\n table,\n isLoading,\n isSelectable,\n pageSize,\n getRowId,\n rowSelection,\n}: DataTableBodyProps<TData>) {\n if (isLoading) {\n return (\n <TableBody>\n {Array.from({ length: pageSize }).map((_unused, index) => (\n <TableRow key={`skeleton-row-${String(index)}`}>\n {isSelectable && (\n <TableCell className=\"w-[50px]\">\n <Skeleton className=\"h-5 w-5\" />\n </TableCell>\n )}\n {table\n .getAllColumns()\n .filter((column) => column.getIsVisible())\n .map((column) => (\n <TableCell key={`skeleton-cell-${String(index)}-${column.id}`} style={{ width: column.columnDef.size }}>\n <div className={cn('flex items-center justify-center', column.id === 'actions' && 'justify-end')}>\n <Skeleton className={cn(column.id === 'actions' ? 'h-8 w-8' : 'h-[20px] w-full')} />\n </div>\n </TableCell>\n ))}\n </TableRow>\n ))}\n </TableBody>\n )\n }\n\n const rows = table.getRowModel().rows\n\n if (!rows.length) {\n return (\n <TableBody>\n <TableRow>\n <TableCell\n colSpan={isSelectable ? table.getAllColumns().length + 1 : table.getAllColumns().length}\n className=\"h-[200px] text-center\"\n >\n <Typography color=\"neutral\">No data available</Typography>\n </TableCell>\n </TableRow>\n </TableBody>\n )\n }\n\n return (\n <TableBody>\n {rows.map((row) => {\n const rowId = getRowId(row.original)\n const isSelected = rowSelection[rowId] ?? false\n return <MemoizedTableRow key={rowId} row={row} isSelected={isSelected} />\n })}\n </TableBody>\n )\n}\n"],"names":["MemoizedTableRow"],"mappings":"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"}
|