/* Copyright (C) 2025 QuantumNous This program is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more details. You should have received a copy of the GNU Affero General Public License along with this program. If not, see . For commercial licensing, please contact support@quantumnous.com */ import React, { useState, useEffect, useRef } from 'react'; import { useTranslation } from 'react-i18next'; import { Table, Card, Skeleton, Pagination, Empty, Button, Collapsible, } from '@douyinfe/semi-ui'; import { IconChevronDown, IconChevronUp } from '@douyinfe/semi-icons'; import PropTypes from 'prop-types'; import { useIsMobile } from '../../../hooks/common/useIsMobile'; import { useMinimumLoadingTime } from '../../../hooks/common/useMinimumLoadingTime'; /** * CardTable 响应式表格组件 * * 在桌面端渲染 Semi-UI 的 Table 组件,在移动端则将每一行数据渲染成 Card 形式。 * 该组件与 Table 组件的大部分 API 保持一致,只需将原 Table 换成 CardTable 即可。 */ const CardTable = ({ columns = [], dataSource = [], loading = false, rowKey = 'key', hidePagination = false, ...tableProps }) => { const isMobile = useIsMobile(); const { t } = useTranslation(); const showSkeleton = useMinimumLoadingTime(loading); const getRowKey = (record, index) => { if (typeof rowKey === 'function') return rowKey(record); return record[rowKey] !== undefined ? record[rowKey] : index; }; if (!isMobile) { const finalTableProps = hidePagination ? { ...tableProps, pagination: false } : tableProps; return ( ); } if (showSkeleton) { const visibleCols = columns.filter((col) => { if (tableProps?.visibleColumns && col.key) { return tableProps.visibleColumns[col.key]; } return true; }); const renderSkeletonCard = (key) => { const placeholder = (
{visibleCols.map((col, idx) => { if (!col.title) { return (
); } return (
); })}
); return ( ); }; return (
{[1, 2, 3].map((i) => renderSkeletonCard(i))}
); } const isEmpty = !showSkeleton && (!dataSource || dataSource.length === 0); const MobileRowCard = ({ record, index }) => { const [showDetails, setShowDetails] = useState(false); const rowKeyVal = getRowKey(record, index); const hasDetails = tableProps.expandedRowRender && (!tableProps.rowExpandable || tableProps.rowExpandable(record)); return ( {columns.map((col, colIdx) => { if ( tableProps?.visibleColumns && !tableProps.visibleColumns[col.key] ) { return null; } const title = col.title; const cellContent = col.render ? col.render(record[col.dataIndex], record, index) : record[col.dataIndex]; if (!title) { return (
{cellContent}
); } return (
{title}
{cellContent !== undefined && cellContent !== null ? cellContent : '-'}
); })} {hasDetails && ( <>
{tableProps.expandedRowRender(record, index)}
)}
); }; if (isEmpty) { if (tableProps.empty) return tableProps.empty; return (
); } return (
{dataSource.map((record, index) => ( ))} {!hidePagination && tableProps.pagination && dataSource.length > 0 && (
)}
); }; CardTable.propTypes = { columns: PropTypes.array.isRequired, dataSource: PropTypes.array, loading: PropTypes.bool, rowKey: PropTypes.oneOfType([PropTypes.string, PropTypes.func]), hidePagination: PropTypes.bool, }; export default CardTable;