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 } from 'antd'
|
import { QuestionCircleOutlined } from '@ant-design/icons'
|
|
import { getColumnForm } from './formconfig'
|
import { formRule } from '@/utils/option.js'
|
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'],
|
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 = {
|
visible: PropTypes.bool,
|
column: PropTypes.object,
|
fields: PropTypes.array,
|
submitCol: PropTypes.func, // 提交事件
|
cancelCol: PropTypes.func // 取消时删除事件
|
}
|
|
state = {
|
visible: false,
|
formlist: 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)
|
let _options = fromJS(columnTypeOptions[column.type]).toJS()
|
if (column.type === 'text' || column.type === 'number') {
|
if (column.perspective === 'linkmenu') {
|
_options.push('linkmenu', 'linkfields', 'open')
|
} else if (column.perspective === 'linkurl') {
|
_options.push('linkurl', 'linkfields', 'open')
|
}
|
}
|
|
this.setState({
|
visible: true,
|
type: column.type,
|
formlist: formlist.map(item => {
|
item.hidden = !_options.includes(item.key)
|
|
return item
|
})
|
})
|
if (column.focus) {
|
setTimeout(() => {
|
try {
|
let _form = document.getElementById('label')
|
_form && _form.select()
|
} catch (e) {
|
console.warn('表单focus失败!')
|
}
|
}, 200)
|
}
|
}
|
|
typeChange = (key, value, option) => {
|
if (key === 'type') {
|
let _options = fromJS(columnTypeOptions[value]).toJS()
|
|
this.setState({
|
type: value,
|
formlist: this.state.formlist.map(item => {
|
item.hidden = !_options.includes(item.key)
|
|
return item
|
})
|
}, () => {
|
if (value === 'link' || value === 'textarea' || value === 'picture') {
|
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 === 'action') {
|
this.props.form.setFieldsValue({Align: 'center', label: '操作'})
|
} 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'
|
}
|
|
if (values.type !== this.state.type) {
|
values.perspective = ''
|
let _options = fromJS(columnTypeOptions[values.type]).toJS()
|
|
this.setState({
|
type: values.type,
|
formlist: this.state.formlist.map(item => {
|
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: '%'})
|
}
|
}
|
|
changeRadio = (key, value) => {
|
if (key === 'perspective') {
|
let _options = fromJS(columnTypeOptions[this.state.type]).toJS()
|
|
if (value === 'linkmenu') {
|
_options.push('linkmenu', 'linkfields', 'open')
|
} else if (value === 'linkurl') {
|
_options.push('linkurl', 'linkfields', 'open')
|
}
|
|
this.setState({
|
formlist: this.state.formlist.map(item => {
|
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') {
|
let rules = []
|
if (item.key === 'field') {
|
rules = [{
|
pattern: /^[\u4E00-\u9FA50-9a-zA-Z_]*$/ig,
|
message: '字段名只允许包含数字、字母、汉字以及_'
|
}, {
|
max: formRule.input.max,
|
message: formRule.input.message
|
}]
|
} else if (item.key !== 'linkurl') {
|
rules = [{
|
max: formRule.input.max,
|
message: formRule.input.message
|
}]
|
}
|
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 + '!'
|
},
|
...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.changeRadio(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') {
|
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 = () => {
|
// 表单提交时检查输入值是否正确
|
this.props.form.validateFieldsAndScroll((err, values) => {
|
if (!err) {
|
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="显示列编辑"
|
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)
|