EventDetail.vue 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392
  1. <template>
  2. <Dialog type="xl" :title="title" class="custom-dialog" customShow hide-footer @close="handleClose">
  3. <div class="common-header">
  4. <!-- <div v-if="detailData.del_flag !== '2'" class="common-btn-primary2" @click="handleUpdate">-->
  5. <!-- <i class="edit-icon" />-->
  6. <!-- 编辑-->
  7. <!-- </div>-->
  8. <div v-if="detailData.event_status === '0'" class="common-btn-primary2" style="margin-left: 20px" @click="handleStartEvent">
  9. <i class="command-icon" />
  10. 开始指挥
  11. </div>
  12. <div v-if="detailData.event_status === '2'" class="common-btn-danger" style="margin-left: 20px" @click="handleCloseEvent">
  13. <i class="close-icon" />
  14. 关闭事件
  15. </div>
  16. <div v-if="detailData.del_flag === '0'" class="common-btn-danger" style="margin-left: 20px" @click="handleShowDeleteDialog">删除事件</div>
  17. </div>
  18. <div class="scroll-box">
  19. <div class="common-title-box">基础信息</div>
  20. <div class="form">
  21. <div class="form-item">
  22. <div class="text1">事件编号</div>
  23. <div class="text2">{{ detailData.event_id }}</div>
  24. </div>
  25. <div class="form-item">
  26. <div class="text1">事件类型</div>
  27. <div class="text2">
  28. <dict-tag :options="mm_event_type" :value="detailData.event_type" />
  29. </div>
  30. </div>
  31. <div class="form-item">
  32. <div class="text1">事件等级</div>
  33. <div class="text2">
  34. <dict-tag :options="mm_event_level" :value="detailData.event_level" />
  35. <!-- <el-icon style="margin-left: 20px; cursor: pointer" @click="handleEventLevelOpen"><Fold /></el-icon>-->
  36. </div>
  37. </div>
  38. <div class="form-item">
  39. <div class="text1">事件状态</div>
  40. <div class="text2">
  41. <dict-tag :options="mm_event_state" :value="detailData.event_status" />
  42. </div>
  43. </div>
  44. <div class="form-item">
  45. <div class="text1">事发地点</div>
  46. <div class="text2">{{ detailData.address }}</div>
  47. </div>
  48. <div class="form-item">
  49. <div class="text1">事发时间</div>
  50. <div class="text2">{{ detailData.event_time }}</div>
  51. </div>
  52. <div class="form-item">
  53. <div class="text1">上报时间</div>
  54. <div class="text2">{{ detailData.report_time }}</div>
  55. </div>
  56. <div class="form-item">
  57. <div class="text1">伤亡情况</div>
  58. <div class="text2">
  59. <div v-if="detailData.casualties == '0'" class="flex">
  60. <span>未上报</span>
  61. <!-- <span class="link" style="margin-left: 20px" @click="uploadeCasualties">去上报</span>-->
  62. </div>
  63. <div v-if="detailData.casualties == '1'" class="flex">
  64. <span>已上报</span>
  65. </div>
  66. </div>
  67. </div>
  68. <div class="form-item">
  69. <div class="text1">登记人</div>
  70. <div class="text2">{{ detailData.reported_by }}</div>
  71. </div>
  72. <div class="form-item">
  73. <div class="text1">登记时间</div>
  74. <div class="text2">{{ detailData.report_time }}</div>
  75. </div>
  76. <div class="form-item">
  77. <div class="text1">联系方式</div>
  78. <div class="text2">{{ detailData.contact }}</div>
  79. </div>
  80. <div class="form-item">
  81. <div class="text1">事件来源</div>
  82. <div class="text2">{{ detailData.event_source }}</div>
  83. </div>
  84. </div>
  85. <div class="common-title-box">事件概要</div>
  86. <div v-if="!!detailData.event_description" class="text-box">{{ detailData.event_description }}</div>
  87. <div class="common-title-box">事件跟踪</div>
  88. <div style="padding: 0px 50px 15px 16px">
  89. <Step :step-data="eventTrackState" style="width: 100%" />
  90. </div>
  91. <div class="common-title-box">指挥记录</div>
  92. <div class="info-content">
  93. <div class="list">
  94. <div class="list-item">
  95. <i class="img" />
  96. <div class="item-title">大屏指挥记录</div>
  97. </div>
  98. <div class="list-item">
  99. <i class="img" />
  100. <div class="item-title">中屏指挥记录</div>
  101. </div>
  102. </div>
  103. </div>
  104. <div class="common-title-box">匹配预案</div>
  105. <div v-if="!!planFiles && planFiles.length > 0" class="common-info-content">
  106. <div class="list2">
  107. <div v-for="(item, index) in planFiles" :key="index" class="list-item" style="margin-bottom: 10px">
  108. <div class="link" @click="previewSummaryFile2(item)">{{ item.name }}</div>
  109. <div style="margin-left: 40px" @click="previewSummaryFile2(item)">查看</div>
  110. <div style="margin-left: 20px; display: flex; align-items: center" @click="downloadSummaryFile2(item)">
  111. <span>下载</span>
  112. <el-icon class="icon" style="margin-left: 0px"><Download /></el-icon>
  113. </div>
  114. </div>
  115. </div>
  116. </div>
  117. <!-- <div v-if="!!detailData.plan_name" class="info-content">-->
  118. <!-- <div class="list">-->
  119. <!-- <div class="list-item">-->
  120. <!-- <i class="img" />-->
  121. <!-- <div class="item-title">{{ detailData.plan_name }}</div>-->
  122. <!-- </div>-->
  123. <!-- </div>-->
  124. <!-- </div>-->
  125. <div class="common-title-box">事件总结报告</div>
  126. <div v-if="!!summaryFiles && summaryFiles.length > 0" class="info-content">
  127. <div class="list2">
  128. <div v-for="(item, index) in summaryFiles" :key="index" class="list-item" style="margin-bottom: 10px">
  129. <div class="link" @click="previewSummaryFile(item)">{{ item.file_name }}</div>
  130. <div style="margin-left: 40px" @click="previewSummaryFile(item)">查看</div>
  131. <div style="margin-left: 20px; display: flex; align-items: center" @click="downloadSummaryFile(item)">
  132. <span>下载</span>
  133. <el-icon class="icon" style="margin-left: 0px"><Download /></el-icon>
  134. </div>
  135. </div>
  136. </div>
  137. </div>
  138. <!-- <div v-if="!!summaryFiles && summaryFiles.length > 0" class="info-content">-->
  139. <!-- <div class="list2">-->
  140. <!-- <div v-for="(item, index) in summaryFiles" :key="index" class="list-item" @click="downloadSummaryFile(item.url)">-->
  141. <!-- <div class="link">{{ item.file_name }}</div>-->
  142. <!-- <el-icon class="icon"><Download /></el-icon>-->
  143. <!-- </div>-->
  144. <!-- </div>-->
  145. <!-- </div>-->
  146. </div>
  147. <Dialog v-model="showDeleteDialog" class="tip" type="xs" title="提示">确认删除事件吗</Dialog>
  148. </Dialog>
  149. <Dialog v-if="dialogTableVisible" v-model="dialogTableVisible" title="文件预览" height="1800px">
  150. <div style="font-size: 18px;font-weight: bold;margin-bottom: 10px;">{{ detailInfo.file_name }}</div>
  151. <pdf-viewer :url="baseUrl + '/file/download/' + detailInfo.url" />
  152. </Dialog>
  153. </template>
  154. <script lang="ts" setup name="EventDetail">
  155. import { getEventDetail } from '@/api/duty/eventing';
  156. import { download2 } from '@/utils/request';
  157. import PdfViewer from '@/views/knowledge/HiddenStandards/PdfViewer.vue';
  158. const props = defineProps({
  159. modelValue: {
  160. type: Boolean,
  161. default: () => {
  162. return false;
  163. }
  164. },
  165. eventId: String,
  166. title: String
  167. });
  168. const { proxy } = getCurrentInstance() as ComponentInternalInstance;
  169. const { mm_event_type, mm_event_level, mm_event_state, region } = toRefs<any>(
  170. proxy?.useDict('mm_event_type', 'mm_event_level', 'mm_event_state', 'region')
  171. );
  172. const emits = defineEmits(['update:modelValue']);
  173. const detailData = ref({
  174. event_id: '',
  175. event_title: '',
  176. event_type: '',
  177. event_level: '',
  178. event_status: '',
  179. event_time: '',
  180. address: '',
  181. report_time: '',
  182. event_source: '',
  183. reported_by: '',
  184. event_description: '',
  185. contact: '',
  186. plan_name: '',
  187. casualties: '',
  188. deaths: '',
  189. injuries: '',
  190. missing: '',
  191. del_flag: ''
  192. });
  193. // 事件跟踪数据
  194. let eventTrackState = reactive({
  195. active: 0,
  196. items: [
  197. {
  198. title: '事件登记',
  199. description: ''
  200. },
  201. {
  202. title: '进入指挥',
  203. description: ''
  204. },
  205. {
  206. title: '结束指挥',
  207. description: ''
  208. },
  209. {
  210. title: '关闭事件',
  211. description: ''
  212. }
  213. ]
  214. });
  215. // 事件等级数据
  216. let eventLevelState = reactive({
  217. show: false,
  218. data: []
  219. });
  220. // 下载总结报告
  221. let summaryFiles = ref([]);
  222. let planFiles = ref([]);
  223. const baseUrl = import.meta.env.VITE_APP_BASE_API;
  224. const downloadSummaryFile = (file: any) => {
  225. // window.open(url);
  226. download2(baseUrl + '/file/download/' + file.url, file.file_name);
  227. };
  228. const detailInfo = ref({});
  229. const detailInfo2 = ref({});
  230. const dialogTableVisible = ref(false);
  231. const dialogTableVisible2 = ref(false);
  232. const previewSummaryFile = (file) => {
  233. detailInfo.value = file;
  234. if (file) {
  235. dialogTableVisible.value = true;
  236. }
  237. };
  238. const previewSummaryFile2 = (file) => {
  239. detailInfo2.value = file;
  240. if (file) {
  241. dialogTableVisible2.value = true;
  242. }
  243. };
  244. const downloadSummaryFile2 = (file: any) => {
  245. download2(baseUrl + '/file/download/' + file.url, file.name);
  246. };
  247. const handleClose = () => {
  248. emits('update:modelValue', false);
  249. };
  250. const handleUpdate = () => {
  251. console.log('handleUpdate');
  252. // eventEditDialogState.title = '修改事件';
  253. // eventEditDialogState.show = true;
  254. };
  255. const showDeleteDialog = ref(false);
  256. // 删除事件
  257. const handleShowDeleteDialog = () => {
  258. showDeleteDialog.value = true;
  259. };
  260. const handleDeleteEvent = () => {
  261. deleteEvent({ eventId: props.eventId }).then((res) => {
  262. proxy?.$modal.msgSuccess(res.msg);
  263. fetchEventDetail();
  264. });
  265. };
  266. const initData = () => {
  267. getEventDetail({ event_id: props.eventId }).then((res) => {
  268. detailData.value = res.data;
  269. eventTrackState = res.data.event_status_tracks;
  270. eventLevelState.data = res.data.event_level_tracks;
  271. summaryFiles.value = res.data.summary_file;
  272. planFiles.value = res.data.plan_files;
  273. });
  274. };
  275. onMounted(() => {
  276. initData();
  277. });
  278. </script>
  279. <style lang="scss" scoped>
  280. .custom-dialog {
  281. :deep(.dialog-header) {
  282. pointer-events: none;
  283. .dialog-close {
  284. pointer-events: auto;
  285. }
  286. }
  287. :deep(.dialog-content) {
  288. margin-top: -20px;
  289. }
  290. .tip {
  291. :deep(.dialog-content) {
  292. margin-top: 0;
  293. }
  294. }
  295. .common-header {
  296. display: flex;
  297. align-items: center;
  298. justify-content: flex-end;
  299. }
  300. .form {
  301. display: flex;
  302. flex-wrap: wrap;
  303. align-items: center;
  304. padding: 0 15px 10px;
  305. .form-item {
  306. width: calc(25% - 15px);
  307. margin-left: 20px;
  308. &:nth-child(4n + 1) {
  309. margin-left: 0;
  310. }
  311. .text1 {
  312. font-size: 14px;
  313. color: #8da1be;
  314. line-height: 30px;
  315. }
  316. .text2 {
  317. font-size: 14px;
  318. line-height: 30px;
  319. color: #ffffff;
  320. display: flex;
  321. align-items: center;
  322. }
  323. }
  324. }
  325. .common-title-box {
  326. margin-bottom: 10px;
  327. }
  328. .list {
  329. display: flex;
  330. .list-item {
  331. display: flex;
  332. flex-direction: column;
  333. align-content: center;
  334. justify-content: center;
  335. margin-right: 20px;
  336. cursor: pointer;
  337. position: relative;
  338. .img {
  339. width: 251px;
  340. height: 150px;
  341. background: url('@/assets/images/cardBg.png') no-repeat;
  342. background-size: 100% 100%;
  343. cursor: pointer;
  344. }
  345. .item-title {
  346. position: absolute;
  347. left: 50%;
  348. bottom: 16px;
  349. transform: translateX(-50%);
  350. color: #2c81ff;
  351. }
  352. &:last-child {
  353. margin-right: 0;
  354. }
  355. }
  356. .list-item2 {
  357. margin-right: 40px;
  358. cursor: pointer;
  359. &:last-child {
  360. margin-right: 0;
  361. }
  362. }
  363. }
  364. .info-content {
  365. padding: 20px 35px 30px;
  366. }
  367. .list2 {
  368. .list-item {
  369. display: flex;
  370. align-items: center;
  371. color: #2c81ff;
  372. cursor: pointer;
  373. .icon {
  374. margin-left: 10px;
  375. cursor: pointer;
  376. }
  377. }
  378. }
  379. .scroll-box {
  380. height: 600px;
  381. overflow-y: auto;
  382. }
  383. }
  384. </style>