openvidu/openvidu-components-angular/projects/openvidu-angular/src/lib/components/session/session.component.ts

320 lines
11 KiB
TypeScript
Raw Normal View History

import {
ChangeDetectionStrategy,
Component,
ContentChild,
ElementRef,
EventEmitter,
HostListener,
Input,
OnInit,
Output,
TemplateRef,
ViewChild
} from '@angular/core';
import {
Subscriber,
Session,
StreamEvent,
StreamPropertyChangedEvent,
SessionDisconnectedEvent,
ConnectionEvent,
RecordingEvent
} from 'openvidu-browser';
import { VideoType } from '../../models/video-type.model';
import { ILogger } from '../../models/logger.model';
import { ChatService } from '../../services/chat/chat.service';
import { LoggerService } from '../../services/logger/logger.service';
import { OpenViduService } from '../../services/openvidu/openvidu.service';
import { TokenService } from '../../services/token/token.service';
import { ActionService } from '../../services/action/action.service';
import { Signal } from '../../models/signal.model';
import { ParticipantService } from '../../services/participant/participant.service';
import { MatSidenav } from '@angular/material/sidenav';
import { SidenavMode } from '../../models/layout.model';
import { LayoutService } from '../../services/layout/layout.service';
import { Subscription, skip } from 'rxjs';
import { PanelType } from '../../models/panel.model';
import { PanelService } from '../../services/panel/panel.service';
import { RecordingService } from '../../services/recording/recording.service';
import { TranslateService } from '../../services/translate/translate.service';
/**
* @internal
*/
@Component({
selector: 'ov-session',
templateUrl: './session.component.html',
styleUrls: ['./session.component.css'],
changeDetection: ChangeDetectionStrategy.OnPush
})
export class SessionComponent implements OnInit {
@ContentChild('toolbar', { read: TemplateRef }) toolbarTemplate: TemplateRef<any>;
@ContentChild('panel', { read: TemplateRef }) panelTemplate: TemplateRef<any>;
@ContentChild('layout', { read: TemplateRef }) layoutTemplate: TemplateRef<any>;
@Input() usedInPrejoinPage = false;
@Output() onSessionCreated = new EventEmitter<any>();
session: Session;
sessionScreen: Session;
sideMenu: MatSidenav;
sidenavMode: SidenavMode = SidenavMode.SIDE;
protected readonly SIDENAV_WIDTH_LIMIT_MODE = 790;
protected menuSubscription: Subscription;
protected layoutWidthSubscription: Subscription;
protected updateLayoutInterval: NodeJS.Timer;
protected log: ILogger;
constructor(
protected actionService: ActionService,
protected openviduService: OpenViduService,
protected participantService: ParticipantService,
protected loggerSrv: LoggerService,
protected chatService: ChatService,
protected tokenService: TokenService,
protected layoutService: LayoutService,
protected panelService: PanelService,
private recordingService: RecordingService,
private translateService: TranslateService
) {
this.log = this.loggerSrv.get('SessionComponent');
}
@HostListener('window:beforeunload')
beforeunloadHandler() {
this.leaveSession();
}
@HostListener('window:resize')
sizeChange() {
this.layoutService.update();
}
@ViewChild('sidenav')
set sidenavMenu(menu: MatSidenav) {
setTimeout(() => {
if (menu) {
this.sideMenu = menu;
this.subscribeToTogglingMenu();
}
}, 0);
}
@ViewChild('videoContainer', { static: false, read: ElementRef })
set videoContainer(container: ElementRef) {
setTimeout(() => {
if (container && !this.toolbarTemplate) {
container.nativeElement.style.height = '100%';
container.nativeElement.style.minHeight = '100%';
this.layoutService.update();
}
}, 0);
}
async ngOnInit() {
if (!this.usedInPrejoinPage) {
this.session = this.openviduService.getWebcamSession();
this.sessionScreen = this.openviduService.getScreenSession();
this.subscribeToConnectionCreatedAndDestroyed();
this.subscribeToStreamCreated();
this.subscribeToStreamDestroyed();
this.subscribeToStreamPropertyChange();
this.subscribeToNicknameChanged();
this.chatService.subscribeToChat();
this.subscribeToReconnection();
// if(RecordingEnabled){
this.subscribeToRecordingEvents();
// }
this.onSessionCreated.emit(this.session);
await this.connectToSession();
// Workaround, firefox does not have audio when publisher join with muted camera
// if (this.platformService.isFirefox() && !this.participantService.hasCameraVideoActive()) {
// this.openviduService.publishVideo(this.participantService.getMyCameraPublisher(), true);
// this.openviduService.publishVideo(this.participantService.getMyCameraPublisher(), false);
// }
}
}
ngOnDestroy() {
// Reconnecting session is received in Firefox
// To avoid 'Connection lost' message uses session.off()
this.session?.off('reconnecting');
this.participantService.clear();
this.session = null;
this.sessionScreen = null;
if (this.menuSubscription) this.menuSubscription.unsubscribe();
if (this.layoutWidthSubscription) this.layoutWidthSubscription.unsubscribe();
}
leaveSession() {
this.log.d('Leaving session...');
this.openviduService.disconnect();
}
protected subscribeToTogglingMenu() {
this.sideMenu.openedChange.subscribe(() => {
if (this.updateLayoutInterval) {
clearInterval(this.updateLayoutInterval);
}
this.layoutService.update();
});
this.sideMenu.openedStart.subscribe(() => {
this.updateLayoutInterval = setInterval(() => this.layoutService.update(), 50);
});
this.sideMenu.closedStart.subscribe(() => {
this.updateLayoutInterval = setInterval(() => this.layoutService.update(), 50);
});
this.menuSubscription = this.panelService.panelOpenedObs
.pipe(skip(1))
.subscribe((ev: { opened: boolean; type?: PanelType | string }) => {
if (this.sideMenu) {
ev.opened ? this.sideMenu.open() : this.sideMenu.close();
}
});
}
protected subscribeToLayoutWidth() {
this.layoutWidthSubscription = this.layoutService.layoutWidthObs.subscribe((width) => {
this.sidenavMode = width <= this.SIDENAV_WIDTH_LIMIT_MODE ? SidenavMode.OVER : SidenavMode.SIDE;
});
}
private async connectToSession(): Promise<void> {
try {
if (this.participantService.haveICameraAndScreenActive()) {
await this.openviduService.connectSession(this.openviduService.getWebcamSession(), this.tokenService.getWebcamToken());
await this.openviduService.connectSession(this.openviduService.getScreenSession(), this.tokenService.getScreenToken());
await this.openviduService.publish(this.participantService.getMyCameraPublisher());
await this.openviduService.publish(this.participantService.getMyScreenPublisher());
} else if (this.participantService.isOnlyMyScreenActive()) {
await this.openviduService.connectSession(this.openviduService.getScreenSession(), this.tokenService.getScreenToken());
await this.openviduService.publish(this.participantService.getMyScreenPublisher());
} else {
await this.openviduService.connectSession(this.openviduService.getWebcamSession(), this.tokenService.getWebcamToken());
await this.openviduService.publish(this.participantService.getMyCameraPublisher());
}
} catch (error) {
// this._error.emit({ error: error.error, messgae: error.message, code: error.code, status: error.status });
this.log.e('There was an error connecting to the session:', error.code, error.message);
this.actionService.openDialog(this.translateService.translate('ERRORS.SESSION'), error?.error || error?.message || error);
}
}
private subscribeToConnectionCreatedAndDestroyed() {
this.session.on('connectionCreated', (event: ConnectionEvent) => {
const connectionId = event.connection?.connectionId;
const nickname: string = this.participantService.getNicknameFromConnectionData(event.connection.data);
const isRemoteConnection: boolean = !this.openviduService.isMyOwnConnection(connectionId);
const isCameraConnection: boolean = !nickname?.includes(`_${VideoType.SCREEN}`);
const data = event.connection?.data;
if (isRemoteConnection && isCameraConnection) {
// Adding participant when connection is created and it's not screen
this.participantService.addRemoteConnection(connectionId, data, null);
//Sending nicnkanme signal to new participants
if (this.openviduService.needSendNicknameSignal()) {
const data = { clientData: this.participantService.getMyNickname() };
this.openviduService.sendSignal(Signal.NICKNAME_CHANGED, [event.connection], data);
}
}
});
this.session.on('connectionDestroyed', (event: ConnectionEvent) => {
const nickname: string = this.participantService.getNicknameFromConnectionData(event.connection.data);
const isRemoteConnection: boolean = !this.openviduService.isMyOwnConnection(event.connection.connectionId);
const isCameraConnection: boolean = !nickname?.includes(`_${VideoType.SCREEN}`);
// Deleting participant when connection is destroyed
if (isRemoteConnection && isCameraConnection) {
this.participantService.removeConnectionByConnectionId(event.connection.connectionId);
}
});
}
private subscribeToStreamCreated() {
this.session.on('streamCreated', (event: StreamEvent) => {
const connectionId = event.stream?.connection?.connectionId;
const data = event.stream?.connection?.data;
const isRemoteConnection: boolean = !this.openviduService.isMyOwnConnection(connectionId);
if (isRemoteConnection) {
const subscriber: Subscriber = this.session.subscribe(event.stream, undefined);
this.participantService.addRemoteConnection(connectionId, data, subscriber);
// this.oVSessionService.sendNicknameSignal(event.stream.connection);
}
});
}
private subscribeToStreamDestroyed() {
this.session.on('streamDestroyed', (event: StreamEvent) => {
const connectionId = event.stream.connection.connectionId;
this.participantService.removeConnectionByConnectionId(connectionId);
});
}
private subscribeToStreamPropertyChange() {
this.session.on('streamPropertyChanged', (event: StreamPropertyChangedEvent) => {
const connectionId = event.stream.connection.connectionId;
const isRemoteConnection: boolean = !this.openviduService.isMyOwnConnection(connectionId);
if (isRemoteConnection) {
this.participantService.updateRemoteParticipants();
}
});
}
private subscribeToNicknameChanged() {
this.session.on(`signal:${Signal.NICKNAME_CHANGED}`, (event: any) => {
const connectionId = event.from.connectionId;
const isRemoteConnection: boolean = !this.openviduService.isMyOwnConnection(connectionId);
if (isRemoteConnection) {
const nickname = this.participantService.getNicknameFromConnectionData(event.data);
this.participantService.setRemoteNickname(connectionId, nickname);
}
});
}
private subscribeToReconnection() {
this.session.on('reconnecting', () => {
this.log.w('Connection lost: Reconnecting');
this.actionService.openDialog(
this.translateService.translate('ERRRORS.CONNECTION'),
this.translateService.translate('ERRRORS.RECONNECT'),
false
);
});
this.session.on('reconnected', () => {
this.log.w('Connection lost: Reconnected');
this.actionService.closeDialog();
});
this.session.on('sessionDisconnected', (event: SessionDisconnectedEvent) => {
if (event.reason === 'networkDisconnect') {
this.actionService.closeDialog();
this.leaveSession();
}
});
}
private subscribeToRecordingEvents() {
this.session.on('recordingStarted', (event: RecordingEvent) => {
this.recordingService.startRecording(event);
});
this.session.on('recordingStopped', (event: RecordingEvent) => {
this.recordingService.stopRecording(event);
});
}
}