2018-10-28 23:26:47 +03:00
|
|
|
import { UncontrolledDropdown, DropdownToggle, DropdownMenu, DropdownItem } from 'reactstrap';
|
|
|
|
import { toPairs } from 'ramda';
|
2019-01-06 00:25:54 +03:00
|
|
|
import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
|
|
|
|
import { faSortAmountUp as sortAscIcon, faSortAmountDown as sortDescIcon } from '@fortawesome/free-solid-svg-icons';
|
2018-10-29 00:54:08 +03:00
|
|
|
import classNames from 'classnames';
|
2021-11-06 14:26:20 +03:00
|
|
|
import { determineOrderDir, Order, OrderDir } from './helpers/ordering';
|
2021-12-25 12:31:48 +03:00
|
|
|
import './OrderingDropdown.scss';
|
2018-10-28 23:26:47 +03:00
|
|
|
|
2021-12-25 12:31:48 +03:00
|
|
|
export interface OrderingDropdownProps<T extends string = string> {
|
2020-08-31 19:38:27 +03:00
|
|
|
items: Record<T, string>;
|
2021-11-06 14:26:20 +03:00
|
|
|
order: Order<T>;
|
2020-08-31 19:38:27 +03:00
|
|
|
onChange: (orderField?: T, orderDir?: OrderDir) => void;
|
|
|
|
isButton?: boolean;
|
|
|
|
right?: boolean;
|
|
|
|
}
|
2018-10-29 00:54:08 +03:00
|
|
|
|
2021-12-25 12:31:48 +03:00
|
|
|
export function OrderingDropdown<T extends string = string>(
|
|
|
|
{ items, order, onChange, isButton = true, right = false }: OrderingDropdownProps<T>,
|
2020-08-31 19:38:27 +03:00
|
|
|
) {
|
|
|
|
const handleItemClick = (fieldKey: T) => () => {
|
2021-11-06 14:26:20 +03:00
|
|
|
const newOrderDir = determineOrderDir(fieldKey, order.field, order.dir);
|
2018-10-28 23:26:47 +03:00
|
|
|
|
2018-10-29 00:54:08 +03:00
|
|
|
onChange(newOrderDir ? fieldKey : undefined, newOrderDir);
|
|
|
|
};
|
|
|
|
|
|
|
|
return (
|
|
|
|
<UncontrolledDropdown>
|
|
|
|
<DropdownToggle
|
|
|
|
caret
|
2020-12-25 12:54:49 +03:00
|
|
|
color={isButton ? 'primary' : 'link'}
|
|
|
|
className={classNames({ 'dropdown-btn__toggle btn-block': isButton, 'btn-sm p-0': !isButton })}
|
2018-10-29 00:54:08 +03:00
|
|
|
>
|
2020-12-25 12:54:49 +03:00
|
|
|
{!isButton && <>Order by</>}
|
2021-11-06 14:26:20 +03:00
|
|
|
{isButton && !order.field && <>Order by...</>}
|
|
|
|
{isButton && order.field && `Order by: "${items[order.field]}" - "${order.dir ?? 'DESC'}"`}
|
2018-10-29 00:54:08 +03:00
|
|
|
</DropdownToggle>
|
|
|
|
<DropdownMenu
|
2022-03-11 18:37:41 +03:00
|
|
|
end={right}
|
2021-12-25 12:31:48 +03:00
|
|
|
className={classNames('w-100', { 'ordering-dropdown__menu--link': !isButton })}
|
2018-10-29 00:54:08 +03:00
|
|
|
>
|
|
|
|
{toPairs(items).map(([ fieldKey, fieldValue ]) => (
|
2021-11-06 14:26:20 +03:00
|
|
|
<DropdownItem key={fieldKey} active={order.field === fieldKey} onClick={handleItemClick(fieldKey as T)}>
|
2018-10-29 00:54:08 +03:00
|
|
|
{fieldValue}
|
2021-11-06 14:26:20 +03:00
|
|
|
{order.field === fieldKey && (
|
2018-10-29 00:54:08 +03:00
|
|
|
<FontAwesomeIcon
|
2021-11-06 14:26:20 +03:00
|
|
|
icon={order.dir === 'ASC' ? sortAscIcon : sortDescIcon}
|
2021-12-25 12:31:48 +03:00
|
|
|
className="ordering-dropdown__sort-icon"
|
2018-10-29 00:54:08 +03:00
|
|
|
/>
|
|
|
|
)}
|
|
|
|
</DropdownItem>
|
|
|
|
))}
|
|
|
|
<DropdownItem divider />
|
2021-11-06 14:26:20 +03:00
|
|
|
<DropdownItem disabled={!order.field} onClick={() => onChange()}>
|
2018-10-29 00:54:08 +03:00
|
|
|
<i>Clear selection</i>
|
2018-10-28 23:26:47 +03:00
|
|
|
</DropdownItem>
|
2018-10-29 00:54:08 +03:00
|
|
|
</DropdownMenu>
|
|
|
|
</UncontrolledDropdown>
|
|
|
|
);
|
2020-08-31 19:38:27 +03:00
|
|
|
}
|