mirror of
https://github.com/IRBorisov/ConceptPortal.git
synced 2025-06-26 13:00:39 +03:00
256 lines
8.5 KiB
TypeScript
256 lines
8.5 KiB
TypeScript
'use client';
|
||
|
||
import { zodResolver } from '@hookform/resolvers/zod';
|
||
import clsx from 'clsx';
|
||
import { useRef } from 'react';
|
||
import { Controller, useForm, useWatch } from 'react-hook-form';
|
||
|
||
import { useConceptNavigation } from '@/app/Navigation/NavigationContext';
|
||
import { urls } from '@/app/urls';
|
||
import { useAuthSuspense } from '@/backend/auth/useAuth';
|
||
import { CreateLibraryItemSchema, ICreateLibraryItemDTO } from '@/backend/library/api';
|
||
import { useCreateItem } from '@/backend/library/useCreateItem';
|
||
import { VisibilityIcon } from '@/components/DomainIcons';
|
||
import { IconDownload } from '@/components/Icons';
|
||
import InfoError from '@/components/info/InfoError';
|
||
import SelectAccessPolicy from '@/components/select/SelectAccessPolicy';
|
||
import SelectItemType from '@/components/select/SelectItemType';
|
||
import SelectLocationContext from '@/components/select/SelectLocationContext';
|
||
import SelectLocationHead from '@/components/select/SelectLocationHead';
|
||
import Button from '@/components/ui/Button';
|
||
import Label from '@/components/ui/Label';
|
||
import MiniButton from '@/components/ui/MiniButton';
|
||
import Overlay from '@/components/ui/Overlay';
|
||
import SubmitButton from '@/components/ui/SubmitButton';
|
||
import TextArea from '@/components/ui/TextArea';
|
||
import TextInput from '@/components/ui/TextInput';
|
||
import { AccessPolicy, LibraryItemType, LocationHead } from '@/models/library';
|
||
import { combineLocation } from '@/models/libraryAPI';
|
||
import { useLibrarySearchStore } from '@/stores/librarySearch';
|
||
import { EXTEOR_TRS_FILE } from '@/utils/constants';
|
||
|
||
function FormCreateItem() {
|
||
const router = useConceptNavigation();
|
||
const { user } = useAuthSuspense();
|
||
const { createItem, isPending, error: serverError, reset: clearServerError } = useCreateItem();
|
||
|
||
const searchLocation = useLibrarySearchStore(state => state.location);
|
||
const setSearchLocation = useLibrarySearchStore(state => state.setLocation);
|
||
|
||
const {
|
||
register,
|
||
handleSubmit,
|
||
clearErrors,
|
||
setValue,
|
||
control,
|
||
formState: { errors }
|
||
} = useForm<ICreateLibraryItemDTO>({
|
||
resolver: zodResolver(CreateLibraryItemSchema),
|
||
defaultValues: {
|
||
item_type: LibraryItemType.RSFORM,
|
||
access_policy: AccessPolicy.PUBLIC,
|
||
visible: true,
|
||
read_only: false,
|
||
location: !!searchLocation ? searchLocation : LocationHead.USER
|
||
}
|
||
});
|
||
const itemType = useWatch({ control, name: 'item_type' });
|
||
const file = useWatch({ control, name: 'file' });
|
||
const inputRef = useRef<HTMLInputElement | null>(null);
|
||
|
||
function resetErrors() {
|
||
clearServerError();
|
||
clearErrors();
|
||
}
|
||
|
||
function handleCancel() {
|
||
if (router.canBack()) {
|
||
router.back();
|
||
} else {
|
||
router.push(urls.library);
|
||
}
|
||
}
|
||
|
||
function handleFileChange(event: React.ChangeEvent<HTMLInputElement>) {
|
||
if (event.target.files && event.target.files.length > 0) {
|
||
setValue('file', event.target.files[0]);
|
||
setValue('fileName', event.target.files[0].name);
|
||
} else {
|
||
setValue('file', undefined);
|
||
setValue('fileName', '');
|
||
}
|
||
}
|
||
|
||
function handleItemTypeChange(value: LibraryItemType) {
|
||
if (value !== LibraryItemType.RSFORM) {
|
||
setValue('file', undefined);
|
||
setValue('fileName', '');
|
||
}
|
||
setValue('item_type', value);
|
||
}
|
||
|
||
function onSubmit(data: ICreateLibraryItemDTO) {
|
||
createItem(data, newItem => {
|
||
setSearchLocation(data.location);
|
||
if (newItem.item_type == LibraryItemType.RSFORM) {
|
||
router.push(urls.schema(newItem.id));
|
||
} else {
|
||
router.push(urls.oss(newItem.id));
|
||
}
|
||
});
|
||
}
|
||
|
||
return (
|
||
<form
|
||
className={clsx('cc-fade-in cc-column', 'min-w-[30rem] max-w-[30rem] mx-auto', 'px-6 py-3')}
|
||
onSubmit={event => void handleSubmit(onSubmit)(event)}
|
||
onChange={resetErrors}
|
||
>
|
||
<h1 className='select-none'>
|
||
{itemType == LibraryItemType.RSFORM ? (
|
||
<Overlay position='top-0 right-[0.5rem]'>
|
||
<Controller
|
||
control={control}
|
||
name='file'
|
||
render={() => (
|
||
<input
|
||
id='schema_file'
|
||
ref={inputRef}
|
||
type='file'
|
||
style={{ display: 'none' }}
|
||
accept={EXTEOR_TRS_FILE}
|
||
onChange={handleFileChange}
|
||
/>
|
||
)}
|
||
/>
|
||
<MiniButton
|
||
title='Загрузить из Экстеор'
|
||
icon={<IconDownload size='1.25rem' className='icon-primary' />}
|
||
onClick={() => inputRef.current?.click()}
|
||
/>
|
||
</Overlay>
|
||
) : null}
|
||
Создание схемы
|
||
</h1>
|
||
|
||
{file ? <Label className='text-wrap' text={`Загружен файл: ${file.name}`} /> : null}
|
||
|
||
<TextInput
|
||
id='schema_title'
|
||
{...register('title')}
|
||
label='Полное название'
|
||
placeholder={file && 'Загрузить из файла'}
|
||
error={errors.title}
|
||
/>
|
||
|
||
<div className='flex justify-between gap-3'>
|
||
<TextInput
|
||
id='schema_alias'
|
||
{...register('alias')}
|
||
label='Сокращение'
|
||
placeholder={file && 'Загрузить из файла'}
|
||
className='w-[16rem]'
|
||
error={errors.alias}
|
||
/>
|
||
<div className='flex flex-col items-center gap-2'>
|
||
<Label text='Тип схемы' className='self-center select-none' />
|
||
<Controller
|
||
control={control}
|
||
name='item_type'
|
||
render={({ field }) => (
|
||
<SelectItemType
|
||
value={field.value} //
|
||
onChange={handleItemTypeChange}
|
||
/>
|
||
)}
|
||
/>
|
||
</div>
|
||
|
||
<div className='flex flex-col gap-2'>
|
||
<Label text='Доступ' className='self-center select-none' />
|
||
<div className='ml-auto cc-icons'>
|
||
<Controller
|
||
control={control} //
|
||
name='access_policy'
|
||
render={({ field }) => (
|
||
<SelectAccessPolicy
|
||
value={field.value} //
|
||
onChange={field.onChange}
|
||
/>
|
||
)}
|
||
/>
|
||
<Controller
|
||
control={control}
|
||
name='visible'
|
||
render={({ field }) => (
|
||
<MiniButton
|
||
title={field.value ? 'Библиотека: отображать' : 'Библиотека: скрывать'}
|
||
icon={<VisibilityIcon value={field.value} />}
|
||
onClick={() => field.onChange(!field.value)}
|
||
/>
|
||
)}
|
||
/>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<TextArea
|
||
id='schema_comment'
|
||
{...register('comment')}
|
||
label='Описание'
|
||
placeholder={file && 'Загрузить из файла'}
|
||
error={errors.comment}
|
||
/>
|
||
|
||
<div className='flex justify-between gap-3 flex-grow'>
|
||
<div className='flex flex-col gap-2 min-w-[7rem] h-min'>
|
||
<Label text='Корень' />
|
||
<Controller
|
||
control={control} //
|
||
name='location'
|
||
render={({ field }) => (
|
||
<SelectLocationHead
|
||
value={field.value.substring(0, 2) as LocationHead}
|
||
onChange={newValue => field.onChange(combineLocation(newValue, field.value.substring(3)))}
|
||
excluded={!user.is_staff ? [LocationHead.LIBRARY] : []}
|
||
/>
|
||
)}
|
||
/>
|
||
</div>
|
||
<Controller
|
||
control={control} //
|
||
name='location'
|
||
render={({ field }) => (
|
||
<SelectLocationContext
|
||
value={field.value} //
|
||
onChange={field.onChange}
|
||
/>
|
||
)}
|
||
/>
|
||
<Controller
|
||
control={control} //
|
||
name='location'
|
||
render={({ field }) => (
|
||
<TextArea
|
||
id='dlg_cst_body'
|
||
label='Путь'
|
||
rows={4}
|
||
value={field.value.substring(3)}
|
||
onChange={event => field.onChange(combineLocation(field.value.substring(0, 2), event.target.value))}
|
||
error={errors.location}
|
||
/>
|
||
)}
|
||
/>
|
||
</div>
|
||
|
||
<div className='flex justify-around gap-6 py-3'>
|
||
<SubmitButton text='Создать схему' loading={isPending} className='min-w-[10rem]' />
|
||
<Button text='Отмена' className='min-w-[10rem]' onClick={() => handleCancel()} />
|
||
</div>
|
||
{serverError ? <InfoError error={serverError} /> : null}
|
||
</form>
|
||
);
|
||
}
|
||
|
||
export default FormCreateItem;
|