533 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
		
		
			
		
	
	
			533 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
|  | <!-- | ||
|  |  * @Author: SunTao 328867980@qq.com | ||
|  |  * @Date: 2024-10-22 15:30:25 | ||
|  |  * @LastEditors: SunTao 328867980@qq.com | ||
|  |  * @LastEditTime: 2024-10-23 10:38:26 | ||
|  |  * @FilePath: \znxjxt-ui\src\views\big-screen\components\img-dialog.vue | ||
|  |  * @Description: 大屏查看图片弹窗 | ||
|  | --> | ||
|  | 
 | ||
|  | <template> | ||
|  |   <div class="app"> | ||
|  |     <div class="sidebar" ref="sidebar" @scroll="handleScroll"> | ||
|  |       <img | ||
|  |         v-for="(item, index) in defectData" | ||
|  |         :key="item.id" | ||
|  |         :src="item.media[0].img" | ||
|  |         :alt="'Image ' + (index + 1)" | ||
|  |         @click="showImage(index)" | ||
|  |         :class="{ selected: currentIndex === index }" | ||
|  |       /> | ||
|  |     </div> | ||
|  | 
 | ||
|  |     <div class="main-content"> | ||
|  |       <!-- 搜索表单 --> | ||
|  |       <div class="toolbar"> | ||
|  |         <el-form | ||
|  |           :model="imgForm" | ||
|  |           ref="imgForm" | ||
|  |           size="small" | ||
|  |           :inline="true" | ||
|  |           label-width="5rem" | ||
|  |         > | ||
|  |           <el-row :gutter="24"> | ||
|  |             <el-col :span="8"> | ||
|  |               <el-form-item label="路段名称"> | ||
|  |                 <el-select | ||
|  |                   v-model="imgForm.roadName" | ||
|  |                   placeholder="请选择路段名称" | ||
|  |                   :popper-append-to-body="false" | ||
|  |                   clearable | ||
|  |                 > | ||
|  |                   <el-option | ||
|  |                     v-for="item in roadTypeList" | ||
|  |                     :key="item.value" | ||
|  |                     :label="item.label" | ||
|  |                     :value="item.value" | ||
|  |                   /> | ||
|  |                 </el-select> | ||
|  |               </el-form-item> | ||
|  |             </el-col> | ||
|  |             <el-col :span="10"> | ||
|  |               <el-form-item label="公里桩"> | ||
|  |                 <el-input | ||
|  |                   v-model="imgForm.stakeStart" | ||
|  |                   placeholder="起始公里桩" | ||
|  |                   style="width: 8rem" | ||
|  |                   clearable | ||
|  |                 /> | ||
|  |                 <span style="margin: 0 5px">-</span> | ||
|  |                 <el-input | ||
|  |                   v-model="imgForm.stakeEnd" | ||
|  |                   placeholder="终止公里桩" | ||
|  |                   style="width: 8rem" | ||
|  |                   clearable | ||
|  |                 /> | ||
|  |               </el-form-item> | ||
|  |             </el-col> | ||
|  |             <el-col :span="6"> | ||
|  |               <el-button | ||
|  |                 type="primary" | ||
|  |                 icon="el-icon-search" | ||
|  |                 size="mini" | ||
|  |                 @click="handleQuery" | ||
|  |                 >搜索</el-button | ||
|  |               > | ||
|  |               <el-button icon="el-icon-refresh" size="mini" @click="resetQuery" | ||
|  |                 >重置</el-button | ||
|  |               > | ||
|  |             </el-col> | ||
|  |           </el-row> | ||
|  |         </el-form> | ||
|  |       </div> | ||
|  |       <!-- 图片展示 --> | ||
|  |       <div class="image-viewer"> | ||
|  |         <div class="image-container" ref="imageContainer"> | ||
|  |           <img | ||
|  |             :src="currentImage" | ||
|  |             alt="Main Image" | ||
|  |             ref="mainImage" | ||
|  |             @load="updateRects" | ||
|  |           /> | ||
|  |           <div | ||
|  |             v-for="(rect, index) in rects" | ||
|  |             :key="index" | ||
|  |             class="rect-overlay" | ||
|  |             :style="getRectStyle(rect)" | ||
|  |           ></div> | ||
|  |         </div> | ||
|  |         <div class="image-info"> | ||
|  |           采集时间: 2024-08-26 15:44:42<br /> | ||
|  |           起始桩号: K0003+204 终止桩号: K0003+204<br /> | ||
|  |           路产状态: 新增 融合状态: 未融合 | ||
|  |         </div> | ||
|  |       </div> | ||
|  |       <!-- 小图展示 --> | ||
|  |       <div class="thumbnail-row"> | ||
|  |         <img | ||
|  |           v-for="(mediaItem, index) in currentThumbnails" | ||
|  |           :key="'thumb-' + index" | ||
|  |           :src="mediaItem.img" | ||
|  |           :alt="'Thumb ' + (index + 1)" | ||
|  |           @click="showThumbnailImage(index)" | ||
|  |           :class="{ selected: selectedThumbnail === index }" | ||
|  |         /> | ||
|  |       </div> | ||
|  |     </div> | ||
|  |   </div> | ||
|  | </template> | ||
|  | 
 | ||
|  | <script> | ||
|  | import { listDefect, getSegment } from "@/api/xj/disease"; | ||
|  | 
 | ||
|  | export default { | ||
|  |   name: "ImgDialog", | ||
|  |   data() { | ||
|  |     return { | ||
|  |       // 左侧图片数据
 | ||
|  |       defectData: [], | ||
|  |       // 当前选择的index
 | ||
|  |       currentIndex: 0, | ||
|  |       // 点击的index绑定
 | ||
|  |       selectedThumbnail: null, | ||
|  |       // 加载状态
 | ||
|  |       loading: false, | ||
|  |       // 表单绑定
 | ||
|  |       imgForm: { | ||
|  |         // 路段名称
 | ||
|  |         roadName: "", | ||
|  |         // 起始公里桩
 | ||
|  |         stakeStart: "", | ||
|  |         // 终止公里桩
 | ||
|  |         stakeEnd: "", | ||
|  |       }, | ||
|  |       // 路段下拉数据
 | ||
|  |       roadTypeList: [], | ||
|  |       // 分页绑定
 | ||
|  |       params: { | ||
|  |         page: 1, | ||
|  |         size: 10, | ||
|  |       }, | ||
|  |       rects: [], | ||
|  |     }; | ||
|  |   }, | ||
|  |   computed: { | ||
|  |     currentImage() { | ||
|  |       const thumbnails = this.currentThumbnails; | ||
|  |       return ( | ||
|  |         thumbnails[this.selectedThumbnail]?.img || thumbnails[0]?.img || "" | ||
|  |       ); | ||
|  |     }, | ||
|  |     currentThumbnails() { | ||
|  |       return this.defectData[this.currentIndex]?.media || []; | ||
|  |     }, | ||
|  |   }, | ||
|  |   created() { | ||
|  |     this.getSegmentList(); | ||
|  |   }, | ||
|  |   methods: { | ||
|  |     /* 获取路段下拉数据 */ | ||
|  |     getSegmentList() { | ||
|  |       getSegment().then(({ code, data }) => { | ||
|  |         if (code === 200) { | ||
|  |           this.roadTypeList = data; | ||
|  |         } | ||
|  |       }); | ||
|  |     }, | ||
|  |     /* 获取图片列表/点击搜索事件 */ | ||
|  |     getList() { | ||
|  |       this.loading = true; | ||
|  |       const data = { | ||
|  |         ...this.imgForm, | ||
|  |         ...this.params, | ||
|  |       }; | ||
|  |       listDefect(data) | ||
|  |         .then((response) => { | ||
|  |           this.defectData.push(...response.rows); | ||
|  |           this.loading = false; | ||
|  |         }) | ||
|  |         .catch(() => { | ||
|  |           this.loading = false; | ||
|  |         }); | ||
|  |     }, | ||
|  |     /* 点击搜索事件 */ | ||
|  |     handleQuery() { | ||
|  |       const stakeReg = /^K\d{4}\+\d{3}$/; | ||
|  |       if (this.imgForm.stakeStart) { | ||
|  |         if (stakeReg.test(this.imgForm.stakeStart)) { | ||
|  |           this.defectData = []; | ||
|  |           this.params = { | ||
|  |             page: 1, | ||
|  |             size: 10, | ||
|  |           }; | ||
|  |           this.getList(); | ||
|  |           this.showImage(0); | ||
|  |         } else { | ||
|  |           this.$modal.msgWarning("请按照'K0000+000'格式填写公里桩进行修改"); | ||
|  |         } | ||
|  |       } else if (this.imgForm.stakeEnd) { | ||
|  |         if (stakeReg.test(this.imgForm.stakeEnd)) { | ||
|  |           this.defectData = []; | ||
|  |           this.params = { | ||
|  |             page: 1, | ||
|  |             size: 10, | ||
|  |           }; | ||
|  |           this.getList(); | ||
|  |           this.showImage(0); | ||
|  |         } else { | ||
|  |           this.$modal.msgWarning("请按照'K0000+000'格式填写公里桩进行修改"); | ||
|  |         } | ||
|  |       } else { | ||
|  |         this.defectData = []; | ||
|  | 
 | ||
|  |         this.params = { | ||
|  |           page: 1, | ||
|  |           size: 10, | ||
|  |         }; | ||
|  |         this.getList(); | ||
|  |         this.showImage(0); | ||
|  |       } | ||
|  |     }, | ||
|  |     /* 重置事件 */ | ||
|  |     resetQuery() { | ||
|  |       this.defectData = []; | ||
|  |       this.imgForm = { | ||
|  |         roadName: "", | ||
|  |         stakeStart: "", | ||
|  |         stakeEnd: "", | ||
|  |       }; | ||
|  |       this.params = { | ||
|  |         page: 1, | ||
|  |         size: 10, | ||
|  |       }; | ||
|  |       this.getList(); | ||
|  |       this.showImage(0); | ||
|  |     }, | ||
|  |     /* 左侧点击图片事件 */ | ||
|  |     showImage(index) { | ||
|  |       this.currentIndex = index; | ||
|  |       this.selectedThumbnail = 0; // Reset to the first thumbnail when changing the main item
 | ||
|  |       this.scrollToCurrentImage(); | ||
|  |       this.$nextTick(() => { | ||
|  |         this.updateRects(); | ||
|  |       }); | ||
|  |     }, | ||
|  |     showThumbnailImage(index) { | ||
|  |       this.selectedThumbnail = index; | ||
|  |       this.updateRects(); | ||
|  |     }, | ||
|  |     scrollToCurrentImage() { | ||
|  |       const sidebarImages = this.$refs.sidebar.querySelectorAll("img"); | ||
|  |       const currentImageElement = sidebarImages[this.currentIndex]; | ||
|  |       if (currentImageElement) { | ||
|  |         currentImageElement.scrollIntoView({ | ||
|  |           behavior: "smooth", | ||
|  |           block: "nearest", | ||
|  |         }); | ||
|  |       } | ||
|  |     }, | ||
|  |     /* 滚动触发事件 */ | ||
|  |     handleScroll() { | ||
|  |       // 滚动高度+容器高度  滚动区域高度
 | ||
|  |       const sidebar = this.$refs.sidebar; | ||
|  |       if ( | ||
|  |         sidebar.scrollTop + sidebar.clientHeight >= sidebar.scrollHeight - 1 && | ||
|  |         !this.loading | ||
|  |       ) { | ||
|  |         this.loadMoreImages(); | ||
|  |       } | ||
|  |     }, | ||
|  |     /* 加载更多图片方法 */ | ||
|  |     loadMoreImages() { | ||
|  |       this.params.page += 1; | ||
|  |       this.getList(); | ||
|  |     }, | ||
|  |     /* 键盘事件 */ | ||
|  |     handleKeydown(event) { | ||
|  |       if (event.key === "ArrowUp") { | ||
|  |         if (this.currentIndex > 0) { | ||
|  |           this.currentIndex--; | ||
|  |           this.selectedThumbnail = 0; // Reset when changing main images via keyboard
 | ||
|  |           this.scrollToCurrentImage(); | ||
|  |           this.updateRects(); | ||
|  |         } | ||
|  |       } else if (event.key === "ArrowDown") { | ||
|  |         if (this.currentIndex <= this.defectData.length - 1) { | ||
|  |           this.currentIndex === this.defectData.length - 1 | ||
|  |             ? this.currentIndex | ||
|  |             : this.currentIndex++; | ||
|  |           this.selectedThumbnail = 0; // Reset when changing main images via keyboard
 | ||
|  |           this.scrollToCurrentImage(); | ||
|  |           this.updateRects(); | ||
|  |         } else { | ||
|  |           this.loadMoreImages(); | ||
|  |         } | ||
|  |       } | ||
|  |     }, | ||
|  |     /* 图片位置信息获取 */ | ||
|  |     updateRects() { | ||
|  |       const image = this.$refs.mainImage; | ||
|  |       const rects = | ||
|  |         this.defectData[this.currentIndex]?.media[this.selectedThumbnail]?.rect | ||
|  |           .split(",") | ||
|  |           .map(Number) || []; | ||
|  |       this.rects = [ | ||
|  |         { | ||
|  |           left: rects[0], | ||
|  |           top: rects[1], | ||
|  |           width: rects[2], | ||
|  |           height: rects[3], | ||
|  |         }, | ||
|  |       ]; | ||
|  |     }, | ||
|  |     /* 图片红框位置 */ | ||
|  |     getRectStyle({ left, top, width, height }) { | ||
|  |       const image = this.$refs.mainImage; | ||
|  |       const container = this.$refs.imageContainer; | ||
|  |       if (!image || !container) return {}; | ||
|  | 
 | ||
|  |       const scaleX = container.clientWidth / image.naturalWidth; | ||
|  |       const scaleY = container.clientHeight / image.naturalHeight; | ||
|  |       const scale = Math.min(scaleX, scaleY); | ||
|  | 
 | ||
|  |       const renderedWidth = image.naturalWidth * scale; | ||
|  |       const renderedHeight = image.naturalHeight * scale; | ||
|  | 
 | ||
|  |       const offsetX = (container.clientWidth - renderedWidth) / 2; | ||
|  |       const offsetY = (container.clientHeight - renderedHeight) / 2; | ||
|  | 
 | ||
|  |       return { | ||
|  |         position: "absolute", | ||
|  |         left: `${left * scale + offsetX}px`, | ||
|  |         top: `${top * scale + offsetY}px`, | ||
|  |         width: `${width * scale}px`, | ||
|  |         height: `${height * scale}px`, | ||
|  |         border: "2px solid #FF0000", | ||
|  |         boxSizing: "border-box", | ||
|  |       }; | ||
|  |     }, | ||
|  |   }, | ||
|  |   mounted() { | ||
|  |     this.getList(); | ||
|  |     window.addEventListener("keydown", this.handleKeydown); | ||
|  |     window.addEventListener("resize", this.updateRects); | ||
|  |   }, | ||
|  |   beforeDestroy() { | ||
|  |     window.removeEventListener("keydown", this.handleKeydown); | ||
|  |     window.removeEventListener("resize", this.updateRects); | ||
|  |   }, | ||
|  | }; | ||
|  | </script> | ||
|  | 
 | ||
|  | <style lang="scss" scoped> | ||
|  | .app { | ||
|  |   width: 100%; | ||
|  |   display: flex; | ||
|  |   flex-direction: row; | ||
|  |   height: 45rem; | ||
|  |   background-color: #2e3a46; | ||
|  | } | ||
|  | 
 | ||
|  | .sidebar { | ||
|  |   width: 250px; | ||
|  |   background-color: #1f292e; | ||
|  |   padding: 10px; | ||
|  |   overflow-y: auto; | ||
|  | } | ||
|  | 
 | ||
|  | .sidebar::-webkit-scrollbar { | ||
|  |   width: 8px; | ||
|  | } | ||
|  | 
 | ||
|  | .sidebar::-webkit-scrollbar-track { | ||
|  |   background: #2e3a46; | ||
|  |   border-radius: 10px; | ||
|  | } | ||
|  | 
 | ||
|  | .sidebar::-webkit-scrollbar-thumb { | ||
|  |   background: #444c51; | ||
|  |   border-radius: 10px; | ||
|  | } | ||
|  | 
 | ||
|  | .sidebar::-webkit-scrollbar-thumb:hover { | ||
|  |   background: #00aaff; | ||
|  | } | ||
|  | 
 | ||
|  | .sidebar img { | ||
|  |   width: 100%; | ||
|  |   margin-bottom: 5px; | ||
|  |   cursor: pointer; | ||
|  |   border-radius: 3px; | ||
|  |   transition: transform 0.2s, border 0.2s; | ||
|  |   box-sizing: border-box; | ||
|  | } | ||
|  | 
 | ||
|  | .sidebar img.selected { | ||
|  |   border: 3px solid #00aaff; | ||
|  |   transform: scale(1.05); | ||
|  | } | ||
|  | 
 | ||
|  | .main-content { | ||
|  |   width: calc(100% - 250px); | ||
|  |   background-color: #1b2126; | ||
|  |   padding: 10px; | ||
|  |   display: flex; | ||
|  |   flex-direction: column; | ||
|  | } | ||
|  | 
 | ||
|  | .toolbar { | ||
|  |   width: 100%; | ||
|  |   height: 5rem; | ||
|  |   display: flex; | ||
|  |   align-items: center; | ||
|  |   background-color: #333c42; | ||
|  |   padding: 5px; | ||
|  |   border-bottom: 1px solid #333c42; | ||
|  | 
 | ||
|  |   ::v-deep .el-form { | ||
|  |     width: 100%; | ||
|  | 
 | ||
|  |     .el-form-item--small .el-form-item__label { | ||
|  |       color: #ffffff; | ||
|  |     } | ||
|  | 
 | ||
|  |     .el-select { | ||
|  |       width: 100%; | ||
|  | 
 | ||
|  |       .el-input--small .el-input__inner { | ||
|  |         color: #ffffff; | ||
|  |         background-color: #333c42; | ||
|  |       } | ||
|  | 
 | ||
|  |       .el-select-dropdown { | ||
|  |         background-color: #102649; | ||
|  |         border-color: #08204f; | ||
|  |         .el-scrollbar { | ||
|  |           .el-select-dropdown__wrap { | ||
|  |             .el-scrollbar__view { | ||
|  |               .el-select-dropdown__item { | ||
|  |                 color: #ffffff; | ||
|  |               } | ||
|  |               .el-select-dropdown__item:hover { | ||
|  |                 background-color: #2b4c7e; | ||
|  |               } | ||
|  | 
 | ||
|  |               .el-select-dropdown__item.selected { | ||
|  |                 background-color: #2b4c7e; | ||
|  |               } | ||
|  | 
 | ||
|  |               .el-select-dropdown__item.hover { | ||
|  |                 background-color: #2b4c7e; | ||
|  |               } | ||
|  |             } | ||
|  | 
 | ||
|  |             .el-select-dropdown__list { | ||
|  |               background-color: #333c42; | ||
|  |             } | ||
|  |           } | ||
|  |         } | ||
|  |       } | ||
|  |     } | ||
|  | 
 | ||
|  |     .el-input--small .el-input__inner { | ||
|  |       color: #ffffff; | ||
|  |       background-color: #333c42; | ||
|  |     } | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | .image-container { | ||
|  |   position: relative; | ||
|  |   width: 100%; | ||
|  |   height: 80%; | ||
|  |   display: flex; | ||
|  |   align-items: center; | ||
|  |   justify-content: center; | ||
|  |   overflow: hidden; | ||
|  | } | ||
|  | 
 | ||
|  | .image-container img { | ||
|  |   max-width: 100%; | ||
|  |   max-height: 100%; | ||
|  |   object-fit: contain; | ||
|  | } | ||
|  | 
 | ||
|  | .rect-overlay { | ||
|  |   position: absolute; | ||
|  |   pointer-events: none; | ||
|  |   border: 2px solid red; | ||
|  | } | ||
|  | 
 | ||
|  | .image-viewer { | ||
|  |   height: 33rem; | ||
|  |   display: flex; | ||
|  |   flex-direction: column; | ||
|  |   align-items: center; | ||
|  |   justify-content: center; | ||
|  |   background-color: #22272b; | ||
|  |   position: relative; | ||
|  | } | ||
|  | 
 | ||
|  | .thumbnail-row { | ||
|  |   display: flex; | ||
|  |   justify-content: center; | ||
|  |   overflow-x: auto; | ||
|  |   padding: 10px; | ||
|  |   background-color: #333c42; | ||
|  |   gap: 10px; | ||
|  | } | ||
|  | 
 | ||
|  | .thumbnail-row img { | ||
|  |   width: 100px; | ||
|  |   height: 60px; | ||
|  |   cursor: pointer; | ||
|  |   border-radius: 3px; | ||
|  |   transition: transform 0.2s, border 0.2s; | ||
|  |   box-sizing: border-box; | ||
|  | } | ||
|  | 
 | ||
|  | .thumbnail-row img.selected { | ||
|  |   border: 3px solid #00aaff; | ||
|  | } | ||
|  | </style> |