Merge branch 'dev' of github.com:jetlinks/jetlinks-ui-vue into dev
This commit is contained in:
		
						commit
						b706ff1c09
					
				|  | @ -0,0 +1,13 @@ | ||||||
|  | import server from '@/utils/request' | ||||||
|  | 
 | ||||||
|  | export default { | ||||||
|  |     // 列表
 | ||||||
|  |     list: (data: any) => server.post(`/media/device/_query/`, data), | ||||||
|  |     // 详情
 | ||||||
|  |     detail: (id: string): any => server.get(`/media/device/${id}`), | ||||||
|  |     // 新增
 | ||||||
|  |     save: (data: any) => server.post(`/media/device/${data.channel}`, data), | ||||||
|  |     // 修改
 | ||||||
|  |     update: (data: any) => server.put(`/media/device/${data.channel}/${data.id}`, data), | ||||||
|  |     del: (id: string) => server.remove(`/media/device/${id}`), | ||||||
|  | } | ||||||
|  | @ -45,7 +45,10 @@ const iconKeys = [ | ||||||
|     'InfoCircleOutlined', |     'InfoCircleOutlined', | ||||||
|     'SearchOutlined', |     'SearchOutlined', | ||||||
|     'EllipsisOutlined', |     'EllipsisOutlined', | ||||||
|     'ClockCircleOutlined' |     'ClockCircleOutlined', | ||||||
|  |     'PartitionOutlined', | ||||||
|  |     'ShareAltOutlined', | ||||||
|  |     'playCircleOutlined', | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
| const Icon = (props: {type: string}) => { | const Icon = (props: {type: string}) => { | ||||||
|  |  | ||||||
|  | @ -4,14 +4,14 @@ | ||||||
|       <a-popconfirm v-bind="popConfirm" @confirm="conform" :disabled="!isPermission || props.disabled"> |       <a-popconfirm v-bind="popConfirm" @confirm="conform" :disabled="!isPermission || props.disabled"> | ||||||
|         <a-tooltip v-if="tooltip" v-bind="tooltip"> |         <a-tooltip v-if="tooltip" v-bind="tooltip"> | ||||||
|           <slot v-if="noButton"></slot> |           <slot v-if="noButton"></slot> | ||||||
|           <a-button v-else v-bind="_buttonProps" :disabled="_isPermission" @click="handleClick"> |           <a-button v-else v-bind="_buttonProps" :disabled="_isPermission" > | ||||||
|             <slot></slot> |             <slot></slot> | ||||||
|             <template #icon> |             <template #icon> | ||||||
|               <slot name="icon"></slot> |               <slot name="icon"></slot> | ||||||
|             </template> |             </template> | ||||||
|           </a-button> |           </a-button> | ||||||
|         </a-tooltip> |         </a-tooltip> | ||||||
|         <a-button v-else v-bind="_buttonProps" :disabled="_isPermission" @click="handleClick"> |         <a-button v-else v-bind="_buttonProps" :disabled="_isPermission" > | ||||||
|           <slot></slot> |           <slot></slot> | ||||||
|           <template #icon> |           <template #icon> | ||||||
|             <slot name="icon"></slot> |             <slot name="icon"></slot> | ||||||
|  | @ -22,7 +22,7 @@ | ||||||
|     <template v-else-if="tooltip"> |     <template v-else-if="tooltip"> | ||||||
|       <a-tooltip v-bind="tooltip"> |       <a-tooltip v-bind="tooltip"> | ||||||
|         <slot v-if="noButton"></slot> |         <slot v-if="noButton"></slot> | ||||||
|         <a-button v-else v-bind="_buttonProps" :disabled="_isPermission" @click="handleClick"> |         <a-button v-else v-bind="_buttonProps" :disabled="_isPermission" > | ||||||
|           <slot></slot> |           <slot></slot> | ||||||
|           <template #icon> |           <template #icon> | ||||||
|             <slot name="icon"></slot> |             <slot name="icon"></slot> | ||||||
|  | @ -32,7 +32,7 @@ | ||||||
|     </template> |     </template> | ||||||
|     <template v-else> |     <template v-else> | ||||||
|       <slot v-if="noButton"></slot> |       <slot v-if="noButton"></slot> | ||||||
|       <a-button v-else v-bind="_buttonProps" :disabled="_isPermission" @click="handleClick"> |       <a-button v-else v-bind="_buttonProps" :disabled="_isPermission" > | ||||||
|         <slot></slot> |         <slot></slot> | ||||||
|         <template #icon> |         <template #icon> | ||||||
|           <slot name="icon"></slot> |           <slot name="icon"></slot> | ||||||
|  | @ -42,7 +42,7 @@ | ||||||
|   </template> |   </template> | ||||||
|   <a-tooltip v-else title="没有权限"> |   <a-tooltip v-else title="没有权限"> | ||||||
|     <slot v-if="noButton"></slot> |     <slot v-if="noButton"></slot> | ||||||
|     <a-button v-else v-bind="_buttonProps" :disabled="_isPermission" @click="handleClick"> |     <a-button v-else v-bind="_buttonProps" :disabled="_isPermission" > | ||||||
|       <slot></slot> |       <slot></slot> | ||||||
|       <template #icon> |       <template #icon> | ||||||
|         <slot name="icon"></slot> |         <slot name="icon"></slot> | ||||||
|  | @ -56,11 +56,11 @@ import { TooltipProps, PopconfirmProps } from 'ant-design-vue/es' | ||||||
| import { buttonProps } from 'ant-design-vue/es/button/button' | import { buttonProps } from 'ant-design-vue/es/button/button' | ||||||
| import { usePermissionStore } from '@/store/permission'; | import { usePermissionStore } from '@/store/permission'; | ||||||
| 
 | 
 | ||||||
| interface PermissionButtonEmits { | // interface PermissionButtonEmits { | ||||||
|   (e: 'click', data: MouseEvent): void; | //   (e: 'click', data: MouseEvent): void; | ||||||
| } | // } | ||||||
| 
 | 
 | ||||||
| const emits = defineEmits<PermissionButtonEmits>() | // const emits = defineEmits<PermissionButtonEmits>() | ||||||
| 
 | 
 | ||||||
| // interface PermissionButtonProps extends ButtonProps { | // interface PermissionButtonProps extends ButtonProps { | ||||||
| //   tooltip?: TooltipProps; | //   tooltip?: TooltipProps; | ||||||
|  | @ -105,9 +105,6 @@ const _isPermission = computed(() => | ||||||
|       : false |       : false | ||||||
|     : true |     : true | ||||||
| ) | ) | ||||||
| const handleClick = (e: MouseEvent) => { |  | ||||||
|   emits('click', e) |  | ||||||
| } |  | ||||||
| 
 | 
 | ||||||
| const conform = (e: MouseEvent) => { | const conform = (e: MouseEvent) => { | ||||||
|   props.popConfirm?.onConfirm?.(e) |   props.popConfirm?.onConfirm?.(e) | ||||||
|  |  | ||||||
|  | @ -0,0 +1,7 @@ | ||||||
|  | <template> | ||||||
|  |     <div class="page-container">save</div> | ||||||
|  | </template> | ||||||
|  | 
 | ||||||
|  | <script setup lang="ts"></script> | ||||||
|  | 
 | ||||||
|  | <style lang="less" scoped></style> | ||||||
|  | @ -0,0 +1,334 @@ | ||||||
|  | <template> | ||||||
|  |     <div class="page-container"> | ||||||
|  |         <Search | ||||||
|  |             :columns="columns" | ||||||
|  |             target="notice-config" | ||||||
|  |             @search="handleSearch" | ||||||
|  |         /> | ||||||
|  |         <JTable | ||||||
|  |             ref="listRef" | ||||||
|  |             :columns="columns" | ||||||
|  |             :request="DeviceApi.list" | ||||||
|  |             :defaultParams="{ | ||||||
|  |                 sorts: [{ name: 'createTime', order: 'desc' }], | ||||||
|  |             }" | ||||||
|  |             :params="params" | ||||||
|  |         > | ||||||
|  |             <template #headerTitle> | ||||||
|  |                 <a-button type="primary" @click="handleAdd"> 新增 </a-button> | ||||||
|  |             </template> | ||||||
|  |             <template #card="slotProps"> | ||||||
|  |                 <CardBox | ||||||
|  |                     :value="slotProps" | ||||||
|  |                     :actions="getActions(slotProps, 'card')" | ||||||
|  |                     v-bind="slotProps" | ||||||
|  |                     :showStatus="true" | ||||||
|  |                     :status=" | ||||||
|  |                         slotProps.state.value === 'online' ? 'success' : 'error' | ||||||
|  |                     " | ||||||
|  |                     :statusText="slotProps.state.text" | ||||||
|  |                     :statusNames="{ success: 'success', error: 'error' }" | ||||||
|  |                 > | ||||||
|  |                     <template #img> | ||||||
|  |                         <slot name="img"> | ||||||
|  |                             <img :src="getImage('/device-media.png')" /> | ||||||
|  |                         </slot> | ||||||
|  |                     </template> | ||||||
|  |                     <template #content> | ||||||
|  |                         <h3 class="card-item-content-title"> | ||||||
|  |                             {{ slotProps.name }} | ||||||
|  |                         </h3> | ||||||
|  |                         <a-row> | ||||||
|  |                             <a-col :span="12"> | ||||||
|  |                                 <div class="card-item-content-text">厂商</div> | ||||||
|  |                                 <div>{{ slotProps.manufacturer }}</div> | ||||||
|  |                             </a-col> | ||||||
|  |                             <a-col :span="12"> | ||||||
|  |                                 <div class="card-item-content-text"> | ||||||
|  |                                     通道数量 | ||||||
|  |                                 </div> | ||||||
|  |                                 <div>{{ slotProps.channelNumber }}</div> | ||||||
|  |                             </a-col> | ||||||
|  |                             <a-col :span="12"> | ||||||
|  |                                 <div class="card-item-content-text">型号</div> | ||||||
|  |                                 <div>{{ slotProps.model }}</div> | ||||||
|  |                             </a-col> | ||||||
|  |                             <a-col :span="12"> | ||||||
|  |                                 <div class="card-item-content-text"> | ||||||
|  |                                     接入方式 | ||||||
|  |                                 </div> | ||||||
|  |                                 <div> | ||||||
|  |                                     {{ providerType[slotProps.provider] }} | ||||||
|  |                                 </div> | ||||||
|  |                             </a-col> | ||||||
|  |                         </a-row> | ||||||
|  |                     </template> | ||||||
|  |                     <template #actions="item"> | ||||||
|  |                         <a-tooltip | ||||||
|  |                             v-bind="item.tooltip" | ||||||
|  |                             :title="item.disabled && item.tooltip.title" | ||||||
|  |                         > | ||||||
|  |                             <a-popconfirm | ||||||
|  |                                 v-if="item.popConfirm" | ||||||
|  |                                 v-bind="item.popConfirm" | ||||||
|  |                                 :disabled="item.disabled" | ||||||
|  |                             > | ||||||
|  |                                 <a-button :disabled="item.disabled"> | ||||||
|  |                                     <AIcon type="DeleteOutlined" /> | ||||||
|  |                                 </a-button> | ||||||
|  |                             </a-popconfirm> | ||||||
|  |                             <template v-else> | ||||||
|  |                                 <a-button | ||||||
|  |                                     :disabled="item.disabled" | ||||||
|  |                                     @click="item.onClick" | ||||||
|  |                                 > | ||||||
|  |                                     <AIcon :type="item.icon" /> | ||||||
|  |                                     <span>{{ item.text }}</span> | ||||||
|  |                                 </a-button> | ||||||
|  |                             </template> | ||||||
|  |                         </a-tooltip> | ||||||
|  |                     </template> | ||||||
|  |                 </CardBox> | ||||||
|  |             </template> | ||||||
|  |             <template #action="slotProps"> | ||||||
|  |                 <a-space :size="16"> | ||||||
|  |                     <a-tooltip | ||||||
|  |                         v-for="i in getActions(slotProps, 'table')" | ||||||
|  |                         :key="i.key" | ||||||
|  |                         v-bind="i.tooltip" | ||||||
|  |                     > | ||||||
|  |                         <a-popconfirm | ||||||
|  |                             v-if="i.popConfirm" | ||||||
|  |                             v-bind="i.popConfirm" | ||||||
|  |                             :disabled="i.disabled" | ||||||
|  |                         > | ||||||
|  |                             <a-button | ||||||
|  |                                 :disabled="i.disabled" | ||||||
|  |                                 style="padding: 0" | ||||||
|  |                                 type="link" | ||||||
|  |                                 ><AIcon :type="i.icon" | ||||||
|  |                             /></a-button> | ||||||
|  |                         </a-popconfirm> | ||||||
|  |                         <a-button | ||||||
|  |                             style="padding: 0" | ||||||
|  |                             type="link" | ||||||
|  |                             v-else | ||||||
|  |                             @click="i.onClick && i.onClick(slotProps)" | ||||||
|  |                         > | ||||||
|  |                             <a-button | ||||||
|  |                                 :disabled="i.disabled" | ||||||
|  |                                 style="padding: 0" | ||||||
|  |                                 type="link" | ||||||
|  |                                 ><AIcon :type="i.icon" | ||||||
|  |                             /></a-button> | ||||||
|  |                         </a-button> | ||||||
|  |                     </a-tooltip> | ||||||
|  |                 </a-space> | ||||||
|  |             </template> | ||||||
|  |         </JTable> | ||||||
|  |     </div> | ||||||
|  | </template> | ||||||
|  | 
 | ||||||
|  | <script setup lang="ts"> | ||||||
|  | import DeviceApi from '@/api/media/device'; | ||||||
|  | import type { ActionsType } from '@/components/Table/index.vue'; | ||||||
|  | import { message } from 'ant-design-vue'; | ||||||
|  | import { getImage } from '@/utils/comm'; | ||||||
|  | 
 | ||||||
|  | const providerType = { | ||||||
|  |     'gb28181-2016': 'GB/T28181', | ||||||
|  |     'fixed-media': '固定地址', | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | const router = useRouter(); | ||||||
|  | 
 | ||||||
|  | const listRef = ref<Record<string, any>>({}); | ||||||
|  | const params = ref<Record<string, any>>({}); | ||||||
|  | 
 | ||||||
|  | const columns = [ | ||||||
|  |     { | ||||||
|  |         title: 'ID', | ||||||
|  |         dataIndex: 'id', | ||||||
|  |         key: 'id', | ||||||
|  |         search: { | ||||||
|  |             type: 'string', | ||||||
|  |         }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         title: '名称', | ||||||
|  |         dataIndex: 'name', | ||||||
|  |         key: 'name', | ||||||
|  |         search: { | ||||||
|  |             type: 'string', | ||||||
|  |         }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         title: '接入方式', | ||||||
|  |         dataIndex: 'type', | ||||||
|  |         key: 'type', | ||||||
|  |         scopedSlots: true, | ||||||
|  |         search: { | ||||||
|  |             type: 'select', | ||||||
|  |             options: [ | ||||||
|  |                 { label: '固定地址', value: 'fixed-media' }, | ||||||
|  |                 { label: 'GB/T28181', value: 'gb28181-2016' }, | ||||||
|  |             ], | ||||||
|  |             handleValue: (v: any) => { | ||||||
|  |                 return '123'; | ||||||
|  |             }, | ||||||
|  |         }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         title: '通道数量', | ||||||
|  |         dataIndex: 'channelNumber', | ||||||
|  |         key: 'channelNumber', | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         title: '厂商', | ||||||
|  |         dataIndex: 'manufacturer', | ||||||
|  |         key: 'manufacturer', | ||||||
|  |         search: { | ||||||
|  |             type: 'string', | ||||||
|  |         }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         title: '产品名称', | ||||||
|  |         dataIndex: 'productId', | ||||||
|  |         key: 'productId', | ||||||
|  |         scopedSlots: true, | ||||||
|  |         search: { | ||||||
|  |             type: 'select', | ||||||
|  |             options: [ | ||||||
|  |                 { label: '固定地址', value: 'fixed-media' }, | ||||||
|  |                 { label: 'GB/T28181', value: 'gb28181-2016' }, | ||||||
|  |             ], | ||||||
|  |             handleValue: (v: any) => { | ||||||
|  |                 return '123'; | ||||||
|  |             }, | ||||||
|  |         }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         title: '状态', | ||||||
|  |         dataIndex: 'state', | ||||||
|  |         key: 'state', | ||||||
|  |         scopedSlots: true, | ||||||
|  |         search: { | ||||||
|  |             type: 'select', | ||||||
|  |             options: [ | ||||||
|  |                 { label: '禁用', value: 'notActive' }, | ||||||
|  |                 { label: '离线', value: 'offline' }, | ||||||
|  |                 { label: '在线', value: 'online' }, | ||||||
|  |             ], | ||||||
|  |             handleValue: (v: any) => { | ||||||
|  |                 return '123'; | ||||||
|  |             }, | ||||||
|  |         }, | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |         title: '操作', | ||||||
|  |         key: 'action', | ||||||
|  |         fixed: 'right', | ||||||
|  |         width: 250, | ||||||
|  |         scopedSlots: true, | ||||||
|  |     }, | ||||||
|  | ]; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * 搜索 | ||||||
|  |  * @param params | ||||||
|  |  */ | ||||||
|  | const handleSearch = (e: any) => { | ||||||
|  |     // console.log('handleSearch:', e); | ||||||
|  |     params.value = e; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | /** | ||||||
|  |  * 新增 | ||||||
|  |  */ | ||||||
|  | const handleAdd = () => { | ||||||
|  |     router.push(`/media/device/Save`); | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | const getActions = ( | ||||||
|  |     data: Partial<Record<string, any>>, | ||||||
|  |     type: 'card' | 'table', | ||||||
|  | ): ActionsType[] => { | ||||||
|  |     if (!data) return []; | ||||||
|  |     const actions = [ | ||||||
|  |         { | ||||||
|  |             key: 'edit', | ||||||
|  |             text: '编辑', | ||||||
|  |             tooltip: { | ||||||
|  |                 title: '编辑', | ||||||
|  |             }, | ||||||
|  |             icon: 'EditOutlined', | ||||||
|  |             onClick: () => { | ||||||
|  |                 router.push(`/media/device/Save?id=${data.id}`); | ||||||
|  |             }, | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |             key: 'view', | ||||||
|  |             text: '查看通道', | ||||||
|  |             tooltip: { | ||||||
|  |                 title: '查看通道', | ||||||
|  |             }, | ||||||
|  |             icon: 'PartitionOutlined', | ||||||
|  |             onClick: () => { | ||||||
|  |                 router.push( | ||||||
|  |                     `/media/device/Channel?id=${data.id}&type=${data.provider}`, | ||||||
|  |                 ); | ||||||
|  |             }, | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |             key: 'debug', | ||||||
|  |             text: '更新通道', | ||||||
|  |             tooltip: { | ||||||
|  |                 title: | ||||||
|  |                     data.provider === 'fixed-media' | ||||||
|  |                         ? '固定地址无法更新通道' | ||||||
|  |                         : data.state.value === 'offline' | ||||||
|  |                         ? '设备已离线' | ||||||
|  |                         : data.state.value === 'notActive' | ||||||
|  |                         ? '设备已禁用' | ||||||
|  |                         : '', | ||||||
|  |             }, | ||||||
|  |             disabled: | ||||||
|  |                 data.state.value === 'offline' || | ||||||
|  |                 data.state.value === 'notActive' || | ||||||
|  |                 data.provider === 'fixed-media', | ||||||
|  |             icon: 'SyncOutlined', | ||||||
|  |             onClick: () => { | ||||||
|  |                 // updateChannel() | ||||||
|  |             }, | ||||||
|  |         }, | ||||||
|  |         { | ||||||
|  |             key: 'delete', | ||||||
|  |             text: '删除', | ||||||
|  |             tooltip: { | ||||||
|  |                 title: '在线设备无法删除', | ||||||
|  |             }, | ||||||
|  |             disabled: data.state.value === 'online', | ||||||
|  |             popConfirm: { | ||||||
|  |                 title: '确认删除?', | ||||||
|  |                 onConfirm: async () => { | ||||||
|  |                     const resp = await DeviceApi.del(data.id); | ||||||
|  |                     if (resp.status === 200) { | ||||||
|  |                         message.success('操作成功!'); | ||||||
|  |                         listRef.value?.reload(); | ||||||
|  |                     } else { | ||||||
|  |                         message.error('操作失败!'); | ||||||
|  |                     } | ||||||
|  |                 }, | ||||||
|  |             }, | ||||||
|  |             icon: 'DeleteOutlined', | ||||||
|  |         }, | ||||||
|  |     ]; | ||||||
|  |     return actions; | ||||||
|  | }; | ||||||
|  | </script> | ||||||
|  | <style lang="less" scoped> | ||||||
|  | .page-container { | ||||||
|  |     background: #f0f2f5; | ||||||
|  |     padding: 24px; | ||||||
|  | } | ||||||
|  | </style> | ||||||
|  | @ -0,0 +1,24 @@ | ||||||
|  | type BaseItem = { | ||||||
|  |     id: string; | ||||||
|  |     name: string; | ||||||
|  | }; | ||||||
|  | type State = { | ||||||
|  |     value: string; | ||||||
|  |     text: string; | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | export type DeviceItem = { | ||||||
|  |     photoUrl?: string; | ||||||
|  |     channelNumber: number; | ||||||
|  |     createTime: number; | ||||||
|  |     firmware: string; | ||||||
|  |     gatewayId: string; | ||||||
|  |     host: string; | ||||||
|  |     manufacturer: string; | ||||||
|  |     model: string; | ||||||
|  |     port: number; | ||||||
|  |     provider: string; | ||||||
|  |     state: State; | ||||||
|  |     streamMode: string; | ||||||
|  |     transport: string; | ||||||
|  | } & BaseItem; | ||||||
		Loading…
	
		Reference in New Issue