| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523 |
- import React from 'react'
- import { Layout, Row, Col, Input, Button, Table, Icon, Menu, Dropdown, Card, Breadcrumb, Popover, Tree, Tag } from 'antd'
- import { connect } from 'dva'
- import { dateFormat } from '../../utils/baseUtils'
- import GroupSelector from './groupSelector'
- import TransferBox from '../common/selectUserBox/selectUserBox'
- import DeleteBox from '../common/deleteBox/deleteBox'
- import DataPreview from '../common/dataPreview/dataPreview'
- import './list.less'
- const { Content } = Layout
- const { Search } = Input
- const { TreeNode } = Tree
- class DataSource extends React.Component {
- constructor(props) {
- super(props);
- this.state = {
- selectedRecord: null, // 当前选中的dataSource
- visibleGroupMenu: false, // 显示分组菜单
- visibleSetGroupMenu: false, //
- visibleTransferBox: false,
- visibleDeleteBox: false,
- visibleDataPreviewBox: false, // 显示数据预览
- groupEditing: false, // 是否处于编辑状态
- }
- };
- hideTransferBox = () => {
- this.setState({ visibleTransferBox: false})
- }
- componentDidMount() {
- const { dispatch } = this.props;
- this.setScrollTableHeight();
- dispatch({ type: 'dataSource/fetchList' });
- dispatch({ type: 'dataSource/remoteGroupList' });
- }
- /**
- * 根据视图设置表格高度以呈现滚动条
- */
- setScrollTableHeight() {
- const mainContent = document.getElementsByClassName('main-content')[0];
- const toolbar = mainContent.getElementsByClassName('datasource-tools')[0];
- const tableHeader = mainContent.getElementsByClassName('ant-table-header')[0];
- const tableBody = mainContent.getElementsByClassName('ant-table-body')[0];
- tableBody.style.maxHeight=`${mainContent.offsetHeight - toolbar.offsetHeight - tableHeader.offsetHeight - 58}px`;
- }
- onGroup() {
- const { dataSource } = this.props;
- const groupList = dataSource.groupList;
- const list = dataSource.list;
- const currentGroup = dataSource.currentGroup;
-
- let groupFilter = groupList.concat({ code: '-1', label: '未分组' }).filter(g => (
- currentGroup[0].code === 'all' ||
- (
- currentGroup[0].code === '-1' ? g.code === '-1' : (
- currentGroup[1] ? (g.code === currentGroup[1].code) :
- (
- g.code === currentGroup[0].code ||
- g.pcode === currentGroup[0].code
- )
- )
- )
- )).map(g => g.code);
-
- return list.filter(l => groupFilter.indexOf(l.groupCode+'') !== -1);
- }
- onSearch(list, text) {
- const reg = new RegExp('([+ \\- & | ! ( ) { } \\[ \\] ^ \" ~ * ? : ( ) \/])', 'g'); // 需要转义的字符
- let filterLabel = (text || '').replace(new RegExp('(\\\\)', 'g'), '\\$1').replace(reg, '\\$1'); // 添加转义符号
- return list.map(l => {
- let o = Object.assign({}, l);
- let reg = new RegExp('('+ filterLabel +'){1}', 'ig');
- if(o.name && o.name.search(reg) !== -1) {
- return o;
- }else if(o.description && o.description.search(reg) !== -1) {
- return o;
- }else {
- return null
- }
- }).filter(a => a!==null);
- }
- onSort(list) {
- return list.sort((a, b) => {
- return new Date(b.createTime) - new Date(a.createTime);
- });
- }
- handleVisibleChange = (flag) => {
- this.setState({ visibleGroupMenu: flag });
- }
- createGroupMenu = (selectedRecord) => {
- const { dataSource, dispatch } = this.props;
- const groupList = dataSource.groupList;
- const pGroups = groupList.filter(d => d.pcode === '-1').sort((a, b) => a.index - b.index);
- const cGroups = groupList.filter(d => d.pcode !== '-1');
- let allGroups = selectedRecord ? [
- { code: '-1', label: '未分组' }
- ].concat(pGroups) : [
- { code: 'all', label: '全部分组' },
- { code: '-1', label: '未分组' }
- ].concat(pGroups);
- return allGroups.map(p => {
- let c = cGroups.filter(c => c.pcode === p.code).sort((a, b) => a.index - b.index);
- return c.length > 0 ? (
- <Menu.SubMenu key={p.code} title={<span style={{ fontWeight: selectedRecord ?
- (p.code + '' === selectedRecord.groupCode + '' ? 'bold' : (
- c.find(ch => ch.code + '' === selectedRecord.groupCode + '') && c.find(ch => ch.code + '' === selectedRecord.groupCode + '').pcode === p.code ? 'bold' : 'normal'
- ))
- : dataSource.currentGroup[0].code === p.code ? 'bold' : 'normal'
- }}>{p.label}</span>} onTitleClick={(item) => {
- // dispatch({ type: 'dataSource/setCurrentGroup', group1: p }); // 尝试禁用分组后跳转逻辑
- if (selectedRecord) {
- dispatch({ type: 'dataSource/remoteSetDataSourceGroup', dataSource: selectedRecord, group: p });
- }
- this.hideGroupMenu();
- }}>
- {c.map(c => {
- return (<Menu.Item key={c.code} onClick={(item) => {
- // dispatch({ type: 'dataSource/setCurrentGroup', group1: p, group2: c }); // 尝试禁用分组后跳转逻辑
- if(selectedRecord) {
- dispatch({ type: 'dataSource/remoteSetDataSourceGroup', dataSource: selectedRecord, group: c });
- }
- }}><span style={{ fontWeight: selectedRecord ? (
- selectedRecord.groupCode+'' === c.code+'' ? 'bold' : 'normal'
- ) : (dataSource.currentGroup[1] && (dataSource.currentGroup[1].code === c.code) ? 'bold' : 'normal') }}>{c.label}</span></Menu.Item>)
- })}
- </Menu.SubMenu>
- ) : (
- <Menu.Item key={p.code} onClick={() => {
- // dispatch({ type: 'dataSource/setCurrentGroup', group1: p }); // 尝试禁用分组后跳转逻辑
- if(selectedRecord) {
- dispatch({ type: 'dataSource/remoteSetDataSourceGroup', dataSource: selectedRecord, group: p });
- }
- this.hideGroupMenu();
- }}><span className={selectedRecord ? (
- selectedRecord.groupCode+'' === p.code+'' ? 'selected' : ''
- ) : dataSource.currentGroup[0] && (dataSource.currentGroup[0].code === p.code) ? 'selected' : ''} style={{ fontWeight: selectedRecord ? (
- selectedRecord.groupCode+'' === p.code+'' ? 'bold' : 'normal'
- ) : dataSource.currentGroup[0] && (dataSource.currentGroup[0].code === p.code) ? 'bold' : 'normal' }}>{p.label}</span></Menu.Item>
- );
- });
- }
- createSubGroupMenu = () => {
- const { dataSource, dispatch } = this.props;
- const groupList = dataSource.groupList;
- const parentGroup = dataSource.currentGroup[0];
- const children = groupList.filter(d => d.pcode === parentGroup.code);
- const subGroup = dataSource.currentGroup[1];
- return children.map(c => {
- return (
- <Menu.Item key={c.code} onClick={() => {
- dispatch({ type: 'dataSource/setCurrentGroup', group1: parentGroup, group2: c });
- }}><span className={subGroup && (subGroup.code === c.code) ? 'selected' : ''} style={{ fontWeight: subGroup && (subGroup.code === c.code) ? 'bold' : 'normal' }}>{c.label}</span></Menu.Item>
- );
- })
- }
- createGroupTree(modify) {
- const { dispatch, dataSource } = this.props;
- const { groupEditing } = this.state;
- const groupList = dataSource.groupList;
- let parent = groupList.filter(d => d.pcode === '-1').sort((a, b) => a.index - b.index);
- let children = groupList.filter(d => d.pcode !== '-1');
- let groupTree = parent.map(p => {
- return (
- <TreeNode disabled={groupEditing} title={
- modify ? (<div><Icon style={{ cursor: 'move' }} type='drag'/>
- <Input value={p.label} size='small' focus={'true'} onFocus={() => {
- this.setState({
- groupEditing: true
- });
- }} onChange={(e) => {
- dispatch({ type: 'dataSource/modifyGroup', group: {...p, label:e.target.value} });
- }} onBlur={(e) => {
- this.setState({
- groupEditing: false
- });
- dispatch({ type: 'dataSource/remoteModifyGroup', group: {...p, label:e.target.value} });
- }} onPressEnter={(e) => {
- dispatch({ type: 'dataSource/remoteModifyGroup', group: {...p, label:e.target.value} });
- }} /><Icon type='plus-circle-o' onClick={() => {
- dispatch({ type: 'dataSource/remoteAddGroup', pgroup: p });
- }}/><Icon type='minus-circle' onClick={() => {
- dispatch({ type: 'dataSource/remoteDeleteGroup', group: p });
- }}/></div>) : p.label} key={p.code}>
- {
- children.filter(c => c.pcode === p.code).sort((a, b) => a.index - b.index).map(c => {
- return (
- <TreeNode disabled={groupEditing} title={
- modify ? (<div><Icon style={{ cursor: 'move' }} type='drag'/>
- <Input value={c.label} size='small' onFocus={() => {
- this.setState({
- groupEditing: true
- });
- }} onChange={(e) => {
- dispatch({ type: 'dataSource/modifyGroup', group: {...c, label:e.target.value} });
- }} onBlur={(e) => {
- this.setState({
- groupEditing: false
- });
- dispatch({ type: 'dataSource/remoteModifyGroup', group: {...c, label:e.target.value} });
- }} onPressEnter={(e) => {
- dispatch({ type: 'dataSource/remoteModifyGroup', group: {...c, label:e.target.value} });
- }} onCompositionEnd={(e) => {
- }}/><Icon type='minus-circle' onClick={() => {
- dispatch({ type: 'dataSource/remoteDeleteGroup', group: c });
- }}/></div>) : p.label
- } key={c.code} />
- )
- })
- }
- </TreeNode>
- )
- });
- return groupTree;
- }
- hideGroupMenu = () => {
- this.setState({
- visibleGroupMenu: false
- });
- }
- onDrop = (info) => {
- const { dispatch } = this.props;
- const dropCode = info.node.props.eventKey;
- const dragCode = info.dragNode.props.eventKey;
- const dropPos = info.node.props.pos.split('-');
- const dropPosition = info.dropPosition - Number(dropPos[dropPos.length - 1]); // -1/0/1 -> 兄/子/弟
- dispatch({ type: 'dataSource/remoteMoveGroup', dragCode, dropCode, dropPosition });
- }
- render() {
-
- const { main, dataSource, dispatch } = this.props;
- const { selectedRecord, visibleTransferBox, visibleDeleteBox, visibleDataPreviewBox } = this.state;
- const { currentUser } = main;
- const reg = new RegExp('([+ \\- & | ! ( ) { } \\[ \\] ^ \" ~ * ? : ( ) \/])', 'g'); // 需要转义的字符
- let filterLabel = dataSource.filterLabel.replace(new RegExp('(\\\\)', 'g'), '\\$1').replace(reg, '\\$1'); // 添加转义符号
- const TAG_COLOR = ['blue'];
- // const TAG_COLOR1 = ['magenta', 'red', 'volcano', 'orange', 'gold', 'lime', 'green', 'cyan', 'blue', 'geekblue', 'purple'];
- const moreOperatingMenu = (
- <Menu className='operationmenu' visible={true}>
- <Menu.Item
- onClick={() => {
- dispatch({ type: 'chartDesigner/remoteQucikAdd', dataSource: selectedRecord });
- }}
- >
- <Icon type="file-add" />创建图表
- </Menu.Item>
- {/* {selectedRecord && currentUser.code === selectedRecord.creatorCode && <Menu.Item onClick={(e) => {
- dispatch({ type: 'dataSource/resetNewModel' });
- let selectedModel = dataSource.list.find((i) => { return i.code === selectedRecord.code })
- dispatch({type: 'main/redirect', path: {pathname: '/datasource/'+ selectedModel.type +'/' + selectedModel.code + '/base'}});
- }}>
- <Icon type="info-circle-o" />属性设置
- </Menu.Item>} */}
- <Menu.Item onClick={() => {
- this.setState({
- visibleDataPreviewBox: true
- });
- }}><Icon type="search" />预览数据</Menu.Item>
- <Menu.Divider />
- { selectedRecord && currentUser.code === selectedRecord.creatorCode && <Menu.SubMenu className='setgroupmenu' title={<div><Icon style={{ marginRight: '6px' }} type='profile' />移动到</div>}>
- {this.createGroupMenu(selectedRecord)}
- </Menu.SubMenu>}
- <Menu.Divider />
- { selectedRecord && currentUser.code === selectedRecord.creatorCode && <Menu.Item
- onClick={()=>{
- this.setState({ visibleTransferBox: true})
- }}
- >
- <Icon type="swap" />移交
- </Menu.Item>}
- { selectedRecord && currentUser.code === selectedRecord.creatorCode && <Menu.Item
- onClick={(e) => {
- this.setState({ visibleDeleteBox: true})
- }}
- >
- <Icon type="delete" />删除
-
- </Menu.Item>}
- </Menu>
- );
- const dataSourceColumns = [{
- title: '名称',
- dataIndex: 'name',
- key: 'name',
- width: 100,
- render: (text, record) => {
- return <div className='datasource-name'>
- <div className={`datasource-type`}>
- {record.type === 'database' ? <Icon type="database" theme="outlined" /> : <Icon type="file-excel" theme="outlined" />}
- </div>
- <div>
- <span style={{ color: '#1890ff', cursor: 'pointer' }} onClick={() => {
- dispatch({ type: 'dataSource/resetNewModel' });
- dispatch({type: 'main/redirect', path: {pathname: '/datasource/'+ record.type +'/' + record.code + '/base'}});
- }}>
- { filterLabel ?
- ((text || '').split(new RegExp(`(${filterLabel})`, 'i')).map((fragment, i) => {
- return (
- fragment.toLowerCase().replace(new RegExp('(\\\\)', 'g'), '\\$1').replace(reg, '\\$1') === filterLabel.toLowerCase() ?
- <span key={i} style={{fontWeight: 'bold', color: 'red'}} className="highlight">{fragment}</span> :
- fragment
- )
- }
- )) : text
- }
- </span>
- </div>
- </div>
- }
- }, {
- title: '说明',
- dataIndex: 'description',
- key: 'description',
- width: 200,
- render: (text, record) => {
- return (
- <span>
- { filterLabel ?
- ((text || '').split(new RegExp(`(${filterLabel})`, 'i')).map((fragment, i) => {
- return (
- fragment.toLowerCase().replace(new RegExp('(\\\\)', 'g'), '\\$1').replace(reg, '\\$1') === filterLabel.toLowerCase() ?
- <span key={i} style={{fontWeight: 'bold', color: 'red'}} className="highlight">{fragment}</span> :
- fragment
- )
- }
- )) : text
- }
- </span>
- )
- }
- }, {
- title: '创建人',
- dataIndex: 'creatorName',
- key: 'creatorName',
- width: 100
- }, {
- title: '创建时间',
- dataIndex: 'createTime',
- key: 'createTime',
- render: (text, record) => dateFormat(text, 'yyyy-MM-dd hh:mm:ss'),
- width: 100
- }, {
- title: '操作',
- key: 'action',
- render: (text, record, index) => (
- <Dropdown code={record.code} overlay={moreOperatingMenu} trigger={['click']} >
- <Icon type="setting" />
- </Dropdown>
- ),
- width: 50
- }];
- return (
- <Layout className='datasource-view'>
- <Content>
- <Card className='datasource-body' title={
- <Row className='datasource-tools' type='flex' justify='space-between'>
- <Col style={{ display: 'flex' }}>
- <Popover overlayClassName='popover-group' title={
- <Row className='grouptree-title' type='flex' justify='space-between'>
- <Col>
- 分组管理
- </Col>
- <Col>
- <div className='create-group' onClick={() => {
- dispatch({ type: 'dataSource/remoteAddGroup' });
- }}>添加分组<Icon type="plus-circle-o" /></div>
- </Col>
- </Row>
- } trigger="click" placement="bottomLeft" content={(
- <Tree
- className='tree-group'
- showLine
- defaultExpandAll
- draggable
- onDragStart={this.onDragStart}
- onDrop={this.onDrop}
- >
- {
- this.createGroupTree(true)
- }
- </Tree>
- )}>
- <Icon type="bars" />
- </Popover>
- <Breadcrumb className='group' separator=">">
- <Breadcrumb.Item>
- <GroupSelector model={dataSource} modelName='dataSource'>
- <Tag color={TAG_COLOR[Math.ceil(Math.random()*TAG_COLOR.length) - 1]} >
- {dataSource.currentGroup[0].label}
- </Tag>
- </GroupSelector>
- </Breadcrumb.Item>
- {dataSource.currentGroup[1] && (<Breadcrumb.Item>
- <GroupSelector model={dataSource} modelName='dataSource'>
- <Tag color={TAG_COLOR[Math.ceil(Math.random()*TAG_COLOR.length) - 1]}>
- {dataSource.currentGroup[1].label}
- </Tag>
- </GroupSelector>
- </Breadcrumb.Item>)}
- </Breadcrumb>
- </Col>
- <Col className='search'>
- <Col>
- <Search
- value={dataSource.filterLabel}
- placeholder="请输入关键字"
- onChange={e => {
- dispatch({ type: 'dataSource/setFilterLabel', label: e.target.value });
- }}
- />
- </Col>
- <Col>
- <Dropdown overlay={(
- <Menu onClick={(item, key, keyPath) => {
- const type = item.key;
- dispatch({ type: 'dataSource/resetNewModel' });
- dispatch({ type: 'dataConnect/resetSelected' });
- dispatch({ type: 'dataSource/setNewModelField', name: 'type', value: type });
- dispatch({type: 'main/redirect', path: {pathname: '/datasource/'+ type +'/create/base'}});
- }}>
- { currentUser.role === 'admin' && <Menu.Item key='database'>来自数据库</Menu.Item>}
- <Menu.Item disabled key='file'>来自文件</Menu.Item>
- </Menu>
- )} trigger={['click']}>
- <Button>
- <Icon type="plus" />添加数据源
- </Button>
- </Dropdown>
- </Col>
- </Col>
- </Row>
- }>
- <Table
- className='datasource-table'
- columns={dataSourceColumns}
- dataSource={
- this.onSort(
- this.onSearch(this.onGroup(), dataSource.filterLabel)
- )
- }
- size='small'
- scroll={{x: false, y: true}}
- pagination={false}
- onRow={(record) => {
- return {
- onClick: () => {this.setState({ selectedRecord: record})}
- }
- }}
- />
- <TransferBox
- visibleBox={visibleTransferBox}
- title='选择移交对象'
- okHandler={(user) => {
- dispatch({ type: 'dataSource/transfer', dataSourceCode: this.state.selectedRecord.code, userCode: user.code });
- }}
- hideBox={() => {
- this.setState({
- visibleTransferBox: false
- })
- }}
- onlyAdmin={true}
- />
- {visibleDeleteBox && <DeleteBox
- visibleBox={visibleDeleteBox}
- text={<div><span>确定要删除数据源【{selectedRecord.name}】吗?</span><br/><span style={{ color: 'red' }}>(此操作将会导致使用该数据源的图表失效!)</span></div>}
- hideBox={() => {
- this.setState({
- visibleDeleteBox: false
- })
- }}
- okHandler={() =>{
- dispatch({ type: 'dataSource/remoteDelete', code: selectedRecord.code }).then(() => {
- dispatch({ type: 'chart/fetchList', mandatory: true });
- })
- }}
- />}
- {visibleDataPreviewBox && <DataPreview
- visibleBox={visibleDataPreviewBox}
- hideBox={() => {
- this.setState({
- visibleDataPreviewBox: false
- });
- }}
- fetchFunction={(page, pageSize) => {
- dispatch({ type: 'dataSource/remoteDataList', code: selectedRecord.code, page, pageSize });
- }}
- />}
- </Card>
- </Content>
- </Layout>
- )
- }
- }
- function mapStateToProps({present: {main, dataSource, dataConnect}}) {
- return { main, dataSource, dataConnect }
- }
- export default connect(mapStateToProps)(DataSource)
|