Mapeate/Public/ecma/Mapeate.ecma.js

717 lines
28 KiB
JavaScript

Mapeate = function(entradas){
const self = this,
configuracion_por_defecto = {
nombre_objeto : "mapeate",
nulos : false,
valor_por_defecto : null,
autoiniciar : true,
archivos_de_configuracion_por_defecto : [
"/json/Mapeate.configuracion.json",
"/json/Mapeate.configuracion.secretos.json"
],
texto_por_defecto : "",
configuracion_sobreescribir : false,
i18n_sobreescribir : false,
formato_terminal : "[{tipo}] {yyyy}{mm}{dd} {hh}{ii}{ss} [{linea}]{archivo}({metodo}): {mensaje}",
terminal_tipos_por_defecto : [
["unkn", "unknown", "desconocido"],
[" ok ", "ok", "si", "correcto"],
["info", "informacion", "information"],
["note", "nota", "comment", "comentario"],
["warn", "warning", "aviso", "advertencia"],
["erro", "error", "incrrecto"],
["fail", "failure", "fallo"],
["exce", "excepcion"]
],
archivos_de_i18n_por_defecto : [
"/json/i18n/Mapeate.i18n.espanol.json"
],
ajax_timeout : 2000,
terminal_unkn_oscuro : "color : #AAA;",
terminal_unkn_claro : "color : #666;",
terminal_ok_oscuro : "color : #9F9;",
terminal_ok_claro : "color : #090;",
terminal_info_oscuro : "color : #99F;",
terminal_info_claro : "color : #009;",
terminal_note_oscuro : "color : #222;",
terminal_note_claro : "color : #EFEFEF;"
},
configuracion = {},
textos = {
espanol : {
mapeate : "Mapeate",
mapeate_construyendose : "La aplicación 'Mapéate' se está construyendo...",
mapeate_construido : "La aplicación 'Mapeate' se construyó completamente.",
mapeate_iniciando : "La aplicación 'Mapeate' se está iniciando...",
cargando_configuracion : "Cargando la configuración...",
configuracion_cargada : "La configuración ha sido cargada completamente.",
cargando_terminal_tipos : "Cargando los tipos de salidas de terminal...",
terminal_tipos_cargados : "Los tipos de terminal han sido cargados completamente.",
cargando_i18n : "Cargando los textos internacionalizados...",
i18n_cargada : "Los textos internacionalizados han sido cargados completamente.",
mapeate_vistas_construyendose : "El módulo de gestión de vistas de la aplicación 'Mapeatea' se está construyendo...",
mapeate_vistas_construido : "El módulo de gestión de vistas de la aplicación 'Mapeatea' se construyó completamente.",
mapeate_mapas_construyendose : "El módulo de gestión de mapas de la aplicación 'Mapeatea' se está construyendo...",
mapeate_mapas_construido : "El módulo de gestión de mapas de la aplicación 'Mapeatea' se construyó completamente.",
mapeate_mapas_selector_construyendose : "El módulo de gestión de selección de mapas de la aplicación 'Mapeatea' se está construyendo...",
mapeate_mapas_selector_construido : "El módulo de gestión de selección de mapas de la aplicación 'Mapeatea' se construyó completamente.",
mapeate_base_construyendose : "El módulo de gestión básica de DPIs de la aplicación 'Mapeatea' se está construyendo...",
mapeate_base_construido : "El módulo de gestión básica de DPIs de la aplicación 'Mapeatea' se construyó completamente."
}
},
terminal_tipos = [],
hashes = [],
hilos_de_proceso = [];
let iniciado = false,
configuracion_sobreescribir = false,
i18n_sobreescribir = false,
idioma_por_defecto = null,
idioma = null,
modo_dispositivo = "por_defecto",
modo_gui = "por_defecto",
terminal_unkn_oscuro, terminal_unkn_claro,
terminal_ok_oscuro, terminal_ok_claro,
terminal_info_oscuro, terminal_info_claro,
terminal_note_oscuro, terminal_note_claro,
modo_gui_oscuro = window.matchMedia && window.matchMedia("(prefers-color-scheme: dark)").matches,
formato_terminal = "[{tipo}] {yyyy}{mm}{dd} {hh}{ii}{ss} [{linea}]{archivo}({metodo}): {mensaje}",
hash_alfabeto, hash_longitud,
intervalo_de_procesos, precarga_timeout,
construido = false,
claves_padre_por_defecto = ["tagName", "nodeName"];
let hash_self = this.hash_self;
let si_mismo = this.si_mismo = document;
let nombre_objeto = this.nombre_objeto;
let vistas = this.vistas;
let mapas = this.mapas;
let base = this.base;
const re_bloque_de_traza = new RegExp("^(" + [
/\s*at\s+(([^\s]+)\s+\()?(([^\(\)\:]+\:)?[^\(\)\:]+)(\:([0-9]+)\:[0-9]+)?\)?/.source, // Webkit
/([^\@]+)\@([^:]+\:[^\:]+)\:([0-9]+)\:[0-9]+/.source, // Gecko
].join("|") + ")$");
const constructor = () => {
nombre_objeto = self.nombre_objeto = self.configuracion("nombre_objeto");
configuracion_por_defecto.terminal_tipos_por_defecto.forEach(entrada => terminal_tipos.push(entrada));
establecer_variables_comunes();
self.print("info", "mapeate_construyendose");
Mapeate.Vistas && (vistas = this.vistas = new Mapeate.Vistas(self, entradas));
Mapeate.Mapas && (mapas = this.mapas = new Mapeate.Mapas(self, entradas));
Mapeate.Base && (base = this.base = new Mapeate.Base(self, entradas));
web = this.web = Mapeate.Web ? new Mapeate.Web(self, entradas) : null;
self.print("ok", "mapeate_construido");
self.configuracion("autoiniciar") && self.iniciar();
};
const establecer_variables_comunes = () => {
configuracion_sobreescribir = self.configuracion(["configuracion_sobreescribir", "sobreescribir"]);
formato_terminal = self.configuracion("formato_terminal");
terminal_unkn_oscuro = self.configuracion("terminal_unkn_oscuro");
terminal_unkn_claro = self.configuracion("terminal_ounkn_claro");
terminal_ok_oscuro = self.configuracion("terminal_ok_oscuro");
terminal_ok_claro = self.configuracion("terminal_ok_claro");
terminal_info_oscuro = self.configuracion("terminal_info_oscuro");
terminal_info_claro = self.configuracion("terminal_info_claro");
terminal_note_oscuro = self.configuracion("terminal_note_oscuro");
terminal_note_claro = self.configuracion("terminal_note_claro");
};
this.ejecutar = (callback, ...argumentos) => typeof callback == "function" && callback(...argumentos);
this.iniciar = callback => {
const terminar = estado => self.ejecutar(callback, estado);
self.print("info", "mapeate_iniciando");
if(iniciado){
self.print("warn", "mapeate_ya_iniciado");
terminar(false);
return false;
};
iniciado = true;
establecer_variables_comunes();
self.print("info", "cargando_configuracion");
self.ejecutar_array_asincrono([
"archivos_de_configuracion_por_defecto",
"archivos_de_configuracion"
], (clave, callback) => self.configuracion_anadir(self.configuracion(clave), true, () => {
establecer_variables_comunes();
self.ejecutar(callback)
}), () => {
self.print("ok", "configuracion_cargada");
self.print("info", "cargando_terminal_tipos");
self.ejecutar_array_asincrono([
"terminal_tipos_por_defecto",
"terminal_tipos"
], (clave, callback) => self.terminal_tipos_anadir(self.configuracion(clave), callback), () => {
self.print("ok", "terminal_tipos_cargados");
const nuevas_claves_padre = self.coger_claves(self.configuracion(("claves_padre_por_defecto", "claves_padre")));
i18n_sobreescribir = self.configuracion(["i18n_sobreescribir", "sobreescribir"]);
idioma_por_defecto = self.configuracion(["idioma_por_defecto", "idioma"]);
idioma = self.configuracion(["idioma", "idioma_por_defecto"]);
hash_alfabeto = self.configuracion("hash_alfabeto");
hash_longitud = self.configuracion("hash_longitud");
precarga_timeout = self.configuracion("precarga_timeout");
nuevas_claves_padre.length && (claves_padre_por_defecto = nuevas_claves_padre);
intervalo_de_procesos = setInterval(ejecutar_hilos_de_proceso, 1000 / self.configuracion("fotogramas_por_segundo"));
self.print("info", "cargando_i18n");
self.ejecutar_array_asincrono([
"archivos_de_i18n_por_defecto",
"archivos_de_i18n"
], (clave, callback) => self.i18n_anadir(self.configuracion(clave), true, callback), () => {
self.print("ok", "i18n_cargada");
self.ejecutar_array_asincrono(web ? [vistas, base, web] : [vistas, mapas, base], (elemento, callback) => elemento.iniciar(callback), () => {
self.print("ok", "mapeate_iniciado");
terminar(true);
});
});
});
});
return true;
};
this.nulos = nulos => typeof nulos != "boolean" ? self.configuracion("nulos", null, false, false) : nulos;
this.valor_por_defecto = (valor_por_defecto, nulos) => (
valor_por_defecto !== undefined && (self.nulos(nulos) || valor_por_defecto !== null) ? valor_por_defecto :
self.configuracion("valor_por_defecto", null, null, true)
);
this.coger_array = valor => valor && typeof valor == "object" && valor instanceof Array ? valor : [valor];
this.coger_array_de_diccionarios = valor => valor && typeof valor == "object" ? valor instanceof Array ? valor : [valor] : [];
this.coger_array_de_strings = valor => (
valor && typeof valor == "object" && valor instanceof Array ? valor : [valor]
).filter(elemento => typeof elemento == "string" && elemento.trim()).map(elemento => elemento.trim());
this.coger_claves = valor => self.coger_array_de_strings(valor).filter(elemento => elemento);
this.configuracion = (claves, entradas, valor_por_defecto, nulos) => {
const m = (claves = self.coger_array_de_strings(claves)).length;
if(m){
const l = (entradas = [].concat(
self.coger_array_de_diccionarios(entradas),
[entradas, configuracion, configuracion_por_defecto]
)).length;
nulos = self.nulos(nulos);
for(let i = 0; i < l; i ++)
if(entradas[i] && typeof entradas[i] == "object")
for(let j = 0; j < m; j ++)
if(entradas[i][claves[j]] !== undefined && (nulos || entradas[i][claves[j]] !== null))
return entradas[i][claves[j]];
};
return self.valor_por_defecto(valor_por_defecto, nulos);
};
this.leer_archivo = (url, entradas) => {
let terminado = false;
const ajax = new XMLHttpRequest(),
callback = self.configuracion("callback",
typeof entradas == "function" ? (entradas = {callback : entradas}) :
entradas && typeof entradas == "object" ? entradas :
{}),
timeout = self.configuracion(["ajax_timeout", "timeout"], entradas),
terminar = codigo => (
!terminado &&
(terminado = true) &&
self.ejecutar(callback, ajax.responseText, ajax.status, ajax.readyState, codigo)
),
fecha = Date.now();
ajax.open("get", url, true);
ajax.timeout = timeout;
ajax.onreadystatechange = () => {
if(terminado)
return;
if(ajax.readyState == 4)
terminar(
(ajax.status >= 200 && ajax.status < 300) || [301, 302, 304].includes(ajax.status) ? "OK" :
"HTTP_ERROR");
else if(Date.now() - fecha > timeout)
terminar("TIMEOUT_FORZADO");
};
ajax.send(null);
ajax.onabort = () => terminar("ABORTADO");
ajax.onerror = () => terminar("ERROR");
ajax.ontimeout = () => terminar("TIMEOUT");
return ajax;
};
const ejecutar_array_asincrono = (elementos, callback_parcial, callback_final, i) => {
if(!elementos || i >= elementos.length){
self.ejecutar(callback_final);
return;
};
const terminar = () => ejecutar_array_asincrono(elementos, callback_parcial, callback_final, i + 1);
if(!elementos[i]){
terminar();
return;
};
if(typeof elementos[i] == "object" && elementos[i] instanceof Array)
ejecutar_array_asincrono(elementos[i], terminar, 0);
else
self.ejecutar(callback_parcial, elementos[i], terminar);
};
this.ejecutar_array_asincrono = (elementos, callback_parcial, callback_final) => ejecutar_array_asincrono(
typeof elementos == "object" && elementos instanceof Array ? elementos : [elementos],
callback_parcial,
callback_final,
0
);
this.procesar_json = (string, si, no) => {
let json;
try{
json = JSON.parse(string);
}catch(excepcion){};
if(json){
typeof si == "function" && si(json);
return true;
};
typeof no == "function" && no(null);
return false;
};
this.cargar_diccionarios = (entradas, callback_parcial, callback_final, i) => {
if(!entradas || (i || (i = 0)) >= entradas.length){
self.ejecutar(callback_final);
return;
};
const terminar = () => self.cargar_diccionarios(entradas, callback_parcial, callback_final, i + 1);
if(!entradas[i])
terminar();
else if(typeof entradas[i] == "string")
self.procesar_json(entradas[i], json => self.cargar_diccionarios(self.coger_array(json), callback_parcial, terminar, 0), () => {
self.leer_archivo(entradas[i], respuesta => self.procesar_json(respuesta, json => self.cargar_diccionarios(self.coger_array(json), callback_parcial, terminar, 0), terminar));
});
else if(typeof entradas[i] == "object"){
if(entradas[i] instanceof Array)
self.cargar_diccionarios(entradas[i], callback_parcial, terminar, 0);
else
self.ejecutar(callback_parcial, entradas[i], terminar);
}else
terminar();
};
this.configuracion_anadir = (entradas, sobreescribir, callback) => {
typeof sobreescribir == "boolean" || (sobreescribir = configuracion_sobreescribir);
self.cargar_diccionarios(self.coger_array(entradas), (nueva_configuracion, callback) => {
for(const clave in nueva_configuracion)
(sobreescribir || configuracion[clave] === undefined) && (configuracion[clave] = nueva_configuracion[clave]);
self.ejecutar(callback);
}, callback, 0);
};
const formatear_parametro = (clave, string) => ' ' + ([
"onclick", "onmouseover", "onmouseout", "onmousedown", "onmouseup", "onmousemove"
].includes(clave) ? '' : 'data-') + clave.replace(/[^a-z0-9]+/g, '-') + '="' + ("" + string).replace(/"/g, "\\\"") + '"';
this.string_variables = (string, variables, por_defecto) => {
const l = (variables = self.coger_array_de_diccionarios(variables)).length,
valores_por_defecto = {
nombre_objeto : nombre_objeto,
hash_self : hash_self
};
return (string || "" + string).replace(/(nombre_objeto)/g, "{$1}").replace(/\{{2}([^\{\}]+)\}{2}|\{([^\{\}]+)\}| +data-([^=\s]+)="\[{2}\]{2}"/g, (crudo, i18n, clave, parametros) => {
if(i18n){
let texto = i18n;
const argumentos = i18n.match(/^(((?!(_descripcion)).)+)(_descripcion)?$/),
[clave, descripcion] = argumentos ? [argumentos[1], argumentos[4]] : [i18n, null];
for(let i = 0; i < l; i ++)
if(typeof variables[i][clave] == "string" && /^[a-z0-9_]+$/i.test(variables[i][clave])){
texto = variables[i][clave];
break;
};
return self.i18n(texto + (descripcion || ""), variables);
}if(parametros){
for(let i = 0; i < l; i ++)
switch(typeof variables[i][parametros]){
case "object":
let html = '';
for(const clave in variables[i][parametros])
html += formatear_parametro(clave, variables[i][parametros][clave]);
return html;
case "string":
default:
if(variables[i][parametros] !== undefined)
return formatear_parametro(parametros, variables[i][parametros])
};
return "";
};
for(let i = 0; i < l; i ++)
if(variables[i][clave] !== undefined)
return variables[i][clave];
if(valores_por_defecto[clave] !== undefined)
return valores_por_defecto[clave];
return por_defecto === undefined ? crudo : por_defecto;
});
};
const i18n = (claves, por_defecto) => {
const m = (claves = self.coger_array_de_strings(claves)).length,
idiomas = [idioma_por_defecto, idioma].concat(Object.keys(textos)),
l = idiomas.length;
for(let i = 0; i < l; i ++)
if(typeof idiomas[i] == "string" && textos[idiomas[i]])
for(let j = 0; j < m; j ++)
if(textos[idiomas[i]][claves[j]] !== undefined)
return textos[idiomas[i]][claves[j]];
return (
por_defecto !== undefined ? por_defecto :
m ? claves[0] :
self.configuracion("texto_por_defecto"));
};
this.i18n = (claves, variables, por_defecto) => self.string_variables(i18n(claves, por_defecto), variables);
this.i18n_anadir = (entradas, sobreescribir, callback) => {
typeof sobreescribir == "boolean" || (sobreescribir = i18n_sobreescribir);
self.cargar_diccionarios(self.coger_array_de_strings(entradas), (nuevos_textos, callback) => {
for(const idioma in nuevos_textos){
textos[idioma] || (textos[idioma] = {});
if(nuevos_textos[idioma] && typeof nuevos_textos[idioma] == "object")
for(const clave in nuevos_textos[idioma])
(sobreescribir || textos[idioma][clave] === undefined) && (textos[idioma][clave] = nuevos_textos[idioma][clave]);
};
self.ejecutar(callback)
}, callback, 0);
};
const terminal_tipos_anadir = (entradas, callback, i, clave) => {
if(i > 10)
return;
if(!entradas || i >= entradas.length){
typeof callback == "function" && callback();
return;
};
const terminar = () => terminal_tipos_anadir(entradas, callback, i + 1, clave);
if(!entradas[i]){
terminar();
return;
};
if(typeof entradas[i] == "string"){
if(/^ *[a-z0-9]+ *$/i.test(entradas[i])){
if(clave === undefined && !terminal_tipos.some((tipos, j) => tipos.includes(entradas[i]) && !isNaN(clave = j))){
clave = terminal_tipos.length;
terminal_tipos.push([entradas[i]]);
}else
!terminal_tipos[clave].includes(entradas[i]) && terminal_tipos[clave].push(entradas[i]);
terminar();
return;
};
const procesar = string => self.procesar_json(
string,
json => terminal_tipos_anadir(json instanceof Array ? json : [json], terminar, 0),
terminar
);
if(/^(\[(.|[\r\n])+\])$/.test(entradas[i]))
procesar(entradas[i]);
else
self.leer_archivo(entradas[i], procesar);
}else if(typeof entradas[i] == "object" && entradas[i] instanceof Array)
terminal_tipos_anadir(entradas[i], terminar, 0);
else
terminar();
};
this.es_dispositivo_movil = () => modo_dispositivo == "movil" || (modo_dispositivo != "escritorio" && es_movil);
this.modo_gui_oscuro = () => modo_gui == "oscuro" || (modo_gui != "claro" && modo_gui_oscuro);
this.terminal_tipos_anadir = (entradas, callback) => terminal_tipos_anadir(entradas, callback, 0);
this.coger_id_tipo_terminal = tipo => {
const l = terminal_tipos.length;
if(typeof tipo == "string" && (tipo = tipo.trim().toLowerCase()))
for(let i = 0; i < l; i ++)
if(terminal_tipos[i].includes(tipo))
return i;
return 0;
};
this.coger_tipo_terminal = tipo => terminal_tipos[self.coger_id_tipo_terminal(tipo)][0].toUpperCase();
this.coger_traza = i => (new Error()).stack.replace(/^Error\s*?[\r\n]+/, "").trim().split(/[\r\n]+/).slice(1 + (i || 0)).map(linea => {
const matches = linea.match(re_bloque_de_traza);
return matches ? {
archivo : matches[4] || matches[9],
metodo : matches[3] || matches[8] || "",
linea : Number(matches[7] || matches[10])
} : null;
}).filter(linea => linea);
this.print = (tipo, mensaje, variables, i) => {
const fecha = new Date(),
conjunto = {
tipo_crudo : tipo,
tipo : self.coger_tipo_terminal(tipo),
i18n : mensaje,
...(typeof variables == "object" ? variables : {}),
...(self.coger_traza(isNaN(i) ? 1 : i)[0] || {
linea : -1,
metodo : "UNKNOWN",
archivo : "UNKNOWN"
})
};
let mensaje_procesado;
conjunto.line = conjunto.linea;
conjunto.method = conjunto.metodo;
conjunto.file = conjunto.archivo;
["year", "month", "date", "hours", "minutes", "seconds"].forEach(clave => {
let k = clave.substring(0, 1);
const valor = fecha["get" + (clave == "year" ? "FullYear" : k.toUpperCase() + clave.substring(1))]();
conjunto[clave == "date" ? "day" : clave] = valor;
conjunto[clave == "minutes" ? k = "i" : k] = valor;
conjunto[k + k] = ("00" + valor).slice(-2);
});
conjunto["yyyy"] = conjunto["year"];
conjunto.mensaje = self.i18n(mensaje, conjunto);
mensaje_procesado = self.string_variables(formato_terminal, conjunto);
switch(conjunto.tipo){
case " OK ":
console.log("%c" + mensaje_procesado, self.modo_gui_oscuro() ? terminal_ok_oscuro : terminal_ok_claro);
break;
case "INFO":
console.log("%c" + mensaje_procesado, self.modo_gui_oscuro() ? terminal_info_oscuro : terminal_info_claro);
break;
case "UNKN":
console.log("%c" + mensaje_procesado, self.modo_gui_oscuro() ? terminal_note_oscuro : terminal_note_claro);
break;
case "ERRO":
case "FAIL":
case "EXCE":
console.error(mensaje_procesado);
break;
case "WARN":
console.warn(mensaje_procesado);
break;
case "UNKN":
default:
console.log("%c" + mensaje_procesado, self.modo_gui_oscuro() ? terminal_unkn_oscuro : terminal_unkn_claro);
break;
};
};
this.hash = () => {
let hash;
const l = hash_alfabeto.length;
do{
hash = "";
while((hash += hash_alfabeto[Math.random() * l >> 0]).length < hash_longitud);
}while(
hashes.includes(hash) ||
/^[0-9]/.test(hash) ||
document.querySelector("#" + hash + ",." + hash + ",[name=" + hash + "]")
);
hashes.push(hash);
return hash;
};
const ejecutar_hilos_de_proceso = () => hilos_de_proceso.forEach(hilo => hilo && hilo());
this.anadir_hilo_de_proceso = metodo => {
if(typeof metodo != "function")
return null;
let i = 0;
const l = hilos_de_proceso.length;
for(; i < l; i ++)
if(!hilos_de_proceso[i])
break;
hilos_de_proceso[i] = metodo;
return i;
};
this.eliminar_hilo_de_proceso = i => !isNaN(i) && hilos_de_proceso[i] && (hilos_de_proceso[i] = null);
this.precargar = (selector, callback) => {
const terminar = elemento => {
if(typeof callback == "function")
try{
callback(elemento);
}catch(exception){
console.error(exception);
};
};
if(!selector)
terminar(null);
else if(selector.nodeName || selector.tagName)
terminar(selector);
else if(typeof selector == "string" && (selector = selector.trim())){
let elemento;
try{
if(elemento = si_mismo.querySelector(selector)){
terminar(elemento);
return;
};
}catch(excepcion){
terminar(null);
return;
};
const fecha = Date.now(),
precarga = self.anadir_hilo_de_proceso(() => {
if(elemento = si_mismo.querySelector(selector)){
self.eliminar_hilo_de_proceso(precarga);
terminar(elemento);
}else if(Date.now() - fecha > precarga_timeout){
self.eliminar_hilo_de_proceso(precarga);
terminar(null);
};
});
}else
terminar(null);
};
this.coger_hash = () => hash_self || (hash_self = self.hash_self = mapeate.hash());
this.establecer_si_mismo = elemento => (!si_mismo.constructor || si_mismo.constructor.name == "HTMLDocument") && (si_mismo = self.si_mismo = elemento);
this.porcentuar = (x1, y1, x2, y2, cubrir) => {
const r1 = x1 / y1,
r2 = x2 / y2;
if((cubrir && r1 > r2) || (!cubrir && r1 < r2)){
const x = x1 * y2 / y1;
return [(x2 - x) / 2, 0, x, y2];
};
const y = y1 * x2 / x1;
return [0, (y2 - y) / 2, x2, y];
};
this.consola = (texto, variables) => {
si_mismo && (si_mismo.querySelector(".juego .consola section").appendChild(document.createElement("div")).innerText = self.i18n(texto, variables));
self.print("test", texto, variables);
};
this.coger_padre = (elemento, objetivo, atributos) => {
(atributos = self.coger_claves(atributos)).length ||
(atributos = claves_padre_por_defecto);
objetivo = objetivo.toLowerCase();
while(elemento && elemento.hasAttribute && !atributos.some(atributo => elemento[atributo].toLowerCase() == objetivo) && (elemento = elemento.parentNode));
return (elemento && elemento.hasAttribute && elemento) || null;
};
this.alerta = (mensaje, callback, datos) => base.abrir_dialogo(mensaje, datos, [
{i18n : "aceptar", accion : () => {
base.ocultar_dialogo();
self.ejecutar(callback);
}}
]);
this.confirmar = (mensaje, si, no, datos) => base.abrir_dialogo(mensaje, datos, [
{i18n : "aceptar", accion : () => {
base.ocultar_dialogo();
self.ejecutar(si);
}},
{i18n : "cancelar", accion : () => {
base.ocultar_dialogo();
self.ejecutar(no);
}}
]);
constructor();
};