mirror of https://github.com/portainer/portainer
111 lines
2.4 KiB
TypeScript
111 lines
2.4 KiB
TypeScript
import clsx from 'clsx';
|
|
import { PropsWithChildren, ReactNode } from 'react';
|
|
import { TableHeaderProps } from 'react-table';
|
|
|
|
import { TableHeaderSortIcons } from './TableHeaderSortIcons';
|
|
import styles from './TableHeaderCell.module.css';
|
|
|
|
interface Props {
|
|
canFilter: boolean;
|
|
canSort: boolean;
|
|
headerProps: TableHeaderProps;
|
|
isSorted: boolean;
|
|
isSortedDesc?: boolean;
|
|
onSortClick: (desc: boolean) => void;
|
|
render: () => ReactNode;
|
|
renderFilter: () => ReactNode;
|
|
}
|
|
|
|
export function TableHeaderCell({
|
|
headerProps: { className, role, style },
|
|
canSort,
|
|
render,
|
|
onSortClick,
|
|
isSorted,
|
|
isSortedDesc = true,
|
|
canFilter,
|
|
renderFilter,
|
|
}: Props) {
|
|
return (
|
|
<th role={role} style={style} className={className}>
|
|
<div className="flex h-full flex-row flex-nowrap items-center gap-1">
|
|
<SortWrapper
|
|
canSort={canSort}
|
|
onClick={onSortClick}
|
|
isSorted={isSorted}
|
|
isSortedDesc={isSortedDesc}
|
|
>
|
|
{render()}
|
|
</SortWrapper>
|
|
{canFilter ? renderFilter() : null}
|
|
</div>
|
|
</th>
|
|
);
|
|
}
|
|
|
|
interface SortWrapperProps {
|
|
canSort: boolean;
|
|
isSorted: boolean;
|
|
isSortedDesc?: boolean;
|
|
onClick?: (desc: boolean) => void;
|
|
}
|
|
|
|
function SortWrapper({
|
|
canSort,
|
|
children,
|
|
onClick = () => {},
|
|
isSorted,
|
|
isSortedDesc = true,
|
|
}: PropsWithChildren<SortWrapperProps>) {
|
|
if (!canSort) {
|
|
return <>{children}</>;
|
|
}
|
|
|
|
return (
|
|
<button
|
|
type="button"
|
|
onClick={() => onClick(!isSortedDesc)}
|
|
className={clsx(
|
|
'!ml-0 h-full w-full border-none !bg-transparent !px-0 focus:border-none',
|
|
styles.sortable,
|
|
isSorted && styles.sortingActive
|
|
)}
|
|
>
|
|
<div className="flex h-full w-full flex-row items-center justify-start">
|
|
{children}
|
|
<TableHeaderSortIcons
|
|
sorted={isSorted}
|
|
descending={isSorted && !!isSortedDesc}
|
|
className="ml-1"
|
|
/>
|
|
</div>
|
|
</button>
|
|
);
|
|
}
|
|
|
|
interface TableColumnHeaderAngularProps {
|
|
colTitle: string;
|
|
canSort: boolean;
|
|
isSorted?: boolean;
|
|
isSortedDesc?: boolean;
|
|
}
|
|
|
|
export function TableColumnHeaderAngular({
|
|
canSort,
|
|
isSorted,
|
|
colTitle,
|
|
isSortedDesc = true,
|
|
}: TableColumnHeaderAngularProps) {
|
|
return (
|
|
<div className="flex h-full flex-row flex-nowrap">
|
|
<SortWrapper
|
|
canSort={canSort}
|
|
isSorted={!!isSorted}
|
|
isSortedDesc={isSortedDesc}
|
|
>
|
|
{colTitle}
|
|
</SortWrapper>
|
|
</div>
|
|
);
|
|
}
|