index.vue 49 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636
  1. <template>
  2. <div class="page-container">
  3. <!-- 搜索表单 -->
  4. <div class="search-form">
  5. <el-form :model="searchForm" inline data-id="search-form">
  6. <el-form-item>
  7. <el-input
  8. v-model.trim="searchForm.lssId"
  9. placeholder="LSS ID"
  10. clearable
  11. data-id="search-keyword"
  12. @keyup.enter="handleSearch"
  13. />
  14. </el-form-item>
  15. <el-form-item>
  16. <el-input
  17. v-model.trim="searchForm.lssName"
  18. placeholder="名称"
  19. clearable
  20. data-id="search-keyword"
  21. @keyup.enter="handleSearch"
  22. />
  23. </el-form-item>
  24. <el-form-item>
  25. <el-select v-model="searchForm.status" :placeholder="t('心跳')" clearable data-id="search-enabled">
  26. <el-option label="全部" value="" />
  27. <el-option label="active" :value="1" />
  28. <el-option label="hold" :value="2" />
  29. <el-option label="dead" :value="3" />
  30. </el-select>
  31. </el-form-item>
  32. <el-form-item>
  33. <el-button type="primary" :icon="Search" data-id="btn-search" @click="handleSearch">
  34. {{ t('查询') }}
  35. </el-button>
  36. <el-button :icon="RefreshRight" data-id="btn-reset" @click="handleReset">{{ t('重置') }}</el-button>
  37. </el-form-item>
  38. </el-form>
  39. </div>
  40. <!-- 数据表格 -->
  41. <div class="table-wrapper">
  42. <el-table
  43. ref="tableRef"
  44. v-loading="loading"
  45. :data="lssList"
  46. stripe
  47. size="default"
  48. data-id="lss-table"
  49. height="100%"
  50. @sort-change="handleSortChange"
  51. >
  52. <el-table-column prop="lssId" label="LSS ID" min-width="120" sortable="custom" show-overflow-tooltip />
  53. <el-table-column prop="lssName" :label="t('名称')" min-width="140" sortable="custom" show-overflow-tooltip />
  54. <el-table-column prop="address" :label="t('地址')" min-width="180" sortable="custom" show-overflow-tooltip />
  55. <el-table-column prop="ip" :label="t('IP')" min-width="180" sortable="custom" show-overflow-tooltip />
  56. <el-table-column :label="t('心跳')" width="220" align="center">
  57. <template #default="{ row }">
  58. {{
  59. row.status === 'active'
  60. ? t('活跃')
  61. : row.status === 'hold'
  62. ? t('hold')
  63. : row.status === 'dead'
  64. ? t('离线')
  65. : '-'
  66. }}
  67. |
  68. {{ formatTime(row.lastHeartbeatAt) }}
  69. </template>
  70. </el-table-column>
  71. <el-table-column :label="t('设备列表')" align="center">
  72. <template #default="{ row }">
  73. <el-button type="primary" link @click="handleCameraList(row)">
  74. <Icon icon="mdi:cctv" />
  75. </el-button>
  76. </template>
  77. </el-table-column>
  78. <el-table-column :label="t('ably')" align="center" fixed="right">
  79. <template #default="{ row }">
  80. {{ row.ablyClientId }}
  81. </template>
  82. </el-table-column>
  83. <!-- <el-table-column prop="status" :label="t('状态')" min-width="100" sortable="custom">
  84. <template #default="{ row }">
  85. <el-tag :type="getStatusTagType(row.status)" size="small">
  86. {{ formatStatus(row.status) }}
  87. </el-tag>
  88. </template>
  89. </el-table-column> -->
  90. <!-- <el-table-column prop="currentTasks" :label="t('当前任务')" min-width="100" sortable="custom" align="center">
  91. <template #default="{ row }">{{ row.currentTasks }} / {{ row.maxTasks }}</template>
  92. </el-table-column> -->
  93. <!-- <el-table-column prop="enabled" :label="t('启用')" min-width="80" align="center">
  94. <template #default="{ row }">
  95. <el-switch v-model="row.enabled" :loading="row._switching"
  96. @change="(val: boolean) => handleToggleEnabled(row, val)" />
  97. </template>
  98. </el-table-column> -->
  99. <!-- <el-table-column prop="ffmpegVersion" label="FFmpeg" show-overflow-tooltip /> -->
  100. <!-- <el-table-column label="詳情" align="center">
  101. <template #default="{ row }">
  102. <el-button type="primary" link :icon="View" @click="handleViewDetail(row)" />
  103. </template>
  104. </el-table-column>
  105. <el-table-column label="status" align="center">
  106. <template #default="{ row }">
  107. {{ row.status === 'ONLINE' ? '在线' : '离线' }}
  108. </template>
  109. </el-table-column> -->
  110. <el-table-column :label="t('操作')" align="center" fixed="right">
  111. <template #default="{ row }">
  112. <el-button type="primary" link :icon="Edit" @click="handleEdit(row)" />
  113. <el-button type="primary" link @click="handleScanDevices(row)">
  114. <Icon icon="mdi:radar" />
  115. </el-button>
  116. <el-button type="danger" link :icon="Delete" @click="handleDelete(row)" />
  117. </template>
  118. </el-table-column>
  119. </el-table>
  120. </div>
  121. <!-- LSS 详情抽屉 -->
  122. <el-drawer v-model="detailDrawerVisible" title="LSS 节点详情" direction="rtl" size="500px" destroy-on-close>
  123. <el-descriptions :column="1" border>
  124. <el-descriptions-item label="LSS ID">{{ currentLss?.lssId }}</el-descriptions-item>
  125. <el-descriptions-item label="名称">{{ currentLss?.lssName }}</el-descriptions-item>
  126. <el-descriptions-item label="地址">{{ currentLss?.address }}</el-descriptions-item>
  127. <el-descriptions-item label="机器 ID">{{ currentLss?.machineId || '-' }}</el-descriptions-item>
  128. <el-descriptions-item label="状态">
  129. <el-tag :type="getStatusTagType(currentLss?.status)" size="small">
  130. {{ formatStatus(currentLss?.status) }}
  131. </el-tag>
  132. </el-descriptions-item>
  133. <el-descriptions-item label="任务数">
  134. {{ currentLss?.currentTasks }} / {{ currentLss?.maxTasks }}
  135. </el-descriptions-item>
  136. <el-descriptions-item label="FFmpeg 版本">{{ currentLss?.ffmpegVersion || '-' }}</el-descriptions-item>
  137. <el-descriptions-item label="系统信息">{{ currentLss?.systemInfo || '-' }}</el-descriptions-item>
  138. <el-descriptions-item label="启用状态">
  139. <el-tag :type="currentLss?.enabled ? 'success' : 'info'" size="small">
  140. {{ currentLss?.enabled ? '已启用' : '已禁用' }}
  141. </el-tag>
  142. </el-descriptions-item>
  143. <el-descriptions-item label="创建时间">{{ formatTime(currentLss?.createdAt) }}</el-descriptions-item>
  144. <el-descriptions-item label="更新时间">{{ formatTime(currentLss?.updatedAt) }}</el-descriptions-item>
  145. </el-descriptions>
  146. </el-drawer>
  147. <!-- LSS 编辑抽屉 -->
  148. <el-drawer
  149. v-model="lssEditDrawerVisible"
  150. direction="rtl"
  151. :size="editDrawerSize"
  152. :with-header="false"
  153. destroy-on-close
  154. class="lss-edit-drawer"
  155. >
  156. <div class="drawer-content">
  157. <!-- 顶部 Tabs -->
  158. <el-tabs v-model="editActiveTab" class="drawer-tabs">
  159. <el-tab-pane label="LSS详情" name="detail" />
  160. <el-tab-pane label="摄像头列表" name="camera" />
  161. <el-tab-pane label="推币机列表" name="pusher" />
  162. </el-tabs>
  163. <div class="drawer-body">
  164. <!-- LSS 详情 Tab -->
  165. <div v-show="editActiveTab === 'detail'" class="lss-detail-form">
  166. <el-form ref="lssEditFormRef" :model="lssEditForm" label-width="80px" label-position="left">
  167. <el-form-item label="LSS ID:">
  168. <span class="form-value">{{ currentLss?.lssId }}</span>
  169. </el-form-item>
  170. <el-form-item :label="t('名称') + ':'" prop="lssName">
  171. <el-input v-model="lssEditForm.lssName" placeholder="请输入名称" style="width: 180px" />
  172. </el-form-item>
  173. <el-form-item :label="t('地址') + ':'" prop="address">
  174. <el-input v-model="lssEditForm.address" placeholder="请输入地址" />
  175. </el-form-item>
  176. <el-form-item :label="t('IP') + ':'">
  177. <span class="form-value">{{ currentLss?.ip }}</span>
  178. </el-form-item>
  179. <el-form-item :label="t('心跳') + ':'">
  180. <span class="heartbeat-status" :class="getHeartbeatClass(currentLss?.status)">
  181. {{ formatHeartbeat(currentLss) }}
  182. <span class="heartbeat-dot" :class="getHeartbeatClass(currentLss?.status)"></span>
  183. </span>
  184. <el-tooltip placement="right" effect="light">
  185. <template #content>
  186. <div class="heartbeat-tooltip">
  187. <div class="tooltip-title">心跳状态:</div>
  188. <div>active - 持续返回中,并且频繁</div>
  189. <div>hold - 五分钟内有返回</div>
  190. <div>dead - 五分钟内没有返回</div>
  191. <div class="tooltip-format">表现形式为:</div>
  192. <div class="tooltip-example">Status [yy-mm-dd 00:00:00]</div>
  193. </div>
  194. </template>
  195. <el-icon class="heartbeat-info-icon">
  196. <QuestionFilled />
  197. </el-icon>
  198. </el-tooltip>
  199. </el-form-item>
  200. <el-form-item :label="t('ably') + ':'" prop="ably">
  201. <div class="textarea-wrapper">
  202. <el-input
  203. disabled
  204. type="textarea"
  205. :rows="8"
  206. v-model="lssEditForm.ably"
  207. placeholder="请输入ably信息"
  208. maxlength="1000"
  209. show-word-limit
  210. />
  211. </div>
  212. </el-form-item>
  213. </el-form>
  214. </div>
  215. <!-- 摄像头列表 Tab -->
  216. <div v-show="editActiveTab === 'camera'" v-loading="cameraLoading" class="tab-content-wrapper">
  217. <div class="camera-toolbar">
  218. <el-form :model="cameraSearchForm" inline>
  219. <el-form-item>
  220. <el-input
  221. v-model.trim="cameraSearchForm.keyword"
  222. placeholder="IP / 设备ID / 名称"
  223. clearable
  224. style="width: 200px"
  225. @keyup.enter="handleCameraSearch"
  226. />
  227. </el-form-item>
  228. <el-form-item>
  229. <el-select v-model="cameraSearchForm.status" placeholder="状态" clearable style="width: 120px">
  230. <el-option label="全部" value="" />
  231. <el-option label="在线" value="ONLINE" />
  232. <el-option label="离线" value="OFFLINE" />
  233. </el-select>
  234. </el-form-item>
  235. <el-form-item>
  236. <el-button type="primary" :icon="Search" @click="handleCameraSearch">查询</el-button>
  237. <el-button :icon="RefreshRight" @click="handleCameraReset">重置</el-button>
  238. </el-form-item>
  239. </el-form>
  240. <el-button type="primary" :icon="Plus" @click="handleAddCamera">{{ t('新增') }}</el-button>
  241. </div>
  242. <el-empty v-if="!cameraLoading && cameraList.length === 0" description="暂无关联设备" />
  243. <el-table v-else :data="cameraList" stripe size="small" border>
  244. <!-- <el-table-column prop="ip" label="本地IP" min-width="110" /> -->
  245. <el-table-column prop="cameraId" label="设备ID" min-width="100" show-overflow-tooltip />
  246. <el-table-column prop="cameraName" label="名称" min-width="100" show-overflow-tooltip />
  247. <el-table-column :label="t('状态(心跳)')" min-width="140">
  248. <template #default="{ row }">
  249. <span :class="['status-text', row.status === 'active' ? 'status-active' : 'status-dead']">
  250. {{ formatCameraStatus(row) }}
  251. </span>
  252. </template>
  253. </el-table-column>
  254. <el-table-column label="参数配置" min-width="80" align="center">
  255. <template #default="{ row }">
  256. <el-button type="primary" link @click="handleViewConfig(row)">查看</el-button>
  257. </template>
  258. </el-table-column>
  259. <el-table-column label="运行参数" min-width="80" align="center">
  260. <template #default="{ row }">
  261. <el-button type="primary" link @click="handleViewRunParams(row)">查看</el-button>
  262. </template>
  263. </el-table-column>
  264. <el-table-column prop="brand" :label="t('厂商')" min-width="90">
  265. <template #default="{ row }">
  266. {{ formatBrand(row.brand) }}
  267. </template>
  268. </el-table-column>
  269. <el-table-column prop="model" label="型号" min-width="130" show-overflow-tooltip />
  270. <el-table-column label="添加时间" min-width="140">
  271. <template #default="{ row }">
  272. {{ formatTime(row.createdAt) }}
  273. </template>
  274. </el-table-column>
  275. <el-table-column label="设备控制" min-width="100" align="center" fixed="right">
  276. <template #default="{ row }">
  277. <el-button type="primary" link :icon="Edit" @click="handleEditCamera(row)" />
  278. <el-button type="danger" link :icon="Delete" @click="handleDeleteCamera(row)" />
  279. <el-button link :class="['crosshairs-btn', { active: !row.streamSn }]" @click="handleViewCamera(row)">
  280. <Icon icon="mdi:crosshairs" />
  281. </el-button>
  282. </template>
  283. </el-table-column>
  284. </el-table>
  285. <div v-if="cameraList.length > 0" class="camera-count">共 {{ cameraList.length }} 个设备</div>
  286. </div>
  287. <!-- 推币机列表 Tab -->
  288. <div v-show="editActiveTab === 'pusher'" class="tab-content">
  289. <el-empty description="暂无推币机数据" />
  290. </div>
  291. </div>
  292. <div class="drawer-footer">
  293. <el-button @click="lssEditDrawerVisible = false">{{ t('取消') }}</el-button>
  294. <el-button type="primary" :loading="lssUpdating" @click="handleUpdateLss">{{ t('更新') }}</el-button>
  295. </div>
  296. </div>
  297. </el-drawer>
  298. <!-- 设备列表抽屉 -->
  299. <el-drawer
  300. v-model="cameraDrawerVisible"
  301. :title="`设备列表 - ${currentLss?.lssName || ''}`"
  302. direction="rtl"
  303. size="80%"
  304. destroy-on-close
  305. class="device-drawer"
  306. >
  307. <el-tabs v-model="deviceActiveTab" class="device-tabs">
  308. <el-tab-pane label="摄像头列表" name="camera">
  309. <div v-loading="cameraLoading" class="tab-content-wrapper">
  310. <div class="camera-toolbar">
  311. <el-form :model="cameraSearchForm" inline>
  312. <el-form-item>
  313. <el-input
  314. v-model.trim="cameraSearchForm.keyword"
  315. placeholder="IP / 设备ID / 名称"
  316. clearable
  317. style="width: 200px"
  318. @keyup.enter="handleCameraSearch"
  319. />
  320. </el-form-item>
  321. <el-form-item>
  322. <el-select v-model="cameraSearchForm.status" placeholder="状态" clearable style="width: 120px">
  323. <el-option label="全部" value="" />
  324. <el-option label="在线" value="ONLINE" />
  325. <el-option label="离线" value="OFFLINE" />
  326. </el-select>
  327. </el-form-item>
  328. <el-form-item>
  329. <el-button type="primary" :icon="Search" @click="handleCameraSearch">查询</el-button>
  330. <el-button :icon="RefreshRight" @click="handleCameraReset">重置</el-button>
  331. </el-form-item>
  332. </el-form>
  333. <el-button type="primary" :icon="Plus" @click="handleAddCamera">{{ t('新增') }}</el-button>
  334. </div>
  335. <el-empty v-if="!cameraLoading && cameraList.length === 0" description="暂无关联设备" />
  336. <el-table v-else :data="cameraList" stripe size="small" border>
  337. <!-- <el-table-column prop="ip" label="本地IP" min-width="110" /> -->
  338. <el-table-column prop="cameraId" label="设备ID" min-width="100" show-overflow-tooltip />
  339. <el-table-column prop="cameraName" label="名称" min-width="100" show-overflow-tooltip />
  340. <el-table-column :label="t('状态(心跳)')" min-width="140">
  341. <template #default="{ row }">
  342. <span :class="['status-text', row.status === 'active' ? 'status-active' : 'status-dead']">
  343. {{ formatCameraStatus(row) }}
  344. </span>
  345. </template>
  346. </el-table-column>
  347. <el-table-column label="参数配置" min-width="80" align="center">
  348. <template #default="{ row }">
  349. <el-button type="primary" link @click="handleViewConfig(row)">查看</el-button>
  350. </template>
  351. </el-table-column>
  352. <el-table-column label="运行参数" min-width="80" align="center">
  353. <template #default="{ row }">
  354. <el-button type="primary" link @click="handleViewRunParams(row)">查看</el-button>
  355. </template>
  356. </el-table-column>
  357. <el-table-column prop="brand" :label="t('厂商')" min-width="90">
  358. <template #default="{ row }">
  359. {{ formatBrand(row.brand) }}
  360. </template>
  361. </el-table-column>
  362. <el-table-column prop="model" label="型号" min-width="130" show-overflow-tooltip />
  363. <el-table-column label="添加时间" min-width="140">
  364. <template #default="{ row }">
  365. {{ formatTime(row.createdAt) }}
  366. </template>
  367. </el-table-column>
  368. <el-table-column label="设备控制" min-width="100" align="center" fixed="right">
  369. <template #default="{ row }">
  370. <el-button type="primary" link :icon="Edit" @click="handleEditCamera(row)" />
  371. <el-button type="danger" link :icon="Delete" @click="handleDeleteCamera(row)" />
  372. <el-button link :class="['crosshairs-btn', { active: !row.streamSn }]" @click="handleViewCamera(row)">
  373. <Icon icon="mdi:crosshairs" />
  374. </el-button>
  375. </template>
  376. </el-table-column>
  377. </el-table>
  378. <div v-if="cameraList.length > 0" class="camera-count">共 {{ cameraList.length }} 个设备</div>
  379. </div>
  380. </el-tab-pane>
  381. <el-tab-pane label="推币机列表" name="pusher">
  382. <div class="tab-content-wrapper">
  383. <div class="camera-toolbar">
  384. <el-form inline>
  385. <el-form-item>
  386. <el-input placeholder="设备ID / 名称" clearable style="width: 200px" />
  387. </el-form-item>
  388. <el-form-item>
  389. <el-select placeholder="状态" clearable style="width: 120px">
  390. <el-option label="全部" value="" />
  391. <el-option label="在线" value="ONLINE" />
  392. <el-option label="离线" value="OFFLINE" />
  393. </el-select>
  394. </el-form-item>
  395. <el-form-item>
  396. <el-button type="primary" :icon="Search">查询</el-button>
  397. <el-button :icon="RefreshRight">重置</el-button>
  398. </el-form-item>
  399. </el-form>
  400. <el-button type="primary" :icon="Plus">{{ t('新增') }}</el-button>
  401. </div>
  402. <el-empty description="暂无推币机数据" />
  403. </div>
  404. </el-tab-pane>
  405. <el-tab-pane label="其他设备" name="other">
  406. <div class="tab-content-wrapper">
  407. <div class="camera-toolbar">
  408. <el-form inline>
  409. <el-form-item>
  410. <el-input placeholder="设备ID / 名称" clearable style="width: 200px" />
  411. </el-form-item>
  412. <el-form-item>
  413. <el-select placeholder="状态" clearable style="width: 120px">
  414. <el-option label="全部" value="" />
  415. <el-option label="在线" value="ONLINE" />
  416. <el-option label="离线" value="OFFLINE" />
  417. </el-select>
  418. </el-form-item>
  419. <el-form-item>
  420. <el-button type="primary" :icon="Search">查询</el-button>
  421. <el-button :icon="RefreshRight">重置</el-button>
  422. </el-form-item>
  423. </el-form>
  424. <el-button type="primary" :icon="Plus">{{ t('新增') }}</el-button>
  425. </div>
  426. <el-empty description="暂无其他设备数据" />
  427. </div>
  428. </el-tab-pane>
  429. </el-tabs>
  430. </el-drawer>
  431. <!-- 摄像头编辑抽屉 -->
  432. <el-drawer
  433. v-model="cameraDialogVisible"
  434. :title="isEditCamera ? '编辑摄像头' : '添加摄像头'"
  435. direction="rtl"
  436. size="500px"
  437. :close-on-click-modal="false"
  438. destroy-on-close
  439. class="camera-edit-drawer"
  440. >
  441. <el-form ref="cameraFormRef" :model="cameraForm" :rules="cameraRules" label-width="100px">
  442. <!-- <el-form-item label="IP 地址" prop="ip">
  443. <el-input v-model="cameraForm.ip" :disabled="isEditCamera" placeholder="请输入 IP 地址" />
  444. </el-form-item> -->
  445. <el-form-item label="设备ID" prop="cameraId">
  446. <el-input v-model="cameraForm.cameraId" :disabled="isEditCamera" placeholder="请输入设备ID" />
  447. </el-form-item>
  448. <el-form-item label="设备名称" prop="cameraName">
  449. <el-input v-model="cameraForm.cameraName" placeholder="请输入设备名称" />
  450. </el-form-item>
  451. <el-form-item label="厂商" prop="vendorName">
  452. <el-select
  453. v-model="cameraForm.vendorName"
  454. placeholder="请选择摄像头"
  455. style="width: 100%"
  456. filterable
  457. @change="handleVendorSelect"
  458. >
  459. <el-option
  460. v-for="vendor in [
  461. { id: 'hikvision', name: '海康威视' },
  462. { id: 'dahua', name: '大华' },
  463. { id: 'uniview', name: '宇视' },
  464. { id: 'other', name: '其他' }
  465. ]"
  466. :key="vendor.id"
  467. :label="vendor.name"
  468. :value="vendor.id"
  469. />
  470. </el-select>
  471. </el-form-item>
  472. <el-form-item label="型号" prop="model">
  473. <el-input v-model="cameraForm.model" placeholder="请输入型号" />
  474. </el-form-item>
  475. <!-- <el-form-item label="摄像头型号" prop="cameraId">
  476. <el-select v-model="cameraForm.selectedVendorId" placeholder="请选择摄像头" style="width: 100%" filterable
  477. @change="handleVendorSelect">
  478. <el-option v-for="vendor in cameraVendorList" :key="vendor.id" :label="vendor.name" :value="vendor.id" />
  479. </el-select>
  480. </el-form-item> -->
  481. <!-- <el-form-item label="名称" prop="name">
  482. <el-input v-model="cameraForm.name" placeholder="请输入名称" />
  483. </el-form-item> -->
  484. <!-- <el-form-item label="端口" prop="port">
  485. <el-input-number v-model="cameraForm.port" :min="1" :max="65535" style="width: 100%" />
  486. </el-form-item> -->
  487. <!-- <el-form-item label="用户名" prop="username">
  488. <el-input v-model="cameraForm.username" placeholder="请输入用户名" />
  489. </el-form-item>
  490. <el-form-item label="密码" prop="password">
  491. <el-input v-model="cameraForm.password" type="password" show-password placeholder="请输入密码" />
  492. </el-form-item> -->
  493. <el-form-item label="参数配置">
  494. <JsonEditor v-model="cameraForm.paramConfig" height="200px" placeholder="请输入参数配置 (JSON)" />
  495. </el-form-item>
  496. <el-form-item label="设备运行参数">
  497. <JsonEditor v-model="cameraForm.runtimeParams" height="200px" placeholder="设备运行参数 (JSON)" />
  498. </el-form-item>
  499. </el-form>
  500. <template #footer>
  501. <div class="drawer-footer">
  502. <el-button @click="cameraDialogVisible = false">{{ t('取消') }}</el-button>
  503. <el-button type="primary" :loading="cameraSubmitting" @click="handleSubmitCamera">{{ t('确定') }}</el-button>
  504. </div>
  505. </template>
  506. </el-drawer>
  507. <!-- 参数配置/运行参数抽屉 -->
  508. <el-drawer
  509. v-model="paramsDialogVisible"
  510. :title="paramsDialogTitle"
  511. direction="rtl"
  512. size="500px"
  513. :close-on-click-modal="false"
  514. destroy-on-close
  515. class="params-drawer"
  516. >
  517. <JsonEditor
  518. v-model="paramsContent"
  519. height="500px"
  520. :placeholder="paramsDialogType === 'config' ? '请输入参数配置(JSON 格式)' : '请输入运行参数(JSON 格式)'"
  521. />
  522. <template #footer>
  523. <div class="drawer-footer">
  524. <el-button @click="paramsDialogVisible = false">{{ t('取消') }}</el-button>
  525. <el-button type="primary" :loading="paramsSubmitting" @click="handleSaveParams">{{ t('更新') }}</el-button>
  526. </div>
  527. </template>
  528. </el-drawer>
  529. <!-- 分页 -->
  530. <div class="pagination-container">
  531. <el-pagination
  532. v-model:current-page="currentPage"
  533. v-model:page-size="pageSize"
  534. :page-sizes="[10, 20, 50, 100]"
  535. :total="total"
  536. layout="total, sizes, prev, pager, next, jumper"
  537. background
  538. @size-change="handleSizeChange"
  539. @current-change="handleCurrentChange"
  540. />
  541. </div>
  542. </div>
  543. </template>
  544. <script setup lang="ts">
  545. import { ref, reactive, onMounted, computed, watch } from 'vue'
  546. import { Search, RefreshRight, Delete, View, Edit, VideoCamera, Plus, QuestionFilled } from '@element-plus/icons-vue'
  547. import { ElMessage, ElMessageBox } from 'element-plus'
  548. import { listLssNodes, deleteLssNode, setLssNodeEnabled, updateLssNode } from '@/api/lss'
  549. import { adminListCameras, adminAddCamera, adminUpdateCamera, adminDeleteCamera, adminGetCamera } from '@/api/camera'
  550. import { listCameraVendors } from '@/api/camera-vendor'
  551. import { Icon } from '@iconify/vue'
  552. import JsonEditor from '@/components/JsonEditor.vue'
  553. import type {
  554. LssNodeDTO,
  555. LssNodeStatus,
  556. LssNodeListRequest,
  557. LssHeartbeatStatus,
  558. CameraInfoDTO,
  559. CameraAddRequest,
  560. CameraUpdateRequest,
  561. CameraVendorDTO,
  562. IAbly
  563. } from '@/types'
  564. import type { FormInstance, FormRules } from 'element-plus'
  565. import dayjs from 'dayjs'
  566. import { useI18n } from 'vue-i18n'
  567. import { useRouter } from 'vue-router'
  568. const { t } = useI18n({ useScope: 'global' })
  569. const router = useRouter()
  570. // 格式化状态显示
  571. function formatStatus(status: LssNodeStatus | undefined): string {
  572. switch (status) {
  573. case 'active':
  574. return '在线'
  575. case 'hold':
  576. return '离线'
  577. case 'dead':
  578. return '离线'
  579. default:
  580. return '离线'
  581. }
  582. }
  583. // 获取状态标签类型
  584. function getStatusTagType(status: LssNodeStatus | undefined): 'success' | 'danger' | 'warning' | 'info' {
  585. switch (status) {
  586. case 'ONLINE':
  587. return 'success'
  588. case 'OFFLINE':
  589. return 'danger'
  590. case 'BUSY':
  591. return 'warning'
  592. case 'MAINTENANCE':
  593. return 'info'
  594. default:
  595. return 'info'
  596. }
  597. }
  598. // 格式化时间
  599. function formatTime(time: string | undefined): string {
  600. if (!time) return '-'
  601. return dayjs(time).format('YYYY-MM-DD HH:mm:ss')
  602. }
  603. // 格式化摄像头状态
  604. function formatCameraStatus(row: CameraInfoDTO): string {
  605. if (row.status === 'active') {
  606. // 大约5秒钟
  607. return `active [${formatTime(row.updatedAt)}]`
  608. } else if (row.status === 'hold') {
  609. // 大约5分钟没有返回
  610. return `hold [${formatTime(row.updatedAt)}]`
  611. } else {
  612. // 大约10分钟没有返回
  613. return `dead (离线)`
  614. }
  615. }
  616. // 当前激活的摄像头 ID
  617. const activeCameraId = ref<number | null>(null)
  618. function handleViewCamera(row: CameraInfoDTO) {
  619. // 切换激活状态
  620. router.push(`/live-stream?cameraId=${row.cameraId}`)
  621. }
  622. // 格式化品牌
  623. function formatBrand(brand: string | undefined): string {
  624. const brandMap: Record<string, string> = {
  625. hikvision: 'HIKVISION',
  626. dahua: 'DAHUA',
  627. uniview: 'UNIVIEW',
  628. other: '其他'
  629. }
  630. return brand ? brandMap[brand] || brand.toUpperCase() : '-'
  631. }
  632. // 验证 JSON 格式
  633. function isValidJson(str: string): boolean {
  634. if (!str || !str.trim()) return true // 空值视为有效
  635. try {
  636. JSON.parse(str)
  637. return true
  638. } catch {
  639. return false
  640. }
  641. }
  642. // 格式化心跳状态
  643. function formatHeartbeat(lss: LssNodeDTO | null | undefined): string {
  644. if (!lss) return '-'
  645. const status = lss.heartbeat || (lss.status === 'active' ? 'active' : lss.status === 'hold' ? 'hold' : 'dead')
  646. const time = lss.heartbeatTime || lss.updatedAt
  647. if (status === 'active') {
  648. return `active [${formatTime(time)}]`
  649. } else if (status === 'hold') {
  650. return `hold [${formatTime(time)}]`
  651. } else {
  652. return `dead (离线)`
  653. }
  654. }
  655. // 获取心跳状态样式类
  656. function getHeartbeatClass(status: LssHeartbeatStatus | undefined): string {
  657. switch (status) {
  658. case 'active':
  659. return 'status-active'
  660. case 'hold':
  661. return 'status-hold'
  662. case 'dead':
  663. default:
  664. return 'status-dead'
  665. }
  666. }
  667. // 查看参数配置
  668. function handleViewConfig(row: CameraInfoDTO) {
  669. paramsCamera.value = row
  670. paramsDialogType.value = 'config'
  671. paramsDialogTitle.value = `参数配置 - ${row.cameraName}`
  672. paramsContent.value = row.paramConfig || ''
  673. paramsDialogVisible.value = true
  674. }
  675. // 查看运行参数
  676. function handleViewRunParams(row: CameraInfoDTO) {
  677. paramsCamera.value = row
  678. paramsDialogType.value = 'run'
  679. paramsDialogTitle.value = `运行参数 - ${row.cameraName}`
  680. paramsContent.value = row.runtimeParams || ''
  681. paramsDialogVisible.value = true
  682. }
  683. // 保存参数配置/运行参数
  684. async function handleSaveParams() {
  685. if (!paramsCamera.value) return
  686. paramsSubmitting.value = true
  687. try {
  688. const data: CameraUpdateRequest = {
  689. id: paramsCamera.value.id
  690. }
  691. if (paramsDialogType.value === 'config') {
  692. data.paramConfig = paramsContent.value
  693. } else {
  694. data.runtimeParams = paramsContent.value
  695. }
  696. const res = await adminUpdateCamera(data)
  697. if (res.success) {
  698. ElMessage.success('保存成功')
  699. paramsDialogVisible.value = false
  700. // 更新本地数据
  701. if (paramsDialogType.value === 'config') {
  702. paramsCamera.value.configParams = paramsContent.value
  703. } else {
  704. paramsCamera.value.runParams = paramsContent.value
  705. }
  706. } else {
  707. ElMessage.error(res.errMessage || '保存失败')
  708. }
  709. } catch (error) {
  710. console.error('保存参数失败', error)
  711. ElMessage.error('保存失败')
  712. } finally {
  713. paramsSubmitting.value = false
  714. }
  715. }
  716. const loading = ref(false)
  717. const lssList = ref<(LssNodeDTO & { _switching?: boolean })[]>([])
  718. const tableRef = ref()
  719. // 抽屉状态
  720. const detailDrawerVisible = ref(false)
  721. const currentLss = ref<LssNodeDTO | null>(null)
  722. // LSS 编辑抽屉状态
  723. const lssEditDrawerVisible = ref(false)
  724. const lssUpdating = ref(false)
  725. const editActiveTab = ref('detail')
  726. const lssEditFormRef = ref<FormInstance>()
  727. const lssEditForm = reactive({
  728. lssName: '',
  729. address: '',
  730. ip: '',
  731. ably: ''
  732. })
  733. // 根据当前 tab 计算抽屉宽度
  734. const editDrawerSize = computed(() => {
  735. return editActiveTab.value === 'detail' ? '500px' : '80%'
  736. })
  737. // 设备列表抽屉状态
  738. const cameraDrawerVisible = ref(false)
  739. const cameraLoading = ref(false)
  740. const deviceActiveTab = ref('camera')
  741. const cameraList = ref<CameraInfoDTO[]>([])
  742. const cameraVendorList = ref<CameraVendorDTO[]>([])
  743. // 摄像头搜索表单
  744. const cameraSearchForm = reactive({
  745. keyword: '',
  746. status: '' as 'ONLINE' | 'OFFLINE' | ''
  747. })
  748. // 摄像头编辑弹窗状态
  749. const cameraDialogVisible = ref(false)
  750. const cameraFormRef = ref<FormInstance>()
  751. const isEditCamera = ref(false)
  752. const cameraSubmitting = ref(false)
  753. const currentCamera = ref<CameraInfoDTO | null>(null)
  754. // 参数配置/运行参数弹窗状态
  755. const paramsDialogVisible = ref(false)
  756. const paramsDialogTitle = ref('')
  757. const paramsDialogType = ref<'config' | 'run'>('config')
  758. const paramsContent = ref('')
  759. const paramsSubmitting = ref(false)
  760. const paramsCamera = ref<CameraInfoDTO | null>(null)
  761. // 摄像头表单
  762. const cameraForm = reactive({
  763. selectedVendorId: null as number | null,
  764. cameraId: '',
  765. cameraName: '',
  766. vendorName: '',
  767. model: '',
  768. ip: '',
  769. port: 80,
  770. username: '',
  771. password: '',
  772. brand: '',
  773. capability: 'switch_only' as 'switch_only' | 'ptz_enabled',
  774. rtspUrl: '',
  775. channelNo: '',
  776. remark: '',
  777. enabled: true,
  778. paramConfig: '',
  779. runtimeParams: ''
  780. })
  781. // 摄像头表单验证规则(动态)
  782. const cameraRules = computed<FormRules>(() => ({}))
  783. // 排序状态
  784. const sortState = reactive<{
  785. sortBy: string
  786. sortDir: 'ASC' | 'DESC' | undefined
  787. }>({
  788. sortBy: '',
  789. sortDir: undefined
  790. })
  791. // 搜索表单
  792. const searchForm = reactive<{
  793. lssId: string
  794. lssName: string
  795. status: LssNodeStatus | ''
  796. }>({
  797. lssId: '',
  798. lssName: '',
  799. status: ''
  800. })
  801. // 分页相关
  802. const currentPage = ref(1)
  803. const pageSize = ref(20)
  804. const total = ref(0)
  805. async function getList() {
  806. loading.value = true
  807. try {
  808. const params: LssNodeListRequest = {
  809. page: currentPage.value,
  810. size: pageSize.value
  811. }
  812. if (searchForm.lssId) {
  813. params.lssId = searchForm.lssId
  814. }
  815. if (searchForm.lssName) {
  816. params.lssName = searchForm.lssName
  817. }
  818. if (searchForm.status) {
  819. params.status = searchForm.status
  820. }
  821. if (sortState.sortBy) {
  822. params.sortBy = sortState.sortBy
  823. params.sortDir = sortState.sortDir
  824. }
  825. const res = await listLssNodes(params)
  826. if (res.success && res.data) {
  827. lssList.value = res.data.list
  828. total.value = res.data.total || 0
  829. } else {
  830. ElMessage.error(res.errMessage || '获取列表失败')
  831. }
  832. } catch (error) {
  833. console.error('获取 LSS 列表失败', error)
  834. ElMessage.error('获取列表失败')
  835. } finally {
  836. loading.value = false
  837. }
  838. }
  839. function handleSearch() {
  840. currentPage.value = 1
  841. getList()
  842. }
  843. function handleReset() {
  844. searchForm.keyword = ''
  845. searchForm.status = ''
  846. searchForm.enabled = null
  847. currentPage.value = 1
  848. sortState.sortBy = ''
  849. sortState.sortDir = undefined
  850. getList()
  851. }
  852. function handleSortChange({ prop, order }: { prop: string; order: 'ascending' | 'descending' | null }) {
  853. sortState.sortBy = prop || ''
  854. sortState.sortDir = order === 'ascending' ? 'ASC' : order === 'descending' ? 'DESC' : undefined
  855. getList()
  856. }
  857. function handleSizeChange(val: number) {
  858. pageSize.value = val
  859. currentPage.value = 1
  860. getList()
  861. }
  862. function handleCurrentChange(val: number) {
  863. currentPage.value = val
  864. getList()
  865. }
  866. function handleViewDetail(row: LssNodeDTO) {
  867. currentLss.value = row
  868. detailDrawerVisible.value = true
  869. }
  870. function handleScanDevices(row: LssNodeDTO) {
  871. console.log(row)
  872. }
  873. function handleEdit(row: LssNodeDTO) {
  874. currentLss.value = row
  875. lssEditForm.lssName = row.lssName || ''
  876. lssEditForm.address = row.address || ''
  877. lssEditForm.ably = JSON.stringify(row.ably)
  878. editActiveTab.value = 'detail'
  879. lssEditDrawerVisible.value = true
  880. }
  881. async function handleCameraList(row: LssNodeDTO) {
  882. currentLss.value = row
  883. cameraSearchForm.keyword = ''
  884. cameraSearchForm.status = ''
  885. deviceActiveTab.value = 'camera'
  886. cameraDrawerVisible.value = true
  887. await loadCameraList()
  888. }
  889. async function handleUpdateLss() {
  890. if (!currentLss.value) return
  891. lssUpdating.value = true
  892. try {
  893. const res = await updateLssNode({
  894. lssId: currentLss.value.lssId,
  895. lssName: lssEditForm.lssName,
  896. address: lssEditForm.address,
  897. ablyInfo: lssEditForm.ablyInfo
  898. })
  899. if (res.success) {
  900. ElMessage.success('更新成功')
  901. lssEditDrawerVisible.value = false
  902. getList()
  903. } else {
  904. ElMessage.error(res.errMessage || '更新失败')
  905. }
  906. } catch (error) {
  907. console.error('更新 LSS 失败', error)
  908. ElMessage.error('更新失败')
  909. } finally {
  910. lssUpdating.value = false
  911. }
  912. }
  913. async function loadCameraList() {
  914. if (!currentLss.value) return
  915. cameraLoading.value = true
  916. cameraList.value = []
  917. try {
  918. const params: any = { lssId: currentLss.value.lssId }
  919. if (cameraSearchForm.keyword) {
  920. params.keyword = cameraSearchForm.keyword
  921. }
  922. if (cameraSearchForm.status) {
  923. params.status = cameraSearchForm.status
  924. }
  925. const res = await adminListCameras(params)
  926. if (res.success && res.data) {
  927. cameraList.value = res.data.list || []
  928. } else {
  929. ElMessage.error(res.errMessage || '获取摄像头列表失败')
  930. }
  931. } catch (error) {
  932. console.error('获取摄像头列表失败', error)
  933. ElMessage.error('获取摄像头列表失败')
  934. } finally {
  935. cameraLoading.value = false
  936. }
  937. }
  938. function handleCameraSearch() {
  939. loadCameraList()
  940. }
  941. function handleCameraReset() {
  942. cameraSearchForm.keyword = ''
  943. cameraSearchForm.status = ''
  944. loadCameraList()
  945. }
  946. function resetCameraForm() {
  947. cameraForm.selectedVendorId = null
  948. cameraForm.cameraId = ''
  949. cameraForm.cameraName = ''
  950. cameraForm.vendorName = ''
  951. cameraForm.model = ''
  952. cameraForm.ip = ''
  953. cameraForm.port = 80
  954. cameraForm.username = ''
  955. cameraForm.password = ''
  956. cameraForm.brand = ''
  957. cameraForm.capability = 'switch_only'
  958. cameraForm.rtspUrl = ''
  959. cameraForm.model = ''
  960. cameraForm.channelNo = ''
  961. cameraForm.remark = ''
  962. cameraForm.enabled = true
  963. cameraForm.paramConfig = ''
  964. cameraForm.runtimeParams = ''
  965. cameraFormRef.value?.clearValidate()
  966. }
  967. async function loadCameraVendorList() {
  968. try {
  969. const res = await listCameraVendors({ enabled: true })
  970. if (res.success && res.data) {
  971. cameraVendorList.value = res.data.list || []
  972. }
  973. } catch (error) {
  974. console.error('获取厂商列表失败', error)
  975. }
  976. }
  977. function handleVendorSelect(vendorId: number) {
  978. const vendor = cameraVendorList.value.find((v) => v.id === vendorId)
  979. if (vendor) {
  980. cameraForm.brand = vendor.code
  981. // 设置厂商默认端口
  982. if (vendor.defaultPort) {
  983. cameraForm.port = vendor.defaultPort
  984. }
  985. // 根据厂商设置默认能力
  986. cameraForm.capability = vendor.supportPtz ? 'ptz_enabled' : 'switch_only'
  987. }
  988. }
  989. async function handleAddCamera() {
  990. isEditCamera.value = false
  991. currentCamera.value = null
  992. resetCameraForm()
  993. await loadCameraVendorList()
  994. cameraDialogVisible.value = true
  995. }
  996. async function handleEditCamera(row: CameraInfoDTO) {
  997. isEditCamera.value = true
  998. try {
  999. // 通过 API 获取摄像头详情
  1000. const res = await adminGetCamera(row.id)
  1001. if (!res.success || !res.data) {
  1002. ElMessage.error(res.errMessage || '获取摄像头详情失败')
  1003. return
  1004. }
  1005. const camera = res.data
  1006. currentCamera.value = camera
  1007. cameraForm.selectedVendorId = null
  1008. cameraForm.cameraId = camera.cameraId
  1009. cameraForm.cameraName = camera.cameraName
  1010. cameraForm.vendorName = camera.vendorName || ''
  1011. cameraForm.model = camera.model || ''
  1012. cameraForm.ip = camera.ip
  1013. cameraForm.port = camera.port || 80
  1014. cameraForm.username = camera.username || ''
  1015. cameraForm.password = ''
  1016. cameraForm.brand = camera.brand || ''
  1017. cameraForm.capability = camera.capability || 'switch_only'
  1018. cameraForm.rtspUrl = camera.rtspUrl || ''
  1019. cameraForm.model = camera.model || ''
  1020. cameraForm.channelNo = camera.channelNo || ''
  1021. cameraForm.remark = camera.remark || ''
  1022. cameraForm.enabled = camera.enabled
  1023. cameraForm.paramConfig = camera.paramConfig || ''
  1024. cameraForm.runtimeParams = camera.runtimeParams || ''
  1025. await loadCameraVendorList()
  1026. cameraDialogVisible.value = true
  1027. } catch (error) {
  1028. console.error('获取摄像头详情失败', error)
  1029. ElMessage.error('获取摄像头详情失败')
  1030. }
  1031. }
  1032. async function handleSubmitCamera() {
  1033. if (!cameraFormRef.value) return
  1034. await cameraFormRef.value.validate(async (valid) => {
  1035. if (!valid) return
  1036. // 验证 JSON 格式
  1037. if (!isValidJson(cameraForm.paramConfig)) {
  1038. ElMessage.error('参数配置格式错误,请输入有效的 JSON')
  1039. return
  1040. }
  1041. if (!isValidJson(cameraForm.runtimeParams)) {
  1042. ElMessage.error('设备运行参数格式错误,请输入有效的 JSON')
  1043. return
  1044. }
  1045. cameraSubmitting.value = true
  1046. try {
  1047. if (isEditCamera.value) {
  1048. // 编辑模式:更新摄像头信息
  1049. if (!currentCamera.value) {
  1050. ElMessage.error('摄像头信息错误')
  1051. return
  1052. }
  1053. const data: CameraUpdateRequest = {
  1054. id: currentCamera.value.id,
  1055. cameraName: cameraForm.cameraName,
  1056. vendorName: cameraForm.vendorName,
  1057. model: cameraForm.model,
  1058. port: cameraForm.port,
  1059. username: cameraForm.username,
  1060. brand: cameraForm.brand,
  1061. capability: cameraForm.capability,
  1062. lssId: currentLss.value?.lssId,
  1063. rtspUrl: cameraForm.rtspUrl,
  1064. channelNo: cameraForm.channelNo,
  1065. remark: cameraForm.remark,
  1066. enabled: cameraForm.enabled,
  1067. paramConfig: cameraForm.paramConfig,
  1068. runtimeParams: cameraForm.runtimeParams
  1069. }
  1070. if (cameraForm.password) {
  1071. data.password = cameraForm.password
  1072. }
  1073. const res = await adminUpdateCamera(data)
  1074. if (res.success) {
  1075. ElMessage.success('更新成功')
  1076. cameraDialogVisible.value = false
  1077. loadCameraList()
  1078. } else {
  1079. ElMessage.error(res.errMessage || '更新失败')
  1080. }
  1081. } else {
  1082. // 新增模式:创建摄像头并绑定到当前 LSS
  1083. const data: CameraAddRequest = {
  1084. cameraId: cameraForm.cameraId,
  1085. cameraName: cameraForm.cameraName,
  1086. vendorName: cameraForm.vendorName,
  1087. model: cameraForm.model,
  1088. paramConfig: cameraForm.paramConfig,
  1089. runtimeParams: cameraForm.runtimeParams,
  1090. lssId: currentLss.value?.lssId
  1091. }
  1092. const res = await adminAddCamera(data)
  1093. if (res.success) {
  1094. ElMessage.success('添加成功')
  1095. cameraDialogVisible.value = false
  1096. loadCameraList()
  1097. } else {
  1098. ElMessage.error(res.errMessage || '添加失败')
  1099. }
  1100. }
  1101. } catch (error) {
  1102. console.error('保存摄像头失败', error)
  1103. ElMessage.error('操作失败')
  1104. } finally {
  1105. cameraSubmitting.value = false
  1106. }
  1107. })
  1108. }
  1109. async function handleDeleteCamera(row: CameraInfoDTO) {
  1110. try {
  1111. await ElMessageBox.confirm(`确定要删除摄像头 "${row.name}" 吗?`, '提示', {
  1112. type: 'warning'
  1113. })
  1114. const res = await adminDeleteCamera(row.id)
  1115. if (res.success) {
  1116. ElMessage.success('删除成功')
  1117. loadCameraList()
  1118. } else {
  1119. ElMessage.error(res.errMessage || '删除失败')
  1120. }
  1121. } catch (error) {
  1122. if (error !== 'cancel') {
  1123. console.error('删除摄像头失败', error)
  1124. ElMessage.error('删除失败')
  1125. }
  1126. }
  1127. }
  1128. async function handleToggleEnabled(row: LssNodeDTO & { _switching?: boolean }, enabled: boolean) {
  1129. row._switching = true
  1130. try {
  1131. const res = await setLssNodeEnabled(row.lssId, enabled)
  1132. if (res.success) {
  1133. ElMessage.success(enabled ? '已启用' : '已禁用')
  1134. } else {
  1135. // 恢复原状态
  1136. row.enabled = !enabled
  1137. ElMessage.error(res.errMessage || '操作失败')
  1138. }
  1139. } catch (error) {
  1140. row.enabled = !enabled
  1141. console.error('切换启用状态失败', error)
  1142. ElMessage.error('操作失败')
  1143. } finally {
  1144. row._switching = false
  1145. }
  1146. }
  1147. async function handleDelete(row: LssNodeDTO) {
  1148. try {
  1149. await ElMessageBox.confirm(`确定要删除 LSS 节点 "${row.lssName}" 吗?`, '提示', {
  1150. type: 'warning'
  1151. })
  1152. const res = await deleteLssNode(row.lssId)
  1153. if (res.success) {
  1154. ElMessage.success('删除成功')
  1155. getList()
  1156. } else {
  1157. ElMessage.error(res.errMessage || '删除失败')
  1158. }
  1159. } catch (error) {
  1160. if (error !== 'cancel') {
  1161. console.error('删除失败', error)
  1162. ElMessage.error('删除失败')
  1163. }
  1164. }
  1165. }
  1166. // 监听 tab 切换,加载对应数据
  1167. watch(editActiveTab, (newTab) => {
  1168. if (newTab === 'camera' && currentLss.value) {
  1169. cameraSearchForm.keyword = ''
  1170. cameraSearchForm.status = ''
  1171. loadCameraList()
  1172. }
  1173. })
  1174. onMounted(() => {
  1175. getList()
  1176. })
  1177. </script>
  1178. <style lang="scss" scoped>
  1179. .page-container {
  1180. display: flex;
  1181. flex-direction: column;
  1182. box-sizing: border-box;
  1183. }
  1184. .search-form {
  1185. flex-shrink: 0;
  1186. margin-bottom: 16px;
  1187. padding: 16px 16px 4px 16px;
  1188. background: #f5f7fa;
  1189. :deep(.el-form-item) {
  1190. margin-bottom: 12px;
  1191. margin-right: 16px;
  1192. }
  1193. :deep(.el-input),
  1194. :deep(.el-select) {
  1195. width: 160px;
  1196. }
  1197. :deep(.el-button--primary) {
  1198. background-color: #4f46e5;
  1199. border-color: #4f46e5;
  1200. &:hover,
  1201. &:focus {
  1202. background-color: #6366f1;
  1203. border-color: #6366f1;
  1204. }
  1205. }
  1206. }
  1207. .table-wrapper {
  1208. flex: 1;
  1209. min-height: 0;
  1210. overflow: hidden;
  1211. }
  1212. .pagination-container {
  1213. flex-shrink: 0;
  1214. display: flex;
  1215. justify-content: flex-end;
  1216. padding-top: 16px;
  1217. :deep(.el-pagination) {
  1218. .el-pager li.is-active {
  1219. background-color: #4f46e5;
  1220. color: #fff;
  1221. }
  1222. .el-pager li:not(.is-active):hover {
  1223. color: #4f46e5;
  1224. }
  1225. .btn-prev:hover,
  1226. .btn-next:hover {
  1227. color: #4f46e5;
  1228. }
  1229. }
  1230. }
  1231. .camera-toolbar {
  1232. display: flex;
  1233. justify-content: space-between;
  1234. align-items: flex-start;
  1235. margin-bottom: 16px;
  1236. :deep(.el-form) {
  1237. .el-form-item {
  1238. margin-bottom: 0;
  1239. margin-right: 12px;
  1240. }
  1241. }
  1242. }
  1243. .camera-count {
  1244. margin-top: 16px;
  1245. text-align: right;
  1246. color: #666;
  1247. font-size: 14px;
  1248. }
  1249. .status-text {
  1250. font-size: 12px;
  1251. }
  1252. // 十字瞄准按钮样式
  1253. .crosshairs-btn {
  1254. color: #909399;
  1255. transition: color 0.2s;
  1256. &:hover {
  1257. color: #606266;
  1258. }
  1259. &.active {
  1260. color: red;
  1261. &:hover {
  1262. color: red;
  1263. }
  1264. }
  1265. }
  1266. .status-active {
  1267. color: #67c23a;
  1268. }
  1269. .status-hold {
  1270. color: #e6a23c;
  1271. }
  1272. .status-dead {
  1273. color: #f56c6c;
  1274. }
  1275. // 抽屉样式
  1276. :deep(.el-drawer) {
  1277. .el-drawer__header {
  1278. margin-bottom: 0;
  1279. padding: 16px 20px;
  1280. border-bottom: 1px solid #e5e7eb;
  1281. }
  1282. .el-drawer__body {
  1283. padding: 16px;
  1284. }
  1285. .el-descriptions {
  1286. .el-descriptions__label {
  1287. width: 100px;
  1288. font-weight: 600;
  1289. }
  1290. }
  1291. }
  1292. // LSS 编辑抽屉样式
  1293. .lss-edit-drawer {
  1294. :deep(.el-drawer__body) {
  1295. padding: 0;
  1296. display: flex;
  1297. flex-direction: column;
  1298. height: 100%;
  1299. }
  1300. }
  1301. .drawer-content {
  1302. display: flex;
  1303. flex-direction: column;
  1304. height: 100%;
  1305. }
  1306. .drawer-tabs {
  1307. flex-shrink: 0;
  1308. border-bottom: 1px solid #e5e7eb;
  1309. :deep(.el-tabs__header) {
  1310. margin: 0;
  1311. padding: 0 20px;
  1312. }
  1313. :deep(.el-tabs__nav-wrap::after) {
  1314. display: none;
  1315. }
  1316. :deep(.el-tabs__item) {
  1317. height: 48px;
  1318. line-height: 48px;
  1319. font-size: 14px;
  1320. color: #606266;
  1321. &.is-active {
  1322. color: #409eff;
  1323. font-weight: 500;
  1324. }
  1325. &:hover {
  1326. color: #409eff;
  1327. }
  1328. }
  1329. :deep(.el-tabs__active-bar) {
  1330. background-color: #409eff;
  1331. }
  1332. }
  1333. .drawer-body {
  1334. flex: 1;
  1335. overflow-y: auto;
  1336. padding: 20px;
  1337. }
  1338. .lss-detail-form {
  1339. :deep(.el-form-item) {
  1340. margin-bottom: 18px;
  1341. }
  1342. :deep(.el-form-item__label) {
  1343. color: #606266;
  1344. font-size: 14px;
  1345. }
  1346. .form-value {
  1347. line-height: 32px;
  1348. color: #303133;
  1349. font-size: 14px;
  1350. }
  1351. .textarea-wrapper {
  1352. width: 100%;
  1353. }
  1354. }
  1355. .heartbeat-status {
  1356. display: inline-flex;
  1357. align-items: center;
  1358. gap: 6px;
  1359. line-height: 32px;
  1360. font-size: 14px;
  1361. }
  1362. .heartbeat-dot {
  1363. display: inline-block;
  1364. width: 8px;
  1365. height: 8px;
  1366. border-radius: 50%;
  1367. &.status-active {
  1368. background-color: #67c23a;
  1369. }
  1370. &.status-hold {
  1371. background-color: #e6a23c;
  1372. }
  1373. &.status-dead {
  1374. background-color: #f56c6c;
  1375. }
  1376. }
  1377. .heartbeat-info-icon {
  1378. margin-left: 8px;
  1379. color: #909399;
  1380. cursor: pointer;
  1381. &:hover {
  1382. color: #409eff;
  1383. }
  1384. }
  1385. .heartbeat-tooltip {
  1386. font-size: 12px;
  1387. line-height: 1.6;
  1388. .tooltip-title {
  1389. font-weight: 500;
  1390. margin-bottom: 4px;
  1391. }
  1392. .tooltip-format {
  1393. margin-top: 8px;
  1394. font-weight: 500;
  1395. }
  1396. .tooltip-example {
  1397. color: #409eff;
  1398. }
  1399. }
  1400. .tab-content {
  1401. min-height: 200px;
  1402. }
  1403. // 设备列表抽屉样式
  1404. .device-drawer {
  1405. :deep(.el-drawer__body) {
  1406. padding: 0;
  1407. }
  1408. }
  1409. .device-tabs {
  1410. height: 100%;
  1411. display: flex;
  1412. flex-direction: column;
  1413. :deep(.el-tabs__header) {
  1414. margin: 0;
  1415. padding: 0 20px;
  1416. flex-shrink: 0;
  1417. border-bottom: 1px solid #e5e7eb;
  1418. }
  1419. :deep(.el-tabs__nav-wrap::after) {
  1420. display: none;
  1421. }
  1422. :deep(.el-tabs__item) {
  1423. height: 48px;
  1424. line-height: 48px;
  1425. font-size: 14px;
  1426. color: #606266;
  1427. &.is-active {
  1428. color: #409eff;
  1429. font-weight: 500;
  1430. }
  1431. &:hover {
  1432. color: #409eff;
  1433. }
  1434. }
  1435. :deep(.el-tabs__active-bar) {
  1436. background-color: #409eff;
  1437. }
  1438. :deep(.el-tabs__content) {
  1439. flex: 1;
  1440. overflow: hidden;
  1441. padding: 16px;
  1442. }
  1443. :deep(.el-tab-pane) {
  1444. height: 100%;
  1445. overflow-y: auto;
  1446. }
  1447. }
  1448. .tab-content-wrapper {
  1449. height: 100%;
  1450. }
  1451. .drawer-footer {
  1452. flex-shrink: 0;
  1453. display: flex;
  1454. justify-content: flex-end;
  1455. padding: 12px 20px;
  1456. border-top: 1px solid #e5e7eb;
  1457. gap: 12px;
  1458. }
  1459. // 表格样式
  1460. :deep(.el-table) {
  1461. --el-table-row-hover-bg-color: #f0f0ff;
  1462. .el-table__row--striped td.el-table__cell {
  1463. background-color: #f8f9fc;
  1464. }
  1465. .el-table__header th {
  1466. background-color: #f5f7fa;
  1467. color: #333;
  1468. font-weight: 600;
  1469. }
  1470. .el-button--primary.is-link {
  1471. color: #4f46e5;
  1472. &:hover {
  1473. color: #6366f1;
  1474. }
  1475. }
  1476. .caret-wrapper {
  1477. .sort-caret.ascending {
  1478. border-bottom-color: #4f46e5;
  1479. }
  1480. .sort-caret.descending {
  1481. border-top-color: #4f46e5;
  1482. }
  1483. }
  1484. }
  1485. </style>