524 lines
14 KiB
Vue
524 lines
14 KiB
Vue
<script setup lang="ts">
|
|
import type { PageParam } from '@vben/request';
|
|
|
|
import type { IotDeviceApi } from '#/api/iot/device/device';
|
|
import type { IotDeviceGroupApi } from '#/api/iot/device/group';
|
|
import type { IotProductApi } from '#/api/iot/product/product';
|
|
|
|
import { nextTick, onMounted, ref } from 'vue';
|
|
import { useRoute, useRouter } from 'vue-router';
|
|
|
|
import { Page, useVbenModal } from '@vben/common-ui';
|
|
import { DICT_TYPE } from '@vben/constants';
|
|
import { getDictOptions } from '@vben/hooks';
|
|
import { IconifyIcon } from '@vben/icons';
|
|
import { downloadFileFromBlobPart, isEmpty } from '@vben/utils';
|
|
|
|
import {
|
|
Button,
|
|
Card,
|
|
Input,
|
|
message,
|
|
Select,
|
|
Space,
|
|
Tag,
|
|
} from 'ant-design-vue';
|
|
|
|
import { ACTION_ICON, TableAction, useVbenVxeGrid } from '#/adapter/vxe-table';
|
|
import {
|
|
deleteDevice,
|
|
deleteDeviceList,
|
|
exportDeviceExcel,
|
|
getDevicePage,
|
|
} from '#/api/iot/device/device';
|
|
import { getSimpleDeviceGroupList } from '#/api/iot/device/group';
|
|
import { getSimpleProductList } from '#/api/iot/product/product';
|
|
import { $t } from '#/locales';
|
|
|
|
import { useGridColumns } from './data';
|
|
import DeviceCardView from './modules/card-view.vue';
|
|
import DeviceForm from './modules/form.vue';
|
|
import DeviceGroupForm from './modules/group-form.vue';
|
|
import DeviceImportForm from './modules/import-form.vue';
|
|
|
|
/** IoT 设备列表 */
|
|
defineOptions({ name: 'IoTDevice' });
|
|
|
|
const route = useRoute();
|
|
const router = useRouter();
|
|
const products = ref<IotProductApi.Product[]>([]);
|
|
const deviceGroups = ref<IotDeviceGroupApi.DeviceGroup[]>([]);
|
|
const viewMode = ref<'card' | 'list'>('card');
|
|
const cardViewRef = ref();
|
|
const checkedIds = ref<number[]>([]);
|
|
|
|
/** 判断是否为列表视图 */
|
|
const isListView = () => viewMode.value === 'list';
|
|
|
|
const [DeviceFormModal, deviceFormModalApi] = useVbenModal({
|
|
connectedComponent: DeviceForm,
|
|
destroyOnClose: true,
|
|
});
|
|
|
|
const [DeviceGroupFormModal, deviceGroupFormModalApi] = useVbenModal({
|
|
connectedComponent: DeviceGroupForm,
|
|
destroyOnClose: true,
|
|
});
|
|
|
|
const [DeviceImportFormModal, deviceImportFormModalApi] = useVbenModal({
|
|
connectedComponent: DeviceImportForm,
|
|
destroyOnClose: true,
|
|
});
|
|
|
|
const queryParams = ref<Partial<PageParam>>({
|
|
deviceName: '',
|
|
nickname: '',
|
|
productId: undefined,
|
|
deviceType: undefined,
|
|
status: undefined,
|
|
groupId: undefined,
|
|
}); // 搜索参数
|
|
|
|
/** 搜索 */
|
|
function handleSearch() {
|
|
if (viewMode.value === 'list') {
|
|
gridApi.formApi.setValues(queryParams.value);
|
|
}
|
|
gridApi.query();
|
|
}
|
|
|
|
/** 重置搜索 */
|
|
function handleReset() {
|
|
queryParams.value.deviceName = '';
|
|
queryParams.value.nickname = '';
|
|
queryParams.value.productId = undefined;
|
|
queryParams.value.deviceType = undefined;
|
|
queryParams.value.status = undefined;
|
|
queryParams.value.groupId = undefined;
|
|
handleSearch();
|
|
}
|
|
|
|
/** 刷新表格 */
|
|
function handleRefresh() {
|
|
gridApi.query();
|
|
}
|
|
|
|
/** 视图切换 */
|
|
async function handleViewModeChange(mode: 'card' | 'list') {
|
|
if (viewMode.value === mode) {
|
|
return; // 如果已经是目标视图,不需要切换
|
|
}
|
|
viewMode.value = mode;
|
|
// 等待视图更新后再触发查询
|
|
await nextTick();
|
|
gridApi.query();
|
|
}
|
|
|
|
/** 导出表格 */
|
|
async function handleExport() {
|
|
const data = await exportDeviceExcel({
|
|
...queryParams.value,
|
|
pageNo: 1,
|
|
pageSize: 999_999,
|
|
} as PageParam);
|
|
downloadFileFromBlobPart({ fileName: '物联网设备.xls', source: data });
|
|
}
|
|
|
|
/** 打开设备详情 */
|
|
function openDetail(id: number) {
|
|
router.push({ name: 'IoTDeviceDetail', params: { id } });
|
|
}
|
|
|
|
/** 跳转到产品详情页面 */
|
|
function openProductDetail(productId: number) {
|
|
router.push({ name: 'IoTProductDetail', params: { id: productId } });
|
|
}
|
|
|
|
/** 打开物模型数据 */
|
|
function openModel(id: number) {
|
|
router.push({
|
|
name: 'IoTDeviceDetail',
|
|
params: { id },
|
|
query: { tab: 'model' },
|
|
});
|
|
}
|
|
|
|
/** 新增设备 */
|
|
function handleCreate() {
|
|
deviceFormModalApi.setData(null).open();
|
|
}
|
|
|
|
/** 编辑设备 */
|
|
function handleEdit(row: IotDeviceApi.Device) {
|
|
deviceFormModalApi.setData(row).open();
|
|
}
|
|
|
|
/** 删除设备 */
|
|
async function handleDelete(row: IotDeviceApi.Device) {
|
|
const hideLoading = message.loading({
|
|
content: $t('ui.actionMessage.deleting', [row.deviceName]),
|
|
duration: 0,
|
|
});
|
|
try {
|
|
await deleteDevice(row.id!);
|
|
message.success($t('ui.actionMessage.deleteSuccess', [row.deviceName]));
|
|
handleRefresh();
|
|
} finally {
|
|
hideLoading();
|
|
}
|
|
}
|
|
|
|
/** 批量删除设备 */
|
|
async function handleDeleteBatch() {
|
|
if (checkedIds.value.length === 0) {
|
|
message.warning('请选择要删除的设备');
|
|
return;
|
|
}
|
|
const hideLoading = message.loading({
|
|
content: $t('ui.actionMessage.deletingBatch'),
|
|
duration: 0,
|
|
});
|
|
try {
|
|
await deleteDeviceList(checkedIds.value);
|
|
message.success($t('ui.actionMessage.deleteSuccess'));
|
|
checkedIds.value = [];
|
|
handleRefresh();
|
|
} finally {
|
|
hideLoading();
|
|
}
|
|
}
|
|
|
|
/** 添加到分组 */
|
|
function handleAddToGroup() {
|
|
if (checkedIds.value.length === 0) {
|
|
message.warning('请选择要添加到分组的设备');
|
|
return;
|
|
}
|
|
deviceGroupFormModalApi.setData(checkedIds.value).open();
|
|
}
|
|
|
|
/** 设备导入 */
|
|
function handleImport() {
|
|
deviceImportFormModalApi.open();
|
|
}
|
|
|
|
function handleRowCheckboxChange({
|
|
records,
|
|
}: {
|
|
records: IotDeviceApi.Device[];
|
|
}) {
|
|
checkedIds.value = records.map((item) => item.id!);
|
|
}
|
|
|
|
const [Grid, gridApi] = useVbenVxeGrid<IotDeviceApi.Device>({
|
|
gridOptions: {
|
|
checkboxConfig: {
|
|
highlight: true,
|
|
reserve: true,
|
|
},
|
|
columns: useGridColumns(),
|
|
height: 'auto',
|
|
keepSource: true,
|
|
proxyConfig: {
|
|
ajax: {
|
|
query: async ({
|
|
page,
|
|
}: {
|
|
page: { currentPage: number; pageSize: number };
|
|
}) => {
|
|
return await getDevicePage({
|
|
pageNo: page.currentPage,
|
|
pageSize: page.pageSize,
|
|
...queryParams.value,
|
|
} as PageParam);
|
|
},
|
|
},
|
|
},
|
|
rowConfig: {
|
|
keyField: 'id',
|
|
isHover: true,
|
|
},
|
|
toolbarConfig: {
|
|
refresh: true,
|
|
search: true,
|
|
},
|
|
},
|
|
gridEvents: {
|
|
checkboxAll: handleRowCheckboxChange,
|
|
checkboxChange: handleRowCheckboxChange,
|
|
},
|
|
});
|
|
|
|
/** 包装 gridApi.query() 方法,统一列表视图和卡片视图的查询接口 */
|
|
const originalQuery = gridApi.query.bind(gridApi);
|
|
gridApi.query = async (params?: Record<string, any>) => {
|
|
if (viewMode.value === 'list') {
|
|
return await originalQuery(params);
|
|
} else {
|
|
// 卡片视图:调用卡片组件的 query 方法
|
|
cardViewRef.value?.query();
|
|
}
|
|
};
|
|
|
|
/** 初始化 */
|
|
onMounted(async () => {
|
|
// 获取产品列表
|
|
products.value = await getSimpleProductList();
|
|
// 获取分组列表
|
|
deviceGroups.value = await getSimpleDeviceGroupList();
|
|
|
|
// 处理 productId 参数
|
|
const { productId } = route.query;
|
|
if (productId) {
|
|
queryParams.value.productId = Number(productId);
|
|
// 自动触发搜索
|
|
handleSearch();
|
|
}
|
|
});
|
|
</script>
|
|
|
|
<template>
|
|
<Page auto-content-height>
|
|
<DeviceFormModal @success="handleRefresh" />
|
|
<DeviceGroupFormModal @success="handleRefresh" />
|
|
<DeviceImportFormModal @success="handleRefresh" />
|
|
|
|
<!-- 统一搜索工具栏 -->
|
|
<Card :body-style="{ padding: '16px' }" class="mb-4">
|
|
<!-- 搜索表单 -->
|
|
<div class="mb-3 flex flex-wrap items-center gap-3">
|
|
<Select
|
|
v-model:value="queryParams.productId"
|
|
placeholder="请选择产品"
|
|
allow-clear
|
|
style="width: 200px"
|
|
>
|
|
<Select.Option
|
|
v-for="product in products"
|
|
:key="product.id"
|
|
:value="product.id"
|
|
>
|
|
{{ product.name }}
|
|
</Select.Option>
|
|
</Select>
|
|
<Input
|
|
v-model:value="queryParams.deviceName"
|
|
placeholder="请输入 DeviceName"
|
|
allow-clear
|
|
style="width: 200px"
|
|
@press-enter="handleSearch"
|
|
/>
|
|
<Input
|
|
v-model:value="queryParams.nickname"
|
|
placeholder="请输入备注名称"
|
|
allow-clear
|
|
style="width: 200px"
|
|
@press-enter="handleSearch"
|
|
/>
|
|
<Select
|
|
v-model:value="queryParams.deviceType"
|
|
placeholder="请选择设备类型"
|
|
allow-clear
|
|
style="width: 200px"
|
|
>
|
|
<Select.Option
|
|
v-for="dict in getDictOptions(
|
|
DICT_TYPE.IOT_PRODUCT_DEVICE_TYPE,
|
|
'number',
|
|
)"
|
|
:key="dict.value"
|
|
:value="dict.value"
|
|
>
|
|
{{ dict.label }}
|
|
</Select.Option>
|
|
</Select>
|
|
<Select
|
|
v-model:value="queryParams.status"
|
|
placeholder="请选择设备状态"
|
|
allow-clear
|
|
style="width: 200px"
|
|
>
|
|
<Select.Option
|
|
v-for="dict in getDictOptions(DICT_TYPE.IOT_DEVICE_STATE, 'number')"
|
|
:key="dict.value"
|
|
:value="dict.value"
|
|
>
|
|
{{ dict.label }}
|
|
</Select.Option>
|
|
</Select>
|
|
<Select
|
|
v-model:value="queryParams.groupId"
|
|
placeholder="请选择设备分组"
|
|
allow-clear
|
|
style="width: 200px"
|
|
>
|
|
<Select.Option
|
|
v-for="group in deviceGroups"
|
|
:key="group.id"
|
|
:value="group.id"
|
|
>
|
|
{{ group.name }}
|
|
</Select.Option>
|
|
</Select>
|
|
<Button type="primary" @click="handleSearch">
|
|
<IconifyIcon icon="ant-design:search-outlined" class="mr-1" />
|
|
{{ $t('common.search') }}
|
|
</Button>
|
|
<Button @click="handleReset">
|
|
<IconifyIcon icon="ant-design:reload-outlined" class="mr-1" />
|
|
{{ $t('common.reset') }}
|
|
</Button>
|
|
</div>
|
|
|
|
<!-- 操作按钮 -->
|
|
<div class="flex items-center justify-between">
|
|
<TableAction
|
|
:actions="[
|
|
{
|
|
label: $t('ui.actionTitle.create', ['设备']),
|
|
type: 'primary',
|
|
icon: ACTION_ICON.ADD,
|
|
auth: ['iot:device:create'],
|
|
onClick: handleCreate,
|
|
},
|
|
{
|
|
label: $t('ui.actionTitle.export'),
|
|
type: 'primary',
|
|
icon: ACTION_ICON.DOWNLOAD,
|
|
auth: ['iot:device:export'],
|
|
onClick: handleExport,
|
|
},
|
|
{
|
|
label: $t('ui.actionTitle.import'),
|
|
type: 'primary',
|
|
icon: ACTION_ICON.UPLOAD,
|
|
auth: ['iot:device:import'],
|
|
onClick: handleImport,
|
|
},
|
|
{
|
|
label: '添加到分组',
|
|
type: 'primary',
|
|
icon: 'ant-design:folder-add-outlined',
|
|
auth: ['iot:device:update'],
|
|
ifShow: isListView,
|
|
disabled: isEmpty(checkedIds),
|
|
onClick: handleAddToGroup,
|
|
},
|
|
{
|
|
label: $t('ui.actionTitle.deleteBatch'),
|
|
type: 'primary',
|
|
danger: true,
|
|
icon: ACTION_ICON.DELETE,
|
|
auth: ['iot:device:delete'],
|
|
ifShow: isListView,
|
|
disabled: isEmpty(checkedIds),
|
|
onClick: handleDeleteBatch,
|
|
},
|
|
]"
|
|
/>
|
|
<!-- 视图切换 -->
|
|
<Space :size="4">
|
|
<Button
|
|
:type="viewMode === 'card' ? 'primary' : 'default'"
|
|
@click="handleViewModeChange('card')"
|
|
>
|
|
<IconifyIcon icon="ant-design:appstore-outlined" />
|
|
</Button>
|
|
<Button
|
|
:type="viewMode === 'list' ? 'primary' : 'default'"
|
|
@click="handleViewModeChange('list')"
|
|
>
|
|
<IconifyIcon icon="ant-design:unordered-list-outlined" />
|
|
</Button>
|
|
</Space>
|
|
</div>
|
|
</Card>
|
|
|
|
<!-- 列表视图 -->
|
|
<Grid table-title="设备列表" v-show="viewMode === 'list'">
|
|
<template #product="{ row }">
|
|
<a
|
|
class="cursor-pointer text-primary"
|
|
@click="openProductDetail(row.productId)"
|
|
>
|
|
{{ products.find((p) => p.id === row.productId)?.name || '-' }}
|
|
</a>
|
|
</template>
|
|
<template #groups="{ row }">
|
|
<template v-if="row.groupIds?.length">
|
|
<Tag
|
|
v-for="groupId in row.groupIds"
|
|
:key="groupId"
|
|
size="small"
|
|
class="mr-1"
|
|
>
|
|
{{ deviceGroups.find((g) => g.id === groupId)?.name }}
|
|
</Tag>
|
|
</template>
|
|
<span v-else>-</span>
|
|
</template>
|
|
<template #actions="{ row }">
|
|
<TableAction
|
|
:actions="[
|
|
{
|
|
label: $t('common.detail'),
|
|
type: 'link',
|
|
onClick: openDetail.bind(null, row.id!),
|
|
},
|
|
{
|
|
label: '日志',
|
|
type: 'link',
|
|
onClick: openModel.bind(null, row.id!),
|
|
},
|
|
{
|
|
label: $t('common.edit'),
|
|
type: 'link',
|
|
icon: ACTION_ICON.EDIT,
|
|
onClick: handleEdit.bind(null, row),
|
|
},
|
|
{
|
|
label: $t('common.delete'),
|
|
type: 'link',
|
|
danger: true,
|
|
icon: ACTION_ICON.DELETE,
|
|
popConfirm: {
|
|
title: $t('ui.actionMessage.deleteConfirm', [row.deviceName]),
|
|
confirm: handleDelete.bind(null, row),
|
|
},
|
|
},
|
|
]"
|
|
/>
|
|
</template>
|
|
</Grid>
|
|
|
|
<!-- 卡片视图 -->
|
|
<DeviceCardView
|
|
v-show="viewMode === 'card'"
|
|
ref="cardViewRef"
|
|
:products="products"
|
|
:device-groups="deviceGroups"
|
|
:search-params="{
|
|
deviceName: queryParams.deviceName || '',
|
|
nickname: queryParams.nickname || '',
|
|
productId: queryParams.productId,
|
|
deviceType: queryParams.deviceType,
|
|
status: queryParams.status,
|
|
groupId: queryParams.groupId,
|
|
}"
|
|
@create="handleCreate"
|
|
@edit="handleEdit"
|
|
@delete="handleDelete"
|
|
@detail="openDetail"
|
|
@model="openModel"
|
|
@product-detail="openProductDetail"
|
|
/>
|
|
</Page>
|
|
</template>
|
|
|
|
<style scoped>
|
|
/* 隐藏 VxeGrid 自带的搜索表单区域 */
|
|
:deep(.vxe-grid--form-wrapper) {
|
|
display: none !important;
|
|
}
|
|
</style>
|