| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719 |
- <template>
- <div class="page-container">
- <!-- 搜索表单 -->
- <div class="search-form">
- <el-form :model="searchForm" inline data-id="search-form">
- <el-form-item>
- <el-input
- v-model.trim="searchForm.lssId"
- :placeholder="t('LSS ID')"
- clearable
- data-id="search-keyword"
- @keyup.enter="handleSearch"
- />
- </el-form-item>
- <el-form-item>
- <el-input
- v-model.trim="searchForm.lssName"
- :placeholder="t('名称')"
- clearable
- data-id="search-keyword"
- @keyup.enter="handleSearch"
- />
- </el-form-item>
- <el-form-item>
- <el-select v-model="searchForm.status" :placeholder="t('心跳')" clearable data-id="search-enabled">
- <el-option :label="t('全部')" value="" />
- <el-option :label="t('活跃')" value="active" />
- <el-option :label="t('待机')" value="hold" />
- <el-option :label="t('离线')" value="dead" />
- </el-select>
- </el-form-item>
- <el-form-item>
- <el-button type="primary" data-id="btn-search" @click="handleSearch">
- <Icon icon="mdi:magnify" width="16" height="16" style="margin-right: 4px" />
- {{ t('查询') }}
- </el-button>
- <el-button type="info" data-id="btn-reset" @click="handleReset">
- <Icon icon="mdi:refresh" width="16" height="16" style="margin-right: 4px" />
- {{ t('重置') }}
- </el-button>
- </el-form-item>
- </el-form>
- </div>
- <!-- 数据表格 -->
- <div class="table-wrapper">
- <el-table
- ref="tableRef"
- v-loading="loading"
- :data="lssList"
- stripe
- size="default"
- data-id="lss-table"
- height="100%"
- @sort-change="handleSortChange"
- >
- <el-table-column prop="lssId" :label="t('LSS ID')" min-width="120" sortable="custom" show-overflow-tooltip />
- <el-table-column prop="lssName" :label="t('名称')" min-width="140" sortable="custom" show-overflow-tooltip />
- <el-table-column prop="address" :label="t('地址')" min-width="180" sortable="custom" show-overflow-tooltip />
- <el-table-column prop="ip" :label="t('IP')" min-width="180" sortable="custom" show-overflow-tooltip />
- <el-table-column :label="t('心跳')" width="220" align="center">
- <template #default="{ row }">
- <span :class="getHeartbeatClass(row.status)">
- {{ t(row.status) || '-' }}
- </span>
- | {{ formatTime(row.lastHeartbeatAt) }}
- </template>
- </el-table-column>
- <el-table-column :label="t('设备列表')" align="center">
- <template #default="{ row }">
- <el-button type="primary" link @click="handleEdit(row, 'camera')">
- <Icon icon="mdi:cctv" width="20" height="20" />
- </el-button>
- </template>
- </el-table-column>
- <el-table-column :label="t('ably')" align="center" fixed="right">
- <template #default="{ row }">
- {{ row.ablyClientId }}
- </template>
- </el-table-column>
- <el-table-column :label="t('操作')" width="130" align="center" fixed="right">
- <template #default="{ row }">
- <el-button type="primary" link @click="handleEdit(row, 'detail')">
- <Icon icon="mdi:note-edit-outline" width="20" height="20" />
- </el-button>
- <el-button type="primary" link @click="handleScanDevices(row)">
- <Icon icon="mdi:radar" width="20" height="20" />
- </el-button>
- <el-button type="danger" link @click="handleDelete(row)">
- <Icon icon="mdi:delete" width="20" height="20" />
- </el-button>
- </template>
- </el-table-column>
- </el-table>
- </div>
- <!-- LSS 详情抽屉 -->
- <el-drawer v-model="detailDrawerVisible" :title="t('LSS 节点详情')" direction="rtl" size="500px" destroy-on-close>
- <el-descriptions :column="1" border>
- <el-descriptions-item :label="t('LSS ID')">{{ currentLss?.lssId }}</el-descriptions-item>
- <el-descriptions-item :label="t('名称')">{{ currentLss?.lssName }}</el-descriptions-item>
- <el-descriptions-item :label="t('地址')">{{ currentLss?.address }}</el-descriptions-item>
- <el-descriptions-item :label="t('机器 ID')">{{ currentLss?.machineId || '-' }}</el-descriptions-item>
- <el-descriptions-item :label="t('状态')">
- <el-tag :type="getStatusTagType(currentLss?.status)" size="small">
- {{ formatStatus(currentLss?.status) }}
- </el-tag>
- </el-descriptions-item>
- <el-descriptions-item :label="t('任务数')">
- {{ currentLss?.currentTasks }} / {{ currentLss?.maxTasks }}
- </el-descriptions-item>
- <el-descriptions-item :label="t('FFmpeg 版本')">{{ currentLss?.ffmpegVersion || '-' }}</el-descriptions-item>
- <el-descriptions-item :label="t('系统信息')">{{ currentLss?.systemInfo || '-' }}</el-descriptions-item>
- <el-descriptions-item :label="t('启用状态')">
- <el-tag :type="currentLss?.enabled ? 'success' : 'info'" size="small">
- {{ currentLss?.enabled ? '已启用' : '已禁用' }}
- </el-tag>
- </el-descriptions-item>
- <el-descriptions-item :label="t('创建时间')">{{ formatTime(currentLss?.createdAt) }}</el-descriptions-item>
- <el-descriptions-item :label="t('更新时间')">{{ formatTime(currentLss?.updatedAt) }}</el-descriptions-item>
- </el-descriptions>
- </el-drawer>
- <!-- LSS 编辑抽屉 -->
- <el-drawer
- v-model="lssEditDrawerVisible"
- direction="rtl"
- :size="editDrawerSize"
- :with-header="false"
- destroy-on-close
- class="lss-edit-drawer"
- >
- <div class="drawer-content">
- <!-- 顶部 Tabs -->
- <el-tabs v-model="editActiveTab" class="drawer-tabs">
- <el-tab-pane :label="t('LSS详情')" name="detail" />
- <el-tab-pane :label="t('摄像头列表')" name="camera" />
- <el-tab-pane :label="t('推币机列表')" name="pusher" />
- </el-tabs>
- <div class="drawer-body">
- <!-- LSS 详情 Tab -->
- <div v-show="editActiveTab === 'detail'" class="lss-detail-form">
- <el-form ref="lssEditFormRef" :model="lssEditForm" label-width="80px" label-position="left">
- <el-form-item :label="t('LSS ID') + ':'">
- <span class="form-value">{{ currentLss?.lssId }}</span>
- </el-form-item>
- <el-form-item :label="t('名称') + ':'" prop="lssName">
- <el-input v-model="lssEditForm.lssName" :placeholder="t('请输入名称')" style="width: 180px" />
- </el-form-item>
- <el-form-item :label="t('地址') + ':'" prop="address">
- <el-input v-model="lssEditForm.address" :placeholder="t('请输入地址')" />
- </el-form-item>
- <el-form-item :label="t('IP') + ':'">
- <span class="form-value">{{ currentLss?.ip }}</span>
- </el-form-item>
- <el-form-item :label="t('心跳') + ':'">
- <span class="heartbeat-status" :class="getHeartbeatClass(currentLss?.status)">
- {{ formatHeartbeat(currentLss) }}
- <span class="heartbeat-dot" :class="getHeartbeatClass(currentLss?.status)"></span>
- </span>
- <el-tooltip placement="right" effect="light">
- <template #content>
- <div class="heartbeat-tooltip">
- <div class="tooltip-title">{{ t('心跳状态') }}:</div>
- <div>{{ t('活跃') }} - {{ t('持续返回中,并且频繁') }}</div>
- <div>{{ t('待机') }} - {{ t('五分钟内有返回') }}</div>
- <div>{{ t('离线') }} - {{ t('五分钟内没有返回') }}</div>
- <div class="tooltip-format">{{ t('表现形式为') }}:</div>
- <div class="tooltip-example">{{ t('Status') }} [yy-mm-dd 00:00:00]</div>
- </div>
- </template>
- <Icon icon="mdi:help-circle" class="heartbeat-info-icon" width="16" height="16" />
- </el-tooltip>
- </el-form-item>
- <el-form-item :label="t('ably') + ':'" prop="ably">
- <div class="textarea-wrapper">
- <el-input
- disabled
- type="textarea"
- :rows="8"
- v-model="lssEditForm.ably"
- :placeholder="t('请输入ably信息')"
- maxlength="1000"
- show-word-limit
- />
- </div>
- </el-form-item>
- </el-form>
- </div>
- <!-- 摄像头列表 Tab -->
- <div v-show="editActiveTab === 'camera'" v-loading="cameraLoading" class="tab-content-wrapper">
- <div class="camera-toolbar">
- <el-form :model="cameraSearchForm" inline>
- <el-form-item>
- <el-input
- v-model.trim="cameraSearchForm.cameraId"
- :placeholder="t('设备ID')"
- clearable
- style="width: 200px"
- @keyup.enter="handleCameraSearch"
- />
- </el-form-item>
- <el-form-item>
- <el-input
- v-model.trim="cameraSearchForm.cameraName"
- :placeholder="t('名称')"
- clearable
- style="width: 200px"
- @keyup.enter="handleCameraSearch"
- />
- </el-form-item>
- <el-form-item>
- <el-select v-model="cameraSearchForm.status" :placeholder="t('状态')" clearable style="width: 120px">
- <el-option :label="t('全部')" value="" />
- <el-option label="active" value="active" />
- <el-option label="hold" value="hold" />
- <el-option label="dead" value="dead" />
- </el-select>
- </el-form-item>
- <el-form-item>
- <el-button type="primary" @click="handleCameraSearch">
- <Icon icon="mdi:magnify" width="16" height="16" style="margin-right: 4px" />
- {{ t('查询') }}
- </el-button>
- <el-button type="info" @click="handleCameraReset">
- <Icon icon="mdi:refresh" width="16" height="16" style="margin-right: 4px" />
- {{ t('重置') }}
- </el-button>
- <el-button type="primary" @click="handleAddCamera">
- <Icon icon="mdi:plus" width="16" height="16" style="margin-right: 4px" />
- {{ t('新增') }}
- </el-button>
- </el-form-item>
- </el-form>
- </div>
- <el-table :data="cameraList" stripe size="small" border>
- <template #empty>
- <el-empty :description="t('暂无关联设备')" />
- </template>
- <el-table-column prop="cameraId" :label="t('设备ID')" min-width="100" show-overflow-tooltip />
- <el-table-column prop="cameraName" :label="t('名称')" min-width="100" show-overflow-tooltip />
- <el-table-column :label="t('状态(心跳)')" min-width="140">
- <template #default="{ row }">
- <span :class="['status-text', row.status === 'active' ? 'status-active' : 'status-dead']">
- {{ formatCameraStatus(row) }}
- </span>
- </template>
- </el-table-column>
- <el-table-column :label="t('参数配置')" min-width="80" align="center">
- <template #default="{ row }">
- <el-button type="primary" link @click="handleViewConfig(row)">{{ t('查看') }}</el-button>
- </template>
- </el-table-column>
- <el-table-column :label="t('运行参数')" min-width="80" align="center">
- <template #default="{ row }">
- <el-button type="primary" link @click="handleViewRunParams(row)">{{ t('查看') }}</el-button>
- </template>
- </el-table-column>
- <el-table-column prop="brand" :label="t('厂商')" min-width="90">
- <template #default="{ row }">
- {{ formatBrand(row.vendorName) }}
- </template>
- </el-table-column>
- <el-table-column prop="model" :label="t('型号')" min-width="130" show-overflow-tooltip />
- <el-table-column :label="t('添加时间')" min-width="140">
- <template #default="{ row }">
- {{ formatTime(row.createdAt) }}
- </template>
- </el-table-column>
- <el-table-column :label="t('设备控制')" width="130" align="center">
- <template #default="{ row }">
- <el-button type="primary" link @click="handleEditCamera(row)">
- <Icon icon="mdi:note-edit-outline" width="20" height="20" />
- </el-button>
- <!-- :tooltip="t('查看Cloudflare Stream')" -->
- <el-button type="primary" link @click="handleViewCamera(row)">
- <Icon
- icon="mdi:controller-right"
- :class="['crosshairs-btn', { active: row.streamSn }]"
- width="20"
- height="20"
- />
- </el-button>
- <el-button type="danger" link @click="handleDeleteCamera(row)">
- <Icon icon="mdi:delete" width="20" height="20" />
- </el-button>
- </template>
- </el-table-column>
- </el-table>
- <div class="camera-pagination">
- <el-pagination
- v-model:current-page="cameraCurrentPage"
- v-model:page-size="cameraPageSize"
- :page-sizes="[10, 15, 20, 50]"
- :total="cameraTotal"
- layout="total, sizes, prev, pager, next"
- background
- small
- @size-change="handleCameraSizeChange"
- @current-change="handleCameraPageChange"
- />
- </div>
- </div>
- <!-- 推币机列表 Tab -->
- <div v-show="editActiveTab === 'pusher'" class="tab-content">
- <el-empty :description="t('暂无推币机数据')" />
- </div>
- </div>
- <div v-show="editActiveTab === 'detail'" class="drawer-footer">
- <el-button @click="lssEditDrawerVisible = false">{{ t('取消') }}</el-button>
- <el-button type="primary" :loading="lssUpdating" @click="handleUpdateLss">{{ t('更新') }}</el-button>
- </div>
- </div>
- </el-drawer>
- <!-- 设备列表抽屉 -->
- <el-drawer
- v-model="cameraDrawerVisible"
- :title="`${t('设备列表')} - ${currentLss?.lssName || ''}`"
- direction="rtl"
- size="80%"
- destroy-on-close
- class="device-drawer"
- >
- <el-tabs v-model="deviceActiveTab" class="device-tabs">
- <el-tab-pane :label="t('摄像头列表')" name="camera">
- <div v-loading="cameraLoading" class="tab-content-wrapper">
- <div class="camera-toolbar">
- <el-form :model="cameraSearchForm" inline>
- <el-form-item>
- <el-input
- v-model.trim="cameraSearchForm.cameraId"
- :placeholder="t('设备ID')"
- clearable
- style="width: 150px"
- @keyup.enter="handleCameraSearch"
- />
- </el-form-item>
- <el-form-item>
- <el-input
- v-model.trim="cameraSearchForm.cameraName"
- :placeholder="t('设备名称')"
- clearable
- style="width: 150px"
- @keyup.enter="handleCameraSearch"
- />
- </el-form-item>
- <el-form-item>
- <el-select v-model="cameraSearchForm.status" :placeholder="t('状态')" clearable style="width: 120px">
- <el-option :label="t('全部')" value="" />
- <el-option label="active" value="active" />
- <el-option label="hold" value="hold" />
- <el-option label="dead" value="dead" />
- </el-select>
- </el-form-item>
- <el-form-item>
- <el-button type="primary" @click="handleCameraSearch">
- <Icon icon="mdi:magnify" width="16" height="16" style="margin-right: 4px" />
- {{ t('查询') }}
- </el-button>
- <el-button type="info" @click="handleCameraReset">
- <Icon icon="mdi:refresh" width="16" height="16" style="margin-right: 4px" />
- {{ t('重置') }}
- </el-button>
- <el-button type="primary" @click="handleAddCamera">
- <Icon icon="mdi:plus" width="16" height="16" style="margin-right: 4px" />
- {{ t('新增') }}
- </el-button>
- </el-form-item>
- </el-form>
- </div>
- <el-table :data="cameraList" stripe size="small" border>
- <template #empty>
- <el-empty :description="t('暂无关联设备')" />
- </template>
- <!-- <el-table-column prop="ip" label="本地IP" min-width="110" /> -->
- <el-table-column prop="cameraId" :label="t('设备ID')" min-width="100" show-overflow-tooltip />
- <el-table-column prop="cameraName" :label="t('名称')" min-width="100" show-overflow-tooltip />
- <el-table-column :label="t('状态(心跳)')" min-width="140">
- <template #default="{ row }">
- <span :class="['status-text', row.status === 'active' ? 'status-active' : 'status-dead']">
- {{ formatCameraStatus(row) }}
- </span>
- </template>
- </el-table-column>
- <el-table-column :label="t('参数配置')" min-width="80" align="center">
- <template #default="{ row }">
- <el-button type="primary" link @click="handleViewConfig(row)">查看</el-button>
- </template>
- </el-table-column>
- <el-table-column :label="t('运行参数')" min-width="80" align="center">
- <template #default="{ row }">
- <el-button type="primary" link @click="handleViewRunParams(row)">查看</el-button>
- </template>
- </el-table-column>
- <el-table-column prop="vendorName" :label="t('厂商')" min-width="90">
- <template #default="{ row }">
- {{ formatBrand(row.vendorName) }}
- </template>
- </el-table-column>
- <el-table-column prop="model" :label="t('型号')" min-width="130" show-overflow-tooltip />
- <el-table-column :label="t('添加时间')" min-width="140">
- <template #default="{ row }">
- {{ formatTime(row.createdAt) }}
- </template>
- </el-table-column>
- <el-table-column :label="t('设备控制')" min-width="100" align="center" fixed="right">
- <template #default="{ row }">
- <el-button type="primary" link @click="handleEditCamera(row)">
- <Icon icon="mdi:note-edit-outline" width="20" height="20" />
- </el-button>
- <el-button type="danger" link @click="handleDeleteCamera(row)">
- <Icon icon="mdi:delete" width="20" height="20" />
- </el-button>
- <el-button link :class="['crosshairs-btn', { active: !row.streamSn }]" @click="handleViewCamera(row)">
- <Icon icon="mdi:crosshairs" />
- </el-button>
- </template>
- </el-table-column>
- </el-table>
- <div v-if="cameraList.length > 0" class="camera-count">共 {{ cameraList.length }} 个设备</div>
- </div>
- </el-tab-pane>
- <el-tab-pane :label="t('推币机列表')" name="pusher">
- <div class="tab-content-wrapper">
- <div class="camera-toolbar">
- <el-form inline>
- <el-form-item>
- <el-input :placeholder="t('设备ID / 名称')" clearable style="width: 200px" />
- </el-form-item>
- <el-form-item>
- <el-select :placeholder="t('状态')" clearable style="width: 120px">
- <el-option :label="t('全部')" value="" />
- <el-option :label="t('在线')" value="ONLINE" />
- <el-option :label="t('离线')" value="OFFLINE" />
- </el-select>
- </el-form-item>
- <el-form-item>
- <el-button type="primary">
- <Icon icon="mdi:magnify" width="16" height="16" style="margin-right: 4px" />
- {{ t('查询') }}
- </el-button>
- <el-button type="info">
- <Icon icon="mdi:refresh" width="16" height="16" style="margin-right: 4px" />
- {{ t('重置') }}
- </el-button>
- </el-form-item>
- </el-form>
- <el-button type="primary">
- <Icon icon="mdi:plus" width="16" height="16" style="margin-right: 4px" />
- {{ t('新增') }}
- </el-button>
- </div>
- <el-empty :description="t('暂无推币机数据')" />
- </div>
- </el-tab-pane>
- <el-tab-pane :label="t('其他设备')" name="other">
- <div class="tab-content-wrapper">
- <div class="camera-toolbar">
- <el-form inline>
- <el-form-item>
- <el-input :placeholder="t('设备ID / 名称')" clearable style="width: 200px" />
- </el-form-item>
- <el-form-item>
- <el-select :placeholder="t('状态')" clearable style="width: 120px">
- <el-option :label="t('全部')" value="" />
- <el-option :label="t('在线')" value="ONLINE" />
- <el-option :label="t('离线')" value="OFFLINE" />
- </el-select>
- </el-form-item>
- <el-form-item>
- <el-button type="primary">
- <Icon icon="mdi:magnify" width="16" height="16" style="margin-right: 4px" />
- {{ t('查询') }}
- </el-button>
- <el-button type="info">
- <Icon icon="mdi:refresh" width="16" height="16" style="margin-right: 4px" />
- {{ t('重置') }}
- </el-button>
- </el-form-item>
- </el-form>
- <el-button type="primary">
- <Icon icon="mdi:plus" width="16" height="16" style="margin-right: 4px" />
- {{ t('新增') }}
- </el-button>
- </div>
- <el-empty :description="t('暂无其他设备数据')" />
- </div>
- </el-tab-pane>
- </el-tabs>
- </el-drawer>
- <!-- 摄像头编辑抽屉 -->
- <el-drawer
- v-model="cameraDialogVisible"
- :title="isEditCamera ? t('摄像头详情') : t('添加摄像头')"
- direction="rtl"
- size="500px"
- :close-on-click-modal="false"
- :show-close="false"
- destroy-on-close
- class="camera-edit-drawer"
- >
- <el-form ref="cameraFormRef" :model="cameraForm" :rules="cameraRules" label-width="auto">
- <!-- <el-form-item label="IP 地址" prop="ip">
- <el-input v-model="cameraForm.ip" :disabled="isEditCamera" placeholder="请输入 IP 地址" />
- </el-form-item> -->
- <el-form-item :label="t('设备ID') + ':'" prop="cameraId">
- <el-input v-model="cameraForm.cameraId" :disabled="isEditCamera" :placeholder="t('请输入设备ID')" />
- </el-form-item>
- <el-form-item :label="t('设备名称') + ':'" prop="cameraName">
- <el-input v-model="cameraForm.cameraName" :placeholder="t('请输入设备名称')" />
- </el-form-item>
- <el-form-item :label="t('厂商') + ':'" prop="vendorName">
- <el-select
- v-model="cameraForm.vendorName"
- :placeholder="t('请选择摄像头')"
- style="width: 100%"
- filterable
- @change="handleVendorSelect"
- >
- <el-option
- v-for="vendor in [
- { id: 'hikvision', name: '海康威视' },
- { id: 'dahua', name: '大华' },
- { id: 'uniview', name: '宇视' },
- { id: 'other', name: '其他' }
- ]"
- :key="vendor.id"
- :label="vendor.name"
- :value="vendor.id"
- />
- </el-select>
- </el-form-item>
- <el-form-item :label="t('型号') + ':'" prop="model">
- <el-input v-model="cameraForm.model" :placeholder="t('请输入型号')" />
- </el-form-item>
- <el-form-item v-if="isEditCamera" :label="t('添加时间') + ':'">
- <span class="form-value">{{ formatTime(currentCamera?.createdAt) }}</span>
- </el-form-item>
- <!-- <el-form-item label="摄像头型号" prop="cameraId">
- <el-select v-model="cameraForm.selectedVendorId" placeholder="请选择摄像头" style="width: 100%" filterable
- @change="handleVendorSelect">
- <el-option v-for="vendor in cameraVendorList" :key="vendor.id" :label="vendor.name" :value="vendor.id" />
- </el-select>
- </el-form-item> -->
- <!-- <el-form-item label="名称" prop="name">
- <el-input v-model="cameraForm.name" placeholder="请输入名称" />
- </el-form-item> -->
- <!-- <el-form-item label="端口" prop="port">
- <el-input-number v-model="cameraForm.port" :min="1" :max="65535" style="width: 100%" />
- </el-form-item> -->
- <!-- <el-form-item label="用户名" prop="username">
- <el-input v-model="cameraForm.username" placeholder="请输入用户名" />
- </el-form-item>
- <el-form-item label="密码" prop="password">
- <el-input v-model="cameraForm.password" type="password" show-password placeholder="请输入密码" />
- </el-form-item> -->
- <el-form-item :label="t('参数配置') + ':'">
- <CodeEditor
- v-model="cameraForm.paramConfig"
- language="json"
- height="200px"
- :placeholder="t('请输入参数配置 (JSON)')"
- />
- </el-form-item>
- <br />
- <el-form-item :label="t('运行参数') + ':'">
- <CodeEditor
- v-model="cameraForm.runtimeParams"
- language="json"
- height="200px"
- :placeholder="t('设备运行参数 (JSON)')"
- />
- </el-form-item>
- <el-form-item :label="t('添加时间') + ':'">
- {{ formatTime(cameraForm.createdAt) }}
- </el-form-item>
- </el-form>
- <template #footer>
- <div class="drawer-footer">
- <el-button @click="cameraDialogVisible = false">{{ t('取消') }}</el-button>
- <el-button type="primary" :loading="cameraSubmitting" @click="handleSubmitCamera">{{ t('更新') }}</el-button>
- </div>
- </template>
- </el-drawer>
- <!-- 参数配置/运行参数抽屉 -->
- <el-drawer
- v-model="paramsDialogVisible"
- :title="paramsDialogTitle"
- direction="rtl"
- size="500px"
- :close-on-click-modal="false"
- :show-close="false"
- destroy-on-close
- class="params-drawer"
- >
- <CodeEditor
- v-model="paramsContent"
- language="json"
- height="500px"
- :placeholder="
- paramsDialogType === 'config' ? t('请输入参数配置(JSON 格式)') : t('请输入运行参数(JSON 格式)')
- "
- />
- <template #footer>
- <div class="drawer-footer">
- <el-button @click="paramsDialogVisible = false">{{ t('取消') }}</el-button>
- <el-button type="primary" :loading="paramsSubmitting" @click="handleSaveParams">{{ t('更新') }}</el-button>
- </div>
- </template>
- </el-drawer>
- <!-- 分页 -->
- <div class="pagination-container">
- <el-pagination
- v-model:current-page="currentPage"
- v-model:page-size="pageSize"
- :page-sizes="[10, 15, 20, 50, 100]"
- :total="total"
- layout="total, sizes, prev, pager, next, jumper"
- background
- @size-change="handleSizeChange"
- @current-change="handleCurrentChange"
- />
- </div>
- </div>
- </template>
- <script setup lang="ts">
- import { ref, reactive, onMounted, computed, watch } from 'vue'
- // Element Plus icons removed - using Iconify instead
- import { ElMessage, ElMessageBox } from 'element-plus'
- import { Icon } from '@iconify/vue'
- import type { FormInstance, FormRules } from 'element-plus'
- import { useI18n } from 'vue-i18n'
- import { formatTime } from '@/utils/dayjs'
- import { useRouter } from 'vue-router'
- import { listLssNodes, deleteLssNode, setLssNodeEnabled, updateLssNode } from '@/api/lss'
- import { adminListCameras, adminAddCamera, adminUpdateCamera, adminDeleteCamera, adminGetCamera } from '@/api/camera'
- import { listCameraVendors } from '@/api/camera-vendor'
- import CodeEditor from '@/components/CodeEditor.vue'
- import type {
- LssNodeDTO,
- LssNodeStatus,
- LssNodeListRequest,
- LssHeartbeatStatus,
- CameraInfoDTO,
- CameraAddRequest,
- CameraUpdateRequest,
- CameraVendorDTO,
- IAbly,
- CameraHeartbeatStatus
- } from '@/types'
- const { t } = useI18n({ useScope: 'global' })
- const router = useRouter()
- // 格式化状态显示
- function formatStatus(status: LssNodeStatus | undefined): string {
- switch (status) {
- case 'active':
- return '在线'
- case 'hold':
- return '离线'
- case 'dead':
- return '离线'
- default:
- return '离线'
- }
- }
- // 获取状态标签类型
- function getStatusTagType(status: LssNodeStatus | undefined): 'success' | 'danger' | 'warning' | 'info' {
- switch (status) {
- case 'active':
- return 'success'
- case 'hold':
- return 'danger'
- case 'dead':
- return 'warning'
- default:
- return 'info'
- }
- }
- // 格式化摄像头状态
- function formatCameraStatus(row: CameraInfoDTO): string {
- if (row.status === 'active') {
- // 大约5秒钟
- return `active [${formatTime(row.updatedAt)}]`
- }
- if (row.status === 'hold') {
- // 大约5分钟没有返回
- return `hold [${formatTime(row.updatedAt)}]`
- }
- // 大约10分钟没有返回
- return `dead (离线)`
- }
- // 当前激活的摄像头 ID
- const activeCameraId = ref<number | null>(null)
- async function handleViewCamera(row: CameraInfoDTO) {
- // 如果没有 streamSn,显示提示对话框
- if (!row.streamSn) {
- try {
- await ElMessageBox.confirm(t('请先新增 Live Stream,才能进行后续操作。'), t('尚未建立 Live Stream'), {
- confirmButtonText: t('新增 Live Stream'),
- cancelButtonText: t('取消'),
- type: 'warning',
- center: true,
- customClass: 'live-stream-dialog',
- distinguishCancelAndClose: true
- })
- router.push(`/live-stream?cameraId=${row.cameraId}&lssId=${row.lssId}&action=create`)
- } catch {
- // 用户点击了取消,不做任何操作
- }
- return
- }
- // 有 streamSn,正常跳转
- router.push(`/live-stream?cameraId=${row.cameraId}`)
- }
- // 格式化品牌
- function formatBrand(brand: string | undefined): string {
- const brandMap: Record<string, string> = {
- hikvision: 'HIKVISION',
- dahua: 'DAHUA',
- uniview: 'UNIVIEW',
- other: '其他'
- }
- return brand ? brandMap[brand] || brand.toUpperCase() : '-'
- }
- // 验证 JSON 格式
- function isValidJson(str: string): boolean {
- if (!str || !str.trim()) return true // 空值视为有效
- try {
- JSON.parse(str)
- return true
- } catch {
- return false
- }
- }
- // 格式化心跳状态
- function formatHeartbeat(lss: LssNodeDTO | null | undefined): string {
- if (!lss) return '-'
- const status = lss.heartbeat || (lss.status === 'active' ? 'active' : lss.status === 'hold' ? 'hold' : 'dead')
- const time = lss.heartbeatTime || lss.updatedAt
- if (status === 'active') {
- return `active [${formatTime(time)}]`
- }
- if (status === 'hold') {
- return `hold [${formatTime(time)}]`
- }
- return `dead (离线)`
- }
- // 获取心跳状态样式类
- function getHeartbeatClass(status: LssHeartbeatStatus | undefined): string {
- switch (status) {
- case 'active':
- return 'status-active'
- case 'hold':
- return 'status-hold'
- case 'dead':
- default:
- return 'status-dead'
- }
- }
- // 查看参数配置
- function handleViewConfig(row: CameraInfoDTO) {
- paramsCamera.value = row
- paramsDialogType.value = 'config'
- paramsDialogTitle.value = `参数配置 - ${row.cameraName}`
- paramsContent.value = row.paramConfig || ''
- paramsDialogVisible.value = true
- }
- // 查看运行参数
- function handleViewRunParams(row: CameraInfoDTO) {
- paramsCamera.value = row
- paramsDialogType.value = 'run'
- paramsDialogTitle.value = `运行参数 - ${row.cameraName}`
- paramsContent.value = row.runtimeParams || ''
- paramsDialogVisible.value = true
- }
- // 保存参数配置/运行参数
- async function handleSaveParams() {
- if (!paramsCamera.value) return
- paramsSubmitting.value = true
- try {
- const data: CameraUpdateRequest = {
- id: paramsCamera.value.id
- }
- if (paramsDialogType.value === 'config') {
- data.paramConfig = paramsContent.value
- } else {
- data.runtimeParams = paramsContent.value
- }
- const res = await adminUpdateCamera(data)
- if (res.success) {
- ElMessage.success('保存成功')
- paramsDialogVisible.value = false
- // 更新本地数据
- if (paramsDialogType.value === 'config') {
- paramsCamera.value.configParams = paramsContent.value
- } else {
- paramsCamera.value.runParams = paramsContent.value
- }
- } else {
- ElMessage.error(res.errMessage || '保存失败')
- }
- } catch (error) {
- console.error('保存参数失败', error)
- ElMessage.error('保存失败')
- } finally {
- paramsSubmitting.value = false
- }
- }
- const loading = ref(false)
- const lssList = ref<(LssNodeDTO & { _switching?: boolean })[]>([])
- const tableRef = ref()
- // 抽屉状态
- const detailDrawerVisible = ref(false)
- const currentLss = ref<LssNodeDTO | null>(null)
- // LSS 编辑抽屉状态
- const lssEditDrawerVisible = ref(false)
- const lssUpdating = ref(false)
- const editActiveTab = ref('detail')
- const lssEditFormRef = ref<FormInstance>()
- const lssEditForm = reactive({
- lssName: '',
- address: '',
- ip: '',
- ably: ''
- })
- // 根据当前 tab 计算抽屉宽度
- const editDrawerSize = computed(() => {
- return editActiveTab.value === 'detail' ? '500px' : '80%'
- })
- // 设备列表抽屉状态
- const cameraDrawerVisible = ref(false)
- const cameraLoading = ref(false)
- const deviceActiveTab = ref('camera')
- const cameraList = ref<CameraInfoDTO[]>([])
- const cameraVendorList = ref<CameraVendorDTO[]>([])
- // 摄像头分页
- const cameraCurrentPage = ref(1)
- const cameraPageSize = ref(15)
- const cameraTotal = ref(0)
- // 摄像头搜索表单
- const cameraSearchForm = reactive({
- cameraId: '',
- cameraName: '',
- status: '' as CameraHeartbeatStatus | ''
- })
- // 摄像头编辑弹窗状态
- const cameraDialogVisible = ref(false)
- const cameraFormRef = ref<FormInstance>()
- const isEditCamera = ref(false)
- const cameraSubmitting = ref(false)
- const currentCamera = ref<CameraInfoDTO | null>(null)
- // 参数配置/运行参数弹窗状态
- const paramsDialogVisible = ref(false)
- const paramsDialogTitle = ref('')
- const paramsDialogType = ref<'config' | 'run'>('config')
- const paramsContent = ref('')
- const paramsSubmitting = ref(false)
- const paramsCamera = ref<CameraInfoDTO | null>(null)
- // 摄像头表单
- const cameraForm = reactive({
- selectedVendorId: null as number | null,
- cameraId: '',
- cameraName: '',
- vendorName: '',
- model: '',
- ip: '',
- port: 80,
- username: '',
- password: '',
- brand: '',
- capability: 'switch_only' as 'switch_only' | 'ptz_enabled',
- rtspUrl: '',
- channelNo: '',
- remark: '',
- enabled: true,
- paramConfig: '',
- runtimeParams: '',
- createdAt: '',
- updatedAt: ''
- })
- // 摄像头表单验证规则(动态)
- const cameraRules = computed<FormRules>(() => ({
- cameraId: [{ required: true, message: t('请输入设备ID'), trigger: 'blur' }]
- }))
- // 排序状态
- const sortState = reactive<{
- sortBy: string
- sortDir: 'ASC' | 'DESC' | undefined
- }>({
- sortBy: '',
- sortDir: undefined
- })
- // 搜索表单
- const searchForm = reactive<{
- lssId: string
- lssName: string
- status: LssNodeStatus | ''
- }>({
- lssId: '',
- lssName: '',
- status: ''
- })
- // 分页相关
- const currentPage = ref(1)
- const pageSize = ref(15)
- const total = ref(0)
- async function getList() {
- loading.value = true
- try {
- const params: LssNodeListRequest = {
- page: currentPage.value,
- size: pageSize.value
- }
- if (searchForm.lssId) {
- params.lssId = searchForm.lssId
- }
- if (searchForm.lssName) {
- params.lssName = searchForm.lssName
- }
- if (searchForm.status) {
- params.status = searchForm.status
- }
- if (sortState.sortBy) {
- params.sortBy = sortState.sortBy
- params.sortDir = sortState.sortDir
- }
- const res = await listLssNodes(params)
- if (res.success && res.data) {
- lssList.value = res.data.list
- total.value = res.data.total || 0
- } else {
- ElMessage.error(res.errMessage || '获取列表失败')
- }
- } catch (error) {
- console.error('获取 LSS 列表失败', error)
- ElMessage.error('获取列表失败')
- } finally {
- loading.value = false
- }
- }
- function handleSearch() {
- currentPage.value = 1
- getList()
- }
- function handleReset() {
- searchForm.lssId = ''
- searchForm.lssName = ''
- searchForm.status = ''
- sortState.sortBy = ''
- sortState.sortDir = undefined
- currentPage.value = 1
- getList()
- }
- function handleSortChange({ prop, order }: { prop: string; order: 'ascending' | 'descending' | null }) {
- sortState.sortBy = prop || ''
- sortState.sortDir = order === 'ascending' ? 'ASC' : order === 'descending' ? 'DESC' : undefined
- getList()
- }
- function handleSizeChange(val: number) {
- pageSize.value = val
- currentPage.value = 1
- getList()
- }
- function handleCurrentChange(val: number) {
- currentPage.value = val
- getList()
- }
- function handleViewDetail(row: LssNodeDTO) {
- currentLss.value = row
- detailDrawerVisible.value = true
- }
- function handleScanDevices(row: LssNodeDTO) {
- console.log(row)
- }
- function handleEdit(row: LssNodeDTO, tab: 'detail' | 'camera' | 'pusher') {
- currentLss.value = row
- lssEditForm.lssName = row.lssName || ''
- lssEditForm.address = row.address || ''
- lssEditForm.ably = JSON.stringify(row.ably)
- editActiveTab.value = tab
- lssEditDrawerVisible.value = true
- // 每次打开抽屉都刷新摄像头列表
- cameraSearchForm.cameraId = ''
- cameraSearchForm.cameraName = ''
- cameraSearchForm.status = ''
- loadCameraList()
- }
- async function handleCameraList(row: LssNodeDTO) {
- currentLss.value = row
- cameraSearchForm.cameraId = ''
- cameraSearchForm.cameraName = ''
- cameraSearchForm.status = ''
- deviceActiveTab.value = 'camera'
- cameraDrawerVisible.value = true
- await loadCameraList()
- }
- async function handleUpdateLss() {
- if (!currentLss.value) return
- lssUpdating.value = true
- try {
- const res = await updateLssNode({
- lssId: currentLss.value.lssId,
- lssName: lssEditForm.lssName,
- address: lssEditForm.address,
- ablyInfo: lssEditForm.ably
- })
- if (res.success) {
- ElMessage.success('更新成功')
- lssEditDrawerVisible.value = false
- getList()
- } else {
- ElMessage.error(res.errMessage || '更新失败')
- }
- } catch (error) {
- console.error('更新 LSS 失败', error)
- ElMessage.error('更新失败')
- } finally {
- lssUpdating.value = false
- }
- }
- async function loadCameraList() {
- if (!currentLss.value) return
- cameraLoading.value = true
- cameraList.value = []
- try {
- const params: any = {
- lssId: currentLss.value.lssId,
- page: cameraCurrentPage.value,
- size: cameraPageSize.value
- }
- if (cameraSearchForm.cameraId) {
- params.cameraId = cameraSearchForm.cameraId
- }
- if (cameraSearchForm.cameraName) {
- params.cameraName = cameraSearchForm.cameraName
- }
- if (cameraSearchForm.status) {
- params.status = cameraSearchForm.status
- }
- const res = await adminListCameras(params)
- if (res.success && res.data) {
- cameraList.value = res.data.list || []
- cameraTotal.value = res.data.total || 0
- } else {
- ElMessage.error(res.errMessage || '获取摄像头列表失败')
- }
- } catch (error) {
- console.error('获取摄像头列表失败', error)
- ElMessage.error('获取摄像头列表失败')
- } finally {
- cameraLoading.value = false
- }
- }
- function handleCameraSearch() {
- cameraCurrentPage.value = 1
- loadCameraList()
- }
- function handleCameraReset() {
- cameraSearchForm.cameraId = ''
- cameraSearchForm.cameraName = ''
- cameraSearchForm.status = ''
- cameraCurrentPage.value = 1
- loadCameraList()
- }
- function handleCameraSizeChange(val: number) {
- cameraPageSize.value = val
- cameraCurrentPage.value = 1
- loadCameraList()
- }
- function handleCameraPageChange(val: number) {
- cameraCurrentPage.value = val
- loadCameraList()
- }
- function resetCameraForm() {
- cameraForm.selectedVendorId = null
- cameraForm.cameraId = ''
- cameraForm.cameraName = ''
- cameraForm.vendorName = ''
- cameraForm.model = ''
- cameraForm.ip = ''
- cameraForm.port = 80
- cameraForm.username = ''
- cameraForm.password = ''
- cameraForm.brand = ''
- cameraForm.capability = 'switch_only'
- cameraForm.rtspUrl = ''
- cameraForm.model = ''
- cameraForm.channelNo = ''
- cameraForm.remark = ''
- cameraForm.enabled = true
- cameraForm.paramConfig = ''
- cameraForm.runtimeParams = ''
- cameraForm.createdAt = ''
- cameraForm.updatedAt = ''
- cameraFormRef.value?.clearValidate()
- }
- async function loadCameraVendorList() {
- try {
- const res = await listCameraVendors({ enabled: true })
- if (res.success && res.data) {
- cameraVendorList.value = res.data.list || []
- }
- } catch (error) {
- console.error('获取厂商列表失败', error)
- }
- }
- function handleVendorSelect(vendorId: number) {
- const vendor = cameraVendorList.value.find((v) => v.id === vendorId)
- if (vendor) {
- cameraForm.brand = vendor.code
- // 设置厂商默认端口
- if (vendor.defaultPort) {
- cameraForm.port = vendor.defaultPort
- }
- // 根据厂商设置默认能力
- cameraForm.capability = vendor.supportPtz ? 'ptz_enabled' : 'switch_only'
- }
- }
- async function handleAddCamera() {
- isEditCamera.value = false
- currentCamera.value = null
- resetCameraForm()
- await loadCameraVendorList()
- cameraDialogVisible.value = true
- }
- async function handleEditCamera(row: CameraInfoDTO) {
- isEditCamera.value = true
- try {
- // 通过 API 获取摄像头详情
- const res = await adminGetCamera(row.id)
- if (!res.success || !res.data) {
- ElMessage.error(res.errMessage || '获取摄像头详情失败')
- return
- }
- const camera = res.data
- currentCamera.value = camera
- cameraForm.selectedVendorId = null
- cameraForm.cameraId = camera.cameraId
- cameraForm.cameraName = camera.cameraName
- cameraForm.vendorName = camera.vendorName || ''
- cameraForm.model = camera.model || ''
- cameraForm.ip = camera.ip
- cameraForm.port = camera.port || 80
- cameraForm.username = camera.username || ''
- cameraForm.password = ''
- cameraForm.brand = camera.brand || ''
- cameraForm.capability = camera.capability || 'switch_only'
- cameraForm.rtspUrl = camera.rtspUrl || ''
- cameraForm.model = camera.model || ''
- cameraForm.channelNo = camera.channelNo || ''
- cameraForm.remark = camera.remark || ''
- cameraForm.createdAt = camera.createdAt || ''
- cameraForm.updatedAt = camera.updatedAt || ''
- cameraForm.enabled = camera.enabled
- cameraForm.paramConfig = camera.paramConfig || ''
- cameraForm.runtimeParams = camera.runtimeParams || ''
- await loadCameraVendorList()
- cameraDialogVisible.value = true
- } catch (error) {
- console.error('获取摄像头详情失败', error)
- ElMessage.error('获取摄像头详情失败')
- }
- }
- async function handleSubmitCamera() {
- if (!cameraFormRef.value) return
- await cameraFormRef.value.validate(async (valid) => {
- if (!valid) return
- // 验证 JSON 格式
- if (!isValidJson(cameraForm.paramConfig)) {
- ElMessage.error('参数配置格式错误,请输入有效的 JSON')
- return
- }
- if (!isValidJson(cameraForm.runtimeParams)) {
- ElMessage.error('设备运行参数格式错误,请输入有效的 JSON')
- return
- }
- cameraSubmitting.value = true
- try {
- if (isEditCamera.value) {
- // 编辑模式:更新摄像头信息
- if (!currentCamera.value) {
- ElMessage.error('摄像头信息错误')
- return
- }
- const data: CameraUpdateRequest = {
- id: currentCamera.value.id,
- cameraName: cameraForm.cameraName,
- vendorName: cameraForm.vendorName,
- model: cameraForm.model,
- port: cameraForm.port,
- username: cameraForm.username,
- brand: cameraForm.brand,
- capability: cameraForm.capability,
- lssId: currentLss.value?.lssId,
- rtspUrl: cameraForm.rtspUrl,
- channelNo: cameraForm.channelNo,
- remark: cameraForm.remark,
- enabled: cameraForm.enabled,
- paramConfig: cameraForm.paramConfig,
- runtimeParams: cameraForm.runtimeParams
- }
- if (cameraForm.password) {
- data.password = cameraForm.password
- }
- const res = await adminUpdateCamera(data)
- if (res.success) {
- ElMessage.success('更新成功')
- cameraDialogVisible.value = false
- loadCameraList()
- } else {
- ElMessage.error(res.errMessage || '更新失败')
- }
- } else {
- // 新增模式:创建摄像头并绑定到当前 LSS
- const data: CameraAddRequest = {
- cameraId: cameraForm.cameraId,
- cameraName: cameraForm.cameraName,
- vendorName: cameraForm.vendorName,
- model: cameraForm.model,
- paramConfig: cameraForm.paramConfig,
- runtimeParams: cameraForm.runtimeParams,
- lssId: currentLss.value?.lssId
- }
- const res = await adminAddCamera(data)
- if (res.success) {
- ElMessage.success('添加成功')
- cameraDialogVisible.value = false
- loadCameraList()
- } else {
- ElMessage.error(res.errMessage || '添加失败')
- }
- }
- } catch (error) {
- console.error('保存摄像头失败', error)
- ElMessage.error('操作失败')
- } finally {
- cameraSubmitting.value = false
- }
- })
- }
- async function handleDeleteCamera(row: CameraInfoDTO) {
- try {
- await ElMessageBox.confirm(`确定要删除摄像头 "${row.cameraName}" 吗?`, '提示', {
- type: 'warning'
- })
- const res = await adminDeleteCamera(row.id)
- if (res.success) {
- ElMessage.success('删除成功')
- loadCameraList()
- } else {
- ElMessage.error(res.errMessage || '删除失败')
- }
- } catch (error) {
- if (error !== 'cancel') {
- console.error('删除摄像头失败', error)
- ElMessage.error('删除失败')
- }
- }
- }
- async function handleToggleEnabled(row: LssNodeDTO & { _switching?: boolean }, enabled: boolean) {
- row._switching = true
- try {
- const res = await setLssNodeEnabled(row.lssId, enabled)
- if (res.success) {
- ElMessage.success(enabled ? '已启用' : '已禁用')
- } else {
- // 恢复原状态
- row.enabled = !enabled
- ElMessage.error(res.errMessage || '操作失败')
- }
- } catch (error) {
- row.enabled = !enabled
- console.error('切换启用状态失败', error)
- ElMessage.error('操作失败')
- } finally {
- row._switching = false
- }
- }
- async function handleDelete(row: LssNodeDTO) {
- try {
- await ElMessageBox.confirm(`确定要删除 LSS 节点 "${row.lssName}" 吗?`, '提示', {
- type: 'warning'
- })
- const res = await deleteLssNode(row.lssId)
- if (res.success) {
- ElMessage.success('删除成功')
- getList()
- } else {
- ElMessage.error(res.errMessage || '删除失败')
- }
- } catch (error) {
- if (error !== 'cancel') {
- console.error('删除失败', error)
- ElMessage.error('删除失败')
- }
- }
- }
- // 监听 tab 切换,加载对应数据
- watch(editActiveTab, (newTab) => {
- if (newTab === 'camera' && currentLss.value) {
- cameraSearchForm.cameraId = ''
- cameraSearchForm.cameraName = ''
- cameraSearchForm.status = ''
- cameraCurrentPage.value = 1
- loadCameraList()
- }
- })
- onMounted(() => {
- getList()
- })
- </script>
- <style lang="scss" scoped>
- .page-container {
- display: flex;
- flex-direction: column;
- box-sizing: border-box;
- }
- .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;
- }
- .camera-toolbar {
- display: flex;
- justify-content: space-between;
- align-items: flex-start;
- margin-bottom: 16px;
- :deep(.el-form) {
- .el-form-item {
- margin-bottom: 0;
- margin-right: 12px;
- }
- }
- }
- .camera-pagination {
- margin-top: 16px;
- display: flex;
- justify-content: flex-end;
- }
- .status-text {
- font-size: 12px;
- }
- // 十字瞄准按钮样式
- .crosshairs-btn {
- color: #bbbbbb;
- &.active {
- color: var(--color-primary);
- }
- }
- .status-active {
- color: #67c23a;
- }
- .status-hold {
- color: #e6a23c;
- }
- .status-dead {
- color: #f56c6c;
- }
- // 抽屉样式
- :deep(.el-drawer) {
- .el-drawer__header {
- margin-bottom: 0;
- padding: 16px 20px;
- border-bottom: 1px solid #e5e7eb;
- }
- .el-drawer__body {
- padding: 16px;
- }
- .el-descriptions {
- .el-descriptions__label {
- width: 100px;
- font-weight: 600;
- }
- }
- }
- // LSS 编辑抽屉样式
- .lss-edit-drawer {
- :deep(.el-drawer__body) {
- padding: 0;
- display: flex;
- flex-direction: column;
- height: 100%;
- }
- }
- .drawer-content {
- display: flex;
- flex-direction: column;
- height: 100%;
- }
- .drawer-tabs {
- flex-shrink: 0;
- border-bottom: 1px solid #e5e7eb;
- :deep(.el-tabs__header) {
- margin: 0;
- padding: 0 20px;
- }
- :deep(.el-tabs__nav-wrap::after) {
- display: none;
- }
- :deep(.el-tabs__item) {
- height: 48px;
- line-height: 48px;
- font-size: 14px;
- color: #606266;
- &.is-active {
- color: #409eff;
- font-weight: 500;
- }
- &:hover {
- color: #409eff;
- }
- }
- :deep(.el-tabs__active-bar) {
- background-color: #409eff;
- }
- }
- .drawer-body {
- flex: 1;
- overflow-y: auto;
- padding: 20px;
- }
- .lss-detail-form {
- :deep(.el-form-item) {
- margin-bottom: 18px;
- }
- :deep(.el-form-item__label) {
- color: #606266;
- font-size: 14px;
- }
- .form-value {
- line-height: 32px;
- color: #303133;
- font-size: 14px;
- }
- .textarea-wrapper {
- width: 100%;
- }
- }
- .heartbeat-status {
- display: inline-flex;
- align-items: center;
- gap: 6px;
- line-height: 32px;
- font-size: 14px;
- }
- .heartbeat-dot {
- display: inline-block;
- width: 8px;
- height: 8px;
- border-radius: 50%;
- &.status-active {
- background-color: #67c23a;
- }
- &.status-hold {
- background-color: #e6a23c;
- }
- &.status-dead {
- background-color: #f56c6c;
- }
- }
- .heartbeat-info-icon {
- margin-left: 8px;
- color: #909399;
- cursor: pointer;
- &:hover {
- color: #409eff;
- }
- }
- .heartbeat-tooltip {
- font-size: 12px;
- line-height: 1.6;
- .tooltip-title {
- font-weight: 500;
- margin-bottom: 4px;
- }
- .tooltip-format {
- margin-top: 8px;
- font-weight: 500;
- }
- .tooltip-example {
- color: #409eff;
- }
- }
- .tab-content {
- min-height: 200px;
- }
- // 设备列表抽屉样式
- .device-drawer {
- :deep(.el-drawer__body) {
- padding: 0;
- }
- }
- .device-tabs {
- height: 100%;
- display: flex;
- flex-direction: column;
- :deep(.el-tabs__header) {
- margin: 0;
- padding: 0 20px;
- flex-shrink: 0;
- border-bottom: 1px solid #e5e7eb;
- }
- :deep(.el-tabs__nav-wrap::after) {
- display: none;
- }
- :deep(.el-tabs__item) {
- height: 48px;
- line-height: 48px;
- font-size: 14px;
- color: #606266;
- &.is-active {
- color: #409eff;
- font-weight: 500;
- }
- &:hover {
- color: #409eff;
- }
- }
- :deep(.el-tabs__active-bar) {
- background-color: #409eff;
- }
- :deep(.el-tabs__content) {
- flex: 1;
- overflow: hidden;
- padding: 16px;
- }
- :deep(.el-tab-pane) {
- height: 100%;
- overflow-y: auto;
- }
- }
- .tab-content-wrapper {
- height: 100%;
- }
- .drawer-footer {
- flex-shrink: 0;
- display: flex;
- justify-content: flex-end;
- padding: 12px 20px;
- border-top: 1px solid #e5e7eb;
- gap: 12px;
- }
- // 表格样式
- :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;
- }
- }
- </style>
|