partey_workadventure/front/src/WebRtc/MediaManager.ts

219 lines
7.7 KiB
TypeScript
Raw Normal View History

import { HtmlUtils } from "./HtmlUtils";
import type { UserInputManager } from "../Phaser/UserInput/UserInputManager";
import { localStreamStore } from "../Stores/MediaStore";
import { screenSharingLocalStreamStore } from "../Stores/ScreenSharingStore";
import { helpCameraSettingsVisibleStore } from "../Stores/HelpCameraSettingsStore";
export type StartScreenSharingCallback = (media: MediaStream) => void;
export type StopScreenSharingCallback = (media: MediaStream) => void;
import { cowebsiteCloseButtonId } from "./CoWebsiteManager";
import { gameOverlayVisibilityStore } from "../Stores/GameOverlayStoreVisibility";
2021-08-03 11:13:08 +02:00
import { layoutManagerActionStore, layoutManagerVisibilityStore } from "../Stores/LayoutManagerStore";
import { get } from "svelte/store";
2020-06-23 12:24:36 +02:00
export class MediaManager {
startScreenSharingCallBacks: Set<StartScreenSharingCallback> = new Set<StartScreenSharingCallback>();
stopScreenSharingCallBacks: Set<StopScreenSharingCallback> = new Set<StopScreenSharingCallback>();
private triggerCloseJistiFrame: Map<String, Function> = new Map<String, Function>();
2020-11-17 18:03:44 +01:00
private userInputManager?: UserInputManager;
constructor() {
localStreamStore.subscribe((result) => {
if (result.type === "error") {
console.error(result.error);
2021-08-03 11:13:08 +02:00
layoutManagerActionStore.addAction({
uuid: "cameraAccessDenied",
2021-08-03 11:13:08 +02:00
type: "warning",
message: "Camera access denied. Click here and check your browser permissions.",
callback: () => {
helpCameraSettingsVisibleStore.set(true);
},
2021-08-03 11:13:08 +02:00
userInputManager: this.userInputManager,
});
//remove it after 10 sec
setTimeout(() => {
layoutManagerActionStore.removeAction("cameraAccessDenied");
}, 10000);
return;
}
});
screenSharingLocalStreamStore.subscribe((result) => {
if (result.type === "error") {
console.error(result.error);
2021-08-03 11:13:08 +02:00
layoutManagerActionStore.addAction({
uuid: "screenSharingAccessDenied",
2021-08-03 11:13:08 +02:00
type: "warning",
message: "Screen sharing denied. Click here and check your browser permissions.",
callback: () => {
helpCameraSettingsVisibleStore.set(true);
},
2021-08-03 11:13:08 +02:00
userInputManager: this.userInputManager,
});
//remove it after 10 sec
setTimeout(() => {
layoutManagerActionStore.removeAction("screenSharingAccessDenied");
}, 10000);
return;
}
});
2020-04-26 20:55:20 +02:00
}
public showGameOverlay(): void {
const gameOverlay = HtmlUtils.getElementByIdOrFail("game-overlay");
gameOverlay.classList.add("active");
2020-11-17 18:03:44 +01:00
2021-06-03 20:05:39 +02:00
const buttonCloseFrame = HtmlUtils.getElementByIdOrFail(cowebsiteCloseButtonId);
2020-11-17 18:03:44 +01:00
const functionTrigger = () => {
this.triggerCloseJitsiFrameButton();
};
buttonCloseFrame.removeEventListener("click", () => {
2021-06-03 20:05:39 +02:00
buttonCloseFrame.blur();
functionTrigger();
});
gameOverlayVisibilityStore.showGameOverlay();
}
public hideGameOverlay(): void {
const gameOverlay = HtmlUtils.getElementByIdOrFail("game-overlay");
gameOverlay.classList.remove("active");
2020-11-17 18:03:44 +01:00
2021-06-03 20:05:39 +02:00
const buttonCloseFrame = HtmlUtils.getElementByIdOrFail(cowebsiteCloseButtonId);
2020-11-17 18:03:44 +01:00
const functionTrigger = () => {
this.triggerCloseJitsiFrameButton();
};
buttonCloseFrame.addEventListener("click", () => {
2021-06-03 20:05:39 +02:00
buttonCloseFrame.blur();
functionTrigger();
});
gameOverlayVisibilityStore.hideGameOverlay();
}
private getScreenSharingId(userId: string): string {
return `screen-sharing-${userId}`;
}
disabledMicrophoneByUserId(userId: number) {
2020-06-09 23:13:26 +02:00
const element = document.getElementById(`microphone-${userId}`);
if (!element) {
return;
}
element.classList.add("active"); //todo: why does a method 'disable' add a class 'active'?
}
enabledMicrophoneByUserId(userId: number) {
2020-06-09 23:13:26 +02:00
const element = document.getElementById(`microphone-${userId}`);
if (!element) {
return;
}
element.classList.remove("active"); //todo: why does a method 'enable' remove a class 'active'?
}
disabledVideoByUserId(userId: number) {
let element = document.getElementById(`${userId}`);
2020-05-14 20:54:34 +02:00
if (element) {
element.style.opacity = "0";
}
element = document.getElementById(`name-${userId}`);
if (element) {
element.style.display = "block";
}
}
enabledVideoByUserId(userId: number) {
let element = document.getElementById(`${userId}`);
if (element) {
2020-05-14 20:54:34 +02:00
element.style.opacity = "1";
}
element = document.getElementById(`name-${userId}`);
if (element) {
element.style.display = "none";
}
}
toggleBlockLogo(userId: number, show: boolean): void {
const blockLogoElement = HtmlUtils.getElementByIdOrFail<HTMLImageElement>("blocking-" + userId);
show ? blockLogoElement.classList.add("active") : blockLogoElement.classList.remove("active");
}
isError(userId: string): void {
console.info("isError", `div-${userId}`);
2020-06-09 23:13:26 +02:00
const element = document.getElementById(`div-${userId}`);
if (!element) {
return;
}
const errorDiv = element.getElementsByClassName("rtc-error").item(0) as HTMLDivElement | null;
if (errorDiv === null) {
return;
}
errorDiv.style.display = "block";
}
isErrorScreenSharing(userId: string): void {
this.isError(this.getScreenSharingId(userId));
}
private getSpinner(userId: string): HTMLDivElement | null {
2020-06-09 23:13:26 +02:00
const element = document.getElementById(`div-${userId}`);
if (!element) {
return null;
}
const connectingSpinnerDiv = element
.getElementsByClassName("connecting-spinner")
.item(0) as HTMLDivElement | null;
2021-06-17 18:48:56 +02:00
return connectingSpinnerDiv;
}
2020-06-03 11:55:31 +02:00
public addTriggerCloseJitsiFrameButton(id: String, Function: Function) {
2020-11-17 18:03:44 +01:00
this.triggerCloseJistiFrame.set(id, Function);
}
public removeTriggerCloseJitsiFrameButton(id: String) {
2020-11-17 18:03:44 +01:00
this.triggerCloseJistiFrame.delete(id);
}
private triggerCloseJitsiFrameButton(): void {
for (const callback of this.triggerCloseJistiFrame.values()) {
callback();
}
}
public setUserInputManager(userInputManager: UserInputManager) {
this.userInputManager = userInputManager;
2020-11-10 12:38:32 +01:00
}
public hasNotification(): boolean {
return Notification.permission === "granted";
}
public requestNotification() {
if (window.Notification && Notification.permission !== "granted") {
return Notification.requestPermission();
} else {
return Promise.reject();
}
}
public createNotification(userName: string) {
if (document.hasFocus()) {
return;
}
if (this.hasNotification()) {
const title = `${userName} wants to discuss with you`;
const options = {
icon: "/resources/logos/logo-WA-min.png",
image: "/resources/logos/logo-WA-min.png",
badge: "/resources/logos/logo-WA-min.png",
};
new Notification(title, options);
}
}
2020-06-03 11:55:31 +02:00
}
export const mediaManager = new MediaManager();