index.vue 49 KB

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