|
|
|
<template>
|
|
|
|
<div class="width100 flex-column flex-justify-start flex-align-start" style="background-color: white;">
|
|
|
|
|
|
|
|
<p class="fz16 ml10 mt15 mb10 color-text-title color-font-bold" style="color: #939393">
|
|
|
|
Before starting manually, please select the publish mode and livestream type
|
|
|
|
</p>
|
|
|
|
<div
|
|
|
|
class="mt15 flex-row flex-align-center flex-justify-between"
|
|
|
|
style="width: 100%;">
|
|
|
|
<p class="ml10 mb0 fz16" style="color: black">
|
|
|
|
Select Video Publish Mode:
|
|
|
|
</p>
|
|
|
|
<a-select
|
|
|
|
style="width: 200px; margin-right: 20px;"
|
|
|
|
placeholder="Select Mode"
|
|
|
|
@select="onPublishModeSelect"
|
|
|
|
>
|
|
|
|
<a-select-option
|
|
|
|
v-for="item in publishModeList"
|
|
|
|
:key="item.label"
|
|
|
|
:value="item.value"
|
|
|
|
>
|
|
|
|
{{ item.label }}
|
|
|
|
</a-select-option>
|
|
|
|
</a-select>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div class="ml10 mr10" style="width: 96%; margin-top: -10px;">
|
|
|
|
<a-divider />
|
|
|
|
</div>
|
|
|
|
<div
|
|
|
|
class="flex-row flex-align-center flex-justify-between"
|
|
|
|
style="width: 100%; margin-top: -10px;"
|
|
|
|
>
|
|
|
|
<p class="ml10 mb0 fz16">Select Livestream Type:</p>
|
|
|
|
<a-select
|
|
|
|
style="width: 200px; margin-right: 20px;"
|
|
|
|
placeholder="Select Live Type"
|
|
|
|
:value="liveStreamStatus.type"
|
|
|
|
@select="onLiveTypeSelect"
|
|
|
|
>
|
|
|
|
<a-select-option
|
|
|
|
v-for="item in liveTypeList"
|
|
|
|
:key="item.label"
|
|
|
|
:value="item.value"
|
|
|
|
>
|
|
|
|
{{ item.label }}
|
|
|
|
</a-select-option>
|
|
|
|
</a-select>
|
|
|
|
</div>
|
|
|
|
<div class="ml10 mr10" style="width: 96%; margin-top: -10px;">
|
|
|
|
<a-divider />
|
|
|
|
</div>
|
|
|
|
<div class="width-100" style="margin-top: -10px;">
|
|
|
|
<div class="ml10" style="width: 97%;">
|
|
|
|
<span class="fz16">Param: </span>
|
|
|
|
<span v-if="liveStreamStatus.type === ELiveTypeValue.Agora" style="word-break: break-all; color: #75c5f6;">
|
|
|
|
<div class="flex-col flex-justify-center flex-align-center">
|
|
|
|
<div>
|
|
|
|
<span class="ml10">Token:</span>
|
|
|
|
<a-input
|
|
|
|
class="ml10"
|
|
|
|
v-model:value="agoraParam.token"
|
|
|
|
placeholder="Token"
|
|
|
|
></a-input>
|
|
|
|
</div>
|
|
|
|
<div>
|
|
|
|
<span class="ml10">Channel:</span>
|
|
|
|
<a-input
|
|
|
|
class="ml10"
|
|
|
|
v-model:value="agoraParam.channelId"
|
|
|
|
placeholder="Channel"
|
|
|
|
></a-input>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</span>
|
|
|
|
<span v-else-if="liveStreamStatus.type === ELiveTypeValue.RTMP" style="word-break: break-all; color: #75c5f6;">{{ rtmpParam }}</span>
|
|
|
|
<span v-else-if="liveStreamStatus.type === ELiveTypeValue.RTSP" style="word-break: break-all; color: #75c5f6;">{{ rtspParam }}</span>
|
|
|
|
<span v-else-if="liveStreamStatus.type === ELiveTypeValue.GB28181" style="word-break: break-all; color: #75c5f6;">{{ gb28181Param }}</span>
|
|
|
|
<span v-else></span>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
</div>
|
|
|
|
<div class="ml10 mr10" style="width: 96%; margin-top: -10px;">
|
|
|
|
<a-divider />
|
|
|
|
</div>
|
|
|
|
<div class="mb20 flex-row flex-align-center flex-justify-center"
|
|
|
|
style="width: 100%; ">
|
|
|
|
<a-button class="flex-column fz20 flex-align-center flex-justify-center" style="width: 100px;" type="ghost" @click="onPlay">Play</a-button>
|
|
|
|
<a-button class="flex-column fz20 flex-align-center flex-justify-center ml40" style="width: 100px;" type="ghost" @click="onStop">Stop</a-button>
|
|
|
|
</div>
|
|
|
|
<a-button v-if="playVisiable" class="flex-column flex-align-center" shape="circle" @click="showLivingStatus"
|
|
|
|
style="position: fixed; top: 13vh; left: 5vw; opacity: 0.8; background-color: rgb(0,0,0,0)">
|
|
|
|
<template #icon><CaretRightFilled style="font-size: 26px; color: " /></template>
|
|
|
|
</a-button>
|
|
|
|
|
|
|
|
<a-drawer placement="right" v-model:visible="drawerVisible" width="280px" :mask="false" @close="closeDrawer">
|
|
|
|
<div class="fz16 width-100">
|
|
|
|
<div class="mt20" style=" margin-bottom: -10px;">
|
|
|
|
<span class="fz20 flex-row flex-align-center flex-justify-center">
|
|
|
|
<font :color="liveState === EStatusValue.LIVING ? 'green' : liveState === EStatusValue.CONNECTED ? 'blue' : 'red'">{{ liveState }}</font></span>
|
|
|
|
</div>
|
|
|
|
<a-divider />
|
|
|
|
<div style=" margin-top: -10px; margin-bottom: -15px;">
|
|
|
|
<span>Frame Rate:</span><span style="float: right; color: #75c5f6;">{{ liveStreamStatus.fps }}<span v-if="liveStreamStatus.fps != -1"> fps</span></span><br/>
|
|
|
|
</div>
|
|
|
|
<a-divider />
|
|
|
|
<div style=" margin-top: -10px; margin-bottom: -10px;">
|
|
|
|
<span>Video Bit Rate:</span><span style="float: right; color: #75c5f6;">{{ liveStreamStatus.videoBitRate }}<span v-if="liveStreamStatus.videoBitRate != -1"> kbps</span></span><br/>
|
|
|
|
</div>
|
|
|
|
<a-divider />
|
|
|
|
<div style=" margin-top: -10px; margin-bottom: -10px;">
|
|
|
|
<span>Audio Bit Rate:</span><span style="float: right; color: #75c5f6;">{{ liveStreamStatus.audioBitRate }}<span v-if="liveStreamStatus.audioBitRate != -1"> kbps</span></span><br/>
|
|
|
|
</div>
|
|
|
|
<a-divider />
|
|
|
|
<div style=" margin-top: -10px; margin-bottom: -10px;">
|
|
|
|
<span>Packet Loss Rate:</span><span style="float: right; color: #75c5f6;">{{ liveStreamStatus.dropRate }}<span v-if="liveStreamStatus.dropRate != -1"> %</span></span><br/>
|
|
|
|
</div>
|
|
|
|
<a-divider />
|
|
|
|
<div style=" margin-top: -10px; margin-bottom: -10px;">
|
|
|
|
<span>RTT:</span><span style="float: right; color: #75c5f6;">{{ liveStreamStatus.rtt }}<span v-if="liveStreamStatus.rtt != -1"> ms</span></span><br/>
|
|
|
|
</div>
|
|
|
|
<a-divider />
|
|
|
|
<div style=" margin-top: -10px;">
|
|
|
|
<span >Jitter:</span><span style="float: right; color: #75c5f6;">{{ liveStreamStatus.jitter }}</span><br/>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</a-drawer>
|
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<script lang="ts" setup>
|
|
|
|
import { message } from 'ant-design-vue'
|
|
|
|
import { onMounted, reactive, ref, UnwrapRef } from 'vue'
|
|
|
|
import { CURRENT_CONFIG as config, CURRENT_CONFIG } from '/@/api/http/config'
|
|
|
|
import { ELiveTypeName, ELiveTypeValue, GB28181Param, LiveConfigParam, LiveStreamStatus, RTSPParam, EVideoPublishType } from '/@/types/live-stream'
|
|
|
|
import apiPilot from '/@/api/pilot-bridge'
|
|
|
|
import { getRoot } from '/@/root'
|
|
|
|
import { ELiveStatusValue, EStatusValue } from '/@/types'
|
|
|
|
import { CaretRightFilled } from '@ant-design/icons-vue'
|
|
|
|
|
|
|
|
const root = getRoot()
|
|
|
|
|
|
|
|
const publishModeList = [
|
|
|
|
{
|
|
|
|
value: EVideoPublishType.VideoOnDemand,
|
|
|
|
label: EVideoPublishType.VideoOnDemand
|
|
|
|
},
|
|
|
|
{
|
|
|
|
value: EVideoPublishType.VideoByManual,
|
|
|
|
label: EVideoPublishType.VideoByManual
|
|
|
|
},
|
|
|
|
{
|
|
|
|
value: EVideoPublishType.VideoDemandAuxManual,
|
|
|
|
label: EVideoPublishType.VideoDemandAuxManual
|
|
|
|
}
|
|
|
|
]
|
|
|
|
const liveTypeList = [
|
|
|
|
{
|
|
|
|
value: ELiveTypeValue.Agora,
|
|
|
|
label: ELiveTypeName.Agora
|
|
|
|
},
|
|
|
|
{
|
|
|
|
value: ELiveTypeValue.RTMP,
|
|
|
|
label: ELiveTypeName.RTMP
|
|
|
|
},
|
|
|
|
{
|
|
|
|
value: ELiveTypeValue.RTSP,
|
|
|
|
label: ELiveTypeName.RTSP
|
|
|
|
},
|
|
|
|
{
|
|
|
|
value: ELiveTypeValue.GB28181,
|
|
|
|
label: ELiveTypeName.GB28181
|
|
|
|
}
|
|
|
|
]
|
|
|
|
const agoraParam = reactive({
|
|
|
|
uid: '2892130292',
|
|
|
|
token: config.agoraToken,
|
|
|
|
channelId: config.agoraChannel
|
|
|
|
})
|
|
|
|
const rtmpParam = {
|
|
|
|
url: config.rtmpURL + new Date().getTime()
|
|
|
|
}
|
|
|
|
const rtspParam: RTSPParam = {
|
|
|
|
userName: CURRENT_CONFIG.rtspUserName,
|
|
|
|
password: CURRENT_CONFIG.rtspPassword,
|
|
|
|
port: CURRENT_CONFIG.rtspPort
|
|
|
|
}
|
|
|
|
const gb28181Param: GB28181Param = {
|
|
|
|
serverIp: CURRENT_CONFIG.gbServerIp,
|
|
|
|
serverPort: CURRENT_CONFIG.gbServerPort,
|
|
|
|
serverId: CURRENT_CONFIG.gbServerId,
|
|
|
|
agentId: CURRENT_CONFIG.gbAgentId,
|
|
|
|
password: CURRENT_CONFIG.gbPassword,
|
|
|
|
agentPort: CURRENT_CONFIG.gbAgentPort,
|
|
|
|
agentChannel: CURRENT_CONFIG.gbAgentChannel,
|
|
|
|
}
|
|
|
|
|
|
|
|
const playVisiable = ref(false)
|
|
|
|
const drawerVisible = ref(false)
|
|
|
|
const liveState = ref(EStatusValue.DISCONNECT)
|
|
|
|
const liveTypeSelected = ref<string>()
|
|
|
|
const publishModeSelected = ref<string>()
|
|
|
|
const liveStreamStatus: LiveStreamStatus = reactive({
|
|
|
|
audioBitRate: -1,
|
|
|
|
dropRate: -1,
|
|
|
|
fps: -1,
|
|
|
|
jitter: -1,
|
|
|
|
quality: -1,
|
|
|
|
rtt: -1,
|
|
|
|
status: -1,
|
|
|
|
type: -1,
|
|
|
|
videoBitRate: -1
|
|
|
|
})
|
|
|
|
|
|
|
|
onMounted(() => {
|
|
|
|
const config: LiveConfigParam = JSON.parse(apiPilot.getLiveshareConfig())
|
|
|
|
liveStreamStatus.type = config.type
|
|
|
|
refreshLiveType()
|
|
|
|
|
|
|
|
window.liveStatusCallback = arg => {
|
|
|
|
liveStatusCallback(arg)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
const liveStatusCallback = async (arg: LiveStreamStatus) => {
|
|
|
|
liveStreamStatus.fps = arg.fps
|
|
|
|
liveStreamStatus.audioBitRate = arg.audioBitRate
|
|
|
|
liveStreamStatus.dropRate = arg.dropRate
|
|
|
|
liveStreamStatus.jitter = arg.jitter
|
|
|
|
liveStreamStatus.rtt = arg.rtt
|
|
|
|
liveStreamStatus.videoBitRate = arg.videoBitRate
|
|
|
|
liveStreamStatus.quality = arg.quality
|
|
|
|
liveStreamStatus.type = arg.type
|
|
|
|
liveStreamStatus.status = arg.status
|
|
|
|
|
|
|
|
switch (liveStreamStatus.status) {
|
|
|
|
case ELiveStatusValue.LIVING:
|
|
|
|
liveState.value = EStatusValue.LIVING
|
|
|
|
break
|
|
|
|
case ELiveStatusValue.CONNECTED:
|
|
|
|
liveState.value = EStatusValue.CONNECTED
|
|
|
|
break
|
|
|
|
default:
|
|
|
|
liveState.value = EStatusValue.DISCONNECT
|
|
|
|
}
|
|
|
|
}
|
|
|
|
function refreshLiveType () {
|
|
|
|
switch (liveStreamStatus.type) {
|
|
|
|
case ELiveTypeValue.Agora:
|
|
|
|
liveTypeSelected.value = ELiveTypeName.Agora
|
|
|
|
break
|
|
|
|
case ELiveTypeValue.RTMP:
|
|
|
|
liveTypeSelected.value = ELiveTypeName.RTMP
|
|
|
|
break
|
|
|
|
case ELiveTypeValue.RTSP:
|
|
|
|
liveTypeSelected.value = ELiveTypeName.RTSP
|
|
|
|
break
|
|
|
|
case ELiveTypeValue.GB28181:
|
|
|
|
liveTypeSelected.value = ELiveTypeName.GB28181
|
|
|
|
break
|
|
|
|
default:
|
|
|
|
liveTypeSelected.value = ELiveTypeName.Unknown
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const onLiveTypeSelect = (val: number) => {
|
|
|
|
liveStreamStatus.type = val
|
|
|
|
refreshLiveType()
|
|
|
|
}
|
|
|
|
const onPublishModeSelect = (val: string) => {
|
|
|
|
publishModeSelected.value = val
|
|
|
|
apiPilot.setVideoPublishType(publishModeSelected.value)
|
|
|
|
}
|
|
|
|
const onPlay = () => {
|
|
|
|
console.info(JSON.stringify(agoraParam))
|
|
|
|
if (!publishModeSelected.value) {
|
|
|
|
message.warn('Please select publish mode!')
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if (liveTypeSelected.value === ELiveTypeName.Unknown) {
|
|
|
|
message.warn('Please select livestream type!')
|
|
|
|
return
|
|
|
|
}
|
|
|
|
switch (liveStreamStatus.type) {
|
|
|
|
case 1: {
|
|
|
|
apiPilot.setLiveshareConfig(ELiveTypeValue.Agora, JSON.stringify(agoraParam))
|
|
|
|
break
|
|
|
|
}
|
|
|
|
case 2: {
|
|
|
|
apiPilot.setLiveshareConfig(ELiveTypeValue.RTMP, JSON.stringify(rtmpParam))
|
|
|
|
break
|
|
|
|
}
|
|
|
|
case 3: {
|
|
|
|
apiPilot.setLiveshareConfig(ELiveTypeValue.RTSP, JSON.stringify(rtspParam))
|
|
|
|
break
|
|
|
|
}
|
|
|
|
case 4: {
|
|
|
|
apiPilot.setLiveshareConfig(ELiveTypeValue.GB28181, JSON.stringify(gb28181Param))
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const status = apiPilot.startLiveshare()
|
|
|
|
if (status) {
|
|
|
|
playVisiable.value = true
|
|
|
|
drawerVisible.value = true
|
|
|
|
message.success('success')
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const showLivingStatus = () => {
|
|
|
|
drawerVisible.value = !drawerVisible.value
|
|
|
|
}
|
|
|
|
|
|
|
|
const onStop = () => {
|
|
|
|
const status = apiPilot.stopLiveshare()
|
|
|
|
if (status) {
|
|
|
|
message.success('success')
|
|
|
|
playVisiable.value = false
|
|
|
|
drawerVisible.value = false
|
|
|
|
setTimeout(() => {
|
|
|
|
let key: (keyof LiveStreamStatus)
|
|
|
|
for (key in liveStreamStatus) {
|
|
|
|
if (key === 'type') {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
liveStreamStatus[key] = -1
|
|
|
|
}
|
|
|
|
}, 2000)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
</script>
|
|
|
|
|
|
|
|
<style lang="scss" scoped>
|
|
|
|
// @import '/@/styles/index.scss';
|
|
|
|
</style>
|