RocketChat/Rocket.Chat

View on GitHub
apps/meteor/client/views/admin/customUserStatus/CustomUserStatusForm.tsx

Summary

Maintainability
A
3 hrs
Test Coverage
import type { IUserStatus } from '@rocket.chat/core-typings';
import type { SelectOption } from '@rocket.chat/fuselage';
import { FieldGroup, Button, ButtonGroup, TextInput, Field, FieldLabel, FieldRow, FieldError, Select, Box } from '@rocket.chat/fuselage';
import { useUniqueId } from '@rocket.chat/fuselage-hooks';
import { useSetModal, useRoute, useToastMessageDispatch, useTranslation, useEndpoint } from '@rocket.chat/ui-contexts';
import type { ReactElement } from 'react';
import React, { useCallback } from 'react';
import { useForm, Controller } from 'react-hook-form';

import { ContextualbarScrollableContent, ContextualbarFooter } from '../../../components/Contextualbar';
import GenericModal from '../../../components/GenericModal';

type CustomUserStatusFormProps = {
    onClose: () => void;
    onReload: () => void;
    status?: IUserStatus;
};

const CustomUserStatusForm = ({ onClose, onReload, status }: CustomUserStatusFormProps): ReactElement => {
    const t = useTranslation();
    const { _id, name, statusType } = status || {};
    const setModal = useSetModal();
    const route = useRoute('user-status');
    const dispatchToastMessage = useToastMessageDispatch();
    const formId = useUniqueId();

    const {
        register,
        control,
        handleSubmit,
        formState: { isDirty, errors },
    } = useForm({
        defaultValues: { name: status?.name ?? '', statusType: status?.statusType ?? '' },
    });

    const saveStatus = useEndpoint('POST', _id ? '/v1/custom-user-status.update' : '/v1/custom-user-status.create');
    const deleteStatus = useEndpoint('POST', '/v1/custom-user-status.delete');

    const handleSave = useCallback(
        async (data) => {
            try {
                await saveStatus({ _id, name, statusType, ...data });

                dispatchToastMessage({
                    type: 'success',
                    message: t('Custom_User_Status_Updated_Successfully'),
                });

                onReload();
                route.push({});
            } catch (error) {
                dispatchToastMessage({ type: 'error', message: error });
            }
        },
        [saveStatus, _id, name, statusType, route, dispatchToastMessage, t, onReload],
    );

    const handleDeleteStatus = useCallback(() => {
        const handleCancel = (): void => {
            setModal(null);
        };

        const handleDelete = async (): Promise<void> => {
            try {
                await deleteStatus({ customUserStatusId: _id || '' });
                dispatchToastMessage({ type: 'success', message: t('Custom_User_Status_Has_Been_Deleted') });
                onReload();
                route.push({});
            } catch (error) {
                dispatchToastMessage({ type: 'error', message: error });
            } finally {
                setModal(null);
            }
        };

        setModal(() => (
            <GenericModal variant='danger' onConfirm={handleDelete} onCancel={handleCancel} onClose={handleCancel} confirmText={t('Delete')}>
                {t('Custom_User_Status_Delete_Warning')}
            </GenericModal>
        ));
    }, [_id, route, deleteStatus, dispatchToastMessage, onReload, setModal, t]);

    const presenceOptions: SelectOption[] = [
        ['online', t('Online')],
        ['busy', t('Busy')],
        ['away', t('Away')],
        ['offline', t('Offline')],
    ];

    return (
        <>
            <ContextualbarScrollableContent>
                <FieldGroup id={formId} is='form' onSubmit={handleSubmit(handleSave)}>
                    <Field>
                        <FieldLabel>{t('Name')}</FieldLabel>
                        <FieldRow>
                            <TextInput {...register('name', { required: true })} placeholder={t('Name')} />
                        </FieldRow>
                        {errors?.name && <FieldError>{t('error-the-field-is-required', { field: t('Name') })}</FieldError>}
                    </Field>
                    <Field>
                        <FieldLabel>{t('Presence')}</FieldLabel>
                        <FieldRow>
                            <Controller
                                name='statusType'
                                control={control}
                                rules={{ required: true }}
                                render={({ field }): ReactElement => <Select {...field} placeholder={t('Presence')} options={presenceOptions} />}
                            />
                        </FieldRow>
                        {errors?.statusType && <FieldError>{t('error-the-field-is-required', { field: t('Presence') })}</FieldError>}
                    </Field>
                </FieldGroup>
            </ContextualbarScrollableContent>
            <ContextualbarFooter>
                <ButtonGroup stretch>
                    <Button onClick={onClose}>{t('Cancel')}</Button>
                    <Button form={formId} primary type='submit' disabled={!isDirty}>
                        {t('Save')}
                    </Button>
                </ButtonGroup>
                {_id && (
                    <Box mbs={8}>
                        <ButtonGroup stretch>
                            <Button icon='trash' danger onClick={handleDeleteStatus}>
                                {t('Delete')}
                            </Button>
                        </ButtonGroup>
                    </Box>
                )}
            </ContextualbarFooter>
        </>
    );
};

export default CustomUserStatusForm;