- Added `activeKey` prop to `CustomCollapse` for better control over active panels. - Updated styles in `CustomCollapse` for improved visual consistency. - Refactored `KnowledgeContent` to include expand/collapse functionality for file and directory sections, enhancing user experience. - Added translations for "collapse" in multiple languages. - Minor style adjustments across various components for better UI consistency.
279 lines
8.5 KiB
TypeScript
279 lines
8.5 KiB
TypeScript
import {
|
|
DeleteOutlined,
|
|
EditOutlined,
|
|
ExclamationCircleOutlined,
|
|
FileImageOutlined,
|
|
FilePdfOutlined,
|
|
FileTextOutlined,
|
|
SortAscendingOutlined,
|
|
SortDescendingOutlined
|
|
} from '@ant-design/icons'
|
|
import { Navbar, NavbarCenter } from '@renderer/components/app/Navbar'
|
|
import TextEditPopup from '@renderer/components/Popups/TextEditPopup'
|
|
import db from '@renderer/databases'
|
|
import { useProviders } from '@renderer/hooks/useProvider'
|
|
import FileManager from '@renderer/services/FileManager'
|
|
import store from '@renderer/store'
|
|
import { FileType, FileTypes } from '@renderer/types'
|
|
import { formatFileSize } from '@renderer/utils'
|
|
import type { MenuProps } from 'antd'
|
|
import { Button, Empty, Flex, Menu, Popconfirm } from 'antd'
|
|
import dayjs from 'dayjs'
|
|
import { useLiveQuery } from 'dexie-react-hooks'
|
|
import { FC, useState } from 'react'
|
|
import { useTranslation } from 'react-i18next'
|
|
import styled from 'styled-components'
|
|
|
|
import FileList from './FileList'
|
|
|
|
type SortField = 'created_at' | 'size' | 'name'
|
|
type SortOrder = 'asc' | 'desc'
|
|
|
|
const FilesPage: FC = () => {
|
|
const { t } = useTranslation()
|
|
const [fileType, setFileType] = useState<string>('document')
|
|
const [sortField, setSortField] = useState<SortField>('created_at')
|
|
const [sortOrder, setSortOrder] = useState<SortOrder>('desc')
|
|
const { providers } = useProviders()
|
|
|
|
const geminiProviders = providers.filter((provider) => provider.type === 'gemini')
|
|
|
|
const tempFilesSort = (files: FileType[]) => {
|
|
return files.sort((a, b) => {
|
|
const aIsTemp = a.origin_name.startsWith('temp_file')
|
|
const bIsTemp = b.origin_name.startsWith('temp_file')
|
|
if (aIsTemp && !bIsTemp) return 1
|
|
if (!aIsTemp && bIsTemp) return -1
|
|
return 0
|
|
})
|
|
}
|
|
|
|
const sortFiles = (files: FileType[]) => {
|
|
return [...files].sort((a, b) => {
|
|
let comparison = 0
|
|
switch (sortField) {
|
|
case 'created_at':
|
|
comparison = dayjs(a.created_at).unix() - dayjs(b.created_at).unix()
|
|
break
|
|
case 'size':
|
|
comparison = a.size - b.size
|
|
break
|
|
case 'name':
|
|
comparison = a.origin_name.localeCompare(b.origin_name)
|
|
break
|
|
}
|
|
return sortOrder === 'asc' ? comparison : -comparison
|
|
})
|
|
}
|
|
|
|
const files = useLiveQuery<FileType[]>(() => {
|
|
if (fileType === 'all') {
|
|
return db.files.orderBy('count').toArray().then(tempFilesSort)
|
|
}
|
|
return db.files.where('type').equals(fileType).sortBy('count').then(tempFilesSort)
|
|
}, [fileType])
|
|
|
|
const sortedFiles = files ? sortFiles(files) : []
|
|
|
|
const handleDelete = async (fileId: string) => {
|
|
const file = await FileManager.getFile(fileId)
|
|
|
|
const paintings = await store.getState().paintings.paintings
|
|
const paintingsFiles = paintings.flatMap((p) => p.files)
|
|
|
|
if (paintingsFiles.some((p) => p.id === fileId)) {
|
|
window.modal.warning({ content: t('files.delete.paintings.warning'), centered: true })
|
|
return
|
|
}
|
|
|
|
if (file) {
|
|
await FileManager.deleteFile(fileId, true)
|
|
}
|
|
|
|
const topics = await db.topics
|
|
.filter((topic) => topic.messages.some((message) => message.files?.some((f) => f.id === fileId)))
|
|
.toArray()
|
|
|
|
if (topics.length > 0) {
|
|
for (const topic of topics) {
|
|
const updatedMessages = topic.messages.map((message) => ({
|
|
...message,
|
|
files: message.files?.filter((f) => f.id !== fileId)
|
|
}))
|
|
await db.topics.update(topic.id, { messages: updatedMessages })
|
|
}
|
|
}
|
|
}
|
|
|
|
const handleRename = async (fileId: string) => {
|
|
const file = await FileManager.getFile(fileId)
|
|
if (file) {
|
|
const newName = await TextEditPopup.show({ text: file.origin_name })
|
|
if (newName) {
|
|
FileManager.updateFile({ ...file, origin_name: newName })
|
|
}
|
|
}
|
|
}
|
|
|
|
const dataSource = sortedFiles?.map((file) => {
|
|
return {
|
|
key: file.id,
|
|
file: <span onClick={() => window.api.file.openPath(file.path)}>{FileManager.formatFileName(file)}</span>,
|
|
size: formatFileSize(file.size),
|
|
size_bytes: file.size,
|
|
count: file.count,
|
|
path: file.path,
|
|
ext: file.ext,
|
|
created_at: dayjs(file.created_at).format('MM-DD HH:mm'),
|
|
created_at_unix: dayjs(file.created_at).unix(),
|
|
actions: (
|
|
<Flex align="center" gap={0} style={{ opacity: 0.7 }}>
|
|
<Button type="text" icon={<EditOutlined />} onClick={() => handleRename(file.id)} />
|
|
<Popconfirm
|
|
title={t('files.delete.title')}
|
|
description={t('files.delete.content')}
|
|
okText={t('common.confirm')}
|
|
cancelText={t('common.cancel')}
|
|
onConfirm={() => handleDelete(file.id)}
|
|
icon={<ExclamationCircleOutlined style={{ color: 'red' }} />}>
|
|
<Button type="text" danger icon={<DeleteOutlined />} />
|
|
</Popconfirm>
|
|
</Flex>
|
|
)
|
|
}
|
|
})
|
|
|
|
const menuItems = [
|
|
{ key: FileTypes.DOCUMENT, label: t('files.document'), icon: <FilePdfOutlined /> },
|
|
{ key: FileTypes.IMAGE, label: t('files.image'), icon: <FileImageOutlined /> },
|
|
{ key: FileTypes.TEXT, label: t('files.text'), icon: <FileTextOutlined /> },
|
|
...geminiProviders.map((provider) => ({
|
|
key: 'gemini_' + provider.id,
|
|
label: provider.name,
|
|
icon: <FilePdfOutlined />
|
|
})),
|
|
{ key: 'all', label: t('files.all'), icon: <FileTextOutlined /> }
|
|
].filter(Boolean) as MenuProps['items']
|
|
|
|
return (
|
|
<Container>
|
|
<Navbar>
|
|
<NavbarCenter style={{ borderRight: 'none' }}>{t('files.title')}</NavbarCenter>
|
|
</Navbar>
|
|
<ContentContainer id="content-container">
|
|
<SideNav>
|
|
<Menu selectedKeys={[fileType]} items={menuItems} onSelect={({ key }) => setFileType(key as FileTypes)} />
|
|
</SideNav>
|
|
<MainContent>
|
|
<SortContainer>
|
|
{['created_at', 'size', 'name'].map((field) => (
|
|
<SortButton
|
|
key={field}
|
|
active={sortField === field}
|
|
onClick={() => {
|
|
if (sortField === field) {
|
|
setSortOrder(sortOrder === 'asc' ? 'desc' : 'asc')
|
|
} else {
|
|
setSortField(field as 'created_at' | 'size' | 'name')
|
|
setSortOrder('desc')
|
|
}
|
|
}}>
|
|
{t(`files.${field}`)}
|
|
{sortField === field && (sortOrder === 'desc' ? <SortDescendingOutlined /> : <SortAscendingOutlined />)}
|
|
</SortButton>
|
|
))}
|
|
</SortContainer>
|
|
{dataSource && dataSource?.length > 0 ? (
|
|
<FileList id={fileType} list={dataSource} files={sortedFiles} />
|
|
) : (
|
|
<Empty />
|
|
)}
|
|
</MainContent>
|
|
</ContentContainer>
|
|
</Container>
|
|
)
|
|
}
|
|
|
|
const Container = styled.div`
|
|
display: flex;
|
|
flex: 1;
|
|
flex-direction: column;
|
|
height: calc(100vh - var(--navbar-height));
|
|
`
|
|
|
|
const MainContent = styled.div`
|
|
display: flex;
|
|
flex: 1;
|
|
flex-direction: column;
|
|
`
|
|
|
|
const SortContainer = styled.div`
|
|
display: flex;
|
|
align-items: center;
|
|
gap: 8px;
|
|
padding: 8px 16px;
|
|
border-bottom: 0.5px solid var(--color-border);
|
|
`
|
|
|
|
const ContentContainer = styled.div`
|
|
display: flex;
|
|
flex: 1;
|
|
flex-direction: row;
|
|
min-height: 100%;
|
|
`
|
|
|
|
const SideNav = styled.div`
|
|
width: var(--settings-width);
|
|
border-right: 0.5px solid var(--color-border);
|
|
padding: 7px 12px;
|
|
user-select: none;
|
|
|
|
.ant-menu {
|
|
border-inline-end: none !important;
|
|
background: transparent;
|
|
}
|
|
|
|
.ant-menu-item {
|
|
height: 36px;
|
|
line-height: 36px;
|
|
margin: 4px 0;
|
|
width: 100%;
|
|
border-radius: var(--list-item-border-radius);
|
|
border: 0.5px solid transparent;
|
|
|
|
&:hover {
|
|
background-color: var(--color-background-soft) !important;
|
|
}
|
|
|
|
&.ant-menu-item-selected {
|
|
background-color: var(--color-background-soft);
|
|
color: var(--color-primary);
|
|
border: 0.5px solid var(--color-border);
|
|
color: var(--color-text);
|
|
}
|
|
}
|
|
`
|
|
|
|
const SortButton = styled(Button)<{ active?: boolean }>`
|
|
display: flex;
|
|
align-items: center;
|
|
gap: 4px;
|
|
padding: 4px 12px;
|
|
height: 30px;
|
|
border-radius: var(--list-item-border-radius);
|
|
border: 0.5px solid ${(props) => (props.active ? 'var(--color-border)' : 'transparent')};
|
|
background-color: ${(props) => (props.active ? 'var(--color-background-soft)' : 'transparent')};
|
|
color: ${(props) => (props.active ? 'var(--color-text)' : 'var(--color-text-secondary)')};
|
|
|
|
&:hover {
|
|
background-color: var(--color-background-soft);
|
|
color: var(--color-text);
|
|
}
|
|
|
|
.anticon {
|
|
font-size: 12px;
|
|
}
|
|
`
|
|
|
|
export default FilesPage
|