/*
* Web Inquirer V 0.7
* Copyright 2016 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.7c",//
alertIcon
; //
// Locals
var inq = this, inqapp;
// Tools
function objectResume(obj) {
var r, k, v;
if (obj === null)
return "<null>";
if (obj === undefined)
return "<undefined>";
switch (typeof obj) {
case "string":
if (obj.length < 200)
return '"' + obj + '"';
else
return '"' + obj.substr(0, 200) + "...";
case "function":
return "function";
case "object":
r = "";
if (obj instanceof HTMLElement)
r += "<" + obj.tagName + ">:: ";
else if (obj instanceof HTMLDocument)
r += "DOCUMENT:: ";
else if (obj instanceof Window)
r += "WINDOW:: ";
r += "{\n";
for (k in obj) {
r += k + ":" + objectResume(obj[k]) + ",\n";
if (r.length > 500) {
r += " ...";
break;
}
}
r += "}";
return r;
default:
return "" + obj;
}
}
function objectResumeLink(obj) {
var a = document.createElement("a");
a.href = "#";
a.onclick = function () {
inq.inspect(obj);
return false;
}
a.appendChild(document.createTextNode(objectResume(obj)));
a.style.merge({
color: "black",
textDecoration: "none"
});
return a;
}
function errorLink(obj) {
var a = document.createElement("a");
a.href = "#";
a.onclick = function () {
getApp().showError(obj);
return false;
}
if (obj instanceof Error)
a.appendChild(document.createTextNode(obj));
else
a.appendChild(document.createTextNode(obj.message));
a.style.merge({
color: "black",
textDecoration: "none"
});
return a;
}
// Methods
function getApp() {
var
win_err_width = 400,
win_err_height = 150;
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 () {
this.getApp().windows.wPref.show();
}
}),
],
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: [merger.ui.window("Wmain", {
title: "Error",
width: win_err_width,
height: win_err_height,
visible: false,
content: [merger.ui.picture("Iico", {
src: bugico,
top: 0,
left: 0,
width: 32,
height: 32,
}),
merger.ui.label("Lerror", {
top: 0,
left: 37,
width: win_err_width - 37,
height: 32,
text: "",
}),
merger.ui.label("Tinfo", {
top: 37,
left: -5,
width: win_err_width,
height: win_err_height - 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: win_err_height - 20,
left: win_err_width - 35,
width: 35,
height: 20,
text: "Ok",
onClick: function (e) {
this.getWindow().close();
}
}),
],
onClose: function () {
this.hide();
//merger.leave();
},
setError: function (message, data) {
this.content.Lerror.setText(message);
this.content.Tinfo.setText(data);
}
}), merger.ui.window("Wconsole", {
title: "Console",
width: 500,
height: 350,
visible: false,
content: [
merger.ui.html("Hconsole", {
top: 0,
left: 0,
width: 500,
height: 330,
style: {
overflow: "scroll",
},
}),
merger.ui.textbox("Tinput", {
top: 330,
left: -5,
width: 500 + 10,
height: 25,
placeholder: "javascript code here",
//multiple: true,
style: {
whiteSpace: "pre",
fontFamily: "Lucida Console, Monospace",
border: "0",
overflow: "scroll",
background: "#EEE",
padding: "5px",
},
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();
},
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(objectResumeLink(object));
d.style.borderBottom = "1px solid #EEE";
head.appendChild(document.createTextNode("<"));
head.style.color = "#DDD";
break;
case 'e':
d.appendChild(errorLink(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 err, 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]);
}
}),
merger.ui.window("wPref", {
title: "Preferences",
width: 175,
height: 100,
visible: false,
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: 80,
left: 150,
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;
}
}),
],
onLoad: function () {
},
onAbout: function () {
merger.dialogs.messageBox(this, "Inquirer v" + VERSION + " ©2016-2017 XWolf Override.<br>Debugger application layer for web pages. Useful for embedded browser debugging", "About Inquirer", null, this.icon, 100);
},
onFocus: function () {
removeIcon();
if (!inq.autoShow)
this.windows.Wconsole.show();
},
showError: function (err) {
if (err instanceof Error) {
this.windows.Wmain.setError(err.message, err.stack);
} else {
this.windows.Wmain.setError(err.message, "File: " + err.source + "\r\n\r\nAt: " + err.at);
}
this.windows.Wmain.show();
},
showConsole: function () {
this.windows.Wconsole.show();
}
});
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({
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() {
var wcon = getApp().windows.Wconsole;
if (!window.console)
window.console = {};
function consoleOverride(method, funct) {
var old = window.console[method];
window.console[method] = function () {
funct.apply(this, arguments);
if (old)
old.apply(this, arguments);
};
window.console[method].old = old;
}
consoleOverride("log", function () {
wcon.write.apply(wcon, arguments);
})
consoleOverride("debug", function () {
wcon.write.apply(wcon, arguments);
})
consoleOverride("error", function () {
wcon.writeError.apply(wcon, arguments);
})
}
// Install hooks
function hook() {
hookOnError();
hookKonamiCode();
hookConsole();
}
function error(error) {
getApp().windows.Wconsole.writeError(error);
if (this.autoShow) {
getApp().showError(error);
show();
}
if (this.alertIcon) {
showIcon();
}
}
function inspect(object) {
}
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
});
// Hook the page now
hook();
window.addEventListener("load", function () {
getApp().windows.Wconsole.write("Inquirer " + VERSION + " console.");
})
}();