Files
tonePage/tone-page-web/app/console/(with-menu)/storage/page.tsx
2025-06-10 14:57:52 +08:00

197 lines
7.9 KiB
TypeScript
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
'use client';
import { Button } from '@/components/ui/button';
import { Checkbox } from '@/components/ui/checkbox';
import { Table, TableBody, TableCaption, TableCell, TableHead, TableHeader, TableRow } from '@/components/ui/table';
import { useOssStore } from '@/hooks/admin/web/blog/use-oss-store';
import { ObjectMeta } from 'ali-oss';
import { Delete, Download, Edit, RefreshCcw, Upload } from 'lucide-react';
import { useEffect, useMemo, useState } from 'react';
import { toast } from 'sonner';
import {
DropdownMenu,
DropdownMenuContent,
DropdownMenuItem,
DropdownMenuTrigger,
} from "@/components/ui/dropdown-menu"
import { UploadManager } from './components/UploadManager';
const formatSizeNumber = (n: number) => {
const unit = ['B', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB'];
for (const [i, u] of unit.entries()) {
if (n < 1024 ** (i + 1)) {
if (i <= 0) {
return `${(n)}${unit[i]}`;
} else {
return `${(n / 1024 ** i).toFixed(2)}${unit[i]}`;
}
}
}
}
interface FileItem {
id: string;
name: string;
size: number;// Byte
lastModified: Date;
isChecked: boolean;
};
export default function Page() {
const { stsTokenData, isLoading, error, store } = useOssStore();
const [fileList, setFileList] = useState<null | FileItem[]>(null);
const handleRefreshFileList = async () => {
if (!store || !stsTokenData) return toast.error('初始化失败,请刷新界面重试');
setFileList(null);
const res = await store.listV2({ prefix: `tone-page/${stsTokenData.userId}` }, {}).catch(e => {
toast.error('文件列表加载失败');
});
if (res && res.objects) {
setFileList(res.objects.map(v => ({
id: v.name,
name: v.name.replace(`tone-page/${stsTokenData.userId}/`, ''),
size: v.size,
lastModified: new Date(v.lastModified),
isChecked: false,
})));
}
}
const handleCheckboxChange = (id: string, checked: boolean) => {
setFileList((prevFileList) => {
if (!prevFileList) return null;
return prevFileList.map(item => {
if (item.id === id) {
return { ...item, isChecked: checked };
}
return item;
})
})
}
useEffect(() => {
store && stsTokenData && handleRefreshFileList();
}, [stsTokenData]);
const checkedFileIds = useMemo(() => {
if (!fileList) return null;
return fileList.filter(i => i.isChecked).map(i => i.id);
}, [fileList])
const handleDeleteCheckedFiles = async () => {
if (!store || !stsTokenData) return toast.error('初始化未完成或失败,请等待或重新加载');
const deleteFiles = (fileList || []).filter(i => i.isChecked);
if (!deleteFiles) return toast.error('请选择需要删除的文件');
let failedCount = 0;
for (let item of deleteFiles) {
await deleteFile(item.name).catch(e => { failedCount++; return e; });
}
if (failedCount > 0) {
toast.warning(`删除完成,共有${failedCount}个文件删除失败`)
} else {
toast.success(`${deleteFiles.length}个文件删除完成`)
}
handleRefreshFileList();
}
const handleDeleteFile = async (fileItem: FileItem) => {
await deleteFile(fileItem.name)
.then(() => toast.success('删除完成'))
.catch(() => toast.error('删除失败'));
handleRefreshFileList();
}
const deleteFile = async (localFilename: string) => {
if (!store) throw new Error('store未初始化');
if (!stsTokenData) throw new Error('sts服务未初始化');
return store.delete(`/tone-page/${stsTokenData.userId}/${localFilename}`);
}
const downloadFile = (localFilename: string) => {
if (!store) throw new Error('store未初始化');
if (!stsTokenData) throw new Error('sts服务未初始化');
const url = store.signatureUrl(`/tone-page/${stsTokenData.userId}/${localFilename}`);
const a = document.createElement('a');
a.href = url;
a.download = localFilename;
document.body.appendChild(a);
a.click();
document.body.removeChild(a);
}
const handleDownloadFile = async (fileItem: FileItem) => {
downloadFile(fileItem.name);
}
return (
<div>
<div className='mt-1 flex gap-2'>
<UploadManager
store={store}
basePath={stsTokenData && `/tone-page/${stsTokenData.userId}`}
handleRefreshFileList={handleRefreshFileList}
>
<Button variant='default' className='cursor-pointer'><Upload /></Button>
</UploadManager>
<Button
variant='destructive'
className='cursor-pointer'
disabled={(checkedFileIds?.length || 0) <= 0}
onClick={() => handleDeleteCheckedFiles()}
><Delete /></Button>
<div className='flex items-center'>
<Button variant='secondary' size='icon' className='cursor-pointer' onClick={() => handleRefreshFileList()}><RefreshCcw /></Button>
<div className='text-sm ml-2'> {fileList?.length ?? 0} 100</div>
</div>
</div>
<Table>
<TableCaption>
{(isLoading || (fileList == null && !error)) && <div>...</div>}
{error && <div>{`${error}`}</div>}
{fileList && fileList.length === 0 && <div></div>}
</TableCaption>
<TableHeader>
<TableRow>
<TableHead className='w-10'></TableHead>
<TableHead></TableHead>
<TableHead></TableHead>
<TableHead></TableHead>
</TableRow>
</TableHeader>
<TableBody>
{
fileList && fileList.map(d => (
<TableRow key={d.name} >
<TableCell>
<Checkbox checked={d.isChecked} onCheckedChange={v => handleCheckboxChange(d.id, Boolean(v))} />
</TableCell>
<TableCell>
<DropdownMenu>
<DropdownMenuTrigger className='whitespace-normal break-all cursor-pointer text-left'>
{d.name}
</DropdownMenuTrigger>
<DropdownMenuContent>
<DropdownMenuItem onClick={() => handleDownloadFile(d)}><Download /></DropdownMenuItem>
{/* <DropdownMenuItem><Edit />编辑</DropdownMenuItem> */}
<DropdownMenuItem onClick={() => handleDeleteFile(d)}><Delete /></DropdownMenuItem>
</DropdownMenuContent>
</DropdownMenu>
</TableCell>
<TableCell>{formatSizeNumber(d.size)}</TableCell>
<TableCell className='whitespace-normal break-words'>{d.lastModified.toLocaleString()}</TableCell>
</TableRow>
))
}
</TableBody>
</Table>
</div >
)
}