| | |
| | | import React, {Component} from 'react' |
| | | import PropTypes from 'prop-types' |
| | | import { fromJS } from 'immutable' |
| | | import { Modal, Form, Row, Col, Select, Icon, Radio, Tooltip, Input, InputNumber, Cascader, Tabs } from 'antd' |
| | | import { is, fromJS } from 'immutable' |
| | | import { Modal, Form, Row, Col, Select, Icon, Radio, Tooltip, Input, InputNumber, Tabs, Button } from 'antd' |
| | | |
| | | import Utils from '@/utils/utils.js' |
| | | import { chartColors } from '@/utils/option.js' |
| | | import { getBarOrLineChartOptionForm } from './formconfig' |
| | | import { minkeColorSystem, colorTransform } from '@/utils/option.js' |
| | | import asyncComponent from '@/utils/asyncComponent' |
| | | import ColorSketch from '@/mob/colorsketch' |
| | | import './index.scss' |
| | | |
| | | const { TabPane } = Tabs |
| | | const EditTable = asyncComponent(() => import('@/templates/zshare/editTable')) |
| | | |
| | | class LineChartDrawerForm extends Component { |
| | | static propTpyes = { |
| | | MenuType: PropTypes.any, |
| | | dict: PropTypes.object, |
| | | plot: PropTypes.object, |
| | | sysRoles: PropTypes.array, |
| | | config: PropTypes.object, |
| | | plotchange: PropTypes.func |
| | | } |
| | |
| | | state = { |
| | | view: 'normal', |
| | | visible: false, |
| | | disabled: true, |
| | | datatype: '', |
| | | plot: null, |
| | | formlist: null, |
| | | fieldName: null, |
| | | colorOptions: fromJS(minkeColorSystem).toJS().map(option => { |
| | | option.children = option.children.map(cell => { |
| | | let _cell = {} |
| | | _cell.label = <div className={'background ' + cell.value}>{cell.value}</div> |
| | | _cell.value = colorTransform[cell.value] |
| | | |
| | | return _cell |
| | | }) |
| | | return option |
| | | }), |
| | | shapeOptions: [ |
| | | colorColumns: [ |
| | | { |
| | | value: 'line', |
| | | label: '折线', |
| | | children: [ |
| | | { value: 'smooth', label: 'smooth' }, |
| | | { value: 'line', label: 'line' }, |
| | | { value: 'dot', label: 'dot' }, |
| | | { value: 'dash', label: 'dash' }, |
| | | { value: 'hv', label: 'hv' }, |
| | | { value: 'vh', label: 'vh' }, |
| | | { value: 'hvh', label: 'hvh' }, |
| | | { value: 'vhv', label: 'vhv' } |
| | | title: '指标', |
| | | dataIndex: 'label', |
| | | editable: false, |
| | | width: '40%' |
| | | }, |
| | | { |
| | | title: '颜色', |
| | | dataIndex: 'color', |
| | | inputType: 'color', |
| | | editable: true, |
| | | width: '40%', |
| | | render: (text, record) => { |
| | | return (<div style={{width: '80px', height: '23px', background: text}}></div>) |
| | | } |
| | | }, |
| | | ], |
| | | statColorColumns: [ |
| | | { |
| | | title: '指标', |
| | | dataIndex: 'type', |
| | | inputType: 'input', |
| | | editable: true, |
| | | width: '40%' |
| | | }, |
| | | { |
| | | title: '颜色', |
| | | dataIndex: 'color', |
| | | inputType: 'color', |
| | | editable: true, |
| | | width: '40%', |
| | | render: (text, record) => { |
| | | return (<div style={{width: '80px', height: '23px', background: text}}></div>) |
| | | } |
| | | }, |
| | | ], |
| | | cusColumns: [ |
| | | { |
| | | title: '指标', |
| | | dataIndex: 'name', |
| | | editable: false, |
| | | width: '20%' |
| | | }, |
| | | { |
| | | title: '形状', |
| | | dataIndex: 'shape', |
| | | inputType: 'cascader', |
| | | editable: true, |
| | | width: '20%', |
| | | render: (text, record) => { |
| | | return text.join(' / ').replace('line', '折线').replace('bar', '柱形') |
| | | }, |
| | | options: [ |
| | | { |
| | | value: 'line', |
| | | label: '折线', |
| | | children: [ |
| | | { value: 'smooth', label: 'smooth' }, |
| | | { value: 'line', label: 'line' }, |
| | | { value: 'dot', label: 'dot' }, |
| | | { value: 'dash', label: 'dash' }, |
| | | { value: 'hv', label: 'hv' }, |
| | | { value: 'vh', label: 'vh' }, |
| | | { value: 'hvh', label: 'hvh' }, |
| | | { value: 'vhv', label: 'vhv' } |
| | | ] |
| | | }, |
| | | { |
| | | value: 'bar', |
| | | label: '柱形', |
| | | children: [ |
| | | { value: 'rect', label: 'rect' }, |
| | | { value: 'hollow-rect', label: 'hollow-rect' }, |
| | | { value: 'line', label: 'line' }, |
| | | { value: 'tick', label: 'tick' }, |
| | | { value: 'funnel', label: 'funnel' }, |
| | | { value: 'pyramid', label: 'pyramid' } |
| | | ], |
| | | } |
| | | ] |
| | | }, |
| | | { |
| | | value: 'bar', |
| | | label: '柱形', |
| | | children: [ |
| | | { value: 'rect', label: 'rect' }, |
| | | { value: 'hollow-rect', label: 'hollow-rect' }, |
| | | { value: 'line', label: 'line' }, |
| | | { value: 'tick', label: 'tick' }, |
| | | { value: 'funnel', label: 'funnel' }, |
| | | { value: 'pyramid', label: 'pyramid' } |
| | | title: '坐标轴', |
| | | dataIndex: 'axis', |
| | | inputType: 'select', |
| | | editable: true, |
| | | width: '20%', |
| | | options: [ |
| | | { value: 'true', text: '显示'}, |
| | | { value: 'false', text: '隐藏'} |
| | | ], |
| | | } |
| | | render: (text, record) => { |
| | | let trans = {'true': '显示', 'false': '隐藏'} |
| | | return trans[text] || '隐藏' |
| | | } |
| | | }, |
| | | { |
| | | title: '标注', |
| | | dataIndex: 'label', |
| | | inputType: 'select', |
| | | editable: true, |
| | | width: '20%', |
| | | options: [ |
| | | { value: 'true', text: '显示'}, |
| | | { value: 'false', text: '隐藏'} |
| | | ], |
| | | render: (text, record) => { |
| | | let trans = {'true': '显示', 'false': '隐藏'} |
| | | return trans[text] || '隐藏' |
| | | } |
| | | }, |
| | | ] |
| | | } |
| | | |
| | | showDrawer = () => { |
| | | const { config } = this.props |
| | | const { config, sysRoles, MenuType } = this.props |
| | | |
| | | let fieldName = {} |
| | | config.columns.forEach(col => { |
| | |
| | | } |
| | | }) |
| | | |
| | | if (config.plot.correction) { |
| | | delete config.plot.correction // 数据修正(已弃用) |
| | | config.plot.barSize = 35 |
| | | } |
| | | |
| | | this.setState({ |
| | | visible: true, |
| | | view: 'normal', |
| | | disabled: config.plot.datatype === 'statistics', |
| | | datatype: config.plot.datatype || 'query', |
| | | fieldName: fieldName, |
| | | plot: fromJS(config.plot).toJS(), |
| | | formlist: getBarOrLineChartOptionForm(config.plot, config.columns, config.setting) |
| | | formlist: getBarOrLineChartOptionForm(config.plot, config.columns, sysRoles, MenuType) |
| | | }) |
| | | } |
| | | |
| | |
| | | |
| | | if (key === 'datatype') { |
| | | this.setState({ |
| | | disabled: val === 'statistics', |
| | | datatype: val, |
| | | formlist: formlist.map(item => { |
| | | if (['Yaxis'].includes(item.key)) { |
| | | item.hidden = val === 'statistics' |
| | |
| | | return fields |
| | | } |
| | | |
| | | getCustomFields = () => { |
| | | const { getFieldDecorator } = this.props.form |
| | | const { plot, fieldName, enabled, colorOptions, shapeOptions } = this.state |
| | | const fields = [] |
| | | |
| | | if (!plot.customs) return null |
| | | |
| | | fields.push(<Col span={12} key="enabled"> |
| | | <Form.Item label="是否启用" style={{marginBottom: 10}}> |
| | | {getFieldDecorator('enabled', { |
| | | initialValue: plot.enabled || 'false' |
| | | })( |
| | | <Radio.Group onChange={this.enabledChange}> |
| | | <Radio value="true">是</Radio> |
| | | <Radio value="false">否</Radio> |
| | | </Radio.Group> |
| | | )} |
| | | </Form.Item> |
| | | </Col>) |
| | | |
| | | plot.customs.forEach((item, i) => { |
| | | fields.push(<Col span={24} key={'field' + i}> |
| | | <p className="field-title">{fieldName[item.field]}</p> |
| | | </Col>) |
| | | fields.push(<Col span={12} key={'shape' + i}> |
| | | <Form.Item label="形状"> |
| | | {getFieldDecorator(item.field + '$shape', { |
| | | initialValue: item.$shape, |
| | | rules: [ |
| | | { |
| | | required: enabled === 'true', |
| | | message: this.props.dict['form.required.select'] + '形状!' |
| | | } |
| | | ] |
| | | })( |
| | | <Cascader |
| | | disabled={enabled === 'false'} |
| | | options={shapeOptions} |
| | | placeholder="" |
| | | displayRender={(label, selectedOptions) => selectedOptions[0] ? selectedOptions[0].label + ' / ' + selectedOptions[1].value : ''} |
| | | /> |
| | | )} |
| | | </Form.Item> |
| | | </Col>) |
| | | fields.push(<Col span={12} key={'color' + i}> |
| | | <Form.Item label="颜色"> |
| | | {getFieldDecorator(item.field + '$color', { |
| | | initialValue: item.$color, |
| | | rules: [ |
| | | { |
| | | required: enabled === 'true', |
| | | message: this.props.dict['form.required.select'] + '颜色!' |
| | | } |
| | | ] |
| | | })( |
| | | <Cascader |
| | | disabled={enabled === 'false'} |
| | | options={colorOptions} |
| | | placeholder="" |
| | | getPopupContainer={() => document.getElementById('chart-custom-drawer-form')} |
| | | displayRender={(label, selectedOptions) => selectedOptions[0] ? <div style={{background: selectedOptions[1].value, width: '20px', height: '20px'}}></div> : ''} |
| | | /> |
| | | )} |
| | | </Form.Item> |
| | | </Col>) |
| | | fields.push(<Col span={12} key={'axis' + i}> |
| | | <Form.Item label="坐标轴"> |
| | | {getFieldDecorator(item.field + '$axis', { |
| | | initialValue: item.axis |
| | | })( |
| | | <Radio.Group disabled={enabled === 'false'} onChange={this.axisChange}> |
| | | <Radio value="left">左侧</Radio> |
| | | <Radio value="right">右侧</Radio> |
| | | <Radio value="unset">不显示</Radio> |
| | | </Radio.Group> |
| | | )} |
| | | </Form.Item> |
| | | </Col>) |
| | | fields.push(<Col span={12} key={'label' + i}> |
| | | <Form.Item label="标注-值"> |
| | | {getFieldDecorator(item.field + '$label', { |
| | | initialValue: item.label |
| | | })( |
| | | <Radio.Group disabled={enabled === 'false'}> |
| | | <Radio value="true">显示</Radio> |
| | | <Radio value="false">隐藏</Radio> |
| | | </Radio.Group> |
| | | )} |
| | | </Form.Item> |
| | | </Col>) |
| | | }) |
| | | |
| | | return fields |
| | | } |
| | | |
| | | axisChange = (e) => { |
| | | enabledChange = (e) => { |
| | | const { plot } = this.state |
| | | let val = e.target.value |
| | | let fieldvalue = {} |
| | | |
| | | plot.customs.forEach(item => { |
| | | if (this.props.form.getFieldValue(item.field + '$axis') === val) { |
| | | fieldvalue[item.field + '$axis'] = 'unset' |
| | | } |
| | | }) |
| | | |
| | | this.props.form.setFieldsValue(fieldvalue) |
| | | } |
| | | |
| | | enabledChange = (e) => { |
| | | let val = e.target.value |
| | | |
| | | this.setState({enabled: val}) |
| | | this.setState({plot: {...plot, enabled: val}}) |
| | | } |
| | | |
| | | onSubmit = () => { |
| | | const { config } = this.props |
| | | const { plot, view, datatype } = this.state |
| | | const { plot, view } = this.state |
| | | |
| | | if (view !== 'custom') { |
| | | if (view === 'normal') { |
| | | this.props.form.validateFieldsAndScroll((err, values) => { |
| | | if (!err) { |
| | | let _plot = {...plot, ...values} |
| | | |
| | | if (datatype === 'query') { |
| | | if (_plot.enabled === 'true') { |
| | | if (_plot.customs.map(_cell => _cell.field).sort().toString() !== _plot.Yaxis.sort().toString()) { |
| | | _plot.customs = null |
| | | _plot.enabled = 'false' |
| | | } |
| | | } |
| | | } else { |
| | | _plot.customs = null |
| | | if (values.datatype === 'statistics' || values.datatype !== plot.datatype) { |
| | | _plot.enabled = 'false' |
| | | } else if (!values.Yaxis || !plot.Yaxis || !is(fromJS(values.Yaxis), fromJS(plot.Yaxis))) { |
| | | _plot.enabled = 'false' |
| | | _plot.colors = null |
| | | } |
| | | |
| | | |
| | | if (values.datatype !== plot.datatype) { |
| | | _plot.colors = null |
| | | } |
| | | |
| | | this.setState({ |
| | | plot: _plot, |
| | | visible: false |
| | |
| | | } |
| | | }) |
| | | } else { |
| | | this.props.form.validateFieldsAndScroll((err, values) => { |
| | | if (!err) { |
| | | let _plot = {...plot, enabled: values.enabled} |
| | | |
| | | if (_plot.enabled === 'true') { |
| | | _plot.customs = _plot.Yaxis.map(field => { |
| | | let _item = {field: field} |
| | | |
| | | _item.$shape = values[field + '$shape'] |
| | | _item.chartType = _item.$shape[0] |
| | | _item.shape = _item.$shape[1] |
| | | |
| | | _item.$color = values[field + '$color'] |
| | | _item.color = _item.$color[1] |
| | | |
| | | _item.axis = values[field + '$axis'] |
| | | |
| | | _item.label = values[field + '$label'] |
| | | |
| | | return _item |
| | | }) |
| | | } else { |
| | | _plot.customs = null |
| | | } |
| | | |
| | | this.setState({ |
| | | plot: _plot, |
| | | visible: false |
| | | }) |
| | | |
| | | this.props.plotchange(_plot) |
| | | } |
| | | this.setState({ |
| | | visible: false |
| | | }) |
| | | this.props.plotchange({...config, plot}) |
| | | } |
| | | } |
| | | |
| | | changeTab = (tab) => { |
| | | const { config } = this.props |
| | | const { plot } = this.state |
| | | const { plot, view } = this.state |
| | | |
| | | if (tab === 'custom') { |
| | | if (view === 'normal') { |
| | | this.props.form.validateFieldsAndScroll((err, values) => { |
| | | if (!err) { |
| | | let _plot = {...plot, ...values} |
| | | |
| | | if (_plot.enabled !== 'true' || _plot.customs.map(_cell => _cell.field).sort().toString() !== _plot.Yaxis.sort().toString()) { |
| | | if (values.datatype === 'statistics' || values.datatype !== plot.datatype) { |
| | | _plot.enabled = 'false' |
| | | _plot.customs = _plot.Yaxis.map((field, i) => { |
| | | let _item = {field: field} |
| | | _item.$shape = [_plot.chartType, _plot.shape] |
| | | _item.$color = [] |
| | | _item.axis = i === 0 ? 'left' : 'unset' |
| | | _item.label = _plot.label |
| | | |
| | | return _item |
| | | _plot.customs = [] |
| | | } else if (!values.Yaxis || !plot.Yaxis || !is(fromJS(values.Yaxis), fromJS(plot.Yaxis))) { |
| | | _plot.enabled = 'false' |
| | | _plot.customs = [] |
| | | _plot.colors = null |
| | | } |
| | | |
| | | let labels = {} |
| | | config.columns.forEach(col => { |
| | | labels[col.field] = col.label |
| | | }) |
| | | |
| | | if (values.datatype !== 'statistics' && (!_plot.customs || _plot.customs.length === 0)) { |
| | | _plot.customs = _plot.Yaxis.map((item, i) => { |
| | | return { |
| | | uuid: Utils.getuuid(), |
| | | type: item, |
| | | name: labels[item] || item, |
| | | axis: i === 0 ? 'true' : 'false', |
| | | label: 'false', |
| | | shape: _plot.chartType === 'bar' ? ['bar', 'rect'] : ['line', 'smooth'] |
| | | } |
| | | }) |
| | | } |
| | | |
| | | if (values.datatype !== plot.datatype || !_plot.colors) { |
| | | _plot.colors = [] |
| | | if (_plot.datatype === 'query') { |
| | | let limit = chartColors.length |
| | | |
| | | _plot.colors = _plot.Yaxis.map((item, i) => { |
| | | return { |
| | | uuid: Utils.getuuid(), |
| | | type: item, |
| | | label: labels[item] || item, |
| | | color: chartColors[i % limit] |
| | | } |
| | | }) |
| | | } |
| | | } |
| | | |
| | | this.setState({ |
| | | enabled: _plot.enabled || 'false', |
| | | datatype: _plot.datatype, |
| | | plot: _plot, |
| | | view: tab |
| | | }) |
| | | } |
| | | }) |
| | | } else { |
| | | this.props.form.validateFieldsAndScroll((err, values) => { |
| | | if (!err) { |
| | | let _values = {enabled: values.enabled} |
| | | |
| | | if (values.enabled === 'true') { |
| | | let _customs = [] |
| | | plot.Yaxis.forEach(field => { |
| | | let _item = {field: field} |
| | | |
| | | _item.$shape = values[field + '$shape'] |
| | | _item.chartType = _item.$shape[0] |
| | | _item.shape = _item.$shape[1] |
| | | |
| | | _item.$color = values[field + '$color'] |
| | | _item.color = _item.$color[1] |
| | | |
| | | _item.axis = values[field + '$axis'] |
| | | _item.label = values[field + '$label'] |
| | | |
| | | _customs.push(_item) |
| | | }) |
| | | |
| | | _values.customs = _customs |
| | | } else { |
| | | _values.customs = null |
| | | } |
| | | |
| | | let _plot = {...plot, ..._values} |
| | | |
| | | this.setState({ |
| | | plot: _plot, |
| | | view: tab, |
| | | formlist: getBarOrLineChartOptionForm(_plot, config.columns) |
| | | }) |
| | | } |
| | | this.setState({ |
| | | view: tab |
| | | }) |
| | | } |
| | | } |
| | | |
| | | addColor = () => { |
| | | let plot = fromJS(this.state.plot).toJS() |
| | | plot.colors = plot.colors || [] |
| | | |
| | | plot.colors.push({ |
| | | uuid: Utils.getuuid(), |
| | | type: `指标${plot.colors.length}`, |
| | | color: 'rgb(91, 143, 249)' |
| | | }) |
| | | |
| | | this.setState({plot}) |
| | | } |
| | | |
| | | changeColor = (colors) => { |
| | | const { plot } = this.state |
| | | |
| | | this.setState({plot: {...plot, colors}}) |
| | | } |
| | | |
| | | changeCustom = (customs) => { |
| | | const { plot } = this.state |
| | | |
| | | this.setState({plot: {...plot, customs}}) |
| | | } |
| | | |
| | | render() { |
| | | const { view, visible, disabled } = this.state |
| | | const { view, visible, datatype, plot, colorColumns, statColorColumns, cusColumns } = this.state |
| | | const formItemLayout = { |
| | | labelCol: { |
| | | xs: { span: 24 }, |
| | |
| | | > |
| | | <Tabs activeKey={view} className="menu-chart-edit-box" onChange={this.changeTab}> |
| | | <TabPane tab="基础设置" key="normal"> |
| | | {view === 'normal' ? <Form {...formItemLayout}> |
| | | <Form {...formItemLayout}> |
| | | <Row gutter={16}>{this.getFields()}</Row> |
| | | </Form> : null} |
| | | </Form> |
| | | </TabPane> |
| | | <TabPane tab="自定义设置" disabled={disabled} key="custom"> |
| | | {view === 'custom' ? <Form {...formItemLayout}> |
| | | <Row gutter={16}>{this.getCustomFields()}</Row> |
| | | </Form> : null} |
| | | </TabPane> |
| | | {plot ? <TabPane tab="颜色设置" key="color"> |
| | | <div> |
| | | {datatype === 'statistics' ? <Button className="color-add mk-green" onClick={this.addColor}>{this.props.dict['model.add']}</Button> : null} |
| | | {datatype === 'statistics' ? <EditTable data={plot.colors || []} columns={statColorColumns} onChange={this.changeColor}/> : null} |
| | | {datatype !== 'statistics' ? <EditTable actions={['edit']} data={plot.colors || []} columns={colorColumns} onChange={this.changeColor}/> : null} |
| | | </div> |
| | | </TabPane> : null} |
| | | {plot ? <TabPane tab="自定义设置" disabled={datatype === 'statistics'} key="custom"> |
| | | <Col span={12}> |
| | | <Form {...formItemLayout}> |
| | | <Form.Item label="是否启用" style={{marginBottom: 10}}> |
| | | <Radio.Group value={plot.enabled || 'false'} onChange={this.enabledChange}> |
| | | <Radio value="true">是</Radio> |
| | | <Radio value="false">否</Radio> |
| | | </Radio.Group> |
| | | </Form.Item> |
| | | </Form> |
| | | </Col> |
| | | <Col style={{fontSize: '12px', color: '#757575', paddingLeft: '10px'}} span={24}>注:使用自定义设置时,显示的坐标轴第一个在左侧,第二个在右侧,多余的不生效;柱形图只可以添加一个(设置多个时,第一个生效)。</Col> |
| | | <EditTable actions={['edit', 'up', 'down']} data={plot.customs || []} columns={cusColumns} onChange={this.changeCustom}/> |
| | | </TabPane> : null} |
| | | </Tabs> |
| | | </Modal> |
| | | </div> |