fix: 合并冲突
This commit is contained in:
		
						commit
						21863de3b1
					
				|  | @ -2,6 +2,7 @@ import { LocalStore } from '@/utils/comm' | |||
| import server from '@/utils/request' | ||||
| import { BASE_API_PATH, TOKEN_KEY } from '@/utils/variable' | ||||
| import { DeviceInstance } from '@/views/device/Instance/typings' | ||||
| import { UnitType } from '@/views/device/Product/typings'; | ||||
| 
 | ||||
| /** | ||||
|  * 删除设备物模型 | ||||
|  | @ -308,4 +309,8 @@ export const startGateway = (id: string) => server.post(`/gateway/device/${id}/_ | |||
| export const getGatewayDetail = (id: string) => server.get(`/gateway/device/${id}`) | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
| /* | ||||
|  * 获取单位列表 | ||||
|  * @returns 单位列表 | ||||
|  */ | ||||
| export const getUnit = () => server.get<UnitType[]>(`/protocol/units`) | ||||
|  |  | |||
|  | @ -1,4 +1,5 @@ | |||
| import server from '@/utils/request' | ||||
| import type { ProductType } from '@/views/media/Device/typings'; | ||||
| 
 | ||||
| export default { | ||||
|     // 列表
 | ||||
|  | @ -10,4 +11,16 @@ export default { | |||
|     // 修改
 | ||||
|     update: (data: any) => server.put(`/media/device/${data.channel}/${data.id}`, data), | ||||
|     del: (id: string) => server.remove(`/media/device/${id}`), | ||||
|     // 更新通道
 | ||||
|     updateChannels: (id: string) => server.post(`/media/device/${id}/channels/_sync`), | ||||
|     // 查询产品列表
 | ||||
|     queryProductList: (data: any) => server.post<ProductType[]>(`/device/product/_query/no-paging`, data), | ||||
|     // 快速添加产品
 | ||||
|     saveProduct: (data: any) => server.post<any>(`/device/product`, data), | ||||
|     // 产品发布
 | ||||
|     deployProductById: (id: string) => server.post<any>(`/device/product/${id}/deploy`), | ||||
|     // 查询设备接入配置
 | ||||
|     queryProvider: (data?: any) => server.post<any>(`/gateway/device/detail/_query`, data), | ||||
|     // 查询网关配置
 | ||||
|     getConfiguration: (id: string, transport: string) => server.get<any>(`/protocol/${id}/${transport}/configuration`), | ||||
| } | ||||
|  | @ -0,0 +1,40 @@ | |||
| <template> | ||||
|   <a-select v-model:value="_value" mode="tags" :options="options" :size="size" @change="change"></a-select> | ||||
| </template> | ||||
| <script setup lang="ts" name="InputSelect"> | ||||
| import { SizeType } from 'ant-design-vue/es/config-provider'; | ||||
| import { DefaultOptionType, SelectValue } from 'ant-design-vue/es/select'; | ||||
| import { PropType } from 'vue'; | ||||
| type valueType = string | number | ||||
| type Emits = { | ||||
|   (e: 'update:value', data: valueType | undefined): void; | ||||
|   (e: 'change'): void; | ||||
| }; | ||||
| const emit = defineEmits<Emits>(); | ||||
| const props = defineProps({ | ||||
|   value: [String, Number], | ||||
|   options: { | ||||
|     type: Array as PropType<DefaultOptionType[]> | undefined, | ||||
|   }, | ||||
|   size: String as PropType<SizeType> | ||||
| }) | ||||
| const _value = ref<valueType[]>(); | ||||
| watch( | ||||
|   () => props.value, | ||||
|   (val) => { | ||||
|     _value.value = val ? [val] : undefined | ||||
|   }, | ||||
|   { immediate: true } | ||||
| ) | ||||
| 
 | ||||
| const change = (value: SelectValue) => { | ||||
|   const _val = (value as valueType[]) | ||||
|   if (_val.length > 1) { | ||||
|     emit('update:value', _val.slice(_val.length - 1)?.[0]) | ||||
|   } else { | ||||
|     emit('update:value', value?.[0]) | ||||
|   } | ||||
| } | ||||
| </script> | ||||
| <style scoped lang="less"> | ||||
| </style> | ||||
|  | @ -10,7 +10,7 @@ | |||
|                 @change="handleChange" | ||||
|                 :action="FILE_UPLOAD" | ||||
|                 :headers="{ | ||||
|                     'X-Access-Token': LocalStore.get(TOKEN_KEY) | ||||
|                     'X-Access-Token': LocalStore.get(TOKEN_KEY), | ||||
|                 }" | ||||
|                 v-bind="props" | ||||
|             > | ||||
|  | @ -26,8 +26,23 @@ | |||
|                         <div class="upload-image-mask">点击修改</div> | ||||
|                     </template> | ||||
|                     <template v-else> | ||||
|                         <AIcon type="LoadingOutlined" v-if="loading" style="font-size: 20px" /> | ||||
|                         <AIcon v-else type="PlusOutlined" style="font-size: 20px" /> | ||||
|                         <AIcon | ||||
|                             type="LoadingOutlined" | ||||
|                             v-if="loading" | ||||
|                             style="font-size: 20px" | ||||
|                         /> | ||||
|                         <template v-else-if="bgImage"> | ||||
|                             <div | ||||
|                                 class="upload-image" | ||||
|                                 :style="`background-image: url(${bgImage});`" | ||||
|                             ></div> | ||||
|                             <div class="upload-image-mask">点击修改</div> | ||||
|                         </template> | ||||
|                         <AIcon | ||||
|                             v-else | ||||
|                             type="PlusOutlined" | ||||
|                             style="font-size: 20px" | ||||
|                         /> | ||||
|                     </template> | ||||
|                 </div> | ||||
|             </a-upload> | ||||
|  | @ -41,7 +56,7 @@ | |||
| 
 | ||||
| <script lang="ts" setup> | ||||
| import { message, UploadChangeParam, UploadProps } from 'ant-design-vue'; | ||||
| import { FILE_UPLOAD } from '@/api/comm' | ||||
| import { FILE_UPLOAD } from '@/api/comm'; | ||||
| import { TOKEN_KEY } from '@/utils/variable'; | ||||
| import { LocalStore } from '@/utils/comm'; | ||||
| import { CSSProperties } from 'vue'; | ||||
|  | @ -56,6 +71,7 @@ interface JUploadProps extends UploadProps { | |||
|     errorMessage?: string; | ||||
|     size?: number; | ||||
|     style?: CSSProperties; | ||||
|     bgImage?: string; | ||||
| } | ||||
| 
 | ||||
| const emit = defineEmits<Emits>(); | ||||
|  | @ -63,35 +79,42 @@ const emit = defineEmits<Emits>(); | |||
| const props: JUploadProps = defineProps({ | ||||
|     modelValue: { | ||||
|         type: String, | ||||
|         default: '' | ||||
|         default: '', | ||||
|     }, | ||||
|     disabled: { | ||||
|         type: Boolean, | ||||
|         default: false | ||||
|         default: false, | ||||
|     }, | ||||
| }) | ||||
|     bgImage: { | ||||
|         type: String, | ||||
|         default: '', | ||||
|     }, | ||||
| }); | ||||
| 
 | ||||
| const loading = ref<boolean>(false) | ||||
| const imageUrl = ref<string>(props?.modelValue || '') | ||||
| const loading = ref<boolean>(false); | ||||
| const imageUrl = ref<string>(props?.modelValue || ''); | ||||
| const imageTypes = props.types ? props.types : ['image/jpeg', 'image/png']; | ||||
| 
 | ||||
| watch(() => props.modelValue, | ||||
| (newValue)=> { | ||||
|   console.log(newValue) | ||||
|   imageUrl.value = newValue | ||||
| }, { | ||||
| watch( | ||||
|     () => props.modelValue, | ||||
|     (newValue) => { | ||||
|         console.log(newValue); | ||||
|         imageUrl.value = newValue; | ||||
|     }, | ||||
|     { | ||||
|         deep: true, | ||||
|   immediate: true | ||||
| }) | ||||
|         immediate: true, | ||||
|     }, | ||||
| ); | ||||
| 
 | ||||
| const handleChange = (info: UploadChangeParam) => { | ||||
|     if (info.file.status === 'uploading') { | ||||
|         loading.value = true; | ||||
|     } | ||||
|     if (info.file.status === 'done') { | ||||
|         imageUrl.value = info.file.response?.result | ||||
|         imageUrl.value = info.file.response?.result; | ||||
|         loading.value = false; | ||||
|         emit('update:modelValue', info.file.response?.result) | ||||
|         emit('update:modelValue', info.file.response?.result); | ||||
|     } | ||||
|     if (info.file.status === 'error') { | ||||
|         loading.value = false; | ||||
|  |  | |||
|  | @ -0,0 +1,67 @@ | |||
| <template> | ||||
|   <a-popover :visible="visible" placement="left"> | ||||
|     <template #title> | ||||
|       <div style="display: flex; justify-content: space-between; align-items: center;"> | ||||
|         <div style="width: 150px;">配置元素</div> | ||||
|         <close-outlined @click="visible = false" /> | ||||
|       </div> | ||||
|     </template> | ||||
|     <template #content> | ||||
|       <div style="max-width: 400px;"> | ||||
|         <a-form layout="vertical" :model="_value"> | ||||
|           <value-type-form v-model:value="_value" :name="[]" isSub key="sub"></value-type-form> | ||||
|           <a-form-item label="说明" name="description" :rules="[ | ||||
|             { max: 200, message: '最多可输入200个字符' }, | ||||
|           ]"> | ||||
|             <a-textarea v-model:value="_value.description" size="small"></a-textarea> | ||||
|           </a-form-item> | ||||
|         </a-form> | ||||
|       </div> | ||||
|     </template> | ||||
|     <a-button type="dashed" block @click="visible = true"> | ||||
|       配置元素<edit-outlined class="item-icon" /> | ||||
|     </a-button> | ||||
|   </a-popover> | ||||
| 
 | ||||
| </template> | ||||
| <script setup lang="ts" name="ArrayParam"> | ||||
| import ValueTypeForm from '@/views/device/components/Metadata/Base/Edit/ValueTypeForm.vue'; | ||||
| import { EditOutlined, CloseOutlined } from '@ant-design/icons-vue'; | ||||
| import { PropType } from 'vue'; | ||||
| 
 | ||||
| type ValueType = Record<any, any>; | ||||
| 
 | ||||
| const props = defineProps({ | ||||
|   value: { | ||||
|     type: Object as PropType<ValueType>, | ||||
|     default: () => ({ extends: {} }) | ||||
|   }, | ||||
|   name: { | ||||
|     type: Array as PropType<string[]>, | ||||
|     required: true | ||||
|   } | ||||
| }) | ||||
| interface Emits { | ||||
|   (e: 'update:value', data: ValueType): void; | ||||
| } | ||||
| const emit = defineEmits<Emits>() | ||||
| 
 | ||||
| const _value = computed({ | ||||
|   get: () => props.value, | ||||
|   set: val => { | ||||
|     emit('update:value', val) | ||||
|   } | ||||
| }) | ||||
| 
 | ||||
| const visible = ref(false) | ||||
| 
 | ||||
| onMounted(() => { | ||||
|   emit('update:value', { extends: {}, ...props.value }) | ||||
| }) | ||||
| </script> | ||||
| <style lang="less" scoped> | ||||
| .item-icon { | ||||
|   color: rgb(136, 136, 136); | ||||
|   font-size: 12px; | ||||
| } | ||||
| </style> | ||||
|  | @ -0,0 +1,81 @@ | |||
| <template> | ||||
|   <div class="boolean-param"> | ||||
|     <a-row :gutter="4"> | ||||
|       <a-col :span="12"> | ||||
|         <a-form-item label=" " :name="name.concat(['trueText'])" :rules="[ | ||||
|           { required: true, message: '请输入trueText' }, | ||||
|         ]"> | ||||
|           <a-input v-model:value="value.trueText" placeholder="trueText" size="small" /> | ||||
|         </a-form-item> | ||||
|       </a-col> | ||||
|       <a-col :span="12"> | ||||
|         <a-form-item label="-" :name="name.concat(['trueValue'])" :rules="[ | ||||
|           { required: true, message: '请输入trueValue' }, | ||||
|         ]"> | ||||
|           <a-input v-model:value="value.trueValue" placeholder="trueValue" size="small"/> | ||||
|         </a-form-item> | ||||
|       </a-col> | ||||
|       <a-col :span="12"> | ||||
|         <a-form-item label=" " :name="name.concat(['falseText'])" :rules="[ | ||||
|           { required: true, message: '请输入falseText' }, | ||||
|         ]"> | ||||
|           <a-input v-model:value="value.falseText" placeholder="falseText" size="small" /> | ||||
|         </a-form-item> | ||||
|       </a-col> | ||||
|       <a-col :span="12"> | ||||
|         <a-form-item label="-" :name="name.concat(['falseValue'])" :rules="[ | ||||
|           { required: true, message: '请输入falseValue' }, | ||||
|         ]"> | ||||
|           <a-input v-model:value="value.falseValue" placeholder="falseValue" size="small" /> | ||||
|         </a-form-item> | ||||
|       </a-col> | ||||
|     </a-row> | ||||
|   </div> | ||||
| </template> | ||||
| <script setup lang="ts" name="BooleanParam"> | ||||
| import { PropType } from 'vue'; | ||||
| type ModelValueType = Record<string, string> | ||||
| 
 | ||||
| interface Emits { | ||||
|   (e: 'update:value', data: ModelValueType): void; | ||||
| } | ||||
| const emit = defineEmits<Emits>() | ||||
| const props = defineProps({ | ||||
|   value: { | ||||
|     type: Object as PropType<ModelValueType>, | ||||
|     default: () => ({ | ||||
|     }) | ||||
|   }, | ||||
|   name: { | ||||
|     type: Array as PropType<string[]>, | ||||
|     required: true | ||||
|   } | ||||
| }) | ||||
| 
 | ||||
| onMounted(() => { | ||||
|   emit('update:value', | ||||
|     { | ||||
|       trueText: '是', | ||||
|       trueValue: 'true', | ||||
|       falseText: '否', | ||||
|       falseValue: 'false', | ||||
|       ...props.value | ||||
|     }) | ||||
| }) | ||||
| 
 | ||||
| </script> | ||||
| <style lang="less" scoped> | ||||
| .boolean-param { | ||||
|   :deep(.ant-form-item) { | ||||
|     flex-direction: row; | ||||
| 
 | ||||
|     .ant-form-item-label { | ||||
|       >label { | ||||
|         margin: 0 10px 0 0; | ||||
|         height: 28px; | ||||
|         line-height: 28px; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
| </style> | ||||
|  | @ -0,0 +1,160 @@ | |||
| <template> | ||||
|   <div class="enum-param"> | ||||
|     <div class="list-item" v-for="(item, index) in _value" :key="index"> | ||||
|       <div class="item-left"> | ||||
|         <menu-outlined class="item-drag item-icon" /> | ||||
|       </div> | ||||
|       <div class="item-middle item-editable"> | ||||
|         <a-popover :visible="editIndex === index" placement="top"> | ||||
|           <template #title> | ||||
|             <div class="edit-title" style="display: flex; justify-content: space-between; align-items: center;"> | ||||
|               <div style="width: 150px;">枚举项配置</div> | ||||
|               <close-outlined @click="handleClose" /> | ||||
|             </div> | ||||
|           </template> | ||||
|           <template #content> | ||||
|             <a-form :model="_value[index]" layout="vertical"> | ||||
|               <a-form-item label="Value" name="value" :rules="[ | ||||
|                 { required: true, message: '请输入Value' }, | ||||
|               ]"> | ||||
|                 <a-input v-model:value="_value[index].value" size="small"></a-input> | ||||
|               </a-form-item> | ||||
|               <a-form-item label="Text" name="text" :rules="[ | ||||
|                 { required: true, message: '请输入Text' }, | ||||
|               ]"> | ||||
|                 <a-input v-model:value="_value[index].text" size="small"></a-input> | ||||
|               </a-form-item> | ||||
|             </a-form> | ||||
|           </template> | ||||
|           <div class="item-edit" @click="handleEdit(index)"> | ||||
|             {{ item.text || '枚举项配置' }} | ||||
|             <edit-outlined class="item-icon" /> | ||||
|           </div> | ||||
|         </a-popover> | ||||
|       </div> | ||||
|       <div class="item-right"> | ||||
|         <delete-outlined @click="handleDelete(index)"/> | ||||
|       </div> | ||||
|     </div> | ||||
|     <a-button type="dashed" block @click="handleAdd"> | ||||
|       <template #icon><plus-outlined class="item-icon" /></template> | ||||
|       新增枚举型 | ||||
|     </a-button> | ||||
|   </div> | ||||
| </template> | ||||
| <script setup lang="ts" name="BooleanParam"> | ||||
| import { PropType } from 'vue' | ||||
| import { MenuOutlined, EditOutlined, DeleteOutlined, PlusOutlined, CloseOutlined } from '@ant-design/icons-vue'; | ||||
| 
 | ||||
| type EnumType = { | ||||
|   text?: string, | ||||
|   value?: string, | ||||
| } | ||||
| 
 | ||||
| interface Emits { | ||||
|   (e: 'update:value', data: EnumType[]): void; | ||||
| } | ||||
| const emit = defineEmits<Emits>() | ||||
| const props = defineProps({ | ||||
|   value: { | ||||
|     type: Object as PropType<EnumType[]>, | ||||
|     default: () => ([]) | ||||
|   } | ||||
| }) | ||||
| 
 | ||||
| const _value = ref<EnumType[]>([]) | ||||
| watchEffect(() => { | ||||
|   _value.value = props.value | ||||
| }) | ||||
| 
 | ||||
| watch(_value, | ||||
| () => { | ||||
|   emit('update:value', _value.value) | ||||
| }, | ||||
| { deep: true }) | ||||
| 
 | ||||
| const editIndex = ref<number>(-1) | ||||
| const handleEdit = (index: number) => { | ||||
|   editIndex.value = index | ||||
| } | ||||
| const handleDelete = (index: number) => { | ||||
|   editIndex.value = -1 | ||||
|   _value.value.splice(index, 1) | ||||
| } | ||||
| const handleClose = () => { | ||||
|   editIndex.value = -1 | ||||
| } | ||||
| const handleAdd = () => { | ||||
|   _value.value.push({}) | ||||
|   emit('update:value', _value.value) | ||||
| } | ||||
| </script> | ||||
| <style lang="less" scoped> | ||||
| .enum-param { | ||||
|   .list-item { | ||||
|     border: 1px solid #f0f0f0; | ||||
|     display: flex; | ||||
|     justify-content: space-between; | ||||
|     color: rgba(0, 0, 0, 0.85); | ||||
|     padding: 3px 6px; | ||||
|     margin-bottom: 10px; | ||||
|     background-color: #fff; | ||||
|     line-height: 26px; | ||||
|     font-size: 14px; | ||||
| 
 | ||||
|     // .item-left { | ||||
|     //   .item-drag { | ||||
|     //     cursor: move; | ||||
|     //   } | ||||
|     // } | ||||
|     .item-edit { | ||||
|       cursor: pointer; | ||||
|     } | ||||
| 
 | ||||
|     .item-icon { | ||||
|       color: rgb(136, 136, 136); | ||||
|       font-size: 12px; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| :deep(.ant-form-item-label) { | ||||
|   line-height: 1; | ||||
| 
 | ||||
|   >label { | ||||
|     font-size: 12px; | ||||
| 
 | ||||
|     &.ant-form-item-required:not(.ant-form-item-required-mark-optional)::before { | ||||
|       font-size: 12px; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| :deep(.ant-form-item-explain) { | ||||
|   font-size: 12px; | ||||
| } | ||||
| 
 | ||||
| :deep(.ant-form-item-with-help) { | ||||
|   .ant-form-item-explain { | ||||
|     min-height: 20px; | ||||
|     line-height: 20px; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| :deep(.ant-form-item) { | ||||
|   margin-bottom: 20px; | ||||
| 
 | ||||
|   &.ant-form-item-with-help { | ||||
|     margin-bottom: 0; | ||||
|   } | ||||
| 
 | ||||
|   input { | ||||
|     font-size: 12px; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| :deep(.ant-input), | ||||
| :deep(.ant-select) { | ||||
|   font-size: 12px; | ||||
| } | ||||
| </style> | ||||
|  | @ -0,0 +1,172 @@ | |||
| <template> | ||||
|   <div class="json-param"> | ||||
|     <div class="list-item" v-for="(item, index) in _value" :key="`object_${index}`"> | ||||
|       <div class="item-left"> | ||||
|         <menu-outlined class="item-drag item-icon" /> | ||||
|       </div> | ||||
|       <div class="item-middle item-editable"> | ||||
|         <a-popover :visible="editIndex === index" placement="left"> | ||||
|           <template #title> | ||||
|             <div class="edit-title" style="display: flex; justify-content: space-between; align-items: center;"> | ||||
|               <div style="width: 150px;">配置参数</div> | ||||
|               <close-outlined @click="handleClose" /> | ||||
|             </div> | ||||
|           </template> | ||||
|           <template #content> | ||||
|             <div style="max-width: 400px;"> | ||||
|               <a-form :model="_value[index]" layout="vertical"> | ||||
|                 <a-form-item label="标识" name="id" :rules="[ | ||||
|                   { required: true, message: '请输入标识' }, | ||||
|                   { max: 64, message: '最多可输入64个字符' }, | ||||
|                   { | ||||
|                     pattern: /^[a-zA-Z0-9_]+$/, | ||||
|                     message: '请输入英文或者数字或者-或者_', | ||||
|                   }, | ||||
|                 ]"> | ||||
|                   <a-input v-model:value="_value[index].id" size="small"></a-input> | ||||
|                 </a-form-item> | ||||
|                 <a-form-item label="名称" name="name" :rules="[ | ||||
|                   { required: true, message: '请输入名称' }, | ||||
|                   { max: 64, message: '最多可输入64个字符' }, | ||||
|                 ]"> | ||||
|                   <a-input v-model:value="_value[index].name" size="small"></a-input> | ||||
|                 </a-form-item> | ||||
|                 <value-type-form v-model:value="_value[index].valueType" :name="['valueType']" isSub | ||||
|                   key="json_sub"></value-type-form> | ||||
|               </a-form> | ||||
|             </div> | ||||
|           </template> | ||||
|           <div class="item-edit" @click="handleEdit(index)"> | ||||
|             {{ item.name || '配置参数' }} | ||||
|             <edit-outlined class="item-icon" /> | ||||
|           </div> | ||||
|         </a-popover> | ||||
|       </div> | ||||
|       <div class="item-right"> | ||||
|         <delete-outlined @click="handleDelete(index)" /> | ||||
|       </div> | ||||
|     </div> | ||||
|     <a-button type="dashed" block @click="handleAdd"> | ||||
|       <template #icon><plus-outlined class="item-icon" /></template> | ||||
|       添加参数 | ||||
|     </a-button> | ||||
|   </div> | ||||
| </template> | ||||
| <script setup lang="ts" name="JsonParam"> | ||||
| import { PropType } from 'vue' | ||||
| import { MenuOutlined, EditOutlined, DeleteOutlined, PlusOutlined, CloseOutlined } from '@ant-design/icons-vue'; | ||||
| import ValueTypeForm from '@/views/device/components/Metadata/Base/Edit/ValueTypeForm.vue'; | ||||
| 
 | ||||
| type JsonType = Record<any, any>; | ||||
| 
 | ||||
| interface Emits { | ||||
|   (e: 'update:value', data: JsonType[]): void; | ||||
| } | ||||
| const emit = defineEmits<Emits>() | ||||
| const props = defineProps({ | ||||
|   value: { | ||||
|     type: Object as PropType<JsonType[]>, | ||||
|     default: () => ([]) | ||||
|   } | ||||
| }) | ||||
| 
 | ||||
| const _value = ref<JsonType[]>([]) | ||||
| watchEffect(() => { | ||||
|   _value.value = props.value | ||||
| }) | ||||
| 
 | ||||
| watch(_value, | ||||
|   () => { | ||||
|     emit('update:value', _value.value) | ||||
|   }, | ||||
|   { deep: true }) | ||||
| 
 | ||||
| const editIndex = ref<number>(-1) | ||||
| const handleEdit = (index: number) => { | ||||
|   editIndex.value = index | ||||
| } | ||||
| const handleDelete = (index: number) => { | ||||
|   editIndex.value = -1 | ||||
|   _value.value.slice(index, 1) | ||||
| } | ||||
| const handleClose = () => { | ||||
|   editIndex.value = -1 | ||||
| } | ||||
| const handleAdd = () => { | ||||
|   _value.value.push({ | ||||
|     valueType: { | ||||
|       expands: {} | ||||
|     }, | ||||
|   }) | ||||
|   emit('update:value', _value.value) | ||||
| } | ||||
| </script> | ||||
| <style lang="less" scoped> | ||||
| .json-param { | ||||
|   .list-item { | ||||
|     border: 1px solid #f0f0f0; | ||||
|     display: flex; | ||||
|     justify-content: space-between; | ||||
|     color: rgba(0, 0, 0, 0.85); | ||||
|     padding: 3px 6px; | ||||
|     margin-bottom: 10px; | ||||
|     background-color: #fff; | ||||
|     line-height: 26px; | ||||
|     font-size: 14px; | ||||
| 
 | ||||
|     // .item-left { | ||||
|     //   .item-drag { | ||||
|     //     cursor: move; | ||||
|     //   } | ||||
|     // } | ||||
|     .item-edit { | ||||
|       cursor: pointer; | ||||
|     } | ||||
| 
 | ||||
|     .item-icon { | ||||
|       color: rgb(136, 136, 136); | ||||
|       font-size: 12px; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| :deep(.ant-form-item-label) { | ||||
|   line-height: 1; | ||||
| 
 | ||||
|   >label { | ||||
|     font-size: 12px; | ||||
| 
 | ||||
|     &.ant-form-item-required:not(.ant-form-item-required-mark-optional)::before { | ||||
|       font-size: 12px; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| :deep(.ant-form-item-explain) { | ||||
|   font-size: 12px; | ||||
| } | ||||
| 
 | ||||
| :deep(.ant-form-item-with-help) { | ||||
|   .ant-form-item-explain { | ||||
|     min-height: 20px; | ||||
|     line-height: 20px; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| :deep(.ant-form-item) { | ||||
|   margin-bottom: 20px; | ||||
| 
 | ||||
|   &.ant-form-item-with-help { | ||||
|     margin-bottom: 0; | ||||
|   } | ||||
| 
 | ||||
|   input { | ||||
|     font-size: 12px; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| :deep(.ant-input), | ||||
| :deep(.ant-select) { | ||||
|   font-size: 12px; | ||||
| } | ||||
| </style> | ||||
|  | @ -1,3 +1,5 @@ | |||
| import { MetadataItem } from "../Product/typings"; | ||||
| 
 | ||||
| export type DeviceInstance = { | ||||
|   id: string; | ||||
|   name: string; | ||||
|  |  | |||
|  | @ -0,0 +1,99 @@ | |||
| <template> | ||||
|   <a-form ref="addFormRef" :model="form.model" layout="vertical"> | ||||
|     <a-form-item label="标识" name="id" :rules="[ | ||||
|       { required: true, message: '请输入标识' }, | ||||
|       { max: 64, message: '最多可输入64个字符' }, | ||||
|       { | ||||
|         pattern: /^[a-zA-Z0-9_]+$/, | ||||
|         message: '请输入英文或者数字或者-或者_', | ||||
|       }, | ||||
|     ]"> | ||||
|       <a-input v-model:value="form.model.id" size="small"></a-input> | ||||
|     </a-form-item> | ||||
|     <a-form-item label="名称" name="name" :rules="[ | ||||
|       { required: true, message: '请输入名称' }, | ||||
|       { max: 64, message: '最多可输入64个字符' }, | ||||
|     ]"> | ||||
|       <a-input v-model:value="form.model.name" size="small"></a-input> | ||||
|     </a-form-item> | ||||
|     <ValueTypeForm :name="['valueType']" v-model:value="form.model.valueType" key="property"></ValueTypeForm> | ||||
|     <a-form-item label="读写类型" :name="['expands', 'type']" :rules="[ | ||||
|       { required: true, message: '请选择读写类型' }, | ||||
|     ]"> | ||||
|       <a-select v-model:value="form.model.expands.type" :options="form.expandsType" mode="multiple" size="small"></a-select> | ||||
|     </a-form-item> | ||||
|     <a-form-item label="说明" name="description" :rules="[ | ||||
|       { max: 200, message: '最多可输入200个字符' }, | ||||
|     ]"> | ||||
|       <a-textarea v-model:value="form.model.description" size="small"></a-textarea> | ||||
|     </a-form-item> | ||||
|   </a-form> | ||||
| </template> | ||||
| <script setup lang="ts" name="PropertyForm"> | ||||
| import ValueTypeForm from './ValueTypeForm.vue' | ||||
| 
 | ||||
| const form = reactive({ | ||||
|   model: { | ||||
|     valueType: { | ||||
|       expands: {} | ||||
|     }, | ||||
|     expands: {} | ||||
|   } as any, | ||||
|   expandsType: [ | ||||
|     { | ||||
|       label: '读', | ||||
|       value: 'read', | ||||
|     }, | ||||
|     { | ||||
|       label: '写', | ||||
|       value: 'write', | ||||
|     }, | ||||
|     { | ||||
|       label: '上报', | ||||
|       value: 'report', | ||||
|     }, | ||||
|   ] | ||||
| }) | ||||
| 
 | ||||
| </script> | ||||
| <style scoped lang="less"> | ||||
| :deep(.ant-form-item-label) { | ||||
|   line-height: 1; | ||||
| 
 | ||||
|   >label { | ||||
|     font-size: 12px; | ||||
| 
 | ||||
|     &.ant-form-item-required:not(.ant-form-item-required-mark-optional)::before { | ||||
|       font-size: 12px; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| :deep(.ant-form-item-explain) { | ||||
|   font-size: 12px; | ||||
| } | ||||
| 
 | ||||
| :deep(.ant-form-item-with-help) { | ||||
|   .ant-form-item-explain { | ||||
|     min-height: 20px; | ||||
|     line-height: 20px; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| :deep(.ant-form-item) { | ||||
|   margin-bottom: 20px; | ||||
| 
 | ||||
|   &.ant-form-item-with-help { | ||||
|     margin-bottom: 0; | ||||
|   } | ||||
| 
 | ||||
|   input { | ||||
|     font-size: 12px; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| :deep(.ant-input), | ||||
| :deep(.ant-select) { | ||||
|   font-size: 12px; | ||||
| } | ||||
| </style> | ||||
|  | @ -0,0 +1,151 @@ | |||
| <template> | ||||
|   <a-form-item label="数据类型" :name="name.concat(['type'])" :rules="[ | ||||
|     { required: true, message: '请选择数据类型' }, | ||||
|   ]"> | ||||
|     <a-select v-model:value="value.type" :options="_dataTypeList" size="small"></a-select> | ||||
|   </a-form-item> | ||||
|   <a-form-item label="单位" :name="name.concat(['unit'])" v-if="['int', 'float', 'long', 'double'].includes(value.type)"> | ||||
|     <InputSelect v-model:value="value.unit" :options="unit.unitOptions" size="small"></InputSelect> | ||||
|   </a-form-item> | ||||
|   <a-form-item label="精度" :name="name.concat(['scale'])" v-if="['float', 'double'].includes(value.type)"> | ||||
|     <a-input-number v-model:value="value.scale" size="small" :min="0" :max="2147483647" :precision="0" | ||||
|       :default-value="2" style="width: 100%"></a-input-number> | ||||
|   </a-form-item> | ||||
|   <a-form-item label="布尔值" name="booleanConfig" v-if="['boolean'].includes(value.type)"> | ||||
|     <BooleanParam  | ||||
|       :name="name" | ||||
|       v-model:value="_value" | ||||
|     ></BooleanParam> | ||||
|   </a-form-item> | ||||
|   <a-form-item label="枚举项" :name="name.concat(['elements'])" v-if="['enum'].includes(value.type)" :rules="[ | ||||
|     { required: true, message: '请配置枚举项' } | ||||
|   ]"> | ||||
|     <EnumParam v-model:value="value.elements"></EnumParam> | ||||
|   </a-form-item> | ||||
|   <a-form-item :name="name.concat(['expands', 'maxLength'])" v-if="['string', 'password'].includes(value.type)"> | ||||
|     <template #label> | ||||
|       <a-space> | ||||
|         最大长度 | ||||
|         <a-tooltip title="字节"> | ||||
|           <question-circle-outlined style="color: rgb(136, 136, 136); font-size: 12px;" /> | ||||
|         </a-tooltip> | ||||
|       </a-space> | ||||
|     </template> | ||||
|     <a-input-number v-model:value="value.expands.maxLength" size="small" :max="2147483647" :min="1" :precision="0" | ||||
|       style="width: 100%;"></a-input-number> | ||||
|   </a-form-item> | ||||
|   <a-form-item label="元素配置" :name="name.concat(['elementType'])" v-if="['array'].includes(value.type)"> | ||||
|     <ArrayParam v-model:value="value.elementType" :name="name.concat(['elementType'])"></ArrayParam> | ||||
|   </a-form-item> | ||||
|   <a-form-item label="JSON对象" :name="name.concat(['properties'])" v-if="['object'].includes(value.type)"> | ||||
|     <JsonParam v-model:value="value.jsonConfig" :name="name.concat(['properties'])"></JsonParam> | ||||
|   </a-form-item> | ||||
|   <a-form-item label="文件类型" :name="name.concat(['fileType'])" v-if="['file'].includes(value.type)" initialValue="url" :rules="[ | ||||
|     { required: true, message: '请选择文件类型' }, | ||||
|   ]"> | ||||
|     <a-select v-model:value="value.fileType" :options="FileTypeList" size="small"></a-select> | ||||
|   </a-form-item> | ||||
| </template> | ||||
| <script lang="ts" setup mame="BaseForm"> | ||||
| import { DataTypeList, FileTypeList } from '@/views/device/data'; | ||||
| import { DefaultOptionType } from 'ant-design-vue/es/select'; | ||||
| import { PropType } from 'vue' | ||||
| import { getUnit } from '@/api/device/instance'; | ||||
| import { Store } from 'jetlinks-store'; | ||||
| import InputSelect from '@/components/InputSelect/index.vue'; | ||||
| import BooleanParam from '@/components/Metadata/BooleanParam/index.vue' | ||||
| import EnumParam from '@/components/Metadata/EnumParam/index.vue' | ||||
| import ArrayParam from '@/components/Metadata/ArrayParam/index.vue' | ||||
| import JsonParam from '@/components/Metadata/JsonParam/index.vue' | ||||
| 
 | ||||
| type ValueType = Record<any, any>; | ||||
| const props = defineProps({ | ||||
|   value: { | ||||
|     type: Object as PropType<ValueType>, | ||||
|     default: () => ({ | ||||
|       extends: {} | ||||
|     }) | ||||
|   }, | ||||
|   isSub: { | ||||
|     type: Boolean, | ||||
|     default: false | ||||
|   }, | ||||
|   name: { | ||||
|     type: Array as PropType<string[]>, | ||||
|     default: () => ([]), | ||||
|     required: true | ||||
|   } | ||||
| }) | ||||
| 
 | ||||
| interface Emits { | ||||
|   (e: 'update:value', data: ValueType): void; | ||||
| } | ||||
| const emit = defineEmits<Emits>() | ||||
| 
 | ||||
| 
 | ||||
| const _value = computed({ | ||||
|   get: () => props.value, | ||||
|   set: val => { | ||||
|     emit('update:value', val) | ||||
|   } | ||||
| }) | ||||
| 
 | ||||
| const unit = { | ||||
|   unitOptions: [] as DefaultOptionType[], | ||||
|   getUnit: () => { | ||||
|     getUnit().then(resp => { | ||||
|       const _data = resp.result.map(item => ({ | ||||
|         label: item.description, | ||||
|         value: item.id, | ||||
|       })); | ||||
|       // 缓存单位数据 | ||||
|       Store.set('units', _data); | ||||
|       unit.unitOptions = _data; | ||||
|     }) | ||||
|   }, | ||||
| } | ||||
| unit.getUnit() | ||||
| 
 | ||||
| const _dataTypeList = computed(() => props.isSub ? DataTypeList.filter(item => item.value !== 'array' && item.value !== 'object') : DataTypeList) | ||||
| </script> | ||||
| <style lang="less" scoped> | ||||
| :deep(.ant-form-item-label) { | ||||
|   line-height: 1; | ||||
| 
 | ||||
|   >label { | ||||
|     font-size: 12px; | ||||
| 
 | ||||
|     &.ant-form-item-required:not(.ant-form-item-required-mark-optional)::before { | ||||
|       font-size: 12px; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| :deep(.ant-form-item-explain) { | ||||
|   font-size: 12px; | ||||
| } | ||||
| 
 | ||||
| :deep(.ant-form-item-with-help) { | ||||
|   .ant-form-item-explain { | ||||
|     min-height: 20px; | ||||
|     line-height: 20px; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| :deep(.ant-form-item) { | ||||
|   margin-bottom: 20px; | ||||
| 
 | ||||
|   &.ant-form-item-with-help { | ||||
|     margin-bottom: 0; | ||||
|   } | ||||
| 
 | ||||
|   input { | ||||
|     font-size: 12px; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| :deep(.ant-input), | ||||
| :deep(.ant-select) { | ||||
|   font-size: 12px; | ||||
| } | ||||
| </style> | ||||
|  | @ -1,10 +1,10 @@ | |||
| <template> | ||||
|   <a-drawer :mask-closable="false" width="25vw" visible :title="`新增${typeMapping[metadataStore.model.type]}`" | ||||
|   <a-drawer :mask-closable="false" width="25vw" visible :title="`${title}-${typeMapping[metadataStore.model.type]}`" | ||||
|     @close="close" destroy-on-close :z-index="1000" placement="right"> | ||||
|     <template #extra> | ||||
|       <a-button :loading="save.loading" type="primary" @click="save.saveMetadata">保存</a-button> | ||||
|     </template> | ||||
|     <a-form ref="addFormRef" :model="form.model"></a-form> | ||||
|     <PropertyForm v-if="metadataStore.model.type === 'properties'"></PropertyForm> | ||||
|   </a-drawer> | ||||
| </template> | ||||
| <script lang="ts" setup name="Edit"> | ||||
|  | @ -19,6 +19,7 @@ import { Store } from 'jetlinks-store'; | |||
| import { SystemConst } from '@/utils/consts'; | ||||
| import { detail } from '@/api/device/instance'; | ||||
| import { DeviceInstance } from '@/views/device/Instance/typings'; | ||||
| import PropertyForm from './PropertyForm.vue'; | ||||
| 
 | ||||
| interface Props { | ||||
|   type: 'product' | 'device'; | ||||
|  | @ -41,6 +42,8 @@ const close = () => { | |||
|   metadataStore.set('item', {}) | ||||
| } | ||||
| 
 | ||||
| const title = computed(() => metadataStore.model.action === 'add' ? '新增' : '修改') | ||||
| 
 | ||||
| const addFormRef = ref<FormInstance>() | ||||
| /** | ||||
|  * 保存按钮 | ||||
|  | @ -113,7 +116,7 @@ const save = reactive({ | |||
| }) | ||||
| 
 | ||||
| const form = reactive({ | ||||
|   model: {} | ||||
|   model: {} as Record<string, any> | ||||
| }) | ||||
| </script> | ||||
| <style lang="less" scoped> | ||||
|  |  | |||
|  | @ -2,8 +2,10 @@ | |||
|   <JTable :loading="loading" :data-source="data" size="small" :columns="columns" row-key="id" model="TABLE"> | ||||
|     <template #headerTitle> | ||||
|       <a-input-search v-model:value="searchValue" placeholder="请输入名称" @search="handleSearch"></a-input-search> | ||||
|     </template> | ||||
|     <template #rightExtraRender> | ||||
|       <PermissionButton type="primary" :uhas-permission="`${permission}:update`" key="add" @click="handleAddClick" | ||||
|         :udisabled="operateLimits('add', type)" :tooltip="{ | ||||
|         :disabled="operateLimits('add', type)" :tooltip="{ | ||||
|           title: operateLimits('add', type) ? '当前的存储方式不支持新增' : '新增', | ||||
|         }"> | ||||
|         <template #icon> | ||||
|  | @ -31,20 +33,20 @@ | |||
|       </a-tag> | ||||
|     </template> | ||||
|     <template #action="slotProps"> | ||||
|       <PermissionButton :has-permission="`${permission}:update`" type="link" key="edit" style="padding: 0" | ||||
|         :disabled="operateLimits('updata', type)" @click="handleEditClick(slotProps)" :tooltip="{ | ||||
|       <PermissionButton :uhas-permission="`${permission}:update`" type="link" key="edit" style="padding: 0" | ||||
|         :udisabled="operateLimits('updata', type)" @click="handleEditClick(slotProps)" :tooltip="{ | ||||
|           title: operateLimits('updata', type) ? '当前的存储方式不支持编辑' : '编辑', | ||||
|         }"> | ||||
|         <EditOutlined /> | ||||
|       </PermissionButton>, | ||||
|       <PermissionButton :has-permission="`${permission}:delete`" type="link" key="delete" style="padding: 0" | ||||
|       </PermissionButton> | ||||
|       <PermissionButton :uhas-permission="`${permission}:delete`" type="link" key="delete" style="padding: 0" | ||||
|         :pop-confirm="{ | ||||
|           title: '确认删除?', onConfirm: async () => { | ||||
|             await removeItem(slotProps); | ||||
|           }, | ||||
|         }" :tooltip="{ | ||||
|           title: '删除', | ||||
| }"> | ||||
|         }"> | ||||
|         <DeleteOutlined /> | ||||
|       </PermissionButton> | ||||
|     </template> | ||||
|  | @ -58,12 +60,13 @@ import { useInstanceStore } from '@/store/instance' | |||
| import { useProductStore } from '@/store/product' | ||||
| import { useMetadataStore } from '@/store/metadata' | ||||
| import PermissionButton from '@/components/PermissionButton/index.vue' | ||||
| import { PlusOutlined } from '@ant-design/icons-vue' | ||||
| import { DeleteOutlined, EditOutlined, PlusOutlined } from '@ant-design/icons-vue' | ||||
| import { message } from 'ant-design-vue/es' | ||||
| import { SystemConst } from '@/utils/consts' | ||||
| import { Store } from 'jetlinks-store' | ||||
| import { asyncUpdateMetadata, removeMetadata } from '../metadata' | ||||
| import { detail } from '@/api/device/instance' | ||||
| import Edit from './Edit/index.vue' | ||||
| // import { detail } from '@/api/device/instance' | ||||
| // import { detail as productDetail } from '@/api/device/product' | ||||
| interface Props { | ||||
|  |  | |||
|  | @ -0,0 +1,135 @@ | |||
| import { isNoCommunity } from '@/utils/utils'; | ||||
| 
 | ||||
| export const DataTypeList: { label: string; value: string }[] = [ | ||||
|   { | ||||
|     value: 'int', | ||||
|     label: 'int(整数型)', | ||||
|   }, | ||||
|   { | ||||
|     value: 'long', | ||||
|     label: 'long(长整数型)', | ||||
|   }, | ||||
|   { | ||||
|     value: 'float', | ||||
|     label: 'float(单精度浮点型)', | ||||
|   }, | ||||
|   { | ||||
|     value: 'double', | ||||
|     label: 'double(双精度浮点数)', | ||||
|   }, | ||||
|   { | ||||
|     value: 'string', | ||||
|     label: 'text(字符串)', | ||||
|   }, | ||||
|   { | ||||
|     value: 'boolean', | ||||
|     label: 'boolean(布尔型)', | ||||
|   }, | ||||
|   { | ||||
|     value: 'date', | ||||
|     label: 'date(时间型)', | ||||
|   }, | ||||
|   { | ||||
|     value: 'enum', | ||||
|     label: 'enum(枚举)', | ||||
|   }, | ||||
|   { | ||||
|     value: 'array', | ||||
|     label: 'array(数组)', | ||||
|   }, | ||||
|   { | ||||
|     value: 'object', | ||||
|     label: 'object(结构体)', | ||||
|   }, | ||||
|   { | ||||
|     value: 'file', | ||||
|     label: 'file(文件)', | ||||
|   }, | ||||
|   { | ||||
|     value: 'password', | ||||
|     label: 'password(密码)', | ||||
|   }, | ||||
|   { | ||||
|     value: 'geoPoint', | ||||
|     label: 'geoPoint(地理位置)', | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| export const PropertySource: { label: string; value: string }[] = isNoCommunity | ||||
|   ? [ | ||||
|       { | ||||
|         value: 'device', | ||||
|         label: '设备', | ||||
|       }, | ||||
|       { | ||||
|         value: 'manual', | ||||
|         label: '手动', | ||||
|       }, | ||||
|       { | ||||
|         value: 'rule', | ||||
|         label: '规则', | ||||
|       }, | ||||
|     ] | ||||
|   : [ | ||||
|       { | ||||
|         value: 'device', | ||||
|         label: '设备', | ||||
|       }, | ||||
|       { | ||||
|         value: 'manual', | ||||
|         label: '手动', | ||||
|       }, | ||||
|     ]; | ||||
| 
 | ||||
| export const FileTypeList: { label: string; value: string }[] = [ | ||||
|   { | ||||
|     label: 'URL(链接)', | ||||
|     value: 'url', | ||||
|   }, | ||||
|   { | ||||
|     label: 'Base64(Base64编码)', | ||||
|     value: 'base64', | ||||
|   }, | ||||
|   { | ||||
|     label: 'binary', | ||||
|     value: 'Binary(二进制)', | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| export const EventLevel: { label: string; value: string }[] = [ | ||||
|   { | ||||
|     label: '普通', | ||||
|     value: 'ordinary', | ||||
|   }, | ||||
|   { | ||||
|     label: '警告', | ||||
|     value: 'warn', | ||||
|   }, | ||||
|   { | ||||
|     value: 'urgent', | ||||
|     label: '紧急', | ||||
|   }, | ||||
| ]; | ||||
| 
 | ||||
| export const DateTypeList = [ | ||||
|   // {
 | ||||
|   //   label: 'String类型的UTC时间戳 (毫秒)',
 | ||||
|   //   value: 'string',
 | ||||
|   // },
 | ||||
|   { | ||||
|     label: 'yyyy-MM-dd', | ||||
|     value: 'yyyy-MM-dd', | ||||
|   }, | ||||
|   { | ||||
|     label: 'yyyy-MM-DD HH:mm:ss', | ||||
|     value: 'yyyy-MM-DD HH:mm:ss', | ||||
|   }, | ||||
|   // {
 | ||||
|   //   label: 'yyyy-MM-dd HH:mm:ss EE',
 | ||||
|   //   value: 'yyyy-MM-dd HH:mm:ss EE',
 | ||||
|   // },
 | ||||
|   // {
 | ||||
|   //   label: 'yyyy-MM-dd HH:mm:ss zzz',
 | ||||
|   //   value: 'yyyy-MM-dd HH:mm:ss zzz',
 | ||||
|   // },
 | ||||
| ]; | ||||
|  | @ -0,0 +1,282 @@ | |||
| <template> | ||||
|     <a-modal | ||||
|         v-model:visible="_vis" | ||||
|         title="快速添加" | ||||
|         cancelText="取消" | ||||
|         okText="确定" | ||||
|         @ok="handleOk" | ||||
|         @cancel="handleCancel" | ||||
|         :confirmLoading="btnLoading" | ||||
|         width="660px" | ||||
|     > | ||||
|         <a-form layout="vertical"> | ||||
|             <a-form-item label="产品名称" v-bind="validateInfos.name"> | ||||
|                 <a-input | ||||
|                     v-model:value="formData.name" | ||||
|                     placeholder="请输入名称" | ||||
|                 /> | ||||
|             </a-form-item> | ||||
|             <template v-if="channel === 'gb28181-2016' && formData.accessId"> | ||||
|                 <a-form-item | ||||
|                     label="接入密码" | ||||
|                     v-bind="validateInfos['configuration.access_pwd']" | ||||
|                 > | ||||
|                     <a-input-password | ||||
|                         v-model:value="formData.configuration.access_pwd" | ||||
|                         placeholder="请输入接入密码" | ||||
|                     /> | ||||
|                 </a-form-item> | ||||
|                 <a-form-item label="流传输模式"> | ||||
|                     <a-select | ||||
|                         v-model:value="formData.configuration.stream_mode" | ||||
|                         placeholder="请选择流传输模式" | ||||
|                         :options="streamMode" | ||||
|                     /> | ||||
|                 </a-form-item> | ||||
|             </template> | ||||
|             <a-form-item label="接入网关" v-bind="validateInfos.accessId"> | ||||
|                 <div class="gateway-box"> | ||||
|                     <div v-if="!gatewayList.length"> | ||||
|                         暂无数据,请先 | ||||
|                         <a-button type="link"> | ||||
|                             添加{{ providerType[props.channel] }} 接入网关 | ||||
|                         </a-button> | ||||
|                     </div> | ||||
|                     <div | ||||
|                         class="gateway-item" | ||||
|                         v-for="(item, index) in gatewayList" | ||||
|                         :key="index" | ||||
|                     > | ||||
|                         <CardBox | ||||
|                             @click="handleClick" | ||||
|                             :active="_selectedRowKeys.includes(item.id)" | ||||
|                             :value="item" | ||||
|                             v-bind="item" | ||||
|                             :status="item.state?.value" | ||||
|                             :statusText="item.state?.text" | ||||
|                             :statusNames="{ | ||||
|                                 online: 'enabled', | ||||
|                                 offline: 'disabled', | ||||
|                             }" | ||||
|                         > | ||||
|                             <template #img> | ||||
|                                 <slot name="img"> | ||||
|                                     <img | ||||
|                                         :src="getImage('/device-access.png')" | ||||
|                                     /> | ||||
|                                 </slot> | ||||
|                             </template> | ||||
|                             <template #content> | ||||
|                                 <h3 class="card-item-content-title"> | ||||
|                                     {{ item.name }} | ||||
|                                 </h3> | ||||
|                                 <div class="desc">{{ item.description }}</div> | ||||
|                                 <a-row v-if="props.channel === 'gb28181-2016'"> | ||||
|                                     <a-col :span="12"> | ||||
|                                         {{ item.channelInfo?.name }} | ||||
|                                     </a-col> | ||||
|                                     <a-col :span="12"> | ||||
|                                         {{ item.protocolDetail.name }} | ||||
|                                     </a-col> | ||||
|                                     <a-col :span="12"> | ||||
|                                         <p | ||||
|                                             v-for="(i, idx) in item.channelInfo | ||||
|                                                 ?.addresses" | ||||
|                                             :key="`${i.address}_address${idx}`" | ||||
|                                         > | ||||
|                                             <a-badge | ||||
|                                                 :text="i.address" | ||||
|                                                 :color=" | ||||
|                                                     i.health === -1 | ||||
|                                                         ? 'red' | ||||
|                                                         : 'green' | ||||
|                                                 " | ||||
|                                             /> | ||||
|                                         </p> | ||||
|                                     </a-col> | ||||
|                                 </a-row> | ||||
|                                 <a-row v-else> | ||||
|                                     <a-col :span="24"> | ||||
|                                         <div class="subtitle"> | ||||
|                                             {{ item.protocolDetail.name }} | ||||
|                                         </div> | ||||
|                                         <p> | ||||
|                                             {{ | ||||
|                                                 item.protocolDetail.description | ||||
|                                             }} | ||||
|                                         </p> | ||||
|                                     </a-col> | ||||
|                                 </a-row> | ||||
|                             </template> | ||||
|                         </CardBox> | ||||
|                     </div> | ||||
|                 </div> | ||||
|             </a-form-item> | ||||
|         </a-form> | ||||
|     </a-modal> | ||||
| </template> | ||||
| 
 | ||||
| <script setup lang="ts"> | ||||
| import { Form, message } from 'ant-design-vue'; | ||||
| import { PropType } from 'vue'; | ||||
| import { streamMode } from '@/views/media/Device/const'; | ||||
| import DeviceApi from '@/api/media/device'; | ||||
| import { getImage } from '@/utils/comm'; | ||||
| import { gatewayType } from '@/views/media/Device/typings'; | ||||
| import { providerType } from '../const'; | ||||
| 
 | ||||
| const useForm = Form.useForm; | ||||
| 
 | ||||
| type Emits = { | ||||
|     (e: 'update:visible', data: boolean): void; | ||||
|     (e: 'update:productId', data: string): void; | ||||
|     (e: 'close'): void; | ||||
| }; | ||||
| const emit = defineEmits<Emits>(); | ||||
| 
 | ||||
| const props = defineProps({ | ||||
|     visible: { type: Boolean, default: false }, | ||||
|     productId: { type: String, default: '' }, | ||||
|     channel: { type: String, default: '' }, | ||||
| }); | ||||
| 
 | ||||
| const _vis = computed({ | ||||
|     get: () => props.visible, | ||||
|     set: (val) => emit('update:visible', val), | ||||
| }); | ||||
| 
 | ||||
| /** | ||||
|  * 获取接入网关 | ||||
|  */ | ||||
| const gatewayList = ref<gatewayType[]>([]); | ||||
| const getGatewayList = async () => { | ||||
|     const params = { | ||||
|         pageSize: 100, | ||||
|         sorts: [{ name: 'createTime', order: 'desc' }], | ||||
|         terms: [{ column: 'provider', value: props.channel }], | ||||
|     }; | ||||
|     const { result } = await DeviceApi.queryProvider(params); | ||||
|     gatewayList.value = result.data; | ||||
| }; | ||||
| 
 | ||||
| /** | ||||
|  * 点击接入网关, 获取对应配置 | ||||
|  * @param e | ||||
|  */ | ||||
| const _selectedRowKeys = ref<string[]>([]); | ||||
| const handleClick = async (e: any) => { | ||||
|     _selectedRowKeys.value = [e.id]; | ||||
|     formData.value.accessId = e.id; | ||||
|     formData.value.accessName = e.name; | ||||
|     formData.value.accessProvider = e.provider; | ||||
|     formData.value.messageProtocol = e.provider; | ||||
|     formData.value.protocolName = e.protocolDetail.name; | ||||
|     formData.value.transportProtocol = e.transport; | ||||
| 
 | ||||
|     const { result } = await DeviceApi.getConfiguration( | ||||
|         props.channel, | ||||
|         e.transport, | ||||
|     ); | ||||
|     console.log('result: ', result); | ||||
| }; | ||||
| 
 | ||||
| watch( | ||||
|     () => _vis.value, | ||||
|     (val) => { | ||||
|         if (val) { | ||||
|             getGatewayList(); | ||||
| 
 | ||||
|             formRules.value['configuration.access_pwd'][0].required = | ||||
|                 props.channel === 'gb28181-2016'; | ||||
|             validate(); | ||||
|         } else { | ||||
|             emit('close'); | ||||
|         } | ||||
|     }, | ||||
| ); | ||||
| 
 | ||||
| // 表单数据 | ||||
| const formData = ref({ | ||||
|     accessId: '', | ||||
|     accessName: '', | ||||
|     accessProvider: '', | ||||
|     configuration: { | ||||
|         access_pwd: '', | ||||
|         stream_mode: 'UDP', | ||||
|     }, | ||||
|     deviceType: 'device', | ||||
|     messageProtocol: '', | ||||
|     name: '', | ||||
|     protocolName: '', | ||||
|     transportProtocol: '', | ||||
| }); | ||||
| 
 | ||||
| // 验证规则 | ||||
| const formRules = ref({ | ||||
|     name: [{ required: true, message: '请输入产品名称' }], | ||||
|     'configuration.access_pwd': [{ required: true, message: '请输入接入密码' }], | ||||
|     accessId: [{ required: true, message: '请选择接入网关' }], | ||||
| }); | ||||
| 
 | ||||
| const { resetFields, validate, validateInfos, clearValidate } = useForm( | ||||
|     formData.value, | ||||
|     formRules.value, | ||||
| ); | ||||
| 
 | ||||
| /** | ||||
|  * 提交 | ||||
|  */ | ||||
| const btnLoading = ref(false); | ||||
| const handleOk = () => { | ||||
|     // console.log('formData.value: ', formData.value); | ||||
|     validate() | ||||
|         .then(async () => { | ||||
|             btnLoading.value = true; | ||||
|             const res = await DeviceApi.saveProduct(formData.value); | ||||
|             if (res.success) { | ||||
|                 emit('update:productId', res.result.id); | ||||
|                 const deployResp = await DeviceApi.deployProductById( | ||||
|                     res.result.id, | ||||
|                 ); | ||||
|                 if (deployResp.success) { | ||||
|                     message.success('操作成功'); | ||||
|                     handleCancel(); | ||||
|                 } | ||||
|             } | ||||
|             btnLoading.value = false; | ||||
|         }) | ||||
|         .catch((err) => { | ||||
|             console.log('err: ', err); | ||||
|         }); | ||||
| }; | ||||
| 
 | ||||
| const handleCancel = () => { | ||||
|     _vis.value = false; | ||||
|     resetFields(); | ||||
| }; | ||||
| </script> | ||||
| 
 | ||||
| <style lang="less" scoped> | ||||
| .gateway-box { | ||||
|     max-height: 450px; | ||||
|     overflow-y: auto; | ||||
|     text-align: center; | ||||
|     .gateway-item { | ||||
|         padding: 16px; | ||||
|         .card-item-content-title, | ||||
|         .desc, | ||||
|         .subtitle { | ||||
|             overflow: hidden; | ||||
|             white-space: nowrap; | ||||
|             text-overflow: ellipsis; | ||||
|         } | ||||
|         .desc, | ||||
|         .subtitle { | ||||
|             margin-top: 10px; | ||||
|             color: #666; | ||||
|             font-weight: 400; | ||||
|             font-size: 12px; | ||||
|         } | ||||
|     } | ||||
| } | ||||
| </style> | ||||
|  | @ -13,82 +13,18 @@ | |||
|                                 layout="horizontal" | ||||
|                                 :options="PROVIDER_OPTIONS" | ||||
|                                 :checkStyle="true" | ||||
|                                 :disabled="!!formData.id" | ||||
|                                 :disabled="!!route.query.id" | ||||
|                                 v-model="formData.channel" | ||||
|                             /> | ||||
|                         </a-form-item> | ||||
|                         <a-row :gutter="24"> | ||||
|                             <a-col :span="8"> | ||||
|                                 <!-- <div class="upload-image-warp-logo"> | ||||
|                                     <div class="upload-image-border-logo"> | ||||
|                                         <a-upload | ||||
|                                             name="file" | ||||
|                                             :action="FILE_UPLOAD" | ||||
|                                             :headers="{ | ||||
|                                                 [TOKEN_KEY]: | ||||
|                                                     LocalStore.get(TOKEN_KEY), | ||||
|                                             }" | ||||
|                                             :showUploadList="false" | ||||
|                                             accept="image/jpeg', 'image/png" | ||||
|                                         > | ||||
|                                             <div | ||||
|                                                 class="upload-image-content-logo" | ||||
|                                             > | ||||
|                                                 <div | ||||
|                                                     class="loading-logo" | ||||
|                                                     v-if="form.logoLoading" | ||||
|                                                 > | ||||
|                                                     <LoadingOutlined | ||||
|                                                         style="font-size: 28px" | ||||
|                                 <JUpload | ||||
|                                     v-model:modelValue="formData.photoUrl" | ||||
|                                     :bgImage="formData.photoUrl" | ||||
|                                 /> | ||||
|                                                 </div> | ||||
|                                                 <div | ||||
|                                                     class="upload-image" | ||||
|                                                     style="height: 100%" | ||||
|                                                     v-if="formValue.logo" | ||||
|                                                     :style=" | ||||
|                                                         formValue.logo | ||||
|                                                             ? `background-image: url(${formValue.logo});` | ||||
|                                                             : '' | ||||
|                                                     " | ||||
|                                                 ></div> | ||||
|                                                 <div | ||||
|                                                     v-if="formValue.logo" | ||||
|                                                     class="upload-image-mask" | ||||
|                                                 > | ||||
|                                                     点击修改 | ||||
|                                                 </div> | ||||
|                                                 <div v-else> | ||||
|                                                     <div | ||||
|                                                         v-if="form.logoLoading" | ||||
|                                                     > | ||||
|                                                         <LoadingOutlined | ||||
|                                                             style=" | ||||
|                                                                 font-size: 28px; | ||||
|                                                             " | ||||
|                                                         /> | ||||
|                                                     </div> | ||||
|                                                     <div v-else> | ||||
|                                                         <PlusOutlined | ||||
|                                                             style=" | ||||
|                                                                 font-size: 28px; | ||||
|                                                             " | ||||
|                                                         /> | ||||
|                                                     </div> | ||||
|                                                 </div> | ||||
|                                             </div> | ||||
|                                         </a-upload> | ||||
|                                         <div v-if="form.logoLoading"> | ||||
|                                             <div class="upload-loading-mask"> | ||||
|                                                 <LoadingOutlined | ||||
|                                                     style="font-size: 28px" | ||||
|                                                 /> | ||||
|                                             </div> | ||||
|                                         </div> | ||||
|                                     </div> | ||||
|                                 </div> --> | ||||
|                             </a-col> | ||||
|                             <a-col :span="12"> | ||||
|                             <a-col :span="16"> | ||||
|                                 <a-form-item | ||||
|                                     label="ID" | ||||
|                                     v-bind="validateInfos.id" | ||||
|  | @ -96,6 +32,7 @@ | |||
|                                     <a-input | ||||
|                                         v-model:value="formData.id" | ||||
|                                         placeholder="请输入" | ||||
|                                         :disabled="!!route.query.id" | ||||
|                                     /> | ||||
|                                 </a-form-item> | ||||
|                                 <a-form-item | ||||
|  | @ -113,31 +50,78 @@ | |||
|                             label="所属产品" | ||||
|                             v-bind="validateInfos.productId" | ||||
|                         > | ||||
|                             <div> | ||||
|                             <a-row :gutter="[0, 10]"> | ||||
|                                 <a-col :span="22"> | ||||
|                                     <a-select | ||||
|                                         v-model:value="formData.productId" | ||||
|                                         placeholder="请选择所属产品" | ||||
|                                         :disabled="!!route.query.id" | ||||
|                                     > | ||||
|                                     <!-- <a-select-option | ||||
|                                     v-for="(item, index) in NOTICE_METHOD" | ||||
|                                         <a-select-option | ||||
|                                             v-for="(item, index) in productList" | ||||
|                                             :key="index" | ||||
|                                     :value="item.value" | ||||
|                                             :value="item.id" | ||||
|                                         > | ||||
|                                     {{ item.label }} | ||||
|                                 </a-select-option> --> | ||||
|                                             {{ item.name }} | ||||
|                                         </a-select-option> | ||||
|                                     </a-select> | ||||
|                                 <AIcon type="PlusCircleOutlined" /> | ||||
|                             </div> | ||||
|                                 </a-col> | ||||
|                                 <a-col :span="2"> | ||||
|                                     <a-button | ||||
|                                         type="link" | ||||
|                                         @click="saveProductVis = true" | ||||
|                                     > | ||||
|                                         <AIcon type="PlusOutlined" /> | ||||
|                                     </a-button> | ||||
|                                 </a-col> | ||||
|                             </a-row> | ||||
|                         </a-form-item> | ||||
|                         <a-form-item | ||||
|                             label="接入密码" | ||||
|                             v-bind="validateInfos['others.access_pwd']" | ||||
|                             v-if="formData.channel === 'gb28181-2016'" | ||||
|                         > | ||||
|                             <a-input-password | ||||
|                                 v-model:value="formData.others.access_pwd" | ||||
|                                 placeholder="请输入接入密码" | ||||
|                             /> | ||||
|                         </a-form-item> | ||||
|                         <template v-if="!!route.query.id"> | ||||
|                             <a-form-item | ||||
|                                 label="流传输模式" | ||||
|                                 v-bind="validateInfos.streamMode" | ||||
|                             > | ||||
|                                 <a-radio-group | ||||
|                                     button-style="solid" | ||||
|                                     v-model:value="formData.streamMode" | ||||
|                                 > | ||||
|                                     <a-radio-button value="UDP"> | ||||
|                                         UDP | ||||
|                                     </a-radio-button> | ||||
|                                     <a-radio-button value="TCP_PASSIVE"> | ||||
|                                         TCP被动 | ||||
|                                     </a-radio-button> | ||||
|                                 </a-radio-group> | ||||
|                             </a-form-item> | ||||
|                             <a-form-item label="设备厂商"> | ||||
|                                 <a-input | ||||
|                                     v-model:value="formData.manufacturer" | ||||
|                                     placeholder="请输入设备厂商" | ||||
|                                 /> | ||||
|                             </a-form-item> | ||||
|                             <a-form-item label="设备型号"> | ||||
|                                 <a-input | ||||
|                                     v-model:value="formData.model" | ||||
|                                     placeholder="请输入设备型号" | ||||
|                                 /> | ||||
|                             </a-form-item> | ||||
|                             <a-form-item label="固件版本"> | ||||
|                                 <a-input | ||||
|                                     v-model:value="formData.firmware" | ||||
|                                     placeholder="请输入固件版本" | ||||
|                                 /> | ||||
|                             </a-form-item> | ||||
|                         </template> | ||||
| 
 | ||||
|                         <a-form-item label="说明"> | ||||
|                             <a-textarea | ||||
|  | @ -250,6 +234,13 @@ | |||
|                 </a-col> | ||||
|             </a-row> | ||||
|         </a-card> | ||||
| 
 | ||||
|         <SaveProduct | ||||
|             v-model:visible="saveProductVis" | ||||
|             v-model:productId="formData.productId" | ||||
|             :channel="formData.channel" | ||||
|             @close="getProductList" | ||||
|         /> | ||||
|     </div> | ||||
| </template> | ||||
| 
 | ||||
|  | @ -258,12 +249,11 @@ import { getImage } from '@/utils/comm'; | |||
| import { Form } from 'ant-design-vue'; | ||||
| import { message } from 'ant-design-vue'; | ||||
| 
 | ||||
| import templateApi from '@/api/notice/template'; | ||||
| import DeviceApi from '@/api/media/device'; | ||||
| 
 | ||||
| import { FILE_UPLOAD } from '@/api/comm'; | ||||
| import { LocalStore } from '@/utils/comm'; | ||||
| import { TOKEN_KEY } from '@/utils/variable'; | ||||
| import { PROVIDER_OPTIONS } from '@/views/media/Device/const'; | ||||
| import type { ProductType } from '@/views/media/Device/typings'; | ||||
| import SaveProduct from './SaveProduct.vue'; | ||||
| 
 | ||||
| const router = useRouter(); | ||||
| const route = useRoute(); | ||||
|  | @ -274,18 +264,26 @@ const formData = ref({ | |||
|     id: '', | ||||
|     name: '', | ||||
|     channel: 'gb28181-2016', | ||||
|     photoUrl: '', | ||||
|     photoUrl: getImage('/device-media.png'), | ||||
|     productId: '', | ||||
|     others: { | ||||
|         access_pwd: '', | ||||
|     }, | ||||
|     description: '', | ||||
|     // 编辑字段 | ||||
|     streamMode: 'UDP', | ||||
|     manufacturer: '', | ||||
|     model: '', | ||||
|     firmware: '', | ||||
| }); | ||||
| 
 | ||||
| // 验证规则 | ||||
| const formRules = ref({ | ||||
|     id: [ | ||||
|         { required: true, message: '请输入ID' }, | ||||
|         { | ||||
|             required: true, | ||||
|             message: '请输入ID', | ||||
|         }, | ||||
|         { max: 64, message: '最多输入64个字符' }, | ||||
|         { | ||||
|             pattern: /^[a-zA-Z0-9_\-]+$/, | ||||
|  | @ -300,8 +298,20 @@ const formRules = ref({ | |||
|     channel: [{ required: true, message: '请选择接入方式' }], | ||||
|     'others.access_pwd': [{ required: true, message: '请输入接入密码' }], | ||||
|     description: [{ max: 200, message: '最多可输入200个字符' }], | ||||
|     streamMode: [{ required: true, message: '请选择流传输模式' }], | ||||
| }); | ||||
| 
 | ||||
| watch( | ||||
|     () => formData.value.channel, | ||||
|     (val) => { | ||||
|         formRules.value['id'][0].required = val === 'gb28181-2016'; | ||||
|         formRules.value['others.access_pwd'][0].required = | ||||
|             val === 'gb28181-2016'; | ||||
|         validate(); | ||||
|         getProductList(); | ||||
|     }, | ||||
| ); | ||||
| 
 | ||||
| const { resetFields, validate, validateInfos, clearValidate } = useForm( | ||||
|     formData.value, | ||||
|     formRules.value, | ||||
|  | @ -309,36 +319,64 @@ const { resetFields, validate, validateInfos, clearValidate } = useForm( | |||
| 
 | ||||
| const clearValid = () => { | ||||
|     setTimeout(() => { | ||||
|         formData.value.variableDefinitions = []; | ||||
|         clearValidate(); | ||||
|     }, 200); | ||||
| }; | ||||
| 
 | ||||
| /** | ||||
|  * 获取所属产品 | ||||
|  */ | ||||
| const productList = ref<ProductType[]>([]); | ||||
| const getProductList = async () => { | ||||
|     // console.log('formData.productId: ', formData.value.productId); | ||||
|     const params = { | ||||
|         paging: false, | ||||
|         sorts: [{ name: 'createTime', order: 'desc' }], | ||||
|         terms: [ | ||||
|             { column: 'accessProvider', value: formData.value.channel }, | ||||
|             { column: 'state', value: 1 }, | ||||
|         ], | ||||
|     }; | ||||
|     const { result } = await DeviceApi.queryProductList(params); | ||||
|     productList.value = result; | ||||
| }; | ||||
| getProductList(); | ||||
| 
 | ||||
| /** | ||||
|  * 新增产品 | ||||
|  */ | ||||
| const saveProductVis = ref(false); | ||||
| 
 | ||||
| /** | ||||
|  * 获取详情 | ||||
|  */ | ||||
| const getDetail = async () => { | ||||
|     const res = await templateApi.detail(route.params.id as string); | ||||
|     const res = await DeviceApi.detail(route.query.id as string); | ||||
|     // console.log('res: ', res); | ||||
|     formData.value = res.result; | ||||
|     // console.log('formData.value: ', formData.value); | ||||
|     formData.value.channel = res.result.provider; | ||||
|     console.log('formData.value: ', formData.value); | ||||
|     // validate(); | ||||
| }; | ||||
| // getDetail(); | ||||
| 
 | ||||
| onMounted(() => { | ||||
|     getDetail(); | ||||
| }); | ||||
| 
 | ||||
| /** | ||||
|  * 表单提交 | ||||
|  */ | ||||
| const btnLoading = ref<boolean>(false); | ||||
| const handleSubmit = () => { | ||||
|     // console.log('formData.value: ', formData.value); | ||||
|     console.log('formData.value: ', formData.value); | ||||
|     validate() | ||||
|         .then(async () => { | ||||
|             btnLoading.value = true; | ||||
|             let res; | ||||
|             if (!formData.value.id) { | ||||
|                 res = await templateApi.save(formData.value); | ||||
|             if (!route.query.id) { | ||||
|                 res = await DeviceApi.save(formData.value); | ||||
|             } else { | ||||
|                 res = await templateApi.update(formData.value); | ||||
|                 res = await DeviceApi.update(formData.value); | ||||
|             } | ||||
|             // console.log('res: ', res); | ||||
|             if (res?.success) { | ||||
|  | @ -360,70 +398,5 @@ const handleSubmit = () => { | |||
| .page-container { | ||||
|     background: #f0f2f5; | ||||
|     padding: 24px; | ||||
|     .upload-image-warp-logo { | ||||
|         display: flex; | ||||
|         justify-content: flex-start; | ||||
|         .upload-image-border-logo { | ||||
|             position: relative; | ||||
|             overflow: hidden; | ||||
|             border: 1px dashed #d9d9d9; | ||||
|             transition: all 0.3s; | ||||
|             width: 160px; | ||||
|             height: 150px; | ||||
|             &:hover { | ||||
|                 border: 1px dashed #1890ff; | ||||
|                 display: flex; | ||||
|             } | ||||
|             .upload-image-content-logo { | ||||
|                 align-items: center; | ||||
|                 justify-content: center; | ||||
|                 position: relative; | ||||
|                 display: flex; | ||||
|                 flex-direction: column; | ||||
|                 width: 160px; | ||||
|                 height: 150px; | ||||
|                 padding: 8px; | ||||
|                 background-color: rgba(0, 0, 0, 0.06); | ||||
|                 cursor: pointer; | ||||
|                 .loading-logo { | ||||
|                     position: absolute; | ||||
|                     top: 50%; | ||||
|                 } | ||||
|                 .loading-icon { | ||||
|                     position: absolute; | ||||
|                 } | ||||
|                 .upload-image { | ||||
|                     width: 100%; | ||||
|                     height: 100%; | ||||
|                     background-repeat: no-repeat; | ||||
|                     background-position: 50%; | ||||
|                     background-size: cover; | ||||
|                 } | ||||
|                 .upload-image-icon { | ||||
|                     width: 100%; | ||||
|                     height: 100%; | ||||
|                     background-repeat: no-repeat; | ||||
|                     background-position: 50%; | ||||
|                     background-size: inherit; | ||||
|                 } | ||||
|                 .upload-image-mask { | ||||
|                     align-items: center; | ||||
|                     justify-content: center; | ||||
|                     position: absolute; | ||||
|                     top: 0; | ||||
|                     left: 0; | ||||
|                     display: none; | ||||
|                     width: 100%; | ||||
|                     height: 100%; | ||||
|                     color: #fff; | ||||
|                     font-size: 16px; | ||||
|                     background-color: rgba(0, 0, 0, 0.35); | ||||
|                 } | ||||
|                 &:hover .upload-image-mask { | ||||
|                     display: flex; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
| </style> | ||||
|  |  | |||
|  | @ -2,3 +2,12 @@ export const PROVIDER_OPTIONS = [ | |||
|     { label: '固定地址', value: 'fixed-media' }, | ||||
|     { label: 'GB/T28181', value: 'gb28181-2016' }, | ||||
| ] | ||||
| export const streamMode = [ | ||||
|     { label: 'UDP', value: 'UDP' }, | ||||
|     { label: 'TCP被动', value: 'TCP_PASSIVE' }, | ||||
| ] | ||||
| 
 | ||||
| export const providerType = { | ||||
|     'gb28181-2016': 'GB/T28181', | ||||
|     'fixed-media': '固定地址', | ||||
| }; | ||||
|  | @ -135,11 +135,7 @@ import type { ActionsType } from '@/components/Table/index.vue'; | |||
| import { message } from 'ant-design-vue'; | ||||
| import { getImage } from '@/utils/comm'; | ||||
| import { PROVIDER_OPTIONS } from '@/views/media/Device/const'; | ||||
| 
 | ||||
| const providerType = { | ||||
|     'gb28181-2016': 'GB/T28181', | ||||
|     'fixed-media': '固定地址', | ||||
| }; | ||||
| import { providerType } from './const'; | ||||
| 
 | ||||
| const router = useRouter(); | ||||
| 
 | ||||
|  |  | |||
|  | @ -22,3 +22,64 @@ export type DeviceItem = { | |||
|     streamMode: string; | ||||
|     transport: string; | ||||
| } & BaseItem; | ||||
| 
 | ||||
| export type ProductType = { | ||||
|     accessId: string; | ||||
|     accessName: string; | ||||
|     accessProvider: string; | ||||
|     createTime: number; | ||||
|     creatorId: string; | ||||
|     deviceType: { | ||||
|         text: string; | ||||
|         value: string; | ||||
|     }; | ||||
|     id: string; | ||||
|     messageProtocol: string; | ||||
|     metadata: string; | ||||
|     modifierId: string; | ||||
|     modifyTime: number; | ||||
|     name: string; | ||||
|     protocolName: string; | ||||
|     state: number; | ||||
|     transportProtocol: string; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| type addressesType = { | ||||
|     address: string; | ||||
|     bad: boolean; | ||||
|     disabled: boolean; | ||||
|     health: number; | ||||
|     ok: boolean; | ||||
| } | ||||
| export type gatewayType = { | ||||
|     channel: string; | ||||
|     channelId: string; | ||||
|     channelInfo: { | ||||
|         id: string; | ||||
|         name: string; | ||||
|         addresses: addressesType[]; | ||||
|     }; | ||||
|     id: string; | ||||
|     name: string; | ||||
|     protocol: string; | ||||
|     protocolDetail: { | ||||
|         id: string; | ||||
|         name: string; | ||||
|         description?: string; | ||||
|     }; | ||||
|     provider: string; | ||||
|     state: { | ||||
|         text: string; | ||||
|         value: string; | ||||
|     }; | ||||
|     transport: string; | ||||
|     transportDetail: { | ||||
|         id: string; | ||||
|         name: string; | ||||
|         metadata: string; | ||||
|         features: string[]; | ||||
|         routes: string[]; | ||||
|     }; | ||||
|     description?: string; | ||||
| } | ||||
		Loading…
	
		Reference in New Issue