diff --git a/.changeset/perfect-ties-taste.md b/.changeset/perfect-ties-taste.md new file mode 100644 index 00000000..28282483 --- /dev/null +++ b/.changeset/perfect-ties-taste.md @@ -0,0 +1,5 @@ +--- +"@4design/for-ui": patch +--- + +feat(Table): ローディングを追加 diff --git a/packages/for-ui/src/table/ColumnDef.ts b/packages/for-ui/src/table/ColumnDef.ts index c4a6b5c7..ad9fa4b4 100644 --- a/packages/for-ui/src/table/ColumnDef.ts +++ b/packages/for-ui/src/table/ColumnDef.ts @@ -1,3 +1 @@ -import type { ColumnDef } from '@tanstack/react-table'; - -export { ColumnDef }; +export type { ColumnDef } from '@tanstack/react-table'; diff --git a/packages/for-ui/src/table/Table.stories.tsx b/packages/for-ui/src/table/Table.stories.tsx index 939bc61e..9058dce3 100644 --- a/packages/for-ui/src/table/Table.stories.tsx +++ b/packages/for-ui/src/table/Table.stories.tsx @@ -1,4 +1,4 @@ -import React, { useState } from 'react'; +import React, { ReactNode, useState } from 'react'; import { MdMoreVert, MdOutlineDelete, MdOutlineEdit } from 'react-icons/md'; import { Meta, Story } from '@storybook/react/types-6-0'; import { Badge } from '../badge'; @@ -18,11 +18,13 @@ export default { component: Table, } as Meta; -// eslint-disable-next-line @typescript-eslint/no-explicit-any -const columns: ColumnDef[] = [ +const columns: ColumnDef[] = [ { header: 'ID', accessorKey: 'id', + meta: { + width: '16px', + }, cell: (cell) => {cell.renderValue()}, }, { @@ -44,6 +46,18 @@ const columns: ColumnDef[] = [ export const Base: Story = () => columns={columns} data={StaticPersonData} />; +export const Loading: Story = () => loading loadingRows={20} columns={columns} />; + +export const LoadingWithSelect: Story = () => ( + + loading + loadingRows={10} + columns={columns} + getRowId={(row) => row.id.toString()} + onSelectRow={(row) => console.info('Selected row: ', row)} + /> +); + export const WithSelect: Story = () => ( columns={columns} diff --git a/packages/for-ui/src/table/Table.tsx b/packages/for-ui/src/table/Table.tsx index 5dd9cabe..1789bd70 100644 --- a/packages/for-ui/src/table/Table.tsx +++ b/packages/for-ui/src/table/Table.tsx @@ -11,7 +11,6 @@ import { useState, } from 'react'; import { - ColumnDef, ColumnSort, flexRender, getCoreRowModel, @@ -28,12 +27,14 @@ import { } from '@tanstack/react-table'; import { Checkbox } from '../checkbox'; import { Radio } from '../radio'; +import { Skeleton } from '../skeleton'; import { fsx } from '../system/fsx'; import { Text } from '../text'; +import { ColumnDef } from './ColumnDef'; import { SortableTableCellHead, TableCell } from './TableCell'; import { TablePagination } from './TablePagination'; -export type TableProps = Pick, 'data' | 'columns' | 'getRowId'> & { +export type TableProps = Pick, 'columns' | 'getRowId'> & { disablePagination?: boolean | undefined; defaultSortColumn?: ColumnSort; /** onRowClick is called when each row is clicked regardless of the type of table (selectable or not) */ @@ -57,8 +58,126 @@ export type TableProps = Pick, 'data' | 'colu /** If wanting to use selectable table, specify _onSelectRow_ or _onSelectRows_ exclusively */ onSelectRows?: ((ids: string[]) => void) | undefined; } + ) & + ( + | { + /** + * 読み込み中であることを示す時に指定 + * + * @default false + */ + loading?: false | undefined; + + /** + * 読み込み中であることを示す時にスケルトンローディングで表示する行数を指定 + * + * @default 10 + */ + loadingRows?: never; + data: TableOptions['data']; + } + | { + /** + * 読み込み中であることを示す時に指定 + * + * @default false + */ + loading: true; + + /** + * 読み込み中であることを示す時にスケルトンローディングで表示する行数を指定 + * + * @default 10 + */ + loadingRows: number; + data?: never; + } ); +const getSelectColumn = ({ + id, + multiple, + loading, + onSelectRow, +}: { + id: string; + multiple?: boolean; + loading?: boolean; + onSelectRow?: (row: RowType) => void; +}): ColumnDef => { + return { + id, + meta: { + minWidth: '20px', + width: '20px', + maxWidth: '20px', + }, + header: ({ table }) => ( + + {multiple && ( + + すべての行を選択 + + } + disabled={loading} + className={fsx(`flex`)} + checked={table.getIsAllRowsSelected()} + indeterminate={!table.getIsAllRowsSelected() && table.getIsSomeRowsSelected()} + onChange={table.getToggleAllRowsSelectedHandler()} + /> + )} + + ), + cell: ({ row }) => ( + + {multiple ? ( + + 行を選択 + + } + disabled={loading} + className={fsx(`flex`)} + checked={row.getIsSelected()} + onClick={(e) => { + onSelectRow?.(row); + e.stopPropagation(); + }} + /> + ) : ( + + 行を選択 + + } + disabled={loading} + className={fsx(`flex`)} + checked={row.getIsSelected()} + onClick={(e) => { + onSelectRow?.(row); + e.stopPropagation(); + }} + /> + )} + + ), + }; +}; + +const makeColumnsLoading = (columns: ColumnDef[]) => + columns.map((column) => ({ + ...column, + cell: () => ( + + + + ), + })); + export const Table = ({ data, disablePagination, @@ -68,19 +187,23 @@ export const Table = ({ onRowClick, rowRenderer, getRowId, - columns, + columns: passedColumns, pageCount, pageSize = 20, className, page, defaultPage = 1, onChangePage, + loading, + loadingRows = 10, }: TableProps) => { const [sorting, setSorting] = useState(defaultSortColumn ? [defaultSortColumn] : []); const [rowSelection, setRowSelection] = useState({}); const prevRowSelection = useRef({}); const tableId = useId(); + const selectable = !!(onSelectRow || onSelectRows); + const onRowSelectionChange: OnChangeFn = useCallback( (updater) => { // updater is designed to be passed to setState like `setState((prev) => updater(prev))` @@ -110,91 +233,45 @@ export const Table = ({ const RowComponent: FC> = rowRenderer || Row; - const selectableColumns = useMemo(() => { - // Not selectable table - if (!(onSelectRow || onSelectRows)) { - return columns; - } + const loadingDummyData = Array(loadingRows).fill( + Object.fromEntries( + passedColumns.map((column) => [column.id || ('accessorKey' in column && column.accessorKey), '']), + ), + ); - const selectColumn: ColumnDef = { - // FIXME: use useId instead - id: 'select', - meta: { - minWidth: '20px', - width: '20px', - maxWidth: '20px', - }, - header: ({ table }) => ( - - {!!onSelectRows && ( - - すべての行を選択 - - } - className={fsx(`flex`)} - checked={table.getIsAllRowsSelected()} - indeterminate={!table.getIsAllRowsSelected() && table.getIsSomeRowsSelected()} - onChange={table.getToggleAllRowsSelectedHandler()} - /> - )} - - ), - cell: ({ row }) => ( - - {!!onSelectRows && ( - - 行を選択 - - } - className={fsx(`flex`)} - checked={row.getIsSelected()} - onClick={(e) => { - selectRow(row); - e.stopPropagation(); - }} - /> - )} - {!!onSelectRow && ( - - 行を選択 - - } - className={fsx(`flex`)} - checked={row.getIsSelected()} - onClick={(e) => { - selectRow(row); - e.stopPropagation(); - }} - /> - )} - - ), - }; - return [selectColumn, ...columns]; - }, [onSelectRow, onSelectRows, selectRow, columns]); + const columns = useMemo(() => { + if (!selectable && !loading) { + return passedColumns; + } + const cols = loading ? makeColumnsLoading(passedColumns) : passedColumns; + if (!selectable) { + return cols; + } + const selectColumn = getSelectColumn({ + id: `${tableId}-select`, + multiple: !!onSelectRows, + loading, + onSelectRow: selectRow, + }); + return [selectColumn, ...cols]; + }, [tableId, onSelectRows, loading, selectRow, passedColumns, selectable]); const table = useReactTable({ - data, - columns: selectableColumns, + data: loading ? loadingDummyData : data, + columns, pageCount: disablePagination ? undefined : pageCount, state: { sorting, rowSelection, }, getRowId, - onRowSelectionChange, + onRowSelectionChange: loading ? undefined : onRowSelectionChange, onSortingChange: setSorting, getCoreRowModel: getCoreRowModel(), getSortedRowModel: getSortedRowModel(), getFilteredRowModel: getFilteredRowModel(), getPaginationRowModel: !disablePagination ? getPaginationRowModel() : undefined, - enableRowSelection: !!(onSelectRow || onSelectRows), + enableRowSelection: selectable, enableMultiRowSelection: !!onSelectRows, }); @@ -204,7 +281,7 @@ export const Table = ({ return (
- + {table.getHeaderGroups().map((headerGroup) => ( @@ -212,6 +289,7 @@ export const Table = ({ ({ { @@ -246,6 +324,7 @@ export const Table = ({ {!disablePagination && (
(
@@ -303,6 +384,7 @@ export const TableRow = forwardRef = { row: RowType; selectable: boolean; + clickable?: boolean; onClick?: (e: MouseEvent, row: RowType) => void; className?: string; }; diff --git a/packages/for-ui/src/table/TableCell.tsx b/packages/for-ui/src/table/TableCell.tsx index 34879ba1..c386c223 100644 --- a/packages/for-ui/src/table/TableCell.tsx +++ b/packages/for-ui/src/table/TableCell.tsx @@ -44,9 +44,10 @@ export const SortableTableCellHead: FC< sorted: false | 'asc' | 'desc'; nextSortingOrder: false | 'asc' | 'desc'; children: ReactNode; + disabled?: boolean; onClick?: HTMLAttributes['onClick']; } -> = ({ sortable, sorted, nextSortingOrder, onClick, children, ...rest }) => ( +> = ({ sortable, sorted, nextSortingOrder, onClick, disabled, children, ...rest }) => ( {sortable ? (