commit
5d2adcac19
|
@ -11,7 +11,7 @@ export interface ThingModelData {
|
|||
productId?: number // 产品编号
|
||||
productKey?: string // 产品标识
|
||||
dataType: string // 数据类型,与 dataSpecs 的 dataType 保持一致
|
||||
type: ProductFunctionTypeEnum // 功能类型
|
||||
type: number // 功能类型
|
||||
property: ThingModelProperty // 属性
|
||||
event?: ThingModelEvent // 事件
|
||||
service?: ThingModelService // 服务
|
||||
|
@ -38,19 +38,6 @@ export interface ThingModelService {
|
|||
[key: string]: any
|
||||
}
|
||||
|
||||
// IOT 产品功能(物模型)类型枚举类
|
||||
export enum ProductFunctionTypeEnum {
|
||||
PROPERTY = 1, // 属性
|
||||
SERVICE = 2, // 服务
|
||||
EVENT = 3 // 事件
|
||||
}
|
||||
|
||||
// IOT 产品功能(物模型)访问模式枚举类
|
||||
export enum ProductFunctionAccessModeEnum {
|
||||
READ_WRITE = 'rw', // 读写
|
||||
READ_ONLY = 'r' // 只读
|
||||
}
|
||||
|
||||
// IoT 产品物模型 API
|
||||
export const ThingModelApi = {
|
||||
// 查询产品物模型分页
|
||||
|
|
|
@ -238,7 +238,7 @@ export enum DICT_TYPE {
|
|||
IOT_DEVICE_STATUS = 'iot_device_status', // IOT 设备状态
|
||||
IOT_PRODUCT_THING_MODEL_TYPE = 'iot_product_thing_model_type', // IOT 产品功能类型
|
||||
IOT_DATA_TYPE = 'iot_data_type', // IOT 数据类型
|
||||
IOT_UNIT_TYPE = 'iot_unit_type', // IOT 单位类型
|
||||
IOT_PRODUCT_THING_MODEL_UNIT = 'iot_product_thing_model_unit', // IOT 物模型单位
|
||||
IOT_RW_TYPE = 'iot_rw_type', // IOT 读写类型
|
||||
IOT_PLUGIN_DEPLOY_TYPE = 'iot_plugin_deploy_type', // IOT 插件部署类型
|
||||
IOT_PLUGIN_STATUS = 'iot_plugin_status', // IOT 插件状态
|
||||
|
|
|
@ -0,0 +1,47 @@
|
|||
<template>
|
||||
<el-form-item
|
||||
:rules="[{ required: true, message: '请选择事件类型', trigger: 'change' }]"
|
||||
label="事件类型"
|
||||
prop="event.type"
|
||||
>
|
||||
<el-radio-group v-model="thingModelEvent.type">
|
||||
<el-radio :value="ThingModelEventType.INFO.value">
|
||||
{{ ThingModelEventType.INFO.label }}
|
||||
</el-radio>
|
||||
<el-radio :value="ThingModelEventType.ALERT.value">
|
||||
{{ ThingModelEventType.ALERT.label }}
|
||||
</el-radio>
|
||||
<el-radio :value="ThingModelEventType.ERROR.value">
|
||||
{{ ThingModelEventType.ERROR.label }}
|
||||
</el-radio>
|
||||
</el-radio-group>
|
||||
</el-form-item>
|
||||
<el-form-item label="输出参数">
|
||||
<ThingModelInputOutputParam
|
||||
v-model="thingModelEvent.outputParams"
|
||||
:direction="ThingModelParamDirection.OUTPUT"
|
||||
/>
|
||||
</el-form-item>
|
||||
</template>
|
||||
|
||||
<script lang="ts" setup>
|
||||
import ThingModelInputOutputParam from './ThingModelInputOutputParam.vue'
|
||||
import { useVModel } from '@vueuse/core'
|
||||
import { ThingModelEvent } from '@/api/iot/thingmodel'
|
||||
import { ThingModelParamDirection, ThingModelEventType } from './config'
|
||||
|
||||
/** IoT 物模型事件 */
|
||||
defineOptions({ name: 'ThingModelEvent' })
|
||||
|
||||
const props = defineProps<{ modelValue: any; isStructDataSpecs?: boolean }>()
|
||||
const emits = defineEmits(['update:modelValue'])
|
||||
const thingModelEvent = useVModel(props, 'modelValue', emits) as Ref<ThingModelEvent>
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
:deep(.el-form-item) {
|
||||
.el-form-item {
|
||||
margin-bottom: 0;
|
||||
}
|
||||
}
|
||||
</style>
|
|
@ -25,10 +25,26 @@
|
|||
<el-input v-model="formData.identifier" placeholder="请输入标识符" />
|
||||
</el-form-item>
|
||||
<!-- 属性配置 -->
|
||||
<ThingModelDataSpecs
|
||||
v-if="formData.type === ProductFunctionTypeEnum.PROPERTY"
|
||||
<ThingModelProperty
|
||||
v-if="formData.type === ThingModelType.PROPERTY"
|
||||
v-model="formData.property"
|
||||
/>
|
||||
<!-- 服务配置 -->
|
||||
<ThingModelService
|
||||
v-if="formData.type === ThingModelType.SERVICE"
|
||||
v-model="formData.service"
|
||||
/>
|
||||
<!-- 事件配置 -->
|
||||
<ThingModelEvent v-if="formData.type === ThingModelType.EVENT" v-model="formData.event" />
|
||||
<el-form-item label="描述" prop="description">
|
||||
<el-input
|
||||
v-model="formData.description"
|
||||
:maxlength="200"
|
||||
:rows="3"
|
||||
placeholder="请输入属性描述"
|
||||
type="textarea"
|
||||
/>
|
||||
</el-form-item>
|
||||
</el-form>
|
||||
|
||||
<template #footer>
|
||||
|
@ -40,12 +56,15 @@
|
|||
|
||||
<script lang="ts" setup>
|
||||
import { ProductVO } from '@/api/iot/product/product'
|
||||
import ThingModelDataSpecs from './ThingModelDataSpecs.vue'
|
||||
import { ProductFunctionTypeEnum, ThingModelApi, ThingModelData } from '@/api/iot/thingmodel'
|
||||
import ThingModelProperty from './ThingModelProperty.vue'
|
||||
import ThingModelService from './ThingModelService.vue'
|
||||
import ThingModelEvent from './ThingModelEvent.vue'
|
||||
import { ThingModelApi, ThingModelData } from '@/api/iot/thingmodel'
|
||||
import { IOT_PROVIDE_KEY } from '@/views/iot/utils/constants'
|
||||
import { DataSpecsDataType, ThingModelFormRules } from './config'
|
||||
import { DataSpecsDataType, ThingModelFormRules, ThingModelType } from './config'
|
||||
import { cloneDeep } from 'lodash-es'
|
||||
import { DICT_TYPE, getIntDictOptions } from '@/utils/dict'
|
||||
import { isEmpty } from '@/utils/is'
|
||||
|
||||
/** IoT 物模型数据表单 */
|
||||
defineOptions({ name: 'IoTProductThingModelForm' })
|
||||
|
@ -60,14 +79,16 @@ const dialogTitle = ref('') // 弹窗的标题
|
|||
const formLoading = ref(false) // 表单的加载中:1)修改时的数据加载;2)提交的按钮禁用
|
||||
const formType = ref('') // 表单的类型:create - 新增;update - 修改
|
||||
const formData = ref<ThingModelData>({
|
||||
type: ProductFunctionTypeEnum.PROPERTY,
|
||||
type: ThingModelType.PROPERTY,
|
||||
dataType: DataSpecsDataType.INT,
|
||||
property: {
|
||||
dataType: DataSpecsDataType.INT,
|
||||
dataSpecs: {
|
||||
dataType: DataSpecsDataType.INT
|
||||
}
|
||||
}
|
||||
},
|
||||
service: {},
|
||||
event: {}
|
||||
})
|
||||
|
||||
const formRef = ref() // 表单 Ref
|
||||
|
@ -92,6 +113,7 @@ defineExpose({ open, close: () => (dialogVisible.value = false) })
|
|||
/** 提交表单 */
|
||||
const emit = defineEmits(['success'])
|
||||
const submitForm = async () => {
|
||||
debugger
|
||||
await formRef.value.validate()
|
||||
formLoading.value = true
|
||||
try {
|
||||
|
@ -99,10 +121,7 @@ const submitForm = async () => {
|
|||
// 信息补全
|
||||
data.productId = product!.value.id
|
||||
data.productKey = product!.value.productKey
|
||||
data.description = data.property.description
|
||||
data.dataType = data.property.dataType
|
||||
data.property.identifier = data.identifier
|
||||
data.property.name = data.name
|
||||
fillExtraAttributes(data)
|
||||
if (formType.value === 'create') {
|
||||
await ThingModelApi.createThingModel(data)
|
||||
message.success(t('common.createSuccess'))
|
||||
|
@ -117,17 +136,60 @@ const submitForm = async () => {
|
|||
}
|
||||
}
|
||||
|
||||
/** 填写额外的属性 */
|
||||
const fillExtraAttributes = (data: any) => {
|
||||
// 处理不同类型的情况
|
||||
// 属性
|
||||
if (data.type === ThingModelType.PROPERTY) {
|
||||
removeDataSpecs(data.property)
|
||||
data.dataType = data.property.dataType
|
||||
data.property.identifier = data.identifier
|
||||
data.property.name = data.name
|
||||
delete data.service
|
||||
delete data.event
|
||||
}
|
||||
// 服务
|
||||
if (data.type === ThingModelType.SERVICE) {
|
||||
removeDataSpecs(data.service)
|
||||
data.dataType = data.service.dataType
|
||||
data.service.identifier = data.identifier
|
||||
data.service.name = data.name
|
||||
delete data.property
|
||||
delete data.event
|
||||
}
|
||||
// 事件
|
||||
if (data.type === ThingModelType.EVENT) {
|
||||
removeDataSpecs(data.event)
|
||||
data.dataType = data.event.dataType
|
||||
data.event.identifier = data.identifier
|
||||
data.event.name = data.name
|
||||
delete data.property
|
||||
delete data.service
|
||||
}
|
||||
}
|
||||
/** 处理 dataSpecs 为空的情况 */
|
||||
const removeDataSpecs = (val: any) => {
|
||||
if (isEmpty(val.dataSpecs)) {
|
||||
delete val.dataSpecs
|
||||
}
|
||||
if (isEmpty(val.dataSpecsList)) {
|
||||
delete val.dataSpecsList
|
||||
}
|
||||
}
|
||||
|
||||
/** 重置表单 */
|
||||
const resetForm = () => {
|
||||
formData.value = {
|
||||
type: ProductFunctionTypeEnum.PROPERTY,
|
||||
type: ThingModelType.PROPERTY,
|
||||
dataType: DataSpecsDataType.INT,
|
||||
property: {
|
||||
dataType: DataSpecsDataType.INT,
|
||||
dataSpecs: {
|
||||
dataType: DataSpecsDataType.INT
|
||||
}
|
||||
}
|
||||
},
|
||||
service: {},
|
||||
event: {}
|
||||
}
|
||||
formRef.value?.resetFields()
|
||||
}
|
||||
|
|
|
@ -0,0 +1,153 @@
|
|||
<template>
|
||||
<div
|
||||
v-for="(item, index) in thingModelParams"
|
||||
:key="index"
|
||||
class="w-1/1 param-item flex justify-between px-10px mb-10px"
|
||||
>
|
||||
<span>参数名称:{{ item.name }}</span>
|
||||
<div class="btn">
|
||||
<el-button link type="primary" @click="openParamForm(item)">编辑</el-button>
|
||||
<el-divider direction="vertical" />
|
||||
<el-button link type="danger" @click="deleteParamItem(index)">删除</el-button>
|
||||
</div>
|
||||
</div>
|
||||
<el-button link type="primary" @click="openParamForm(null)">+新增参数</el-button>
|
||||
|
||||
<!-- param 表单 -->
|
||||
<Dialog v-model="dialogVisible" :title="dialogTitle" append-to-body>
|
||||
<el-form
|
||||
ref="paramFormRef"
|
||||
v-loading="formLoading"
|
||||
:model="formData"
|
||||
:rules="ThingModelFormRules"
|
||||
label-width="100px"
|
||||
>
|
||||
<el-form-item label="参数名称" prop="name">
|
||||
<el-input v-model="formData.name" placeholder="请输入功能名称" />
|
||||
</el-form-item>
|
||||
<el-form-item label="标识符" prop="identifier">
|
||||
<el-input v-model="formData.identifier" placeholder="请输入标识符" />
|
||||
</el-form-item>
|
||||
<!-- 属性配置 -->
|
||||
<ThingModelProperty v-model="formData.property" is-params />
|
||||
</el-form>
|
||||
|
||||
<template #footer>
|
||||
<el-button :disabled="formLoading" type="primary" @click="submitForm">确 定</el-button>
|
||||
<el-button @click="dialogVisible = false">取 消</el-button>
|
||||
</template>
|
||||
</Dialog>
|
||||
</template>
|
||||
|
||||
<script lang="ts" setup>
|
||||
import { useVModel } from '@vueuse/core'
|
||||
import ThingModelProperty from './ThingModelProperty.vue'
|
||||
import { DataSpecsDataType, ThingModelFormRules } from './config'
|
||||
import { isEmpty } from '@/utils/is'
|
||||
|
||||
/** 输入输出参数配置组件 */
|
||||
defineOptions({ name: 'ThingModelInputOutputParam' })
|
||||
|
||||
const props = defineProps<{ modelValue: any; direction: string }>()
|
||||
const emits = defineEmits(['update:modelValue'])
|
||||
const thingModelParams = useVModel(props, 'modelValue', emits) as Ref<any[]>
|
||||
const dialogVisible = ref(false) // 弹窗的是否展示
|
||||
const dialogTitle = ref('新增参数') // 弹窗的标题
|
||||
const formLoading = ref(false) // 表单的加载中:1)修改时的数据加载;2)提交的按钮禁用
|
||||
const paramFormRef = ref() // 表单 ref
|
||||
const formData = ref<any>({
|
||||
dataType: DataSpecsDataType.INT,
|
||||
property: {
|
||||
dataType: DataSpecsDataType.INT,
|
||||
dataSpecs: {
|
||||
dataType: DataSpecsDataType.INT
|
||||
}
|
||||
}
|
||||
})
|
||||
|
||||
/** 打开 param 表单 */
|
||||
const openParamForm = (val: any) => {
|
||||
dialogVisible.value = true
|
||||
resetForm()
|
||||
if (isEmpty(val)) {
|
||||
return
|
||||
}
|
||||
// 编辑时回显数据
|
||||
formData.value = {
|
||||
identifier: val.identifier,
|
||||
name: val.name,
|
||||
description: val.description,
|
||||
property: {
|
||||
dataType: val.dataType,
|
||||
dataSpecs: val.dataSpecs,
|
||||
dataSpecsList: val.dataSpecsList
|
||||
}
|
||||
}
|
||||
}
|
||||
/** 删除 param 项 */
|
||||
const deleteParamItem = (index: number) => {
|
||||
thingModelParams.value.splice(index, 1)
|
||||
}
|
||||
|
||||
/** 添加参数 */
|
||||
const submitForm = async () => {
|
||||
// 初始化参数列表
|
||||
if (isEmpty(thingModelParams.value)) {
|
||||
thingModelParams.value = []
|
||||
}
|
||||
// 校验参数
|
||||
await paramFormRef.value.validate()
|
||||
try {
|
||||
const data = unref(formData)
|
||||
// 构建数据对象
|
||||
const item = {
|
||||
identifier: data.identifier,
|
||||
name: data.name,
|
||||
description: data.description,
|
||||
dataType: data.property.dataType,
|
||||
paraOrder: 0, // TODO @puhui999: 先写死默认看看后续
|
||||
direction: props.direction,
|
||||
dataSpecs:
|
||||
!!data.property.dataSpecs && Object.keys(data.property.dataSpecs).length > 1
|
||||
? data.property.dataSpecs
|
||||
: undefined,
|
||||
dataSpecsList: isEmpty(data.property.dataSpecsList) ? undefined : data.property.dataSpecsList
|
||||
}
|
||||
|
||||
// 查找是否已有相同 identifier 的项
|
||||
const existingIndex = thingModelParams.value.findIndex(
|
||||
(spec) => spec.identifier === data.identifier
|
||||
)
|
||||
if (existingIndex > -1) {
|
||||
// 更新已有项
|
||||
thingModelParams.value[existingIndex] = item
|
||||
} else {
|
||||
// 添加新项
|
||||
thingModelParams.value.push(item)
|
||||
}
|
||||
} finally {
|
||||
// 隐藏对话框
|
||||
dialogVisible.value = false
|
||||
}
|
||||
}
|
||||
|
||||
/** 重置表单 */
|
||||
const resetForm = () => {
|
||||
formData.value = {
|
||||
dataType: DataSpecsDataType.INT,
|
||||
property: {
|
||||
dataType: DataSpecsDataType.INT,
|
||||
dataSpecs: {
|
||||
dataType: DataSpecsDataType.INT
|
||||
}
|
||||
}
|
||||
}
|
||||
paramFormRef.value?.resetFields()
|
||||
}
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
.param-item {
|
||||
background-color: #e4f2fd;
|
||||
}
|
||||
</style>
|
|
@ -75,26 +75,26 @@
|
|||
v-if="property.dataType === DataSpecsDataType.STRUCT"
|
||||
v-model="property.dataSpecsList"
|
||||
/>
|
||||
<el-form-item v-if="!isStructDataSpecs" label="读写类型" prop="property.accessMode">
|
||||
<el-form-item v-if="!isStructDataSpecs && !isParams" label="读写类型" prop="property.accessMode">
|
||||
<el-radio-group v-model="property.accessMode">
|
||||
<el-radio label="rw">读写</el-radio>
|
||||
<el-radio label="r">只读</el-radio>
|
||||
<el-radio :label="ThingModelAccessMode.READ_WRITE.value">
|
||||
{{ ThingModelAccessMode.READ_WRITE.label }}
|
||||
</el-radio>
|
||||
<el-radio :label="ThingModelAccessMode.READ_ONLY.value">
|
||||
{{ ThingModelAccessMode.READ_ONLY.label }}
|
||||
</el-radio>
|
||||
</el-radio-group>
|
||||
</el-form-item>
|
||||
<el-form-item label="属性描述" prop="description">
|
||||
<el-input
|
||||
v-model="property.description"
|
||||
:maxlength="200"
|
||||
:rows="3"
|
||||
placeholder="请输入属性描述"
|
||||
type="textarea"
|
||||
/>
|
||||
</el-form-item>
|
||||
</template>
|
||||
|
||||
<script lang="ts" setup>
|
||||
import { useVModel } from '@vueuse/core'
|
||||
import { DataSpecsDataType, dataTypeOptions, validateBoolName } from './config'
|
||||
import {
|
||||
DataSpecsDataType,
|
||||
dataTypeOptions,
|
||||
ThingModelAccessMode,
|
||||
validateBoolName
|
||||
} from './config'
|
||||
import {
|
||||
ThingModelArrayDataSpecs,
|
||||
ThingModelEnumDataSpecs,
|
||||
|
@ -103,10 +103,10 @@ import {
|
|||
} from './dataSpecs'
|
||||
import { ThingModelProperty } from '@/api/iot/thingmodel'
|
||||
|
||||
/** IoT 物模型数据 */
|
||||
defineOptions({ name: 'ThingModelDataSpecs' })
|
||||
/** IoT 物模型属性 */
|
||||
defineOptions({ name: 'ThingModelProperty' })
|
||||
|
||||
const props = defineProps<{ modelValue: any; isStructDataSpecs?: boolean }>()
|
||||
const props = defineProps<{ modelValue: any; isStructDataSpecs?: boolean; isParams?: boolean }>()
|
||||
const emits = defineEmits(['update:modelValue'])
|
||||
const property = useVModel(props, 'modelValue', emits) as Ref<ThingModelProperty>
|
||||
const getDataTypeOptions = computed(() => {
|
||||
|
@ -117,12 +117,14 @@ const getDataTypeOptions = computed(() => {
|
|||
!([DataSpecsDataType.STRUCT, DataSpecsDataType.ARRAY] as any[]).includes(item.value)
|
||||
)
|
||||
}) // 获得数据类型列表
|
||||
|
||||
/** 属性值的数据类型切换时初始化相关数据 */
|
||||
const handleChange = (dataType: any) => {
|
||||
property.value.dataSpecsList = []
|
||||
property.value.dataSpecs = {}
|
||||
|
||||
property.value.dataSpecs.dataType = dataType
|
||||
// 不是列表型数据才设置 dataSpecs.dataType
|
||||
![DataSpecsDataType.ENUM, DataSpecsDataType.BOOL, DataSpecsDataType.STRUCT].includes(dataType) &&
|
||||
(property.value.dataSpecs.dataType = dataType)
|
||||
switch (dataType) {
|
||||
case DataSpecsDataType.ENUM:
|
||||
property.value.dataSpecsList.push({
|
|
@ -0,0 +1,50 @@
|
|||
<template>
|
||||
<el-form-item
|
||||
:rules="[{ required: true, message: '请选择调用方式', trigger: 'change' }]"
|
||||
label="调用方式"
|
||||
prop="service.callType"
|
||||
>
|
||||
<el-radio-group v-model="service.callType">
|
||||
<el-radio :value="ThingModelServiceCallType.ASYNC.value">
|
||||
{{ ThingModelServiceCallType.ASYNC.label }}
|
||||
</el-radio>
|
||||
<el-radio :value="ThingModelServiceCallType.SYNC.value">
|
||||
{{ ThingModelServiceCallType.SYNC.label }}
|
||||
</el-radio>
|
||||
</el-radio-group>
|
||||
</el-form-item>
|
||||
<el-form-item label="输入参数">
|
||||
<ThingModelInputOutputParam
|
||||
v-model="service.inputParams"
|
||||
:direction="ThingModelParamDirection.INPUT"
|
||||
/>
|
||||
</el-form-item>
|
||||
<el-form-item label="输出参数">
|
||||
<ThingModelInputOutputParam
|
||||
v-model="service.outputParams"
|
||||
:direction="ThingModelParamDirection.OUTPUT"
|
||||
/>
|
||||
</el-form-item>
|
||||
</template>
|
||||
|
||||
<script lang="ts" setup>
|
||||
import ThingModelInputOutputParam from './ThingModelInputOutputParam.vue'
|
||||
import { useVModel } from '@vueuse/core'
|
||||
import { ThingModelService } from '@/api/iot/thingmodel'
|
||||
import { ThingModelParamDirection, ThingModelServiceCallType } from './config'
|
||||
|
||||
/** IoT 物模型服务 */
|
||||
defineOptions({ name: 'ThingModelService' })
|
||||
|
||||
const props = defineProps<{ modelValue: any; isStructDataSpecs?: boolean }>()
|
||||
const emits = defineEmits(['update:modelValue'])
|
||||
const service = useVModel(props, 'modelValue', emits) as Ref<ThingModelService>
|
||||
</script>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
:deep(.el-form-item) {
|
||||
.el-form-item {
|
||||
margin-bottom: 0;
|
||||
}
|
||||
}
|
||||
</style>
|
|
@ -1,4 +1,4 @@
|
|||
import {isEmpty} from '@/utils/is'
|
||||
import { isEmpty } from '@/utils/is'
|
||||
|
||||
/** dataSpecs 数值型数据结构 */
|
||||
export interface DataSpecsNumberDataVO {
|
||||
|
@ -48,9 +48,69 @@ export const dataTypeOptions = [
|
|||
|
||||
/** 获得物体模型数据类型配置项名称 */
|
||||
export const getDataTypeOptionsLabel = (value: string) => {
|
||||
if (isEmpty(value)) {
|
||||
return value
|
||||
}
|
||||
return dataTypeOptions.find((option) => option.value === value)?.label
|
||||
}
|
||||
|
||||
// IOT 产品物模型类型枚举类
|
||||
export const ThingModelType = {
|
||||
PROPERTY: 1, // 属性
|
||||
SERVICE: 2, // 服务
|
||||
EVENT: 3 // 事件
|
||||
} as const
|
||||
|
||||
// IOT 产品物模型访问模式枚举类
|
||||
export const ThingModelAccessMode = {
|
||||
READ_WRITE: {
|
||||
label: '读写',
|
||||
value: 'rw'
|
||||
},
|
||||
READ_ONLY: {
|
||||
label: '只读',
|
||||
value: 'r'
|
||||
}
|
||||
} as const
|
||||
|
||||
// IOT 产品物模型服务调用方式枚举
|
||||
export const ThingModelServiceCallType = {
|
||||
ASYNC: {
|
||||
label: '异步调用',
|
||||
value: 'async'
|
||||
},
|
||||
SYNC: {
|
||||
label: '同步调用',
|
||||
value: 'sync'
|
||||
}
|
||||
} as const
|
||||
export const getCallTypeByValue = (value: string): string | undefined =>
|
||||
Object.values(ThingModelServiceCallType).find((type) => type.value === value)?.label
|
||||
|
||||
// IOT 产品物模型事件类型枚举
|
||||
export const ThingModelEventType = {
|
||||
INFO: {
|
||||
label: '信息',
|
||||
value: 'info'
|
||||
},
|
||||
ALERT: {
|
||||
label: '告警',
|
||||
value: 'alert'
|
||||
},
|
||||
ERROR: {
|
||||
label: '故障',
|
||||
value: 'error'
|
||||
}
|
||||
} as const
|
||||
export const getEventTypeByValue = (value: string): string | undefined =>
|
||||
Object.values(ThingModelEventType).find((type) => type.value === value)?.label
|
||||
|
||||
// IOT 产品物模型参数是输入参数还是输出参数
|
||||
export const ThingModelParamDirection = {
|
||||
INPUT: 'input', // 输入参数
|
||||
OUTPUT: 'output' // 输出参数
|
||||
} as const
|
||||
|
||||
/** 公共校验规则 */
|
||||
export const ThingModelFormRules = {
|
||||
name: [
|
||||
|
|
|
@ -47,10 +47,10 @@
|
|||
@change="unitChange"
|
||||
>
|
||||
<el-option
|
||||
v-for="(item, index) in UnifyUnitSpecsDTO"
|
||||
v-for="(item, index) in getStrDictOptions(DICT_TYPE.IOT_PRODUCT_THING_MODEL_UNIT)"
|
||||
:key="index"
|
||||
:label="item.Name + '-' + item.Symbol"
|
||||
:value="item.Name + '-' + item.Symbol"
|
||||
:label="item.label + '-' + item.value"
|
||||
:value="item.label + '-' + item.value"
|
||||
/>
|
||||
</el-select>
|
||||
</el-form-item>
|
||||
|
@ -58,8 +58,8 @@
|
|||
|
||||
<script lang="ts" setup>
|
||||
import { useVModel } from '@vueuse/core'
|
||||
import { UnifyUnitSpecsDTO } from '@/views/iot/utils/constants'
|
||||
import { DataSpecsNumberDataVO } from '../config'
|
||||
import { DICT_TYPE, getStrDictOptions } from '@/utils/dict'
|
||||
|
||||
/** 数值型的 dataSpecs 配置组件 */
|
||||
defineOptions({ name: 'ThingModelNumberDataSpecs' })
|
||||
|
|
|
@ -35,7 +35,7 @@
|
|||
<el-input v-model="formData.identifier" placeholder="请输入标识符" />
|
||||
</el-form-item>
|
||||
<!-- 属性配置 -->
|
||||
<ThingModelDataSpecs v-model="formData.property" is-struct-data-specs />
|
||||
<ThingModelProperty v-model="formData.property" is-struct-data-specs />
|
||||
</el-form>
|
||||
|
||||
<template #footer>
|
||||
|
@ -47,7 +47,7 @@
|
|||
|
||||
<script lang="ts" setup>
|
||||
import { useVModel } from '@vueuse/core'
|
||||
import ThingModelDataSpecs from '../ThingModelDataSpecs.vue'
|
||||
import ThingModelProperty from '../ThingModelProperty.vue'
|
||||
import { DataSpecsDataType, ThingModelFormRules } from '../config'
|
||||
import { isEmpty } from '@/utils/is'
|
||||
|
||||
|
@ -111,7 +111,7 @@ const submitForm = async () => {
|
|||
!!data.property.dataSpecs && Object.keys(data.property.dataSpecs).length > 1
|
||||
? data.property.dataSpecs
|
||||
: undefined,
|
||||
dataSpecsList: data.property.dataSpecsList
|
||||
dataSpecsList: isEmpty(data.property.dataSpecsList) ? undefined : data.property.dataSpecsList
|
||||
}
|
||||
|
||||
// 查找是否已有相同 identifier 的项
|
||||
|
|
|
@ -56,13 +56,63 @@
|
|||
<el-table-column align="center" label="标识符" prop="identifier" />
|
||||
<el-table-column align="center" label="数据类型" prop="identifier">
|
||||
<template #default="{ row }">
|
||||
{{ dataTypeOptionsLabel(row.property.dataType) ?? '-' }}
|
||||
{{ dataTypeOptionsLabel(row.property?.dataType) ?? '-' }}
|
||||
</template>
|
||||
</el-table-column>
|
||||
<el-table-column align="center" label="数据定义" prop="identifier">
|
||||
<el-table-column align="left" label="数据定义" prop="identifier">
|
||||
<template #default="{ row }">
|
||||
<!-- TODO puhui999: 数据定义展示待完善 -->
|
||||
{{ row.property.dataSpecs ?? row.property.dataSpecsList }}
|
||||
<!-- 属性 -->
|
||||
<template v-if="row.type === ThingModelType.PROPERTY">
|
||||
<!-- 非列表型:数值 -->
|
||||
<div
|
||||
v-if="
|
||||
[
|
||||
DataSpecsDataType.INT,
|
||||
DataSpecsDataType.DOUBLE,
|
||||
DataSpecsDataType.FLOAT
|
||||
].includes(row.property.dataType)
|
||||
"
|
||||
>
|
||||
取值范围:{{ `${row.property.dataSpecs.min}~${row.property.dataSpecs.max}` }}
|
||||
</div>
|
||||
<!-- 非列表型:文本 -->
|
||||
<div v-if="DataSpecsDataType.TEXT === row.property.dataType">
|
||||
数据长度:{{ row.property.dataSpecs.length }}
|
||||
</div>
|
||||
<!-- 列表型: 数组、结构、时间(特殊) -->
|
||||
<div
|
||||
v-if="
|
||||
[
|
||||
DataSpecsDataType.ARRAY,
|
||||
DataSpecsDataType.STRUCT,
|
||||
DataSpecsDataType.DATE
|
||||
].includes(row.property.dataType)
|
||||
"
|
||||
>
|
||||
-
|
||||
</div>
|
||||
<!-- 列表型: 布尔值、枚举 -->
|
||||
<div
|
||||
v-if="
|
||||
[DataSpecsDataType.BOOL, DataSpecsDataType.ENUM].includes(row.property.dataType)
|
||||
"
|
||||
>
|
||||
<div>
|
||||
{{ DataSpecsDataType.BOOL === row.property.dataType ? '布尔值' : '枚举值' }}:
|
||||
</div>
|
||||
<div v-for="item in row.property.dataSpecsList" :key="item.value">
|
||||
{{ `${item.name}-${item.value}` }}
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
<!-- 服务 -->
|
||||
<div v-if="row.type === ThingModelType.SERVICE">
|
||||
调用方式:{{ getCallTypeByValue(row.service.callType) }}
|
||||
</div>
|
||||
<!-- 事件 -->
|
||||
<div v-if="row.type === ThingModelType.EVENT">
|
||||
事件类型:{{ getEventTypeByValue(row.event.type) }}
|
||||
</div>
|
||||
</template>
|
||||
</el-table-column>
|
||||
<el-table-column align="center" label="操作">
|
||||
|
@ -104,7 +154,14 @@ import { DICT_TYPE, getIntDictOptions } from '@/utils/dict'
|
|||
import ThingModelForm from './ThingModelForm.vue'
|
||||
import { ProductVO } from '@/api/iot/product/product'
|
||||
import { IOT_PROVIDE_KEY } from '@/views/iot/utils/constants'
|
||||
import { getDataTypeOptionsLabel } from '@/views/iot/thingmodel/config'
|
||||
import {
|
||||
DataSpecsDataType,
|
||||
getCallTypeByValue,
|
||||
getDataTypeOptionsLabel,
|
||||
getEventTypeByValue,
|
||||
ThingModelType
|
||||
} from './config'
|
||||
import { ThingModelNumberDataSpecs } from '@/views/iot/thingmodel/dataSpecs'
|
||||
|
||||
defineOptions({ name: 'IoTProductThingModel' })
|
||||
|
||||
|
|
|
@ -2,534 +2,3 @@
|
|||
export const IOT_PROVIDE_KEY = {
|
||||
PRODUCT: 'IOT_PRODUCT'
|
||||
}
|
||||
// TODO puhui999: 物模型数字数据类型单位类型,后面改成字典获取
|
||||
export const UnifyUnitSpecsDTO = [
|
||||
{
|
||||
Symbol: 'L/min',
|
||||
Name: '升每分钟'
|
||||
},
|
||||
{
|
||||
Symbol: 'mg/kg',
|
||||
Name: '毫克每千克'
|
||||
},
|
||||
{
|
||||
Symbol: 'NTU',
|
||||
Name: '浊度'
|
||||
},
|
||||
{
|
||||
Symbol: 'pH',
|
||||
Name: 'PH值'
|
||||
},
|
||||
{
|
||||
Symbol: 'dS/m',
|
||||
Name: '土壤EC值'
|
||||
},
|
||||
{
|
||||
Symbol: 'W/㎡',
|
||||
Name: '太阳总辐射'
|
||||
},
|
||||
{
|
||||
Symbol: 'mm/hour',
|
||||
Name: '降雨量'
|
||||
},
|
||||
{
|
||||
Symbol: 'var',
|
||||
Name: '乏'
|
||||
},
|
||||
{
|
||||
Symbol: 'cP',
|
||||
Name: '厘泊'
|
||||
},
|
||||
{
|
||||
Symbol: 'aw',
|
||||
Name: '饱和度'
|
||||
},
|
||||
{
|
||||
Symbol: 'pcs',
|
||||
Name: '个'
|
||||
},
|
||||
{
|
||||
Symbol: 'cst',
|
||||
Name: '厘斯'
|
||||
},
|
||||
{
|
||||
Symbol: 'bar',
|
||||
Name: '巴'
|
||||
},
|
||||
{
|
||||
Symbol: 'ppt',
|
||||
Name: '纳克每升'
|
||||
},
|
||||
{
|
||||
Symbol: 'ppb',
|
||||
Name: '微克每升'
|
||||
},
|
||||
{
|
||||
Symbol: 'uS/cm',
|
||||
Name: '微西每厘米'
|
||||
},
|
||||
{
|
||||
Symbol: 'N/C',
|
||||
Name: '牛顿每库仑'
|
||||
},
|
||||
{
|
||||
Symbol: 'V/m',
|
||||
Name: '伏特每米'
|
||||
},
|
||||
{
|
||||
Symbol: 'ml/min',
|
||||
Name: '滴速'
|
||||
},
|
||||
{
|
||||
Symbol: 'mmHg',
|
||||
Name: '毫米汞柱'
|
||||
},
|
||||
{
|
||||
Symbol: 'mmol/L',
|
||||
Name: '血糖'
|
||||
},
|
||||
{
|
||||
Symbol: 'mm/s',
|
||||
Name: '毫米每秒'
|
||||
},
|
||||
{
|
||||
Symbol: 'turn/m',
|
||||
Name: '转每分钟'
|
||||
},
|
||||
{
|
||||
Symbol: 'count',
|
||||
Name: '次'
|
||||
},
|
||||
{
|
||||
Symbol: 'gear',
|
||||
Name: '档'
|
||||
},
|
||||
{
|
||||
Symbol: 'stepCount',
|
||||
Name: '步'
|
||||
},
|
||||
{
|
||||
Symbol: 'Nm3/h',
|
||||
Name: '标准立方米每小时'
|
||||
},
|
||||
{
|
||||
Symbol: 'kV',
|
||||
Name: '千伏'
|
||||
},
|
||||
{
|
||||
Symbol: 'kVA',
|
||||
Name: '千伏安'
|
||||
},
|
||||
{
|
||||
Symbol: 'kVar',
|
||||
Name: '千乏'
|
||||
},
|
||||
{
|
||||
Symbol: 'uw/cm2',
|
||||
Name: '微瓦每平方厘米'
|
||||
},
|
||||
{
|
||||
Symbol: '只',
|
||||
Name: '只'
|
||||
},
|
||||
{
|
||||
Symbol: '%RH',
|
||||
Name: '相对湿度'
|
||||
},
|
||||
{
|
||||
Symbol: 'm³/s',
|
||||
Name: '立方米每秒'
|
||||
},
|
||||
{
|
||||
Symbol: 'kg/s',
|
||||
Name: '公斤每秒'
|
||||
},
|
||||
{
|
||||
Symbol: 'r/min',
|
||||
Name: '转每分钟'
|
||||
},
|
||||
{
|
||||
Symbol: 't/h',
|
||||
Name: '吨每小时'
|
||||
},
|
||||
{
|
||||
Symbol: 'KCL/h',
|
||||
Name: '千卡每小时'
|
||||
},
|
||||
{
|
||||
Symbol: 'L/s',
|
||||
Name: '升每秒'
|
||||
},
|
||||
{
|
||||
Symbol: 'Mpa',
|
||||
Name: '兆帕'
|
||||
},
|
||||
{
|
||||
Symbol: 'm³/h',
|
||||
Name: '立方米每小时'
|
||||
},
|
||||
{
|
||||
Symbol: 'kvarh',
|
||||
Name: '千乏时'
|
||||
},
|
||||
{
|
||||
Symbol: 'μg/L',
|
||||
Name: '微克每升'
|
||||
},
|
||||
{
|
||||
Symbol: 'kcal',
|
||||
Name: '千卡路里'
|
||||
},
|
||||
{
|
||||
Symbol: 'GB',
|
||||
Name: '吉字节'
|
||||
},
|
||||
{
|
||||
Symbol: 'MB',
|
||||
Name: '兆字节'
|
||||
},
|
||||
{
|
||||
Symbol: 'KB',
|
||||
Name: '千字节'
|
||||
},
|
||||
{
|
||||
Symbol: 'B',
|
||||
Name: '字节'
|
||||
},
|
||||
{
|
||||
Symbol: 'μg/(d㎡·d)',
|
||||
Name: '微克每平方分米每天'
|
||||
},
|
||||
{
|
||||
Symbol: '',
|
||||
Name: '无'
|
||||
},
|
||||
{
|
||||
Symbol: 'ppm',
|
||||
Name: '百万分率'
|
||||
},
|
||||
{
|
||||
Symbol: 'pixel',
|
||||
Name: '像素'
|
||||
},
|
||||
{
|
||||
Symbol: 'Lux',
|
||||
Name: '照度'
|
||||
},
|
||||
{
|
||||
Symbol: 'grav',
|
||||
Name: '重力加速度'
|
||||
},
|
||||
{
|
||||
Symbol: 'dB',
|
||||
Name: '分贝'
|
||||
},
|
||||
{
|
||||
Symbol: '%',
|
||||
Name: '百分比'
|
||||
},
|
||||
{
|
||||
Symbol: 'lm',
|
||||
Name: '流明'
|
||||
},
|
||||
{
|
||||
Symbol: 'bit',
|
||||
Name: '比特'
|
||||
},
|
||||
{
|
||||
Symbol: 'g/mL',
|
||||
Name: '克每毫升'
|
||||
},
|
||||
{
|
||||
Symbol: 'g/L',
|
||||
Name: '克每升'
|
||||
},
|
||||
{
|
||||
Symbol: 'mg/L',
|
||||
Name: '毫克每升'
|
||||
},
|
||||
{
|
||||
Symbol: 'μg/m³',
|
||||
Name: '微克每立方米'
|
||||
},
|
||||
{
|
||||
Symbol: 'mg/m³',
|
||||
Name: '毫克每立方米'
|
||||
},
|
||||
{
|
||||
Symbol: 'g/m³',
|
||||
Name: '克每立方米'
|
||||
},
|
||||
{
|
||||
Symbol: 'kg/m³',
|
||||
Name: '千克每立方米'
|
||||
},
|
||||
{
|
||||
Symbol: 'nF',
|
||||
Name: '纳法'
|
||||
},
|
||||
{
|
||||
Symbol: 'pF',
|
||||
Name: '皮法'
|
||||
},
|
||||
{
|
||||
Symbol: 'μF',
|
||||
Name: '微法'
|
||||
},
|
||||
{
|
||||
Symbol: 'F',
|
||||
Name: '法拉'
|
||||
},
|
||||
{
|
||||
Symbol: 'Ω',
|
||||
Name: '欧姆'
|
||||
},
|
||||
{
|
||||
Symbol: 'μA',
|
||||
Name: '微安'
|
||||
},
|
||||
{
|
||||
Symbol: 'mA',
|
||||
Name: '毫安'
|
||||
},
|
||||
{
|
||||
Symbol: 'kA',
|
||||
Name: '千安'
|
||||
},
|
||||
{
|
||||
Symbol: 'A',
|
||||
Name: '安培'
|
||||
},
|
||||
{
|
||||
Symbol: 'mV',
|
||||
Name: '毫伏'
|
||||
},
|
||||
{
|
||||
Symbol: 'V',
|
||||
Name: '伏特'
|
||||
},
|
||||
{
|
||||
Symbol: 'ms',
|
||||
Name: '毫秒'
|
||||
},
|
||||
{
|
||||
Symbol: 's',
|
||||
Name: '秒'
|
||||
},
|
||||
{
|
||||
Symbol: 'min',
|
||||
Name: '分钟'
|
||||
},
|
||||
{
|
||||
Symbol: 'h',
|
||||
Name: '小时'
|
||||
},
|
||||
{
|
||||
Symbol: 'day',
|
||||
Name: '日'
|
||||
},
|
||||
{
|
||||
Symbol: 'week',
|
||||
Name: '周'
|
||||
},
|
||||
{
|
||||
Symbol: 'month',
|
||||
Name: '月'
|
||||
},
|
||||
{
|
||||
Symbol: 'year',
|
||||
Name: '年'
|
||||
},
|
||||
{
|
||||
Symbol: 'kn',
|
||||
Name: '节'
|
||||
},
|
||||
{
|
||||
Symbol: 'km/h',
|
||||
Name: '千米每小时'
|
||||
},
|
||||
{
|
||||
Symbol: 'm/s',
|
||||
Name: '米每秒'
|
||||
},
|
||||
{
|
||||
Symbol: '″',
|
||||
Name: '秒'
|
||||
},
|
||||
{
|
||||
Symbol: '′',
|
||||
Name: '分'
|
||||
},
|
||||
{
|
||||
Symbol: '°',
|
||||
Name: '度'
|
||||
},
|
||||
{
|
||||
Symbol: 'rad',
|
||||
Name: '弧度'
|
||||
},
|
||||
{
|
||||
Symbol: 'Hz',
|
||||
Name: '赫兹'
|
||||
},
|
||||
{
|
||||
Symbol: 'μW',
|
||||
Name: '微瓦'
|
||||
},
|
||||
{
|
||||
Symbol: 'mW',
|
||||
Name: '毫瓦'
|
||||
},
|
||||
{
|
||||
Symbol: 'kW',
|
||||
Name: '千瓦特'
|
||||
},
|
||||
{
|
||||
Symbol: 'W',
|
||||
Name: '瓦特'
|
||||
},
|
||||
{
|
||||
Symbol: 'cal',
|
||||
Name: '卡路里'
|
||||
},
|
||||
{
|
||||
Symbol: 'kW·h',
|
||||
Name: '千瓦时'
|
||||
},
|
||||
{
|
||||
Symbol: 'Wh',
|
||||
Name: '瓦时'
|
||||
},
|
||||
{
|
||||
Symbol: 'eV',
|
||||
Name: '电子伏'
|
||||
},
|
||||
{
|
||||
Symbol: 'kJ',
|
||||
Name: '千焦'
|
||||
},
|
||||
{
|
||||
Symbol: 'J',
|
||||
Name: '焦耳'
|
||||
},
|
||||
{
|
||||
Symbol: '℉',
|
||||
Name: '华氏度'
|
||||
},
|
||||
{
|
||||
Symbol: 'K',
|
||||
Name: '开尔文'
|
||||
},
|
||||
{
|
||||
Symbol: 't',
|
||||
Name: '吨'
|
||||
},
|
||||
{
|
||||
Symbol: '°C',
|
||||
Name: '摄氏度'
|
||||
},
|
||||
{
|
||||
Symbol: 'mPa',
|
||||
Name: '毫帕'
|
||||
},
|
||||
{
|
||||
Symbol: 'hPa',
|
||||
Name: '百帕'
|
||||
},
|
||||
{
|
||||
Symbol: 'kPa',
|
||||
Name: '千帕'
|
||||
},
|
||||
{
|
||||
Symbol: 'Pa',
|
||||
Name: '帕斯卡'
|
||||
},
|
||||
{
|
||||
Symbol: 'mg',
|
||||
Name: '毫克'
|
||||
},
|
||||
{
|
||||
Symbol: 'g',
|
||||
Name: '克'
|
||||
},
|
||||
{
|
||||
Symbol: 'kg',
|
||||
Name: '千克'
|
||||
},
|
||||
{
|
||||
Symbol: 'N',
|
||||
Name: '牛'
|
||||
},
|
||||
{
|
||||
Symbol: 'mL',
|
||||
Name: '毫升'
|
||||
},
|
||||
{
|
||||
Symbol: 'L',
|
||||
Name: '升'
|
||||
},
|
||||
{
|
||||
Symbol: 'mm³',
|
||||
Name: '立方毫米'
|
||||
},
|
||||
{
|
||||
Symbol: 'cm³',
|
||||
Name: '立方厘米'
|
||||
},
|
||||
{
|
||||
Symbol: 'km³',
|
||||
Name: '立方千米'
|
||||
},
|
||||
{
|
||||
Symbol: 'm³',
|
||||
Name: '立方米'
|
||||
},
|
||||
{
|
||||
Symbol: 'h㎡',
|
||||
Name: '公顷'
|
||||
},
|
||||
{
|
||||
Symbol: 'c㎡',
|
||||
Name: '平方厘米'
|
||||
},
|
||||
{
|
||||
Symbol: 'm㎡',
|
||||
Name: '平方毫米'
|
||||
},
|
||||
{
|
||||
Symbol: 'k㎡',
|
||||
Name: '平方千米'
|
||||
},
|
||||
{
|
||||
Symbol: '㎡',
|
||||
Name: '平方米'
|
||||
},
|
||||
{
|
||||
Symbol: 'nm',
|
||||
Name: '纳米'
|
||||
},
|
||||
{
|
||||
Symbol: 'μm',
|
||||
Name: '微米'
|
||||
},
|
||||
{
|
||||
Symbol: 'mm',
|
||||
Name: '毫米'
|
||||
},
|
||||
{
|
||||
Symbol: 'cm',
|
||||
Name: '厘米'
|
||||
},
|
||||
{
|
||||
Symbol: 'dm',
|
||||
Name: '分米'
|
||||
},
|
||||
{
|
||||
Symbol: 'km',
|
||||
Name: '千米'
|
||||
},
|
||||
{
|
||||
Symbol: 'm',
|
||||
Name: '米'
|
||||
}
|
||||
]
|
||||
|
|
Loading…
Reference in New Issue