RocketChat/Rocket.Chat

View on GitHub
apps/meteor/ee/client/omnichannel/cannedResponses/components/cannedResponseForm.tsx

Summary

Maintainability
F
3 days
Test Coverage
import { css } from '@rocket.chat/css-in-js';
import { Box, Field, FieldLabel, FieldRow, FieldError, TextInput, FieldGroup, RadioButton, FieldHint, Option } from '@rocket.chat/fuselage';
import { useUniqueId } from '@rocket.chat/fuselage-hooks';
import { usePermission, useTranslation } from '@rocket.chat/ui-contexts';
import React, { useState } from 'react';
import { useFormContext, Controller } from 'react-hook-form';

import AutoCompleteDepartment from '../../../../../client/components/AutoCompleteDepartment';
import Tags from '../../../../../client/components/Omnichannel/Tags';
import CannedResponsesComposer from './CannedResponsesComposer/CannedResponsesComposer';
import CannedResponsesComposerPreview from './CannedResponsesComposer/CannedResponsesComposerPreview';

// TODO: refactor Tags field to get proper validation
const CannedResponseForm = () => {
    const t = useTranslation();
    const hasManagerPermission = usePermission('view-all-canned-responses');
    const hasMonitorPermission = usePermission('save-department-canned-responses');

    const {
        control,
        formState: { errors },
        watch,
    } = useFormContext();

    const clickable = css`
        cursor: pointer;
    `;

    const { text, scope } = watch();
    const [preview, setPreview] = useState(false);

    const shortcutField = useUniqueId();
    const messageField = useUniqueId();
    const publicRadioField = useUniqueId();
    const departmentRadioField = useUniqueId();
    const privateRadioField = useUniqueId();
    const departmentField = useUniqueId();

    return (
        <FieldGroup>
            <Field>
                <FieldLabel htmlFor={shortcutField}>{t('Shortcut')}</FieldLabel>
                <Controller
                    name='shortcut'
                    control={control}
                    rules={{ required: t('The_field_is_required', t('Shortcut')) }}
                    render={({ field }) => (
                        <TextInput
                            {...field}
                            id={shortcutField}
                            placeholder={`!${t('shortcut_name')}`}
                            aria-required={true}
                            aria-invalid={Boolean(errors?.shortcut)}
                            aria-describedby={`${shortcutField}-error`}
                        />
                    )}
                />
                {errors?.shortcut && (
                    <FieldError aria-live='assertive' id={`${shortcutField}-error`}>
                        {errors.shortcut.message}
                    </FieldError>
                )}
            </Field>
            <Field>
                <FieldLabel htmlFor={messageField} w='full' display='flex' flexDirection='row' justifyContent='space-between'>
                    {t('Message')}
                    {text !== '' && (
                        <Box className={clickable} color='info' onClick={() => setPreview(!preview)}>
                            {preview ? t('Editor') : t('Preview')}
                        </Box>
                    )}
                </FieldLabel>
                {preview ? (
                    <CannedResponsesComposerPreview text={text} />
                ) : (
                    <Controller
                        name='text'
                        control={control}
                        rules={{ required: t('The_field_is_required', t('Message')) }}
                        render={({ field: { value, onChange, name, onBlur } }) => (
                            <CannedResponsesComposer
                                id={messageField}
                                value={value}
                                onChange={onChange}
                                name={name}
                                onBlur={onBlur}
                                aria-describedby={`${messageField}-error`}
                                aria-required={true}
                                aria-invalid={Boolean(errors.text)}
                            />
                        )}
                    />
                )}
                {errors?.text && (
                    <FieldError aria-live='assertive' id={`${messageField}-error`}>
                        {errors.text.message}
                    </FieldError>
                )}
            </Field>
            <Field>
                <Controller name='tags' control={control} render={({ field: { value, onChange } }) => <Tags handler={onChange} tags={value} />} />
            </Field>
            {(hasManagerPermission || hasMonitorPermission) && (
                <>
                    <Field>
                        <FieldRow>
                            <FieldLabel htmlFor={publicRadioField}>{t('Public')}</FieldLabel>
                            <Controller
                                name='scope'
                                control={control}
                                render={({ field: { onChange, value, ...field } }) => (
                                    <RadioButton
                                        {...field}
                                        id={publicRadioField}
                                        onChange={() => onChange('global')}
                                        disabled={hasMonitorPermission && !hasManagerPermission}
                                        checked={value === 'global'}
                                        aria-describedby={`${publicRadioField}-hint`}
                                        data-qa-id='canned-response-public-radio'
                                    />
                                )}
                            />
                        </FieldRow>
                        <FieldHint id={`${publicRadioField}-hint`}>{t('Canned_Response_Sharing_Public_Description')}</FieldHint>
                    </Field>
                    <Field>
                        <FieldRow>
                            <FieldLabel htmlFor={departmentRadioField}>{t('Department')}</FieldLabel>
                            <Controller
                                name='scope'
                                control={control}
                                render={({ field: { onChange, value, ...field } }) => (
                                    <RadioButton
                                        {...field}
                                        id={departmentRadioField}
                                        onChange={() => onChange('department')}
                                        checked={value === 'department'}
                                        aria-describedby={`${departmentRadioField}-hint`}
                                    />
                                )}
                            />
                        </FieldRow>
                        <FieldHint id={`${departmentRadioField}-hint`}>{t('Canned_Response_Sharing_Department_Description')}</FieldHint>
                    </Field>
                    <Field>
                        <FieldRow>
                            <FieldLabel htmlFor={privateRadioField}>{t('Private')}</FieldLabel>
                            <Controller
                                name='scope'
                                control={control}
                                render={({ field: { onChange, value, ...field } }) => (
                                    <RadioButton
                                        {...field}
                                        id={privateRadioField}
                                        onChange={() => onChange('user')}
                                        checked={value === 'user'}
                                        aria-describedby={`${privateRadioField}-hint`}
                                    />
                                )}
                            />
                        </FieldRow>
                        <FieldHint id={`${privateRadioField}-hint`}>{t('Canned_Response_Sharing_Private_Description')}</FieldHint>
                    </Field>
                    {scope === 'department' && (
                        <Field>
                            <FieldLabel htmlFor={departmentField}>{t('Department')}</FieldLabel>
                            <FieldRow>
                                <Controller
                                    name='departmentId'
                                    control={control}
                                    rules={{ required: t('The_field_is_required', t('Department')) }}
                                    render={({ field: { value, onChange } }) => (
                                        <AutoCompleteDepartment
                                            {...(hasMonitorPermission && { onlyMyDepartments: hasMonitorPermission })}
                                            id={departmentField}
                                            value={value}
                                            onChange={onChange}
                                            aria-describedby={`${departmentField}-error`}
                                            aria-required={true}
                                            aria-invalid={Boolean(errors?.departmentId)}
                                            withTitle={false}
                                            renderItem={({ label, ...props }) => (
                                                <Option {...props} label={<span style={{ whiteSpace: 'normal' }}>{label}</span>} />
                                            )}
                                        />
                                    )}
                                />
                            </FieldRow>
                            {errors?.departmentId && (
                                <FieldError aria-live='assertive' id={departmentField}>
                                    {errors.departmentId.message}
                                </FieldError>
                            )}
                        </Field>
                    )}
                </>
            )}
        </FieldGroup>
    );
};

export default CannedResponseForm;