Skip to content
This repository has been archived by the owner on Aug 31, 2022. It is now read-only.

Commit

Permalink
Merge pull request #107 from Synthetixio/feat/table
Browse files Browse the repository at this point in the history
feat: add table component
  • Loading branch information
Rickk137 authored Jul 12, 2022
2 parents db03d17 + 3ef3866 commit a70ef06
Show file tree
Hide file tree
Showing 8 changed files with 416 additions and 37 deletions.
3 changes: 3 additions & 0 deletions .vscode/settings.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
{
"typescript.tsdk": "node_modules/typescript/lib"
}
8 changes: 5 additions & 3 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@
"@synthetixio/transaction-notifier": "^2.70.1",
"clsx": "^1.1.1",
"lodash": "^4.17.21",
"react-table": "^7.8.0",
"react-transition-group": "^4.4.2"
},
"devDependencies": {
Expand All @@ -52,6 +53,7 @@
"@types/node": "^16.11.13",
"@types/react": "^18.0.0",
"@types/react-dom": "^18.0.0",
"@types/react-table": "^7.7.12",
"@types/react-transition-group": "^4.4.4",
"@typescript-eslint/eslint-plugin": "^5.7.0",
"@typescript-eslint/parser": "^5.7.0",
Expand All @@ -71,8 +73,8 @@
"lint-staged": "^12.1.2",
"postcss": "^8.4.5",
"prettier": "^2.5.1",
"react": "^18.0.0",
"react-dom": "^18.0.0",
"react": "^18.2.0",
"react-dom": "^18.2.0",
"react-scripts": "5.0.0",
"rollup": "^2.61.1",
"rollup-plugin-copy": "^3.4.0",
Expand All @@ -83,7 +85,7 @@
"semantic-release": "19.0.2",
"storybook-dark-mode": "^1.1.0",
"tailwindcss": "^3.0.3",
"typescript": "^4.5.4",
"typescript": "^4.7.4",
"web-vitals": "^2.1.2",
"webpack": "5"
},
Expand Down
64 changes: 34 additions & 30 deletions src/components/Pagination/Pagination.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,7 @@ export const Pagination: React.FC<PaginationProps> = ({
localization = {
of: 'of'
},
className
className = 'ui-max-w-[350px]'
}) => {
const pageCount = Math.ceil(length / pageSize);

Expand All @@ -35,25 +35,27 @@ export const Pagination: React.FC<PaginationProps> = ({
return (
<div
className={clsx(
'ui-flex ui-text-white ui-justify-around ui-items-center ui-gap-5 ui-max-w-[350px]',
'ui-flex ui-text-xl ui-text-gray-650 ui-justify-between ui-items-center ui-gap-5',
className
)}
>
<Icon
className={clsx({
'ui-cursor-pointer ui-text-primary hover:ui-text-blue-light-2': pageIndex !== 0
})}
name='Left-3'
onClick={() => gotoPage(0)}
/>
<div>
<Icon
className={clsx('ui-mr-3', {
'ui-cursor-pointer ui-text-primary hover:ui-text-blue-light-2': pageIndex !== 0
})}
name='Left-3'
onClick={() => gotoPage(0)}
/>

<Icon
className={clsx({
'ui-cursor-pointer ui-text-primary hover:ui-text-blue-light-2': pageIndex !== 0
})}
name='Left-4'
onClick={() => gotoPage(Math.max(pageIndex - 1, 0))}
/>
<Icon
className={clsx({
'ui-cursor-pointer ui-text-primary hover:ui-text-blue-light-2': pageIndex !== 0
})}
name='Left-4'
onClick={() => gotoPage(Math.max(pageIndex - 1, 0))}
/>
</div>

<h6 className='ui-tg-caption ui-text-gray-500 ui-select-none'>
{startIndex + 1}-{endIndex}
Expand All @@ -62,21 +64,23 @@ export const Pagination: React.FC<PaginationProps> = ({
{length}
</h6>

<Icon
className={clsx({
'ui-cursor-pointer ui-text-primary hover:ui-text-blue-light-2': canNextPage
})}
name='Right-4'
onClick={() => canNextPage && gotoPage(pageIndex + 1)}
/>
<div>
<Icon
className={clsx({
'ui-cursor-pointer ui-text-primary hover:ui-text-blue-light-2': canNextPage
})}
name='Right-4'
onClick={() => canNextPage && gotoPage(pageIndex + 1)}
/>

<Icon
className={clsx({
'ui-cursor-pointer ui-text-primary hover:ui-text-blue-light-2': canNextPage
})}
name='Right-3'
onClick={() => gotoPage(pageCount - 1)}
/>
<Icon
className={clsx({
'ui-cursor-pointer ui-text-primary hover:ui-text-blue-light-2': canNextPage
})}
name='Right-3'
onClick={() => gotoPage(pageCount - 1)}
/>
</div>
</div>
);
};
95 changes: 95 additions & 0 deletions src/components/Table/Table.stories.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,95 @@
import { ComponentMeta, ComponentStory } from '@storybook/react';
import { Card } from 'components/Card/Card';
import { Column } from 'react-table';

import { Table } from './Table';

export default {
title: 'Table',
component: Table,
decorators: [(Story) => <Story />]
} as ComponentMeta<typeof Table>;

const Template: ComponentStory<typeof Table> = (args) => (
<Card variant='dark-blue'>
<Table {...args} />
</Card>
);

export const Primary = Template.bind({});

type Crypto = {
coin: string;
symbol: string;
price: number;
percent1H: number;
percent24H: number;
percent7D: number;
last7Days: string;
};

const data: Crypto[] = [
{
coin: 'Synthetix',
symbol: 'sUSD',
price: 11,
percent1H: -1.3,
percent24H: -3.5,
percent7D: -8.6,
last7Days: 'https://picsum.photos/50'
},
{
coin: 'Synthetix',
symbol: 'SNX',
price: 2.71,
percent1H: -5.3,
percent24H: 9.5,
percent7D: 2.6,
last7Days: 'https://picsum.photos/50'
}
];

const columns: Column<Crypto>[] = [
{
disableSortBy: true,
Header: 'Coin',
accessor: 'symbol',
columnClass: 'ui-text-left',
cellClass: 'ui-text-left',
Cell: ({ row }) => (
<div className='ui-flex ui-items-center'>
<img
alt=''
className='ui-w-6 ui-h-6 ui-rounded-full ui-mr-3'
src='https://picsum.photos/50'
/>
<span className='ui-tg-caption-bold lg:ui-tg-content-bold'>{row.original.symbol}</span>
<span className='ui-hidden ui-tg-content ui-ml-3 lg:ui-block'>{row.original.coin}</span>
</div>
)
},
{
Header: 'Price',
accessor: (row) => row.price,
sortType: (a, b) => a.original.price - b.original.price
},
{
Header: '1h',
accessor: 'percent1H',
Cell: (row) => row.value,
sortType: (a, b) => a.original.percent1H - b.original.percent1H
},
{
Header: '7h',
accessor: 'percent7D',
Cell: (row) => row.value,
sortType: (a, b) => a.original.percent7D - b.original.percent7D
}
];

Primary.args = {
className: '',
data: [...data, ...data],
columns: columns,
initialState: { pageSize: 3 }
};
139 changes: 139 additions & 0 deletions src/components/Table/Table.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,139 @@
/* eslint-disable react/jsx-key */
import clsx from 'clsx';
import { Icon } from 'components/Icon/Icon';
import { Pagination, PaginationLocalization } from 'components/Pagination/Pagination';
import React, { ReactElement, ReactNode, useMemo } from 'react';
import {
Row,
TableHeaderProps,
TableOptions,
useFlexLayout,
usePagination,
useSortBy,
useTable
} from 'react-table';

export interface TableProps<T extends Record<string, unknown>> extends TableOptions<T> {
className?: string;
paginationLocalization?: PaginationLocalization;
onClick?: (row: Row<T>) => void;
}

export const Table = <T extends Record<string, unknown>>(props: TableProps<T>): ReactElement => {
const { className, paginationLocalization, onClick, ...rest } = props;

const defaultColumn = useMemo(
() => ({
width: 150,
minWidth: 100
}),
[]
);

const tableInstance = useTable(
{ ...rest, defaultColumn },
useSortBy,
useFlexLayout,
usePagination
);

const {
headerGroups,
prepareRow,
getTableProps,
getTableBodyProps,
page,
state,
gotoPage,
rows
} = tableInstance;

const { pageIndex, pageSize } = state;

return (
<>
<div className={clsx('ui-overflow-auto', className)}>
<table {...getTableProps()} className='ui-w-full'>
<thead>
{headerGroups.map((headerGroup) => (
<tr {...headerGroup.getHeaderGroupProps()}>
{headerGroup.headers.map((column) => (
<TableHeader
className={clsx(
'ui-text-gray-500 ui-select-none ui-p-2 ui-tg-caption-bold lg:ui-tg-content-bold ui-transition-colors',
column.columnClass || 'ui-text-right',
{
'hover:ui-text-primary': column.canSort
}
)}
{...column.getHeaderProps(column.getSortByToggleProps())}
>
<>
{column.render('Header')}
<span>
{column.isSorted && (
<Icon
className='ui-text-xs ui-ml-1'
name={column.isSortedDesc ? 'Top' : 'Bottom'}
/>
)}
</span>
</>
</TableHeader>
))}
</tr>
))}
</thead>

<tbody {...getTableBodyProps()}>
{page.map((row) => {
prepareRow(row);
return (
<tr
onClick={() => onClick?.(row)}
{...row.getRowProps()}
className={clsx(
'ui-text-white ui-border-t last:ui-border-b ui-border-solid ui-border-gray-700',
{
'ui-cursor-pointer': !!onClick
}
)}
>
{row.cells.map((cell) => (
<td
className={clsx(
'ui-p-2 ui-tg-caption lg:ui-tg-content',
cell.column.cellClass || 'ui-text-right'
)}
{...cell.getCellProps()}
>
<span className='ui-inline-flex ui-items-center ui-h-full'>
{cell.render('Cell') as ReactNode}
</span>
</td>
))}
</tr>
);
})}
</tbody>
</table>
</div>
<Pagination
className='ui-mt-6 ui-w-full'
gotoPage={gotoPage}
length={rows.length}
localization={paginationLocalization}
pageIndex={pageIndex}
pageSize={pageSize}
/>
</>
);
};

interface HeaderProps extends TableHeaderProps {
children: React.ReactNode;
}

const TableHeader: React.FC<HeaderProps> = ({ children, ...props }) => {
return <th {...props}>{children}</th>;
};
Loading

0 comments on commit a70ef06

Please sign in to comment.