2023-02-06 10:01:03 -08:00
|
|
|
import clsx from 'clsx';
|
2022-03-21 11:09:01 -07:00
|
|
|
import React from 'react';
|
|
|
|
import { defineMessages, useIntl } from 'react-intl';
|
|
|
|
import { Link } from 'react-router-dom';
|
|
|
|
|
2022-11-25 09:04:11 -08:00
|
|
|
import { HStack, Text } from 'soapbox/components/ui';
|
2022-04-07 11:47:06 -07:00
|
|
|
import SvgIcon from 'soapbox/components/ui/icon/svg-icon';
|
2022-03-24 16:56:22 -07:00
|
|
|
|
2022-03-21 11:09:01 -07:00
|
|
|
const sizes = {
|
|
|
|
md: 'p-4 sm:rounded-xl',
|
|
|
|
lg: 'p-4 sm:p-6 sm:rounded-xl',
|
|
|
|
xl: 'p-4 sm:p-10 sm:rounded-3xl',
|
|
|
|
};
|
|
|
|
|
|
|
|
const messages = defineMessages({
|
|
|
|
back: { id: 'card.back.label', defaultMessage: 'Back' },
|
|
|
|
});
|
|
|
|
|
2023-04-05 07:30:05 -07:00
|
|
|
export type CardSizes = keyof typeof sizes
|
|
|
|
|
2022-03-21 11:09:01 -07:00
|
|
|
interface ICard {
|
2022-04-30 21:39:58 -07:00
|
|
|
/** The type of card. */
|
2023-04-11 05:55:17 -07:00
|
|
|
variant?: 'default' | 'rounded' | 'slim'
|
2022-04-30 21:39:58 -07:00
|
|
|
/** Card size preset. */
|
2023-04-05 07:30:05 -07:00
|
|
|
size?: CardSizes
|
2022-04-30 21:39:58 -07:00
|
|
|
/** Extra classnames for the <div> element. */
|
2022-10-04 12:08:22 -07:00
|
|
|
className?: string
|
2022-04-30 21:39:58 -07:00
|
|
|
/** Elements inside the card. */
|
2022-10-04 12:08:22 -07:00
|
|
|
children: React.ReactNode
|
2022-03-21 11:09:01 -07:00
|
|
|
}
|
|
|
|
|
2022-04-30 21:39:58 -07:00
|
|
|
/** An opaque backdrop to hold a collection of related elements. */
|
2022-08-12 10:58:35 -07:00
|
|
|
const Card = React.forwardRef<HTMLDivElement, ICard>(({ children, variant = 'default', size = 'md', className, ...filteredProps }, ref): JSX.Element => (
|
2022-03-21 11:09:01 -07:00
|
|
|
<div
|
|
|
|
ref={ref}
|
|
|
|
{...filteredProps}
|
2023-02-06 10:01:03 -08:00
|
|
|
className={clsx({
|
2023-04-05 11:16:18 -07:00
|
|
|
'bg-white dark:bg-primary-900 text-gray-900 dark:text-gray-100 shadow-lg dark:shadow-none': variant === 'rounded',
|
2022-07-15 15:03:21 -07:00
|
|
|
[sizes[size]]: variant === 'rounded',
|
2023-04-11 05:55:17 -07:00
|
|
|
'py-4': variant === 'slim',
|
2022-03-24 12:27:27 -07:00
|
|
|
}, className)}
|
2022-03-21 11:09:01 -07:00
|
|
|
>
|
|
|
|
{children}
|
|
|
|
</div>
|
|
|
|
));
|
|
|
|
|
|
|
|
interface ICardHeader {
|
2023-02-15 13:26:27 -08:00
|
|
|
backHref?: string
|
2022-03-21 11:09:01 -07:00
|
|
|
onBackClick?: (event: React.MouseEvent) => void
|
2022-11-30 09:19:16 -08:00
|
|
|
className?: string
|
2023-01-10 15:03:15 -08:00
|
|
|
children?: React.ReactNode
|
2022-03-21 11:09:01 -07:00
|
|
|
}
|
|
|
|
|
2022-05-11 12:35:56 -07:00
|
|
|
/**
|
|
|
|
* Card header container with back button.
|
|
|
|
* Typically holds a CardTitle.
|
|
|
|
*/
|
2022-11-30 09:19:16 -08:00
|
|
|
const CardHeader: React.FC<ICardHeader> = ({ className, children, backHref, onBackClick }): JSX.Element => {
|
2022-03-21 11:09:01 -07:00
|
|
|
const intl = useIntl();
|
|
|
|
|
|
|
|
const renderBackButton = () => {
|
|
|
|
if (!backHref && !onBackClick) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Comp: React.ElementType = backHref ? Link : 'button';
|
|
|
|
const backAttributes = backHref ? { to: backHref } : { onClick: onBackClick };
|
|
|
|
|
|
|
|
return (
|
2023-03-20 17:37:46 -07:00
|
|
|
<Comp {...backAttributes} className='rounded-full text-gray-900 focus:ring-2 focus:ring-primary-500 dark:text-gray-100' aria-label={intl.formatMessage(messages.back)}>
|
2022-12-04 17:40:09 -08:00
|
|
|
<SvgIcon src={require('@tabler/icons/arrow-left.svg')} className='h-6 w-6 rtl:rotate-180' />
|
2022-05-07 10:04:28 -07:00
|
|
|
<span className='sr-only' data-testid='back-button'>{intl.formatMessage(messages.back)}</span>
|
2022-03-21 11:09:01 -07:00
|
|
|
</Comp>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
return (
|
2023-04-05 11:16:18 -07:00
|
|
|
<HStack alignItems='center' space={2} className={className}>
|
2022-03-21 11:09:01 -07:00
|
|
|
{renderBackButton()}
|
|
|
|
|
|
|
|
{children}
|
2022-11-25 09:04:11 -08:00
|
|
|
</HStack>
|
2022-03-21 11:09:01 -07:00
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
interface ICardTitle {
|
2022-05-24 08:16:07 -07:00
|
|
|
title: React.ReactNode
|
2022-03-21 11:09:01 -07:00
|
|
|
}
|
|
|
|
|
2022-04-30 21:39:58 -07:00
|
|
|
/** A card's title. */
|
2022-05-01 11:29:36 -07:00
|
|
|
const CardTitle: React.FC<ICardTitle> = ({ title }): JSX.Element => (
|
|
|
|
<Text size='xl' weight='bold' tag='h1' data-testid='card-title' truncate>{title}</Text>
|
2022-03-21 11:09:01 -07:00
|
|
|
);
|
|
|
|
|
2022-11-26 13:15:58 -08:00
|
|
|
interface ICardBody {
|
|
|
|
/** Classnames for the <div> element. */
|
|
|
|
className?: string
|
2023-01-10 15:03:15 -08:00
|
|
|
/** Children to appear inside the card. */
|
|
|
|
children: React.ReactNode
|
2022-11-26 13:15:58 -08:00
|
|
|
}
|
|
|
|
|
2022-04-30 21:39:58 -07:00
|
|
|
/** A card's body. */
|
2022-11-26 13:15:58 -08:00
|
|
|
const CardBody: React.FC<ICardBody> = ({ className, children }): JSX.Element => (
|
|
|
|
<div data-testid='card-body' className={className}>{children}</div>
|
2022-03-21 11:09:01 -07:00
|
|
|
);
|
|
|
|
|
|
|
|
export { Card, CardHeader, CardTitle, CardBody };
|