2022-10-21 13:14:17 +08:00
|
|
|
import type { AdminConsoleKey } from '@logto/phrases';
|
|
|
|
import type { Application } from '@logto/schemas';
|
|
|
|
import type { KeyboardEvent } from 'react';
|
|
|
|
import { useRef } from 'react';
|
2022-06-27 20:10:27 +08:00
|
|
|
import { Controller, FormProvider, useForm } from 'react-hook-form';
|
2022-06-29 10:26:04 +08:00
|
|
|
import { toast } from 'react-hot-toast';
|
2022-06-27 20:10:27 +08:00
|
|
|
import { useTranslation } from 'react-i18next';
|
|
|
|
import useSWR from 'swr';
|
|
|
|
|
|
|
|
import Button from '@/components/Button';
|
|
|
|
import FormField from '@/components/FormField';
|
|
|
|
import { convertRhfErrorMessage, createValidatorForRhf } from '@/components/MultiTextInput/utils';
|
2022-11-29 18:44:50 +08:00
|
|
|
import MultiTextInputField from '@/components/MultiTextInputField';
|
2022-06-27 20:10:27 +08:00
|
|
|
import TextInput from '@/components/TextInput';
|
2022-10-21 13:14:17 +08:00
|
|
|
import type { RequestError } from '@/hooks/use-api';
|
|
|
|
import useApi from '@/hooks/use-api';
|
|
|
|
import type { GuideForm } from '@/types/guide';
|
2023-02-15 10:26:22 +08:00
|
|
|
import { uriValidator } from '@/utils/validator';
|
2022-06-27 20:10:27 +08:00
|
|
|
|
|
|
|
import * as styles from './index.module.scss';
|
|
|
|
|
|
|
|
type Props = {
|
|
|
|
appId: string;
|
|
|
|
name: 'redirectUris' | 'postLogoutRedirectUris';
|
2022-07-11 17:28:51 +08:00
|
|
|
title: AdminConsoleKey;
|
2022-06-27 20:10:27 +08:00
|
|
|
isSingle?: boolean;
|
|
|
|
};
|
|
|
|
|
|
|
|
const UriInputField = ({ appId, name, title, isSingle = false }: Props) => {
|
|
|
|
const methods = useForm<Partial<GuideForm>>();
|
|
|
|
const {
|
|
|
|
control,
|
|
|
|
getValues,
|
|
|
|
handleSubmit,
|
|
|
|
reset,
|
|
|
|
formState: { isSubmitting },
|
|
|
|
} = methods;
|
|
|
|
|
2023-02-10 01:34:23 +08:00
|
|
|
const { data, mutate } = useSWR<Application, RequestError>(`api/applications/${appId}`);
|
2022-06-27 20:10:27 +08:00
|
|
|
|
|
|
|
const ref = useRef<HTMLDivElement>(null);
|
|
|
|
const { t } = useTranslation(undefined, { keyPrefix: 'admin_console' });
|
|
|
|
const api = useApi();
|
|
|
|
|
|
|
|
const onSubmit = async (value: string[]) => {
|
|
|
|
const updatedApp = await api
|
2023-02-10 01:34:23 +08:00
|
|
|
.patch(`api/applications/${appId}`, {
|
2022-06-27 20:10:27 +08:00
|
|
|
json: {
|
|
|
|
oidcClientMetadata: {
|
|
|
|
[name]: value.filter(Boolean),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
})
|
|
|
|
.json<Application>();
|
|
|
|
void mutate(updatedApp);
|
2022-06-29 10:26:04 +08:00
|
|
|
toast.success(t('general.saved'));
|
2022-06-27 20:10:27 +08:00
|
|
|
|
|
|
|
// Reset form to set 'isDirty' to false
|
|
|
|
reset(getValues());
|
|
|
|
};
|
|
|
|
|
|
|
|
const onKeyPress = (event: KeyboardEvent<HTMLInputElement>, value: string[]) => {
|
|
|
|
if (event.key === 'Enter') {
|
|
|
|
event.preventDefault();
|
|
|
|
void handleSubmit(async () => onSubmit(value))();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
return (
|
|
|
|
<FormProvider {...methods}>
|
|
|
|
<form>
|
2022-11-29 18:44:50 +08:00
|
|
|
<Controller
|
|
|
|
name={name}
|
|
|
|
control={control}
|
|
|
|
defaultValue={data?.oidcClientMetadata[name]}
|
|
|
|
rules={{
|
|
|
|
validate: createValidatorForRhf({
|
|
|
|
required: t(
|
|
|
|
isSingle ? 'errors.required_field_missing' : 'errors.required_field_missing_plural',
|
|
|
|
{ field: title }
|
|
|
|
),
|
|
|
|
pattern: {
|
|
|
|
verify: (value) => !value || uriValidator(value),
|
|
|
|
message: t('errors.invalid_uri_format'),
|
|
|
|
},
|
|
|
|
}),
|
|
|
|
}}
|
|
|
|
render={({ field: { onChange, value = [] }, fieldState: { error, isDirty } }) => {
|
|
|
|
const errorObject = convertRhfErrorMessage(error?.message);
|
2022-06-27 20:10:27 +08:00
|
|
|
|
2022-11-29 18:44:50 +08:00
|
|
|
return (
|
|
|
|
<div ref={ref} className={styles.wrapper}>
|
|
|
|
{isSingle && (
|
|
|
|
<FormField
|
|
|
|
isRequired={name === 'redirectUris'}
|
|
|
|
className={styles.field}
|
|
|
|
title={title}
|
|
|
|
>
|
2022-06-27 20:10:27 +08:00
|
|
|
<TextInput
|
|
|
|
className={styles.field}
|
|
|
|
value={value[0]}
|
|
|
|
errorMessage={errorObject?.required ?? errorObject?.inputs?.[0]}
|
|
|
|
onChange={({ currentTarget: { value } }) => {
|
|
|
|
onChange([value]);
|
|
|
|
}}
|
|
|
|
onKeyPress={(event) => {
|
|
|
|
onKeyPress(event, value);
|
|
|
|
}}
|
|
|
|
/>
|
2022-11-29 18:44:50 +08:00
|
|
|
</FormField>
|
|
|
|
)}
|
|
|
|
{!isSingle && (
|
|
|
|
<MultiTextInputField
|
|
|
|
isRequired={name === 'redirectUris'}
|
|
|
|
formFieldClassName={styles.field}
|
|
|
|
title={title}
|
|
|
|
value={value}
|
|
|
|
error={errorObject}
|
|
|
|
className={styles.multiTextInput}
|
|
|
|
onChange={onChange}
|
|
|
|
onKeyPress={(event) => {
|
|
|
|
onKeyPress(event, value);
|
|
|
|
}}
|
2022-06-27 20:10:27 +08:00
|
|
|
/>
|
2022-11-29 18:44:50 +08:00
|
|
|
)}
|
|
|
|
<Button
|
|
|
|
className={styles.saveButton}
|
|
|
|
disabled={!isDirty}
|
|
|
|
isLoading={isSubmitting}
|
|
|
|
title="general.save"
|
|
|
|
type="primary"
|
|
|
|
onClick={handleSubmit(async () => onSubmit(value))}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}}
|
|
|
|
/>
|
2022-06-27 20:10:27 +08:00
|
|
|
</form>
|
|
|
|
</FormProvider>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
export default UriInputField;
|