pleroma/app/soapbox/components/ui/streamfield/streamfield.tsx

95 lines
2.6 KiB
TypeScript
Raw Normal View History

2022-05-02 19:10:47 -07:00
import React from 'react';
2022-05-02 20:05:43 -07:00
import { useIntl, defineMessages } from 'react-intl';
2022-05-02 19:10:47 -07:00
import Button from '../button/button';
2022-05-02 19:38:15 -07:00
import HStack from '../hstack/hstack';
import IconButton from '../icon-button/icon-button';
2022-05-02 19:10:47 -07:00
import Stack from '../stack/stack';
import Text from '../text/text';
2022-05-02 19:10:47 -07:00
2022-05-02 20:05:43 -07:00
const messages = defineMessages({
add: { id: 'streamfield.add', defaultMessage: 'Add' },
remove: { id: 'streamfield.remove', defaultMessage: 'Remove' },
});
2022-05-02 19:10:47 -07:00
interface IStreamfield {
/** Array of values for the streamfield. */
2022-05-02 19:10:47 -07:00
values: any[],
/** Input label message. */
labelText?: React.ReactNode,
/** Input hint message. */
hintText?: React.ReactNode,
/** Callback to add an item. */
2022-05-02 19:10:47 -07:00
onAddItem?: () => void,
/** Callback to remove an item by index. */
2022-05-02 19:38:15 -07:00
onRemoveItem?: (i: number) => void,
/** Callback when values are changed. */
2022-05-02 19:10:47 -07:00
onChange: (values: any[]) => void,
/** Input to render for each value. */
2022-05-02 19:10:47 -07:00
component: React.ComponentType<{ onChange: (value: any) => void, value: any }>,
/** Maximum number of allowed inputs. */
2022-05-02 19:10:47 -07:00
maxItems?: number,
}
/** List of inputs that can be added or removed. */
const Streamfield: React.FC<IStreamfield> = ({
values,
labelText,
hintText,
2022-05-02 19:10:47 -07:00
onAddItem,
2022-05-02 19:38:15 -07:00
onRemoveItem,
2022-05-02 19:10:47 -07:00
onChange,
component: Component,
maxItems = Infinity,
}) => {
2022-05-02 20:05:43 -07:00
const intl = useIntl();
2022-05-02 19:10:47 -07:00
const handleChange = (i: number) => {
return (value: any) => {
const newData = [...values];
newData[i] = value;
onChange(newData);
};
};
return (
<Stack space={4}>
<Stack>
{labelText && <Text size='sm' weight='medium'>{labelText}</Text>}
{hintText && <Text size='xs' theme='muted'>{hintText}</Text>}
</Stack>
2022-05-02 19:10:47 -07:00
<Stack>
2022-05-02 19:38:15 -07:00
{values.map((value, i) => (
<HStack space={2} alignItems='center'>
2022-05-02 19:10:47 -07:00
<Component key={i} onChange={handleChange(i)} value={value} />
2022-05-02 19:38:15 -07:00
{onRemoveItem && (
<IconButton
iconClassName='w-4 h-4'
className='bg-transparent text-gray-400 hover:text-gray-600'
src={require('@tabler/icons/icons/x.svg')}
onClick={() => onRemoveItem(i)}
2022-05-02 20:05:43 -07:00
title={intl.formatMessage(messages.remove)}
2022-05-02 19:38:15 -07:00
/>
)}
</HStack>
))}
2022-05-02 19:10:47 -07:00
</Stack>
{onAddItem && (
<Button
icon={require('@tabler/icons/icons/plus.svg')}
onClick={onAddItem}
disabled={values.length >= maxItems}
theme='ghost'
block
>
2022-05-02 20:05:43 -07:00
{intl.formatMessage(messages.add)}
2022-05-02 19:10:47 -07:00
</Button>
)}
</Stack>
);
};
export default Streamfield;