1
0
mirror of https://github.com/haiwen/seahub.git synced 2025-09-21 19:37:28 +00:00
Files
seahub/frontend/src/pages/data-grid/app-main.js

254 lines
6.3 KiB
JavaScript
Raw Normal View History

import React from 'react';
import PropTypes from 'prop-types';
2019-04-26 07:16:01 +08:00
import ReactDataGrid from '@seafile/react-data-grid/';
import update from 'immutability-helper';
2019-04-26 07:16:01 +08:00
import { Menu, Editors } from '@seafile/react-data-grid-addons';
import GridHeaderContextMenu from './grid-header-contextmenu';
import GridContentContextMenu from './grid-content-contextmenu';
import ModalPortal from '../../components/modal-portal';
import NewColumnDialog from './new-column-dialog';
import isHotkey from 'is-hotkey';
const propTypes = {
initData: PropTypes.object.isRequired,
};
class AppMain extends React.Component {
constructor(props, context) {
super(props, context);
this._columns = [
{
2019-04-26 07:45:07 +08:00
key: 'name',
name: 'Name',
width: 80,
2019-04-26 07:45:07 +08:00
editable: true,
resizable: true
}
];
2019-05-06 14:36:47 +08:00
let initData = props.initData;
2019-05-06 14:36:47 +08:00
this.state = {
2019-04-26 07:16:01 +08:00
columns: initData.columns.length ? this.deseralizeGridData(initData.columns) : this._columns,
rows: initData.rows.length ? initData.rows : this.createRows(1),
isNewColumnDialogShow: false,
};
2019-05-06 14:36:47 +08:00
this.dTableStore = new DTableStore(initData);
}
componentDidMount() {
document.addEventListener('keydown', this.onHotKey);
}
componentWillReceiveProps(nextProps) {
if (nextProps.isContentChanged) {
return;
}
let data = nextProps.initData;
this.deseralizeGridData(data);
}
componentWillUnmount() {
document.removeEventListener('keydown', this.onHotKey);
}
createRows = (numberOfRows) => {
let rows = [];
for (let i = 0; i < numberOfRows; i++) {
rows[i] = this.createFakeRowObjectData(i);
}
return rows;
};
createFakeRowObjectData = (index) => {
2019-04-26 07:45:07 +08:00
return {name: 'name_' + index};
};
getColumns = () => {
let clonedColumns = this.state.columns.slice();
return clonedColumns;
};
handleGridRowsUpdated = ({ fromRow, toRow, updated }) => {
let rows = this.state.rows.slice();
for (let i = fromRow; i <= toRow; i++) {
let rowToUpdate = rows[i];
let updatedRow = update(rowToUpdate, {$merge: updated});
rows[i] = updatedRow;
}
this.setState({ rows });
this.props.onContentChanged();
};
handleAddRow = ({ newRowIndex }) => {
const newRow = {
2019-04-26 07:45:07 +08:00
name: 'name_' + newRowIndex,
};
let rows = this.state.rows.slice();
rows = update(rows, {$push: [newRow]});
this.setState({ rows });
this.props.onContentChanged();
};
getRowAt = (index) => {
if (index < 0 || index > this.getSize()) {
return undefined;
}
return this.state.rows[index];
};
getSize = () => {
return this.state.rows.length;
};
onInsertRow = () => {
let newRowIndex = this.getSize();
2019-05-06 14:36:47 +08:00
let rows = this.dTableStore.insertRow(newRowIndex);
this.setState({rows});
this.props.onContentChanged();
}
onInsertColumn = () => {
2019-04-26 07:16:01 +08:00
this.setState({isNewColumnDialogShow: true});
this.props.onContentChanged();
}
onColumnResize = (index, width) => {
let columns = this.state.columns.slice();
columns[index - 1].width = width;
this.setState({columns: columns});
this.props.onContentChanged();
2019-04-26 07:16:01 +08:00
}
onNewColumn = (columnName, columnType) => {
2019-05-06 14:36:47 +08:00
let idx = this.state.columns.length;
let columns = this.dTableStore.insertColumn(idx, columnName, columnType);
columns = this.formatColumnsData(columns);
this.setState({columns: columns});
2019-04-26 07:16:01 +08:00
this.onNewColumnCancel();
this.props.onContentChanged();
2019-04-26 07:16:01 +08:00
}
onNewColumnCancel = () => {
this.setState({isNewColumnDialogShow: false});
this.props.onContentChanged();
2019-04-26 07:16:01 +08:00
}
onRowDelete = (e, data) => {
let { rowIdx } = data;
2019-05-06 14:36:47 +08:00
let rows = this.dTableStore.deleteRow(rowIdx);
this.setState({rows});
this.props.onContentChanged();
2019-04-26 07:16:01 +08:00
}
onColumnDelete = (e, data) => {
let column = data.column;
2019-05-06 14:36:47 +08:00
let idx = column.idx - 1;
let columns = this.dTableStore.deleteColumn(idx);
this.setState({columns});
this.props.onContentChanged();
}
serializeGridData = () => {
let gridData = {
columns: JSON.stringify(this.state.columns),
rows: JSON.stringify(this.state.rows),
};
return gridData;
}
deseralizeGridData = (data) => {
2019-04-26 07:16:01 +08:00
let columns = JSON.parse(data.columns);
let rows = JSON.parse(data.rows);
columns = this.formatColumnsData(columns);
this.setState({
2019-04-26 07:16:01 +08:00
columns: columns,
rows: rows,
});
2019-05-06 14:36:47 +08:00
this.dTableStore.updateStoreValues({columns, rows});
2019-04-26 07:16:01 +08:00
}
formatColumnsData = (columns) => {
return columns.map(column => {
if (column.editor) {
let editor = this.createEditor(column.editor);
column.editor = editor;
}
return column;
});
}
2019-04-26 07:16:01 +08:00
createEditor = (editorType) => {
let editor = null;
switch (editorType) {
case 'number':
editor = <Editors.NumberEditor />;
break;
case 'text':
editor = null;
break;
default:
break;
}
return editor;
}
onHotKey = (event) => {
if (isHotkey('mod+s', event)) {
event.preventDefault();
this.props.onSave();
return true;
}
}
render() {
let columns = this.getColumns();
return (
<div id="main">
<ReactDataGrid
ref={ node => this.grid = node }
enableCellSelect={true}
columns={columns}
rowGetter={this.getRowAt}
rowsCount={this.getSize()}
onGridRowsUpdated={this.handleGridRowsUpdated}
enableRowSelect={true}
rowHeight={50}
2019-04-29 10:17:43 +08:00
minHeight={500}
rowScrollTimeout={200}
enableInsertColumn={true}
enableInsertRow={true}
onInsertRow={this.onInsertRow}
onInsertColumn={this.onInsertColumn}
2019-04-26 07:16:01 +08:00
RowsContainer={Menu.ContextMenuTrigger}
headerContextMenu={<GridHeaderContextMenu id="grid-header-contxtmenu" onColumnDelete={this.onColumnDelete} />}
contextMenu={<GridContentContextMenu id="grid-content-contxtmenu" onRowDelete={this.onRowDelete} />}
onColumnResize={this.onColumnResize}
/>
2019-04-26 07:16:01 +08:00
{this.state.isNewColumnDialogShow && (
<ModalPortal>
<NewColumnDialog
onNewColumnCancel={this.onNewColumnCancel}
onNewColumn={this.onNewColumn}
/>
</ModalPortal>
)}
</div>
);
}
}
AppMain.propTypes = propTypes;
export default AppMain;