Files
AnthoLume/frontend/src/pages/AdminImportPage.tsx
2026-03-22 17:21:34 -04:00

174 lines
6.1 KiB
TypeScript

import { useState } from 'react';
import { useGetImportDirectory, usePostImport } from '../generated/anthoLumeAPIV1';
import type { DirectoryItem, DirectoryListResponse } from '../generated/model';
import { getErrorMessage } from '../utils/errors';
import { Button } from '../components/Button';
import { FolderOpenIcon } from '../icons';
import { useToasts } from '../components/ToastContext';
export default function AdminImportPage() {
const [currentPath, setCurrentPath] = useState<string>('');
const [selectedDirectory, setSelectedDirectory] = useState<string>('');
const [importType, setImportType] = useState<'DIRECT' | 'COPY'>('DIRECT');
const { showInfo, showError } = useToasts();
const { data: directoryData, isLoading } = useGetImportDirectory(
currentPath ? { directory: currentPath } : {}
);
const postImport = usePostImport();
const directoryResponse =
directoryData?.status === 200 ? (directoryData.data as DirectoryListResponse) : null;
const directories = directoryResponse?.items ?? [];
const currentPathDisplay = directoryResponse?.current_path ?? currentPath ?? '/data';
const handleSelectDirectory = (directory: string) => {
setSelectedDirectory(`${currentPath}/${directory}`);
};
const handleNavigateUp = () => {
if (currentPathDisplay !== '/') {
const parts = currentPathDisplay.split('/');
parts.pop();
setCurrentPath(parts.join('/') || '');
}
};
const handleImport = () => {
if (!selectedDirectory) return;
postImport.mutate(
{
data: {
directory: selectedDirectory,
type: importType,
},
},
{
onSuccess: _response => {
showInfo('Import completed successfully');
setTimeout(() => {
window.location.href = '/admin/import-results';
}, 1500);
},
onError: error => {
showError('Import failed: ' + getErrorMessage(error));
},
}
);
};
const handleCancel = () => {
setSelectedDirectory('');
};
if (isLoading && !currentPath) {
return <div className="text-content-muted">Loading...</div>;
}
if (selectedDirectory) {
return (
<div className="overflow-x-auto">
<div className="inline-block min-w-full overflow-hidden rounded shadow">
<div className="flex grow flex-col gap-2 rounded bg-surface p-4 text-content-muted shadow-lg">
<p className="text-lg font-semibold text-content">Selected Import Directory</p>
<form className="flex flex-col gap-4" onSubmit={handleImport}>
<div className="flex w-full justify-between gap-4">
<div className="flex items-center gap-4 text-content">
<FolderOpenIcon size={20} />
<p className="break-all text-lg font-medium">{selectedDirectory}</p>
</div>
<div className="mr-4 flex flex-col justify-around gap-2 text-content">
<div className="inline-flex items-center gap-2">
<input
type="radio"
id="direct"
checked={importType === 'DIRECT'}
onChange={() => setImportType('DIRECT')}
/>
<label htmlFor="direct">Direct</label>
</div>
<div className="inline-flex items-center gap-2">
<input
type="radio"
id="copy"
checked={importType === 'COPY'}
onChange={() => setImportType('COPY')}
/>
<label htmlFor="copy">Copy</label>
</div>
</div>
</div>
<div className="flex gap-4">
<Button type="submit" className="px-10 py-2 text-base">
Import Directory
</Button>
<Button
type="button"
variant="secondary"
onClick={handleCancel}
className="px-10 py-2 text-base"
>
Cancel
</Button>
</div>
</form>
</div>
</div>
</div>
);
}
return (
<div className="overflow-x-auto">
<div className="inline-block min-w-full overflow-hidden rounded shadow">
<table className="min-w-full bg-surface text-sm leading-normal text-content">
<thead className="text-content-muted">
<tr>
<th className="w-12 border-b border-border p-3 text-left font-normal"></th>
<th className="break-all border-b border-border p-3 text-left font-normal">
{currentPath}
</th>
</tr>
</thead>
<tbody>
{currentPath !== '/' && (
<tr>
<td className="border-b border-border p-3 text-content-muted"></td>
<td className="border-b border-border p-3">
<button onClick={handleNavigateUp}>
<p>../</p>
</button>
</td>
</tr>
)}
{directories.length === 0 ? (
<tr>
<td className="p-3 text-center" colSpan={2}>
No Folders
</td>
</tr>
) : (
directories.map((item: DirectoryItem) => (
<tr key={item.name}>
<td className="border-b border-border p-3 text-content-muted">
<button onClick={() => item.name && handleSelectDirectory(item.name)}>
<FolderOpenIcon size={20} />
</button>
</td>
<td className="border-b border-border p-3">
<button onClick={() => item.name && handleSelectDirectory(item.name)}>
<p>{item.name ?? ''}</p>
</button>
</td>
</tr>
))
)}
</tbody>
</table>
</div>
</div>
);
}