go-viee-fetch-Demo/src/packages/components/Charts/ConfinedSpace/yushiVideo/index.vue

721 lines
25 KiB
Vue
Raw Normal View History

2025-08-28 10:36:02 +08:00
<!-- eslint-disable vue/multi-word-component-names -->
<template>
<div>
<div :class="option.isOldStyle ? 'video_title' : 'video_title video_title_new'" v-if="option.showBtn">
<div class="title_text" v-if="option.showTree && option.showBtn">{{ option.videoTitle }}</div>
<n-select v-if="!option.showTree" class="video_select" placement="top-end"
v-model:value="option.selectedDataSource" :options="option.dataSource" :style="`width:${w / 2}px;`"
@update:value="handleSelectDataSource" />
<n-button tertiary v-else-if="option.showTree && option.showBtn && option.isOldStyle"
:class="option.isOldStyle ? 'videoChangBtn' : 'videoChangBtn videoChangBtn_new'"
@click="handleClick">切换</n-button>
<n-button tertiary v-else-if="option.showTree && option.showBtn && !option.isOldStyle"
:class="option.isOldStyle ? 'videoChangBtn' : 'videoChangBtn videoChangBtn_new'"
@click="handleClick"></n-button>
<n-modal v-model:show="showDialog" :mask-closable="false" preset="card" title="选择摄像头"
:class="['custom-tab-modal']" :draggable="{ bounds: 'none' }" :style="{ width: '644px', height: '420px' }">
<n-tree-select ref="cameraTree" class="cameraTree" :menu-props="{
class: 'custom-dropdown'
}" v-model:value="option.selectedDataSource" :options="option.dataset.list" clearable
:default-expanded-keys="option.expandedKeys" style="width: 615px; margin-top: 16px" :show="isDropdownOpen"
@update:show="handleShowChange" @update:value="handleSelectDataSource" />
</n-modal>
</div>
<div ref="vYushiVideoRef" class="go-video" :id="uuid"></div>
</div>
</template>
<script setup lang="ts">
//@ts-nocheck
import { PropType, toRefs, shallowReactive, watch, ref, onDeactivated, onMounted, nextTick } from 'vue'
import { useChartDataFetch } from '@/hooks'
import { CreateComponentType } from '@/packages/index.d'
import { useChartEditStore } from '@/store/modules/chartEditStore/chartEditStore'
import { option as configOption } from './config'
import { Base64 } from 'js-base64'
import { md5 } from 'js-md5'
import { PageEnum } from '@/enums/pageEnum'
import axios from 'axios'
import { getUUID, isPreview } from '@/utils'
import { useYushiVideoStore } from '@/store/modules/yushiVideoStore/yushiVideoStore'
const yushiStore = useYushiVideoStore()
let selectedList = []
const cameraTree = ref(null)
const uuid = getUUID()
let keepAliveInterval: any = null
// const expandedKeys = ref([]);
const isDropdownOpen = ref(false)
const props = defineProps({
chartConfig: {
type: Object as PropType<CreateComponentType>,
required: true
}
})
const { w, h } = toRefs(props.chartConfig.attr)
let option = shallowReactive({ ...configOption, expandedKeys: [] })
const showModalRef = ref(false)
//@ts-ignore
const findParentPath = (targetKey, nodes, path = []) => {
for (const node of nodes) {
if (node) {
// 当前路径快照
const currentPath = [...path, { key: node.key, label: node.label }]
console.log(node)
// 找到目标节点时返回完整路径(排除自身)
if (node.key === targetKey) return [...path, { key: node.key, label: node.label }]
// 递归搜索子节点
if (node.children && node.children.length > 0) {
//@ts-ignore
const result = findParentPath(targetKey, node.children, currentPath)
if (result) return result
}
}
}
return null // 未找到
}
//@ts-ignore
const handleShowChange = value => {
// Always set the dropdown to be open
if (!value) {
isDropdownOpen.value = true
}
}
const setNode = (treeData: any) => {
for (let i in treeData) {
if (treeData[i].grade === 'org') {
treeData[i].isLeaf = false
treeData[i].disabled = true
setNode(treeData[i].children)
} else {
treeData[i].isLeaf = true
treeData[i].disabled = false
}
}
}
let showDialog = showModalRef
const handleClick = async () => {
showModalRef.value = true
setTimeout(async () => {
//@ts-ignore
// cameraTree.value.focus();
setNode(option.dataset.list)
isDropdownOpen.value = true
//@ts-ignore
if (option.selectedDataSource === '') {
option.selectedDataSource = option.dataset.defaultValue
}
setTimeout(() => {
scrollIntoView(option.selectedDataSource)
}, 200)
}, 0)
}
const scrollIntoView = (key: any) => {
if (!cameraTree.value) return
const targetElement = document.getElementsByClassName('v-binder-follower-content')
if (targetElement) {
//@ts-ignore
targetElement[0].style.transform = 'translateX(14px) translateY(104px)'
// targetElement.scrollIntoView({ behavior: 'smooth', block: 'center' })
}
}
const getDefaultSelectedCamera = (list, index) => {
for (let i in list) {
if (list[i] && list[i].grade === 'camera') {
if (selectedList.length < Number(index)) {
selectedList.push(list[i].key)
}
} else if (list[i]) {
getDefaultSelectedCamera(list[i].children, index)
}
}
return selectedList[Number(index) - 1]
}
// 预览更新
const vYushiVideoRef = ref(null)
useChartDataFetch(props.chartConfig, useChartEditStore, (newData: any) => {
option.dataset = newData
})
// 编辑更新
watch(
() => props.chartConfig.option,
(newData: any) => {
option = newData
},
{
immediate: true,
deep: true
}
)
// 组织联动放开
watch(
() => yushiStore.$state.currentCameraOrgTree,
(newData: any) => {
option.dataset.list = newData
//切换时同时设置默认摄像头
selectedList = []
option.selectedDataSource = getDefaultSelectedCamera(newData, option.videoIndex)
if (newData) {
let path = findParentPath(option.selectedDataSource, newData)
props.chartConfig.option.videoTitle = path[1].label + '(' + path[path.length - 1].label + ')'
props.chartConfig.option.expandedKeys = path.map((item: any) => {
return item.key
})
}
//@ts-ignore
window.imosPlayer.playLive(window['_iframeId' + option.videoIndex], {
camera: option.selectedDataSource,
//@ts-ignore
title: option.selectedDataSource
})
},
{
immediate: false,
deep: true
}
)
//宇视摄像头
const keepalive = (token: any) => {
let ipaddr = PageEnum.VMIP
let linkPort = PageEnum.VM_PORT
let VIIDPort = PageEnum.VIID_PORT
axios({
method: 'POST',
url: `http://${ipaddr}${VIIDPort && ':' + VIIDPort}/VIID/hadesadapter/user/keepalive`,
headers: {
'Content-Type': 'application/json; charset=utf8',
Authorization: token
},
responseType: 'json'
}).then(res => {
if (res.data.ErrCode === 404) {
// VM B3351以前的用法
//@ts-ignore
const kaCLose = e => {
console.warn('user dead', e)
}
let ws = new WebSocket(`ws://${ipaddr}${VIIDPort && ':' + VIIDPort}/VIID/event`)
ws.onopen = () => {
ws.send(token)
}
ws.onclose = kaCLose
ws.onerror = kaCLose
} else {
// VM B3351以后的用法
// let reConnect = 0
let time = setInterval(() => {
axios({
method: 'POST',
url: `http://${ipaddr}${VIIDPort && ':' + VIIDPort}/VIID/hadesadapter/user/keepalive`,
headers: {
'Content-Type': 'application/json; charset=utf8',
Authorization: token
},
responseType: 'json'
}).then(async res2 => {
if (res2.data.ErrCode !== 0) {
// reConnect++
// if (reConnect > 5) {
clearInterval(time)
await initplayer(newData)
// reConnect = 0
console.warn('user dead')
// }
} else {
// reConnect = 0
}
})
}, 30000)
}
})
}
const getRandom = () => {
function S4() {
return (((1 + Math.random()) * 0x10000) | 0).toString(16).substring(1)
}
return S4() + S4() + '-' + S4() + '-' + S4() + '-' + S4() + '-' + S4() + S4() + S4()
}
//@ts-ignore
const initplayer = async (newData: any) => {
const password = PageEnum.VM_PASS
const username = PageEnum.VM_NAME
let ipaddr = PageEnum.VMIP
let linkPort = PageEnum.VM_PORT
let VIIDPort = PageEnum.VIID_PORT
let date = new Date()
let day = ('0' + date.getDate()).slice(-2)
let month = ('0' + (date.getMonth() + 1)).slice(-2)
let today = date.getFullYear() + '-' + month + '-' + day
//@ts-ignore
let liveNetProtocol = undefined
//@ts-ignore
if (liveNetProtocol === undefined) {
liveNetProtocol = 'tcp'
}
//@ts-ignore
let replayNetProtocol = undefined
if (replayNetProtocol === undefined) {
replayNetProtocol = 'tcp'
}
//@ts-ignore
let liveByMS = true
let nbsp = String.fromCharCode(160)
//@ts-ignore
let arr = []
arr.length = 40
//@ts-ignore
let str = arr.fill(nbsp).join('')
axios({
method: 'POST',
url: `http://${ipaddr}${VIIDPort && ':' + VIIDPort}/VIID/login/v2`,
headers: {
'Content-Type': 'application/json; charset=utf8'
},
responseType: 'json'
})
.then(res => {
const AccessCode = res.data.AccessCode //res.body改为res.data
let usernameEncrypted = Base64.encode(username)
let passwordEncrypted = md5(password)
const loginStr = usernameEncrypted + AccessCode + passwordEncrypted
const LoginSignature = md5(loginStr)
axios({
method: 'POST',
url: `http://${ipaddr}${VIIDPort && ':' + VIIDPort}/VIID/login/v2`,
headers: {
'Content-Type': 'application/json; charset=utf8'
},
responseType: 'json',
data: {
UserName: username,
AccessCode,
LoginSignature
}
})
.then(res => {
// 悬浮方案要求网页title唯一参照接口文档【悬浮播放器概念】
document.title = document.title + str + getRandom()
const token = res.data.AccessToken
//@ts-ignore
window.token = res.data.AccessToken
// if (option.videoIndex === '1') {
keepalive(token)
// }
// 跨nat映射时如果对接端口不是一一映射到8093则本接口必调。
// 或https代理的对接端口不是443本接口必调
// 本接口必须在init之前调用
//@ts-ignore
window.imosPlayer.setLinkPort(linkPort)
//@ts-ignore
window.imosPlayer
.init({
ip: ipaddr, // 必传
token: token, // 必传
title: document.title, // 必传
offset: [0, 0]
})
//@ts-ignore
.then(async res => {
//@ts-ignore
if (res.ErrCode === 0) {
//@ts-ignore
window.imosPlayer.cssScale(option.sca)
//@ts-ignore
window.imosPlayer.setLiveNetLinkMode(liveNetProtocol, liveByMS)
// 国产电脑坐标计算
// window.imosPlayer.setWindowParams(windowParams);
//@ts-ignore
window.__login = true
//console.log('登录成功请创建窗格建议打开F12方便查看运行结果')
// await getDataSource(newData)
//创建视频窗格
//@ts-ignore
let videoDom = await window.imosPlayer.createPanelWindow()
// 窗格默认400*400通过样式改宽高
videoDom.style.width = '100%'
videoDom.style.height = '100%'
//@ts-ignore
window.imosPlayer
.setVoidClassName(videoDom, {
className: 'basic'
})
//@ts-ignore
.then(e => {
console.log(e)
})
//@ts-ignore
//放开后视频播放被覆盖,黑屏看不到
window.imosPlayer.setVoidBroadCastRegion({
voidClassName: 'basic',
region: {
times: 1000,
class: ['n-modal-body-wrapper']
}
})
// let array = document.getElementsByClassName('go-video')
//@ts-ignore
// window.imosPlayer.setViewDomByClassName({
// className: 'basic',
// doms: array
// })
//@ts-ignore
window['_iframeId' + option.videoIndex] = videoDom.id
let divDom = document.getElementById(uuid)
//@ts-ignore
divDom.style.width = '100%'
//@ts-ignore
divDom.style.height = '84%'
//@ts-ignore
if (divDom.children.length === 0) {
//@ts-ignore
divDom.appendChild(videoDom)
}
//@ts-ignore
window.imosPlayer.playLive(videoDom.id, {
camera: option.dataset.defaultValue,
//@ts-ignore
title: option.dataset.defaultValue
})
} else {
//console.log(res.ErrMsg)
}
})
//@ts-ignore
.catch(err => {
console.error(err)
})
})
.catch(e => {
console.log('登录失败!!!')
})
})
.catch(() => console.log('登录失败!!!'))
}
const handleSelectDataSource = (v: any) => {
showModalRef.value = false
option.selectedDataSource = v
let path = findParentPath(v, option.dataset.list)
option.videoTitle = path[1].label + '(' + path[path.length - 1].label + ')'
option.expandedKeys = path.map((item: any) => {
return item.key
})
//@ts-ignore
window.imosPlayer.playLive(window['_iframeId' + option.videoIndex], {
camera: v,
//@ts-ignore
title: v
})
}
const getDataSource = (newData: any) => {
// 查相机编码
let data = {
org: 'iccsid',
condition: {
ItemNum: 3,
Condition: [
{
QueryType: 256,
LogicFlag: 0,
QueryData: '1001'
},
{
QueryType: 257,
LogicFlag: 0,
QueryData: '1'
},
{
QueryType: 1,
LogicFlag: 5,
QueryData: ''
}
],
QueryCount: 1,
PageFirstRowNumber: 0,
PageRowNum: 200
}
}
let conditionEncodeStr1 = encodeURIComponent(JSON.stringify(data.condition))
let url = `http://${PageEnum.VMIP}:${PageEnum.VIID_PORT}/VIID/query?org=${data.org}&condition=${conditionEncodeStr1}`
//@ts-ignore
axios({
method: 'GET',
url: url,
headers: {
//@ts-ignore
Authorization: window.token
},
contentType: 'application/json'
})
//@ts-ignore
.then(res => {
//@ts-ignore
if (res.data.ErrCode === 0) {
//@ts-ignore
let infoList = res.data.Result.InfoList
console.log(infoList)
//@ts-ignore
let arr = []
//@ts-ignore
infoList.forEach(info => {
//@ts-ignore
if (info.ResItemV1.ResStatus === 1) {
// 不要离线的
arr.push({
//@ts-ignore
value: info.ResItemV1.ResCode,
//@ts-ignore
label: info.ResItemV1.ResName
})
}
})
//@ts-ignore
option.dataSource = arr
//@ts-ignore
option.selectedDataSource = arr[0] ? arr[0].value : ''
} else {
console.log(`查询相机失败:[${res.data.ErrCode}] ${res.data.ErrMsg}`)
}
})
}
2025-08-28 11:06:02 +08:00
const fetchCameraTree = async () => {
console.log('fetchCameraTree: 正在获取摄像头树数据...');
try {
const res = await axios.get('/dev/api/camera/tree');
if (res.status === 200 && res.data && res.data.state) {
console.log("API 响应数据 (Camera Tree):", res.data.value);
return res.data.value;
} else {
console.warn('API 返回非预期数据结构或空数据:', res);
return [];
}
} catch (error) {
console.error('获取摄像头树失败:', error);
return [];
}
};
2025-08-28 10:36:02 +08:00
onMounted(async () => {
if (isPreview()) {
yushiStore.setCurrentCameraOrgTree(yushiStore.getCameraOrgTree)
2025-08-28 11:06:02 +08:00
props.chartConfig.option.dataset.list = await fetchCameraTree();
2025-08-28 10:36:02 +08:00
await initplayer(props.chartConfig.option.dataset)
if (props.chartConfig.option.dataset.list) {
//加载title
let path = findParentPath(props.chartConfig.option.dataset.defaultValue, props.chartConfig.option.dataset.list)
props.chartConfig.option.videoTitle = path[1].label + '(' + path[path.length - 1].label + ')'
props.chartConfig.option.expandedKeys = path.map((item: any) => {
return item.key
})
}
}
2025-08-28 11:06:02 +08:00
}
)
2025-08-28 10:36:02 +08:00
onDeactivated(() => {
//@ts-ignore
clearInterval(keepAliveInterval)
})
const dataSetHandle = async (newData: any) => {
await initplayer(newData)
}
const { vChartRef } = useChartDataFetch(props.chartConfig, useChartEditStore, (newData: any) => {
//加载title
if (newData.list) {
let path = findParentPath(newData.dataset.defaultValue, newData.list)
props.chartConfig.option.videoTitle = path[1].label + '(' + path[path.length - 1].label + ')'
props.chartConfig.option.expandedKeys = path.map((item: any) => {
return item.key
})
dataSetHandle(newData)
}
})
</script>
<style lang="scss" scoped>
@include go('video') {
display: block;
object-fit: v-bind('option.fit');
// width: 100%;
// height: v-bind('h * 0.9');
}
.video_title {
margin: 8px 8px;
display: flex;
color: #fff;
justify-content: space-between;
}
.video_title_new {
background: #0d4b61;
padding: 4px 4px 4px 12px;
margin: 4px 0;
}
.videoChangBtn {
padding: 4px 4px;
border: 1px solid rgb(24, 160, 219);
background: transparent;
height: 20px;
}
.videoChangBtn_new {
width: 52px;
height: 23px;
background-repeat: no-repeat;
background-image: url('');
}
</style>
<style lang="scss">
.custom-tab-modal>.n-card-header {
background-color: rgba(26, 56, 113, 1) !important;
background-image: linear-gradient(to right, rgba(8, 100, 177, 0.7), transparent) !important;
padding: 16px !important;
border: 2px solid rgba(62, 200, 244, 1);
border-bottom-width: 0px !important;
}
.custom-tab-modal>.n-card__content {
background-color: rgba(26, 56, 113, 1) !important;
border: 2px solid rgba(62, 200, 244, 1);
padding-left: 12px;
}
.custom-tab-modal__tab .n-button {
--normal-border: #115f8c;
border-radius: 0;
background: linear-gradient(to left, var(--normal-border), var(--normal-border)) left top no-repeat,
linear-gradient(to bottom, var(--normal-border), var(--normal-border)) left top no-repeat,
linear-gradient(to left, var(--normal-border), var(--normal-border)) right top no-repeat,
linear-gradient(to bottom, var(--normal-border), var(--normal-border)) right top no-repeat,
linear-gradient(to left, var(--normal-border), var(--normal-border)) left bottom no-repeat,
linear-gradient(to bottom, var(--normal-border), var(--normal-border)) left bottom no-repeat,
linear-gradient(to left, var(--normal-border), var(--normal-border)) right bottom no-repeat,
linear-gradient(to left, var(--normal-border), var(--normal-border)) right bottom no-repeat;
background-size: 0.1rem 0.5rem, 0.5rem 0.1rem, 0.1rem 0.5rem, 0.5rem 0.1rem;
padding: 0;
}
.custom-tab-modal__tab .n-button:hover {
--active-border: #00e4ff;
font-weight: 700;
background: linear-gradient(to left, var(--active-border), var(--active-border)) left top no-repeat,
linear-gradient(to bottom, var(--active-border), var(--active-border)) left top no-repeat,
linear-gradient(to left, var(--active-border), var(--active-border)) right top no-repeat,
linear-gradient(to bottom, var(--active-border), var(--active-border)) right top no-repeat,
linear-gradient(to left, var(--active-border), var(--active-border)) left bottom no-repeat,
linear-gradient(to bottom, var(--active-border), var(--active-border)) left bottom no-repeat,
linear-gradient(to left, var(--active-border), var(--active-border)) right bottom no-repeat,
linear-gradient(to left, var(--active-border), var(--active-border)) right bottom no-repeat;
background-size: 0.1rem 0.5rem, 0.5rem 0.1rem, 0.1rem 0.5rem, 0.5rem 0.1rem;
}
.custom-tab-modal__tab .active-button {
--active-border: #00e4ff;
font-weight: 700;
background: linear-gradient(to left, var(--active-border), var(--active-border)) left top no-repeat,
linear-gradient(to bottom, var(--active-border), var(--active-border)) left top no-repeat,
linear-gradient(to left, var(--active-border), var(--active-border)) right top no-repeat,
linear-gradient(to bottom, var(--active-border), var(--active-border)) right top no-repeat,
linear-gradient(to left, var(--active-border), var(--active-border)) left bottom no-repeat,
linear-gradient(to bottom, var(--active-border), var(--active-border)) left bottom no-repeat,
linear-gradient(to left, var(--active-border), var(--active-border)) right bottom no-repeat,
linear-gradient(to left, var(--active-border), var(--active-border)) right bottom no-repeat;
background-size: 0.1rem 0.5rem, 0.5rem 0.1rem, 0.1rem 0.5rem, 0.5rem 0.1rem;
}
.custom-tab-modal__tab .button_content {
width: 126px;
display: flex;
justify-content: center;
align-items: center;
height: 32px;
background-color: rgba(10, 166, 254, 0.3) !important;
}
.custom-tab-modal__tab .n-button:hover .button_content {
background-color: rgba(0, 228, 255, 0.3) !important;
}
.custom-tab-modal__tab .active-button .button_content {
background-color: rgba(0, 228, 255, 0.3) !important;
}
.custom-data-table .n-data-table-th,
.custom-data-table .n-data-table-thead,
.custom-data-table .n-data-table-table,
.custom-data-table .n-data-table-tr:not(.n-data-table-tr--summary):hover {
background-color: rgba(60, 124, 211, 0.15) !important;
}
.custom-data-table tbody td:nth-child(odd),
.custom-data-table .n-data-table-td {
background-color: rgba(60, 124, 211, 0.05) !important;
}
.custom-data-table .n-data-table-tr.n-data-table-tr--striped {
background-color: rgba(60, 124, 211, 0.15) !important;
}
.custom-data-table .operation {
color: rgb(0, 228, 255) !important;
}
.custom-data-table .n-data-table-th__title {
color: #fff !important;
}
.custom-dropdown {
background: transparent !important;
border: none !important;
box-shadow: unset !important;
width: 615px !important;
}
.custom-data-table .v-binder-follower-content {
transform: translateX(7px) translateY(104px) !important;
}
</style>