import React, {Component} from 'react'
|
import PropTypes from 'prop-types'
|
import { is, fromJS } from 'immutable'
|
import { Form, Row, Col, Input, Select, InputNumber, Radio, Tooltip, Cascader, Modal, Checkbox, Popover } from 'antd'
|
import { QuestionCircleOutlined } from '@ant-design/icons'
|
|
import { getColumnForm } from './formconfig'
|
import './index.scss'
|
|
const { TextArea } = Input
|
const columnTypeOptions = {
|
text: ['label', 'field', 'type', 'Align', 'Hide', 'IsSort', 'Width', 'prefix', 'postfix', 'textFormat', 'fieldlength', 'blacklist', 'perspective', 'rowspan'],
|
number: ['label', 'field', 'type', 'Align', 'Hide', 'IsSort', 'Width', 'decimal', 'format', 'prefix', 'postfix', 'blacklist', 'perspective', 'sum', 'rowspan'],
|
link: ['label', 'field', 'type', 'Align', 'Hide', 'IsSort', 'joint', 'Width', 'fieldlength', 'blacklist', 'nameField'],
|
textarea: ['label', 'field', 'type', 'Align', 'Hide', 'IsSort', 'Width', 'fieldlength', 'prefix', 'postfix', 'blacklist'],
|
picture: ['label', 'field', 'type', 'Align', 'Hide', 'IsSort', 'Width', 'fieldlength', 'blacklist', 'scale', 'lenWidRadio', 'backgroundSize', 'span'],
|
video: ['label', 'field', 'type', 'Align', 'Hide', 'startTime', 'Width', 'fieldlength', 'blacklist', 'aspectRatio'],
|
colspan: ['label', 'type', 'Align', 'Hide', 'blacklist'],
|
custom: ['label', 'type', 'Align', 'Width', 'blacklist', 'IsSort'],
|
action: ['label', 'type', 'Align', 'Width'],
|
formula: ['label', 'type', 'Align', 'Hide', 'Width', 'prefix', 'postfix', 'eval', 'formula', 'blacklist'],
|
index: ['label', 'type', 'Align', 'Width']
|
}
|
|
class NormalTableColumn extends Component {
|
static propTpyes = {
|
column: PropTypes.object,
|
fields: PropTypes.array,
|
submitCol: PropTypes.func, // 提交事件
|
cancelCol: PropTypes.func // 取消时删除事件
|
}
|
|
state = {
|
visible: false,
|
formlist: null
|
}
|
|
record = null
|
|
UNSAFE_componentWillReceiveProps (nextProps) {
|
if (nextProps.column && !is(fromJS(this.props.column), fromJS(nextProps.column))) {
|
this.editColumn(nextProps.column)
|
}
|
}
|
|
editColumn = (column) => {
|
let fields = fromJS(this.props.fields).toJS().map(item => {
|
item.text = `${item.field}(${item.label})`
|
return item
|
})
|
|
let formlist = getColumnForm(column, fields)
|
|
this.record = {}
|
|
formlist.forEach(item => {
|
this.record[item.key] = item.initVal
|
})
|
|
let _options = this.getOptions()
|
|
this.setState({
|
visible: true,
|
formlist: formlist.map(item => {
|
item.hidden = !_options.includes(item.key)
|
|
if (item.key === 'formula') {
|
item.fields = this.props.fields.map(col => col.field)
|
item.fields = item.fields.join(', ')
|
} else if (this.record.type === 'formula' && item.key === 'decimal') {
|
item.required = false
|
}
|
|
return item
|
})
|
})
|
if (column.focus) {
|
setTimeout(() => {
|
try {
|
let _form = document.getElementById('label')
|
_form && _form.select()
|
} catch (e) {
|
console.warn('表单focus失败!')
|
}
|
}, 200)
|
}
|
}
|
|
getOptions = () => {
|
let _options = fromJS(columnTypeOptions[this.record.type]).toJS()
|
|
if (this.record.type === 'text' || this.record.type === 'number') {
|
if (this.record.perspective === 'linkmenu') {
|
_options.push('linkmenu', 'linkfields', 'open')
|
} else if (this.record.perspective === 'linkurl') {
|
_options.push('linkurl', 'linkfields', 'open')
|
}
|
} else if (this.record.type === 'formula' && this.record.eval === 'true') {
|
_options.push('decimal')
|
} else if (this.record.type === 'custom' && this.record.IsSort === 'true') {
|
_options.push('sortField')
|
}
|
|
return _options
|
}
|
|
typeChange = (key, value, option) => {
|
const { column } = this.props
|
this.record[key] = value
|
|
if (key === 'type') {
|
if (['link', 'textarea', 'picture', 'custom'].includes(value)) {
|
this.record.IsSort = 'false'
|
}
|
|
let _options = this.getOptions()
|
|
this.setState({
|
formlist: this.state.formlist.map(item => {
|
if (item.key === 'decimal') {
|
item.required = !(column.isSub || value === 'formula')
|
if (value === 'formula') {
|
this.record.decimal = ''
|
}
|
}
|
|
item.initVal = this.record[item.key]
|
item.hidden = !_options.includes(item.key)
|
|
return item
|
})
|
}, () => {
|
if (['link', 'textarea', 'picture', 'custom'].includes(value)) {
|
this.props.form.setFieldsValue({IsSort: 'false'})
|
} else if (value === 'text' || value === 'number') {
|
this.props.form.setFieldsValue({perspective: ''})
|
} else if (value === 'colspan') {
|
this.props.form.setFieldsValue({Align: 'center'})
|
} else if (value === 'index') {
|
this.props.form.setFieldsValue({label: '序号'})
|
}
|
})
|
} else if (key === 'field') {
|
let values = {label: option.props.label || option.props.children}
|
if (/Decimal|int/ig.test(option.props.datatype)) {
|
let decimal = 0
|
if (/Decimal/ig.test(option.props.datatype)) {
|
decimal = +option.props.datatype.replace(/Decimal\(18,/ig, '').replace(')', '')
|
}
|
values.type = 'number'
|
values.decimal = decimal
|
} else {
|
values.type = 'text'
|
}
|
|
let _type = this.record.type
|
this.record.type = values.type
|
|
if (values.type !== _type) {
|
values.perspective = ''
|
this.record.perspective = ''
|
|
let _options = this.getOptions()
|
|
this.setState({
|
formlist: this.state.formlist.map(item => {
|
item.initVal = this.record[item.key]
|
item.hidden = !_options.includes(item.key)
|
|
return item
|
})
|
}, () => {
|
this.props.form.setFieldsValue(values)
|
})
|
} else {
|
this.props.form.setFieldsValue(values)
|
}
|
} else if (key === 'format' && value === 'percent') {
|
this.props.form.setFieldsValue({postfix: '%'})
|
} else if (['perspective', 'eval', 'IsSort'].includes(key)) {
|
let _options = this.getOptions()
|
|
this.setState({
|
formlist: this.state.formlist.map(item => {
|
item.initVal = this.record[item.key]
|
item.hidden = !_options.includes(item.key)
|
|
return item
|
})
|
})
|
}
|
}
|
|
getFields() {
|
const { getFieldDecorator } = this.props.form
|
const { formlist } = this.state
|
const fields = []
|
|
if (!formlist) return null
|
|
formlist.forEach((item, index) => {
|
if (item.hidden || item.forbid) return
|
|
if (item.type === 'text') {
|
fields.push(
|
<Col span={12} key={index}>
|
<Form.Item label={item.tooltip ?
|
<Tooltip placement="topLeft" title={item.tooltip}>
|
<QuestionCircleOutlined className="mk-form-tip" />
|
{item.label}
|
</Tooltip> : item.label
|
}>
|
{getFieldDecorator(item.key, {
|
initialValue: item.initVal || '',
|
rules: [
|
{
|
required: !!item.required,
|
message: '请输入' + item.label + '!'
|
},
|
...item.rules
|
]
|
})(<Input placeholder="" autoComplete="off" disabled={item.readonly} onPressEnter={this.handleSubmit} />)}
|
</Form.Item>
|
</Col>
|
)
|
} else if (item.type === 'number') {
|
fields.push(
|
<Col span={12} key={index}>
|
<Form.Item label={item.tooltip ?
|
<Tooltip placement="topLeft" title={item.tooltip}>
|
<QuestionCircleOutlined className="mk-form-tip" />
|
{item.label}
|
</Tooltip> : item.label
|
}>
|
{getFieldDecorator(item.key, {
|
initialValue: item.initVal,
|
rules: [
|
{
|
required: !!item.required,
|
message: '请输入' + item.label + '!'
|
}
|
]
|
})(item.unlimit ? <InputNumber onPressEnter={this.handleSubmit}/> :
|
<InputNumber min={item.min} max={item.max} precision={item.decimal} onPressEnter={this.handleSubmit}/>)}
|
</Form.Item>
|
</Col>
|
)
|
} else if (item.type === 'select') {
|
fields.push(
|
<Col span={12} key={index}>
|
<Form.Item label={item.label}>
|
{getFieldDecorator(item.key, {
|
initialValue: item.initVal || '',
|
rules: [
|
{
|
required: !!item.required,
|
message: '请选择' + item.label + '!'
|
}
|
]
|
})(
|
<Select
|
showSearch
|
filterOption={(input, option) => option.props.children.toLowerCase().indexOf(input.toLowerCase()) >= 0}
|
onChange={(value, option) => {this.typeChange(item.key, value, option)}}
|
getPopupContainer={() => document.getElementById('columnwinter')}
|
>
|
{item.options.map((option, index) =>
|
<Select.Option key={index} datatype={option.datatype || ''} label={option.label || ''} value={(option.value || option.field || option.MenuID)}>
|
{(option.text || option.label || option.MenuName)}
|
</Select.Option>
|
)}
|
</Select>
|
)}
|
</Form.Item>
|
</Col>
|
)
|
} else if (item.type === 'radio') {
|
fields.push(
|
<Col span={12} key={index}>
|
<Form.Item label={item.tooltip ?
|
<Tooltip placement="topLeft" title={item.tooltip}>
|
<QuestionCircleOutlined className="mk-form-tip" />
|
{item.label}
|
</Tooltip> : item.label
|
}>
|
{getFieldDecorator(item.key, {
|
initialValue: item.initVal,
|
rules: [
|
{
|
required: !!item.required,
|
message: '请选择' + item.label + '!'
|
}
|
]
|
})(
|
<Radio.Group onChange={(e) => {this.typeChange(item.key, e.target.value)}}>
|
{item.options.map(option => {
|
return (<Radio key={option.value} value={option.value}>{option.text}</Radio>)
|
})}
|
</Radio.Group>
|
)}
|
</Form.Item>
|
</Col>
|
)
|
} else if (item.type === 'checkbox') {
|
fields.push(
|
<Col span={12} key={index}>
|
<Form.Item label={item.tooltip ?
|
<Tooltip placement="topLeft" title={item.tooltip}>
|
<QuestionCircleOutlined className="mk-form-tip" />
|
{item.label}
|
</Tooltip> : item.label
|
}>
|
{getFieldDecorator(item.key, {
|
initialValue: item.initVal
|
})(
|
<Checkbox.Group>
|
{item.options.map(option => <Checkbox key={option.value} value={option.value}>{option.text}</Checkbox>)}
|
</Checkbox.Group>
|
)}
|
</Form.Item>
|
</Col>
|
)
|
} else if (item.type === 'multiselect') { // 多选
|
fields.push(
|
<Col span={12} key={index}>
|
<Form.Item label={item.label}>
|
{getFieldDecorator(item.key, {
|
initialValue: item.initVal || []
|
})(
|
<Select
|
showSearch
|
mode="multiple"
|
filterOption={(input, option) => option.props.children.toLowerCase().indexOf(input.toLowerCase()) >= 0}
|
>
|
{item.options.map((option, i) =>
|
<Select.Option id={i} key={i} value={option.value || option.field}>{option.text || option.label}</Select.Option>
|
)}
|
</Select>
|
)}
|
</Form.Item>
|
</Col>
|
)
|
} else if (item.type === 'cascader') { // 多选
|
fields.push(
|
<Col span={12} key={index}>
|
<Form.Item label={item.label}>
|
{getFieldDecorator(item.key, {
|
initialValue: item.initVal || [],
|
rules: [
|
{
|
required: !!item.required,
|
message: '请选择' + item.label + '!'
|
}
|
]
|
})(
|
<Cascader
|
options={item.options}
|
placeholder=""
|
getPopupContainer={() => document.getElementById('columnwinter')}
|
/>
|
)}
|
</Form.Item>
|
</Col>
|
)
|
} else if (item.type === 'textarea') {
|
if (item.key === 'formula') {
|
fields.push(
|
<Col span={24} className="textarea" key={index}>
|
<Form.Item label={item.tooltip ?
|
<Tooltip placement="topLeft" title={item.tooltip}>
|
<QuestionCircleOutlined className="mk-form-tip" />
|
{item.label}
|
</Tooltip> : item.label
|
}>
|
{getFieldDecorator(item.key, {
|
initialValue: item.initVal || '',
|
rules: [
|
{
|
required: !!item.required,
|
message: '请输入' + item.label + '!'
|
}
|
]
|
})(<TextArea autoSize={{minRows: 2}} disabled={item.readonly} placeholder={item.placeholder || ''} />)}
|
</Form.Item>
|
<Popover overlayClassName="formula-fields" placement="topLeft" title="" content={<div>{item.fields}</div>} trigger="click">
|
<span className="formula-icon">字段集</span>
|
</Popover>
|
</Col>
|
)
|
} else {
|
fields.push(
|
<Col span={24} key={index} className="textarea">
|
<Form.Item label={item.tooltip ?
|
<Tooltip placement="topLeft" title={item.tooltip}>
|
<QuestionCircleOutlined className="mk-form-tip" />
|
{item.label}
|
</Tooltip> : item.label
|
}>
|
{getFieldDecorator(item.key, {
|
initialValue: item.initVal || '',
|
rules: [
|
{
|
required: !!item.required,
|
message: '请输入' + item.label + '!'
|
}
|
]
|
})(<TextArea rows={2} disabled={item.readonly} placeholder={item.placeholder || ''}/>)}
|
</Form.Item>
|
</Col>
|
)
|
}
|
}
|
})
|
return fields
|
}
|
|
handleSubmit = () => {
|
const { fields } = this.props
|
// 表单提交时检查输入值是否正确
|
this.props.form.validateFieldsAndScroll((err, values) => {
|
if (!err) {
|
// eslint-disable-next-line
|
if (values.type === 'formula' && values.eval === 'true' && /^[\u4E00-\u9FA50-9a-zA-Z_\s@\+\-\*\/]*$/ig.test(values.formula) && /[\+\-\*\/]/ig.test(values.formula)) {
|
let cols = []
|
fields.forEach(col => {
|
if (/^(Int|Decimal)/ig.test(col.datatype)) {
|
cols.push({reg: new RegExp('@' + col.field + '@', 'ig'), value: `(@${col.field}@)`})
|
}
|
})
|
|
cols.forEach(col => {
|
values.formula = values.formula.replace(col.reg, col.value)
|
})
|
}
|
|
this.props.submitCol(values, () => {
|
this.setState({visible: false, formlist: null})
|
})
|
}
|
})
|
}
|
|
editModalCancel = () => {
|
this.setState({visible: false, formlist: null})
|
|
this.props.cancelCol()
|
}
|
|
render() {
|
const { visible } = this.state
|
const formItemLayout = {
|
labelCol: {
|
xs: { span: 24 },
|
sm: { span: 6 }
|
},
|
wrapperCol: {
|
xs: { span: 24 },
|
sm: { span: 18 }
|
}
|
}
|
|
return (
|
<div style={{display: 'inline-block'}}>
|
<Modal
|
title="显示列编辑"
|
wrapClassName="mk-scroll-modal"
|
visible={visible}
|
width={850}
|
maskClosable={false}
|
onOk={this.handleSubmit}
|
onCancel={this.editModalCancel}
|
destroyOnClose
|
>
|
<Form {...formItemLayout} className="commontable-column-form" id="columnwinter">
|
<Row gutter={24}>{this.getFields()}</Row>
|
</Form>
|
</Modal>
|
</div>
|
)
|
}
|
}
|
|
export default Form.create()(NormalTableColumn)
|