ThingModelDataSpecs.vue 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193
  1. <template>
  2. <el-form-item
  3. :rules="[{ required: true, message: '请选择数据类型', trigger: 'change' }]"
  4. label="数据类型"
  5. prop="property.dataType"
  6. >
  7. <el-select v-model="property.dataType" placeholder="请选择数据类型" @change="handleChange">
  8. <el-option
  9. v-for="option in dataTypeOptions"
  10. :key="option.value"
  11. :label="option.label"
  12. :value="option.value"
  13. />
  14. </el-select>
  15. </el-form-item>
  16. <!-- 数值型配置 -->
  17. <ThingModelNumberTypeDataSpecs
  18. v-if="
  19. [DataSpecsDataType.INT, DataSpecsDataType.DOUBLE, DataSpecsDataType.FLOAT].includes(
  20. property.dataType || ''
  21. )
  22. "
  23. v-model="property.dataSpecs"
  24. />
  25. <!-- 枚举型配置 -->
  26. <ThingModelEnumTypeDataSpecs
  27. v-if="property.dataType === DataSpecsDataType.ENUM"
  28. v-model="property.dataSpecsList"
  29. />
  30. <!-- 布尔型配置 -->
  31. <el-form-item
  32. v-if="property.dataType === DataSpecsDataType.BOOL"
  33. :rules="[{ required: true, message: '请输入布尔值名称', trigger: 'blur' }]"
  34. label="布尔值"
  35. prop="property.dataSpecsList"
  36. >
  37. <template v-for="(item, index) in property.dataSpecsList" :key="item.value">
  38. <div class="flex items-center justify-start w-1/1 mb-5px">
  39. <span>{{ item.value }}</span>
  40. <span class="mx-2">-</span>
  41. <el-form-item
  42. :prop="`property.dataSpecsList[${index}].name`"
  43. :rules="[
  44. { required: true, message: '枚举描述不能为空' },
  45. { validator: validateBoolName, trigger: 'blur' }
  46. ]"
  47. class="flex-1 mb-0"
  48. >
  49. <el-input
  50. v-model="item.name"
  51. :placeholder="`如:${item.value === 0 ? '关' : '开'}`"
  52. class="w-255px!"
  53. />
  54. </el-form-item>
  55. </div>
  56. </template>
  57. </el-form-item>
  58. <!-- 文本型配置 -->
  59. <el-form-item
  60. v-if="property.dataType === DataSpecsDataType.TEXT"
  61. :rules="[
  62. { required: true, message: '请输入文本字节长度', trigger: 'blur' },
  63. { validator: validateTextLength, trigger: 'blur' }
  64. ]"
  65. label="数据长度"
  66. prop="property.dataSpecs.length"
  67. >
  68. <el-input v-model="property.dataSpecs.length" class="w-255px!" placeholder="请输入文本字节长度">
  69. <template #append>字节</template>
  70. </el-input>
  71. </el-form-item>
  72. <!-- 时间型配置 -->
  73. <el-form-item v-if="property.dataType === DataSpecsDataType.DATE" label="时间格式" prop="date">
  74. <el-input class="w-255px!" disabled placeholder="String类型的UTC时间戳(毫秒)" />
  75. </el-form-item>
  76. <!-- 数组型配置-->
  77. <ThingModelArrayTypeDataSpecs
  78. v-if="property.dataType === DataSpecsDataType.ARRAY"
  79. v-model="property.dataSpecs"
  80. />
  81. <!-- TODO puhui999: Struct 属性待完善 -->
  82. <el-form-item
  83. :rules="[{ required: true, message: '请选择读写类型', trigger: 'change' }]"
  84. label="读写类型"
  85. prop="property.accessMode"
  86. >
  87. <el-radio-group v-model="property.accessMode">
  88. <el-radio label="rw">读写</el-radio>
  89. <el-radio label="r">只读</el-radio>
  90. </el-radio-group>
  91. </el-form-item>
  92. <el-form-item label="属性描述" prop="description">
  93. <el-input
  94. v-model="property.description"
  95. :maxlength="200"
  96. :rows="3"
  97. placeholder="请输入属性描述"
  98. type="textarea"
  99. />
  100. </el-form-item>
  101. </template>
  102. <script lang="ts" setup>
  103. import { useVModel } from '@vueuse/core'
  104. import { DataSpecsDataType, dataTypeOptions } from './config'
  105. import {
  106. ThingModelArrayTypeDataSpecs,
  107. ThingModelEnumTypeDataSpecs,
  108. ThingModelNumberTypeDataSpecs
  109. } from './dataSpecs'
  110. import { ThingModelProperty } from 'src/api/iot/thingmodel'
  111. import { isEmpty } from '@/utils/is'
  112. /** IoT 物模型数据 */
  113. defineOptions({ name: 'ThingModelDataSpecs' })
  114. const props = defineProps<{ modelValue: any }>()
  115. const emits = defineEmits(['update:modelValue'])
  116. const property = useVModel(props, 'modelValue', emits) as Ref<ThingModelProperty>
  117. /** 属性值的数据类型切换时初始化相关数据 */
  118. const handleChange = (dataType: any) => {
  119. property.value.dataSpecsList = []
  120. property.value.dataSpecs = {}
  121. property.value.dataSpecs.dataType = dataType
  122. switch (dataType) {
  123. case DataSpecsDataType.ENUM:
  124. property.value.dataSpecsList.push({
  125. dataType: DataSpecsDataType.ENUM,
  126. name: '', // 枚举项的名称
  127. value: undefined // 枚举值
  128. })
  129. break
  130. case DataSpecsDataType.BOOL:
  131. for (let i = 0; i < 2; i++) {
  132. property.value.dataSpecsList.push({
  133. dataType: DataSpecsDataType.BOOL,
  134. name: '', // 布尔值的名称
  135. value: i // 布尔值
  136. })
  137. }
  138. break
  139. }
  140. }
  141. // TODO @puhui999:一些校验的规则,是不是写到 utils 里。
  142. /** 校验布尔值名称 */
  143. const validateBoolName = (_: any, value: string, callback: any) => {
  144. if (isEmpty(value)) {
  145. callback(new Error('布尔值名称不能为空'))
  146. return
  147. }
  148. // 检查开头字符
  149. if (!/^[\u4e00-\u9fa5a-zA-Z0-9]/.test(value)) {
  150. callback(new Error('布尔值名称必须以中文、英文字母或数字开头'))
  151. return
  152. }
  153. // 检查整体格式
  154. if (!/^[\u4e00-\u9fa5a-zA-Z0-9][a-zA-Z0-9\u4e00-\u9fa5_-]*$/.test(value)) {
  155. callback(new Error('布尔值名称只能包含中文、英文字母、数字、下划线和短划线'))
  156. return
  157. }
  158. // 检查长度(一个中文算一个字符)
  159. if (value.length > 20) {
  160. callback(new Error('布尔值名称长度不能超过20个字符'))
  161. return
  162. }
  163. callback()
  164. }
  165. /** 校验文本长度 */
  166. const validateTextLength = (_: any, value: any, callback: any) => {
  167. if (isEmpty(value)) {
  168. callback(new Error('文本长度不能为空'))
  169. return
  170. }
  171. if (isNaN(Number(value))) {
  172. callback(new Error('文本长度必须是数字'))
  173. return
  174. }
  175. callback()
  176. }
  177. </script>
  178. <style lang="scss" scoped>
  179. :deep(.el-form-item) {
  180. .el-form-item {
  181. margin-bottom: 0;
  182. }
  183. }
  184. </style>