mirror of
https://github.com/kuaifan/dootask.git
synced 2025-12-11 18:42:54 +00:00
741 lines
30 KiB
Vue
741 lines
30 KiB
Vue
<template>
|
|
<div class="meeting-warp">
|
|
<!-- 加入/新建 -->
|
|
<Modal
|
|
v-model="addShow"
|
|
:title="$L(addData.type === 'join' ? '加入会议' : '新会议')"
|
|
:mask-closable="false"
|
|
:closable="!addData.sharekey">
|
|
<Form ref="addForm" :model="addData" :rules="addRule" v-bind="formOptions" @submit.native.prevent>
|
|
<template v-if="addData.type === 'join'">
|
|
<!-- 加入会议 -->
|
|
<FormItem v-if="addData.name" prop="userids" :label="$L('会议主题')">
|
|
<Input v-model="addData.name" disabled/>
|
|
</FormItem>
|
|
<FormItem v-if="addData.sharekey" prop="username" :label="$L('你的姓名')">
|
|
<Input v-model="addData.username" :placeholder="$L('请输入你的姓名')"/>
|
|
</FormItem>
|
|
<FormItem prop="meetingid" :label="$L('会议频道ID')">
|
|
<Input v-model="addData.meetingid" :disabled="addData.meetingdisabled === true" :placeholder="$L('请输入会议频道ID')"/>
|
|
</FormItem>
|
|
</template>
|
|
<template v-else>
|
|
<!-- 新会议 -->
|
|
<FormItem prop="name" :label="$L('会议主题')">
|
|
<Input v-model="addData.name" :maxlength="50" :placeholder="$L('选填')"/>
|
|
</FormItem>
|
|
<FormItem prop="userids" :label="$L('邀请成员')">
|
|
<UserSelect v-model="addData.userids" :uncancelable="[userId]" :multiple-max="20" :title="$L('选择邀请成员')"/>
|
|
</FormItem>
|
|
</template>
|
|
<FormItem prop="tracks">
|
|
<CheckboxGroup v-model="addData.tracks">
|
|
<Checkbox label="audio">
|
|
<span>{{$L('麦克风')}}</span>
|
|
</Checkbox>
|
|
<Checkbox label="video">
|
|
<span>{{$L('摄像头')}}</span>
|
|
</Checkbox>
|
|
</CheckboxGroup>
|
|
</FormItem>
|
|
</Form>
|
|
<div slot="footer" class="adaption">
|
|
<Button type="default" @click="addShow=false" v-if="!addData.sharekey">{{$L('取消')}}</Button>
|
|
<Button type="primary" :loading="loadIng > 0" @click="onSubmit">{{$L(addData.type === 'join' ? '加入会议' : '开始会议')}}</Button>
|
|
</div>
|
|
</Modal>
|
|
|
|
<!-- 会议中 -->
|
|
<Modal
|
|
v-model="meetingShow"
|
|
:title="addData.name"
|
|
ref="meetingModal"
|
|
:mask="false"
|
|
:mask-closable="false"
|
|
:closable="false"
|
|
:transition-names="['', '']"
|
|
:beforeClose="onClose"
|
|
:class-name="`meeting-manager${meetingMini ? ' meeting-hidden' : ''}`"
|
|
:ignore-remove-last="meetingMini"
|
|
fullscreen>
|
|
<ul>
|
|
<li v-if="localUser.uid">
|
|
<Player :player="localUser" isLocal/>
|
|
</li>
|
|
<li v-for="user in remoteUsers">
|
|
<Player :player="user"/>
|
|
</li>
|
|
</ul>
|
|
<div slot="footer" class="adaption meeting-button-group">
|
|
<Button type="primary" :loading="audioLoad" @click="onAudio">
|
|
<i class="taskfont" v-html="localUser.audioTrack ? '' : ''"></i>
|
|
</Button>
|
|
<Button type="primary" :loading="videoLoad" @click="onVideo">
|
|
<i class="taskfont" v-html="localUser.videoTrack ? '' : ''"></i>
|
|
</Button>
|
|
<template v-if="windowPortrait">
|
|
<Button type="primary" :loading="linkCopyLoad" @click="onInvitation('open')">
|
|
<i class="taskfont"></i>
|
|
</Button>
|
|
<Button type="primary" v-if="!addData.sharekey" @click="meetingMini = true">
|
|
<i class="taskfont"></i>
|
|
</Button>
|
|
<Button type="warning" :loading="loadIng > 0" @click="onClose">
|
|
<i class="taskfont"></i>
|
|
</Button>
|
|
</template>
|
|
<template v-else>
|
|
<Button type="primary" @click="onInvitation('open')">{{$L('邀请')}}</Button>
|
|
<Button type="primary" v-if="!addData.sharekey" @click="meetingMini = true">{{$L('最小化')}}</Button>
|
|
<Button type="warning" :loading="loadIng > 0" @click="onClose">{{$L('离开会议')}}</Button>
|
|
</template>
|
|
</div>
|
|
</Modal>
|
|
<DragBallComponent
|
|
v-if="meetingMini"
|
|
id="meetingDragBall"
|
|
:z-index="3000"
|
|
@on-click="meetingMini=false">
|
|
<div class="meeting-drag-ball">
|
|
<i class="taskfont" v-html="localUser.audioTrack ? '' : ''"></i>
|
|
<i class="taskfont" v-html="localUser.videoTrack ? '' : ''"></i>
|
|
<em>{{$L('会议中')}}</em>
|
|
</div>
|
|
</DragBallComponent>
|
|
|
|
<!-- 邀请 -->
|
|
<Modal
|
|
v-model="invitationShow"
|
|
:title="$L('邀请加入')"
|
|
:mask-closable="false">
|
|
<Form ref="invitationForm" :model="invitationData" v-bind="formOptions" @submit.native.prevent>
|
|
<FormItem prop="userids" :label="$L('邀请成员')">
|
|
<UserSelect v-model="invitationData.userids" :multiple-max="20" :title="$L('选择邀请成员')"/>
|
|
</FormItem>
|
|
</Form>
|
|
<div slot="footer" class="adaption">
|
|
<Button type="default" :loading="linkCopyLoad" @click="linkCopy">{{$L('复制链接')}}</Button>
|
|
<Button type="primary" :loading="invitationLoad" @click="onInvitation('submit')">{{$L('发送邀请')}}</Button>
|
|
</div>
|
|
</Modal>
|
|
</div>
|
|
</template>
|
|
|
|
<script>
|
|
import {mapState} from 'vuex'
|
|
import Player from "./player.vue";
|
|
import DragBallComponent from "../../../../components/DragBallComponent";
|
|
import UserSelect from "../../../../components/UserSelect.vue";
|
|
import emitter from "../../../../store/events";
|
|
import {getErrorMessage} from "./utils";
|
|
import {getLanguage} from "../../../../language";
|
|
|
|
export default {
|
|
name: "MeetingManager",
|
|
components: {UserSelect, DragBallComponent, Player},
|
|
props: {
|
|
id: {
|
|
type: String,
|
|
default: () => {
|
|
return "meeting-player-" + Math.round(Math.random() * 10000);
|
|
}
|
|
}
|
|
},
|
|
data() {
|
|
return {
|
|
loadIng: 0,
|
|
|
|
addShow: false,
|
|
addData: {
|
|
userids: [],
|
|
tracks: ['audio']
|
|
},
|
|
addRule: {
|
|
username: [
|
|
{ required: true, message: this.$L('请输入你的姓名!'), trigger: 'change' },
|
|
]
|
|
},
|
|
|
|
invitationShow: false,
|
|
invitationLoad: false,
|
|
invitationData: {
|
|
userids: [],
|
|
},
|
|
|
|
meetingShow: false,
|
|
meetingMini: false,
|
|
audioLoad: false,
|
|
videoLoad: false,
|
|
|
|
agoraClient: null,
|
|
remoteUsers: [],
|
|
localUser: {
|
|
uid: null,
|
|
audioTrack: null,
|
|
videoTrack: null,
|
|
},
|
|
|
|
linkCopyLoad: false,
|
|
}
|
|
},
|
|
|
|
computed: {
|
|
...mapState(['meetingWindow', 'appMeetingShow', 'formOptions', 'userToken']),
|
|
},
|
|
|
|
mounted() {
|
|
emitter.on('addMeeting', this.onAdd);
|
|
},
|
|
|
|
beforeDestroy() {
|
|
emitter.off('addMeeting', this.onAdd);
|
|
},
|
|
|
|
watch: {
|
|
meetingMini(val) {
|
|
if (!val) {
|
|
this.$refs.meetingModal.modalIndex = this.$refs.meetingModal.handleGetModalIndex()
|
|
}
|
|
},
|
|
meetingWindow: {
|
|
handler(data) {
|
|
switch (data.type) {
|
|
// 创建会议
|
|
case 'add':
|
|
this.addShow = data.show;
|
|
break;
|
|
|
|
// 加入会议(直接加入)
|
|
case 'join':
|
|
case 'direct':
|
|
this.addShow = data.show;
|
|
this.addData.type = 'join';
|
|
if (data.meetingNickname) {
|
|
this.addData.username = data.meetingNickname;
|
|
}
|
|
if (data.meetingAvatar) {
|
|
this.addData.userimg = data.meetingAvatar;
|
|
}
|
|
if ($A.runNum(data.meetingAudio) && !this.addData.tracks.includes('audio')) {
|
|
this.addData.tracks.push('audio')
|
|
}
|
|
if ($A.runNum(data.meetingVideo) && !this.addData.tracks.includes('video')) {
|
|
this.addData.tracks.push('video')
|
|
}
|
|
if (data.meetingSharekey) {
|
|
this.addData.sharekey = data.meetingSharekey;
|
|
this.addData.meetingid = data.meetingid || '';
|
|
this.addData.meetingdisabled = !!data.meetingSharekey;
|
|
}
|
|
if (data.type === 'direct') {
|
|
this.onOpen(true);
|
|
}
|
|
break;
|
|
|
|
// 邀请加入
|
|
case 'invitation':
|
|
this.invitationShow = data.show;
|
|
this.invitationLoad = false;
|
|
this.invitationData.meetingid = data.meetingid;
|
|
break;
|
|
|
|
// 加入失败
|
|
case 'error':
|
|
this.addShow = data.show;
|
|
this.invitationShow = data.show;
|
|
this.invitationLoad = false;
|
|
$A.modalError('加入会议失败');
|
|
break;
|
|
}
|
|
},
|
|
immediate: true,
|
|
}
|
|
},
|
|
|
|
methods: {
|
|
onAdd(data) {
|
|
data = $A.isJson(data) ? data : {};
|
|
// 获取会话成员
|
|
if (/^\d+$/.test(data.dialog_id)) {
|
|
this.loadIng++;
|
|
this.$store.dispatch("call", {
|
|
url: 'dialog/user',
|
|
data: {
|
|
dialog_id: data.dialog_id
|
|
}
|
|
}).then(({data}) => {
|
|
this.$set(this.addData, 'userids', data.filter(item => !item.bot).map(item => item.userid))
|
|
}).finally(_ => {
|
|
this.loadIng--;
|
|
});
|
|
delete data.dialog_id;
|
|
}
|
|
// 加上自己
|
|
if (!$A.isArray(data.userids)) {
|
|
data.userids = []
|
|
}
|
|
if (this.userId && !data.userids.includes(this.userId)) {
|
|
data.userids.push(this.userId)
|
|
}
|
|
// 加上音频
|
|
if (!$A.isArray(data.tracks)) {
|
|
data.tracks = ['audio']
|
|
} else if (!data.tracks.includes('audio')) {
|
|
data.tracks.push('audio')
|
|
}
|
|
this.addData = data;
|
|
this.addShow = true;
|
|
},
|
|
|
|
onSubmit() {
|
|
this.$refs.addForm.validate((valid) => {
|
|
if (valid) {
|
|
this.onOpen()
|
|
}
|
|
});
|
|
},
|
|
|
|
async onOpen(isDirect = false) {
|
|
// 判断是否在会议中
|
|
let isMeeting = false;
|
|
if ($A.isEEUiApp) {
|
|
isMeeting = this.appMeetingShow;
|
|
} else if ($A.Electron) {
|
|
const meetingWindow = await $A.Electron.sendAsync("getChildWindow", 'meeting-window')
|
|
if (meetingWindow) {
|
|
const currentWindow = await $A.Electron.sendAsync("getChildWindow", null)
|
|
isMeeting = currentWindow?.id !== meetingWindow.id;
|
|
}
|
|
} else {
|
|
isMeeting = this.meetingShow;
|
|
}
|
|
if (isMeeting) {
|
|
$A.modalWarning({
|
|
content: "正在会议中,无法进入其他会议室",
|
|
onOk: this.onBeforeClose
|
|
});
|
|
return;
|
|
}
|
|
|
|
// 加载动画
|
|
const loader = (add) => {
|
|
if (isDirect) {
|
|
if (add) {
|
|
this.$store.dispatch('showSpinner');
|
|
} else {
|
|
this.$store.dispatch('hiddenSpinner', 600);
|
|
}
|
|
} else {
|
|
if (add) {
|
|
this.loadIng++;
|
|
} else {
|
|
this.loadIng--;
|
|
}
|
|
}
|
|
}
|
|
|
|
// 加入会议
|
|
loader(true);
|
|
this.$store.dispatch("call", {
|
|
url: 'users/meeting/open',
|
|
data: this.addData
|
|
}).then(({data}) => {
|
|
this.$set(this.addData, 'name', data.name);
|
|
this.$set(this.addData, 'meetingid', data.meetingid);
|
|
this.$set(this.localUser, 'nickname', data.nickname);
|
|
this.$set(this.localUser, 'userimg', data.userimg);
|
|
this.$store.dispatch("saveDialogMsg", data.msgs);
|
|
this.$store.dispatch("updateDialogLastMsg", data.msgs);
|
|
delete data.name;
|
|
delete data.msgs;
|
|
// App 直接使用新窗口打开会议
|
|
if ($A.isEEUiApp) {
|
|
loader(true);
|
|
this.loadNum = 0
|
|
this.loadTimer && clearInterval(this.loadTimer)
|
|
this.loadTimer = setInterval(() => {
|
|
if (this.addShow && ++this.loadNum < 10) {
|
|
return
|
|
}
|
|
clearInterval(this.loadTimer)
|
|
loader(false)
|
|
}, 1000)
|
|
$A.eeuiAppSendMessage({
|
|
action: 'startMeeting',
|
|
meetingParams: {
|
|
name: this.addData.name,
|
|
token: data.token,
|
|
channel: data.channel,
|
|
uuid: data.uid,
|
|
appid: data.appid,
|
|
avatar: data.userimg,
|
|
username: data.nickname,
|
|
video: this.addData.tracks.includes("video"),
|
|
audio: this.addData.tracks.includes("audio"),
|
|
meetingid: data.meetingid,
|
|
sharelink: data.sharelink,
|
|
alert: {
|
|
title: this.$L('温馨提示'),
|
|
message: this.$L('确定要离开会议吗?'),
|
|
cancel: this.$L('继续'),
|
|
confirm: this.$L('退出'),
|
|
}
|
|
}
|
|
});
|
|
return
|
|
}
|
|
// 客户端且未获得邀请链接 获取会议链接之后使用子窗口打开会议
|
|
if ($A.Electron && !this.addData.sharekey) {
|
|
loader(true);
|
|
this.$store.dispatch("call", {
|
|
url: 'users/meeting/link',
|
|
data: {
|
|
meetingid: data.meetingid,
|
|
},
|
|
}).then(linkRes => {
|
|
// 使用子窗口打开会议
|
|
const config = {
|
|
title: this.addData.name,
|
|
titleFixed: true,
|
|
parent: null,
|
|
width: Math.min(window.screen.availWidth, 1440),
|
|
height: Math.min(window.screen.availHeight, 900),
|
|
}
|
|
const meetingLink = $A.leftDelete(linkRes.data, $A.mainUrl())
|
|
.replace(/^\//, '');
|
|
const meetingPath = $A.urlAddParams("/" + meetingLink, {
|
|
type: 'direct',
|
|
nickname: data.nickname,
|
|
avatar: data.userimg,
|
|
audio: this.addData.tracks.includes("audio") ? 1 : 0,
|
|
video: this.addData.tracks.includes("video") ? 1 : 0,
|
|
token: this.userToken,
|
|
});
|
|
this.$store.dispatch('openChildWindow', {
|
|
name: `meeting-window`,
|
|
path: meetingPath,
|
|
force: false,
|
|
config
|
|
});
|
|
// 关闭弹窗
|
|
this.addShow = false;
|
|
}).catch(({ msg }) => {
|
|
$A.modalError({
|
|
content: msg,
|
|
onOk: this.onBeforeClose
|
|
});
|
|
}).finally(_ => {
|
|
loader(false);
|
|
});
|
|
return;
|
|
}
|
|
// Web 加载会议组件
|
|
loader(true);
|
|
$A.loadScript('js/AgoraRTC_N-4.17.0.js').then(_ => {
|
|
this.join(data)
|
|
}).catch(_ => {
|
|
$A.modalError({
|
|
content: "会议组件加载失败!",
|
|
onOk: this.onBeforeClose
|
|
});
|
|
}).finally(_ => {
|
|
loader(false);
|
|
})
|
|
}).catch(({msg}) => {
|
|
$A.modalError({
|
|
content: msg,
|
|
onOk: this.onBeforeClose
|
|
});
|
|
}).finally(_ => {
|
|
loader(false);
|
|
});
|
|
},
|
|
|
|
onAudio() {
|
|
if (this.localUser.audioTrack) {
|
|
this.closeAudio();
|
|
} else {
|
|
this.openAudio();
|
|
}
|
|
},
|
|
|
|
onVideo() {
|
|
if (this.localUser.videoTrack) {
|
|
this.closeVideo();
|
|
} else {
|
|
this.openVideo();
|
|
}
|
|
},
|
|
|
|
onInvitation(type) {
|
|
if (type === 'open') {
|
|
if (this.addData.sharekey && !this.userId) {
|
|
this.linkCopy();
|
|
return;
|
|
}
|
|
this.invitationData = {
|
|
userids: [],
|
|
meetingid: this.addData.meetingid
|
|
};
|
|
this.invitationShow = true;
|
|
} else if (type === 'submit') {
|
|
if (this.invitationData.userids.length === 0) {
|
|
$A.modalWarning("请选择邀请成员");
|
|
return;
|
|
}
|
|
this.invitationLoad = true;
|
|
this.$store.dispatch("call", {
|
|
url: 'users/meeting/invitation',
|
|
data: this.invitationData
|
|
}).then(({data, msg}) => {
|
|
this.invitationShow = false;
|
|
this.$store.dispatch("saveDialogMsg", data.msgs);
|
|
this.$store.dispatch("updateDialogLastMsg", data.msgs);
|
|
$A.messageSuccess(msg);
|
|
}).catch(({msg}) => {
|
|
$A.modalError(msg);
|
|
}).finally(_ => {
|
|
this.invitationLoad = false;
|
|
});
|
|
}
|
|
},
|
|
|
|
onClose() {
|
|
return new Promise(resolve => {
|
|
$A.modalConfirm({
|
|
content: '确定要离开会议吗?',
|
|
cancelText: '继续',
|
|
okText: '退出',
|
|
onOk: async () => {
|
|
await this.onBeforeClose()
|
|
resolve()
|
|
}
|
|
});
|
|
})
|
|
},
|
|
|
|
async onBeforeClose() {
|
|
try {
|
|
await this.leave()
|
|
} catch (e) {
|
|
console.error(e)
|
|
}
|
|
if ($A.isSubElectron) {
|
|
this.$Electron.sendMessage('windowDestroy');
|
|
} else if (this.addData.sharekey) {
|
|
this.addShow = true;
|
|
}
|
|
},
|
|
|
|
linkCopy() {
|
|
this.linkCopyLoad = true;
|
|
this.$store.dispatch("call", {
|
|
url: 'users/meeting/link',
|
|
data: {
|
|
meetingid: this.addData.meetingid || this.invitationData.meetingid,
|
|
sharekey: this.addData.sharekey
|
|
},
|
|
}).then(({ data }) => {
|
|
this.copyText({
|
|
text: data,
|
|
success: '已复制会议邀请链接',
|
|
error: "复制失败"
|
|
});
|
|
this.invitationShow = false;
|
|
}).catch(({ msg }) => {
|
|
$A.modalError(msg);
|
|
}).finally(_ => {
|
|
this.linkCopyLoad = false;
|
|
});
|
|
},
|
|
|
|
async join(options) {
|
|
this.loadIng++;
|
|
try {
|
|
// 音频采集设备状态变化回调
|
|
AgoraRTC.onMicrophoneChanged = async (changedDevice) => {
|
|
// When plugging in a device, switch to a device that is newly plugged in.
|
|
if (changedDevice.state === "ACTIVE") {
|
|
this.localUser.audioTrack?.setDevice(changedDevice.device.deviceId);
|
|
// Switch to an existing device when the current device is unplugged.
|
|
} else if (changedDevice.device.label === this.localUser.audioTrack?.getTrackLabel()) {
|
|
const oldMicrophones = await AgoraRTC.getMicrophones();
|
|
oldMicrophones[0] && this.localUser.audioTrack?.setDevice(oldMicrophones[0].deviceId);
|
|
}
|
|
}
|
|
// 视频采集设备状态变化回调
|
|
AgoraRTC.onCameraChanged = async (changedDevice) => {
|
|
// When plugging in a device, switch to a device that is newly plugged in.
|
|
if (changedDevice.state === "ACTIVE") {
|
|
this.localUser.videoTrack?.setDevice(changedDevice.device.deviceId);
|
|
// Switch to an existing device when the current device is unplugged.
|
|
} else if (changedDevice.device.label === this.localUser.videoTrack?.getTrackLabel()) {
|
|
const oldCameras = await AgoraRTC.getCameras();
|
|
oldCameras[0] && this.localUser.videoTrack?.setDevice(oldCameras[0].deviceId);
|
|
}
|
|
}
|
|
// 音频或视频轨道自动播放失败回调
|
|
AgoraRTC.onAutoplayFailed = () => {
|
|
$A.messageWarning("点击屏幕开始会议");
|
|
}
|
|
// 设置日志级别
|
|
AgoraRTC.setLogLevel(window.systemInfo.debug === "yes" ? 0 : 3);
|
|
|
|
// 创建客户端
|
|
this.agoraClient = AgoraRTC.createClient({mode: "rtc", codec: "vp8"});
|
|
// 添加事件侦听器
|
|
this.agoraClient.on("user-joined", this.handleUserJoined);
|
|
this.agoraClient.on("user-left", this.handleUserLeft);
|
|
this.agoraClient.on("user-published", this.handleUserPublished);
|
|
this.agoraClient.on("user-unpublished", this.handleUserUnpublished);
|
|
// 加入频道、开启音视频
|
|
const localTracks = [];
|
|
try {
|
|
this.localUser.uid = await this.agoraClient.join(options.appid, options.channel, options.token, options.uid)
|
|
// 创建本地音频和视频轨道
|
|
await Promise.all(['audio', 'video'].map(async (trackType) => {
|
|
if (!this.addData.tracks.includes(trackType)) {
|
|
return;
|
|
}
|
|
const createTrack = trackType === 'audio' ? AgoraRTC.createMicrophoneAudioTrack : AgoraRTC.createCameraVideoTrack;
|
|
const trackKey = `${trackType}Track`;
|
|
try {
|
|
this.localUser[trackKey] = await createTrack();
|
|
localTracks.push(this.localUser[trackKey]);
|
|
} catch (e) {
|
|
if (e.code === 'DEVICE_NOT_FOUND') {
|
|
console.warn(`${trackType} device not found:`, e);
|
|
} else {
|
|
throw e;
|
|
}
|
|
}
|
|
}));
|
|
// 将本地视频曲目播放到本地浏览器、将本地音频和视频发布到频道。
|
|
if (localTracks.length > 0) {
|
|
await this.agoraClient.publish(localTracks);
|
|
}
|
|
// 显示会议组件
|
|
this.meetingShow = true;
|
|
} catch (error) {
|
|
console.error(error)
|
|
$A.modalError({
|
|
language: false,
|
|
content: getErrorMessage(error.code, getLanguage()) || this.$L("会议组件加载失败!"),
|
|
onOk: this.onBeforeClose
|
|
});
|
|
}
|
|
} catch (e) { }
|
|
this.addShow = false;
|
|
this.loadIng--;
|
|
},
|
|
|
|
async leave() {
|
|
this.loadIng++;
|
|
try {
|
|
// 删除本地用户和播放器视图。
|
|
['audioTrack', 'videoTrack'].some(trackName => {
|
|
this.localUser[trackName]?.stop();
|
|
this.localUser[trackName]?.close();
|
|
})
|
|
this.localUser = {
|
|
uid: null,
|
|
audioTrack: null,
|
|
videoTrack: null,
|
|
}
|
|
// 删除远程用户和播放器视图。
|
|
this.remoteUsers = [];
|
|
// 离开频道
|
|
await this.agoraClient.leave();
|
|
} catch (e) { }
|
|
this.meetingShow = false;
|
|
this.loadIng--;
|
|
},
|
|
|
|
async openAudio() {
|
|
if (this.audioLoad || this.localUser.audioTrack) return;
|
|
this.audioLoad = true;
|
|
try {
|
|
this.localUser.audioTrack = await AgoraRTC.createMicrophoneAudioTrack()
|
|
await this.agoraClient.publish([this.localUser.audioTrack]);
|
|
} catch (e) {
|
|
$A.modalError({
|
|
language: false,
|
|
content: getErrorMessage(e.code, getLanguage()) || this.$L("开启麦克风失败!"),
|
|
});
|
|
} finally {
|
|
this.audioLoad = false;
|
|
}
|
|
},
|
|
|
|
async closeAudio() {
|
|
if (this.audioLoad || !this.localUser.audioTrack) return;
|
|
this.audioLoad = true;
|
|
await this.agoraClient.unpublish([this.localUser.audioTrack]);
|
|
this.localUser.audioTrack.stop();
|
|
this.localUser.audioTrack.close();
|
|
this.localUser.audioTrack = null;
|
|
this.audioLoad = false;
|
|
},
|
|
|
|
async openVideo() {
|
|
if (this.videoLoad || this.localUser.videoTrack) return;
|
|
this.videoLoad = true;
|
|
try {
|
|
this.localUser.videoTrack = await AgoraRTC.createCameraVideoTrack()
|
|
await this.agoraClient.publish([this.localUser.videoTrack]);
|
|
} catch (e) {
|
|
$A.modalError({
|
|
language: false,
|
|
content: getErrorMessage(e.code, getLanguage()) || this.$L("开启摄像头失败!"),
|
|
});
|
|
} finally {
|
|
this.videoLoad = false;
|
|
}
|
|
},
|
|
|
|
async closeVideo() {
|
|
if (this.videoLoad || !this.localUser.videoTrack) return;
|
|
this.videoLoad = true;
|
|
await this.agoraClient.unpublish([this.localUser.videoTrack]);
|
|
this.localUser.videoTrack.stop();
|
|
this.localUser.videoTrack.close();
|
|
this.localUser.videoTrack = null;
|
|
this.videoLoad = false;
|
|
},
|
|
|
|
async handleUserJoined(user) {
|
|
if (user.uid == this.localUser.uid) {
|
|
return;
|
|
}
|
|
const index = this.remoteUsers.findIndex(item => item.uid == user.uid)
|
|
if (index > -1) {
|
|
this.remoteUsers.splice(index, 1, user)
|
|
} else {
|
|
this.remoteUsers.push(user)
|
|
}
|
|
},
|
|
|
|
async handleUserLeft(user) {
|
|
const index = this.remoteUsers.findIndex(item => item.uid == user.uid)
|
|
if (index > -1) {
|
|
this.remoteUsers.splice(index, 1)
|
|
}
|
|
},
|
|
|
|
async handleUserPublished(user, mediaType) {
|
|
const item = this.remoteUsers.find(item => item.uid == user.uid)
|
|
if (item) {
|
|
await this.agoraClient.subscribe(user, mediaType);
|
|
}
|
|
},
|
|
|
|
async handleUserUnpublished(user, mediaType) {
|
|
const item = this.remoteUsers.find(item => item.uid == user.uid)
|
|
if (item) {
|
|
await this.agoraClient.unsubscribe(user, mediaType);
|
|
}
|
|
},
|
|
}
|
|
}
|
|
</script>
|