2020-09-24 09:24:31 +00:00
|
|
|
import {ConnectionHandler, DisconnectReason} from "./ConnectionHandler";
|
|
|
|
import {Registry} from "./events";
|
2020-09-24 13:51:22 +00:00
|
|
|
import * as loader from "tc-loader";
|
|
|
|
import {Stage} from "tc-loader";
|
2019-04-04 19:47:52 +00:00
|
|
|
|
2020-04-09 13:10:14 +00:00
|
|
|
export let server_connections: ConnectionManager;
|
2020-09-24 13:51:22 +00:00
|
|
|
|
2020-11-07 12:16:07 +00:00
|
|
|
class ReplaceableContainer {
|
|
|
|
placeholder: HTMLDivElement;
|
|
|
|
container: HTMLDivElement;
|
|
|
|
|
|
|
|
constructor(container: HTMLDivElement, placeholder?: HTMLDivElement) {
|
|
|
|
this.container = container;
|
|
|
|
this.placeholder = placeholder || document.createElement("div");
|
|
|
|
}
|
|
|
|
|
|
|
|
replaceWith(target: HTMLDivElement | undefined) {
|
|
|
|
target = target || this.placeholder;
|
|
|
|
this.container.replaceWith(target);
|
|
|
|
this.container = target;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-06 16:22:29 +00:00
|
|
|
export interface ConnectionManagerEvents {
|
|
|
|
notify_handler_created: {
|
|
|
|
handlerId: string,
|
|
|
|
handler: ConnectionHandler
|
|
|
|
},
|
|
|
|
|
|
|
|
/* This will also trigger when a connection gets deleted. So if you're just interested to connect event handler to the active connection,
|
|
|
|
unregister them from the old handler and register them for the new handler every time */
|
|
|
|
notify_active_handler_changed: {
|
|
|
|
oldHandler: ConnectionHandler | undefined,
|
|
|
|
newHandler: ConnectionHandler | undefined,
|
|
|
|
|
|
|
|
oldHandlerId: string | undefined,
|
|
|
|
newHandlerId: string | undefined
|
|
|
|
},
|
|
|
|
|
|
|
|
/* Will never fire on an active connection handler! */
|
|
|
|
notify_handler_deleted: {
|
|
|
|
handlerId: string,
|
|
|
|
handler: ConnectionHandler
|
|
|
|
},
|
|
|
|
|
|
|
|
notify_handler_order_changed: { }
|
|
|
|
}
|
|
|
|
|
2020-04-09 13:10:14 +00:00
|
|
|
export class ConnectionManager {
|
2021-01-06 16:22:29 +00:00
|
|
|
private readonly events_: Registry<ConnectionManagerEvents>;
|
|
|
|
private connectionHandlers: ConnectionHandler[] = [];
|
|
|
|
private activeConnectionHandler: ConnectionHandler | undefined;
|
2019-04-04 19:47:52 +00:00
|
|
|
|
2020-11-07 12:16:07 +00:00
|
|
|
private containerChannelVideo: ReplaceableContainer;
|
2020-12-09 19:44:33 +00:00
|
|
|
|
2020-09-25 16:29:42 +00:00
|
|
|
constructor() {
|
2021-01-06 16:22:29 +00:00
|
|
|
this.events_ = new Registry<ConnectionManagerEvents>();
|
|
|
|
this.events_.enableDebug("connection-manager");
|
2020-12-09 19:44:33 +00:00
|
|
|
|
2021-01-06 16:22:29 +00:00
|
|
|
/* FIXME! */
|
2020-11-07 12:16:07 +00:00
|
|
|
this.containerChannelVideo = new ReplaceableContainer(document.getElementById("channel-video") as HTMLDivElement);
|
2020-04-09 13:10:14 +00:00
|
|
|
this.set_active_connection(undefined);
|
|
|
|
}
|
|
|
|
|
|
|
|
events() : Registry<ConnectionManagerEvents> {
|
2021-01-06 16:22:29 +00:00
|
|
|
return this.events_;
|
2019-04-04 19:47:52 +00:00
|
|
|
}
|
|
|
|
|
2020-04-09 13:10:14 +00:00
|
|
|
spawn_server_connection() : ConnectionHandler {
|
2019-04-04 19:47:52 +00:00
|
|
|
const handler = new ConnectionHandler();
|
2021-01-06 16:22:29 +00:00
|
|
|
handler.initialize_client_state(this.activeConnectionHandler);
|
|
|
|
this.connectionHandlers.push(handler);
|
2020-04-09 13:10:14 +00:00
|
|
|
|
2021-01-06 16:22:29 +00:00
|
|
|
this.events_.fire("notify_handler_created", { handler: handler, handlerId: handler.handlerId });
|
2019-04-04 19:47:52 +00:00
|
|
|
return handler;
|
|
|
|
}
|
|
|
|
|
2020-04-09 13:10:14 +00:00
|
|
|
destroy_server_connection(handler: ConnectionHandler) {
|
2021-01-06 16:22:29 +00:00
|
|
|
if(this.connectionHandlers.length <= 1) {
|
2020-04-09 13:10:14 +00:00
|
|
|
throw "cannot deleted the last connection handler";
|
2021-01-06 16:22:29 +00:00
|
|
|
}
|
2020-04-09 13:10:14 +00:00
|
|
|
|
2021-01-06 16:22:29 +00:00
|
|
|
if(!this.connectionHandlers.remove(handler)) {
|
2020-04-09 13:10:14 +00:00
|
|
|
throw "unknown connection handler";
|
2021-01-06 16:22:29 +00:00
|
|
|
}
|
2020-04-09 13:10:14 +00:00
|
|
|
|
2019-04-04 19:47:52 +00:00
|
|
|
if(handler.serverConnection) {
|
|
|
|
const connected = handler.connected;
|
|
|
|
handler.serverConnection.disconnect("handler destroyed");
|
|
|
|
handler.handleDisconnect(DisconnectReason.HANDLER_DESTROYED, connected);
|
|
|
|
}
|
|
|
|
|
2021-01-06 16:22:29 +00:00
|
|
|
if(handler === this.activeConnectionHandler) {
|
|
|
|
this.set_active_connection_(this.connectionHandlers[0]);
|
|
|
|
}
|
|
|
|
this.events_.fire("notify_handler_deleted", { handler: handler, handlerId: handler.handlerId });
|
2019-08-21 08:00:01 +00:00
|
|
|
|
|
|
|
/* destroy all elements */
|
|
|
|
handler.destroy();
|
2019-04-04 19:47:52 +00:00
|
|
|
}
|
|
|
|
|
2020-04-09 13:10:14 +00:00
|
|
|
set_active_connection(handler: ConnectionHandler) {
|
2021-01-06 16:22:29 +00:00
|
|
|
if(handler && this.connectionHandlers.indexOf(handler) == -1) {
|
2019-08-21 08:00:01 +00:00
|
|
|
throw "Handler hasn't been registered or is already obsolete!";
|
2021-01-06 16:22:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if(handler === this.activeConnectionHandler) {
|
2020-04-09 13:10:14 +00:00
|
|
|
return;
|
2021-01-06 16:22:29 +00:00
|
|
|
}
|
|
|
|
|
2020-04-09 13:10:14 +00:00
|
|
|
this.set_active_connection_(handler);
|
|
|
|
}
|
2019-04-04 19:47:52 +00:00
|
|
|
|
2020-09-25 18:54:26 +00:00
|
|
|
swapHandlerOrder(handlerA: ConnectionHandler, handlerB: ConnectionHandler) {
|
2021-01-06 16:22:29 +00:00
|
|
|
const indexA = this.connectionHandlers.findIndex(handler => handlerA === handler);
|
|
|
|
const indexB = this.connectionHandlers.findIndex(handler => handlerB === handler);
|
2020-09-25 18:54:26 +00:00
|
|
|
|
|
|
|
if(indexA === -1 || indexB === -1 || indexA === indexB) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-01-06 16:22:29 +00:00
|
|
|
let temp = this.connectionHandlers[indexA];
|
|
|
|
this.connectionHandlers[indexA] = this.connectionHandlers[indexB];
|
|
|
|
this.connectionHandlers[indexB] = temp;
|
2020-09-25 18:54:26 +00:00
|
|
|
this.events().fire("notify_handler_order_changed");
|
|
|
|
}
|
|
|
|
|
2020-04-09 13:10:14 +00:00
|
|
|
private set_active_connection_(handler: ConnectionHandler) {
|
2021-01-05 17:13:57 +00:00
|
|
|
/*
|
2020-11-07 12:16:07 +00:00
|
|
|
this.containerChannelVideo.replaceWith(handler?.video_frame.getContainer());
|
2021-01-05 17:13:57 +00:00
|
|
|
*/
|
2020-12-09 19:44:33 +00:00
|
|
|
|
2021-01-06 16:22:29 +00:00
|
|
|
const oldHandler = this.activeConnectionHandler;
|
|
|
|
this.activeConnectionHandler = handler;
|
|
|
|
this.events_.fire("notify_active_handler_changed", {
|
|
|
|
oldHandler: oldHandler,
|
2020-09-24 13:51:22 +00:00
|
|
|
newHandler: handler,
|
|
|
|
|
2021-01-06 16:22:29 +00:00
|
|
|
oldHandlerId: oldHandler?.handlerId,
|
2020-09-24 13:51:22 +00:00
|
|
|
newHandlerId: handler?.handlerId
|
2020-04-09 13:10:14 +00:00
|
|
|
});
|
2021-01-06 16:22:29 +00:00
|
|
|
oldHandler?.events().fire("notify_visibility_changed", { visible: false });
|
2020-06-11 09:17:56 +00:00
|
|
|
handler?.events().fire("notify_visibility_changed", { visible: true });
|
2019-04-04 19:47:52 +00:00
|
|
|
}
|
2020-07-20 17:08:13 +00:00
|
|
|
|
|
|
|
findConnection(handlerId: string) : ConnectionHandler | undefined {
|
2021-01-06 16:22:29 +00:00
|
|
|
return this.connectionHandlers.find(e => e.handlerId === handlerId);
|
2020-07-20 17:08:13 +00:00
|
|
|
}
|
2019-04-04 19:47:52 +00:00
|
|
|
|
2020-04-09 13:10:14 +00:00
|
|
|
active_connection() : ConnectionHandler | undefined {
|
2021-01-06 16:22:29 +00:00
|
|
|
return this.activeConnectionHandler;
|
2019-04-04 19:47:52 +00:00
|
|
|
}
|
|
|
|
|
2020-04-09 13:10:14 +00:00
|
|
|
all_connections() : ConnectionHandler[] {
|
2021-01-06 16:22:29 +00:00
|
|
|
return this.connectionHandlers;
|
2020-12-29 15:53:04 +00:00
|
|
|
}
|
2020-04-09 13:10:14 +00:00
|
|
|
}
|
|
|
|
|
2020-09-24 13:51:22 +00:00
|
|
|
loader.register_task(Stage.JAVASCRIPT_INITIALIZING, {
|
|
|
|
name: "server manager init",
|
|
|
|
function: async () => {
|
2020-09-25 16:29:42 +00:00
|
|
|
server_connections = new ConnectionManager();
|
2020-09-24 13:51:22 +00:00
|
|
|
},
|
|
|
|
priority: 80
|
|
|
|
});
|