类型:开发
描述:
This commit is contained in:
@@ -2,27 +2,24 @@
|
|||||||
<div class="box-2">
|
<div class="box-2">
|
||||||
<div class="left-nav">
|
<div class="left-nav">
|
||||||
<div v-if="showNav" >
|
<div v-if="showNav" >
|
||||||
<div class="ul">
|
<div class="ul">
|
||||||
<div
|
<div
|
||||||
class="li"
|
class="li"
|
||||||
:class="{ active: current == index }"
|
:class="{ active: current == index }"
|
||||||
v-for="(item, index) in navList"
|
v-for="(item, index) in navList"
|
||||||
:key="index"
|
:key="index"
|
||||||
@click="handleNav(index)"
|
@click="handleNav(index)"
|
||||||
>
|
>
|
||||||
<!-- <span v-if="!params.businessScenicArea && index == 0"> 核心路段 </span> -->
|
<span>
|
||||||
<span>
|
{{ item }}
|
||||||
{{ item.dictLabel }}
|
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
:class="{ active: current == navList.length }"
|
:class="{ active: current == navList.length }"
|
||||||
@click="handleNav(navList.length)" class="li">
|
@click="handleNav(navList.length)" class="li">
|
||||||
<span>收藏夹</span>
|
<span>收藏夹</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
<div class="bom-box">
|
<div class="bom-box">
|
||||||
<Title2 title="检索" />
|
<Title2 title="检索" />
|
||||||
@@ -52,7 +49,7 @@
|
|||||||
alt=""
|
alt=""
|
||||||
/>
|
/>
|
||||||
{{resource.regions}}
|
{{resource.regions}}
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
<div v-if="resource.show">
|
<div v-if="resource.show">
|
||||||
<div
|
<div
|
||||||
@@ -64,9 +61,9 @@
|
|||||||
{{ res.cameraName || res.cameraIndexCode }}
|
{{ res.cameraName || res.cameraIndexCode }}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -202,18 +199,19 @@
|
|||||||
postVideoControlApi,
|
postVideoControlApi,
|
||||||
postVideoCollectApi
|
postVideoCollectApi
|
||||||
} from '@/api/monitor'
|
} from '@/api/monitor'
|
||||||
|
import mpegtsjs from 'mpegts.js'
|
||||||
|
|
||||||
import { debounce } from 'lodash'
|
import { debounce } from 'lodash'
|
||||||
import pubSub from 'pubsub-js'
|
import pubSub from 'pubsub-js'
|
||||||
import Hls from 'hls.js'
|
import Hls from 'hls.js'
|
||||||
|
|
||||||
const Z00M_IN = 'ZOOM_IN' // 焦距变大
|
const Z00M_IN = 'zoomin' // 焦距变大
|
||||||
const Z00M_OUT = 'ZOOM_OUT' // 焦距变小
|
const Z00M_OUT = 'zoomout' // 焦距变小
|
||||||
const UP = 'UP' // 上转
|
const UP = 'up' // 上转
|
||||||
const DOWN = 'DOWN' // 下转
|
const DOWN = 'down' // 下转
|
||||||
const LEFT = 'LEFT' // 左转
|
const LEFT = 'left' // 左转
|
||||||
const RIGHT = 'RIGHT' // 右转
|
const RIGHT = 'right' // 右转
|
||||||
const STOP = 'STOP' // 停止操作
|
const STOP = 'stop' // 停止操作
|
||||||
let ACTION = '0'
|
let ACTION = '0'
|
||||||
|
|
||||||
const props = {
|
const props = {
|
||||||
@@ -249,41 +247,97 @@
|
|||||||
let timer = null
|
let timer = null
|
||||||
|
|
||||||
const postVideoRemain = async () => {
|
const postVideoRemain = async () => {
|
||||||
if(!videoList.value.length) return false;
|
// if(!videoList.value.length) return false;
|
||||||
timer = setInterval(() => {
|
// timer = setInterval(() => {
|
||||||
postVideoRemainApi({
|
// postVideoRemainApi({
|
||||||
cameraIndexCode: videoList.value.map((item) => item.cameraIndexCode)
|
// cameraIndexCode: videoList.value.map((item) => item.cameraIndexCode)
|
||||||
})
|
// })
|
||||||
}, 2000)
|
// }, 2000)
|
||||||
}
|
}
|
||||||
|
const createPlayer = (cameraIndexCode,videoElement) => {
|
||||||
|
getPreviewUrlApi({
|
||||||
|
type: 'hls',
|
||||||
|
cameraIndexCode: cameraIndexCode
|
||||||
|
}).then(res=>{
|
||||||
|
const url = res.data.url;
|
||||||
|
if(url.startsWith('ws')){
|
||||||
|
const player = mpegtsjs.createPlayer({
|
||||||
|
url: url,
|
||||||
|
type: 'flv',
|
||||||
|
isLive: true,
|
||||||
|
hasAudio: false
|
||||||
|
})
|
||||||
|
player.attachMediaElement(videoElement)
|
||||||
|
player.load()
|
||||||
|
player.play()
|
||||||
|
// 错误处理和重连机制
|
||||||
|
player.on(mpegtsjs.Events.ERROR, (err) => {
|
||||||
|
console.error('播放器错误【'+cameraIndexCode+'】:', err)
|
||||||
|
player.unload();
|
||||||
|
player.destroy();
|
||||||
|
// 3 秒后尝试重新加载
|
||||||
|
setTimeout(() => {
|
||||||
|
console.error('重新加载【'+cameraIndexCode+'】' )
|
||||||
|
createPlayer(cameraIndexCode,videoElement);
|
||||||
|
}, 3000)
|
||||||
|
|
||||||
const initVideo = () => {
|
})
|
||||||
clearHlsRefs()
|
player.on(mpegtsjs.Events.LOADING_COMPLETE, (err) => {
|
||||||
nextTick(() => {
|
console.error('加载完成:', err)
|
||||||
videoList.value.forEach(async (item, index) => {
|
})
|
||||||
const video = document.getElementById(`monitorVideo${index}`)
|
hlsRefs.push(player)
|
||||||
const hls = new Hls({
|
}
|
||||||
|
else{
|
||||||
|
const player = new Hls({
|
||||||
maxBufferLength: 10, // 最大缓冲长度(秒)
|
maxBufferLength: 10, // 最大缓冲长度(秒)
|
||||||
maxMaxBufferLength: 15, // 缓冲区长度的上限
|
maxMaxBufferLength: 15, // 缓冲区长度的上限
|
||||||
maxBufferSize: 30 * 1000 * 1000 // 最大缓冲大小(字节)
|
maxBufferSize: 30 * 1000 * 1000 // 最大缓冲大小(字节)
|
||||||
})
|
})
|
||||||
hls.loadSource(item.hlsUrl)
|
player.loadSource(url)
|
||||||
hls.attachMedia(video)
|
player.attachMedia(videoElement)
|
||||||
hls.on(Hls.Events.MANIFEST_PARSED, () => {
|
player.on(Hls.Events.MANIFEST_PARSED, () => {
|
||||||
video.play()
|
videoElement.play()
|
||||||
})
|
})
|
||||||
hlsRefs.push(hls)
|
player.on(Hls.Events.ERROR, (event, data) => {
|
||||||
|
// 根据错误类型进行处理
|
||||||
|
if (data.fatal) {
|
||||||
|
switch (data.type) {
|
||||||
|
case Hls.ErrorTypes.NETWORK_ERROR:
|
||||||
|
console.log('网络错误,尝试重新加载');
|
||||||
|
player.startLoad();
|
||||||
|
break;
|
||||||
|
case Hls.ErrorTypes.MEDIA_ERROR:
|
||||||
|
console.log('媒体错误,尝试修复');
|
||||||
|
player.recoverMediaError();
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
console.log('无法恢复的错误,销毁播放器');
|
||||||
|
// hls.destroy();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
hlsRefs.push(player)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
const initVideo = () => {
|
||||||
|
clearHlsRefs()
|
||||||
|
nextTick(() => {
|
||||||
|
videoList.value.forEach(async (item, index) => {
|
||||||
|
const videoElement = document.getElementById(`monitorVideo${index}`)
|
||||||
|
createPlayer(item.cameraIndexCode,videoElement);
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
const onInput = debounce((e) => {
|
const onInput = debounce((e) => {
|
||||||
|
|
||||||
if(e){
|
if(e){
|
||||||
getVideoRegions('search')
|
getVideoRegions('search')
|
||||||
}else{
|
}else{
|
||||||
getVideoRegions()
|
getVideoRegions()
|
||||||
}
|
}
|
||||||
|
|
||||||
}, 500)
|
}, 500)
|
||||||
const clearHlsRefs = () => {
|
const clearHlsRefs = () => {
|
||||||
if (hlsRefs.length > 0) {
|
if (hlsRefs.length > 0) {
|
||||||
@@ -303,7 +357,7 @@
|
|||||||
videoList.value = []
|
videoList.value = []
|
||||||
getVideoList()
|
getVideoList()
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
const getColletList = async () => {
|
const getColletList = async () => {
|
||||||
clearHlsRefs()
|
clearHlsRefs()
|
||||||
@@ -338,30 +392,60 @@
|
|||||||
initVideo()
|
initVideo()
|
||||||
}
|
}
|
||||||
let thisVideo = ref(null)
|
let thisVideo = ref(null)
|
||||||
const handleItemVideo = (url, type, code,item) => {
|
const handleItemVideo = async (url, type, code, item) => {
|
||||||
thisVideo.value = item
|
let res = await getPreviewUrlApi({
|
||||||
|
cameraIndexCode: code,
|
||||||
|
type: 'hls'
|
||||||
|
})
|
||||||
|
url = res.data.url
|
||||||
|
thisVideo.value = item
|
||||||
videoLog.value = 2
|
videoLog.value = 2
|
||||||
cameraIndexCode.value = code
|
cameraIndexCode.value = code
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
hlsRef = new Hls({
|
if (url.startsWith('ws')) {
|
||||||
maxBufferLength: 10, // 最大缓冲长度(秒)
|
hlsRef = mpegtsjs.createPlayer({
|
||||||
maxMaxBufferLength: 15, // 缓冲区长度的上限
|
url: url,
|
||||||
maxBufferSize: 30 * 1000 * 1000 // 最大缓冲大小(字节)
|
type: 'flv',
|
||||||
})
|
isLive: true,
|
||||||
hlsRef.loadSource(url)
|
hasAudio: false
|
||||||
hlsRef.attachMedia(videoRef.value)
|
})
|
||||||
hlsRef.on(Hls.Events.MANIFEST_PARSED, () => {
|
hlsRef.attachMediaElement(videoRef.value)
|
||||||
videoRef.value.play()
|
hlsRef.load()
|
||||||
})
|
hlsRef.play()
|
||||||
|
// 错误处理和重连机制
|
||||||
|
hlsRef.on('error', (err) => {
|
||||||
|
console.error('播放器错误:', err)
|
||||||
|
// 3 秒后尝试重新加载
|
||||||
|
hlsRef.unload()
|
||||||
|
hlsRef.destroy()
|
||||||
|
// 3 秒后尝试重新加载
|
||||||
|
setTimeout(() => {
|
||||||
|
console.error('重新加载【' + cameraIndexCode + '】')
|
||||||
|
handleItemVideo(url, 101, code, item);
|
||||||
|
}, 3000)
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
hlsRef = new Hls({
|
||||||
|
maxBufferLength: 10, // 最大缓冲长度(秒)
|
||||||
|
maxMaxBufferLength: 15, // 缓冲区长度的上限
|
||||||
|
maxBufferSize: 30 * 1000 * 1000 // 最大缓冲大小(字节)
|
||||||
|
})
|
||||||
|
hlsRef.loadSource(url)
|
||||||
|
hlsRef.attachMedia(videoRef.value)
|
||||||
|
hlsRef.on(Hls.Events.MANIFEST_PARSED, () => {
|
||||||
|
videoRef.value.play()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
if (type == 100) initVideo()
|
if (type == 100) initVideo()
|
||||||
}, 1000)
|
}, 1000)
|
||||||
}
|
}
|
||||||
const handleAction = async (e) => {
|
const handleAction = async (e) => {
|
||||||
|
command.value = e
|
||||||
if (e == STOP) {
|
if (e == STOP) {
|
||||||
ACTION = '1'
|
ACTION = '1'
|
||||||
} else {
|
} else {
|
||||||
ACTION = '0'
|
ACTION = '0'
|
||||||
command.value = e
|
|
||||||
}
|
}
|
||||||
await postVideoControlApi({
|
await postVideoControlApi({
|
||||||
action: ACTION,
|
action: ACTION,
|
||||||
@@ -381,18 +465,17 @@
|
|||||||
}
|
}
|
||||||
const handleRegList = (item,i,x)=>{
|
const handleRegList = (item,i,x)=>{
|
||||||
regionList.value[i].resourcesList[x].show = !regionList.value[i].resourcesList[x].show
|
regionList.value[i].resourcesList[x].show = !regionList.value[i].resourcesList[x].show
|
||||||
|
|
||||||
}
|
}
|
||||||
const handleCamera = async (itemCode,resource) => {
|
const handleCamera = async (itemCode,resource) => {
|
||||||
show.value = true
|
show.value = true
|
||||||
let res = await getPreviewUrlApi({
|
let res = await getPreviewUrlApi({
|
||||||
type: 'hls',
|
type: 'hls',
|
||||||
cameraIndexCode:itemCode
|
cameraIndexCode: itemCode
|
||||||
})
|
})
|
||||||
console.log(res,'rrrrrrrrrrrrrr')
|
|
||||||
cameraIndexCode.value = itemCode;
|
cameraIndexCode.value = itemCode;
|
||||||
isCollect.value = resource.isCollect
|
isCollect.value = resource.isCollect
|
||||||
isDiy.value = resource.isDiy
|
isDiy.value = resource.isDiy
|
||||||
videoSrc.value = res.data.url
|
videoSrc.value = res.data.url
|
||||||
}
|
}
|
||||||
const handleCollect = async (id, status, index) => {
|
const handleCollect = async (id, status, index) => {
|
||||||
@@ -451,7 +534,7 @@
|
|||||||
let res = await getVideoListApi(params)
|
let res = await getVideoListApi(params)
|
||||||
total.value = res.total
|
total.value = res.total
|
||||||
if (res.data.length > 0) {
|
if (res.data.length > 0) {
|
||||||
|
|
||||||
videoList.value = res.data
|
videoList.value = res.data
|
||||||
// console.log(videoList.value,'1111111111111111111111')
|
// console.log(videoList.value,'1111111111111111111111')
|
||||||
// postVideoRemain()
|
// postVideoRemain()
|
||||||
@@ -485,13 +568,13 @@
|
|||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
console.log('无法恢复的错误,销毁播放器');
|
console.log('无法恢复的错误,销毁播放器');
|
||||||
|
|
||||||
// hls.destroy();
|
// hls.destroy();
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
hlsRefs.push(hls)
|
hlsRefs.push(hls)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
@@ -520,7 +603,7 @@
|
|||||||
cameraName: cameraName.value,
|
cameraName: cameraName.value,
|
||||||
businessScenicArea: params.businessScenicArea
|
businessScenicArea: params.businessScenicArea
|
||||||
})
|
})
|
||||||
|
|
||||||
regionList.value = res.data
|
regionList.value = res.data
|
||||||
regionList.value.videoResources = res.data.videoResources
|
regionList.value.videoResources = res.data.videoResources
|
||||||
// regionList.value[0].show = true
|
// regionList.value[0].show = true
|
||||||
@@ -532,7 +615,7 @@
|
|||||||
r.show = false
|
r.show = false
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
if(search=='search'){
|
if(search=='search'){
|
||||||
regionList.value.forEach((item,index)=>{
|
regionList.value.forEach((item,index)=>{
|
||||||
item.show = true
|
item.show = true
|
||||||
@@ -541,7 +624,7 @@
|
|||||||
})
|
})
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
console.log(regionList.value, ' regionList.value')
|
console.log(regionList.value, ' regionList.value')
|
||||||
}
|
}
|
||||||
const onMonitorChange = () => {
|
const onMonitorChange = () => {
|
||||||
@@ -557,8 +640,8 @@
|
|||||||
getVideoType()
|
getVideoType()
|
||||||
getVideoRegions()
|
getVideoRegions()
|
||||||
},1500)
|
},1500)
|
||||||
|
|
||||||
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
@@ -588,7 +671,7 @@
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
|
|
||||||
.new-title{
|
.new-title{
|
||||||
margin-bottom:vh(20);
|
margin-bottom:vh(20);
|
||||||
color:#fff;
|
color:#fff;
|
||||||
@@ -671,7 +754,7 @@
|
|||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
|
|
||||||
.search-icon {
|
.search-icon {
|
||||||
width: vw(20);
|
width: vw(20);
|
||||||
height: vw(20);
|
height: vw(20);
|
||||||
@@ -705,7 +788,7 @@
|
|||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
position: relative;
|
position: relative;
|
||||||
padding-top: vh(30);
|
padding-top: vh(30);
|
||||||
|
|
||||||
// border-left: vw(2) solid #37d8fc;
|
// border-left: vw(2) solid #37d8fc;
|
||||||
&:nth-child(1) {
|
&:nth-child(1) {
|
||||||
padding-top: 0;
|
padding-top: 0;
|
||||||
@@ -783,7 +866,7 @@
|
|||||||
overflow-y: auto;
|
overflow-y: auto;
|
||||||
overflow-x: hidden;
|
overflow-x: hidden;
|
||||||
/* 滚动条整体样式 */
|
/* 滚动条整体样式 */
|
||||||
|
|
||||||
&::-webkit-scrollbar {
|
&::-webkit-scrollbar {
|
||||||
width: vw(4); /* 滚动条的宽度 */
|
width: vw(4); /* 滚动条的宽度 */
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user