742 lines
25 KiB
Vue
742 lines
25 KiB
Vue
<!-- 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'
|
||
import axiosInstance from '@/api/axios'
|
||
|
||
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 axiosInstance.get(`/awjt/api/camera/tree/${option.sceneCode}`, { baseURL: '', responseType: 'json' });
|
||
console.log("API 响应数据 (Camera Tree):", res);
|
||
if (res && res.state === true) {
|
||
console.log("API 响应数据 (Camera Tree):", res.value);
|
||
return res.value;
|
||
} else {
|
||
console.warn('API 返回非预期数据结构或空数据:', res);
|
||
return [];
|
||
}
|
||
} catch (error) {
|
||
console.error('获取摄像头树失败使用静态数据:', error);
|
||
return props.chartConfig.option.dataset.list;
|
||
}
|
||
};
|
||
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('');
|
||
}
|
||
|
||
.title_text {
|
||
padding-left: 10px;
|
||
}
|
||
|
||
.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>
|