mirror of
https://github.com/Syngnat/GoNavi.git
synced 2026-05-12 09:59:45 +08:00
- 表头自定义组件保留 width,virtual 模式下选择列正常显示 - 新增后端 ExportQuery,导出当前页/选中行避免长字段 IPC 截断 - 筛选支持更多操作符并统一 WHERE 生成逻辑 Close #57 Close #56
247 lines
10 KiB
TypeScript
247 lines
10 KiB
TypeScript
import React, { useEffect, useState, useCallback, useRef } from 'react';
|
|
import { message } from 'antd';
|
|
import { TabData, ColumnDefinition } from '../types';
|
|
import { useStore } from '../store';
|
|
import { DBQuery, DBGetColumns } from '../../wailsjs/go/app/App';
|
|
import DataGrid, { GONAVI_ROW_KEY } from './DataGrid';
|
|
import { buildWhereSQL, quoteIdentPart, quoteQualifiedIdent } from '../utils/sql';
|
|
|
|
const DataViewer: React.FC<{ tab: TabData }> = ({ tab }) => {
|
|
const [data, setData] = useState<any[]>([]);
|
|
const [columnNames, setColumnNames] = useState<string[]>([]);
|
|
const [pkColumns, setPkColumns] = useState<string[]>([]);
|
|
const [loading, setLoading] = useState(false);
|
|
const { connections, addSqlLog } = useStore();
|
|
const fetchSeqRef = useRef(0);
|
|
const countSeqRef = useRef(0);
|
|
const countKeyRef = useRef<string>('');
|
|
const pkSeqRef = useRef(0);
|
|
const pkKeyRef = useRef<string>('');
|
|
|
|
const [pagination, setPagination] = useState({
|
|
current: 1,
|
|
pageSize: 100,
|
|
total: 0,
|
|
totalKnown: false
|
|
});
|
|
|
|
const [sortInfo, setSortInfo] = useState<{ columnKey: string, order: string } | null>(null);
|
|
|
|
const [showFilter, setShowFilter] = useState(false);
|
|
const [filterConditions, setFilterConditions] = useState<any[]>([]);
|
|
|
|
useEffect(() => {
|
|
setPkColumns([]);
|
|
pkKeyRef.current = '';
|
|
countKeyRef.current = '';
|
|
setPagination(prev => ({ ...prev, current: 1, total: 0, totalKnown: false }));
|
|
}, [tab.connectionId, tab.dbName, tab.tableName]);
|
|
|
|
const fetchData = useCallback(async (page = pagination.current, size = pagination.pageSize) => {
|
|
const seq = ++fetchSeqRef.current;
|
|
setLoading(true);
|
|
const conn = connections.find(c => c.id === tab.connectionId);
|
|
if (!conn) {
|
|
message.error("Connection not found");
|
|
if (fetchSeqRef.current === seq) setLoading(false);
|
|
return;
|
|
}
|
|
|
|
const config = {
|
|
...conn.config,
|
|
port: Number(conn.config.port),
|
|
password: conn.config.password || "",
|
|
database: conn.config.database || "",
|
|
useSSH: conn.config.useSSH || false,
|
|
ssh: conn.config.ssh || { host: "", port: 22, user: "", password: "", keyPath: "" }
|
|
};
|
|
|
|
const dbType = config.type || '';
|
|
|
|
const dbName = tab.dbName || '';
|
|
const tableName = tab.tableName || '';
|
|
|
|
const whereSQL = buildWhereSQL(dbType, filterConditions);
|
|
|
|
const countSql = `SELECT COUNT(*) as total FROM ${quoteQualifiedIdent(dbType, tableName)} ${whereSQL}`;
|
|
|
|
let sql = `SELECT * FROM ${quoteQualifiedIdent(dbType, tableName)} ${whereSQL}`;
|
|
if (sortInfo && sortInfo.order) {
|
|
sql += ` ORDER BY ${quoteIdentPart(dbType, sortInfo.columnKey)} ${sortInfo.order === 'ascend' ? 'ASC' : 'DESC'}`;
|
|
}
|
|
const offset = (page - 1) * size;
|
|
// 大表性能:打开表不阻塞在 COUNT(*),先通过多取 1 条判断是否还有下一页;总数在后台统计并异步回填。
|
|
sql += ` LIMIT ${size + 1} OFFSET ${offset}`;
|
|
|
|
const startTime = Date.now();
|
|
try {
|
|
const pData = DBQuery(config as any, dbName, sql);
|
|
|
|
const resData = await pData;
|
|
const duration = Date.now() - startTime;
|
|
|
|
// Log Execution
|
|
addSqlLog({
|
|
id: `log-${Date.now()}-data`,
|
|
timestamp: Date.now(),
|
|
sql: sql,
|
|
status: resData.success ? 'success' : 'error',
|
|
duration: duration,
|
|
message: resData.success ? '' : resData.message,
|
|
affectedRows: Array.isArray(resData.data) ? resData.data.length : undefined,
|
|
dbName
|
|
});
|
|
|
|
if (pkColumns.length === 0) {
|
|
const pkKey = `${tab.connectionId}|${dbName}|${tableName}`;
|
|
if (pkKeyRef.current !== pkKey) {
|
|
pkKeyRef.current = pkKey;
|
|
const pkSeq = ++pkSeqRef.current;
|
|
DBGetColumns(config as any, dbName, tableName)
|
|
.then((resCols: any) => {
|
|
if (pkSeqRef.current !== pkSeq) return;
|
|
if (pkKeyRef.current !== pkKey) return;
|
|
if (!resCols?.success) return;
|
|
const pks = (resCols.data as ColumnDefinition[]).filter((c: any) => c.key === 'PRI').map((c: any) => c.name);
|
|
setPkColumns(pks);
|
|
})
|
|
.catch(() => {
|
|
if (pkSeqRef.current !== pkSeq) return;
|
|
if (pkKeyRef.current !== pkKey) return;
|
|
});
|
|
}
|
|
}
|
|
|
|
if (resData.success) {
|
|
let resultData = resData.data as any[];
|
|
if (!Array.isArray(resultData)) resultData = [];
|
|
|
|
const hasMore = resultData.length > size;
|
|
if (hasMore) resultData = resultData.slice(0, size);
|
|
|
|
let fieldNames = resData.fields || [];
|
|
if (fieldNames.length === 0 && resultData.length > 0) {
|
|
fieldNames = Object.keys(resultData[0]);
|
|
}
|
|
if (fetchSeqRef.current !== seq) return;
|
|
setColumnNames(fieldNames);
|
|
resultData.forEach((row: any, i: number) => {
|
|
if (row && typeof row === 'object') row[GONAVI_ROW_KEY] = `row-${offset + i}`;
|
|
});
|
|
setData(resultData);
|
|
const countKey = `${tab.connectionId}|${dbName}|${tableName}|${whereSQL}`;
|
|
const derivedTotalKnown = !hasMore;
|
|
const derivedTotal = derivedTotalKnown ? offset + resultData.length : page * size + 1;
|
|
if (derivedTotalKnown) countKeyRef.current = countKey;
|
|
|
|
setPagination(prev => {
|
|
if (derivedTotalKnown) {
|
|
return { ...prev, current: page, pageSize: size, total: derivedTotal, totalKnown: true };
|
|
}
|
|
if (prev.totalKnown && countKeyRef.current === countKey) {
|
|
return { ...prev, current: page, pageSize: size };
|
|
}
|
|
return { ...prev, current: page, pageSize: size, total: derivedTotal, totalKnown: false };
|
|
});
|
|
|
|
if (!derivedTotalKnown) {
|
|
if (countKeyRef.current !== countKey) {
|
|
countKeyRef.current = countKey;
|
|
const countSeq = ++countSeqRef.current;
|
|
const countStart = Date.now();
|
|
|
|
DBQuery(config as any, dbName, countSql)
|
|
.then((resCount: any) => {
|
|
const countDuration = Date.now() - countStart;
|
|
|
|
addSqlLog({
|
|
id: `log-${Date.now()}-count`,
|
|
timestamp: Date.now(),
|
|
sql: countSql,
|
|
status: resCount.success ? 'success' : 'error',
|
|
duration: countDuration,
|
|
message: resCount.success ? '' : resCount.message,
|
|
dbName
|
|
});
|
|
|
|
if (countSeqRef.current !== countSeq) return;
|
|
if (countKeyRef.current !== countKey) return;
|
|
|
|
if (!resCount.success) return;
|
|
if (!Array.isArray(resCount.data) || resCount.data.length === 0) return;
|
|
|
|
const total = Number(resCount.data[0]?.['total']);
|
|
if (!Number.isFinite(total) || total < 0) return;
|
|
|
|
setPagination(prev => ({ ...prev, total, totalKnown: true }));
|
|
})
|
|
.catch(() => {
|
|
if (countSeqRef.current !== countSeq) return;
|
|
if (countKeyRef.current !== countKey) return;
|
|
// 统计失败不影响主流程,不弹窗;可在日志里查看。
|
|
});
|
|
}
|
|
}
|
|
} else {
|
|
message.error(resData.message);
|
|
}
|
|
} catch (e: any) {
|
|
if (fetchSeqRef.current !== seq) return;
|
|
message.error("Error fetching data: " + e.message);
|
|
addSqlLog({
|
|
id: `log-${Date.now()}-error`,
|
|
timestamp: Date.now(),
|
|
sql: sql,
|
|
status: 'error',
|
|
duration: Date.now() - startTime,
|
|
message: e.message,
|
|
dbName
|
|
});
|
|
}
|
|
if (fetchSeqRef.current === seq) setLoading(false);
|
|
}, [connections, tab, sortInfo, filterConditions, pkColumns.length]);
|
|
// Depend on pkColumns.length to avoid loop? No, pkColumns is updated inside.
|
|
// Actually, 'pkColumns' state shouldn't trigger re-fetch.
|
|
// The 'if (pkColumns.length === 0)' check is inside.
|
|
// So adding pkColumns to dependency is safer but might trigger double fetch if not careful?
|
|
// Only if pkColumns changes. It changes once from [] to [...].
|
|
// So it's fine.
|
|
|
|
// Handlers memoized
|
|
const handleReload = useCallback(() => {
|
|
countKeyRef.current = '';
|
|
fetchData(pagination.current, pagination.pageSize);
|
|
}, [fetchData, pagination.current, pagination.pageSize]);
|
|
const handleSort = useCallback((field: string, order: string) => setSortInfo({ columnKey: field, order }), []);
|
|
const handlePageChange = useCallback((page: number, size: number) => fetchData(page, size), [fetchData]);
|
|
const handleToggleFilter = useCallback(() => setShowFilter(prev => !prev), []);
|
|
const handleApplyFilter = useCallback((conditions: any[]) => setFilterConditions(conditions), []);
|
|
|
|
useEffect(() => {
|
|
fetchData(1, pagination.pageSize);
|
|
}, [tab, sortInfo, filterConditions]); // Initial load and re-load on sort/filter
|
|
|
|
return (
|
|
<div style={{ flex: '1 1 auto', minHeight: 0, height: '100%', width: '100%', overflow: 'hidden', display: 'flex', flexDirection: 'column' }}>
|
|
<DataGrid
|
|
data={data}
|
|
columnNames={columnNames}
|
|
loading={loading}
|
|
tableName={tab.tableName}
|
|
dbName={tab.dbName}
|
|
connectionId={tab.connectionId}
|
|
pkColumns={pkColumns}
|
|
onReload={handleReload}
|
|
onSort={handleSort}
|
|
onPageChange={handlePageChange}
|
|
pagination={pagination}
|
|
showFilter={showFilter}
|
|
onToggleFilter={handleToggleFilter}
|
|
onApplyFilter={handleApplyFilter}
|
|
/>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
export default DataViewer;
|