IotDeviceMainAlarm.vue 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347
  1. <template>
  2. <el-row :gutter="20">
  3. <!-- 左侧部门树 -->
  4. <DeptTree @node-click="handleDeptNodeClick" v-model:collapsed="isLeftContentCollapsed" />
  5. <el-col :xs="24" :span="isLeftContentCollapsed ? 24 : 20">
  6. <ContentWrap>
  7. <!-- 搜索工作栏 -->
  8. <el-form
  9. class="-mb-15px"
  10. :model="queryParams"
  11. ref="queryFormRef"
  12. :inline="true"
  13. label-width="68px"
  14. >
  15. <el-form-item :label="t('iotDevice.code')" prop="deviceCode" style="margin-left: 25px">
  16. <el-input
  17. v-model="queryParams.deviceCode"
  18. :placeholder="t('iotDevice.codeHolder')"
  19. clearable
  20. @keyup.enter="handleQuery"
  21. class="!w-200px"
  22. />
  23. </el-form-item>
  24. <el-form-item :label="t('iotDevice.name')" prop="deviceName">
  25. <el-input
  26. v-model="queryParams.deviceName"
  27. :placeholder="t('iotDevice.nameHolder')"
  28. clearable
  29. @keyup.enter="handleQuery"
  30. class="!w-200px"
  31. />
  32. </el-form-item>
  33. <el-form-item>
  34. <el-button @click="handleQuery"
  35. ><Icon icon="ep:search" class="mr-5px" /> {{ t('file.search') }}</el-button
  36. >
  37. <el-button @click="resetQuery"
  38. ><Icon icon="ep:refresh" class="mr-5px" /> {{ t('file.reset') }}</el-button
  39. >
  40. <el-button
  41. type="success"
  42. plain
  43. @click="handleExport"
  44. :loading="exportLoading"
  45. v-hasPermi="['rq:iot-device:export']"
  46. >
  47. <Icon icon="ep:download" class="mr-5px" /> 导出
  48. </el-button>
  49. </el-form-item>
  50. </el-form>
  51. </ContentWrap>
  52. <!-- 列表 -->
  53. <ContentWrap>
  54. <el-table
  55. height="calc(85vh - 135px)"
  56. v-loading="loading"
  57. :data="list"
  58. :stripe="true"
  59. :show-overflow-tooltip="true"
  60. >
  61. <el-table-column :label="t('monitor.serial')" width="70" align="center">
  62. <template #default="scope">
  63. {{ scope.$index + 1 }}
  64. </template>
  65. </el-table-column>
  66. <el-table-column :label="t('iotDevice.code')" align="center" prop="deviceCode" />
  67. <el-table-column :label="t('iotDevice.name')" align="center" prop="deviceName">
  68. <template #default="scope">
  69. <el-link :underline="false" type="primary" @click="handleDetail(scope.row.id)">
  70. {{ scope.row.deviceName }}
  71. </el-link>
  72. </template>
  73. </el-table-column>
  74. <el-table-column :label="t('bomList.serviceDue')" align="center">
  75. <template #default="scope">
  76. <template v-if="hasMaintenancePlan(scope.row.mainDistance)">
  77. <span :class="getDistanceClass(scope.row.mainDistance)">
  78. {{ scope.row.mainDistance }}
  79. </span>
  80. </template>
  81. <span v-else>无保养计划</span>
  82. </template>
  83. </el-table-column>
  84. <el-table-column
  85. v-if="showRunTime"
  86. label="累计运行时长H"
  87. align="center"
  88. prop="totalRunTime"
  89. />
  90. <el-table-column
  91. v-if="showMileage"
  92. label="累计运行里程KM"
  93. align="center"
  94. prop="totalMileage"
  95. />
  96. <el-table-column v-if="showMultiAttrs" label="多属性累计值" align="center" width="200">
  97. <template #default="scope">
  98. <template v-for="(value, key) in scope.row.multiAttrsTotalRuntime" :key="key">
  99. <span v-if="value && key"> {{ key }}: {{ value }}&nbsp;&nbsp;</span>
  100. </template>
  101. <template v-for="(value, key) in scope.row.multiAttrsTotalMileage" :key="key">
  102. <span v-if="value && key"> {{ key }}: {{ value }}&nbsp;&nbsp;</span>
  103. </template>
  104. </template>
  105. </el-table-column>
  106. <el-table-column :label="t('iotDevice.dept')" align="center" prop="deptName" />
  107. <el-table-column :label="t('devicePerson.rp')" align="center" prop="responsibleNames" />
  108. <el-table-column :label="t('monitor.status')" align="center" prop="deviceStatus">
  109. <template #default="scope">
  110. <dict-tag :type="DICT_TYPE.PMS_DEVICE_STATUS" :value="scope.row.deviceStatus" />
  111. </template>
  112. </el-table-column>
  113. <!-- 工单状态列 -->
  114. <el-table-column :label="t('operationFill.status')" align="center">
  115. <template #default="scope">
  116. <span v-if="scope.row.shouldWorkOrder && scope.row.runningWorkOrder">
  117. {{ t('mainPlan.generatedNotExecuted') }}
  118. </span>
  119. <span v-else-if="scope.row.shouldWorkOrder && !scope.row.runningWorkOrder">
  120. {{ t('mainPlan.notGenerated') }}
  121. </span>
  122. <span v-else>-</span>
  123. </template>
  124. </el-table-column>
  125. <el-table-column :label="t('monitor.operation')" align="center" min-width="60px">
  126. <template #default="scope">
  127. <el-button
  128. link
  129. type="primary"
  130. @click="openBomForm(scope.row)"
  131. v-hasPermi="['rq:iot-device:query']"
  132. v-if="hasMaintenancePlan(scope.row.mainDistance)"
  133. >
  134. {{ t('monitor.details') }}
  135. </el-button>
  136. </template>
  137. </el-table-column>
  138. </el-table>
  139. <!-- 分页 -->
  140. <Pagination
  141. :total="total"
  142. v-model:page="queryParams.pageNo"
  143. v-model:limit="queryParams.pageSize"
  144. @pagination="getList"
  145. />
  146. </ContentWrap>
  147. </el-col>
  148. </el-row>
  149. <DeviceAlarmBomList ref="modelFormRef" :flag="flag" />
  150. </template>
  151. <script setup lang="ts">
  152. import download from '@/utils/download'
  153. import { IotDeviceApi, IotDeviceVO } from '@/api/pms/device'
  154. import { IotMainWorkOrderApi } from '@/api/pms/iotmainworkorder'
  155. import { DICT_TYPE } from '@/utils/dict'
  156. import DeptTree from '@/views/system/user/DeptTree2.vue'
  157. import { useCache } from '@/hooks/web/useCache'
  158. import DeviceAlarmBomList from '@/views/pms/iotmainworkorder/DeviceAlarmBomList.vue'
  159. let isLeftContentCollapsed = ref(false)
  160. const showRunTime = computed(() => {
  161. const all = list.value?.map((item) => item.totalRunTime)
  162. return all.some((item) => Boolean(item))
  163. })
  164. const showMileage = computed(() => {
  165. const all = list.value?.map((item) => item.totalMileage)
  166. return all.some((item) => Boolean(item))
  167. })
  168. const showMultiAttrs = computed(() => {
  169. const all = list.value
  170. ?.map((item) => item.multiAttrsTotalRuntime)
  171. .concat(list.value.map((item) => item.multiAttrsTotalMileage))
  172. .flatMap((item) => Object.values(item ?? {}))
  173. return all.some((item) => Boolean(item))
  174. })
  175. const message = useMessage() // 消息弹窗
  176. const { t } = useI18n() // 国际化
  177. const { push } = useRouter() // 路由跳转
  178. const modelFormRef = ref()
  179. const loading = ref(true) // 列表的加载中
  180. const ifShow = ref(false)
  181. const list = ref<IotDeviceVO[]>([]) // 列表的数据
  182. const total = ref(0) // 列表的总页数
  183. const queryParams = reactive({
  184. pageNo: 1,
  185. pageSize: 10,
  186. deviceCode: undefined,
  187. deviceName: undefined,
  188. brand: undefined,
  189. model: undefined,
  190. deptId: undefined,
  191. deviceStatus: undefined,
  192. assetProperty: undefined,
  193. picUrl: undefined,
  194. remark: undefined,
  195. manufacturerId: undefined,
  196. supplierId: undefined,
  197. manDate: [],
  198. nameplate: undefined,
  199. expires: undefined,
  200. plPrice: undefined,
  201. plDate: [],
  202. plYear: undefined,
  203. plStartDate: [],
  204. plMonthed: undefined,
  205. plAmounted: undefined,
  206. remainAmount: undefined,
  207. infoId: undefined,
  208. infoType: undefined,
  209. infoName: undefined,
  210. infoRemark: undefined,
  211. infoUrl: undefined,
  212. templateJson: undefined,
  213. creator: undefined,
  214. setFlag: ''
  215. })
  216. const queryFormRef = ref() // 搜索的表单
  217. const flag = ref() // 查询保养计划或保养工单的标识
  218. const exportLoading = ref(false) // 导出的加载中
  219. const contentSpan = ref(20)
  220. const treeShow = ref(true)
  221. /** 查询列表 */
  222. const getList = async () => {
  223. loading.value = true
  224. try {
  225. const data = await IotMainWorkOrderApi.deviceMainDistances(queryParams)
  226. list.value = data.list
  227. total.value = data.total
  228. } finally {
  229. loading.value = false
  230. }
  231. }
  232. /** 处理部门被点击 */
  233. const handleDeptNodeClick = async (row) => {
  234. queryParams.deptId = row.id
  235. await getList()
  236. }
  237. /** 搜索按钮操作 */
  238. const handleQuery = () => {
  239. queryParams.pageNo = 1
  240. getList()
  241. }
  242. /** 重置按钮操作 */
  243. const resetQuery = () => {
  244. queryFormRef.value.resetFields()
  245. handleQuery()
  246. }
  247. const getDistanceClass = (distance: number | string | null) => {
  248. if (distance === null || distance === undefined) return ''
  249. // 如果是数字类型,直接处理
  250. if (typeof distance === 'number') {
  251. return distance < 0 ? 'negative-distance' : distance > 0 ? 'positive-distance' : ''
  252. }
  253. // 如果是字符串,提取数字部分
  254. if (typeof distance === 'string') {
  255. // 使用正则提取数字部分(包括负号、小数点和科学计数法)
  256. const numericPart = distance.match(/[-+]?\d*\.?\d+(?:[eE][-+]?\d+)?/)?.[0]
  257. // 如果提取到数字部分,转换为数值
  258. if (numericPart) {
  259. const num = parseFloat(numericPart)
  260. return num < 0 ? 'negative-distance' : num > 0 ? 'positive-distance' : ''
  261. }
  262. }
  263. return ''
  264. }
  265. // 判断是否有保养计划
  266. const hasMaintenancePlan = (mainDistance: any) => {
  267. // 检查:非null、非undefined、非空字符串
  268. return mainDistance != null && mainDistance !== ''
  269. }
  270. const handleDetail = (id: number) => {
  271. push({ name: 'DeviceDetailInfo', params: { id } })
  272. }
  273. const drawerVisible = ref<boolean>(false)
  274. const showDrawer = ref()
  275. const openBomForm = async (row) => {
  276. // 构建设备信息对象,包含所有需要的属性
  277. const deviceInfo = {
  278. deviceId: row.id,
  279. deviceCode: row.deviceCode,
  280. deviceName: row.deviceName,
  281. model: row.model // 新增 model 属性
  282. }
  283. if (row.workOrderId) {
  284. flag.value = 'workOrder'
  285. modelFormRef.value.open(row.workOrderId, flag.value, row.id)
  286. } else if (row.planId) {
  287. flag.value = 'plan'
  288. modelFormRef.value.open(row.planId, flag.value, deviceInfo)
  289. }
  290. }
  291. /** 导出按钮操作 */
  292. const handleExport = async () => {
  293. try {
  294. exportLoading.value = true
  295. const data = await IotDeviceApi.exportIotDeviceMainAlarm(queryParams)
  296. download.excel(data, '保养查询.xls')
  297. } catch {
  298. } finally {
  299. exportLoading.value = false
  300. }
  301. }
  302. const { wsCache } = useCache()
  303. /** 初始化 **/
  304. onMounted(() => {
  305. getList()
  306. })
  307. </script>
  308. <style scoped>
  309. /* 正数样式 - 淡绿色 */
  310. .positive-distance {
  311. color: #67c23a; /* element-plus 成功色 */
  312. background-color: rgba(103, 194, 58, 0.1); /* 10% 透明度的淡绿色背景 */
  313. padding: 2px 8px;
  314. border-radius: 4px;
  315. display: inline-block;
  316. }
  317. /* 负数样式 - 淡红色 */
  318. .negative-distance {
  319. color: #f56c6c; /* element-plus 危险色 */
  320. background-color: rgba(245, 108, 108, 0.1); /* 10% 透明度的淡红色背景 */
  321. padding: 2px 8px;
  322. border-radius: 4px;
  323. display: inline-block;
  324. }
  325. </style>