feat: 【IoT 物联网】场景联动新增 JsonParamsInput 支持事件上报(填写事件的输出参数) 服务调用(填写服务的输入参数)属性设置(填写产品物模型可写属性)

This commit is contained in:
puhui999
2025-08-05 17:33:20 +08:00
parent ab54879203
commit 9917683f0a
6 changed files with 603 additions and 1223 deletions

View File

@ -1,5 +1,4 @@
<!-- 设备控制配置组件 -->
<!-- TODO @puhui999貌似没生效~~~ -->
<template>
<div class="flex flex-col gap-16px">
<!-- 产品和设备选择 - 与触发器保持一致的分离式选择器 -->
@ -51,64 +50,13 @@
<!-- 服务参数配置 -->
<div v-if="action.identifier" class="space-y-16px">
<el-form-item label="服务参数" required>
<div class="w-full space-y-8px">
<!-- JSON 输入框 -->
<div class="relative">
<el-input
v-model="paramsJson"
type="textarea"
:rows="6"
placeholder="请输入JSON格式的服务参数"
@input="handleParamsChange"
:class="{ 'is-error': jsonError }"
/>
<!-- 查看详细示例按钮 -->
<div class="absolute top-8px right-8px">
<el-button
ref="exampleTriggerRef"
type="info"
:icon="InfoFilled"
circle
size="small"
@click="toggleExampleDetail"
title="查看详细示例"
/>
</div>
</div>
<!-- 验证状态和错误提示 -->
<div class="flex items-center justify-between">
<div class="flex items-center gap-8px">
<Icon
:icon="jsonError ? 'ep:warning' : 'ep:circle-check'"
:class="
jsonError ? 'text-[var(--el-color-danger)]' : 'text-[var(--el-color-success)]'
"
class="text-14px"
/>
<span
:class="
jsonError ? 'text-[var(--el-color-danger)]' : 'text-[var(--el-color-success)]'
"
class="text-12px"
>
{{ jsonError || 'JSON格式正确' }}
</span>
</div>
<!-- 快速填充按钮 -->
<div
v-if="selectedService?.inputParams?.length > 0"
class="flex items-center gap-8px"
>
<span class="text-12px text-[var(--el-text-color-secondary)]">快速填充</span>
<el-button size="small" type="primary" plain @click="fillServiceExampleJson">
示例数据
</el-button>
<el-button size="small" type="default" plain @click="clearParams"> 清空 </el-button>
</div>
</div>
</div>
<JsonParamsInput
v-model="paramsValue"
type="service"
:config="{ service: selectedService }"
placeholder="请输入JSON格式的服务参数"
@validate="handleParamsValidate"
/>
</el-form-item>
</div>
</div>
@ -117,209 +65,23 @@
<div v-if="action.productId && isPropertySetAction" class="space-y-16px">
<!-- 参数配置 -->
<el-form-item label="参数" required>
<div class="w-full space-y-8px">
<!-- JSON 输入框 -->
<div class="relative">
<el-input
v-model="paramsJson"
type="textarea"
:rows="6"
placeholder="请输入JSON格式的控制参数"
@input="handleParamsChange"
:class="{ 'is-error': jsonError }"
/>
<!-- 查看详细示例按钮 -->
<div class="absolute top-8px right-8px">
<el-button
ref="exampleTriggerRef"
type="info"
:icon="InfoFilled"
circle
size="small"
@click="toggleExampleDetail"
title="查看详细示例"
/>
</div>
</div>
<!-- 验证状态和错误提示 -->
<div class="flex items-center justify-between">
<div class="flex items-center gap-8px">
<Icon
:icon="jsonError ? 'ep:warning' : 'ep:circle-check'"
:class="
jsonError ? 'text-[var(--el-color-danger)]' : 'text-[var(--el-color-success)]'
"
class="text-14px"
/>
<span
:class="
jsonError ? 'text-[var(--el-color-danger)]' : 'text-[var(--el-color-success)]'
"
class="text-12px"
>
{{ jsonError || 'JSON格式正确' }}
</span>
</div>
<!-- 快速填充按钮 -->
<div v-if="thingModelProperties.length > 0" class="flex items-center gap-8px">
<span class="text-12px text-[var(--el-text-color-secondary)]">快速填充</span>
<el-button size="small" type="primary" plain @click="fillExampleJson">
示例数据
</el-button>
<el-button size="small" type="default" plain @click="clearParams"> 清空 </el-button>
</div>
</div>
</div>
<JsonParamsInput
v-model="paramsValue"
type="property"
:config="{ properties: thingModelProperties }"
placeholder="请输入JSON格式的控制参数"
@validate="handleParamsValidate"
/>
</el-form-item>
<!-- 详细示例弹出层 -->
<Teleport to="body">
<div
v-if="showExampleDetail"
ref="exampleDetailRef"
class="example-detail-popover"
:style="examplePopoverStyle"
>
<div
class="p-16px bg-white rounded-8px shadow-lg border border-[var(--el-border-color)] min-w-400px max-w-500px"
>
<div class="flex items-center gap-8px mb-16px">
<Icon icon="ep:document" class="text-[var(--el-color-info)] text-18px" />
<span class="text-16px font-600 text-[var(--el-text-color-primary)]">
参数配置详细示例
</span>
</div>
<div class="space-y-16px">
<!-- 服务参数示例 - 服务调用时显示 -->
<div v-if="isServiceInvokeAction && selectedService?.inputParams?.length > 0">
<div class="flex items-center gap-8px mb-8px">
<Icon icon="ep:service" class="text-[var(--el-color-success)] text-14px" />
<span class="text-14px font-500 text-[var(--el-text-color-primary)]">
当前服务输入参数
</span>
</div>
<div class="ml-22px space-y-8px">
<div
v-for="param in selectedService.inputParams.slice(0, 4)"
:key="param.identifier"
class="flex items-center justify-between p-8px bg-[var(--el-fill-color-lighter)] rounded-4px"
>
<div class="flex-1">
<div class="text-12px font-500 text-[var(--el-text-color-primary)]">
{{ param.name }}
</div>
<div class="text-11px text-[var(--el-text-color-secondary)]">
{{ param.identifier }}
</div>
</div>
<div class="flex items-center gap-8px">
<el-tag :type="getPropertyTypeTag(param.dataType)" size="small">
{{ getPropertyTypeName(param.dataType) }}
</el-tag>
<span class="text-11px text-[var(--el-text-color-secondary)]">
{{ getExampleValueForParam(param) }}
</span>
</div>
</div>
</div>
<div class="mt-12px ml-22px">
<div class="text-12px text-[var(--el-text-color-secondary)] mb-6px">
完整JSON格式
</div>
<pre
class="p-12px bg-[var(--el-fill-color-light)] rounded-4px text-11px text-[var(--el-text-color-primary)] overflow-x-auto border-l-3px border-[var(--el-color-success)]"
><code>{{ generateServiceExampleJson() }}</code></pre>
</div>
</div>
<!-- 物模型属性示例 - 属性设置时显示 -->
<div v-if="isPropertySetAction && thingModelProperties.length > 0">
<div class="flex items-center gap-8px mb-8px">
<Icon icon="ep:edit" class="text-[var(--el-color-primary)] text-14px" />
<span class="text-14px font-500 text-[var(--el-text-color-primary)]">
当前物模型属性
</span>
</div>
<div class="ml-22px space-y-8px">
<div
v-for="property in thingModelProperties.slice(0, 4)"
:key="property.identifier"
class="flex items-center justify-between p-8px bg-[var(--el-fill-color-lighter)] rounded-4px"
>
<div class="flex-1">
<div class="text-12px font-500 text-[var(--el-text-color-primary)]">
{{ property.name }}
</div>
<div class="text-11px text-[var(--el-text-color-secondary)]">
{{ property.identifier }}
</div>
</div>
<div class="flex items-center gap-8px">
<el-tag :type="getPropertyTypeTag(property.dataType)" size="small">
{{ getPropertyTypeName(property.dataType) }}
</el-tag>
<span class="text-11px text-[var(--el-text-color-secondary)]">
{{ getExampleValue(property) }}
</span>
</div>
</div>
</div>
<div class="mt-12px ml-22px">
<div class="text-12px text-[var(--el-text-color-secondary)] mb-6px">
完整JSON格式
</div>
<pre
class="p-12px bg-[var(--el-fill-color-light)] rounded-4px text-11px text-[var(--el-text-color-primary)] overflow-x-auto border-l-3px border-[var(--el-color-primary)]"
><code>{{ generateExampleJson() }}</code></pre>
</div>
</div>
<!-- 通用示例 -->
<div>
<div class="flex items-center gap-8px mb-8px">
<Icon icon="ep:service" class="text-[var(--el-color-success)] text-14px" />
<span class="text-14px font-500 text-[var(--el-text-color-primary)]">
通用格式示例
</span>
</div>
<div class="ml-22px space-y-8px">
<div class="text-12px text-[var(--el-text-color-secondary)]">
服务调用格式
</div>
<pre
class="p-12px bg-[var(--el-fill-color-light)] rounded-4px text-11px text-[var(--el-text-color-primary)] overflow-x-auto border-l-3px border-[var(--el-color-success)]"
><code>{
"method": "restart",
"params": {
"delay": 5,
"force": false
}
}</code></pre>
</div>
</div>
</div>
<!-- 关闭按钮 -->
<div class="flex justify-end mt-16px">
<el-button size="small" @click="hideExampleDetail">关闭</el-button>
</div>
</div>
</div>
</Teleport>
</div>
</div>
</template>
<script setup lang="ts">
import { useVModel } from '@vueuse/core'
import { InfoFilled } from '@element-plus/icons-vue'
import ProductSelector from '../selectors/ProductSelector.vue'
import DeviceSelector from '../selectors/DeviceSelector.vue'
import JsonParamsInput from '../inputs/JsonParamsInput.vue'
import { Action, ThingModelService } from '@/api/iot/rule/scene/scene.types'
import { IotRuleSceneActionTypeEnum } from '@/views/iot/utils/constants'
@ -336,20 +98,39 @@ const emit = defineEmits<{
const action = useVModel(props, 'modelValue', emit)
const paramsJson = ref('') // 参数JSON字符串
const jsonError = ref('') // JSON格式错误信息
// 简化后的状态变量
const thingModelProperties = ref<any[]>([]) // 物模型属性列表
const loadingThingModel = ref(false) // 物模型加载状态
const propertyValues = ref<Record<string, any>>({}) // 属性值映射
const selectedService = ref<ThingModelService | null>(null) // 选中的服务对象
const serviceList = ref<ThingModelService[]>([]) // 服务列表
const loadingServices = ref(false) // 服务加载状态
const showExampleDetail = ref(false) // 示例详情弹出层显示状态
const exampleTriggerRef = ref() // 示例触发按钮引用
const exampleDetailRef = ref() // 示例详情弹出层引用
const examplePopoverStyle = ref({}) // 示例弹出层样式
// 参数值的计算属性,用于双向绑定
const paramsValue = computed({
get: () => {
if (action.value.params && typeof action.value.params === 'object') {
return JSON.stringify(action.value.params, null, 2)
}
return ''
},
set: (value: string) => {
try {
if (value.trim()) {
action.value.params = JSON.parse(value)
} else {
action.value.params = {}
}
} catch (error) {
console.error('JSON解析错误:', error)
}
}
})
// 参数验证处理
const handleParamsValidate = (result: { valid: boolean; message: string }) => {
// 可以在这里处理验证结果,比如显示错误信息
console.log('参数验证结果:', result)
}
const isPropertySetAction = computed(() => {
// 是否为属性设置类型
@ -371,9 +152,6 @@ const handleProductChange = (productId?: number) => {
action.value.deviceId = undefined
action.value.identifier = undefined // 清空服务标识符
action.value.params = {}
paramsJson.value = ''
jsonError.value = ''
propertyValues.value = {}
selectedService.value = null // 清空选中的服务
serviceList.value = [] // 清空服务列表
}
@ -396,8 +174,6 @@ const handleDeviceChange = (deviceId?: number) => {
// 当设备变化时,清空参数配置
if (action.value.deviceId !== deviceId) {
action.value.params = {}
paramsJson.value = ''
jsonError.value = ''
}
}
@ -410,10 +186,8 @@ const handleServiceChange = (serviceIdentifier?: string) => {
const service = serviceList.value.find((s) => s.identifier === serviceIdentifier) || null
selectedService.value = service
// 当服务变化时,清空参数配置并根据服务输入参数生成默认参数结构
// 当服务变化时,清空参数配置
action.value.params = {}
paramsJson.value = ''
jsonError.value = ''
// 如果选择了服务且有输入参数,生成默认参数结构
if (service && service.inputParams && service.inputParams.length > 0) {
@ -422,50 +196,9 @@ const handleServiceChange = (serviceIdentifier?: string) => {
defaultParams[param.identifier] = getDefaultValueForParam(param)
})
action.value.params = defaultParams
paramsJson.value = JSON.stringify(defaultParams, null, 2)
}
}
/**
* 快速填充示例数据
*/
const fillExampleJson = () => {
const exampleData = generateExampleJson()
paramsJson.value = exampleData
handleParamsChange()
}
/**
* 快速填充服务示例数据
*/
const fillServiceExampleJson = () => {
if (selectedService.value && selectedService.value.inputParams) {
const exampleData = generateServiceExampleJson()
paramsJson.value = exampleData
handleParamsChange()
}
}
/**
* 清空参数
*/
const clearParams = () => {
paramsJson.value = ''
action.value.params = {}
propertyValues.value = {}
jsonError.value = ''
}
// 更新属性值(保留但不在模板中使用)
// const updatePropertyValue = (identifier: string, value: any) => {
// propertyValues.value[identifier] = value
// // 同步更新到 action.params
// action.value.params = { ...propertyValues.value }
// // 同步更新 JSON 显示
// paramsJson.value = JSON.stringify(action.value.params, null, 2)
// jsonError.value = ''
// }
/**
* 加载物模型属性
* @param productId 产品ID
@ -508,12 +241,7 @@ const loadThingModelProperties = async (productId: number) => {
}
]
// 初始化属性值
thingModelProperties.value.forEach((property) => {
if (!(property.identifier in propertyValues.value)) {
propertyValues.value[property.identifier] = ''
}
})
// 属性加载完成,无需额外初始化
} catch (error) {
console.error('加载物模型失败:', error)
thingModelProperties.value = []
@ -561,55 +289,6 @@ const loadServiceFromTSL = async (productId: number, serviceIdentifier: string)
}
}
/**
* 处理参数变化事件
*/
const handleParamsChange = () => {
try {
jsonError.value = '' // 清除之前的错误
if (paramsJson.value.trim()) {
const parsed = JSON.parse(paramsJson.value)
action.value.params = parsed
// 同步更新到属性值
propertyValues.value = { ...parsed }
// 额外的参数验证
if (typeof parsed !== 'object' || parsed === null) {
jsonError.value = '参数必须是一个有效的JSON对象'
return
}
} else {
action.value.params = {}
propertyValues.value = {}
}
} catch (error) {
jsonError.value = `JSON格式错误: ${error instanceof Error ? error.message : '未知错误'}`
console.error('JSON格式错误:', error)
}
}
/**
* 获取属性类型名称
* @param dataType 数据类型
* @returns 类型名称
*/
const getPropertyTypeName = (dataType: string) => {
const typeMap = {
int: '整数',
float: '浮点数',
double: '双精度',
text: '字符串',
bool: '布尔值',
enum: '枚举',
date: '日期',
struct: '结构体',
array: '数组'
}
return typeMap[dataType] || dataType
}
/**
* 根据参数类型获取默认值
* @param param 参数对象
@ -637,243 +316,10 @@ const getDefaultValueForParam = (param: any) => {
}
}
/**
* 获取属性类型标签样式
* @param dataType 数据类型
* @returns 标签类型
*/
const getPropertyTypeTag = (dataType: string) => {
const tagMap = {
int: 'primary',
float: 'success',
double: 'success',
text: 'info',
bool: 'warning',
enum: 'danger',
date: 'primary',
struct: 'info',
array: 'warning'
}
return tagMap[dataType] || 'info'
}
/**
* 获取属性示例值
* @param property 属性对象
* @returns 示例值
*/
const getExampleValue = (property: any) => {
switch (property.dataType) {
case 'int':
return property.identifier === 'BatteryLevel' ? '85' : '25'
case 'float':
case 'double':
return property.identifier === 'Temperature' ? '25.5' : '60.0'
case 'bool':
return 'false'
case 'text':
return '"auto"'
case 'enum':
return '"option1"'
default:
return '""'
}
}
/**
* 获取参数示例值
* @param param 参数对象
* @returns 示例值
*/
const getExampleValueForParam = (param: any) => {
switch (param.dataType) {
case 'int':
return '0'
case 'float':
case 'double':
return '0.0'
case 'bool':
return 'false'
case 'text':
return '"text"'
case 'enum':
if (param.dataSpecs?.dataSpecsList && param.dataSpecs.dataSpecsList.length > 0) {
return `"${param.dataSpecs.dataSpecsList[0].name}"`
}
return '"option1"'
default:
return '""'
}
}
/**
* 生成示例JSON
* @returns JSON字符串
*/
const generateExampleJson = () => {
if (thingModelProperties.value.length === 0) {
return JSON.stringify(
{
BatteryLevel: '',
WaterLeachState: ''
},
null,
2
)
}
const example = {}
thingModelProperties.value.forEach((property) => {
switch (property.dataType) {
case 'int':
example[property.identifier] = property.identifier === 'BatteryLevel' ? 85 : 25
break
case 'float':
case 'double':
example[property.identifier] = property.identifier === 'Temperature' ? 25.5 : 60.0
break
case 'bool':
example[property.identifier] = false
break
case 'text':
example[property.identifier] = 'auto'
break
default:
example[property.identifier] = ''
}
})
return JSON.stringify(example, null, 2)
}
/**
* 生成服务示例JSON
* @returns JSON字符串
*/
const generateServiceExampleJson = () => {
if (!selectedService.value || !selectedService.value.inputParams) {
return JSON.stringify({}, null, 2)
}
const example = {}
selectedService.value.inputParams.forEach((param) => {
example[param.identifier] = getDefaultValueForParam(param)
})
return JSON.stringify(example, null, 2)
}
/**
* 切换示例详情弹出层显示状态
*/
const toggleExampleDetail = () => {
if (showExampleDetail.value) {
hideExampleDetail()
} else {
showExampleDetailPopover()
}
}
/**
* 显示示例详情弹出层
*/
const showExampleDetailPopover = () => {
if (!exampleTriggerRef.value) return
showExampleDetail.value = true
nextTick(() => {
updateExamplePopoverPosition()
})
}
/**
* 隐藏示例详情弹出层
*/
const hideExampleDetail = () => {
showExampleDetail.value = false
}
/**
* 更新示例弹出层位置
*/
const updateExamplePopoverPosition = () => {
if (!exampleTriggerRef.value || !exampleDetailRef.value) return
const triggerEl = exampleTriggerRef.value.$el
const triggerRect = triggerEl.getBoundingClientRect()
// 计算弹出层位置
const left = triggerRect.left + triggerRect.width + 8
const top = triggerRect.top
// 检查是否超出视窗右边界
const popoverWidth = 500 // 最大宽度
const viewportWidth = window.innerWidth
let finalLeft = left
if (left + popoverWidth > viewportWidth - 16) {
// 如果超出右边界,显示在左侧
finalLeft = triggerRect.left - popoverWidth - 8
}
// 检查是否超出视窗下边界
let finalTop = top
const popoverHeight = exampleDetailRef.value.offsetHeight || 300
const viewportHeight = window.innerHeight
if (top + popoverHeight > viewportHeight - 16) {
finalTop = Math.max(16, viewportHeight - popoverHeight - 16)
}
examplePopoverStyle.value = {
position: 'fixed',
left: `${finalLeft}px`,
top: `${finalTop}px`,
zIndex: 9999
}
}
/**
* 点击外部关闭弹出层
* @param event 鼠标事件
*/
const handleClickOutside = (event: MouseEvent) => {
if (
showExampleDetail.value &&
exampleDetailRef.value &&
exampleTriggerRef.value &&
!exampleDetailRef.value.contains(event.target as Node) &&
!exampleTriggerRef.value.$el.contains(event.target as Node)
) {
hideExampleDetail()
}
}
/**
* 监听窗口大小变化,重新计算弹出层位置
*/
const handleResize = () => {
if (showExampleDetail.value) {
updateExamplePopoverPosition()
}
}
/**
* 组件初始化
*/
onMounted(() => {
if (action.value.params && Object.keys(action.value.params).length > 0) {
try {
paramsJson.value = JSON.stringify(action.value.params, null, 2)
propertyValues.value = { ...action.value.params }
jsonError.value = '' // 清除错误状态
} catch (error) {
console.error('初始化参数格式化失败:', error)
jsonError.value = '初始参数格式错误'
}
}
// 如果已经选择了产品且是属性设置类型,加载物模型
if (action.value.productId && isPropertySetAction.value) {
loadThingModelProperties(action.value.productId)
@ -884,47 +330,8 @@ onMounted(() => {
// 加载物模型TSL以获取服务信息
loadServiceFromTSL(action.value.productId, action.value.identifier)
}
// 添加事件监听器
document.addEventListener('click', handleClickOutside)
window.addEventListener('resize', handleResize)
})
/**
* 组件卸载时清理事件监听器
*/
onUnmounted(() => {
document.removeEventListener('click', handleClickOutside)
window.removeEventListener('resize', handleResize)
})
// 监听参数变化
watch(
() => action.value.params,
(newParams) => {
if (newParams && typeof newParams === 'object' && Object.keys(newParams).length > 0) {
try {
const newJsonString = JSON.stringify(newParams, null, 2)
// 只有当JSON字符串真正改变时才更新避免循环更新
if (newJsonString !== paramsJson.value) {
paramsJson.value = newJsonString
jsonError.value = ''
}
} catch (error) {
console.error('参数格式化失败:', error)
jsonError.value = '参数格式化失败'
}
} else {
// 参数为空时清空JSON显示
if (paramsJson.value !== '') {
paramsJson.value = ''
jsonError.value = ''
}
}
},
{ deep: true }
)
// 监听action.value变化处理编辑模式的数据回显
watch(
() => action.value,
@ -944,79 +351,8 @@ watch(
selectedService.value = null
serviceList.value = []
}
// 处理参数回显
if (newAction.params && Object.keys(newAction.params).length > 0) {
try {
const newJsonString = JSON.stringify(newAction.params, null, 2)
if (paramsJson.value !== newJsonString) {
paramsJson.value = newJsonString
propertyValues.value = { ...newAction.params }
jsonError.value = ''
}
} catch (error) {
console.error('参数格式化失败:', error)
jsonError.value = '参数格式化失败'
}
} else {
if (paramsJson.value !== '') {
paramsJson.value = ''
propertyValues.value = {}
jsonError.value = ''
}
}
}
},
{ deep: true, immediate: true }
)
</script>
<style scoped>
/* 参考 PropertySelector 的弹出层样式 */
@keyframes fadeInScale {
from {
opacity: 0;
transform: scale(0.9) translateY(-4px);
}
to {
opacity: 1;
transform: scale(1) translateY(0);
}
}
.example-detail-popover {
animation: fadeInScale 0.2s ease-out;
transform-origin: top left;
}
/* 弹出层箭头效果 */
.example-detail-popover::before {
position: absolute;
top: 20px;
left: -8px;
width: 0;
height: 0;
border-top: 8px solid transparent;
border-right: 8px solid var(--el-border-color);
border-bottom: 8px solid transparent;
content: '';
}
.example-detail-popover::after {
position: absolute;
top: 20px;
left: -7px;
width: 0;
height: 0;
border-top: 8px solid transparent;
border-right: 8px solid white;
border-bottom: 8px solid transparent;
content: '';
}
:deep(.example-content code) {
font-family: 'Courier New', monospace;
color: var(--el-color-primary);
}
</style>