Commit dbb85690e5ee7e65af6e972722a8faf760fd547f
1 parent
6c0755f1
feat: device profile add card mode
Showing
10 changed files
with
764 additions
and
311 deletions
| ... | ... | @@ -2,7 +2,9 @@ import { defHttp } from '/@/utils/http/axios'; |
| 2 | 2 | import { |
| 3 | 3 | TDeviceConfigParams, |
| 4 | 4 | IDeviceConfigAddOrEditModel, |
| 5 | + ProfileRecord, | |
| 5 | 6 | } from '/@/api/device/model/deviceConfigModel'; |
| 7 | +import { PaginationResult } from '/#/axios'; | |
| 6 | 8 | |
| 7 | 9 | enum EDeviceConfigApi { |
| 8 | 10 | /** |
| ... | ... | @@ -52,7 +54,7 @@ export const alarmContactGetPage = () => { |
| 52 | 54 | * 分页查询设备配置页面 |
| 53 | 55 | */ |
| 54 | 56 | export const deviceConfigGetQuery = (params?: TDeviceConfigParams) => { |
| 55 | - return defHttp.get({ | |
| 57 | + return defHttp.get<PaginationResult<ProfileRecord>>({ | |
| 56 | 58 | url: EDeviceConfigApi.DEVICE_CONFIG_GET_PAGE, |
| 57 | 59 | params, |
| 58 | 60 | }); | ... | ... |
| ... | ... | @@ -199,3 +199,45 @@ export interface AlarmLogItem { |
| 199 | 199 | organizationId: string; |
| 200 | 200 | organizationName: string; |
| 201 | 201 | } |
| 202 | + | |
| 203 | +export interface Configuration { | |
| 204 | + type: string; | |
| 205 | +} | |
| 206 | + | |
| 207 | +export interface TransportConfiguration { | |
| 208 | + type: string; | |
| 209 | +} | |
| 210 | + | |
| 211 | +export interface ProvisionConfiguration { | |
| 212 | + type: string; | |
| 213 | + provisionDeviceSecret?: any; | |
| 214 | +} | |
| 215 | + | |
| 216 | +export interface ProfileData { | |
| 217 | + configuration: Configuration; | |
| 218 | + transportConfiguration: TransportConfiguration; | |
| 219 | + provisionConfiguration: ProvisionConfiguration; | |
| 220 | + alarms?: any; | |
| 221 | +} | |
| 222 | + | |
| 223 | +export interface ProfileRecord { | |
| 224 | + id: string; | |
| 225 | + creator: string; | |
| 226 | + createTime: string; | |
| 227 | + updater: string; | |
| 228 | + updateTime: string; | |
| 229 | + name: string; | |
| 230 | + tenantId: string; | |
| 231 | + transportType: string; | |
| 232 | + provisionType: string; | |
| 233 | + deviceType: string; | |
| 234 | + tbProfileId: string; | |
| 235 | + profileData: ProfileData; | |
| 236 | + defaultRuleChainId: string; | |
| 237 | + defaultQueueName: string; | |
| 238 | + image: string; | |
| 239 | + type: string; | |
| 240 | + default: boolean; | |
| 241 | + | |
| 242 | + checked?: boolean; | |
| 243 | +} | ... | ... |
src/components/Widget/CardLayoutButton.vue
0 → 100644
| 1 | +<script lang="ts" setup> | |
| 2 | + import { Popover, Slider, Button } from 'ant-design-vue'; | |
| 3 | + import { LayoutOutlined } from '@ant-design/icons-vue'; | |
| 4 | + import { computed } from 'vue'; | |
| 5 | + | |
| 6 | + const props = withDefaults( | |
| 7 | + defineProps<{ | |
| 8 | + min?: number; | |
| 9 | + max?: number; | |
| 10 | + value?: number; | |
| 11 | + }>(), | |
| 12 | + { | |
| 13 | + min: 4, | |
| 14 | + max: 12, | |
| 15 | + value: 4, | |
| 16 | + } | |
| 17 | + ); | |
| 18 | + | |
| 19 | + const emit = defineEmits(['change', 'update:value']); | |
| 20 | + | |
| 21 | + const generateLayoutMarks = (min: number, max: number) => { | |
| 22 | + const marks = {}; | |
| 23 | + Array.from({ length: max - min + 1 }).forEach((_, index) => { | |
| 24 | + const key = index + min; | |
| 25 | + marks[key] = key; | |
| 26 | + }); | |
| 27 | + return marks; | |
| 28 | + }; | |
| 29 | + | |
| 30 | + const getMarks = computed(() => { | |
| 31 | + const { min, max } = props; | |
| 32 | + return generateLayoutMarks(min, max); | |
| 33 | + }); | |
| 34 | + | |
| 35 | + const handleChange = (value: number) => { | |
| 36 | + emit('update:value', value); | |
| 37 | + emit('change', value); | |
| 38 | + }; | |
| 39 | +</script> | |
| 40 | + | |
| 41 | +<template> | |
| 42 | + <Popover :trigger="['hover']"> | |
| 43 | + <template #content> | |
| 44 | + <div class="w-60"> | |
| 45 | + <div>每行显示数量</div> | |
| 46 | + <Slider | |
| 47 | + :value="props.value" | |
| 48 | + :max="props.max" | |
| 49 | + :min="props.min" | |
| 50 | + :marks="getMarks" | |
| 51 | + @change="handleChange" | |
| 52 | + /> | |
| 53 | + </div> | |
| 54 | + </template> | |
| 55 | + <Button type="primary"> | |
| 56 | + <LayoutOutlined /> | |
| 57 | + </Button> | |
| 58 | + </Popover> | |
| 59 | +</template> | ... | ... |
src/components/Widget/ModeSwitchButton.vue
0 → 100644
| 1 | +<script lang="ts" setup> | |
| 2 | + import { RadioButton, RadioGroup, Tooltip } from 'ant-design-vue'; | |
| 3 | + import { AppstoreOutlined, UnorderedListOutlined } from '@ant-design/icons-vue'; | |
| 4 | + import { Mode } from './const'; | |
| 5 | + const props = defineProps<{ | |
| 6 | + value: Mode; | |
| 7 | + }>(); | |
| 8 | + | |
| 9 | + const emit = defineEmits(['change']); | |
| 10 | + | |
| 11 | + const handleChange = (event: Event) => { | |
| 12 | + const value = (event.target as HTMLInputElement).value; | |
| 13 | + emit('change', value); | |
| 14 | + }; | |
| 15 | +</script> | |
| 16 | + | |
| 17 | +<template> | |
| 18 | + <RadioGroup :value="props.value" button-style="solid" @change="handleChange"> | |
| 19 | + <Tooltip title="卡片模式"> | |
| 20 | + <RadioButton class="cursor-pointer" :value="Mode.CARD"> | |
| 21 | + <AppstoreOutlined /> | |
| 22 | + </RadioButton> | |
| 23 | + </Tooltip> | |
| 24 | + <Tooltip title="列表模式"> | |
| 25 | + <RadioButton class="cursor-pointer" :value="Mode.TABLE"> | |
| 26 | + <UnorderedListOutlined /> | |
| 27 | + </RadioButton> | |
| 28 | + </Tooltip> | |
| 29 | + </RadioGroup> | |
| 30 | +</template> | ... | ... |
src/components/Widget/const.ts
0 → 100644
src/components/Widget/index.ts
0 → 100644
src/views/device/profiles/CardMode.vue
0 → 100644
| 1 | +<script lang="ts" setup> | |
| 2 | + import { PageWrapper } from '/@/components/Page'; | |
| 3 | + import { BasicForm, useForm } from '/@/components/Form'; | |
| 4 | + import { List, Button, Tooltip, Card, PaginationProps, Image } from 'ant-design-vue'; | |
| 5 | + import { ReloadOutlined, EyeOutlined, FormOutlined, MoreOutlined } from '@ant-design/icons-vue'; | |
| 6 | + import { computed, onMounted, reactive, ref, unref } from 'vue'; | |
| 7 | + import { CardLayoutButton, Mode, ModeSwitchButton } from '/@/components/Widget'; | |
| 8 | + import { Authority } from '/@/components/Authority'; | |
| 9 | + import { | |
| 10 | + deviceConfigDelete, | |
| 11 | + deviceConfigGetQuery, | |
| 12 | + setDeviceProfileIsDefaultApi, | |
| 13 | + } from '/@/api/device/deviceConfigApi'; | |
| 14 | + import { ProfileRecord } from '/@/api/device/model/deviceConfigModel'; | |
| 15 | + import { Dropdown } from '/@/components/Dropdown'; | |
| 16 | + import { defaultObj, searchFormSchema, DeviceTypeName } from './device.profile.data'; | |
| 17 | + import { useMessage } from '/@/hooks/web/useMessage'; | |
| 18 | + import { useSyncConfirm } from '/@/hooks/component/useSyncConfirm'; | |
| 19 | + import DeviceProfileModal from './DeviceProfileModal.vue'; | |
| 20 | + import DeviceProfileDrawer from './DeviceProfileDrawer.vue'; | |
| 21 | + import { useModal } from '/@/components/Modal'; | |
| 22 | + import { useDrawer } from '/@/components/Drawer'; | |
| 23 | + | |
| 24 | + defineProps<{ | |
| 25 | + mode: Mode; | |
| 26 | + }>(); | |
| 27 | + | |
| 28 | + const emit = defineEmits(['changeMode']); | |
| 29 | + | |
| 30 | + enum DropMenuEvent { | |
| 31 | + SET_DEFAULT = 'setDefault', | |
| 32 | + DELETE = 'delete', | |
| 33 | + } | |
| 34 | + const IMAGE_FALLBACK = | |
| 35 | + ''; | |
| 36 | + | |
| 37 | + const { createMessage } = useMessage(); | |
| 38 | + const { createSyncConfirm } = useSyncConfirm(); | |
| 39 | + | |
| 40 | + const [register, { getFieldsValue }] = useForm({ | |
| 41 | + showAdvancedButton: true, | |
| 42 | + labelWidth: 100, | |
| 43 | + compact: true, | |
| 44 | + baseColProps: { span: 8 }, | |
| 45 | + schemas: searchFormSchema, | |
| 46 | + submitFunc: async () => { | |
| 47 | + getDataSource(); | |
| 48 | + }, | |
| 49 | + }); | |
| 50 | + | |
| 51 | + const [registerModal, { openModal }] = useModal(); | |
| 52 | + const [registerDrawer, { openDrawer }] = useDrawer(); | |
| 53 | + | |
| 54 | + const loading = ref(false); | |
| 55 | + | |
| 56 | + const pagination = reactive<PaginationProps>({ | |
| 57 | + size: 'small', | |
| 58 | + showTotal: (total: number) => `共 ${total} 条数据`, | |
| 59 | + current: 1, | |
| 60 | + onChange: (page: number) => { | |
| 61 | + pagination.current = page; | |
| 62 | + getDataSource(); | |
| 63 | + }, | |
| 64 | + }); | |
| 65 | + | |
| 66 | + const dataSource = ref<ProfileRecord[]>([]); | |
| 67 | + | |
| 68 | + const colNumber = ref(4); | |
| 69 | + | |
| 70 | + const getSelectAllFlag = computed(() => { | |
| 71 | + return unref(dataSource).every((item) => item.checked); | |
| 72 | + }); | |
| 73 | + | |
| 74 | + const getCheckedRecord = computed(() => { | |
| 75 | + return unref(dataSource) | |
| 76 | + .filter((item) => item.checked) | |
| 77 | + .map((item) => item.id); | |
| 78 | + }); | |
| 79 | + | |
| 80 | + const getDataSource = async () => { | |
| 81 | + try { | |
| 82 | + loading.value = true; | |
| 83 | + const params = getFieldsValue(); | |
| 84 | + const { items, total } = await deviceConfigGetQuery({ | |
| 85 | + page: pagination.current, | |
| 86 | + pageSize: unref(colNumber) * 2, | |
| 87 | + ...params, | |
| 88 | + }); | |
| 89 | + pagination.total = total; | |
| 90 | + dataSource.value = items.map((item) => ({ ...item, checked: false })); | |
| 91 | + } catch (error) { | |
| 92 | + } finally { | |
| 93 | + loading.value = false; | |
| 94 | + } | |
| 95 | + }; | |
| 96 | + | |
| 97 | + const handleModeChange = (mode: Mode) => { | |
| 98 | + emit('changeMode', mode); | |
| 99 | + }; | |
| 100 | + | |
| 101 | + const handleCheckCard = (item: ProfileRecord) => { | |
| 102 | + item.checked = !item.checked; | |
| 103 | + }; | |
| 104 | + | |
| 105 | + const handleSelectAll = () => { | |
| 106 | + dataSource.value = unref(dataSource).map((item) => { | |
| 107 | + return { | |
| 108 | + ...item, | |
| 109 | + checked: !unref(getSelectAllFlag), | |
| 110 | + }; | |
| 111 | + }); | |
| 112 | + }; | |
| 113 | + | |
| 114 | + const handleCreate = () => { | |
| 115 | + openModal(true, { | |
| 116 | + isUpdate: false, | |
| 117 | + }); | |
| 118 | + }; | |
| 119 | + | |
| 120 | + const handleShowDetail = (record: ProfileRecord) => { | |
| 121 | + openDrawer(true, { record }); | |
| 122 | + }; | |
| 123 | + | |
| 124 | + const handleUpdate = (record: ProfileRecord) => { | |
| 125 | + openModal(true, { | |
| 126 | + record, | |
| 127 | + isUpdate: true, | |
| 128 | + }); | |
| 129 | + }; | |
| 130 | + | |
| 131 | + const handleDelete = async (id: string[]) => { | |
| 132 | + try { | |
| 133 | + await createSyncConfirm({ iconType: 'warning', content: '是否确认删除操作?' }); | |
| 134 | + await deviceConfigDelete(id); | |
| 135 | + createMessage.success('删除成功'); | |
| 136 | + await getDataSource(); | |
| 137 | + } catch (error) { | |
| 138 | + throw error; | |
| 139 | + } | |
| 140 | + }; | |
| 141 | + | |
| 142 | + const handleSetDefault = async (record: ProfileRecord) => { | |
| 143 | + try { | |
| 144 | + const { tbProfileId } = record; | |
| 145 | + const data = await setDeviceProfileIsDefaultApi(tbProfileId, 'default', defaultObj); | |
| 146 | + if (!data) return createMessage.error('设置该产品为默认失败'); | |
| 147 | + createMessage.success('设置该产品为默认成功'); | |
| 148 | + await getDataSource(); | |
| 149 | + } catch (error) { | |
| 150 | + throw error; | |
| 151 | + } | |
| 152 | + }; | |
| 153 | + | |
| 154 | + onMounted(() => { | |
| 155 | + getDataSource(); | |
| 156 | + }); | |
| 157 | +</script> | |
| 158 | + | |
| 159 | +<template> | |
| 160 | + <PageWrapper dense contentFullHeight contentClass="flex"> | |
| 161 | + <section class="flex-auto p-4 w-full profile-list"> | |
| 162 | + <div class="flex-auto w-full bg-light-50 dark:bg-dark-900 p-4"> | |
| 163 | + <BasicForm @register="register" /> | |
| 164 | + </div> | |
| 165 | + <List | |
| 166 | + ref="listEl" | |
| 167 | + :loading="loading" | |
| 168 | + class="flex-auto bg-light-50 dark:bg-dark-900 !p-2 !mt-4" | |
| 169 | + position="bottom" | |
| 170 | + :pagination="pagination" | |
| 171 | + :data-source="dataSource" | |
| 172 | + :grid="{ gutter: 4, column: colNumber }" | |
| 173 | + > | |
| 174 | + <template #header> | |
| 175 | + <div class="flex gap-3 justify-end"> | |
| 176 | + <Button type="primary" @click="handleCreate">新增产品</Button> | |
| 177 | + <Button type="primary" @click="handleSelectAll"> | |
| 178 | + {{ getSelectAllFlag ? '反选' : '全选' }} | |
| 179 | + </Button> | |
| 180 | + <Button | |
| 181 | + type="primary" | |
| 182 | + danger | |
| 183 | + :disabled="!getCheckedRecord.length" | |
| 184 | + @click="handleDelete(getCheckedRecord)" | |
| 185 | + > | |
| 186 | + 批量删除 | |
| 187 | + </Button> | |
| 188 | + <ModeSwitchButton :value="$props.mode" @change="handleModeChange" /> | |
| 189 | + <CardLayoutButton v-model:value="colNumber" @change="getDataSource" /> | |
| 190 | + <Tooltip title="刷新"> | |
| 191 | + <Button type="primary" @click="getDataSource"> | |
| 192 | + <ReloadOutlined :spin="loading" /> | |
| 193 | + </Button> | |
| 194 | + </Tooltip> | |
| 195 | + </div> | |
| 196 | + </template> | |
| 197 | + <template #renderItem="{ item }"> | |
| 198 | + <List.Item> | |
| 199 | + <Card | |
| 200 | + hoverable | |
| 201 | + @click="handleCheckCard(item)" | |
| 202 | + :class="item.checked ? '!border-blue-500 !border-2' : ''" | |
| 203 | + > | |
| 204 | + <template #cover> | |
| 205 | + <div class="h-full w-full !flex justify-center items-center text-center"> | |
| 206 | + <Image | |
| 207 | + @click.stop | |
| 208 | + :height="144" | |
| 209 | + :src="item.image" | |
| 210 | + placeholder | |
| 211 | + :fallback="IMAGE_FALLBACK" | |
| 212 | + /> | |
| 213 | + </div> | |
| 214 | + </template> | |
| 215 | + <template class="ant-card-actions" #actions> | |
| 216 | + <Authority> | |
| 217 | + <Tooltip title="详情"> | |
| 218 | + <EyeOutlined key="setting" @click.stop="handleShowDetail(item)" /> | |
| 219 | + </Tooltip> | |
| 220 | + </Authority> | |
| 221 | + <Authority> | |
| 222 | + <Tooltip title="编辑"> | |
| 223 | + <FormOutlined key="edit" @click.stop="handleUpdate(item)" /> | |
| 224 | + </Tooltip> | |
| 225 | + </Authority> | |
| 226 | + <Dropdown | |
| 227 | + :trigger="['hover']" | |
| 228 | + :drop-menu-list="[ | |
| 229 | + { | |
| 230 | + text: '默认', | |
| 231 | + event: DropMenuEvent.SET_DEFAULT, | |
| 232 | + icon: 'ant-design:unordered-list-outlined', | |
| 233 | + onClick: handleSetDefault.bind(null, item), | |
| 234 | + }, | |
| 235 | + { | |
| 236 | + text: '删除', | |
| 237 | + event: DropMenuEvent.DELETE, | |
| 238 | + icon: 'ant-design:delete-outlined', | |
| 239 | + onClick: handleDelete.bind(null, [item.id]), | |
| 240 | + }, | |
| 241 | + ]" | |
| 242 | + > | |
| 243 | + <MoreOutlined @click.stop class="transform rotate-90" /> | |
| 244 | + </Dropdown> | |
| 245 | + </template> | |
| 246 | + <Card.Meta> | |
| 247 | + <template #title> | |
| 248 | + <span class="truncate"> {{ item.name }} </span> | |
| 249 | + </template> | |
| 250 | + <template #description> | |
| 251 | + <div class="truncate h-11"> | |
| 252 | + <div class="truncate">{{ DeviceTypeName[item.deviceType] }} </div> | |
| 253 | + <div class="truncate">{{ item.transportType }} </div> | |
| 254 | + </div> | |
| 255 | + </template> | |
| 256 | + </Card.Meta> | |
| 257 | + </Card> | |
| 258 | + </List.Item> | |
| 259 | + </template> | |
| 260 | + </List> | |
| 261 | + </section> | |
| 262 | + <DeviceProfileModal @register="registerModal" @success="getDataSource" /> | |
| 263 | + <DeviceProfileDrawer @register="registerDrawer" /> | |
| 264 | + </PageWrapper> | |
| 265 | +</template> | |
| 266 | + | |
| 267 | +<style lang="less" scoped> | |
| 268 | + .profile-list:deep(.ant-image-img) { | |
| 269 | + width: 100% !important; | |
| 270 | + height: 100% !important; | |
| 271 | + } | |
| 272 | +</style> | ... | ... |
src/views/device/profiles/TableMode.vue
0 → 100644
| 1 | +<template> | |
| 2 | + <div> | |
| 3 | + <BasicTable | |
| 4 | + class="devide-profiles" | |
| 5 | + @register="registerTable" | |
| 6 | + :rowSelection="{ type: 'checkbox' }" | |
| 7 | + :clickToRowSelect="false" | |
| 8 | + > | |
| 9 | + <template #toolbar> | |
| 10 | + <Authority value="api:yt:deviceProfile:post"> | |
| 11 | + <a-button type="primary" @click="handleCreate"> 新增产品 </a-button> | |
| 12 | + </Authority> | |
| 13 | + <Authority value="api:yt:deviceProfile:import"> | |
| 14 | + <ImpExcel @success="loadDataSuccess" dateFormat="YYYY-MM-DD"> | |
| 15 | + <a-button @click="handleImport"> 导入产品 </a-button> | |
| 16 | + </ImpExcel> | |
| 17 | + </Authority> | |
| 18 | + <Authority value="api:yt:deviceProfile:delete"> | |
| 19 | + <Popconfirm | |
| 20 | + title="您确定要批量删除数据" | |
| 21 | + ok-text="确定" | |
| 22 | + cancel-text="取消" | |
| 23 | + @confirm="handleDeleteOrBatchDelete(null)" | |
| 24 | + > | |
| 25 | + <a-button type="primary" color="error" :disabled="hasBatchDelete"> 批量删除 </a-button> | |
| 26 | + </Popconfirm> | |
| 27 | + </Authority> | |
| 28 | + <ModeSwitchButton :value="$props.mode" @change="handleModeChange" /> | |
| 29 | + </template> | |
| 30 | + <template #img="{ record }"> | |
| 31 | + <TableImg | |
| 32 | + :size="30" | |
| 33 | + :showBadge="false" | |
| 34 | + :simpleShow="true" | |
| 35 | + :imgList=" | |
| 36 | + typeof record.image !== 'undefined' && record.image !== '' && record.image != null | |
| 37 | + ? [record.image] | |
| 38 | + : null | |
| 39 | + " | |
| 40 | + /> | |
| 41 | + </template> | |
| 42 | + <template #action="{ record }"> | |
| 43 | + <TableAction | |
| 44 | + :actions="[ | |
| 45 | + { | |
| 46 | + label: '详情', | |
| 47 | + auth: 'api:yt:deviceProfile:get', | |
| 48 | + icon: 'ant-design:eye-outlined', | |
| 49 | + onClick: handleDetailView.bind(null, record), | |
| 50 | + }, | |
| 51 | + { | |
| 52 | + label: '编辑', | |
| 53 | + auth: 'api:yt:deviceProfile:update', | |
| 54 | + icon: 'clarity:note-edit-line', | |
| 55 | + onClick: handleEdit.bind(null, record), | |
| 56 | + ifShow: () => { | |
| 57 | + return record.name !== 'default' ? true : false; | |
| 58 | + }, | |
| 59 | + }, | |
| 60 | + ]" | |
| 61 | + :drop-down-actions="[ | |
| 62 | + { | |
| 63 | + label: '默认', | |
| 64 | + icon: 'ant-design:profile-outlined', | |
| 65 | + onClick: handleSetDefault.bind(null, record), | |
| 66 | + ifShow: () => { | |
| 67 | + return record.default === false; | |
| 68 | + }, | |
| 69 | + }, | |
| 70 | + { | |
| 71 | + label: '导出', | |
| 72 | + auth: 'api:yt:deviceProfile:export', | |
| 73 | + icon: 'ant-design:login-outlined', | |
| 74 | + onClick: handleExport.bind(null, record), | |
| 75 | + }, | |
| 76 | + { | |
| 77 | + label: '删除', | |
| 78 | + auth: 'api:yt:deviceProfile:delete', | |
| 79 | + icon: 'ant-design:delete-outlined', | |
| 80 | + color: 'error', | |
| 81 | + popConfirm: { | |
| 82 | + title: '是否确认删除', | |
| 83 | + confirm: handleDeleteOrBatchDelete.bind(null, record), | |
| 84 | + }, | |
| 85 | + ifShow: () => { | |
| 86 | + return record.default === false && record.name !== 'default'; | |
| 87 | + }, | |
| 88 | + }, | |
| 89 | + ]" | |
| 90 | + /> | |
| 91 | + </template> | |
| 92 | + </BasicTable> | |
| 93 | + <DeviceProfileModal @register="registerModal" @success="handleSuccess" /> | |
| 94 | + <DeviceProfileDrawer @register="registerDrawer" /> | |
| 95 | + <ExpExcelModal | |
| 96 | + ref="expExcelModalRef" | |
| 97 | + @register="registerExportModal" | |
| 98 | + @success="defaultHeader" | |
| 99 | + /> | |
| 100 | + </div> | |
| 101 | +</template> | |
| 102 | +<script lang="ts" setup> | |
| 103 | + import { ref, nextTick, onUnmounted } from 'vue'; | |
| 104 | + import { BasicTable, TableImg, useTable, TableAction, BasicColumn } from '/@/components/Table'; | |
| 105 | + import { columns, searchFormSchema, defaultObj } from './device.profile.data'; | |
| 106 | + import { useMessage } from '/@/hooks/web/useMessage'; | |
| 107 | + import { | |
| 108 | + deviceConfigGetQuery, | |
| 109 | + deviceConfigDelete, | |
| 110 | + setDeviceProfileIsDefaultApi, | |
| 111 | + } from '/@/api/device/deviceConfigApi'; | |
| 112 | + import { useModal } from '/@/components/Modal'; | |
| 113 | + import { useDrawer } from '/@/components/Drawer'; | |
| 114 | + import DeviceProfileModal from '/@/views/device/profiles/DeviceProfileModal.vue'; | |
| 115 | + import { ImpExcel, ExcelData } from '/@/components/Excel'; | |
| 116 | + import { jsonToSheetXlsx, ExpExcelModal, ExportModalResult } from '/@/components/Excel'; | |
| 117 | + import { Authority } from '/@/components/Authority'; | |
| 118 | + import { useBatchDelete } from '/@/hooks/web/useBatchDelete'; | |
| 119 | + import { Popconfirm } from 'ant-design-vue'; | |
| 120 | + import DeviceProfileDrawer from './DeviceProfileDrawer.vue'; | |
| 121 | + import { Mode, ModeSwitchButton } from '/@/components/Widget'; | |
| 122 | + | |
| 123 | + defineProps<{ | |
| 124 | + mode: Mode; | |
| 125 | + }>(); | |
| 126 | + | |
| 127 | + const emit = defineEmits(['changeMode']); | |
| 128 | + | |
| 129 | + const exportData: any = ref([]); | |
| 130 | + const expExcelModalRef: any = ref(null); | |
| 131 | + const getPathUrl = ref(''); | |
| 132 | + const getPathUrlName = ref(''); | |
| 133 | + const disabled = ref(true); | |
| 134 | + const onCloseVal = ref(0); | |
| 135 | + const immediateStatus = ref(false); | |
| 136 | + const { createMessage } = useMessage(); | |
| 137 | + const [registerModal, { openModal }] = useModal(); | |
| 138 | + const [registerExportModal, { openModal: openModalExcel }] = useModal(); | |
| 139 | + const [registerTable, { setProps, reload, setTableData, getForm }] = useTable({ | |
| 140 | + title: '产品列表', | |
| 141 | + clickToRowSelect: false, | |
| 142 | + api: deviceConfigGetQuery, | |
| 143 | + immediate: immediateStatus.value, | |
| 144 | + columns, | |
| 145 | + formConfig: { | |
| 146 | + labelWidth: 120, | |
| 147 | + schemas: searchFormSchema, | |
| 148 | + }, | |
| 149 | + rowKey: 'id', | |
| 150 | + useSearchForm: true, | |
| 151 | + showTableSetting: true, | |
| 152 | + bordered: true, | |
| 153 | + showIndexColumn: false, | |
| 154 | + actionColumn: { | |
| 155 | + width: 200, | |
| 156 | + title: '操作', | |
| 157 | + dataIndex: 'action', | |
| 158 | + slots: { customRender: 'action' }, | |
| 159 | + fixed: 'right', | |
| 160 | + }, | |
| 161 | + }); | |
| 162 | + const { hasBatchDelete, handleDeleteOrBatchDelete, selectionOptions } = useBatchDelete( | |
| 163 | + deviceConfigDelete, | |
| 164 | + handleSuccess, | |
| 165 | + setProps | |
| 166 | + ); | |
| 167 | + selectionOptions.rowSelection.getCheckboxProps = (record: Recordable) => { | |
| 168 | + // Demo:status为1的选择框禁用 | |
| 169 | + if (record.default === true) { | |
| 170 | + return { disabled: true }; | |
| 171 | + } else if (record.name == 'default') { | |
| 172 | + return { disabled: true }; | |
| 173 | + } else { | |
| 174 | + return { disabled: false }; | |
| 175 | + } | |
| 176 | + }; | |
| 177 | + nextTick(() => { | |
| 178 | + setProps(selectionOptions); | |
| 179 | + }); | |
| 180 | + /** | |
| 181 | + *@param url,name | |
| 182 | + **/ | |
| 183 | + function getParam(url, name) { | |
| 184 | + try { | |
| 185 | + let reg = new RegExp('(^|&)' + name + '=([^&]*)(&|$)'); | |
| 186 | + let r = url.split('?')[1].match(reg); | |
| 187 | + if (r != null) { | |
| 188 | + return r[2]; | |
| 189 | + } | |
| 190 | + return ''; //如果此处只写return;则返回的是undefined | |
| 191 | + } catch (e) { | |
| 192 | + return ''; //如果此处只写return;则返回的是undefined | |
| 193 | + } | |
| 194 | + } | |
| 195 | + getPathUrl.value = window.location.href; | |
| 196 | + const name = 'name'; | |
| 197 | + const getName = getParam(getPathUrl.value, name); | |
| 198 | + getPathUrlName.value = decodeURIComponent(getName); | |
| 199 | + | |
| 200 | + const setRowClassName = async () => { | |
| 201 | + if (getPathUrlName.value !== '') { | |
| 202 | + const { items } = await deviceConfigGetQuery({ | |
| 203 | + page: 1, | |
| 204 | + pageSize: 10, | |
| 205 | + name: getPathUrlName.value, | |
| 206 | + }); | |
| 207 | + nextTick(() => { | |
| 208 | + setTableData(items); | |
| 209 | + const { setFieldsValue, resetFields } = getForm(); | |
| 210 | + setFieldsValue({ | |
| 211 | + name: getPathUrlName.value, | |
| 212 | + }); | |
| 213 | + if (onCloseVal.value == 1) { | |
| 214 | + resetFields(); | |
| 215 | + } | |
| 216 | + }); | |
| 217 | + } else { | |
| 218 | + setTimeout(() => { | |
| 219 | + reload(); | |
| 220 | + }, 80); | |
| 221 | + } | |
| 222 | + }; | |
| 223 | + setRowClassName(); | |
| 224 | + onUnmounted(() => { | |
| 225 | + getPathUrlName.value = ''; | |
| 226 | + onCloseVal.value = 1; | |
| 227 | + }); | |
| 228 | + const tableListRef = ref< | |
| 229 | + { | |
| 230 | + title: string; | |
| 231 | + columns?: any[]; | |
| 232 | + dataSource?: any[]; | |
| 233 | + }[] | |
| 234 | + >([]); | |
| 235 | + | |
| 236 | + function loadDataSuccess(excelDataList: ExcelData[]) { | |
| 237 | + tableListRef.value = []; | |
| 238 | + console.log(excelDataList); | |
| 239 | + for (const excelData of excelDataList) { | |
| 240 | + const { | |
| 241 | + header, | |
| 242 | + results, | |
| 243 | + meta: { sheetName }, | |
| 244 | + } = excelData; | |
| 245 | + const columns: BasicColumn[] = []; | |
| 246 | + for (const title of header) { | |
| 247 | + columns.push({ title, dataIndex: title }); | |
| 248 | + } | |
| 249 | + tableListRef.value.push({ title: sheetName, dataSource: results, columns }); | |
| 250 | + } | |
| 251 | + } | |
| 252 | + //新增 | |
| 253 | + function handleCreate() { | |
| 254 | + openModal(true, { | |
| 255 | + isUpdate: false, | |
| 256 | + }); | |
| 257 | + } | |
| 258 | + //编辑 | |
| 259 | + function handleEdit(record: Recordable) { | |
| 260 | + openModal(true, { | |
| 261 | + record, | |
| 262 | + isUpdate: true, | |
| 263 | + }); | |
| 264 | + } | |
| 265 | + | |
| 266 | + const [registerDrawer, { openDrawer }] = useDrawer(); | |
| 267 | + //详情 | |
| 268 | + function handleDetailView(record: Recordable) { | |
| 269 | + openDrawer(true, { record }); | |
| 270 | + } | |
| 271 | + | |
| 272 | + function defaultHeader({ filename, bookType }: ExportModalResult) { | |
| 273 | + // 默认Object.keys(data[0])作为header | |
| 274 | + const data = exportData.value; | |
| 275 | + jsonToSheetXlsx({ | |
| 276 | + data, | |
| 277 | + filename, | |
| 278 | + write2excelOpts: { | |
| 279 | + bookType, | |
| 280 | + }, | |
| 281 | + }); | |
| 282 | + } | |
| 283 | + //导出 | |
| 284 | + const handleExport = (record: Recordable) => { | |
| 285 | + exportData.value = []; | |
| 286 | + exportData.value.push({ | |
| 287 | + createTime: record.createTime, | |
| 288 | + description: record.description, | |
| 289 | + name: record.name, | |
| 290 | + }); | |
| 291 | + nextTick(() => { | |
| 292 | + openModalExcel(); | |
| 293 | + expExcelModalRef.value?.clearFieldFunc(); | |
| 294 | + }); | |
| 295 | + }; | |
| 296 | + //导入 | |
| 297 | + function handleImport() { | |
| 298 | + console.log('record'); | |
| 299 | + } | |
| 300 | + function handleSuccess() { | |
| 301 | + reload(); | |
| 302 | + } | |
| 303 | + | |
| 304 | + const handleSetDefault = async (record: Recordable) => { | |
| 305 | + let id = record.tbProfileId; | |
| 306 | + const data = await setDeviceProfileIsDefaultApi(id, 'default', defaultObj); | |
| 307 | + if (!data) return createMessage.error('设置该产品为默认失败'); | |
| 308 | + createMessage.success('设置该产品为默认成功'); | |
| 309 | + reload(); | |
| 310 | + disabled.value = true; | |
| 311 | + }; | |
| 312 | + | |
| 313 | + const handleModeChange = (value: Mode) => { | |
| 314 | + emit('changeMode', value); | |
| 315 | + }; | |
| 316 | +</script> | |
| 317 | + | |
| 318 | +<style lang="css"> | |
| 319 | + .devide-profiles .rowcolor { | |
| 320 | + color: red; | |
| 321 | + } | |
| 322 | + | |
| 323 | + .devide-profiles .rowcolor2 { | |
| 324 | + background: #a2c3e6; | |
| 325 | + } | |
| 326 | +</style> | ... | ... |
| ... | ... | @@ -9,6 +9,17 @@ import { FormField, FunctionType } from './step/cpns/physical/cpns/config'; |
| 9 | 9 | import { h } from 'vue'; |
| 10 | 10 | import { Tag } from 'ant-design-vue'; |
| 11 | 11 | |
| 12 | +export enum Mode { | |
| 13 | + CARD = 'card', | |
| 14 | + TABLE = 'table', | |
| 15 | +} | |
| 16 | + | |
| 17 | +export enum DeviceTypeName { | |
| 18 | + DIRECT_CONNECTION = '直连设备', | |
| 19 | + GATEWAY = '网关设备', | |
| 20 | + SENSOR = '网关子设备', | |
| 21 | +} | |
| 22 | + | |
| 12 | 23 | export enum ModelOfMatterPermission { |
| 13 | 24 | CREATE = 'api:yt:things_model:post', |
| 14 | 25 | UPDATE = 'api:yt:things_model:put', | ... | ... |
| 1 | -<template> | |
| 2 | - <div> | |
| 3 | - <BasicTable | |
| 4 | - class="devide-profiles" | |
| 5 | - @register="registerTable" | |
| 6 | - :rowSelection="{ type: 'checkbox' }" | |
| 7 | - :clickToRowSelect="false" | |
| 8 | - > | |
| 9 | - <template #toolbar> | |
| 10 | - <Authority value="api:yt:deviceProfile:post"> | |
| 11 | - <a-button type="primary" @click="handleCreate"> 新增产品 </a-button> | |
| 12 | - </Authority> | |
| 13 | - <Authority value="api:yt:deviceProfile:import"> | |
| 14 | - <ImpExcel @success="loadDataSuccess" dateFormat="YYYY-MM-DD"> | |
| 15 | - <a-button @click="handleImport"> 导入产品 </a-button> | |
| 16 | - </ImpExcel> | |
| 17 | - </Authority> | |
| 18 | - <Authority value="api:yt:deviceProfile:delete"> | |
| 19 | - <Popconfirm | |
| 20 | - title="您确定要批量删除数据" | |
| 21 | - ok-text="确定" | |
| 22 | - cancel-text="取消" | |
| 23 | - @confirm="handleDeleteOrBatchDelete(null)" | |
| 24 | - > | |
| 25 | - <a-button type="primary" color="error" :disabled="hasBatchDelete"> 批量删除 </a-button> | |
| 26 | - </Popconfirm> | |
| 27 | - </Authority> | |
| 28 | - </template> | |
| 29 | - <template #img="{ record }"> | |
| 30 | - <TableImg | |
| 31 | - :size="30" | |
| 32 | - :showBadge="false" | |
| 33 | - :simpleShow="true" | |
| 34 | - :imgList=" | |
| 35 | - typeof record.image !== 'undefined' && record.image !== '' && record.image != null | |
| 36 | - ? [record.image] | |
| 37 | - : null | |
| 38 | - " | |
| 39 | - /> | |
| 40 | - </template> | |
| 41 | - <template #action="{ record }"> | |
| 42 | - <TableAction | |
| 43 | - :actions="[ | |
| 44 | - { | |
| 45 | - label: '详情', | |
| 46 | - auth: 'api:yt:deviceProfile:get', | |
| 47 | - icon: 'ant-design:eye-outlined', | |
| 48 | - onClick: handleDetailView.bind(null, record), | |
| 49 | - }, | |
| 50 | - { | |
| 51 | - label: '编辑', | |
| 52 | - auth: 'api:yt:deviceProfile:update', | |
| 53 | - icon: 'clarity:note-edit-line', | |
| 54 | - onClick: handleEdit.bind(null, record), | |
| 55 | - ifShow: () => { | |
| 56 | - return record.name !== 'default' ? true : false; | |
| 57 | - }, | |
| 58 | - }, | |
| 59 | - ]" | |
| 60 | - :drop-down-actions="[ | |
| 61 | - { | |
| 62 | - label: '默认', | |
| 63 | - icon: 'ant-design:profile-outlined', | |
| 64 | - onClick: handleSetDefault.bind(null, record), | |
| 65 | - ifShow: () => { | |
| 66 | - return record.default === false; | |
| 67 | - }, | |
| 68 | - }, | |
| 69 | - { | |
| 70 | - label: '导出', | |
| 71 | - auth: 'api:yt:deviceProfile:export', | |
| 72 | - icon: 'ant-design:login-outlined', | |
| 73 | - onClick: handleExport.bind(null, record), | |
| 74 | - }, | |
| 75 | - { | |
| 76 | - label: '删除', | |
| 77 | - auth: 'api:yt:deviceProfile:delete', | |
| 78 | - icon: 'ant-design:delete-outlined', | |
| 79 | - color: 'error', | |
| 80 | - popConfirm: { | |
| 81 | - title: '是否确认删除', | |
| 82 | - confirm: handleDeleteOrBatchDelete.bind(null, record), | |
| 83 | - }, | |
| 84 | - ifShow: () => { | |
| 85 | - return record.default === false && record.name !== 'default'; | |
| 86 | - }, | |
| 87 | - }, | |
| 88 | - ]" | |
| 89 | - /> | |
| 90 | - </template> | |
| 91 | - </BasicTable> | |
| 92 | - <DeviceProfileModal @register="registerModal" @success="handleSuccess" /> | |
| 93 | - <DeviceProfileDrawer @register="registerDrawer" /> | |
| 94 | - <ExpExcelModal | |
| 95 | - ref="expExcelModalRef" | |
| 96 | - @register="registerExportModal" | |
| 97 | - @success="defaultHeader" | |
| 98 | - /> | |
| 99 | - </div> | |
| 100 | -</template> | |
| 101 | 1 | <script lang="ts" setup> |
| 102 | - import { ref, nextTick, onUnmounted } from 'vue'; | |
| 103 | - import { BasicTable, TableImg, useTable, TableAction, BasicColumn } from '/@/components/Table'; | |
| 104 | - import { columns, searchFormSchema, defaultObj } from './device.profile.data'; | |
| 105 | - import { useMessage } from '/@/hooks/web/useMessage'; | |
| 106 | - import { | |
| 107 | - deviceConfigGetQuery, | |
| 108 | - deviceConfigDelete, | |
| 109 | - setDeviceProfileIsDefaultApi, | |
| 110 | - } from '/@/api/device/deviceConfigApi'; | |
| 111 | - import { useModal } from '/@/components/Modal'; | |
| 112 | - import { useDrawer } from '/@/components/Drawer'; | |
| 113 | - import DeviceProfileModal from '/@/views/device/profiles/DeviceProfileModal.vue'; | |
| 114 | - import { ImpExcel, ExcelData } from '/@/components/Excel'; | |
| 115 | - import { jsonToSheetXlsx, ExpExcelModal, ExportModalResult } from '/@/components/Excel'; | |
| 116 | - import { Authority } from '/@/components/Authority'; | |
| 117 | - import { useBatchDelete } from '/@/hooks/web/useBatchDelete'; | |
| 118 | - import { Popconfirm } from 'ant-design-vue'; | |
| 119 | - import DeviceProfileDrawer from './DeviceProfileDrawer.vue'; | |
| 120 | - | |
| 121 | - const exportData: any = ref([]); | |
| 122 | - const expExcelModalRef: any = ref(null); | |
| 123 | - const getPathUrl = ref(''); | |
| 124 | - const getPathUrlName = ref(''); | |
| 125 | - const disabled = ref(true); | |
| 126 | - const onCloseVal = ref(0); | |
| 127 | - const immediateStatus = ref(false); | |
| 128 | - const { createMessage } = useMessage(); | |
| 129 | - const [registerModal, { openModal }] = useModal(); | |
| 130 | - const [registerExportModal, { openModal: openModalExcel }] = useModal(); | |
| 131 | - const [registerTable, { setProps, reload, setTableData, getForm }] = useTable({ | |
| 132 | - title: '产品列表', | |
| 133 | - clickToRowSelect: false, | |
| 134 | - api: deviceConfigGetQuery, | |
| 135 | - immediate: immediateStatus.value, | |
| 136 | - columns, | |
| 137 | - formConfig: { | |
| 138 | - labelWidth: 120, | |
| 139 | - schemas: searchFormSchema, | |
| 140 | - }, | |
| 141 | - rowKey: 'id', | |
| 142 | - useSearchForm: true, | |
| 143 | - showTableSetting: true, | |
| 144 | - bordered: true, | |
| 145 | - showIndexColumn: false, | |
| 146 | - actionColumn: { | |
| 147 | - width: 200, | |
| 148 | - title: '操作', | |
| 149 | - dataIndex: 'action', | |
| 150 | - slots: { customRender: 'action' }, | |
| 151 | - fixed: 'right', | |
| 152 | - }, | |
| 153 | - }); | |
| 154 | - const { hasBatchDelete, handleDeleteOrBatchDelete, selectionOptions } = useBatchDelete( | |
| 155 | - deviceConfigDelete, | |
| 156 | - handleSuccess, | |
| 157 | - setProps | |
| 158 | - ); | |
| 159 | - selectionOptions.rowSelection.getCheckboxProps = (record: Recordable) => { | |
| 160 | - // Demo:status为1的选择框禁用 | |
| 161 | - if (record.default === true) { | |
| 162 | - return { disabled: true }; | |
| 163 | - } else if (record.name == 'default') { | |
| 164 | - return { disabled: true }; | |
| 165 | - } else { | |
| 166 | - return { disabled: false }; | |
| 167 | - } | |
| 168 | - }; | |
| 169 | - nextTick(() => { | |
| 170 | - setProps(selectionOptions); | |
| 171 | - }); | |
| 172 | - /** | |
| 173 | - *@param url,name | |
| 174 | - **/ | |
| 175 | - function getParam(url, name) { | |
| 176 | - try { | |
| 177 | - let reg = new RegExp('(^|&)' + name + '=([^&]*)(&|$)'); | |
| 178 | - let r = url.split('?')[1].match(reg); | |
| 179 | - if (r != null) { | |
| 180 | - return r[2]; | |
| 181 | - } | |
| 182 | - return ''; //如果此处只写return;则返回的是undefined | |
| 183 | - } catch (e) { | |
| 184 | - return ''; //如果此处只写return;则返回的是undefined | |
| 185 | - } | |
| 186 | - } | |
| 187 | - getPathUrl.value = window.location.href; | |
| 188 | - const name = 'name'; | |
| 189 | - const getName = getParam(getPathUrl.value, name); | |
| 190 | - getPathUrlName.value = decodeURIComponent(getName); | |
| 191 | - | |
| 192 | - const setRowClassName = async () => { | |
| 193 | - if (getPathUrlName.value !== '') { | |
| 194 | - const { items } = await deviceConfigGetQuery({ | |
| 195 | - page: 1, | |
| 196 | - pageSize: 10, | |
| 197 | - name: getPathUrlName.value, | |
| 198 | - }); | |
| 199 | - nextTick(() => { | |
| 200 | - setTableData(items); | |
| 201 | - const { setFieldsValue, resetFields } = getForm(); | |
| 202 | - setFieldsValue({ | |
| 203 | - name: getPathUrlName.value, | |
| 204 | - }); | |
| 205 | - if (onCloseVal.value == 1) { | |
| 206 | - resetFields(); | |
| 207 | - } | |
| 208 | - }); | |
| 209 | - } else { | |
| 210 | - setTimeout(() => { | |
| 211 | - reload(); | |
| 212 | - }, 80); | |
| 213 | - } | |
| 214 | - }; | |
| 215 | - setRowClassName(); | |
| 216 | - onUnmounted(() => { | |
| 217 | - getPathUrlName.value = ''; | |
| 218 | - onCloseVal.value = 1; | |
| 219 | - }); | |
| 220 | - const tableListRef = ref< | |
| 221 | - { | |
| 222 | - title: string; | |
| 223 | - columns?: any[]; | |
| 224 | - dataSource?: any[]; | |
| 225 | - }[] | |
| 226 | - >([]); | |
| 227 | - | |
| 228 | - function loadDataSuccess(excelDataList: ExcelData[]) { | |
| 229 | - tableListRef.value = []; | |
| 230 | - console.log(excelDataList); | |
| 231 | - for (const excelData of excelDataList) { | |
| 232 | - const { | |
| 233 | - header, | |
| 234 | - results, | |
| 235 | - meta: { sheetName }, | |
| 236 | - } = excelData; | |
| 237 | - const columns: BasicColumn[] = []; | |
| 238 | - for (const title of header) { | |
| 239 | - columns.push({ title, dataIndex: title }); | |
| 240 | - } | |
| 241 | - tableListRef.value.push({ title: sheetName, dataSource: results, columns }); | |
| 242 | - } | |
| 243 | - } | |
| 244 | - //新增 | |
| 245 | - function handleCreate() { | |
| 246 | - openModal(true, { | |
| 247 | - isUpdate: false, | |
| 248 | - }); | |
| 249 | - } | |
| 250 | - //编辑 | |
| 251 | - function handleEdit(record: Recordable) { | |
| 252 | - openModal(true, { | |
| 253 | - record, | |
| 254 | - isUpdate: true, | |
| 255 | - }); | |
| 256 | - } | |
| 257 | - | |
| 258 | - const [registerDrawer, { openDrawer }] = useDrawer(); | |
| 259 | - //详情 | |
| 260 | - function handleDetailView(record: Recordable) { | |
| 261 | - openDrawer(true, { record }); | |
| 262 | - } | |
| 263 | - | |
| 264 | - function defaultHeader({ filename, bookType }: ExportModalResult) { | |
| 265 | - // 默认Object.keys(data[0])作为header | |
| 266 | - const data = exportData.value; | |
| 267 | - jsonToSheetXlsx({ | |
| 268 | - data, | |
| 269 | - filename, | |
| 270 | - write2excelOpts: { | |
| 271 | - bookType, | |
| 272 | - }, | |
| 273 | - }); | |
| 274 | - } | |
| 275 | - //导出 | |
| 276 | - const handleExport = (record: Recordable) => { | |
| 277 | - exportData.value = []; | |
| 278 | - exportData.value.push({ | |
| 279 | - createTime: record.createTime, | |
| 280 | - description: record.description, | |
| 281 | - name: record.name, | |
| 282 | - }); | |
| 283 | - nextTick(() => { | |
| 284 | - openModalExcel(); | |
| 285 | - expExcelModalRef.value?.clearFieldFunc(); | |
| 286 | - }); | |
| 287 | - }; | |
| 288 | - //导入 | |
| 289 | - function handleImport() { | |
| 290 | - console.log('record'); | |
| 291 | - } | |
| 292 | - function handleSuccess() { | |
| 293 | - reload(); | |
| 294 | - } | |
| 295 | - | |
| 296 | - const handleSetDefault = async (record: Recordable) => { | |
| 297 | - let id = record.tbProfileId; | |
| 298 | - const data = await setDeviceProfileIsDefaultApi(id, 'default', defaultObj); | |
| 299 | - if (!data) return createMessage.error('设置该产品为默认失败'); | |
| 300 | - createMessage.success('设置该产品为默认成功'); | |
| 301 | - reload(); | |
| 302 | - disabled.value = true; | |
| 2 | + import { ref } from 'vue'; | |
| 3 | + import TableMode from './TableMode.vue'; | |
| 4 | + import CardMode from './CardMode.vue'; | |
| 5 | + import { Mode } from '/@/components/Widget'; | |
| 6 | + const mode = ref(Mode.CARD); | |
| 7 | + | |
| 8 | + const handleChangeMode = (flag: Mode) => { | |
| 9 | + mode.value = flag; | |
| 303 | 10 | }; |
| 304 | 11 | </script> |
| 305 | 12 | |
| 306 | -<style lang="css"> | |
| 307 | - .devide-profiles .rowcolor { | |
| 308 | - color: red; | |
| 309 | - } | |
| 310 | - | |
| 311 | - .devide-profiles .rowcolor2 { | |
| 312 | - background: #a2c3e6; | |
| 313 | - } | |
| 314 | -</style> | |
| 13 | +<template> | |
| 14 | + <section> | |
| 15 | + <CardMode v-if="mode === Mode.CARD" :mode="mode" @change-mode="handleChangeMode" /> | |
| 16 | + <TableMode v-if="mode === Mode.TABLE" :mode="mode" @change-mode="handleChangeMode" /> | |
| 17 | + </section> | |
| 18 | +</template> | ... | ... |