2019-04-04 19:47:52 +00:00
|
|
|
/// <reference path="ui/frames/chat.ts" />
|
2018-03-07 18:06:52 +00:00
|
|
|
/// <reference path="ui/modal/ModalConnect.ts" />
|
2018-04-16 18:38:35 +00:00
|
|
|
/// <reference path="ui/modal/ModalCreateChannel.ts" />
|
2018-10-20 17:58:06 +00:00
|
|
|
/// <reference path="ui/modal/ModalBanCreate.ts" />
|
2018-04-30 21:57:21 +00:00
|
|
|
/// <reference path="ui/modal/ModalBanClient.ts" />
|
2018-06-20 17:06:55 +00:00
|
|
|
/// <reference path="ui/modal/ModalYesNo.ts" />
|
2018-10-20 17:58:06 +00:00
|
|
|
/// <reference path="ui/modal/ModalBanList.ts" />
|
2018-04-11 15:56:09 +00:00
|
|
|
/// <reference path="settings.ts" />
|
2018-04-16 18:38:35 +00:00
|
|
|
/// <reference path="log.ts" />
|
2018-02-27 16:20:49 +00:00
|
|
|
|
2018-04-16 18:38:35 +00:00
|
|
|
let settings: Settings;
|
2018-04-11 15:56:09 +00:00
|
|
|
|
2018-10-03 20:04:29 +00:00
|
|
|
const js_render = window.jsrender || $;
|
2018-10-06 13:13:45 +00:00
|
|
|
const native_client = window.require !== undefined;
|
|
|
|
|
2018-10-28 22:01:09 +00:00
|
|
|
function getUserMediaFunction() {
|
2018-12-23 21:56:04 +00:00
|
|
|
if((navigator as any).mediaDevices && (navigator as any).mediaDevices.getUserMedia)
|
|
|
|
return (settings, success, fail) => { (navigator as any).mediaDevices.getUserMedia(settings).then(success).catch(fail); };
|
|
|
|
return (navigator as any).getUserMedia || (navigator as any).webkitGetUserMedia || (navigator as any).mozGetUserMedia;
|
2018-10-28 22:01:09 +00:00
|
|
|
}
|
|
|
|
|
2019-03-31 18:07:03 +00:00
|
|
|
interface Window {
|
|
|
|
open_connected_question: () => Promise<boolean>;
|
|
|
|
}
|
|
|
|
|
2018-10-06 13:13:45 +00:00
|
|
|
function setup_close() {
|
|
|
|
window.onbeforeunload = event => {
|
2018-12-28 14:39:23 +00:00
|
|
|
if(profiles.requires_save())
|
|
|
|
profiles.save();
|
2018-10-06 13:13:45 +00:00
|
|
|
|
2018-12-28 14:39:23 +00:00
|
|
|
if(!settings.static(Settings.KEY_DISABLE_UNLOAD_DIALOG, false)) {
|
2019-04-04 19:47:52 +00:00
|
|
|
const active_connections = server_connections.server_connection_handlers().filter(e => e.connected);
|
|
|
|
if(active_connections.length == 0) return;
|
2018-10-06 13:13:45 +00:00
|
|
|
|
2018-12-28 14:39:23 +00:00
|
|
|
if(!native_client) {
|
|
|
|
event.returnValue = "Are you really sure?<br>You're still connected!";
|
|
|
|
} else {
|
2019-03-31 18:07:03 +00:00
|
|
|
if(window.open_connected_question) {
|
|
|
|
event.preventDefault();
|
|
|
|
event.returnValue = "question";
|
|
|
|
window.open_connected_question().then(result => {
|
|
|
|
if(result) {
|
|
|
|
window.onbeforeunload = undefined;
|
|
|
|
|
|
|
|
const {remote} = require('electron');
|
|
|
|
remote.getCurrentWindow().close();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
} else { /* we're in debugging mode */ }
|
2018-12-28 14:39:23 +00:00
|
|
|
}
|
2018-10-06 13:13:45 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
2018-10-03 20:04:29 +00:00
|
|
|
|
2018-10-20 17:58:06 +00:00
|
|
|
declare function moment(...arguments) : any;
|
2018-10-14 11:27:48 +00:00
|
|
|
function setup_jsrender() : boolean {
|
2018-10-03 20:04:29 +00:00
|
|
|
if(!js_render) {
|
|
|
|
displayCriticalError("Missing jsrender extension!");
|
2018-10-14 11:27:48 +00:00
|
|
|
return false;
|
2018-10-03 20:04:29 +00:00
|
|
|
}
|
2018-10-07 16:21:28 +00:00
|
|
|
if(!js_render.views) {
|
|
|
|
displayCriticalError("Missing jsrender viewer extension!");
|
2018-10-14 11:27:48 +00:00
|
|
|
return false;
|
2018-10-07 16:21:28 +00:00
|
|
|
}
|
2018-10-03 20:04:29 +00:00
|
|
|
js_render.views.settings.allowCode(true);
|
|
|
|
js_render.views.tags("rnd", (argument) => {
|
2018-06-24 11:38:53 +00:00
|
|
|
let min = parseInt(argument.substr(0, argument.indexOf('~')));
|
|
|
|
let max = parseInt(argument.substr(argument.indexOf('~') + 1));
|
|
|
|
|
|
|
|
return (Math.round(Math.random() * (min + max + 1) - min)).toString();
|
|
|
|
});
|
2018-10-14 11:27:48 +00:00
|
|
|
|
2019-01-26 16:10:15 +00:00
|
|
|
js_render.views.tags("fmt_date", (...args) => {
|
|
|
|
return moment(args[0]).format(args[1]);
|
2018-10-20 17:58:06 +00:00
|
|
|
});
|
|
|
|
|
2019-01-26 16:10:15 +00:00
|
|
|
js_render.views.tags("tr", (...args) => {
|
|
|
|
return tr(args[0]);
|
2018-12-08 22:13:33 +00:00
|
|
|
});
|
|
|
|
|
2018-10-14 11:27:48 +00:00
|
|
|
$(".jsrender-template").each((idx, _entry) => {
|
|
|
|
if(!js_render.templates(_entry.id, _entry.innerHTML)) { //, _entry.innerHTML
|
|
|
|
console.error("Failed to cache template " + _entry.id + " for js render!");
|
|
|
|
} else
|
|
|
|
console.debug("Successfully loaded jsrender template " + _entry.id);
|
|
|
|
});
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-12-09 19:18:49 +00:00
|
|
|
async function initialize() {
|
2019-01-19 12:16:12 +00:00
|
|
|
settings = new Settings();
|
|
|
|
|
2018-12-15 12:06:41 +00:00
|
|
|
try {
|
|
|
|
await i18n.initialize();
|
|
|
|
} catch(error) {
|
|
|
|
console.error(tr("Failed to initialized the translation system!\nError: %o"), error);
|
|
|
|
displayCriticalError("Failed to setup the translation system");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-04-04 19:47:52 +00:00
|
|
|
bipc.setup();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
async function initialize_app() {
|
|
|
|
const display_load_error = message => {
|
|
|
|
if(typeof(display_critical_load) !== "undefined")
|
|
|
|
display_critical_load(message);
|
|
|
|
else
|
|
|
|
displayCriticalError(message);
|
|
|
|
};
|
|
|
|
|
2018-12-14 23:09:47 +00:00
|
|
|
try { //Initialize main template
|
2019-01-19 12:03:51 +00:00
|
|
|
const main = $("#tmpl_main").renderTag().dividerfy();
|
2019-01-19 12:16:12 +00:00
|
|
|
|
2018-12-14 23:09:47 +00:00
|
|
|
$("body").append(main);
|
|
|
|
} catch(error) {
|
2018-12-30 00:38:13 +00:00
|
|
|
console.error(error);
|
2018-12-14 23:09:47 +00:00
|
|
|
display_load_error(tr("Failed to setup main page!"));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-04-04 19:47:52 +00:00
|
|
|
control_bar = new ControlBar($("#control_bar")); /* setup the control bar */
|
|
|
|
|
|
|
|
if(!audio.player.initialize())
|
|
|
|
console.warn(tr("Failed to initialize audio controller!"));
|
|
|
|
|
|
|
|
sound.initialize().then(() => {
|
|
|
|
console.log(tr("Sounds initialitzed"));
|
|
|
|
});
|
|
|
|
|
2019-01-26 16:10:15 +00:00
|
|
|
await profiles.load();
|
2018-04-11 15:56:09 +00:00
|
|
|
|
2018-12-09 19:18:49 +00:00
|
|
|
try {
|
|
|
|
await ppt.initialize();
|
|
|
|
} catch(error) {
|
|
|
|
console.error(tr("Failed to initialize ppt!\nError: %o"), error);
|
|
|
|
displayCriticalError(tr("Failed to initialize ppt!"));
|
|
|
|
return;
|
|
|
|
}
|
2019-02-17 15:08:10 +00:00
|
|
|
|
|
|
|
setup_close();
|
2018-12-09 19:18:49 +00:00
|
|
|
}
|
|
|
|
|
2019-01-26 16:10:15 +00:00
|
|
|
function ab2str(buf) {
|
|
|
|
return String.fromCharCode.apply(null, new Uint16Array(buf));
|
|
|
|
}
|
|
|
|
|
|
|
|
function str2ab8(str) {
|
|
|
|
const buf = new ArrayBuffer(str.length);
|
|
|
|
const bufView = new Uint8Array(buf);
|
|
|
|
for (let i = 0, strLen = str.length; i < strLen; i++) {
|
|
|
|
bufView[i] = str.charCodeAt(i);
|
|
|
|
}
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* FIXME Dont use atob, because it sucks for non UTF-8 tings */
|
|
|
|
function arrayBufferBase64(base64: string) {
|
|
|
|
base64 = atob(base64);
|
|
|
|
const buf = new ArrayBuffer(base64.length);
|
|
|
|
const bufView = new Uint8Array(buf);
|
|
|
|
for (let i = 0, strLen = base64.length; i < strLen; i++) {
|
|
|
|
bufView[i] = base64.charCodeAt(i);
|
|
|
|
}
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
function base64ArrayBuffer(arrayBuffer) {
|
|
|
|
var base64 = ''
|
|
|
|
var encodings = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/'
|
|
|
|
|
|
|
|
var bytes = new Uint8Array(arrayBuffer)
|
|
|
|
var byteLength = bytes.byteLength
|
|
|
|
var byteRemainder = byteLength % 3
|
|
|
|
var mainLength = byteLength - byteRemainder
|
|
|
|
|
|
|
|
var a, b, c, d
|
|
|
|
var chunk
|
|
|
|
|
|
|
|
// Main loop deals with bytes in chunks of 3
|
|
|
|
for (var i = 0; i < mainLength; i = i + 3) {
|
|
|
|
// Combine the three bytes into a single integer
|
|
|
|
chunk = (bytes[i] << 16) | (bytes[i + 1] << 8) | bytes[i + 2]
|
|
|
|
|
|
|
|
// Use bitmasks to extract 6-bit segments from the triplet
|
|
|
|
a = (chunk & 16515072) >> 18 // 16515072 = (2^6 - 1) << 18
|
|
|
|
b = (chunk & 258048) >> 12 // 258048 = (2^6 - 1) << 12
|
|
|
|
c = (chunk & 4032) >> 6 // 4032 = (2^6 - 1) << 6
|
|
|
|
d = chunk & 63 // 63 = 2^6 - 1
|
|
|
|
|
|
|
|
// Convert the raw binary segments to the appropriate ASCII encoding
|
|
|
|
base64 += encodings[a] + encodings[b] + encodings[c] + encodings[d]
|
|
|
|
}
|
|
|
|
|
|
|
|
// Deal with the remaining bytes and padding
|
|
|
|
if (byteRemainder == 1) {
|
|
|
|
chunk = bytes[mainLength]
|
|
|
|
|
|
|
|
a = (chunk & 252) >> 2 // 252 = (2^6 - 1) << 2
|
|
|
|
|
|
|
|
// Set the 4 least significant bits to zero
|
|
|
|
b = (chunk & 3) << 4 // 3 = 2^2 - 1
|
|
|
|
|
|
|
|
base64 += encodings[a] + encodings[b] + '=='
|
|
|
|
} else if (byteRemainder == 2) {
|
|
|
|
chunk = (bytes[mainLength] << 8) | bytes[mainLength + 1]
|
|
|
|
|
|
|
|
a = (chunk & 64512) >> 10 // 64512 = (2^6 - 1) << 10
|
|
|
|
b = (chunk & 1008) >> 4 // 1008 = (2^6 - 1) << 4
|
|
|
|
|
|
|
|
// Set the 2 least significant bits to zero
|
|
|
|
c = (chunk & 15) << 2 // 15 = 2^4 - 1
|
|
|
|
|
|
|
|
base64 += encodings[a] + encodings[b] + encodings[c] + '='
|
|
|
|
}
|
|
|
|
|
|
|
|
return base64
|
|
|
|
}
|
|
|
|
|
2019-01-28 19:36:11 +00:00
|
|
|
function Base64EncodeUrl(str){
|
2019-01-26 16:10:15 +00:00
|
|
|
return str.replace(/\+/g, '-').replace(/\//g, '_').replace(/\=+$/, '');
|
|
|
|
}
|
|
|
|
|
|
|
|
function Base64DecodeUrl(str: string, pad?: boolean){
|
|
|
|
if(typeof(pad) === 'undefined' || pad)
|
|
|
|
str = (str + '===').slice(0, str.length + (str.length % 4));
|
|
|
|
return str.replace(/-/g, '+').replace(/_/g, '/');
|
|
|
|
}
|
|
|
|
|
2018-12-09 19:18:49 +00:00
|
|
|
function main() {
|
|
|
|
//http://localhost:63343/Web-Client/index.php?_ijt=omcpmt8b9hnjlfguh8ajgrgolr&default_connect_url=true&default_connect_type=teamspeak&default_connect_url=localhost%3A9987&disableUnloadDialog=1&loader_ignore_age=1
|
2019-04-04 19:47:52 +00:00
|
|
|
voice_recoder = new VoiceRecorder();
|
|
|
|
voice_recoder.reinitialiseVAD();
|
|
|
|
|
|
|
|
server_connections = new ServerConnectionManager($("#connection-handlers"));
|
|
|
|
control_bar.initialise(); /* before connection handler to allow property apply */
|
2018-12-09 19:18:49 +00:00
|
|
|
|
2019-04-04 19:47:52 +00:00
|
|
|
const initial_handler = server_connections.spawn_server_connection_handler();
|
|
|
|
initial_handler.acquire_recorder(voice_recoder, false);
|
|
|
|
control_bar.set_connection_handler(initial_handler);
|
2018-04-11 15:56:09 +00:00
|
|
|
/** Setup the XF forum identity **/
|
2018-12-28 14:39:23 +00:00
|
|
|
profiles.identities.setup_forum();
|
2018-04-11 15:56:09 +00:00
|
|
|
|
2019-01-26 17:06:42 +00:00
|
|
|
let _resize_timeout: NodeJS.Timer;
|
2019-04-04 19:47:52 +00:00
|
|
|
$(window).on('resize', event => {
|
|
|
|
if(event.target !== window)
|
|
|
|
return;
|
|
|
|
|
2018-11-03 23:39:29 +00:00
|
|
|
if(_resize_timeout)
|
|
|
|
clearTimeout(_resize_timeout);
|
|
|
|
_resize_timeout = setTimeout(() => {
|
2019-04-04 19:47:52 +00:00
|
|
|
for(const connection of server_connections.server_connection_handlers())
|
|
|
|
connection.invoke_resized_on_activate = true;
|
|
|
|
const active_connection = server_connections.active_connection_handler();
|
|
|
|
if(active_connection)
|
|
|
|
active_connection.resize_elements();
|
2018-11-03 23:39:29 +00:00
|
|
|
}, 1000);
|
2018-10-06 13:13:45 +00:00
|
|
|
});
|
2019-02-25 14:59:42 +00:00
|
|
|
|
|
|
|
stats.initialize({
|
|
|
|
verbose: true,
|
|
|
|
anonymize_ip_addresses: true,
|
|
|
|
volatile_collection_only: false
|
|
|
|
});
|
|
|
|
stats.register_user_count_listener(status => {
|
|
|
|
console.log("Received user count update: %o", status);
|
|
|
|
});
|
2019-03-25 19:04:04 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
setTimeout(() => {
|
|
|
|
Modals.spawnAvatarList(globalClient);
|
|
|
|
}, 1000);
|
|
|
|
*/
|
2019-04-04 19:47:52 +00:00
|
|
|
(<any>window).test_upload = (message?: string) => {
|
|
|
|
message = message || "Hello World";
|
|
|
|
|
|
|
|
const connection = server_connections.active_connection_handler();
|
|
|
|
connection.fileManager.upload_file({
|
|
|
|
size: message.length,
|
2019-03-28 16:30:00 +00:00
|
|
|
overwrite: true,
|
2019-04-04 19:47:52 +00:00
|
|
|
channel: connection.getClient().currentChannel(),
|
2019-03-28 16:30:00 +00:00
|
|
|
name: '/HelloWorld.txt',
|
|
|
|
path: ''
|
|
|
|
}).then(key => {
|
|
|
|
console.log("Got key: %o", key);
|
|
|
|
const upload = new RequestFileUpload(key);
|
|
|
|
|
2019-04-04 19:47:52 +00:00
|
|
|
const buffer = new Uint8Array(message.length);
|
2019-03-28 16:30:00 +00:00
|
|
|
{
|
2019-04-04 19:47:52 +00:00
|
|
|
for(let index = 0; index < message.length; index++)
|
|
|
|
buffer[index] = message.charCodeAt(index);
|
2019-03-28 16:30:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
upload.put_data(buffer).catch(error => {
|
|
|
|
console.error(error);
|
|
|
|
});
|
|
|
|
})
|
|
|
|
};
|
2019-04-04 19:47:52 +00:00
|
|
|
|
|
|
|
server_connections.set_active_connection_handler(server_connections.server_connection_handlers()[0]);
|
|
|
|
|
|
|
|
if(settings.static(Settings.KEY_FLAG_CONNECT_DEFAULT, false) && settings.static(Settings.KEY_CONNECT_ADDRESS, "")) {
|
|
|
|
const profile_uuid = settings.static(Settings.KEY_CONNECT_PROFILE, (profiles.default_profile() || {id: 'default'}).id);
|
|
|
|
console.log("UUID: %s", profile_uuid);
|
|
|
|
const profile = profiles.find_profile(profile_uuid) || profiles.default_profile();
|
|
|
|
const address = settings.static(Settings.KEY_CONNECT_ADDRESS, "");
|
|
|
|
const username = settings.static(Settings.KEY_CONNECT_USERNAME, "Another TeaSpeak user");
|
|
|
|
|
|
|
|
const password = settings.static(Settings.KEY_CONNECT_PASSWORD, "");
|
|
|
|
const password_hashed = settings.static(Settings.KEY_FLAG_CONNECT_PASSWORD, false);
|
|
|
|
|
|
|
|
if(profile && profile.valid()) {
|
|
|
|
const connection = server_connections.active_connection_handler() || server_connections.spawn_server_connection_handler();
|
|
|
|
connection.startConnection(address, profile, username, password.length > 0 ? {
|
|
|
|
password: password,
|
|
|
|
hashed: password_hashed
|
|
|
|
} : undefined);
|
|
|
|
} else {
|
|
|
|
Modals.spawnConnectModal({
|
|
|
|
url: address,
|
|
|
|
enforce: true
|
|
|
|
}, {
|
|
|
|
profile: profile,
|
|
|
|
enforce: true
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
2018-04-19 16:42:34 +00:00
|
|
|
}
|
|
|
|
|
2019-04-04 19:47:52 +00:00
|
|
|
const task_teaweb_starter: loader.Task = {
|
|
|
|
name: "voice app starter",
|
2018-12-30 00:38:13 +00:00
|
|
|
function: async () => {
|
|
|
|
try {
|
2019-04-04 19:47:52 +00:00
|
|
|
await initialize_app();
|
2018-12-30 00:38:13 +00:00
|
|
|
main();
|
|
|
|
if(!audio.player.initialized()) {
|
|
|
|
log.info(LogCategory.VOICE, tr("Initialize audio controller later!"));
|
|
|
|
if(!audio.player.initializeFromGesture) {
|
|
|
|
console.error(tr("Missing audio.player.initializeFromGesture"));
|
|
|
|
} else
|
|
|
|
$(document).one('click', event => audio.player.initializeFromGesture());
|
|
|
|
}
|
|
|
|
} catch (ex) {
|
|
|
|
console.error(ex.stack);
|
|
|
|
if(ex instanceof ReferenceError || ex instanceof TypeError)
|
|
|
|
ex = ex.name + ": " + ex.message;
|
|
|
|
displayCriticalError("Failed to invoke main function:<br>" + ex);
|
2018-09-25 15:39:38 +00:00
|
|
|
}
|
2018-12-30 00:38:13 +00:00
|
|
|
},
|
|
|
|
priority: 10
|
2019-04-04 19:47:52 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
const task_certificate_callback: loader.Task = {
|
|
|
|
name: "certificate accept tester",
|
|
|
|
function: async () => {
|
|
|
|
const certificate_accept = settings.static_global(Settings.KEY_CERTIFICATE_CALLBACK, undefined);
|
|
|
|
if(certificate_accept) {
|
|
|
|
log.info(LogCategory.IPC, tr("Using this instance as certificate callback. ID: %s"), certificate_accept);
|
|
|
|
try {
|
|
|
|
try {
|
|
|
|
await bipc.get_handler().post_certificate_accpected(certificate_accept);
|
|
|
|
} catch(e) {} //FIXME remove!
|
|
|
|
log.info(LogCategory.IPC, tr("Other instance has acknowledged out work. Closing this window."));
|
|
|
|
|
|
|
|
const seconds_tag = $.spawn("a");
|
|
|
|
|
|
|
|
let seconds = 5;
|
|
|
|
let interval_id;
|
|
|
|
interval_id = setInterval(() => {
|
|
|
|
seconds--;
|
|
|
|
seconds_tag.text(seconds.toString());
|
|
|
|
|
|
|
|
if(seconds <= 0) {
|
|
|
|
clearTimeout(interval_id);
|
|
|
|
log.info(LogCategory.GENERAL, tr("Closing window"));
|
|
|
|
window.close();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}, 1000);
|
|
|
|
|
|
|
|
const message =
|
|
|
|
"You've successfully accepted the certificate.{:br:}" +
|
|
|
|
"This page will close in {0} seconds.";
|
|
|
|
createInfoModal(
|
|
|
|
tr("Certificate acccepted successfully"),
|
|
|
|
MessageHelper.formatMessage(tr(message), seconds_tag),
|
|
|
|
{
|
|
|
|
closeable: false,
|
|
|
|
footer: undefined
|
|
|
|
}
|
|
|
|
).open();
|
|
|
|
return;
|
|
|
|
} catch(error) {
|
|
|
|
log.warn(LogCategory.IPC, tr("Failed to successfully post certificate accept status: %o"), error);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
log.info(LogCategory.IPC, tr("We're not used to accept certificated. Booting app."));
|
|
|
|
}
|
|
|
|
|
|
|
|
loader.register_task(loader.Stage.LOADED, task_teaweb_starter);
|
|
|
|
},
|
|
|
|
priority: 10
|
|
|
|
};
|
|
|
|
|
2019-04-25 18:22:13 +00:00
|
|
|
loader.register_task(loader.Stage.JAVASCRIPT_INITIALIZING, {
|
|
|
|
name: "jrendere initialize",
|
|
|
|
function: async () => {
|
|
|
|
try {
|
|
|
|
if(!setup_jsrender())
|
|
|
|
throw "invalid load";
|
|
|
|
} catch (error) {
|
|
|
|
displayCriticalError(tr("Failed to setup jsrender"));
|
|
|
|
console.error(tr("Failed to load jsrender! %o"), error);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
priority: 100
|
|
|
|
})
|
|
|
|
|
2019-04-04 19:47:52 +00:00
|
|
|
loader.register_task(loader.Stage.LOADED, {
|
|
|
|
name: "app starter",
|
|
|
|
function: async () => {
|
|
|
|
try {
|
|
|
|
await initialize();
|
|
|
|
|
|
|
|
if(app.is_web()) {
|
|
|
|
loader.register_task(loader.Stage.LOADED, task_certificate_callback);
|
|
|
|
} else
|
|
|
|
loader.register_task(loader.Stage.LOADED, task_teaweb_starter);
|
|
|
|
} catch (ex) {
|
|
|
|
console.error(ex.stack);
|
|
|
|
if(ex instanceof ReferenceError || ex instanceof TypeError)
|
|
|
|
ex = ex.name + ": " + ex.message;
|
|
|
|
displayCriticalError("Failed to boot app function:<br>" + ex);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
priority: 10
|
2018-10-06 13:13:45 +00:00
|
|
|
});
|
2018-12-30 00:38:13 +00:00
|
|
|
|