feat:采集设备-点位规则添加四则运算规则

pull/1/head
黄伟杰 1 month ago
parent 4fc27674a8
commit dd34e163a7

@ -434,11 +434,83 @@
</el-select> </el-select>
</el-form-item> </el-form-item>
<el-form-item :label="isCountIdentifier ? '四则运算规则' : t('DataCollection.DeviceModel.ruleDialogRule')"> <el-form-item :label="isCountIdentifier ? '四则运算规则' : t('DataCollection.DeviceModel.ruleDialogRule')">
<div class="flex items-center gap-8px"> <div v-if="isCountIdentifier" class="w-full flex flex-col gap-8px">
<div
v-for="(item, index) in countArithmeticRules"
:key="index"
class="flex items-center gap-8px"
>
<el-select
v-model="item.ruleAttributeId"
:placeholder="t('DataCollection.DeviceModel.ruleDialogRuleAttributePlaceholder')"
class="!w-240px"
clearable
>
<el-option
v-for="attr in ruleAttributeOptions"
:key="attr.id"
:label="attr.attributeName || attr.attributeCode"
:value="attr.id"
/>
</el-select>
<el-select
v-model="item.ruleOperator"
:placeholder="t('DataCollection.DeviceModel.ruleDialogRuleOperatorPlaceholder')"
class="!w-160px"
clearable
>
<el-option
v-for="operator in currentRuleOperatorOptions"
:key="operator.value"
:label="operator.label"
:value="operator.value"
/>
</el-select>
<el-input
v-model="item.ruleValue"
:placeholder="t('DataCollection.DeviceModel.ruleDialogRuleValuePlaceholder')"
class="!w-200px"
clearable
/>
<template v-if="index < countArithmeticRules.length - 1">
<el-select
v-model="countArithmeticRules[index].bigOperator"
:placeholder="t('DataCollection.DeviceModel.ruleDialogRuleOperatorPlaceholder')"
class="!w-110px"
clearable
>
<el-option
v-for="operator in countOperatorOptions"
:key="operator.value"
:label="operator.label"
:value="operator.value"
/>
</el-select>
</template>
<template v-else>
<div class="flex items-center gap-4px">
<el-button link type="primary" @click="handleAddCountArithmeticRule">
<Icon icon="ep:plus" />
</el-button>
<el-button
link
type="danger"
:disabled="countArithmeticRules.length <= 1"
@click="handleRemoveCountArithmeticRule(index)"
>
<Icon icon="ep:minus" />
</el-button>
</div>
</template>
</div>
<div class="text-12px text-[#909399] leading-20px">四则运算规则添加顺序从上到下依次计算不按乘除优先</div>
</div>
<div v-else class="flex items-center gap-8px">
<el-select <el-select
v-model="ruleForm.ruleAttributeId" v-model="ruleForm.ruleAttributeId"
:placeholder="t('DataCollection.DeviceModel.ruleDialogRuleAttributePlaceholder')" :placeholder="t('DataCollection.DeviceModel.ruleDialogRuleAttributePlaceholder')"
class="!w-240px" class="!w-240px"
clearable
> >
<el-option <el-option
v-for="item in ruleAttributeOptions" v-for="item in ruleAttributeOptions"
@ -451,6 +523,7 @@
v-model="ruleForm.ruleOperator" v-model="ruleForm.ruleOperator"
:placeholder="t('DataCollection.DeviceModel.ruleDialogRuleOperatorPlaceholder')" :placeholder="t('DataCollection.DeviceModel.ruleDialogRuleOperatorPlaceholder')"
class="!w-160px" class="!w-160px"
clearable
> >
<el-option <el-option
v-for="item in currentRuleOperatorOptions" v-for="item in currentRuleOperatorOptions"
@ -460,10 +533,11 @@
/> />
</el-select> </el-select>
<el-input <el-input
v-if="isCountIdentifier || (ruleForm.ruleOperator !== 'TRUE' && ruleForm.ruleOperator !== 'FALSE')" v-if="ruleForm.ruleOperator !== 'TRUE' && ruleForm.ruleOperator !== 'FALSE'"
v-model="ruleForm.ruleValue" v-model="ruleForm.ruleValue"
:placeholder="t('DataCollection.DeviceModel.ruleDialogRuleValuePlaceholder')" :placeholder="t('DataCollection.DeviceModel.ruleDialogRuleValuePlaceholder')"
class="!w-200px" class="!w-200px"
clearable
/> />
</div> </div>
</el-form-item> </el-form-item>
@ -554,6 +628,7 @@
<el-input <el-input
v-model="createRuleForm.fieldName" v-model="createRuleForm.fieldName"
:placeholder="t('DataCollection.DeviceModel.ruleSearchFieldNamePlaceholder')" :placeholder="t('DataCollection.DeviceModel.ruleSearchFieldNamePlaceholder')"
:disabled="isCreateCountRule"
/> />
</el-form-item> </el-form-item>
<el-form-item :label="t('DataCollection.DeviceModel.ruleDialogDefaultValue')"> <el-form-item :label="t('DataCollection.DeviceModel.ruleDialogDefaultValue')">
@ -1079,6 +1154,18 @@ const ruleForm = reactive<Partial<DevicePointRuleVO>>({
ruleValue: undefined, ruleValue: undefined,
alarmLevel: '', alarmLevel: '',
}) })
type CountArithmeticRule = {
ruleAttributeId?: number
ruleOperator?: string
ruleValue?: string | number
bigOperator?: string
}
const createEmptyCountArithmeticRule = (): CountArithmeticRule => ({
ruleAttributeId: undefined,
ruleOperator: undefined,
ruleValue: undefined,
})
const countArithmeticRules = ref<CountArithmeticRule[]>([createEmptyCountArithmeticRule()])
const extraPointRules = ref< const extraPointRules = ref<
Array<{ Array<{
@ -1098,10 +1185,10 @@ const runningRuleOptions = [
const alarmRuleOptions = [{ value: '5', label: '报警' }] const alarmRuleOptions = [{ value: '5', label: '报警' }]
const countOperatorOptions = [ const countOperatorOptions = [
{ value: '+', label: '' }, { value: '+', label: '+' },
{ value: '-', label: '' }, { value: '-', label: '-' },
{ value: '*', label: '' }, { value: '*', label: '*' },
{ value: '/', label: '' }, { value: '/', label: '/' },
] ]
const currentRuleOptions = computed(() => { const currentRuleOptions = computed(() => {
@ -1139,6 +1226,146 @@ const currentRuleOperatorOptions = computed(() => {
const isRunningIdentifier = computed(() => { const isRunningIdentifier = computed(() => {
return (ruleForm.identifier || '').toString().toUpperCase() === 'RUNNING' return (ruleForm.identifier || '').toString().toUpperCase() === 'RUNNING'
}) })
const resolveRuleAttributeIdByCode = (code: any) => {
if (code == null) return undefined
const targetCode = String(code).trim().toUpperCase()
const target = ruleAttributeOptions.value.find(
(item) => String(item?.attributeCode ?? '').trim().toUpperCase() === targetCode
)
return target?.id
}
const toCountRuleValue = (value: any) => {
const text = String(value ?? '').trim()
if (!text) return ''
if (/^-?\d+(\.\d+)?$/.test(text)) {
return Number(text)
}
return text
}
const parseAlarmRunningFieldRule = (fieldRule: any) => {
if (!fieldRule) return [] as any[]
if (Array.isArray(fieldRule)) return fieldRule
if (typeof fieldRule === 'string') {
try {
const parsed = JSON.parse(fieldRule)
return Array.isArray(parsed) ? parsed : []
} catch {
return []
}
}
return []
}
const buildAlarmRunningFieldRule = () => {
const list: Array<{
code: string
operator: string
operatorRule: string | number
rule: string
}> = []
if (ruleForm.ruleAttributeId && ruleForm.ruleOperator) {
const attr = ruleAttributeOptions.value.find((item) => item.id === ruleForm.ruleAttributeId)
if (attr?.attributeCode) {
list.push({
code: attr.attributeCode,
operator: String(ruleForm.ruleOperator),
operatorRule: (ruleForm.ruleValue as any) ?? '',
rule: String(ruleForm.fieldRule ?? ''),
})
}
}
extraPointRules.value.forEach((item) => {
if (!item.id || !item.operator) return
const attr = ruleAttributeOptions.value.find((opt) => opt.id === item.id)
if (!attr?.attributeCode) return
list.push({
code: attr.attributeCode,
operator: String(item.operator),
operatorRule: (item.operatorRule as any) ?? '',
rule: String(item.rule ?? ruleForm.fieldRule ?? ''),
})
})
return JSON.stringify(list)
}
const buildCountFieldRule = (rules: CountArithmeticRule[]) => {
const validRules = rules.filter((item) => {
return item.ruleAttributeId && item.ruleOperator && String(item.ruleValue ?? '').trim() !== ''
})
if (!validRules.length) return ''
const list = validRules.map((item, index) => {
const attr = ruleAttributeOptions.value.find((opt) => opt.id === item.ruleAttributeId)
if (!attr?.attributeCode) return undefined
const row: any = {
code: attr.attributeCode,
operator: String(item.ruleOperator),
value: toCountRuleValue(item.ruleValue),
}
if (index < validRules.length - 1 && item.bigOperator) {
row.bigOperator = String(item.bigOperator)
}
return row
}).filter(Boolean)
if (!list.length) return ''
return JSON.stringify(list)
}
const parseCountFieldRule = (fieldRule: any): CountArithmeticRule[] => {
if (!fieldRule) return []
const parsed = (() => {
if (typeof fieldRule !== 'string') return fieldRule
try {
return JSON.parse(fieldRule)
} catch {
return undefined
}
})() as any
if (Array.isArray(parsed)) {
return parsed
.map((item: any) => {
const pointId = resolveRuleAttributeIdByCode(item?.code) ?? item?.pointId
return {
ruleAttributeId: pointId != null ? Number(pointId) : undefined,
ruleOperator: item?.operator,
ruleValue: item?.value ?? item?.operatorRule,
bigOperator: item?.bigOperator,
} as CountArithmeticRule
})
.filter((item) => item.ruleAttributeId || item.ruleOperator || item.ruleValue != null || item.bigOperator)
}
const root = parsed?.expr ?? parsed
if (!root || typeof root !== 'object') return []
const rows: CountArithmeticRule[] = []
const walk = (node: any) => {
if (!node || typeof node !== 'object' || !node.op) return
if (node.left && typeof node.left === 'object' && node.left.op) {
walk(node.left)
}
const attrId = resolveRuleAttributeIdByCode(node?.left?.field)
const rightValue =
node?.right?.field != null
? String(node.right.field)
: node?.right?.value != null
? node.right.value
: undefined
rows.push({
ruleAttributeId: attrId,
ruleOperator: node.op,
ruleValue: rightValue,
})
}
walk(root)
return rows.filter((item) => item.ruleOperator || item.ruleAttributeId || item.ruleValue != null)
}
const handleAddCountArithmeticRule = () => {
countArithmeticRules.value.push(createEmptyCountArithmeticRule())
}
const handleRemoveCountArithmeticRule = (index: number) => {
if (countArithmeticRules.value.length <= 1) return
if (index < 0 || index >= countArithmeticRules.value.length) return
countArithmeticRules.value.splice(index, 1)
const list = countArithmeticRules.value
if (list.length > 0) {
list[list.length - 1].bigOperator = undefined
}
}
const openRuleForm = async (row: DevicePointRuleVO & { pointRulesVOList?: any[] }) => { const openRuleForm = async (row: DevicePointRuleVO & { pointRulesVOList?: any[] }) => {
const deviceId = row.deviceId || attributeDeviceId.value const deviceId = row.deviceId || attributeDeviceId.value
@ -1155,12 +1382,27 @@ const openRuleForm = async (row: DevicePointRuleVO & { pointRulesVOList?: any[]
ruleForm.defaultValue = row.defaultValue ruleForm.defaultValue = row.defaultValue
ruleForm.deviceId = row.deviceId ruleForm.deviceId = row.deviceId
ruleForm.alarmLevel = (row as any).alarmLevel ruleForm.alarmLevel = (row as any).alarmLevel
countArithmeticRules.value = [createEmptyCountArithmeticRule()]
extraPointRules.value = [] extraPointRules.value = []
const list = Array.isArray(row.pointRulesVOList) ? row.pointRulesVOList : [] const list = Array.isArray(row.pointRulesVOList) ? row.pointRulesVOList : []
const parsedAlarmRunningList = parseAlarmRunningFieldRule(row.fieldRule)
if (list.length) { const identifier = (row.identifier || '').toString().toUpperCase()
if (identifier !== 'COUNT' && parsedAlarmRunningList.length) {
const first = parsedAlarmRunningList[0] as any
ruleForm.fieldRule = first?.rule ?? ''
ruleForm.ruleAttributeId = resolveRuleAttributeIdByCode(first?.code) as any
ruleForm.ruleOperator = first?.operator as any
ruleForm.ruleValue = first?.operatorRule as any
extraPointRules.value = parsedAlarmRunningList.slice(1).map((item: any) => ({
id: resolveRuleAttributeIdByCode(item?.code),
rule: item?.rule,
operator: item?.operator,
operatorRule: item?.operatorRule,
}))
} else if (list.length) {
const first = list[0] as any const first = list[0] as any
const firstRule = first.rule ?? row.fieldRule const firstRule = first.rule ?? row.fieldRule
const firstAttrId = (() => { const firstAttrId = (() => {
@ -1192,11 +1434,41 @@ const openRuleForm = async (row: DevicePointRuleVO & { pointRulesVOList?: any[]
operator: item.operator, operator: item.operator,
operatorRule: item.operatorRule, operatorRule: item.operatorRule,
})) }))
if (identifier === 'COUNT') {
const parsedFieldRule = parseCountFieldRule(row.fieldRule)
if (parsedFieldRule.length) {
countArithmeticRules.value = parsedFieldRule
} else {
const parsedList = list.map((item: any) => ({
ruleAttributeId: resolveRuleAttributeIdByCode(item?.code),
ruleOperator: item?.operator,
ruleValue: item?.operatorRule,
}))
const filteredParsedList = parsedList.filter((item) => {
return item.ruleAttributeId && item.ruleOperator && String(item.ruleValue ?? '').trim() !== ''
})
countArithmeticRules.value = filteredParsedList.length
? filteredParsedList
: [createEmptyCountArithmeticRule()]
}
}
} else { } else {
ruleForm.fieldRule = row.fieldRule ruleForm.fieldRule = row.fieldRule
ruleForm.ruleAttributeId = row.ruleAttributeId ruleForm.ruleAttributeId = row.ruleAttributeId
ruleForm.ruleOperator = row.ruleOperator ruleForm.ruleOperator = row.ruleOperator
ruleForm.ruleValue = row.ruleValue as any ruleForm.ruleValue = row.ruleValue as any
if (identifier === 'COUNT') {
const parsedFieldRule = parseCountFieldRule(row.fieldRule)
countArithmeticRules.value = parsedFieldRule.length
? parsedFieldRule
: [
{
ruleAttributeId: row.ruleAttributeId,
ruleOperator: row.ruleOperator,
ruleValue: row.ruleValue as any,
},
]
}
} }
const options = currentRuleOptions.value const options = currentRuleOptions.value
@ -1210,42 +1482,67 @@ const handleRuleSubmit = async () => {
if (!ruleForm.id) return if (!ruleForm.id) return
try { try {
ruleFormLoading.value = true ruleFormLoading.value = true
const pointRulesVOList: any[] = [] if (isCountIdentifier.value) {
const hasIncompleteRule = countArithmeticRules.value.some((item) => {
if (ruleForm.ruleAttributeId && ruleForm.ruleOperator) { return !item.ruleAttributeId || !item.ruleOperator || String(item.ruleValue ?? '').trim() === ''
const attr = ruleAttributeOptions.value.find( })
(item) => item.id === ruleForm.ruleAttributeId if (hasIncompleteRule) {
) message.warning('请完整填写每一组四则运算规则')
if (attr && attr.attributeCode) { return
pointRulesVOList.push({
code: attr.attributeCode,
rule: ruleForm.fieldRule,
operator: ruleForm.ruleOperator,
operatorRule: ruleForm.ruleValue,
})
} }
} const normalizedRules = countArithmeticRules.value.filter((item) => {
return item.ruleAttributeId && item.ruleOperator && String(item.ruleValue ?? '').trim() !== ''
extraPointRules.value.forEach((item) => {
if (!item.id || !item.operator) return
const attr = ruleAttributeOptions.value.find((opt) => opt.id === item.id)
if (!attr || !attr.attributeCode) return
pointRulesVOList.push({
code: attr.attributeCode,
rule: item.rule ?? ruleForm.fieldRule,
operator: item.operator,
operatorRule: item.operatorRule,
}) })
}) if (!normalizedRules.length) {
message.warning('请至少填写一组四则运算规则')
return
}
for (let i = 0; i < normalizedRules.length - 1; i++) {
if (!normalizedRules[i].bigOperator) {
message.warning(`请为第${i + 1}组选择连接运算符`)
return
}
}
for (let i = 0; i < normalizedRules.length; i++) {
const item = normalizedRules[i]
if (item.ruleOperator !== '/') continue
const valueText = String(item.ruleValue ?? '').trim()
if (valueText !== '' && !Number.isNaN(Number(valueText)) && Number(valueText) === 0) {
message.warning(`${i + 1}组为除法时数值不能为0`)
return
}
}
const fieldRule = buildCountFieldRule(normalizedRules)
if (!fieldRule) {
message.warning('四则运算规则格式不正确')
return
}
const payload = {
id: ruleForm.id,
identifier: ruleForm.identifier,
fieldName: ruleForm.fieldName,
fieldRule,
defaultValue: ruleForm.defaultValue,
deviceId: ruleForm.deviceId ?? attributeDeviceId.value,
}
await request.put({ url: '/iot/device-point-rules/update', data: payload })
message.success(t('common.updateSuccess'))
ruleDialogVisible.value = false
await getRuleList()
return
}
const alarmRunningFieldRule = buildAlarmRunningFieldRule()
if (!alarmRunningFieldRule || alarmRunningFieldRule === '[]') {
message.warning('请至少填写一组规则')
return
}
const payload = { const payload = {
id: ruleForm.id, id: ruleForm.id,
identifier: ruleForm.identifier, identifier: ruleForm.identifier,
fieldName: ruleForm.fieldName, fieldName: ruleForm.fieldName,
fieldRule: ruleForm.fieldRule, fieldRule: alarmRunningFieldRule,
defaultValue: ruleForm.defaultValue, defaultValue: ruleForm.defaultValue,
deviceId: ruleForm.deviceId ?? attributeDeviceId.value, deviceId: ruleForm.deviceId ?? attributeDeviceId.value,
pointRulesVOList,
} as any } as any
if (!isCountIdentifier.value) { if (!isCountIdentifier.value) {
payload.alarmLevel = ruleForm.alarmLevel payload.alarmLevel = ruleForm.alarmLevel

Loading…
Cancel
Save