mirror of
https://github.com/DerTyp7/teamspeak-obs-overlay.git
synced 2025-10-29 12:52:09 +01:00
133 lines
4.2 KiB
TypeScript
133 lines
4.2 KiB
TypeScript
import { IAuthSenderPayload, IChannel, IClient, IConnection, ITS5ConnectionHandler, ITS5DataHandler, ITS5MessageHandler } from "@interfaces/teamspeak";
|
|
import { TS5DataHandler } from "./dataHandler";
|
|
import { TS5MessageHandler } from "./messageHandler";
|
|
import Logger from "@/utils/logger";
|
|
|
|
|
|
// Establish connection to TS5 client
|
|
// Main class
|
|
export class TS5ConnectionHandler implements ITS5ConnectionHandler {
|
|
ws: WebSocket; // Websocket connection to TS5 client
|
|
authenticated = false; // Is the connection authenticated?
|
|
remoteAppPort: number; // Port of TS5 client
|
|
dataHandler: ITS5DataHandler; // Handles data/lists and states
|
|
messageHandler: ITS5MessageHandler; // Handles messages received from TS5 client
|
|
|
|
constructor(
|
|
// Port of TS5 client
|
|
remoteAppPort: number,
|
|
|
|
// State setters for dataHandler
|
|
setConnections: React.Dispatch<React.SetStateAction<IConnection[]>>,
|
|
setChannels: React.Dispatch<React.SetStateAction<IChannel[]>>,
|
|
setClients: React.Dispatch<React.SetStateAction<IClient[]>>,
|
|
setActiveConnectionStateId: React.Dispatch<React.SetStateAction<number>>,
|
|
) {
|
|
|
|
|
|
// Create websocket connection to TS5 client
|
|
this.remoteAppPort = remoteAppPort;
|
|
this.ws = new WebSocket(`ws://localhost:${this.remoteAppPort}`);
|
|
|
|
// Create dataHandler and messageHandler
|
|
this.dataHandler = new TS5DataHandler(setConnections, setChannels, setClients);
|
|
this.messageHandler = new TS5MessageHandler(this.ws, this.dataHandler, setActiveConnectionStateId);
|
|
}
|
|
|
|
reconnect() {
|
|
Logger.log("Reconnecting...")
|
|
this.ws.close();
|
|
|
|
this.ws = new WebSocket(`ws://localhost:${this.remoteAppPort}`);
|
|
|
|
this.dataHandler.clearAll();
|
|
this.authenticated = false;
|
|
this.connect();
|
|
}
|
|
|
|
// Connect to TS5 client
|
|
connect() {
|
|
Logger.log('Connecting to TS5 client...');
|
|
|
|
// Create authentication payload
|
|
const initalPayload: IAuthSenderPayload = {
|
|
type: "auth",
|
|
payload: {
|
|
identifier: "de.tealfire.obs",
|
|
version: "1.1.3",
|
|
name: "TS5 OBS Overlay",
|
|
description: "A OBS overlay for TS5 by DerTyp876",
|
|
content: {
|
|
apiKey: localStorage.getItem("apiKey") ?? "",
|
|
},
|
|
},
|
|
};
|
|
|
|
this.ws.onopen = () => {
|
|
// Send authentication payload to TS5 client
|
|
this.ws.send(JSON.stringify(initalPayload));
|
|
Logger.wsSent(initalPayload);
|
|
};
|
|
|
|
this.ws.onclose = (event) => {
|
|
Logger.log("WebSocket connection closed", event);
|
|
|
|
// If the connection was closed before authentication, remove the API key from local storage
|
|
// OBS weirdly caches the localstorage and is very stubborn about clearing it (even when clicken "Clear Cache")
|
|
if (!this.authenticated) {
|
|
Logger.log("WebSocket connection closed before authentication");
|
|
localStorage.removeItem("apiKey");
|
|
}
|
|
|
|
setTimeout(() => {
|
|
this.reconnect();
|
|
}, 2000);
|
|
};
|
|
|
|
// Handle messages received from TS5 client
|
|
// See TS5MessageHandler class
|
|
this.ws.onmessage = (event) => {
|
|
const data = JSON.parse(event.data);
|
|
|
|
Logger.wsReceived(data)
|
|
|
|
switch (data.type) {
|
|
case "auth":
|
|
this.messageHandler.handleAuthMessage(data);
|
|
this.authenticated = true;
|
|
break;
|
|
case "clientMoved":
|
|
this.messageHandler.handleClientMovedMessage(data);
|
|
break;
|
|
case "clientPropertiesUpdated":
|
|
this.messageHandler.handleClientPropertiesUpdatedMessage(data);
|
|
break;
|
|
case "talkStatusChanged":
|
|
this.messageHandler.handleTalkStatusChangedMessage(data);
|
|
break;
|
|
case "serverPropertiesUpdated":
|
|
this.messageHandler.handleServerPropertiesUpdatedMessage(data);
|
|
break;
|
|
case "connectStatusChanged":
|
|
this.messageHandler.handleConnectStatusChangedMessage(data);
|
|
break;
|
|
case "clientSelfPropertyUpdated":
|
|
this.messageHandler.handleClientSelfPropertyUpdatedMessage(data);
|
|
break;
|
|
case "channels":
|
|
this.messageHandler.handleChannelsMessage(data);
|
|
break;
|
|
default:
|
|
Logger.log(`No handler for event type: ${data.type}`);
|
|
break;
|
|
}
|
|
};
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|