BigShow.vue 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315
  1. <!--大屏显示组件-->
  2. <template>
  3. <div>
  4. <draggable
  5. v-for="item in currentBigshow"
  6. element="div"
  7. v-model="bigScreenObj[item.ID].signalArr"
  8. :options="dragOptions2"
  9. @change="log"
  10. :class="['split_screen', bigScreenObj[item.ID].splitStatus === 0 ? 'split_screen_one' : bigScreenObj[item.ID].splitStatus === 1 ? 'split_screen_four' : bigScreenObj[item.ID].splitStatus === 2 ? 'split_screen_nine' : 'split_screen_sixteen']"
  11. :style="{
  12. float:'left',
  13. width:item.Width*bigScale + 'px',
  14. height:item.Height*bigScale + 'px',
  15. left:item.Left*bigScale+'px',
  16. top:item.Top*bigScale+'px',
  17. zIndex:item.ZIndex,
  18. display:item.IsVisibility ? 'black' : 'none',
  19. }"
  20. :id="item.ID"
  21. >
  22. <VueDragResize
  23. v-for="itemV in bigScreenObj[item.ID].signalArr" :key="itemV.Id" class="sitem"
  24. :w="item.Width*bigScale/(bigScreenObj[item.ID].splitStatus+1)" :h="item.Height*bigScale/(bigScreenObj[item.ID].splitStatus+1)" :parentLimitation="false"
  25. :x="itemV.left" :y="itemV.top"
  26. @dragstop="(obj) => dragStop(obj,itemV)"
  27. @resizestop="(obj) => reSizeSignal(obj,itemV)"
  28. >
  29. <div ref="signal">
  30. {{itemV.deviceID}}
  31. <!--<VideoPlayer
  32. ref="videoPlayer"
  33. :options="playerOptions"
  34. :playsinline="true"
  35. />-->
  36. <div class="closeBtn" @click.stop="closeSignal(item,itemV)">×</div>
  37. </div>
  38. </VueDragResize>
  39. <div v-for="itemD in bigScreenObj[item.ID].bindList" :key="itemD" class="sitem2">
  40. {{itemD}}
  41. </div>
  42. </draggable>
  43. </div>
  44. </template>
  45. <script>
  46. import {mapState} from 'vuex'
  47. import draggable from 'vuedraggable'
  48. import VueDragResize from 'vue-drag-resize'
  49. import '../assets/less/splitscreen.less'
  50. import {elePosition, getElementLeft, getElementTop, getStaticFile, guId} from "../../utils/tools"
  51. import {reqRefreshView} from "../api"
  52. import VideoPlayer from "./player.vue"
  53. export default {
  54. data() {
  55. return {
  56. streamWindows:[], // 传递出去的信号源位置信息
  57. currentBigshow:[], // 当前显示的大屏
  58. playerOptions: { // rtmp视频流配置
  59. sources: [
  60. {
  61. "src": "rtmp://58.200.131.2:1935/livetv/hunantv",
  62. "type": "rtmp/flv"
  63. }
  64. ],
  65. techOrder: ['flash'],
  66. autoplay: true,
  67. controls: false,
  68. fluid:true,
  69. aspectRatio: '200:93',
  70. preload: 'auto',
  71. notSupportedMessage: '此视频暂无法播放,请稍后再试',
  72. },
  73. screenPosition:{}, // 大屏在视口中的位置
  74. newIndex: 1, // 开窗拖到第几个格子中
  75. }
  76. },
  77. components: {
  78. draggable,
  79. VueDragResize,
  80. VideoPlayer
  81. },
  82. async beforeCreate() {
  83. const bigScreenJson = await getStaticFile('EnityBigScreen.Data')
  84. this.currentBigshow = bigScreenJson.filter(item => item.IsVisibility === true)
  85. // 给大屏对象进行初始化显示
  86. const obj = {}
  87. this.currentBigshow.forEach(item => {
  88. obj[item.ID] = {
  89. bindList:[],
  90. splitStatus:0,
  91. signalArr:[]
  92. }
  93. const num = item.BindList.length
  94. // 拼接屏数量
  95. for(let i=0;i<num;i++){
  96. obj[item.ID].bindList.push(i+1)
  97. }
  98. // 根据拼接屏数量得到分屏状态
  99. obj[item.ID].splitStatus = Math.floor(Math.sqrt(num)-1)
  100. })
  101. this.$store.dispatch('updateBigscreenObj',obj)
  102. /*const num = this.currentBigshow[0].BindList.length
  103. for(let i=0;i<num;i++){
  104. this.$data.bindList.push(i+1)
  105. }
  106. this.$store.dispatch('splitScreen', Math.sqrt(num)-1)*/
  107. },
  108. updated() {
  109. this.currentBigshow.forEach(item => {
  110. // 获取大屏在视口中的位置
  111. const screenDiv = document.getElementById(item.ID)
  112. this.screenPosition[item.ID] = {
  113. top:getElementTop(screenDiv),
  114. left:getElementLeft(screenDiv)
  115. }
  116. })
  117. /*const bigscreenObj = this.$store.state.bigScreenObj
  118. for(const key in bigscreenObj){
  119. const signalArr = bigscreenObj[key].signalArr
  120. let len = signalArr.length
  121. if(len){
  122. // 给新增的信号源进行赋值
  123. const signalObj = signalArr[len-1]
  124. signalObj.Id = guId()
  125. }
  126. }*/
  127. },
  128. methods: {
  129. // 改变信号源
  130. log() {
  131. this.newIndex = this.$store.state.positionNum
  132. },
  133. // 大屏开窗接口提交
  134. reqRefreshBigScreen(arr) {
  135. const streamWindows = []
  136. setTimeout(async () => {
  137. const elementArr = this.$refs.signal ? this.$refs.signal.filter(item => item.clientWidth !== 0) : []
  138. if(elementArr){
  139. elementArr.forEach((item,index) => {
  140. // 获取元素绝对位置的横坐标和纵坐标
  141. const left = getElementLeft(item)
  142. const top = getElementTop(item)
  143. streamWindows.push({
  144. left,
  145. top,
  146. width:arr[index].width,
  147. height:arr[index].height,
  148. id:0,
  149. sourceId:arr[index].sourceId,
  150. widthScale:arr[index].width/(this.currentBigshow[0].Width*this.$store.state.bigScale),
  151. heightScale:arr[index].height/(this.currentBigshow[0].Height*this.$store.state.bigScale),
  152. orginRect:{}
  153. })
  154. })
  155. const data = {
  156. bigScreenId:this.currentBigshow[0].ID,
  157. streamWindows
  158. }
  159. this.streamWindows = streamWindows
  160. // 调用大屏开窗接口
  161. await reqRefreshView(data)
  162. }
  163. })
  164. },
  165. // 拖动结束
  166. dragStop(obj,item) {
  167. /*const arr = this.$store.state.signalPreList.filter(a => a.Id !== item.Id)
  168. item.top = this.screenPosition.top + obj.top
  169. item.left = this.screenPosition.left + obj.left
  170. this.$store.dispatch('updateSignalPreList',[...arr,item])*/
  171. const arr2 = this.$store.state.signalPreList
  172. this.reqRefreshBigScreen(arr2)
  173. },
  174. // 放大缩小信号源(缩放结束事件)
  175. async reSizeSignal(obj,itemV) {
  176. const arr = this.$data.streamWindows
  177. // 找出当前改动的信号源
  178. const currentItem = arr.filter(item => item.sourceId === itemV.sourceId)
  179. currentItem[0].width = obj.width
  180. currentItem[0].height = obj.height
  181. currentItem[0].widthScale = obj.width/(this.currentBigshow[0].Width*this.$store.state.bigScale)
  182. currentItem[0].heightScale = obj.height/(this.currentBigshow[0].Height*this.$store.state.bigScale)
  183. const otherItems = arr.filter(item => item.sourceId !== itemV.sourceId)
  184. const newStreamWindows = [...currentItem,...otherItems]
  185. this.streamWindows = newStreamWindows
  186. this.reqRefreshBigScreen(newStreamWindows)
  187. },
  188. // 关闭信号源
  189. closeSignal(bigscreen,signal) {
  190. const bigScreenObj = this.$store.state.bigScreenObj
  191. // 1.找出是哪个大屏下的信号源发生了改变,过滤掉删除的那个
  192. const newSignalPreList = bigScreenObj[bigscreen.ID].signalArr.filter(item => item.Id !== signal.Id)
  193. // 2.重新赋值
  194. bigScreenObj[bigscreen.ID].signalArr = newSignalPreList
  195. // 3.更新
  196. this.$store.dispatch('updateBigscreenObj',bigScreenObj)
  197. }
  198. },
  199. computed: {
  200. ...mapState(['splitScreenStatus', 'bigScale','positionNum','bigScreenObj']),
  201. signalPreList: {
  202. get() {
  203. return this.$store.state.signalPreList
  204. },
  205. set(arr) {
  206. let len = arr.length
  207. if(len>0){
  208. // 对监视的信号源数组进行深度克隆,解决赋加Id一样的问题
  209. const arr2 = []
  210. arr.forEach(item => {arr2.push(JSON.parse(JSON.stringify(item)))})
  211. // 删除数据的最后一项,重新赋值再添加
  212. arr2.pop()
  213. // 给新增的信号源进行赋值
  214. const signalObj = arr[len-1]
  215. signalObj.Id = guId()
  216. signalObj.splitScreenStatus = this.$store.state.splitScreenStatus + 1
  217. signalObj.width = this.currentBigshow[0].Width * this.$store.state.bigScale / (this.$store.state.splitScreenStatus+1)
  218. signalObj.height = this.currentBigshow[0].Height * this.$store.state.bigScale / (this.$store.state.splitScreenStatus+1)
  219. signalObj.widthScale = signalObj.width / (this.currentBigshow[0].Width * this.$store.state.bigScale)
  220. signalObj.heightScale = signalObj.height / (this.currentBigshow[0].Height * this.$store.state.bigScale)
  221. signalObj.isFixed = true // 是否使用固定定位
  222. this.$store.dispatch('updateSignalPreList', [...arr2,signalObj])
  223. // 添加定时器是为了得到最新的newIndex
  224. setTimeout(() => {
  225. let width = this.currentBigshow[0].Width*this.$store.state.bigScale/(signalObj.splitScreenStatus)
  226. let height = this.currentBigshow[0].Height*this.$store.state.bigScale/(signalObj.splitScreenStatus)
  227. const res = elePosition(width,height,this.newIndex,this.screenPosition.top,this.screenPosition.left,this.$store.state.splitScreenStatus+1)
  228. signalObj.top = res.top - this.screenPosition.top
  229. signalObj.left = res.left - this.screenPosition.left
  230. this.$store.dispatch('updateSignalPreList', [...arr2,signalObj])
  231. })
  232. }else {
  233. this.$store.dispatch('updateSignalPreList', arr)
  234. }
  235. }
  236. },
  237. dragOptions1() {
  238. return {
  239. animation: 0,
  240. group: {
  241. name: "description",
  242. pull: 'clone',
  243. put: false
  244. },
  245. ghostClass: "ghost",
  246. }
  247. },
  248. dragOptions2() {
  249. return {
  250. animation: 0,
  251. group: "description",
  252. disabled:false,
  253. sort:false,
  254. }
  255. },
  256. player() {
  257. return this.$refs.videoPlayer.player
  258. }
  259. },
  260. watch: {
  261. // 监视大屏对象的变化(深度监视)
  262. bigScreenObj: {
  263. handler: function (val, oldVal) {
  264. for (const key in val) {
  265. val[key].signalArr.forEach(item => {
  266. item.Id = guId()
  267. })
  268. }
  269. console.log(val)
  270. this.$store.dispatch('updateBigscreenObj',val)
  271. },
  272. deep: true
  273. },
  274. // 监视大屏中的信号源,只要变动,就调用回调函数
  275. signalPreList: function (arr) {
  276. this.reqRefreshBigScreen(arr)
  277. },
  278. // 监视分屏的状态
  279. splitScreenStatus: function(num) {
  280. this.$data.bindList = []
  281. for (let i=0;i<(num+1)*(num+1);i++){
  282. this.$data.bindList.push(i+1)
  283. }
  284. this.reqRefreshBigScreen(this.$data.streamWindows)
  285. }
  286. }
  287. }
  288. </script>