TeaWeb/shared/js/settings.ts

337 lines
11 KiB
TypeScript
Raw Normal View History

2018-04-16 18:38:35 +00:00
if(typeof(customElements) !== "undefined") {
2019-01-19 12:03:51 +00:00
try {
class X_Properties extends HTMLElement {}
class X_Property extends HTMLElement {}
2018-03-24 22:38:01 +00:00
2019-01-19 12:03:51 +00:00
customElements.define('x-properties', X_Properties, { extends: 'div' });
customElements.define('x-property', X_Property, { extends: 'div' });
} catch(error) {
console.warn("failed to define costum elements");
}
2018-04-16 18:38:35 +00:00
}
2018-03-24 22:38:01 +00:00
2019-03-17 11:15:39 +00:00
/* T = value type */
interface SettingsKey<T> {
key: string;
fallback_keys?: string | string[];
fallback_imports?: {[key: string]:(value: string) => T};
description?: string;
2019-04-04 19:47:52 +00:00
default_value?: T;
2019-03-17 11:15:39 +00:00
}
2019-04-04 19:47:52 +00:00
class SettingsBase {
protected static readonly UPDATE_DIRECT: boolean = true;
2018-02-27 16:20:49 +00:00
2019-03-17 11:15:39 +00:00
protected static transformStO?<T>(input?: string, _default?: T, default_type?: string) : T {
default_type = default_type || typeof _default;
2018-04-19 17:46:47 +00:00
if (typeof input === "undefined") return _default;
2019-03-17 11:15:39 +00:00
if (default_type === "string") return input as any;
else if (default_type === "number") return parseInt(input) as any;
else if (default_type === "boolean") return (input == "1" || input == "true") as any;
else if (default_type === "undefined") return input as any;
2018-04-19 17:46:47 +00:00
return JSON.parse(input) as any;
}
2018-02-27 16:20:49 +00:00
2018-04-19 17:46:47 +00:00
protected static transformOtS?<T>(input: T) : string {
if (typeof input === "string") return input as string;
else if (typeof input === "number") return input.toString();
else if (typeof input === "boolean") return input ? "1" : "0";
else if (typeof input === "undefined") return undefined;
2018-04-19 17:46:47 +00:00
return JSON.stringify(input);
}
2018-02-27 16:20:49 +00:00
2019-03-17 11:15:39 +00:00
protected static resolveKey<T>(key: SettingsKey<T>, _default: T, resolver: (key: string) => string | boolean, default_type?: string) : T {
let value = resolver(key.key);
if(!value) {
/* trying fallbacks */
for(const fallback of key.fallback_keys || []) {
value = resolver(fallback);
if(typeof(value) === "string") {
/* fallback key succeeded */
const importer = (key.fallback_imports || {})[fallback];
if(importer)
return importer(value);
break;
}
}
}
if(typeof(value) !== 'string')
return _default;
2019-04-04 19:47:52 +00:00
return SettingsBase.transformStO(value as string, _default, default_type);
2019-03-17 11:15:39 +00:00
}
protected static keyify<T>(key: string | SettingsKey<T>) : SettingsKey<T> {
if(typeof(key) === "string")
return {key: key};
if(typeof(key) === "object" && key.key)
return key;
throw "key is not a key";
}
2019-04-04 19:47:52 +00:00
}
class StaticSettings extends SettingsBase {
private static _instance: StaticSettings;
static get instance() : StaticSettings {
if(!this._instance)
this._instance = new StaticSettings(true);
return this._instance;
}
2019-03-17 11:15:39 +00:00
2018-04-19 17:46:47 +00:00
protected _handle: StaticSettings;
protected _staticPropsTag: JQuery;
2018-04-16 18:38:35 +00:00
2018-04-19 17:46:47 +00:00
protected constructor(_reserved = undefined) {
2019-04-04 19:47:52 +00:00
super();
2018-04-19 17:46:47 +00:00
if(_reserved && !StaticSettings._instance) {
this._staticPropsTag = $("#properties");
this.initializeStatic();
} else {
this._handle = StaticSettings.instance;
}
2018-04-16 18:38:35 +00:00
}
private initializeStatic() {
2019-04-04 19:47:52 +00:00
let search;
if(window.opener && window.opener !== window) {
search = new URL(window.location.href).search;
} else {
search = location.search;
}
search.substr(1).split("&").forEach(part => {
2018-04-16 18:38:35 +00:00
let item = part.split("=");
$("<x-property></x-property>")
2018-04-16 18:38:35 +00:00
.attr("key", item[0])
.attr("value", item[1])
.appendTo(this._staticPropsTag);
});
}
2019-03-17 11:15:39 +00:00
static?<T>(key: string | SettingsKey<T>, _default?: T, default_type?: string) : T {
if(this._handle) return this._handle.static<T>(key, _default, default_type);
key = StaticSettings.keyify(key);
return StaticSettings.resolveKey(key, _default, key => {
let result = this._staticPropsTag.find("[key='" + key + "']");
if(result.length > 0)
return decodeURIComponent(result.last().attr('value'));
return false;
}, default_type);
2018-02-27 16:20:49 +00:00
}
2019-03-17 11:15:39 +00:00
deleteStatic<T>(key: string | SettingsKey<T>) {
2018-04-19 17:46:47 +00:00
if(this._handle) {
2019-03-17 11:15:39 +00:00
this._handle.deleteStatic<T>(key);
2018-04-19 17:46:47 +00:00
return;
}
2019-03-17 11:15:39 +00:00
key = StaticSettings.keyify(key);
let result = this._staticPropsTag.find("[key='" + key.key + "']");
2018-04-19 17:46:47 +00:00
if(result.length != 0) result.detach();
}
}
class Settings extends StaticSettings {
2019-03-17 11:15:39 +00:00
static readonly KEY_DISABLE_CONTEXT_MENU: SettingsKey<boolean> = {
key: 'disableContextMenu',
description: 'Disable the context menu for the channel tree which allows to debug the DOM easier'
};
static readonly KEY_DISABLE_UNLOAD_DIALOG: SettingsKey<boolean> = {
key: 'disableUnloadDialog',
description: 'Disables the unload popup on side closing'
};
static readonly KEY_DISABLE_VOICE: SettingsKey<boolean> = {
key: 'disableVoice',
description: 'Disables the voice bridge. If disabled, the audio and codec workers aren\'t required anymore'
};
2019-04-04 19:47:52 +00:00
static readonly KEY_DISABLE_MULTI_SESSION: SettingsKey<boolean> = {
key: 'disableMultiSession',
};
2019-03-17 11:15:39 +00:00
2019-03-25 19:04:04 +00:00
static readonly KEY_LOAD_DUMMY_ERROR: SettingsKey<boolean> = {
key: 'dummy_load_error',
description: 'Triggers a loading error at the end of the loading process.'
};
2019-03-17 11:15:39 +00:00
/* Control bar */
static readonly KEY_CONTROL_MUTE_INPUT: SettingsKey<boolean> = {
key: 'mute_input'
};
static readonly KEY_CONTROL_MUTE_OUTPUT: SettingsKey<boolean> = {
key: 'mute_output'
};
static readonly KEY_CONTROL_SHOW_QUERIES: SettingsKey<boolean> = {
key: 'show_server_queries'
};
static readonly KEY_CONTROL_CHANNEL_SUBSCRIBE_ALL: SettingsKey<boolean> = {
key: 'channel_subscribe_all'
};
/* Connect parameters */
static readonly KEY_FLAG_CONNECT_DEFAULT: SettingsKey<boolean> = {
key: 'connect_default'
};
static readonly KEY_CONNECT_ADDRESS: SettingsKey<string> = {
key: 'connect_address'
};
static readonly KEY_CONNECT_PROFILE: SettingsKey<string> = {
key: 'connect_profile'
};
static readonly KEY_CONNECT_USERNAME: SettingsKey<string> = {
key: 'connect_username'
};
static readonly KEY_CONNECT_PASSWORD: SettingsKey<string> = {
key: 'connect_password'
};
static readonly KEY_FLAG_CONNECT_PASSWORD: SettingsKey<boolean> = {
key: 'connect_password_hashed'
};
2019-04-04 19:47:52 +00:00
static readonly KEY_CERTIFICATE_CALLBACK: SettingsKey<string> = {
key: 'certificate_callback'
};
2019-04-29 16:49:01 +00:00
/* sounds */
static readonly KEY_SOUND_MASTER: SettingsKey<number> = {
key: 'audio_master_volume'
};
static readonly KEY_SOUND_MASTER_SOUNDS: SettingsKey<number> = {
key: 'audio_master_volume_sounds'
};
2019-03-17 11:15:39 +00:00
static readonly FN_SERVER_CHANNEL_SUBSCRIBE_MODE: (channel: ChannelEntry) => SettingsKey<ChannelSubscribeMode> = channel => {
return {
key: 'channel_subscribe_mode_' + channel.getChannelId()
}
};
static readonly FN_PROFILE_RECORD: (name: string) => SettingsKey<any> = name => {
return {
key: 'profile_record' + name
}
};
2019-03-17 11:15:39 +00:00
static readonly KEYS = (() => {
const result = [];
for(const key in Settings) {
if(!key.toUpperCase().startsWith("KEY_"))
continue;
if(key.toUpperCase() == "KEYS")
continue;
result.push(key);
}
return result;
})();
2018-04-19 17:46:47 +00:00
private cacheGlobal = {};
private saveWorker: NodeJS.Timer;
private updated: boolean = false;
constructor() {
super();
this.cacheGlobal = JSON.parse(localStorage.getItem("settings.global"));
if(!this.cacheGlobal) this.cacheGlobal = {};
this.saveWorker = setInterval(() => {
if(this.updated)
this.save();
}, 5 * 1000);
2018-04-16 18:38:35 +00:00
}
2019-03-17 11:15:39 +00:00
static_global?<T>(key: string | SettingsKey<T>, _default?: T) : T {
const default_object = { seed: Math.random() } as any;
let _static = this.static(key, default_object, typeof _default);
if(_static !== default_object) return StaticSettings.transformStO(_static, _default);
return this.global<T>(key, _default);
}
2019-03-17 11:15:39 +00:00
global?<T>(key: string | SettingsKey<T>, _default?: T) : T {
return StaticSettings.resolveKey(Settings.keyify(key), _default, key => this.cacheGlobal[key]);
2018-02-27 16:20:49 +00:00
}
2019-03-17 11:15:39 +00:00
changeGlobal<T>(key: string | SettingsKey<T>, value?: T){
key = Settings.keyify(key);
if(this.cacheGlobal[key.key] == value) return;
2018-03-07 18:06:52 +00:00
2018-02-27 16:20:49 +00:00
this.updated = true;
2019-03-17 11:15:39 +00:00
this.cacheGlobal[key.key] = StaticSettings.transformOtS(value);
2018-02-27 16:20:49 +00:00
if(Settings.UPDATE_DIRECT)
this.save();
}
2019-04-04 19:47:52 +00:00
save() {
this.updated = false;
let global = JSON.stringify(this.cacheGlobal);
localStorage.setItem("settings.global", global);
2019-04-15 13:33:51 +00:00
if(localStorage.save)
localStorage.save();
2019-04-04 19:47:52 +00:00
}
}
class ServerSettings extends SettingsBase {
private cacheServer = {};
private currentServer: ServerEntry;
private _server_save_worker: NodeJS.Timer;
private _server_settings_updated: boolean = false;
constructor() {
super();
this._server_save_worker = setInterval(() => {
if(this._server_settings_updated)
this.save();
}, 5 * 1000);
}
server?<T>(key: string | SettingsKey<T>, _default?: T) : T {
return StaticSettings.resolveKey(Settings.keyify(key), _default, key => this.cacheServer[key]);
}
2019-03-17 11:15:39 +00:00
changeServer<T>(key: string | SettingsKey<T>, value?: T) {
key = Settings.keyify(key);
if(this.cacheServer[key.key] == value) return;
2018-03-07 18:06:52 +00:00
2019-04-04 19:47:52 +00:00
this._server_settings_updated = true;
2019-03-17 11:15:39 +00:00
this.cacheServer[key.key] = StaticSettings.transformOtS(value);
2018-02-27 16:20:49 +00:00
if(Settings.UPDATE_DIRECT)
this.save();
}
2018-04-16 18:38:35 +00:00
setServer(server: ServerEntry) {
if(this.currentServer) {
this.save();
2018-03-07 18:06:52 +00:00
this.cacheServer = {};
2018-04-16 18:38:35 +00:00
this.currentServer = undefined;
}
this.currentServer = server;
if(this.currentServer) {
let serverId = this.currentServer.properties.virtualserver_unique_identifier;
this.cacheServer = JSON.parse(localStorage.getItem("settings.server_" + serverId));
if(!this.cacheServer)
this.cacheServer = {};
2018-03-07 18:06:52 +00:00
}
2018-02-27 16:20:49 +00:00
}
save() {
2019-04-04 19:47:52 +00:00
this._server_settings_updated = false;
2018-02-27 16:20:49 +00:00
2018-04-16 18:38:35 +00:00
if(this.currentServer) {
let serverId = this.currentServer.properties.virtualserver_unique_identifier;
2018-03-07 18:06:52 +00:00
let server = JSON.stringify(this.cacheServer);
localStorage.setItem("settings.server_" + serverId, server);
2019-04-15 13:33:51 +00:00
if(localStorage.save)
localStorage.save();
2018-03-07 18:06:52 +00:00
}
2018-02-27 16:20:49 +00:00
}
}