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

736 lines
25 KiB
Vue
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

<!-- eslint-disable vue/multi-word-component-names -->
<template>
<div>
<SmallBaorder01 :titleText="option.titleText" :title-option="option.titleOption"
:headerOption="option.headerOption">
<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"
class="button">切换</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>
</SmallBaorder01>
</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'
import SmallBaorder01 from '../SmallBorder01Co/index.vue'
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}`)
}
})
}
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 [];
}
};
onMounted(async () => {
if (isPreview()) {
yushiStore.setCurrentCameraOrgTree(yushiStore.getCameraOrgTree)
props.chartConfig.option.dataset.list = await fetchCameraTree();
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
})
}
}
}
)
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('');
}
.button {
background-color: #1E5676;
border: 1px solid #2E6E89;
padding: 12px 10px;
margin-right: 20px;
}
</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>