bigbuffet-rw/app/soapbox/components/ui/text/text.tsx

138 lines
3.4 KiB
TypeScript
Raw Normal View History

2023-02-06 10:01:03 -08:00
import clsx from 'clsx';
2022-03-21 11:09:01 -07:00
import React from 'react';
const themes = {
2022-03-23 17:18:37 -07:00
default: 'text-gray-900 dark:text-gray-100',
2022-03-21 11:09:01 -07:00
danger: 'text-danger-600',
primary: 'text-primary-600 dark:text-accent-blue',
muted: 'text-gray-700 dark:text-gray-600',
2022-03-23 17:18:37 -07:00
subtle: 'text-gray-400 dark:text-gray-500',
2022-03-21 11:09:01 -07:00
success: 'text-success-600',
inherit: 'text-inherit',
white: 'text-white',
};
const weights = {
normal: 'font-normal',
medium: 'font-medium',
semibold: 'font-semibold',
bold: 'font-bold',
};
const sizes = {
xs: 'text-xs',
sm: 'text-sm',
md: 'text-base leading-5',
2022-03-21 11:09:01 -07:00
lg: 'text-lg',
xl: 'text-xl',
'2xl': 'text-2xl',
'3xl': 'text-3xl',
};
const alignments = {
left: 'text-left',
center: 'text-center',
right: 'text-right',
};
const trackingSizes = {
normal: 'tracking-normal',
wide: 'tracking-wide',
};
2022-04-12 06:50:04 -07:00
const transformProperties = {
normal: 'normal-case',
uppercase: 'uppercase',
};
2022-03-21 11:09:01 -07:00
const families = {
sans: 'font-sans',
mono: 'font-mono',
};
2022-10-04 12:08:22 -07:00
export type Sizes = keyof typeof sizes
type Tags = 'abbr' | 'p' | 'span' | 'pre' | 'time' | 'h1' | 'h2' | 'h3' | 'h4' | 'h5' | 'h6' | 'label' | 'div' | 'blockquote'
2022-10-04 12:08:22 -07:00
type Directions = 'ltr' | 'rtl'
interface IText extends Pick<React.HTMLAttributes<HTMLParagraphElement>, 'dangerouslySetInnerHTML' | 'tabIndex' | 'lang'> {
/** Text content. */
children?: React.ReactNode
/** How to align the text. */
align?: keyof typeof alignments
/** Extra class names for the outer element. */
className?: string
2022-05-30 12:05:15 -07:00
/** Text direction. */
direction?: Directions
/** Typeface of the text. */
family?: keyof typeof families
2022-04-27 07:11:21 -07:00
/** The "for" attribute specifies which form element a label is bound to. */
htmlFor?: string
/** Font size of the text. */
size?: Sizes
/** HTML element name of the outer element. */
tag?: Tags
/** Theme for the text. */
theme?: keyof typeof themes
/** Letter-spacing of the text. */
tracking?: keyof typeof trackingSizes
/** Transform (eg uppercase) for the text. */
transform?: keyof typeof transformProperties
/** Whether to truncate the text if its container is too small. */
truncate?: boolean
/** Font weight of the text. */
weight?: keyof typeof weights
2022-08-31 15:01:19 -07:00
/** Tooltip title. */
title?: string
2022-03-21 11:09:01 -07:00
}
/** UI-friendly text container with dark mode support. */
const Text = React.forwardRef<any, IText>(
(props, ref) => {
2022-03-21 11:09:01 -07:00
const {
align,
className,
2022-05-30 12:05:15 -07:00
direction,
2022-03-21 11:09:01 -07:00
family = 'sans',
size = 'md',
tag = 'p',
theme = 'default',
tracking = 'normal',
2022-04-12 06:50:04 -07:00
transform = 'normal',
2022-03-21 11:09:01 -07:00
truncate = false,
weight = 'normal',
...filteredProps
} = props;
const Comp: React.ElementType = tag;
const alignmentClass = typeof align === 'string' ? alignments[align] : '';
2022-03-21 11:09:01 -07:00
return (
<Comp
{...filteredProps}
ref={ref}
2022-05-30 12:05:15 -07:00
style={{
textDecoration: tag === 'abbr' ? 'underline dotted' : undefined,
direction,
}}
2023-02-06 10:01:03 -08:00
className={clsx({
2022-03-21 11:09:01 -07:00
'cursor-default': tag === 'abbr',
truncate: truncate,
[sizes[size]]: true,
[themes[theme]]: true,
[weights[weight]]: true,
[trackingSizes[tracking]]: true,
[families[family]]: true,
[alignmentClass]: typeof align !== 'undefined',
2022-04-12 06:50:04 -07:00
[transformProperties[transform]]: typeof transform !== 'undefined',
}, className)}
2022-03-21 11:09:01 -07:00
/>
);
},
);
2022-08-31 15:01:19 -07:00
export {
Text as default,
IText,
};