perf:【IoT 物联网】场景联动触发器样式优化

pull/805/head
puhui999 2025-08-02 13:18:12 +08:00
parent fc09addd7d
commit 26a4f0fee1
4 changed files with 101 additions and 90 deletions

View File

@ -7,6 +7,7 @@
v-model="trigger" v-model="trigger"
:trigger-type="trigger.type" :trigger-type="trigger.type"
@validate="handleMainConditionValidate" @validate="handleMainConditionValidate"
@trigger-type-change="handleTriggerTypeChange"
/> />
</div> </div>
@ -42,6 +43,7 @@ const props = defineProps<{
const emit = defineEmits<{ const emit = defineEmits<{
(e: 'update:modelValue', value: TriggerFormData): void (e: 'update:modelValue', value: TriggerFormData): void
(e: 'validate', value: { valid: boolean; message: string }): void (e: 'validate', value: { valid: boolean; message: string }): void
(e: 'trigger-type-change', type: number): void
}>() }>()
const trigger = useVModel(props, 'modelValue', emit) const trigger = useVModel(props, 'modelValue', emit)
@ -83,6 +85,11 @@ const handleMainConditionValidate = (result: { valid: boolean; message: string }
updateValidationResult() updateValidationResult()
} }
const handleTriggerTypeChange = (type: number) => {
trigger.value.type = type
emit('trigger-type-change', type)
}
// //
const handleConditionGroupValidate = () => { const handleConditionGroupValidate = () => {
updateValidationResult() updateValidationResult()

View File

@ -1,38 +1,33 @@
<!-- 主条件配置组件 --> <!-- 主条件配置组件 -->
<template> <template>
<div class="flex flex-col gap-16px"> <div class="flex flex-col gap-16px">
<!-- 条件配置提示 -->
<div
v-if="!modelValue"
class="p-16px border-2 border-dashed border-[var(--el-border-color)] rounded-8px text-center"
>
<div class="flex flex-col items-center gap-12px">
<Icon icon="ep:plus" class="text-32px text-[var(--el-text-color-placeholder)]" />
<div class="text-[var(--el-text-color-secondary)]">
<p class="text-14px font-500 mb-4px">请配置主条件</p>
<p class="text-12px">主条件是触发器的核心条件必须满足才能触发场景</p>
</div>
<el-button type="primary" @click="addMainCondition">
<Icon icon="ep:plus" />
添加主条件
</el-button>
</div>
</div>
<!-- 主条件配置 --> <!-- 主条件配置 -->
<!-- TODO @puhui999主条件是不是和附加条件组弄成一个风格都是占一行有个绿条 --> <!-- TODO @puhui999主条件是不是和附加条件组弄成一个风格都是占一行有个绿条 -->
<div v-else class="space-y-16px"> <div class="space-y-16px">
<div class="flex items-center justify-between"> <!-- 主条件头部 - 与附加条件组保持一致的绿色风格 -->
<div class="flex items-center gap-8px"> <div
<span class="text-14px font-500 text-[var(--el-text-color-primary)]">主条件</span> class="flex items-center justify-between p-16px bg-gradient-to-r from-green-50 to-emerald-50 border border-green-200 rounded-8px"
<el-tag size="small" type="primary">必须满足</el-tag> >
<div class="flex items-center gap-12px">
<div class="flex items-center gap-8px text-16px font-600 text-green-700">
<div
class="w-24px h-24px bg-green-500 text-white rounded-full flex items-center justify-center text-12px font-bold"
>
</div>
<span>主条件</span>
</div>
<el-tag size="small" type="success">必须满足</el-tag>
</div> </div>
</div> </div>
<!-- 主条件内容配置 -->
<MainConditionInnerConfig <MainConditionInnerConfig
:model-value="modelValue" :model-value="modelValue"
@update:model-value="updateCondition" @update:model-value="updateCondition"
:trigger-type="triggerType" :trigger-type="triggerType"
@validate="handleValidate" @validate="handleValidate"
@trigger-type-change="handleTriggerTypeChange"
/> />
</div> </div>
</div> </div>
@ -45,29 +40,18 @@ import { IotRuleSceneTriggerConditionTypeEnum } from '@/views/iot/utils/constant
/** 主条件配置组件 */ /** 主条件配置组件 */
defineOptions({ name: 'MainConditionConfig' }) defineOptions({ name: 'MainConditionConfig' })
defineProps<{ const props = defineProps<{
modelValue?: TriggerFormData modelValue: TriggerFormData
triggerType: number triggerType: number
}>() }>()
const emit = defineEmits<{ const emit = defineEmits<{
(e: 'update:modelValue', value?: TriggerFormData): void (e: 'update:modelValue', value: TriggerFormData): void
(e: 'validate', result: { valid: boolean; message: string }): void (e: 'validate', result: { valid: boolean; message: string }): void
(e: 'trigger-type-change', type: number): void
}>() }>()
// //
const addMainCondition = () => {
const newCondition: TriggerFormData = {
type: IotRuleSceneTriggerConditionTypeEnum.DEVICE_PROPERTY, //
productId: undefined,
deviceId: undefined,
identifier: '',
operator: '=',
value: ''
}
emit('update:modelValue', newCondition)
}
const updateCondition = (condition: TriggerFormData) => { const updateCondition = (condition: TriggerFormData) => {
emit('update:modelValue', condition) emit('update:modelValue', condition)
} }
@ -76,7 +60,7 @@ const handleValidate = (result: { valid: boolean; message: string }) => {
emit('validate', result) emit('validate', result)
} }
onMounted(() => { const handleTriggerTypeChange = (type: number) => {
addMainCondition() emit('trigger-type-change', type)
}) }
</script> </script>

View File

@ -1,5 +1,23 @@
<template> <template>
<div class="space-y-16px"> <div class="space-y-16px">
<!-- 触发事件类型选择 -->
<el-form-item label="触发事件类型" required>
<el-select
:model-value="triggerType"
@update:model-value="handleTriggerTypeChange"
placeholder="请选择触发事件类型"
class="w-full"
style="width: 100%"
>
<el-option
v-for="option in triggerTypeOptions"
:key="option.value"
:label="option.label"
:value="option.value"
/>
</el-select>
</el-form-item>
<!-- 设备属性条件配置 --> <!-- 设备属性条件配置 -->
<div v-if="isDevicePropertyTrigger" class="space-y-16px"> <div v-if="isDevicePropertyTrigger" class="space-y-16px">
<!-- 产品设备选择 --> <!-- 产品设备选择 -->
@ -57,8 +75,8 @@
<el-col :span="12"> <el-col :span="12">
<el-form-item label="比较值" required> <el-form-item label="比较值" required>
<ValueInput <ValueInput
:model-value="condition.param" :model-value="condition.value"
@update:model-value="(value) => updateConditionField('param', value)" @update:model-value="(value) => updateConditionField('value', value)"
:property-type="propertyType" :property-type="propertyType"
:operator="condition.operator" :operator="condition.operator"
:property-config="propertyConfig" :property-config="propertyConfig"
@ -97,21 +115,22 @@ import PropertySelector from '../selectors/PropertySelector.vue'
import OperatorSelector from '../selectors/OperatorSelector.vue' import OperatorSelector from '../selectors/OperatorSelector.vue'
import ValueInput from '../inputs/ValueInput.vue' import ValueInput from '../inputs/ValueInput.vue'
import DeviceStatusConditionConfig from './DeviceStatusConditionConfig.vue' import DeviceStatusConditionConfig from './DeviceStatusConditionConfig.vue'
import { ConditionFormData } from '@/api/iot/rule/scene/scene.types' import { TriggerFormData } from '@/api/iot/rule/scene/scene.types'
import { IotRuleSceneTriggerTypeEnum } from '@/views/iot/utils/constants' import { IotRuleSceneTriggerTypeEnum, getTriggerTypeOptions } from '@/views/iot/utils/constants'
import { useVModel } from '@vueuse/core' import { useVModel } from '@vueuse/core'
/** 主条件内部配置组件 */ /** 主条件内部配置组件 */
defineOptions({ name: 'MainConditionInnerConfig' }) defineOptions({ name: 'MainConditionInnerConfig' })
const props = defineProps<{ const props = defineProps<{
modelValue: ConditionFormData modelValue: TriggerFormData
triggerType: number triggerType: number
}>() }>()
const emit = defineEmits<{ const emit = defineEmits<{
(e: 'update:modelValue', value: ConditionFormData): void (e: 'update:modelValue', value: TriggerFormData): void
(e: 'validate', result: { valid: boolean; message: string }): void (e: 'validate', result: { valid: boolean; message: string }): void
(e: 'trigger-type-change', value: number): void
}>() }>()
// //
@ -152,17 +171,24 @@ const getTriggerTypeText = (type: number) => {
} }
} }
//
const triggerTypeOptions = getTriggerTypeOptions()
// //
const updateConditionField = (field: keyof ConditionFormData, value: any) => { const updateConditionField = (field: keyof TriggerFormData, value: any) => {
condition.value[field] = value condition.value[field] = value
updateValidationResult() updateValidationResult()
} }
const updateCondition = (value: ConditionFormData) => { const updateCondition = (value: TriggerFormData) => {
emit('update:modelValue', value) emit('update:modelValue', value)
updateValidationResult() updateValidationResult()
} }
const handleTriggerTypeChange = (type: number) => {
emit('trigger-type-change', type)
}
const handleProductChange = () => { const handleProductChange = () => {
// //
condition.value.deviceId = undefined condition.value.deviceId = undefined
@ -231,7 +257,7 @@ const updateValidationResult = () => {
return return
} }
if (!condition.value.param) { if (!condition.value.value) {
isValid.value = false isValid.value = false
validationMessage.value = '请输入比较值' validationMessage.value = '请输入比较值'
emit('validate', { valid: false, message: validationMessage.value }) emit('validate', { valid: false, message: validationMessage.value })
@ -251,7 +277,7 @@ watch(
condition.value.deviceId, condition.value.deviceId,
condition.value.identifier, condition.value.identifier,
condition.value.operator, condition.value.operator,
condition.value.param condition.value.value
], ],
() => { () => {
updateValidationResult() updateValidationResult()

View File

@ -16,20 +16,26 @@
<div class="p-16px space-y-24px"> <div class="p-16px space-y-24px">
<!-- 触发器列表 --> <!-- 触发器列表 -->
<!-- TODO 每个触发器有个外框会不会好点 -->
<div v-if="triggers.length > 0" class="space-y-24px"> <div v-if="triggers.length > 0" class="space-y-24px">
<div <div
v-for="(triggerItem, index) in triggers" v-for="(triggerItem, index) in triggers"
:key="`trigger-${index}`" :key="`trigger-${index}`"
class="border border-[var(--el-border-color-light)] rounded-8px p-16px relative" class="border-2 border-green-200 rounded-8px bg-green-50 shadow-sm hover:shadow-md transition-shadow"
> >
<!-- 触发器头部 --> <!-- 触发器头部 - 绿色主题 -->
<div class="flex items-center justify-between mb-16px"> <div
<div class="flex items-center gap-8px"> class="flex items-center justify-between p-16px bg-gradient-to-r from-green-50 to-emerald-50 border-b border-green-200 rounded-t-6px"
<span class="text-14px font-500 text-[var(--el-text-color-primary)]"> >
触发器 {{ index + 1 }} <div class="flex items-center gap-12px">
</span> <div class="flex items-center gap-8px text-16px font-600 text-green-700">
<el-tag size="small" :type="getTriggerTagType(triggerItem.type)"> <div
class="w-24px h-24px bg-green-500 text-white rounded-full flex items-center justify-center text-12px font-bold"
>
{{ index + 1 }}
</div>
<span>触发器 {{ index + 1 }}</span>
</div>
<el-tag size="small" :type="getTriggerTagType(triggerItem.type)" class="font-500">
{{ getTriggerTypeLabel(triggerItem.type) }} {{ getTriggerTypeLabel(triggerItem.type) }}
</el-tag> </el-tag>
</div> </div>
@ -40,6 +46,7 @@
size="small" size="small"
text text
@click="removeTrigger(index)" @click="removeTrigger(index)"
class="hover:bg-red-50"
> >
<Icon icon="ep:delete" /> <Icon icon="ep:delete" />
删除 删除
@ -47,37 +54,24 @@
</div> </div>
</div> </div>
<!-- 触发事件类型选择 --> <!-- 触发器内容区域 -->
<el-form-item label="触发事件类型" required> <div class="p-16px space-y-16px">
<el-select <!-- 设备触发配置 -->
:model-value="triggerItem.type" <DeviceTriggerConfig
@update:model-value="(value) => updateTriggerType(index, value)" v-if="isDeviceTrigger(triggerItem.type)"
placeholder="请选择触发事件类型" :model-value="triggerItem"
class="w-full" :index="index"
> @update:model-value="(value) => updateTriggerDeviceConfig(index, value)"
<el-option @trigger-type-change="(type) => updateTriggerType(index, type)"
v-for="option in triggerTypeOptions" />
:key="option.value"
:label="option.label"
:value="option.value"
/>
</el-select>
</el-form-item>
<!-- 设备触发配置 --> <!-- 定时触发配置 -->
<DeviceTriggerConfig <TimerTriggerConfig
v-if="isDeviceTrigger(triggerItem.type)" v-else-if="triggerItem.type === TriggerTypeEnum.TIMER"
:model-value="triggerItem" :model-value="triggerItem.cronExpression"
:index="index" @update:model-value="(value) => updateTriggerCronConfig(index, value)"
@update:model-value="(value) => updateTriggerDeviceConfig(index, value)" />
/> </div>
<!-- 定时触发配置 -->
<TimerTriggerConfig
v-else-if="triggerItem.type === TriggerTypeEnum.TIMER"
:model-value="triggerItem.cronExpression"
@update:model-value="(value) => updateTriggerCronConfig(index, value)"
/>
</div> </div>
</div> </div>