vxe-table
Version:
A PC-end table component based on Vxe UI, supporting copy-paste, data pivot table, and high-performance virtual list table solution.
631 lines (607 loc) • 22.6 kB
text/typescript
import { nextTick } from 'vue'
import XEUtils from 'xe-utils'
import { VxeUI } from '../../../ui'
import { eqEmptyValue, getFuncText } from '../../../ui/src/utils'
import { scrollToView } from '../../../ui/src/dom'
import { handleFieldOrColumn, getRowid } from '../../src/util'
import { warnLog, errLog } from '../../../ui/src/log'
import type { TableValidatorMethods, TableValidatorPrivateMethods, VxeTableDefines } from '../../../../types'
const { getConfig, validators, hooks } = VxeUI
/**
* 校验规则
*/
class Rule {
constructor (rule: any) {
Object.assign(this, {
$options: rule,
required: rule.required,
min: rule.min,
max: rule.max,
type: rule.type,
pattern: rule.pattern,
validator: rule.validator,
trigger: rule.trigger,
maxWidth: rule.maxWidth
})
}
/**
* 获取校验不通过的消息
* 支持国际化翻译
*/
get content () {
return getFuncText(this.$options.content || this.$options.message)
}
get message () {
return this.content
}
[key: string]: any
}
// 如果存在 pattern,判断正则
function validREValue (pattern: string | RegExp | undefined, val: string) {
if (pattern && !(XEUtils.isRegExp(pattern) ? pattern : new RegExp(pattern)).test(val)) {
return false
}
return true
}
// 如果存在 max,判断最大值
function validMaxValue (max: string | number | undefined, num: number) {
if (!XEUtils.eqNull(max) && num > XEUtils.toNumber(max)) {
return false
}
return true
}
// 如果存在 min,判断最小值
function validMinValue (min: string | number | undefined, num: number) {
if (!XEUtils.eqNull(min) && num < XEUtils.toNumber(min)) {
return false
}
return true
}
function validRuleValue (rule: VxeTableDefines.ValidatorRule, val: any, required: boolean | undefined) {
const { type, min, max, pattern } = rule
const isArrType = type === 'array'
const isNumType = type === 'number'
const isStrType = type === 'string'
const strVal = `${val}`
if (!validREValue(pattern, strVal)) {
return false
}
if (isArrType) {
if (!XEUtils.isArray(val)) {
return false
}
if (required) {
if (!val.length) {
return false
}
}
if (!validMinValue(min, val.length)) {
return false
}
if (!validMaxValue(max, val.length)) {
return false
}
} else if (isNumType) {
const numVal = Number(val)
if (isNaN(numVal)) {
return false
}
if (!validMinValue(min, numVal)) {
return false
}
if (!validMaxValue(max, numVal)) {
return false
}
} else {
if (isStrType) {
if (!XEUtils.isString(val)) {
return false
}
}
if (required) {
if (!strVal) {
return false
}
}
if (!validMinValue(min, strVal.length)) {
return false
}
if (!validMaxValue(max, strVal.length)) {
return false
}
}
return true
}
function checkRuleStatus (rule: VxeTableDefines.ValidatorRule, val: any) {
const { required } = rule
const isEmptyVal = XEUtils.isArray(val) ? !val.length : eqEmptyValue(val)
if (required) {
if (isEmptyVal) {
return false
}
if (!validRuleValue(rule, val, required)) {
return false
}
} else {
if (!isEmptyVal) {
if (!validRuleValue(rule, val, required)) {
return false
}
}
}
return true
}
const tableValidatorMethodKeys: (keyof TableValidatorMethods)[] = ['fullValidate', 'validate', 'fullValidateField', 'validateField', 'clearValidate']
hooks.add('tableValidatorModule', {
setupTable ($xeTable) {
const { props, reactData, internalData } = $xeTable
const { refValidTooltip } = $xeTable.getRefMaps()
const { computeValidOpts, computeTreeOpts, computeEditOpts, computeAggregateOpts } = $xeTable.getComputeMaps()
let validatorMethods = {} as TableValidatorMethods
let validatorPrivateMethods = {} as TableValidatorPrivateMethods
let validRuleErr: boolean
/**
* 聚焦到校验通过的单元格并弹出校验错误提示
*/
const handleValidError = (params: any): Promise<void> => {
return new Promise(resolve => {
const validOpts = computeValidOpts.value
if (validOpts.autoPos === false) {
$xeTable.dispatchEvent('valid-error', params, null)
resolve()
} else {
$xeTable.handleEdit(params, { type: 'valid-error', trigger: 'call' }).then(() => {
resolve(validatorPrivateMethods.showValidTooltip(params))
})
}
})
}
const handleErrMsgMode = (validErrMaps: Record<string, {
row: any;
column: any;
rule: any;
content: any;
}>) => {
const validOpts = computeValidOpts.value
if (validOpts.msgMode === 'single') {
const keys = Object.keys(validErrMaps)
const resMaps: Record<string, {
row: any;
column: any;
rule: any;
content: any;
}> = {}
if (keys.length) {
const firstKey = keys[0]
resMaps[firstKey] = validErrMaps[firstKey]
}
return resMaps
}
return validErrMaps
}
/**
* 对表格数据进行校验
* 如果不指定数据,则默认只校验临时变动的数据,例如新增或修改
* 如果传 true 则校验当前表格数据
* 如果传 row 指定行记录,则只验证传入的行
* 如果传 rows 为多行记录,则只验证传入的行
* 如果只传 callback 否则默认验证整个表格数据
* 返回 Promise 对象,或者使用回调方式
*/
const beginValidate = (rows: any, cols: VxeTableDefines.ColumnInfo[] | null, cb: any, isFull?: boolean): Promise<any> => {
const validRest: any = {}
const { editRules, treeConfig } = props
const { isRowGroupStatus } = reactData
const { afterFullData, pendingRowMaps, removeRowMaps } = internalData
const treeOpts = computeTreeOpts.value
const aggregateOpts = computeAggregateOpts.value
const validOpts = computeValidOpts.value
let validList
if (rows === true) {
validList = afterFullData
} else if (rows) {
if (XEUtils.isFunction(rows)) {
cb = rows
} else {
validList = XEUtils.isArray(rows) ? rows : [rows]
}
}
if (!validList) {
if ($xeTable.getInsertRecords) {
validList = $xeTable.getInsertRecords().concat($xeTable.getUpdateRecords())
} else {
validList = []
}
}
const rowValidErrs: any = []
internalData._lastCallTime = Date.now()
validRuleErr = false // 如果为快速校验,当存在某列校验不通过时将终止执行
validatorMethods.clearValidate()
const validErrMaps: Record<string, {
row: any;
column: any;
rule: any;
content: any;
}> = {}
if (editRules) {
const columns = cols && cols.length ? cols : $xeTable.getColumns()
const handleVaild = (row: any) => {
const rowid = getRowid($xeTable, row)
// 是否删除
if (removeRowMaps[rowid]) {
return
}
// 是否标记删除
if (pendingRowMaps[rowid]) {
return
}
if ($xeTable.isAggregateRecord(row)) {
return
}
if (isFull || !validRuleErr) {
const colVailds: any[] = []
columns.forEach((column) => {
const field = XEUtils.isString(column) ? column : column.field
if ((isFull || !validRuleErr) && XEUtils.has(editRules, field)) {
colVailds.push(
validatorPrivateMethods.validCellRules('all', row, column)
.catch(({ rule, rules }) => {
const rest = {
rule,
rules,
rowIndex: $xeTable.getRowIndex(row),
row,
columnIndex: $xeTable.getColumnIndex(column),
column,
field,
$table: $xeTable
}
if (!validRest[field]) {
validRest[field] = []
}
validErrMaps[`${getRowid($xeTable, row)}:${column.id}`] = {
column,
row,
rule,
content: rule.content
}
validRest[field].push(rest)
if (!isFull) {
validRuleErr = true
return Promise.reject(rest)
}
})
)
}
})
rowValidErrs.push(Promise.all(colVailds))
}
}
if (isRowGroupStatus) {
XEUtils.eachTree(validList, handleVaild, { children: aggregateOpts.mapChildrenField })
} else if (treeConfig) {
const childrenField = treeOpts.children || treeOpts.childrenField
XEUtils.eachTree(validList, handleVaild, { children: childrenField })
} else {
validList.forEach(handleVaild)
}
return Promise.all(rowValidErrs).then(() => {
const ruleProps = Object.keys(validRest)
reactData.validErrorMaps = handleErrMsgMode(validErrMaps)
return nextTick().then(() => {
if (ruleProps.length) {
return Promise.reject(validRest[ruleProps[0]][0])
}
if (cb) {
cb()
}
})
}).catch(firstErrParams => {
return new Promise<void>((resolve, reject) => {
const finish = () => {
nextTick(() => {
if (cb) {
cb(validRest)
resolve()
} else {
if (getConfig().validToReject === 'obsolete') {
// 已废弃,校验失败将不会执行catch
reject(validRest)
} else {
resolve(validRest)
}
}
})
}
const posAndFinish = () => {
firstErrParams.cell = $xeTable.getCellElement(firstErrParams.row, firstErrParams.column)
scrollToView(firstErrParams.cell)
handleValidError(firstErrParams).then(finish)
}
/**
* 当校验不通过时
* 将表格滚动到可视区
* 由于提示信息至少需要占一行,定位向上偏移一行
*/
if (validOpts.autoPos === false) {
finish()
} else {
const row = firstErrParams.row
const column = firstErrParams.column
$xeTable.scrollToRow(row, column).then(posAndFinish)
}
})
})
} else {
reactData.validErrorMaps = {}
}
return nextTick().then(() => {
if (cb) {
cb()
}
})
}
validatorMethods = {
/**
* 完整校验行,和 validate 的区别就是会给有效数据中的每一行进行校验
*/
fullValidate (rows, cb) {
if (XEUtils.isFunction(cb)) {
warnLog('vxe.error.notValidators', ['fullValidate(rows, callback)', 'fullValidate(rows)'])
}
return beginValidate(rows, null, cb, true)
},
/**
* 快速校验行,如果存在记录不通过的记录,则返回不再继续校验(异步校验除外)
*/
validate (rows, cb) {
return beginValidate(rows, null, cb)
},
/**
* 完整校验单元格,和 validateField 的区别就是会给有效数据中的每一行进行校验
*/
fullValidateField (rows, fieldOrColumn) {
const colList = (XEUtils.isArray(fieldOrColumn) ? fieldOrColumn : (fieldOrColumn ? [fieldOrColumn] : [])).map(column => handleFieldOrColumn($xeTable, column)) as VxeTableDefines.ColumnInfo<any>[]
if (colList.length) {
return beginValidate(rows, colList, null, true)
}
return nextTick()
},
/**
* 快速校验单元格,如果存在记录不通过的记录,则返回不再继续校验(异步校验除外)
*/
validateField (rows, fieldOrColumn) {
const colList = (XEUtils.isArray(fieldOrColumn) ? fieldOrColumn : (fieldOrColumn ? [fieldOrColumn] : [])).map(column => handleFieldOrColumn($xeTable, column)) as VxeTableDefines.ColumnInfo<any>[]
if (colList.length) {
return beginValidate(rows, colList, null)
}
return nextTick()
},
clearValidate (rows, fieldOrColumn) {
const { validErrorMaps } = reactData
const validTip = refValidTooltip.value
const validOpts = computeValidOpts.value
const rowList = XEUtils.isArray(rows) ? rows : (rows ? [rows] : [])
const colList = (XEUtils.isArray(fieldOrColumn) ? fieldOrColumn : (fieldOrColumn ? [fieldOrColumn] : [])).map(column => handleFieldOrColumn($xeTable, column)) as VxeTableDefines.ColumnInfo<any>[]
let validErrMaps: Record<string, {
row: any;
column: any;
rule: any;
content: any;
}> = {}
if (validTip && validTip.reactData.visible) {
validTip.close()
}
// 如果是单个提示模式
if (validOpts.msgMode === 'single') {
reactData.validErrorMaps = {}
return nextTick()
}
if (rowList.length && colList.length) {
validErrMaps = Object.assign({}, validErrorMaps)
rowList.forEach(row => {
colList.forEach((column) => {
const validKey = `${getRowid($xeTable, row)}:${column.id}`
if (validErrMaps[validKey]) {
delete validErrMaps[validKey]
}
})
})
} else if (rowList.length) {
const rowIdList = rowList.map(row => `${getRowid($xeTable, row)}`)
XEUtils.each(validErrorMaps, (item, key) => {
if (rowIdList.indexOf(key.split(':')[0]) > -1) {
validErrMaps[key] = item
}
})
} else if (colList.length) {
const colidList = colList.map(column => `${column.id}`)
XEUtils.each(validErrorMaps, (item, key) => {
if (colidList.indexOf(key.split(':')[1]) > -1) {
validErrMaps[key] = item
}
})
}
reactData.validErrorMaps = validErrMaps
return nextTick()
}
}
validatorPrivateMethods = {
/**
* 校验数据
* 按表格行、列顺序依次校验(同步或异步)
* 校验规则根据索引顺序依次校验,如果是异步则会等待校验完成才会继续校验下一列
* 如果校验失败则,触发回调或者Promise<不通过列的错误消息>
* 如果是传回调方式这返回一个校验不通过列的错误消息
*
* rule 配置:
* required=Boolean 是否必填
* min=Number 最小长度
* max=Number 最大长度
* validator=Function({ cellValue, rule, rules, row, column, rowIndex, columnIndex }) 自定义校验,接收一个 Promise
* trigger=blur|change 触发方式(除非特殊场景,否则默认为空就行)
*/
validCellRules (validType, row, column, val) {
const $xeGrid = $xeTable.xeGrid
const { editRules } = props
const { field } = column
const errorRules: Rule[] = []
const syncValidList: Promise<any>[] = []
if (field && editRules) {
const rules = XEUtils.get(editRules, field)
if (rules) {
const cellValue = XEUtils.isUndefined(val) ? XEUtils.get(row, field) : val
rules.forEach((rule) => {
const { trigger, validator } = rule
if (validType === 'all' || !trigger || validType === trigger) {
if (validator) {
const validParams = {
cellValue,
rule,
rules,
row,
rowIndex: $xeTable.getRowIndex(row),
column,
columnIndex: $xeTable.getColumnIndex(column),
field: column.field,
$table: $xeTable,
$grid: $xeGrid
}
let customValid: any
if (XEUtils.isString(validator)) {
const gvItem = validators.get(validator)
if (gvItem) {
const tcvMethod = gvItem.tableCellValidatorMethod || gvItem.cellValidatorMethod
if (tcvMethod) {
customValid = tcvMethod(validParams)
} else {
errLog('vxe.error.notValidators', [validator])
}
} else {
errLog('vxe.error.notValidators', [validator])
}
} else {
customValid = validator(validParams)
}
if (customValid) {
if (XEUtils.isError(customValid)) {
validRuleErr = true
errorRules.push(new Rule({ type: 'custom', trigger, content: customValid.message, rule: new Rule(rule) }))
} else if (customValid.catch) {
// 如果为异步校验(注:异步校验是并发无序的)
syncValidList.push(
customValid.catch((e: any) => {
validRuleErr = true
errorRules.push(new Rule({ type: 'custom', trigger, content: e && e.message ? e.message : (rule.content || rule.message), rule: new Rule(rule) }))
})
)
}
}
} else {
if (!checkRuleStatus(rule, cellValue)) {
validRuleErr = true
errorRules.push(new Rule(rule))
}
}
}
})
}
}
return Promise.all(syncValidList).then(() => {
if (errorRules.length) {
const rest = { rules: errorRules, rule: errorRules[0] }
return Promise.reject(rest)
}
})
},
hasCellRules (type, row, column) {
const { editRules } = props
const { field } = column
if (field && editRules) {
const rules = XEUtils.get(editRules, field)
return rules && !!XEUtils.find(rules, rule => type === 'all' || !rule.trigger || type === rule.trigger)
}
return false
},
/**
* 触发校验
*/
triggerValidate (type) {
const { editConfig, editRules } = props
const { editStore } = reactData
const { actived } = editStore
const editOpts = computeEditOpts.value
const validOpts = computeValidOpts.value
// 检查清除校验消息
if (editRules && validOpts.msgMode === 'single') {
reactData.validErrorMaps = {}
}
// 校验单元格
if (editConfig && editRules && actived.row) {
const { row, column, cell } = actived.args
if (validatorPrivateMethods.hasCellRules(type, row, column)) {
return validatorPrivateMethods.validCellRules(type, row, column).then(() => {
if (editOpts.mode === 'row') {
validatorMethods.clearValidate(row, column)
}
}).catch(({ rule }: any) => {
// 如果校验不通过与触发方式一致,则聚焦提示错误,否则跳过并不作任何处理
if (!rule.trigger || type === rule.trigger) {
const rest = { rule, row, column, cell }
validatorPrivateMethods.showValidTooltip(rest)
return Promise.reject(rest)
}
return Promise.resolve()
})
}
}
return Promise.resolve()
},
/**
* 弹出校验错误提示
*/
showValidTooltip (params) {
const { height } = props
const { tableData, validStore, validErrorMaps } = reactData
const { rule, row, column, cell } = params
const validOpts = computeValidOpts.value
const validTip = refValidTooltip.value
const content = rule.content
validStore.visible = true
if (validOpts.msgMode === 'single') {
reactData.validErrorMaps = {
[`${getRowid($xeTable, row)}:${column.id}`]: {
column,
row,
rule,
content
}
}
} else {
reactData.validErrorMaps = Object.assign({}, validErrorMaps, {
[`${getRowid($xeTable, row)}:${column.id}`]: {
column,
row,
rule,
content
}
})
}
$xeTable.dispatchEvent('valid-error', params, null)
if (validTip) {
if (validTip && (validOpts.message === 'tooltip' || (validOpts.message === 'default' && !height && tableData.length < 2))) {
return validTip.open(cell, content)
}
}
return nextTick()
}
}
return { ...validatorMethods, ...validatorPrivateMethods }
},
setupGrid ($xeGrid) {
return $xeGrid.extendTableMethods(tableValidatorMethodKeys)
},
setupGantt ($xeGantt) {
return $xeGantt.extendTableMethods(tableValidatorMethodKeys)
}
})