/*
* Web Inquirer V 0.7g
* Copyright 2016-2018 XWolfOverride@gmail.com
*
* Permission is hereby granted, free of charge, to any person obtaining a copy of this
* software and associated documentation files (the "Software"), to deal in the Software
* without restriction, including without limitation the rights to use, copy, modify,
* merge, publish, distribute, sublicense, and/or sell copies of the Software, and to
* permit persons to whom the Software is furnished to do so, subject to the following
* conditions:
*
* The above copyright notice and this permission notice shall be included in all copies
* or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED,
* INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A
* PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
* HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF
* CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE
* OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/
var inquirer = inquirer || new function () {
var icon = "", //
bugico = "", //
VERSION = "0.7g",//
alertIcon,
config = {
color: {
normal: "#000",
string: "#0A0",
number: "#00B",
keyword: "#809",
},
}; //
// Locals
var inq = this, inqapp;
// Types
function BasicError(data) {
this.merge(data);
}
// Tools
function uuid() {
return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function (c) {
var r = Math.random() * 16 | 0, v = c == 'x' ? r : (r & 0x3 | 0x8);
return v.toString(16);
});
}
function objectResumeLink(obj) {
var a = document.createElement("a"), isError;
a.href = "#";
if (obj instanceof Error) {
a.appendChild(document.createTextNode(obj));
isError = true;
}
else if (obj instanceof BasicError) {
a.appendChild(document.createTextNode(obj.message));
isError = true;
} else
a.appendChild(document.createTextNode(objectResumeText(obj)));
a.style.merge({
color: "black",
textDecoration: "none"
});
a.onclick = function () {
if (isError)
getApp().showError(obj);
else
inq.inspect(obj);
return false;
}
return a;
}
/** A formatter tool for console output */
function formatter(rtd) {
var r, p, rtp, i;
r = document.createElement("span");
for (i in rtd) {
rtp = rtd[i];
if (!rtp)
continue;
if (rtp instanceof Node)
r.appendChild(rtp);
else {
p = document.createElement("span");
p.data = rtp;
if (rtp.c)
p.style.color = rtp.c;
p.innerText = rtp.v;
r.appendChild(p);
}
}
return r;
}
/** expandable object resume */
function compactObject(m, d) {
var r, f, mn, dn;
r = document.createElement("span");
f = document.createElement("span");
mn = formatter(m);
function open() {
f.innerText = "▼ ";
if (r.contains(mn))
r.removeChild(mn);
if (dn == null) {
if (typeof d === "function")
d = d();
dn = formatter(d);
}
r.appendChild(dn);
f.onclick = close;
return r;
}
function close() {
f.innerText = "▶︎ ";
if (r.contains(dn))
r.removeChild(dn);
r.appendChild(mn);
f.onclick = open;
return r;
}
f.style.cursor = "pointer";
r.appendChild(f);
r.close = close;
r.open = open;
close();
return r;
}
/** formatice function details */
function functionDetails(f) {
var s, r = [];
if (f) {
s = f.toString().trim();
r.push({ c: config.color.keyword, v: "function " });
if (s.substr(0, 8) === "function")
s = s.substr(8).trim();
r.push({ c: config.color.normal, v: s });
}
return r;
}
/** formatice object resume */
function objectResume(obj) {
var r = [], k, max = 10, first = true;
if (Array.isArray(obj)) {
r.push({ c: config.color.normal, v: "Array (" + obj.length + ")" });
r.push(inspectorLink(obj));
} else {
if (obj.constructor)
r.push({ c: config.color.normal, v: functionName(obj.constructor) + " " });
r.push(inspectorLink(obj));
r.push({ c: config.color.normal, v: "{" });
for (k in obj) {
if (max-- <= 0) {
r.push({ c: config.color.normal, v: "..." });
break;
}
if (first)
first = false;
else
r.push({ c: config.color.normal, v: ", " });
r.push({ c: config.color.keyword, v: k });
}
r.push({ c: config.color.normal, v: "}" });
}
return r;
}
/** foramtice object details */
function objectDetails(obj) {
return function () {
var r = [], k, d;
if (Array.isArray(obj)) {
r.push({ c: config.color.normal, v: "Array (" + obj.length + ")" });
r.push(inspectorLink(obj));
r.push({ c: config.color.normal, v: "[" });
} else {
if (obj.constructor)
r.push({ c: config.color.normal, v: functionName(obj.constructor) + " " });
r.push(inspectorLink(obj));
r.push({ c: config.color.normal, v: "{" });
}
for (k in obj) {
r.push(d = document.createElement("div"));
d.appendChild(formatter([
{ c: config.color.keyword, v: k },
{ c: config.color.normal, v: ": " },
objectRow(obj[k])
]))
d.style.paddingLeft = "8px";
}
r.push({ c: config.color.normal, v: Array.isArray(obj) ? "]" : "}" });
return r;
};
}
/** return the function name */
function functionName(f) {
if (!f)
return "";
if (f.name)
return f.name;
var funcNameRegex, result;
funcNameRegex = /function (.{1,})\(/;
result = (funcNameRegex).exec(f.toString());//(this).constructor.toString());
result = (result && result.length > 1) ? result[1] : "";
if (result.length && result.length > 128)
result = "";
return result;
}
/** Advanced way to show object on screen */
function objectRow(obj, full, expanded) {
var row, robj;
row = document.createElement("span");
if (obj === null)
row.innerText = "<null>";
else if (obj === undefined)
row.innerText = "<undefined>";
else switch (typeof obj) {
case "string":
if (!full && obj.length > 512)
row.appendChild(formatter([
{ c: config.color.normal, v: '"' },
{ c: config.color.string, v: obj.substr(0, 512) },
{ c: config.color.normal, v: '..."' },
]))
else
row.appendChild(formatter([
{ c: config.color.normal, v: '"' },
{ c: config.color.string, v: obj },
{ c: config.color.normal, v: '"' },
]))
break;
case "number":
row.appendChild(formatter([
{ c: config.color.number, v: obj },
]));
break;
case "function":
row.appendChild(robj = compactObject([
{ c: config.color.keyword, v: "function" },
obj.name ? { c: config.color.normal, v: " " + functionName(obj) } : null,
{ c: config.color.normal, v: "() {...}" },
],
functionDetails(obj)
));
break;
case "object":
row.appendChild(robj = compactObject(objectResume(obj), objectDetails(obj)));
break;
default:
row.innerText = obj;
break;
}
if (expanded)
robj.open();
return row;
}
function inspectorLink(object) {
var lnk = document.createElement("IMG");
lnk.src = "";
lnk.style.merge({
verticalAlign: "middle",
cursor: "pointer",
});
lnk.onclick = function () {
openInspector(object);
};
return lnk;
}
// Methods
/**
* Error window handling
* @param {*string} eMessage Error message
* @param {*string} eData Error message extended information
*/
function openErrorBox(eMessage, eData) {
var wid, win, wwidth = 400, wheight = 150;
wid = "error";// + uuid();
win = getApp().windows[wid];
if (!win) {
win = merger.ui.window(wid, {
title: "Error",
width: wwidth,
height: wheight,
content: [merger.ui.picture("Iico", {
src: bugico,
top: 0,
left: 0,
width: 32,
height: 32,
}),
merger.ui.label("Lerror", {
top: 0,
left: 37,
width: wwidth - 37,
height: 32,
text: "",
}),
merger.ui.label("Tinfo", {
top: 37,
left: -5,
width: wwidth,
height: wheight - 37 - 25 - 10,
multiple: true,
style: {
whiteSpace: "pre",
fontFamily: "Lucida Console, Monospace",
border: "0",
overflow: "scroll",
background: "#EEE",
padding: "5px",
},
}),
merger.ui.button("Bok", {
top: wheight - 20,
left: wwidth - 35,
width: 35,
height: 20,
text: "Ok",
onClick: function (e) {
this.getWindow().close();
}
}),
],
onClose: function () {
this.hide();
//getApp().removeWindow(win);
},
load: function () {
win.content.Lerror.setText(eMessage);
win.content.Tinfo.setText(eData);
}
});
getApp().addWindow(win);
}
win.load();
win.show();
return win;
}
/**
* Get the console window
*/
function getConsole() {
var wid, win, wwidth = 300, wheight = 250, inputHeight = 20;
wid = "console";
win = getApp().windows[wid];
if (document.body.scrollWidth)
wwidth = Math.max(wwidth, document.body.scrollWidth - 200);
if (document.body.scrollHeight)
wheight = Math.max(wheight, document.body.scrollHeight - 150);
if (!win) {
win = merger.ui.window(wid, {
title: "Console",
width: wwidth,
height: wheight,
resizable: true,
content: [
merger.ui.html("Hconsole", {
top: 0,
left: 0,
width: wwidth,
height: wheight - (inputHeight - 5),
anchor: "TRBL",
style: {
overflow: "scroll",
},
}),
merger.ui.textbox("Tinput", {
top: wheight - (inputHeight - 5),
left: -5,
width: wwidth + 5,
height: inputHeight,
placeholder: "javascript code here",
anchor: "LBR",
//multiple: true,
style: {
whiteSpace: "pre",
fontFamily: "Lucida Console, Monospace",
border: "0",
overflow: "scroll",
background: "#EEE",
padding: "3px",
},
onkeydown: function (e) {
var t, stop;
switch (e.keyCode) {
case 13:
this.getWindow().exec(this.getText());
t = "";
stop = true;
break;
case 38:
t = this.getWindow().conhis.up();
stop = true;
break;
case 40:
t = this.getWindow().conhis.down();
stop = true;
break;
}
if (t != undefined) {
this.setText(t);
}
if (stop)
return false;
},
}),
],
conhis: {
data: [],
sel: 0,
add: function (code) {
var idx = this.data.indexOf(code);
if (idx >= 0)
this.data.splice(idx, 1);
this.data.push(code);
this.sel = this.data.length;
},
up: function () {
if (this.sel > 0)
this.sel--;
return this.data[this.sel];
},
down: function () {
if (this.sel < this.data.length)
this.sel++;
else return;
if (this.sel == this.data.length)
return "";
return this.data[this.sel];
}
},
onClose: function () {
this.hide();
//getApp().removeWindow(win);
},
log: function (type, object) {
var row = document.createElement("div");
var head = document.createElement("div");
var clear = document.createElement("div");
var d = document.createElement("div");
row.style.merge({
margin: "0 0 0 13px",
});
head.style.merge({
fontFamily: "Lucida Console, Monospace",
fontWeight: "bold",
float: "left",
margin: "0 0 0 -13px",
width: "13px",
});
d.style.merge({
whiteSpace: "pre-wrap",
float: "right",
fontFamily: "Lucida Console, Monospace",
border: "0",
width: "100%",
});
clear.style.merge({
clear: "both",
});
switch (type) {
case 'i':
d.innerText = object;
head.appendChild(document.createTextNode(">"));
head.style.color = "#88F";
d.style.paddingBottom = "3px";
break;
case 'o':
d.appendChild(objectRow(object));
d.style.borderBottom = "1px solid #EEE";
head.appendChild(document.createTextNode("<"));
head.style.color = "#DDD";
break;
case 'e':
d.appendChild(objectRow(object));
head.appendChild(document.createTextNode("!"));
head.style.color = "#F88";
d.style.background = "#FEE";
head.style.background = "#FEE";
break;
}
d.data = object;
row.appendChild(head);
row.appendChild(d);
row.appendChild(clear);
this.content.Hconsole.appendChild(row);
this.content.Hconsole.scrollTop = this.content.Hconsole.scrollHeight;
},
exec: function (code) {
if (code == null || typeof code != "string" || code.length < 1)
return;
this.conhis.add(code);
var obj;
this.log('i', code);
try {
obj = eval('(' + code + ')');
} catch (err) {
window.console.error(err);
return;
}
window.console.log(obj);
},
write: function () {
var k;
for (k in arguments)
this.log('o', arguments[k]);
},
writeError: function () {
var k;
for (k in arguments)
this.log('e', arguments[k]);
},
load: function () {
win.write("Inquirer " + VERSION + " console.");
}
});
getApp().addWindow(win);
win.load();
}
return win;
}
/**
* Open the console window
*/
function openConsole() {
var win = getConsole();
win.show();
return win;
}
/**
* Open the properties window
*/
function openProperties() {
var wid, win, wwidth = 175, wheight = 100;
wid = "preferences";
win = getApp().windows[wid];
if (!win) {
win = merger.ui.window(wid, {
title: "Preferences",
width: wwidth,
height: wheight,
content: [
merger.ui.checkbox("pref_autoshow", {
top: 0,
left: 3,
}),
merger.ui.label("pref_autoshow_lbl", {
top: 0,
left: 20,
text: "Open Inquirer on error"
}),
merger.ui.checkbox("pref_erricon", {
top: 15,
left: 3,
}),
merger.ui.label("pref_erricon_lbl", {
top: 15,
left: 20,
text: "Show small error indicator"
}),
merger.ui.button("pref_save", {
top: wheight - 20,
left: wwidth - 27,
text: "Ok",
onClick: function () {
inq.autoShow = this.getWindow().content.pref_autoshow.checked;
inq.alertIcon = this.getWindow().content.pref_erricon.checked;
this.getWindow().close();
}
}),
],
onShow: function () {
this.getWindow().content.pref_autoshow.checked = inq.autoShow;
this.getWindow().content.pref_erricon.checked = inq.alertIcon;
}
});
getApp().addWindow(win);
}
win.show();
return win;
}
/**
* Open inspector window
* @param {*} object Any object to inspect
*/
function openInspector(object) {
var wid, win, wwidth = 350, wheight = 400, panel;
wid = "inspector_" + uuid();
win = merger.ui.window(wid, {
title: "Inspector",
width: wwidth,
height: wheight,
resizable: true,
content: [
panel = merger.ui.html("panel", {
top: 0,
left: 0,
width: wwidth,
height: wheight - 25,
anchor: "TBRL",
style: {
overflow: "scroll",
},
}),
merger.ui.button("insp_close", {
top: wheight - 20,
left: wwidth - 27,
text: "Ok",
anchor: "BR",
onClick: function () {
this.getWindow().close();
}
}),
],
load: function () {
panel.appendChild(objectRow(object, true, true));
},
onClose: function () {
this.hide();
getApp().removeWindow(win);
}
});
getApp().addWindow(win);
win.load();
win.show();
return win;
}
/**
* Get Inquirer application
*/
function getApp() {
if (!inqapp)
inqapp = merger.app("inquirer", {
title: "Inquirer",
icon: icon,
appMenu: [
merger.ui.menuItem("app_preferences", {
text: "Preferences...",
icon: merger.media.createIcon('#EEE', '⚒'),
onClick: function () {
openProperties();
}
}),
],
menu: [
merger.ui.menuItem("app_tools", {
text: "Tools",
items: [merger.ui.menuItem("tool_console", {
text: "Console",
onClick: function () {
this.getApp().showConsole();
}
})
]
}),
merger.ui.menuItem("app_special", {
text: "Special",
items: [merger.ui.menuItem("special_reload", {
text: "Reload page",
onClick: function () {
location.reload(false);
}
}), merger.ui.menuItem("special_reload_full", {
text: "Reload page (ignore cache)",
onClick: function () {
location.reload(true);
}
}), merger.ui.menuItem("special_debugger", {
text: "Call browser debugger",
onClick: function () {
debugger;
}
})
]
})
],
windows: [
],
onLoad: function () {
},
onAbout: function () {
merger.dialogs.messageBox(this, "Inquirer v" + VERSION + " over Merger UI " + merger.version + " ©2016-2018 XWolf Override.<br>Debugger application layer for web pages. Useful for embedded browser debugging", "About Inquirer", null, this.icon, 100);
},
onFocus: function () {
removeIcon();
if (!getApp().getFocusedWindow())
openConsole();
},
showError: function (err) {
if (err instanceof Error)
openErrorBox(err.message, err.stack);
else
openErrorBox(err.message, "File: " + err.source + "\r\n\r\nAt: " + err.at);
},
showConsole: function () {
openConsole();
}
});
return inqapp;
}
/**
* Show a small yellow mark to indicate some error has been happend
*/
function showIcon() {
if (!alertIcon) {
alertIcon = document.createElement("DIV");
alertIcon.style.merge({
position: "absolute",
top: "0px",
right: "3px",
width: "2px",
height: "1px",
background: "gold",
borderBottom: "goldenRod",
});
document.body.appendChild(alertIcon);
}
alertIcon.style.display = "";
}
/**
* Hides a small yellow mark to indicate some error has been happend
*/
function removeIcon() {
if (!alertIcon)
return;
alertIcon.style.display = "none";
}
// === Public members
function show() {
getApp().show();
}
function hide() {
getApp().windows.Wmain.close();
}
// On error hook
function hookOnError() {
var formerOnError = window.onerror;
window.onerror = function (message, source, lineno, colno, err) {
inq.error(new BasicError({
message: message,
source: source,
at: lineno + ":" + colno,
error: err,
raw: arguments
}));
if (formerOnError)
formerOnError.apply(window, arguments);
}
}
// Konami code hook
function hookKonamiCode() {
var kc = [38, 38, 40, 40, 37, 39, 37, 39, 66, 65], kcindex = 0 // Konami code
document.addEventListener("keydown", function (e) {
if (e.keyCode == kc[kcindex])
kcindex++;
else
kcindex = 0;
if (kcindex == kc.length)
inquirer.show();
}, false);
}
// Console hook
function hookConsole() {
if (!window.console)
window.console = {};
function consoleOverride(method, funct) {
var old = window.console[method];
window.console[method] = function () {
funct.apply(window.console, arguments);
if (old)
old.apply(window.console, arguments);
};
window.console[method].old = old;
}
consoleOverride("log", function () {
openConsole().write.apply(openConsole(), arguments);
})
consoleOverride("info", function () {
openConsole().write.apply(openConsole(), arguments);
})
consoleOverride("debug", function () {
openConsole().write.apply(openConsole(), arguments);
})
consoleOverride("error", function () {
openConsole().writeError.apply(openConsole(), arguments);
})
}
// Install hooks
function hook() {
hookOnError();
hookKonamiCode();
hookConsole();
}
function error(error) {
getConsole().writeError(error);
if (this.autoShow) {
getApp().showError(error);
show();
}
if (this.alertIcon) {
showIcon();
}
}
function inspect(object) {
openInspector(object);
show();
}
function console() {
getApp().showConsole();
show();
}
function log() {
getApp().windows.Wconsole.write.apply(getApp().windows.Wconsole, arguments);
}
// Publish
this.merge({
version: VERSION,
autoShow: false, // Open Inquirer environment when a new unhandled error happen
alertIcon: true, // Small icon on screen to indicate if there is any new error
show: show, // Enters environment
hide: hide, // Exit environent
hook: hook, // Hook the webpage to control errors (Automatically hooked)
error: error, // Collect an error and if autoShow is true enters environment
inspect: inspect, // Enters environment and open variable inspector
console: console, // Enters environment and open inquirer console
log: log, // Log an object into the console
config: config, // Configuration object
});
// Hook the page now
hook();
}();