- Deleted redundant components (`AddActionButton`, `AlertBox`, `AlertStack`, `BackButton`, `CancelButton`, and `CollapsableArea`) and related files. - Removed unused models (`Book`, `BookSerie`, `BookTables`, `Character`, and `Chapter`) to reduce codebase clutter. - Updated project structure and references to reflect these removals.
186 lines
7.3 KiB
TypeScript
186 lines
7.3 KiB
TypeScript
'use client';
|
|
import React, {useCallback, useContext, useMemo, useState} from 'react';
|
|
import {useCharacters, UseCharactersConfig} from '@/hooks/settings/useCharacters';
|
|
import {useTranslations} from '@/lib/i18n';
|
|
import {CharacterProps} from '@/lib/types/character';
|
|
import {SeriesCharacterProps} from '@/lib/types/series';
|
|
import {BookContext, BookContextProps} from '@/context/BookContext';
|
|
import PulseLoader from '@/components/ui/PulseLoader';
|
|
|
|
import ToolDetailHeader from '@/components/book/settings/ToolDetailHeader';
|
|
import SeriesImportSelector from '@/components/form/SeriesImportSelector';
|
|
import AlertBox from '@/components/ui/AlertBox';
|
|
|
|
import CharacterEditorList from './CharacterEditorList';
|
|
import CharacterEditorDetail from './CharacterEditorDetail';
|
|
import CharacterEditorEdit from './CharacterEditorEdit';
|
|
|
|
/**
|
|
* CharacterEditor - Orchestrateur pour ComposerRightBar
|
|
* Mêmes fonctionnalités que CharacterSettings, layout condensé
|
|
*/
|
|
export default function CharacterEditor(): React.JSX.Element {
|
|
const t = useTranslations();
|
|
const {book}: BookContextProps = useContext<BookContextProps>(BookContext);
|
|
const [showDeleteConfirm, setShowDeleteConfirm] = useState<boolean>(false);
|
|
|
|
const config: UseCharactersConfig = useMemo(function (): UseCharactersConfig {
|
|
return {
|
|
entityType: 'book',
|
|
entityId: book?.bookId || '',
|
|
};
|
|
}, [book?.bookId]);
|
|
|
|
const {
|
|
characters,
|
|
seriesCharacters,
|
|
selectedCharacter,
|
|
toolEnabled,
|
|
isLoading,
|
|
bookSeriesId,
|
|
viewMode,
|
|
saveCharacter,
|
|
deleteCharacter,
|
|
updateCharacterField,
|
|
addAttribute,
|
|
removeAttribute,
|
|
toggleTool,
|
|
importFromSeries,
|
|
exportToSeries,
|
|
refreshSeriesCharacters,
|
|
setSelectedCharacter,
|
|
enterDetailMode,
|
|
enterEditMode,
|
|
exitEditMode,
|
|
backToList,
|
|
addNewCharacter,
|
|
} = useCharacters(config);
|
|
|
|
const availableSeriesCharacters = useMemo(function (): SeriesCharacterProps[] {
|
|
return seriesCharacters.filter(function (sc: SeriesCharacterProps): boolean {
|
|
return !characters.some(function (c: CharacterProps): boolean {
|
|
return c.seriesCharacterId === sc.id;
|
|
});
|
|
});
|
|
}, [seriesCharacters, characters]);
|
|
|
|
const handleCharacterChange = useCallback(function (key: keyof CharacterProps, value: string | number | null): void {
|
|
updateCharacterField(key, value);
|
|
}, [updateCharacterField]);
|
|
|
|
async function handleSave(): Promise<void> {
|
|
await exitEditMode(true);
|
|
}
|
|
|
|
function handleCancel(): void {
|
|
exitEditMode(false);
|
|
}
|
|
|
|
async function handleDelete(): Promise<void> {
|
|
if (selectedCharacter?.id) {
|
|
await deleteCharacter(selectedCharacter.id);
|
|
setShowDeleteConfirm(false);
|
|
backToList();
|
|
}
|
|
}
|
|
|
|
function getSeriesCharacterForSelected(): SeriesCharacterProps | null {
|
|
if (!selectedCharacter?.seriesCharacterId) return null;
|
|
return seriesCharacters.find(function (sc: SeriesCharacterProps): boolean {
|
|
return sc.id === selectedCharacter.seriesCharacterId;
|
|
}) || null;
|
|
}
|
|
|
|
if (isLoading) {
|
|
return <PulseLoader size="sm"/>;
|
|
}
|
|
|
|
const isNew: boolean = selectedCharacter?.id === null;
|
|
const canExport: boolean = Boolean(bookSeriesId && selectedCharacter?.id && !selectedCharacter.seriesCharacterId);
|
|
|
|
return (
|
|
<div className="flex flex-col h-full">
|
|
<ToolDetailHeader
|
|
title={selectedCharacter?.name || ''}
|
|
defaultTitle={t('characterDetail.newCharacter')}
|
|
viewMode={viewMode}
|
|
isNew={isNew}
|
|
onBack={backToList}
|
|
onEdit={enterEditMode}
|
|
onSave={handleSave}
|
|
onCancel={handleCancel}
|
|
onDelete={function (): void {
|
|
setShowDeleteConfirm(true);
|
|
}}
|
|
onExport={canExport ? exportToSeries : undefined}
|
|
showExport={canExport}
|
|
showDelete={Boolean(selectedCharacter?.id)}
|
|
/>
|
|
|
|
<div className="flex-1 overflow-y-auto">
|
|
{viewMode === 'list' && (
|
|
<div className="space-y-3 p-2">
|
|
{/* Import from series */}
|
|
{bookSeriesId && availableSeriesCharacters.length > 0 && (
|
|
<SeriesImportSelector
|
|
availableItems={availableSeriesCharacters.map(function (sc: SeriesCharacterProps) {
|
|
return {
|
|
id: sc.id,
|
|
name: `${sc.name}${sc.lastName ? ' ' + sc.lastName : ''}`
|
|
};
|
|
})}
|
|
onImport={importFromSeries}
|
|
placeholder={t('seriesImport.selectElement')}
|
|
label={t('seriesImport.importFromSeries')}
|
|
/>
|
|
)}
|
|
|
|
<CharacterEditorList
|
|
characters={characters}
|
|
onCharacterClick={enterDetailMode}
|
|
onAddCharacter={addNewCharacter}
|
|
/>
|
|
</div>
|
|
)}
|
|
|
|
{viewMode === 'detail' && selectedCharacter && (
|
|
<div className="p-4">
|
|
<CharacterEditorDetail
|
|
character={selectedCharacter}
|
|
seriesCharacter={getSeriesCharacterForSelected()}
|
|
/>
|
|
</div>
|
|
)}
|
|
|
|
{viewMode === 'edit' && selectedCharacter && (
|
|
<div className="p-4">
|
|
<CharacterEditorEdit
|
|
character={selectedCharacter}
|
|
setCharacter={setSelectedCharacter}
|
|
onCharacterChange={handleCharacterChange}
|
|
onAddAttribute={addAttribute}
|
|
onRemoveAttribute={removeAttribute}
|
|
seriesCharacter={getSeriesCharacterForSelected()}
|
|
onSyncComplete={refreshSeriesCharacters}
|
|
/>
|
|
</div>
|
|
)}
|
|
</div>
|
|
|
|
{showDeleteConfirm && selectedCharacter?.id && (
|
|
<AlertBox
|
|
title={t('characterDetail.deleteTitle')}
|
|
message={t('characterDetail.deleteMessage', {name: selectedCharacter.name})}
|
|
type="danger"
|
|
confirmText={t('common.delete')}
|
|
cancelText={t('common.cancel')}
|
|
onConfirm={handleDelete}
|
|
onCancel={function (): void {
|
|
setShowDeleteConfirm(false);
|
|
}}
|
|
/>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|