2018-02-27 16:20:49 +00:00
|
|
|
interface Array<T> {
|
|
|
|
remove(elem?: T): boolean;
|
|
|
|
last?(): T;
|
|
|
|
|
|
|
|
pop_front(): T | undefined;
|
|
|
|
}
|
|
|
|
|
2018-08-10 19:30:58 +00:00
|
|
|
interface JSON {
|
|
|
|
map_to<T>(object: T, json: any, variables?: string | string[], validator?: (map_field: string, map_value: string) => boolean, variable_direction?: number) : T;
|
|
|
|
map_field_to<T>(object: T, value: any, field: string) : T;
|
|
|
|
}
|
|
|
|
|
2018-12-04 17:48:54 +00:00
|
|
|
interface JQuery<TElement = HTMLElement> {
|
2018-06-20 19:05:35 +00:00
|
|
|
render(values?: any) : string;
|
2018-12-04 17:48:54 +00:00
|
|
|
renderTag(values?: any) : JQuery<TElement>;
|
2018-12-23 16:41:14 +00:00
|
|
|
hasScrollBar() : boolean;
|
2018-02-27 16:20:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
interface JQueryStatic<TElement extends Node = HTMLElement> {
|
|
|
|
spawn<K extends keyof HTMLElementTagNameMap>(tagName: K): JQuery<HTMLElementTagNameMap[K]>;
|
2018-06-20 19:05:35 +00:00
|
|
|
views: any;
|
2018-02-27 16:20:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
interface String {
|
|
|
|
format(...fmt): string;
|
2018-04-11 15:56:09 +00:00
|
|
|
format(arguments: string[]): string;
|
2018-02-27 16:20:49 +00:00
|
|
|
}
|
|
|
|
|
2018-08-10 19:30:58 +00:00
|
|
|
if(!JSON.map_to) {
|
2018-09-30 19:50:59 +00:00
|
|
|
JSON.map_to = function <T>(object: T, json: any, variables?: string | string[], validator?: (map_field: string, map_value: string) => boolean, variable_direction?: number): T {
|
|
|
|
if (!validator) validator = (a, b) => true;
|
2018-08-10 19:30:58 +00:00
|
|
|
|
2018-09-30 19:50:59 +00:00
|
|
|
if (!variables) {
|
2018-08-10 19:30:58 +00:00
|
|
|
variables = [];
|
|
|
|
|
2018-09-30 19:50:59 +00:00
|
|
|
if (!variable_direction || variable_direction == 0) {
|
|
|
|
for (let field in json)
|
2018-08-10 19:30:58 +00:00
|
|
|
variables.push(field);
|
2018-09-30 19:50:59 +00:00
|
|
|
} else if (variable_direction == 1) {
|
|
|
|
for (let field in object)
|
2018-08-10 19:30:58 +00:00
|
|
|
variables.push(field);
|
|
|
|
}
|
2018-09-30 19:50:59 +00:00
|
|
|
} else if (!Array.isArray(variables)) {
|
2018-08-10 19:30:58 +00:00
|
|
|
variables = [variables];
|
|
|
|
}
|
|
|
|
|
2018-09-30 19:50:59 +00:00
|
|
|
for (let field of variables) {
|
|
|
|
if (!json[field]) {
|
2018-12-05 19:46:33 +00:00
|
|
|
console.trace(tr("Json does not contains %s"), field);
|
2018-08-10 19:30:58 +00:00
|
|
|
continue;
|
|
|
|
}
|
2018-09-30 19:50:59 +00:00
|
|
|
if (!validator(field, json[field])) {
|
2018-12-05 19:46:33 +00:00
|
|
|
console.trace(tr("Validator results in false for %s"), field);
|
2018-08-10 19:30:58 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
JSON.map_field_to(object, json[field], field);
|
|
|
|
}
|
|
|
|
return object;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!JSON.map_field_to) {
|
|
|
|
JSON.map_field_to = function<T>(object: T, value: any, field: string) : T {
|
|
|
|
let field_type = typeof(object[field]);
|
|
|
|
if(field_type == "string" || field_type == "object" || field_type == "undefined")
|
|
|
|
object[field as string] = value;
|
|
|
|
else if(field_type == "number")
|
|
|
|
object[field as string] = parseFloat(value);
|
|
|
|
else if(field_type == "boolean")
|
|
|
|
object[field as string] = value == "1" || value == "true";
|
2018-12-05 19:46:33 +00:00
|
|
|
else console.warn(tr("Invalid object type %s for entry %s"), field_type, field);
|
2018-08-10 19:30:58 +00:00
|
|
|
|
|
|
|
return object;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-27 16:20:49 +00:00
|
|
|
if (!Array.prototype.remove) {
|
|
|
|
Array.prototype.remove = function<T>(elem?: T): boolean {
|
2018-03-24 22:38:01 +00:00
|
|
|
const index = this.indexOf(elem, 0);
|
2018-02-27 16:20:49 +00:00
|
|
|
if (index > -1) {
|
|
|
|
this.splice(index, 1);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!Array.prototype.pop_front) {
|
|
|
|
Array.prototype.pop_front = function<T>(): T {
|
|
|
|
if(this.length == 0) return undefined;
|
|
|
|
return this.splice(0, 1)[0];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-16 18:38:35 +00:00
|
|
|
|
2018-02-27 16:20:49 +00:00
|
|
|
if (!Array.prototype.last){
|
|
|
|
Array.prototype.last = function(){
|
|
|
|
if(this.length == 0) return undefined;
|
|
|
|
return this[this.length - 1];
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2018-04-11 15:56:09 +00:00
|
|
|
if(typeof ($) !== "undefined") {
|
|
|
|
if(!$.spawn) {
|
|
|
|
$.spawn = function<K extends keyof HTMLElementTagNameMap>(tagName: K): JQuery<HTMLElementTagNameMap[K]> {
|
2018-10-08 23:27:14 +00:00
|
|
|
return $(document.createElement(tagName) as any);
|
2018-04-11 15:56:09 +00:00
|
|
|
}
|
2018-02-27 16:20:49 +00:00
|
|
|
}
|
2018-12-23 16:41:14 +00:00
|
|
|
if(!$.fn.renderTag) {
|
|
|
|
$.fn.renderTag = function (values?: any) : JQuery {
|
2018-10-03 20:04:29 +00:00
|
|
|
let result;
|
|
|
|
if(this.render) {
|
|
|
|
result = $(this.render(values));
|
|
|
|
} else {
|
2018-10-14 11:27:48 +00:00
|
|
|
const template = window.jsrender.render[this.attr("id")];
|
|
|
|
/*
|
|
|
|
result = window.jsrender.templates("tmpl_permission_entry", $("#tmpl_permission_entry").html());
|
|
|
|
result = window.jsrender.templates("xxx", this.html());
|
|
|
|
*/
|
|
|
|
result = template(values);
|
2018-10-03 20:04:29 +00:00
|
|
|
result = $(result);
|
|
|
|
}
|
2018-09-30 20:36:17 +00:00
|
|
|
result.find("node").each((index, element) => {
|
2018-11-03 23:39:29 +00:00
|
|
|
$(element).replaceWith(values[$(element).attr("key")] || (values[0] || [])[$(element).attr("key")]);
|
2018-09-30 20:36:17 +00:00
|
|
|
});
|
2018-09-30 19:50:59 +00:00
|
|
|
return result;
|
2018-06-20 17:06:55 +00:00
|
|
|
}
|
|
|
|
}
|
2018-12-23 16:41:14 +00:00
|
|
|
if(!$.fn.hasScrollBar)
|
|
|
|
$.fn.hasScrollBar = function() {
|
|
|
|
return this.get(0).scrollHeight > this.height();
|
|
|
|
}
|
|
|
|
|
2018-02-27 16:20:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!String.prototype.format) {
|
|
|
|
String.prototype.format = function() {
|
2018-03-24 22:38:01 +00:00
|
|
|
const args = arguments;
|
2018-04-11 15:56:09 +00:00
|
|
|
let array = args.length == 1 && $.isArray(args[0]);
|
2018-02-27 16:20:49 +00:00
|
|
|
return this.replace(/\{\{|\}\}|\{(\d+)\}/g, function (m, n) {
|
|
|
|
if (m == "{{") { return "{"; }
|
|
|
|
if (m == "}}") { return "}"; }
|
2018-04-11 15:56:09 +00:00
|
|
|
return array ? args[0][n] : args[n];
|
2018-02-27 16:20:49 +00:00
|
|
|
});
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
function concatenate(resultConstructor, ...arrays) {
|
|
|
|
let totalLength = 0;
|
|
|
|
for (const arr of arrays) {
|
|
|
|
totalLength += arr.length;
|
|
|
|
}
|
|
|
|
const result = new resultConstructor(totalLength);
|
|
|
|
let offset = 0;
|
|
|
|
for (const arr of arrays) {
|
|
|
|
result.set(arr, offset);
|
|
|
|
offset += arr.length;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
function formatDate(secs: number) : string {
|
|
|
|
let years = Math.floor(secs / (60 * 60 * 24 * 365));
|
|
|
|
let days = Math.floor(secs / (60 * 60 * 24)) % 365;
|
|
|
|
let hours = Math.floor(secs / (60 * 60)) % 24;
|
|
|
|
let minutes = Math.floor(secs / 60) % 60;
|
|
|
|
let seconds = Math.floor(secs % 60);
|
|
|
|
|
|
|
|
let result = "";
|
|
|
|
if(years > 0)
|
2018-12-05 19:46:33 +00:00
|
|
|
result += years + " " + tr("years") + " ";
|
2018-02-27 16:20:49 +00:00
|
|
|
if(years > 0 || days > 0)
|
2018-12-05 19:46:33 +00:00
|
|
|
result += days + " " + tr("days") + " ";
|
2018-02-27 16:20:49 +00:00
|
|
|
if(years > 0 || days > 0 || hours > 0)
|
2018-12-05 19:46:33 +00:00
|
|
|
result += hours + " " + tr("hours") + " ";
|
2018-02-27 16:20:49 +00:00
|
|
|
if(years > 0 || days > 0 || hours > 0 || minutes > 0)
|
2018-12-05 19:46:33 +00:00
|
|
|
result += minutes + " " + tr("minutes") + " ";
|
2018-02-27 16:20:49 +00:00
|
|
|
if(years > 0 || days > 0 || hours > 0 || minutes > 0 || seconds > 0)
|
2018-12-05 19:46:33 +00:00
|
|
|
result += seconds + " " + tr("seconds") + " ";
|
2018-02-27 16:20:49 +00:00
|
|
|
else
|
2018-12-05 19:46:33 +00:00
|
|
|
result = tr("now") + " ";
|
2018-02-27 16:20:49 +00:00
|
|
|
|
|
|
|
return result.substr(0, result.length - 1);
|
2018-08-10 19:30:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function calculate_width(text: string) : number {
|
|
|
|
let element = $.spawn("div");
|
|
|
|
element.text(text)
|
|
|
|
.css("display", "none")
|
|
|
|
.css("margin", 0);
|
|
|
|
$("body").append(element);
|
|
|
|
let size = element.width();
|
|
|
|
element.detach();
|
|
|
|
return size;
|
2018-09-26 13:30:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
declare class webkitAudioContext extends AudioContext {}
|
|
|
|
declare class webkitOfflineAudioContext extends OfflineAudioContext {}
|
|
|
|
interface Window {
|
|
|
|
readonly webkitAudioContext: typeof webkitAudioContext;
|
|
|
|
readonly AudioContext: typeof webkitAudioContext;
|
|
|
|
readonly OfflineAudioContext: typeof OfflineAudioContext;
|
|
|
|
readonly webkitOfflineAudioContext: typeof webkitOfflineAudioContext;
|
|
|
|
readonly RTCPeerConnection: typeof RTCPeerConnection;
|
2018-09-30 19:50:59 +00:00
|
|
|
readonly Pointer_stringify: any;
|
2018-10-04 20:49:20 +00:00
|
|
|
readonly jsrender: any;
|
|
|
|
|
2018-10-02 16:38:58 +00:00
|
|
|
require(id: string): any;
|
2018-09-30 19:50:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
interface Navigator {
|
|
|
|
browserSpecs: {
|
|
|
|
name: string,
|
|
|
|
version: string
|
|
|
|
};
|
2018-02-27 16:20:49 +00:00
|
|
|
}
|