| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741 |
- <template>
- <div class="page-container">
- <!-- 搜索表单 -->
- <div class="search-form">
- <el-form :model="searchForm" inline data-id="search-form">
- <el-form-item :label="t('厂家代码')">
- <el-input
- v-model.trim="searchForm.code"
- placeholder="请输入厂家代码"
- clearable
- data-id="search-code"
- @keyup.enter="handleSearch"
- />
- </el-form-item>
- <el-form-item :label="t('厂家名称')">
- <el-input
- v-model.trim="searchForm.name"
- placeholder="请输入厂家名称"
- clearable
- data-id="search-name"
- @keyup.enter="handleSearch"
- />
- </el-form-item>
- <el-form-item :label="t('启用状态')">
- <el-select v-model="searchForm.enabled" placeholder="全部" clearable data-id="search-enabled">
- <el-option label="全部" value="" />
- <el-option label="已启用" :value="true" />
- <el-option label="已禁用" :value="false" />
- </el-select>
- </el-form-item>
- <el-form-item>
- <el-button type="primary" data-id="btn-search" @click="handleSearch">
- {{ t('查询') }}
- </el-button>
- <el-button :icon="RefreshRight" data-id="btn-reset" @click="handleReset">{{ t('重置') }}</el-button>
- <el-button type="primary" data-id="btn-add-vendor" @click="handleAdd">
- {{ t('新增') }}
- </el-button>
- <!-- <el-button type="success" :icon="Setting" data-id="btn-init" @click="handleInit">
- {{ t('初始化默认数据') }}
- </el-button> -->
- </el-form-item>
- </el-form>
- </div>
- <!-- 批量操作栏 -->
- <div v-if="selectedRows.length > 0" class="batch-actions">
- <span class="batch-info">{{ t('已选择') }} {{ selectedRows.length }} {{ t('项') }}</span>
- <el-button type="danger" :icon="Delete" :loading="deleteLoading" @click="handleBatchDelete">
- {{ t('批量删除') }}
- </el-button>
- <el-button @click="clearSelection">{{ t('取消选择') }}</el-button>
- </div>
- <!-- 数据表格 -->
- <div class="table-wrapper">
- <el-table
- ref="tableRef"
- v-loading="loading"
- :data="sortedList"
- stripe
- size="default"
- data-id="vendor-table"
- height="100%"
- @selection-change="handleSelectionChange"
- @sort-change="handleSortChange"
- >
- <el-table-column type="selection" width="50" align="center" />
- <el-table-column type="index" :label="t('序号')" width="60" align="center" />
- <el-table-column prop="code" :label="t('厂家代码')" min-width="100" sortable="custom" show-overflow-tooltip />
- <el-table-column prop="name" :label="t('厂家名称')" min-width="120" sortable="custom" show-overflow-tooltip>
- <template #default="{ row }">
- <el-link type="primary" :data-id="`link-edit-${row.code}`" @click="handleEdit(row)">
- {{ row.name }}
- </el-link>
- </template>
- </el-table-column>
- <el-table-column prop="description" :label="t('描述')" min-width="150" show-overflow-tooltip />
- <el-table-column :label="t('协议支持')" min-width="200" align="center">
- <template #default="{ row }">
- <el-tag v-if="row.supportOnvif" type="success" size="small" class="protocol-tag">ONVIF</el-tag>
- <el-tag v-if="row.supportPtz" type="primary" size="small" class="protocol-tag">PTZ</el-tag>
- <el-tag v-if="row.supportIsapi" type="warning" size="small" class="protocol-tag">ISAPI</el-tag>
- <el-tag v-if="row.supportGb28181" type="info" size="small" class="protocol-tag">GB28181</el-tag>
- <el-tag v-if="row.supportAudio" type="danger" size="small" class="protocol-tag">Audio</el-tag>
- </template>
- </el-table-column>
- <el-table-column prop="defaultPort" :label="t('默认端口')" width="100" align="center">
- <template #default="{ row }">
- {{ row.defaultPort || '-' }}
- </template>
- </el-table-column>
- <el-table-column prop="defaultRtspPort" :label="t('RTSP端口')" width="100" align="center">
- <template #default="{ row }">
- {{ row.defaultRtspPort || '-' }}
- </template>
- </el-table-column>
- <el-table-column prop="enabled" :label="t('启用')" sortable="custom" width="80" align="center">
- <template #default="{ row }">
- <el-tag :type="row.enabled ? 'success' : 'info'">
- {{ row.enabled ? t('是') : t('否') }}
- </el-tag>
- </template>
- </el-table-column>
- <el-table-column prop="sortOrder" :label="t('排序')" width="80" sortable="custom" align="center">
- <template #default="{ row }">
- {{ row.sortOrder ?? 0 }}
- </template>
- </el-table-column>
- <el-table-column prop="createdAt" :label="t('创建时间')" width="160" sortable="custom" align="center">
- <template #default="{ row }">
- {{ formatDateTime(row.createdAt) }}
- </template>
- </el-table-column>
- <el-table-column :label="t('操作')" min-width="90" align="center" fixed="right">
- <template #default="{ row }">
- <el-button type="primary" link :data-id="`btn-edit-${row.code}`" @click="handleEdit(row)">
- <Icon icon="mdi:note-edit-outline" width="20" height="20" />
- </el-button>
- <el-button
- type="danger"
- link
- :disabled="deleteLoading"
- :data-id="`btn-delete-${row.code}`"
- @click="handleDelete(row)"
- >
- <Icon icon="mdi:delete" width="20" height="20" />
- </el-button>
- </template>
- </el-table-column>
- </el-table>
- </div>
- <!-- 分页 -->
- <div class="pagination-container">
- <el-pagination
- v-model:current-page="currentPage"
- v-model:page-size="pageSize"
- :page-sizes="[10, 20, 50, 100]"
- :total="total"
- layout="total, sizes, prev, pager, next, jumper"
- background
- @size-change="handleSizeChange"
- @current-change="handleCurrentChange"
- />
- </div>
- <!-- 新增/编辑弹窗 -->
- <el-dialog v-model="dialogVisible" :title="dialogTitle" width="600px" destroy-on-close data-id="dialog-vendor">
- <div class="form-container">
- <el-scrollbar>
- <el-form ref="formRef" :model="form" :rules="rules" label-width="120px" data-id="form-vendor">
- <el-form-item :label="t('厂家代码')" prop="code">
- <el-input v-model="form.code" placeholder="请输入厂家代码" :disabled="isEdit" data-id="input-code" />
- </el-form-item>
- <el-form-item :label="t('厂家名称')" prop="name">
- <el-input v-model="form.name" placeholder="请输入厂家名称" data-id="input-name" />
- </el-form-item>
- <el-form-item :label="t('描述')" prop="description">
- <el-input
- v-model="form.description"
- type="textarea"
- :rows="2"
- placeholder="请输入描述"
- data-id="input-description"
- />
- </el-form-item>
- <el-form-item :label="t('Logo URL')" prop="logoUrl">
- <el-input v-model="form.logoUrl" placeholder="请输入Logo URL" data-id="input-logo-url" />
- </el-form-item>
- <el-form-item :label="t('协议支持')">
- <el-checkbox v-model="form.supportOnvif" data-id="check-onvif">ONVIF</el-checkbox>
- <el-checkbox v-model="form.supportPtz" data-id="check-ptz">PTZ</el-checkbox>
- <el-checkbox v-model="form.supportIsapi" data-id="check-isapi">ISAPI</el-checkbox>
- <el-checkbox v-model="form.supportGb28181" data-id="check-gb28181">GB28181</el-checkbox>
- <el-checkbox v-model="form.supportAudio" data-id="check-audio">Audio</el-checkbox>
- </el-form-item>
- <el-form-item :label="t('默认分辨率')" prop="resolution">
- <el-input v-model="form.resolution" placeholder="如: 1920x1080" data-id="input-resolution" />
- </el-form-item>
- <el-row>
- <el-col :span="12">
- <el-form-item :label="t('默认端口')" prop="defaultPort">
- <el-input-number
- v-model="form.defaultPort"
- :min="1"
- :max="65535"
- placeholder="80"
- data-id="input-default-port"
- />
- </el-form-item>
- </el-col>
- <el-col :span="12">
- <el-form-item :label="t('RTSP端口')" prop="defaultRtspPort">
- <el-input-number
- v-model="form.defaultRtspPort"
- :min="1"
- :max="65535"
- placeholder="554"
- data-id="input-rtsp-port"
- />
- </el-form-item>
- </el-col>
- </el-row>
- <el-form-item :label="t('RTSP URL模板')" prop="rtspUrlTemplate">
- <el-input
- v-model="form.rtspUrlTemplate"
- type="textarea"
- :rows="2"
- placeholder="如: rtsp://{username}:{password}@{ip}:{port}/Streaming/Channels/{channel}"
- data-id="input-rtsp-template"
- />
- </el-form-item>
- <el-row>
- <el-col :span="12">
- <el-form-item :label="t('排序号')" prop="sortOrder">
- <el-input-number v-model="form.sortOrder" :min="0" :max="9999" data-id="input-sort-order" />
- </el-form-item>
- </el-col>
- <el-col :span="12">
- <el-form-item :label="t('启用状态')">
- <el-switch v-model="form.enabled" data-id="switch-enabled" />
- </el-form-item>
- </el-col>
- </el-row>
- </el-form>
- </el-scrollbar>
- </div>
- <template #footer>
- <el-button data-id="btn-cancel" @click="dialogVisible = false">{{ t('取消') }}</el-button>
- <el-button type="primary" :loading="submitLoading" data-id="btn-submit" @click="handleSubmit">
- {{ t('确定') }}
- </el-button>
- </template>
- </el-dialog>
- </div>
- </template>
- <script setup lang="ts">
- import { ref, reactive, onMounted, computed } from 'vue'
- import { ElMessage, ElMessageBox, type FormInstance, type FormRules, type TableInstance } from 'element-plus'
- import { Plus, Edit, Delete, Search, RefreshRight, Setting } from '@element-plus/icons-vue'
- import dayjs from 'dayjs'
- import { Icon } from '@iconify/vue'
- import { useI18n } from 'vue-i18n'
- import {
- listCameraVendors,
- addCameraVendor,
- updateCameraVendor,
- deleteCameraVendor,
- initCameraVendors
- } from '@/api/camera-vendor'
- import type { CameraVendorDTO, CameraVendorAddRequest, CameraVendorUpdateRequest } from '@/types'
- const { t } = useI18n({ useScope: 'global' })
- // 格式化时间
- function formatDateTime(dateStr: string | undefined): string {
- if (!dateStr) return '-'
- return dayjs(dateStr).format('YYYY-MM-DD HH:mm')
- }
- const loading = ref(false)
- const submitLoading = ref(false)
- const deleteLoading = ref(false)
- const vendorList = ref<CameraVendorDTO[]>([])
- const dialogVisible = ref(false)
- const formRef = ref<FormInstance>()
- const tableRef = ref<TableInstance>()
- // 选中的行
- const selectedRows = ref<CameraVendorDTO[]>([])
- // 排序状态
- const sortState = reactive<{
- prop: string
- order: 'ascending' | 'descending' | null
- }>({
- prop: '',
- order: null
- })
- // 搜索表单
- const searchForm = reactive<{
- code: string
- name: string
- enabled: boolean | ''
- }>({
- code: '',
- name: '',
- enabled: ''
- })
- // 分页相关
- const currentPage = ref(1)
- const pageSize = ref(20)
- const total = ref(0)
- // 排序后的数据
- const sortedList = computed(() => {
- const list = [...vendorList.value]
- if (sortState.prop && sortState.order) {
- list.sort((a, b) => {
- const aVal = a[sortState.prop as keyof CameraVendorDTO]
- const bVal = b[sortState.prop as keyof CameraVendorDTO]
- // 处理空值
- if (aVal == null && bVal == null) return 0
- if (aVal == null) return sortState.order === 'ascending' ? -1 : 1
- if (bVal == null) return sortState.order === 'ascending' ? 1 : -1
- // 比较
- let result = 0
- if (typeof aVal === 'number' && typeof bVal === 'number') {
- result = aVal - bVal
- } else if (typeof aVal === 'boolean' && typeof bVal === 'boolean') {
- result = aVal === bVal ? 0 : aVal ? 1 : -1
- } else {
- result = String(aVal).localeCompare(String(bVal))
- }
- return sortState.order === 'ascending' ? result : -result
- })
- }
- return list
- })
- const form = reactive<{
- id?: number
- code: string
- name: string
- description: string
- logoUrl: string
- supportOnvif: boolean
- supportPtz: boolean
- supportIsapi: boolean
- supportGb28181: boolean
- supportAudio: boolean
- resolution: string
- defaultPort: number | undefined
- defaultRtspPort: number | undefined
- rtspUrlTemplate: string
- enabled: boolean
- sortOrder: number
- }>({
- code: '',
- name: '',
- description: '',
- logoUrl: '',
- supportOnvif: false,
- supportPtz: false,
- supportIsapi: false,
- supportGb28181: false,
- supportAudio: false,
- resolution: '',
- defaultPort: undefined,
- defaultRtspPort: undefined,
- rtspUrlTemplate: '',
- enabled: true,
- sortOrder: 0
- })
- const isEdit = computed(() => !!form.id)
- const dialogTitle = computed(() => (isEdit.value ? t('编辑厂家') : t('新增厂家')))
- const rules: FormRules = {
- code: [{ required: true, message: t('请输入厂家代码'), trigger: 'blur' }],
- name: [{ required: true, message: t('请输入厂家名称'), trigger: 'blur' }]
- }
- async function getList() {
- loading.value = true
- try {
- // 构建查询参数
- const params: Record<string, any> = {
- page: currentPage.value,
- size: pageSize.value
- }
- // 搜索关键词
- if (searchForm.code || searchForm.name) {
- params.keyword = searchForm.code || searchForm.name
- }
- // 启用状态过滤
- if (searchForm.enabled !== '') {
- params.enabled = searchForm.enabled
- }
- // 排序
- if (sortState.prop && sortState.order) {
- params.sortBy = sortState.prop
- params.sortDir = sortState.order === 'ascending' ? 'ASC' : 'DESC'
- }
- const res = await listCameraVendors(params)
- if (res.success) {
- vendorList.value = res.data.list
- total.value = res.data.total || 0
- }
- } finally {
- loading.value = false
- }
- }
- function handleSearch() {
- currentPage.value = 1
- getList()
- }
- function handleReset() {
- searchForm.code = ''
- searchForm.name = ''
- searchForm.enabled = ''
- currentPage.value = 1
- sortState.prop = ''
- sortState.order = null
- getList()
- }
- // 排序变化处理
- function handleSortChange({ prop, order }: { prop: string; order: 'ascending' | 'descending' | null }) {
- sortState.prop = prop || ''
- sortState.order = order
- getList()
- }
- // 选择变化处理
- function handleSelectionChange(rows: CameraVendorDTO[]) {
- selectedRows.value = rows
- }
- // 清除选择
- function clearSelection() {
- tableRef.value?.clearSelection()
- }
- // 批量删除
- async function handleBatchDelete() {
- if (selectedRows.value.length === 0) return
- try {
- await ElMessageBox.confirm(`${t('确定要删除选中的')} ${selectedRows.value.length} ${t('个厂家吗?')}`, t('提示'), {
- type: 'warning'
- })
- deleteLoading.value = true
- const deletePromises = selectedRows.value.map((row) => deleteCameraVendor(row.id))
- await Promise.all(deletePromises)
- ElMessage.success(`${t('成功删除')} ${selectedRows.value.length} ${t('个厂家')}`)
- clearSelection()
- getList()
- } catch (error) {
- if (error !== 'cancel') {
- console.error(t('批量删除失败'), error)
- ElMessage.error(t('批量删除失败'))
- }
- } finally {
- deleteLoading.value = false
- }
- }
- function handleAdd() {
- Object.assign(form, {
- id: undefined,
- code: '',
- name: '',
- description: '',
- logoUrl: '',
- supportOnvif: false,
- supportPtz: false,
- supportIsapi: false,
- supportGb28181: false,
- supportAudio: false,
- resolution: '',
- defaultPort: undefined,
- defaultRtspPort: undefined,
- rtspUrlTemplate: '',
- enabled: true,
- sortOrder: 0
- })
- dialogVisible.value = true
- }
- function handleEdit(row: CameraVendorDTO) {
- Object.assign(form, {
- id: row.id,
- code: row.code,
- name: row.name,
- description: row.description || '',
- logoUrl: row.logoUrl || '',
- supportOnvif: row.supportOnvif,
- supportPtz: row.supportPtz,
- supportIsapi: row.supportIsapi,
- supportGb28181: row.supportGb28181,
- supportAudio: row.supportAudio,
- resolution: row.resolution || '',
- defaultPort: row.defaultPort,
- defaultRtspPort: row.defaultRtspPort,
- rtspUrlTemplate: row.rtspUrlTemplate || '',
- enabled: row.enabled,
- sortOrder: row.sortOrder ?? 0
- })
- dialogVisible.value = true
- }
- async function handleDelete(row: CameraVendorDTO) {
- if (deleteLoading.value) return
- try {
- await ElMessageBox.confirm(`${t('确定要删除厂家')} "${row.name}" ${t('吗?')}`, t('提示'), {
- type: 'warning'
- })
- deleteLoading.value = true
- const res = await deleteCameraVendor(row.id)
- if (res.success) {
- ElMessage.success(t('删除成功'))
- getList()
- }
- } catch (error) {
- if (error !== 'cancel') {
- console.error(t('删除失败'), error)
- }
- } finally {
- deleteLoading.value = false
- }
- }
- async function handleInit() {
- try {
- await ElMessageBox.confirm(t('确定要初始化默认厂家数据吗?这将添加预设的摄像头厂家信息。'), t('提示'), {
- type: 'warning'
- })
- loading.value = true
- const res = await initCameraVendors()
- if (res.success) {
- ElMessage.success(t('初始化成功'))
- getList()
- }
- } catch (error) {
- if (error !== 'cancel') {
- console.error(t('初始化失败'), error)
- ElMessage.error(t('初始化失败'))
- }
- } finally {
- loading.value = false
- }
- }
- async function handleSubmit() {
- if (!formRef.value) return
- await formRef.value.validate(async (valid) => {
- if (valid) {
- submitLoading.value = true
- try {
- if (isEdit.value) {
- const updateData: CameraVendorUpdateRequest = {
- id: form.id!,
- code: form.code,
- name: form.name,
- description: form.description || undefined,
- logoUrl: form.logoUrl || undefined,
- supportOnvif: form.supportOnvif,
- supportPtz: form.supportPtz,
- supportIsapi: form.supportIsapi,
- supportGb28181: form.supportGb28181,
- supportAudio: form.supportAudio,
- resolution: form.resolution || undefined,
- defaultPort: form.defaultPort,
- defaultRtspPort: form.defaultRtspPort,
- rtspUrlTemplate: form.rtspUrlTemplate || undefined,
- enabled: form.enabled,
- sortOrder: form.sortOrder
- }
- const res = await updateCameraVendor(updateData)
- if (res.success) {
- ElMessage.success(t('修改成功'))
- dialogVisible.value = false
- getList()
- }
- } else {
- const addData: CameraVendorAddRequest = {
- code: form.code,
- name: form.name,
- description: form.description || undefined,
- logoUrl: form.logoUrl || undefined,
- supportOnvif: form.supportOnvif,
- supportPtz: form.supportPtz,
- supportIsapi: form.supportIsapi,
- supportGb28181: form.supportGb28181,
- supportAudio: form.supportAudio,
- resolution: form.resolution || undefined,
- defaultPort: form.defaultPort,
- defaultRtspPort: form.defaultRtspPort,
- rtspUrlTemplate: form.rtspUrlTemplate || undefined,
- enabled: form.enabled,
- sortOrder: form.sortOrder
- }
- const res = await addCameraVendor(addData)
- if (res.success) {
- ElMessage.success(t('新增成功'))
- dialogVisible.value = false
- getList()
- }
- }
- } finally {
- submitLoading.value = false
- }
- }
- })
- }
- function handleSizeChange(val: number) {
- pageSize.value = val
- currentPage.value = 1
- getList()
- }
- function handleCurrentChange(val: number) {
- currentPage.value = val
- getList()
- }
- onMounted(() => {
- getList()
- })
- </script>
- <style lang="scss" scoped>
- .page-container {
- display: flex;
- flex-direction: column;
- box-sizing: border-box;
- }
- .form-container {
- padding: 18px 0;
- }
- // 批量操作栏
- .batch-actions {
- flex-shrink: 0;
- display: flex;
- align-items: center;
- gap: 12px;
- margin-bottom: 12px;
- padding: 12px 16px;
- background: #fef3c7;
- border: 1px solid #f59e0b;
- .batch-info {
- font-size: 14px;
- color: #92400e;
- font-weight: 500;
- }
- :deep(.el-button--danger) {
- background-color: #dc2626;
- border-color: #dc2626;
- &:hover {
- background-color: #ef4444;
- border-color: #ef4444;
- }
- }
- }
- .search-form {
- flex-shrink: 0;
- margin-bottom: 16px;
- padding: 16px 16px 4px 16px;
- background: #f5f7fa;
- :deep(.el-form-item) {
- margin-bottom: 12px;
- margin-right: 16px;
- }
- :deep(.el-input),
- :deep(.el-select) {
- width: 160px;
- }
- }
- .table-wrapper {
- flex: 1;
- min-height: 0;
- overflow: hidden;
- }
- .pagination-container {
- flex-shrink: 0;
- display: flex;
- justify-content: flex-end;
- padding-top: 16px;
- }
- // 协议标签样式
- .protocol-tag {
- margin: 2px;
- }
- // 表格样式
- :deep(.el-table) {
- --el-table-row-hover-bg-color: #f0f0ff;
- .el-table__row--striped td.el-table__cell {
- background-color: #f8f9fc;
- }
- .el-table__header th {
- background-color: #f5f7fa;
- color: #333;
- font-weight: 600;
- }
- }
- // 弹窗 Indigo 主题
- :deep(.el-dialog) {
- .el-dialog__header {
- border-bottom: 1px solid #e5e7eb;
- padding-bottom: 16px;
- }
- .el-dialog__footer {
- border-top: 1px solid #e5e7eb;
- padding-top: 16px;
- }
- .el-switch.is-checked .el-switch__core {
- background-color: #4f46e5;
- border-color: #4f46e5;
- }
- .el-checkbox__input.is-checked .el-checkbox__inner {
- background-color: #4f46e5;
- border-color: #4f46e5;
- }
- .el-checkbox__input.is-checked + .el-checkbox__label {
- color: #4f46e5;
- }
- }
- </style>
|