| | |
| | | <template> |
| | | <dv-full-screen-container class="full-screen-container"> |
| | | <header class="page-header"> |
| | | {{getWorkshopName}} |
| | | <div class="header-right"> |
| | | <a-button type="primary" icon="save" size="large" @click="saveDevicePositionByApi">保存位置</a-button> |
| | | <header class="page-header" :style="{height: pageHeaderHeight+'px'}"> |
| | | {{ workshopDetails.workshopName }} |
| | | <div class="header-left" v-has="'home:saveDevicePositionAndSize'"> |
| | | <a-space> |
| | | <span v-if="!isSwitchChecked">开启功能</span> |
| | | <span v-else>关闭功能</span> |
| | | <a-switch |
| | | checked-children="开" |
| | | un-checked-children="关" |
| | | @change="handleSwitchChange" |
| | | v-model="isSwitchChecked" |
| | | :disabled="isSwitchChecked" |
| | | /> |
| | | </a-space> |
| | | <a-button type="primary" icon="save" size="large" @click="saveDevicePositionAndSizeByApi">保存位置</a-button> |
| | | </div> |
| | | <div class="device-status-info"> |
| | | <a-space v-for="item in deviceStatusList" :key="item.value" class="single-status-info"> |
| | | <div>{{ item.label }}</div> |
| | | <div class="status-square" :style="{ backgroundColor: item.color }"></div> |
| | | <div>{{ getDeviceNumberByStatus(item.value) }}</div> |
| | | </a-space> |
| | | </div> |
| | | </header> |
| | | |
| | | <dv-border-box-8 class="content-container"> |
| | | <div v-for="item in deviceList" :key="item.equipmentId" class="single-device" |
| | | :style="{top:item.top+'px',left:item.left+'px',height:'135px',width:'175px'}" @mousedown="handleMouseDown" |
| | | :id="item.equipmentId"> |
| | | <div class="device-status"> |
| | | <img v-if="item.status==0" src="@/assets/yellow.png" draggable="false"> |
| | | <img v-if="item.status==1" src="@/assets/red.png" draggable="false"> |
| | | <img v-if="item.status==2" src="@/assets/gray.png" draggable="false"> |
| | | <img v-if="item.status==3" src="@/assets/green.png" draggable="false"> |
| | | <img :src="item.equipmentImageUrl" draggable="false"> |
| | | </div> |
| | | <div class="device-id">{{item.equipmentId}}</div> |
| | | <!--<div draggable="false" class="device-info">--> |
| | | <!--<div v-if="item.status==0" class="status-square" style="background-color: gray"></div>--> |
| | | <!--<div v-if="item.status==1" class="status-square" style="background-color: red"></div>--> |
| | | <!--<div v-if="item.status==2" class="status-square" style="background-color: green"></div>--> |
| | | <!--<div v-if="item.status==3" class="status-square" style="background-color: yellow"></div>--> |
| | | <!--<div class="device-id">{{item.equipmentId}}</div>--> |
| | | <!--</div>--> |
| | | <dv-border-box-8> |
| | | <div class="content-container" ref="deviceContainerRef"> |
| | | <VueDragResize |
| | | v-for="(item, index) in deviceList" |
| | | :key="item.equipmentId" |
| | | :w="item.vw" |
| | | :h="item.vh" |
| | | :x="item.coordinateLeft" |
| | | :y="item.coordinateTop" |
| | | v-on:resizing="resize($event, index)" |
| | | v-on:dragging="resize($event, index)" |
| | | :parentLimitation="true" |
| | | :minw="70" |
| | | :minh="70" |
| | | :isDraggable="isDraggable" |
| | | :isResizable="isResizable" |
| | | :stickSize="6" |
| | | > |
| | | <div class="single-device" :style="{ width: item.vw + 'px', height: item.vh + 'px' }" |
| | | @click="openDetail(item)"> |
| | | <div class="device-status"> |
| | | <div |
| | | v-if="item.equipmentStatus == 2 || item.equipmentStatus == 1" |
| | | :style="{ backgroundImage: `url(${require('@/assets/yellow.png')})` }" |
| | | class="status-image" |
| | | ></div> |
| | | <div |
| | | v-if="item.equipmentStatus == 22" |
| | | :style="{ backgroundImage: `url(${require('@/assets/red.png')})` }" |
| | | class="status-image" |
| | | ></div> |
| | | <div |
| | | v-if="item.equipmentStatus == 0" |
| | | :style="{ backgroundImage: `url(${require('@/assets/gray.png')})` }" |
| | | class="status-image" |
| | | ></div> |
| | | <div |
| | | v-if="item.equipmentStatus == 3" |
| | | :style="{ backgroundImage: `url(${require('@/assets/green.png')})` }" |
| | | class="status-image" |
| | | ></div> |
| | | <div |
| | | :style="{ backgroundImage: `url(${getImgView(item.equipmentImage) || require('@/assets/8.png')})` }" |
| | | class="device-image" |
| | | ></div> |
| | | </div> |
| | | <div class="device-id" id="deviceId" |
| | | :style="{ fontSize: item.fontSize + 'px',color:workshopDetails.equipmentIdColor }"> |
| | | {{ item.equipmentId }} |
| | | </div> |
| | | </div> |
| | | </VueDragResize> |
| | | </div> |
| | | <!--<div style="width: 175px;height: 135px;background-color: red;position: absolute;top: 200px;left: 100px"></div>--> |
| | | </dv-border-box-8> |
| | | |
| | | <EquipmentDetailModal ref="EquipmentDetailModal"></EquipmentDetailModal> |
| | | </dv-full-screen-container> |
| | | </template> |
| | | |
| | | <script> |
| | | import VueDragResize from 'vue-drag-resize' |
| | | import api from '@/api/mdc' |
| | | import { getFileAccessHttpUrl } from '@/api/manage' |
| | | import EquipmentDetailModal from './mdc/base/modules/WorkshopSignage/EquipmentDetailModal' |
| | | import { message } from 'ant-design-vue' |
| | | |
| | | message.config({ |
| | | maxCount: 3 |
| | | }) |
| | | |
| | | export default { |
| | | components: { |
| | | VueDragResize, |
| | | EquipmentDetailModal |
| | | }, |
| | | data() { |
| | | return { |
| | | mouseX: 0, // 鼠标在元素内的X坐标 |
| | | mouseY: 0, // 鼠标在元素内的Y坐标, |
| | | dragging: false, //是否在拖拽中 |
| | | left: 0, // 元素左上角距离父容器左侧的距离 |
| | | top: 0, // 元素左上角距离父容器顶部的距离, |
| | | elementId: '',// 被拖拽元素的id属性值 |
| | | elementWidth: '', |
| | | elementHeight: '', |
| | | pageHeaderHeight: 80,// 页头高度 |
| | | taskBarHeight: window.screen.height - window.screen.availHeight,// 屏幕底部任务栏高度 |
| | | utilsBarHeight: window.outerHeight - window.innerHeight,// 浏览器工具栏高度 |
| | | normalPageHeight: 0, |
| | | fullScreenPageHeight: 0, |
| | | workshopDetails: {}, // 车间详细信息, |
| | | isDraggable: false, // 是否开启拖拽 |
| | | isResizable: false, // 是否开启缩放 |
| | | isSwitchChecked: false, // 是否开启功能 |
| | | timingAcquisition: null, // 定时刷新是否开启 |
| | | deviceList: [ |
| | | // { |
| | | // equipmentId: '123213213123232',// 设备ID |
| | | // equipmentImage: require('@/assets/8.png'), // 设备图片 |
| | | // coordinateTop: 200, // 拖拽元素距盒子顶距离 |
| | | // coordinateLeft: 100, // 拖拽元素距盒子左侧距离 |
| | | // vw: 100, // 缩放元素宽度 |
| | | // vh: 100, // 缩放元素高度 |
| | | // fontSize: 12, // 缩放元素字体大小 |
| | | // equipmentStatus: 1 // 设备状态 0:关机 22:报警 2:待机 3:运行 开机:1(没有四色灯标识归为待机) |
| | | // } |
| | | ], // 设备信息列表 |
| | | deviceStatusList: [ |
| | | { |
| | | equipmentId: '123213213123232', |
| | | equipmentImageUrl: require('@/assets/8.png'), |
| | | top: 200, |
| | | left: 100, |
| | | status: 1 |
| | | label: '关机', |
| | | value: 0, |
| | | color: '#A8A8A8' |
| | | }, |
| | | { |
| | | equipmentId: '512346789561232', |
| | | equipmentImageUrl: require('@/assets/8.png'), |
| | | top: 500, |
| | | left: 753, |
| | | status: 0 |
| | | label: '待机', |
| | | value: 2, |
| | | color: '#FFFF00' |
| | | }, |
| | | { |
| | | equipmentId: '64746965647653', |
| | | equipmentImageUrl: require('@/assets/8.png'), |
| | | top: 300, |
| | | left: 860, |
| | | status: 2 |
| | | label: '运行', |
| | | value: 3, |
| | | color: '#00EE00' |
| | | }, |
| | | { |
| | | equipmentId: '33548976965462', |
| | | equipmentImageUrl: require('@/assets/8.png'), |
| | | top: 100, |
| | | left: 380, |
| | | status: 3 |
| | | }, |
| | | { |
| | | equipmentId: '85484913549253', |
| | | equipmentImageUrl: require('@/assets/8.png'), |
| | | top: 200, |
| | | left: 1500, |
| | | status: 2 |
| | | label: '报警', |
| | | value: 22, |
| | | color: '#FF0000' |
| | | } |
| | | ] |
| | | ],// 设备状态指示灯列表, |
| | | isFullscreen: false |
| | | } |
| | | }, |
| | | computed: { |
| | | getWorkshopName() { |
| | | switch (this.$route.params.workshopId) { |
| | | case '5321': |
| | | return '长沙车间' |
| | | case '6312': |
| | | return '天津车间' |
| | | case '3463': |
| | | return '武汉车间' |
| | | case '8421': |
| | | return '北京车间' |
| | | default: |
| | | return '测试车间' |
| | | } |
| | | watch: { |
| | | isSwitchChecked: { |
| | | handler(newVal) { |
| | | if (!newVal) { |
| | | console.log('定时器开启中') |
| | | this.timingAcquisition = setInterval(() => { |
| | | this.getDeviceListByApi(this.$route.params.id) |
| | | }, 2000) |
| | | } else { |
| | | console.log('关闭定时器') |
| | | clearInterval(this.timingAcquisition) |
| | | this.timingAcquisition = null |
| | | } |
| | | }, |
| | | immediate: true |
| | | } |
| | | }, |
| | | methods: { |
| | | /** |
| | | * 通过车间Id调用接口获取设备列表 |
| | | * @param workshopId 车间Id |
| | | * 通过车间Id调用接口获取设备信息列表 |
| | | * @param id 车间Id |
| | | */ |
| | | getDeviceListByApi(workshopId) { |
| | | console.log('车间Id', workshopId) |
| | | getDeviceListByApi(id) { |
| | | console.log('重新刷新') |
| | | api.getDeviceListInWorkshopSignagePageApi(id).then((res) => { |
| | | if (res.result && res.result.length > 0) { |
| | | this.deviceList = res.result |
| | | } |
| | | }) |
| | | }, |
| | | |
| | | /** |
| | | * 拖拽对象鼠标按键按下事件 |
| | | * @param event |
| | | * 通过车间Id调用接口获取车间详细信息 |
| | | * @param id 车间Id |
| | | */ |
| | | handleMouseDown(event) { |
| | | this.dragging = true |
| | | this.mouseX = event.clientX |
| | | this.mouseY = event.clientY |
| | | this.elementId = event.currentTarget.id |
| | | this.top = event.currentTarget.offsetTop |
| | | this.left = event.currentTarget.offsetLeft |
| | | document.addEventListener('mousemove', this.handleMouseMove) |
| | | document.addEventListener('mouseup', this.handleMouseUp) |
| | | this.elementWidth = +event.currentTarget.style.width.replace('px', '') |
| | | this.elementHeight = +event.currentTarget.style.height.replace('px', '') |
| | | getWorkshopDetailsByApi(id) { |
| | | api.getWorkshopDetailByWorkshopIdApi(id).then((res) => { |
| | | this.workshopDetails = res.result |
| | | this.$refs.deviceContainerRef.style.backgroundImage = `url(${this.getImgView( |
| | | this.workshopDetails.backgroundImage |
| | | )})` |
| | | |
| | | this.fullScreenPageHeight = `calc(100% - ${this.pageHeaderHeight}px)` |
| | | this.normalPageHeight = `calc(100% - ${this.pageHeaderHeight}px - ${this.taskBarHeight}px - ${this.utilsBarHeight}px)` |
| | | |
| | | if (this.isFullscreen) { |
| | | this.$refs.deviceContainerRef.style.height = this.fullScreenPageHeight |
| | | } else { |
| | | this.$refs.deviceContainerRef.style.height = this.normalPageHeight |
| | | } |
| | | }) |
| | | }, |
| | | |
| | | /** |
| | | * 文档对象鼠标移动事件 |
| | | * @param event |
| | | * 图片预览 |
| | | * @param text 图片地址 |
| | | */ |
| | | handleMouseMove(event) { |
| | | if (this.dragging) { |
| | | // x方向鼠标偏移量 |
| | | const deltaX = event.clientX - this.mouseX |
| | | // y方向鼠标偏移量 |
| | | const deltaY = event.clientY - this.mouseY |
| | | this.left += deltaX |
| | | this.top += deltaY |
| | | this.deviceList.forEach(item => { |
| | | if (item.equipmentId === this.elementId) { |
| | | item.top = this.top |
| | | item.left = this.left |
| | | // 限制拖拽超出区域(缩放浏览器窗口后距四周距离出现问题) |
| | | if (item.top < 0) { |
| | | item.top = 0 |
| | | } else if (item.top + this.elementHeight + 80 > window.innerHeight) { |
| | | item.top = window.innerHeight- this.elementHeight |
| | | getImgView(text) { |
| | | if (text && text.indexOf(',') > 0) { |
| | | text = text.substring(0, text.indexOf(',')) |
| | | } |
| | | return getFileAccessHttpUrl(text) |
| | | }, |
| | | |
| | | /** |
| | | * 点击保存按钮调用接口保存拖拽后的位置与设备图标尺寸 |
| | | */ |
| | | saveDevicePositionAndSizeByApi() { |
| | | console.log('触发保存') |
| | | if (this.isOperatingDevice) { |
| | | api.saveDevicePositionAndSizeApi(this.deviceList).then((res) => { |
| | | if (res.code === 200) { |
| | | this.$notification.success({ |
| | | message: '消息', |
| | | description: res.message |
| | | }) |
| | | this.isOperatingDevice = false |
| | | if (this.isSwitchChecked) { |
| | | this.isSwitchChecked = false |
| | | this.isResizable = !this.isResizable |
| | | this.isDraggable = !this.isDraggable |
| | | } |
| | | if (item.left < 0) { |
| | | item.left = 0 |
| | | } else if (item.left + this.elementWidth > window.innerWidth) { |
| | | item.left = window.innerWidth - this.elementWidth |
| | | } |
| | | this.getDeviceListByApi(this.$route.params.id) |
| | | } |
| | | }) |
| | | this.mouseX = event.clientX |
| | | this.mouseY = event.clientY |
| | | } else { |
| | | this.$notification.warning({ |
| | | message: '消息', |
| | | description: '请开启功能后再进行保存' |
| | | }) |
| | | } |
| | | }, |
| | | /** |
| | | * 文档对象鼠标按键弹起事件 |
| | | * @param event |
| | | */ |
| | | handleMouseUp(event) { |
| | | this.dragging = false |
| | | document.removeEventListener('mousemove', this.handleMouseMove) |
| | | document.removeEventListener('mouseup', this.handleMouseUp) |
| | | }, |
| | | /** |
| | | * 点击保存按钮调用接口保存拖拽后的位置 |
| | | */ |
| | | saveDevicePositionByApi() { |
| | | console.log('保存位置') |
| | | |
| | | /** |
| | | * 设备拖拽或缩放时触发事件 |
| | | * @param newRect 拖拽或缩放后的尺寸及距离 |
| | | * @param index 拖拽设备在数组中的下标 |
| | | */ |
| | | resize(newRect, index) { |
| | | // if (newRect.width > 100) { |
| | | // if (newRect.width / newRect.height < 2) { |
| | | // this.deviceList[index].fontSize = newRect.width / 10 |
| | | // } else { |
| | | // this.deviceList[index].fontSize = newRect.height / 5 |
| | | // } |
| | | // } else { |
| | | // this.deviceList[index].fontSize = 12 |
| | | // } |
| | | this.deviceList[index].vw = newRect.width |
| | | this.deviceList[index].vh = newRect.height |
| | | this.deviceList[index].coordinateTop = newRect.top |
| | | this.deviceList[index].coordinateLeft = newRect.left |
| | | }, |
| | | |
| | | /** |
| | | * 根据设备状态值获取对应设备数量 |
| | | * @param value 设备状态值 |
| | | * @returns {number} 设备数量 |
| | | */ |
| | | getDeviceNumberByStatus(value) { |
| | | return this.deviceList.filter((item) => item.equipmentStatus === value).length |
| | | }, |
| | | |
| | | /** |
| | | * 开启功能触发事件 |
| | | * @param checked 当前switch状态,是否开启,初始为false |
| | | */ |
| | | handleSwitchChange(checked) { |
| | | this.isOperatingDevice = true |
| | | this.isResizable = !this.isResizable |
| | | this.isDraggable = !this.isDraggable |
| | | }, |
| | | |
| | | openDetail(item) { |
| | | if (!this.isSwitchChecked) { |
| | | if (item.equipmentStatus == 0) { |
| | | // this.$message.warn("设备处于关机状态!"); |
| | | this.$message.warning('设备处于关机状态!') |
| | | return false |
| | | } |
| | | console.log(item) |
| | | console.log(this.$refs.EquipmentDetailModal) |
| | | this.$refs.EquipmentDetailModal.initData(item.equId) |
| | | this.$refs.EquipmentDetailModal.timerModel(item.equId) |
| | | // this.equipMessageTimer = setInterval(() => { |
| | | // setTimeout( this.$refs.equmentDetaiModel.initData(item.equipmentId),0) |
| | | // },1000*10) |
| | | } |
| | | }, |
| | | |
| | | onResize() { |
| | | // 利用屏幕分辨率和window对象的内高度来判断兼容IE |
| | | let winFlag = window.innerHeight === window.screen.height |
| | | // 利用window全屏标识来判断 -- IE无效 |
| | | let isFull = window.fullScreen || document.webkitIsFullScreen |
| | | |
| | | if (isFull === undefined) { |
| | | this.isFullscreen = winFlag |
| | | } else { |
| | | this.isFullscreen = winFlag || isFull |
| | | } |
| | | console.log(winFlag) // true全屏 false不是全屏 |
| | | console.log('任务栏高度', this.taskBarHeight) |
| | | console.log('工具栏高度', this.utilsBarHeight) |
| | | |
| | | //全屏则图纸盒子可视高度减去页头高度,正常窗口则减去页头、屏幕底部任务栏以及浏览器工具栏高度 |
| | | if (winFlag) { |
| | | console.log('触发全屏') |
| | | this.$refs.deviceContainerRef.style.height = this.fullScreenPageHeight |
| | | |
| | | } else { |
| | | console.log('关闭全屏') |
| | | // 如果刷新页面重新加载后默认处于全屏模式,则工具栏高度为负数,因此需要退出全屏模式时重置工具栏高度 |
| | | this.utilsBarHeight = window.outerHeight - window.innerHeight |
| | | this.normalPageHeight = `calc(100% - ${this.pageHeaderHeight}px - ${this.taskBarHeight}px - ${this.utilsBarHeight}px)` |
| | | this.$refs.deviceContainerRef.style.height = this.normalPageHeight |
| | | } |
| | | } |
| | | |
| | | }, |
| | | created() { |
| | | if (this.$route.params.id) { |
| | | this.getDeviceListByApi(this.$route.params.id) |
| | | this.getWorkshopDetailsByApi(this.$route.params.id) |
| | | } |
| | | |
| | | |
| | | let winFlag = window.innerHeight === window.screen.height |
| | | // 利用window全屏标识来判断 -- IE无效 |
| | | let isFull = window.fullScreen || document.webkitIsFullScreen |
| | | |
| | | if (isFull === undefined) { |
| | | this.isFullscreen = winFlag |
| | | } else { |
| | | this.isFullscreen = winFlag || isFull |
| | | } |
| | | |
| | | // window.addEventListener('resize', this.onResize) |
| | | }, |
| | | mounted() { |
| | | // 禁止用户选中内容 |
| | | document.onselectstart = () => false |
| | | }, |
| | | created() { |
| | | if (this.$route.params.workshopId) { |
| | | this.getDeviceListByApi(this.$route.params.workshopId) |
| | | } |
| | | beforeDestroy() { |
| | | // 确保销毁定时器、事件及回收资源 |
| | | clearInterval(this.timingAcquisition) |
| | | this.timingAcquisition = null |
| | | // window.removeEventListener('resize', this.onResize) |
| | | } |
| | | } |
| | | </script> |
| | |
| | | color: #fff; |
| | | |
| | | .page-header { |
| | | height: 80px; |
| | | font-size: 50px; |
| | | text-align: center; |
| | | position: relative; |
| | | .header-right { |
| | | |
| | | .header-left { |
| | | width: 450px; |
| | | position: absolute; |
| | | right: 200px; |
| | | top: 0; |
| | | left: 0px; |
| | | top: 35px; |
| | | display: flex; |
| | | justify-content: space-evenly; |
| | | align-items: center; |
| | | font-size: 16px; |
| | | } |
| | | |
| | | .device-status-info { |
| | | font-size: 16px; |
| | | width: 400px; |
| | | position: absolute; |
| | | top: 40px; |
| | | right: 40px; |
| | | display: flex; |
| | | align-items: center; |
| | | justify-content: space-between; |
| | | |
| | | .single-status-info { |
| | | /*width: 70px;*/ |
| | | /*display: flex;*/ |
| | | /*align-items: center;*/ |
| | | /*justify-content: space-between;*/ |
| | | |
| | | .status-square { |
| | | width: 14px; |
| | | height: 14px; |
| | | border: 1px solid #fff; |
| | | border-radius: 2px; |
| | | } |
| | | } |
| | | } |
| | | } |
| | | |
| | | .content-container { |
| | | position: relative; |
| | | width: 100%; |
| | | /*height: calc(100% - 200px);*/ |
| | | /*height: 100%;*/ |
| | | background-repeat: no-repeat; |
| | | background-size: 100% 100%; |
| | | |
| | | .single-device { |
| | | position: absolute; |
| | | border: 1px solid transparent; |
| | |
| | | align-items: center; |
| | | justify-content: space-between; |
| | | cursor: default; |
| | | |
| | | &:active { |
| | | border: 1px solid #00b3ff; |
| | | border: 1px solid #1890ff; |
| | | } |
| | | |
| | | .device-status { |
| | | width: 145px; |
| | | width: 100%; |
| | | height: 100%; |
| | | display: flex; |
| | | justify-content: space-between; |
| | | align-items: end; |
| | | |
| | | .status-image { |
| | | background-size: 100% 100%; |
| | | background-repeat: no-repeat; |
| | | width: 10px; |
| | | height: 60%; |
| | | margin-right: 5px; |
| | | } |
| | | |
| | | .device-image { |
| | | background-size: 100% 100%; |
| | | background-repeat: no-repeat; |
| | | width: 100%; |
| | | height: 100%; |
| | | } |
| | | } |
| | | .device-id { |
| | | font-size: 20px; |
| | | } |
| | | /*.device-info {*/ |
| | | /*width: 100%;*/ |
| | | /*display: flex;*/ |
| | | /*align-items: center;*/ |
| | | /*justify-content: space-between;*/ |
| | | /*.status-square {*/ |
| | | /*width: 14px;*/ |
| | | /*height: 14px;*/ |
| | | /*border: 1px solid #fff;*/ |
| | | /*border-radius: 2px;*/ |
| | | /*}*/ |
| | | /*}*/ |
| | | } |
| | | } |
| | | } |