// ==UserScript==
// @name frisch's UserScript Extender
// @namespace http://null.frisch-live.de/
// @version 0.60
// @description Extends the document with a new namespace for user script crosswide functions to utilize. Has no use alone but can be accessed by other scripts using document.fExt
// @author frisch
// @include *
// @require http://ajax.googleapis.com/ajax/libs/jquery/1/jquery.min.js
// @grant GM_setClipboard
// ==/UserScript==
if(document.fExt === undefined) {
console.log("Initializing frisch's UserScript Extender...");
// Initialization
fExt = {};
fExt.jq = $;
// Overwrites the contains expression to be case-insensitive
fExt.jq.expr[':'].contains = function(a, i, m) {
return fExt.jq(a).text().toUpperCase().indexOf(m[3].toUpperCase()) >= 0;
};
fExt.settings = {
minShowTime: 3000, // in milliseconds
maxShowTime: 9000, // in milliseconds
position: "BottomRight", // Position for the popup menu: TopRight, TopLeft, BottomRight, BottomLeft
animationType: "fade", // Animation for the popup menu: slide, fade, none
animationSpeed: 500,
customContextMenu: true,
hideContextMenuOnLeave: true, // automatically fades out the custom context menu when it loses focus, otherwhise only hides when you click somewhere
progressType: "progressbar", //Possible values: hourglass, progressbar
toleranceX: 20, // pixels as tolerance for the context menu, adjust the position slightly
toleranceY: 20, // pixels as tolerance for the context menu, adjust the position slightly
};
fExt.popupQueue = [];
fExt.popping = false;
var hourGlass0 = "";
var hourGlass25 = "";
var hourGlass50 = "";
var hourGlass75 = "";
var hourGlass100 = "";
var pbar0 = "";
var pbar25 = "";
var pbar50 = "";
var pbar75 = "";
var pbar100 = "";
var loader0, loader25, loader50, loader75, loader100;
switch(fExt.settings.progressType){
case "progressbar":
loader0 = pbar0;
loader25 = pbar25;
loader50 = pbar50;
loader75 = pbar75;
loader100 = pbar100;
break;
case "hourglass":
loader0 = hourGlass0;
loader25 = hourGlass25;
loader50 = hourGlass50;
loader75 = hourGlass75;
loader100 = hourGlass100;
break;
default:
loader0 = hourGlass0;
loader25 = hourGlass25;
loader50 = hourGlass50;
loader75 = hourGlass75;
loader100 = hourGlass100;
break;
}
// Custom Elements
fExt.fExtPopup = $('<div id="fExtPopup" class="fExtElement" style="display:none; position:fixed; width:auto; height:auto; background-color:#545454; padding:10px; border-color:white; border-style:groove; color:white; z-index:10000; text-align:center; font-size:12px;"></div>');
$("body").append(fExt.fExtPopup);
fExt.fExtMessage = $('<div id="fExtMessage" class="fExtElement" style="display:none; position:fixed; width:auto; height:auto; background-color:#545454; padding:25px; border-color:white; border-style:groove; color:white; z-index:10000; text-align:center; font-size:large; padding:20px;"></div>');
fExt.fExtMessageText = $('<span style="display: inline-block;"></span>');
fExt.fExtMessageText.appendTo(fExt.fExtMessage);
fExt.fExtMessage.appendTo("body");
// Functions
fExt.setLoading = function(percentage){
if(percentage >= 100){
$("link[rel='icon']").attr("href", loader100);
$("link[rel='shortcut icon']").attr("href", loader100);
setTimeout(function(){ fExt.setLoading(-1); },fExt.settings.minShowTime);
}
else if(percentage >= 75){
$("link[rel='icon']").attr("href", loader75);
$("link[rel='shortcut icon']").attr("href", loader75);
}
else if(percentage >= 50){
$("link[rel='icon']").attr("href", loader50);
$("link[rel='shortcut icon']").attr("href", loader50);
}
else if(percentage >= 25){
$("link[rel='icon']").attr("href", loader25);
$("link[rel='shortcut icon']").attr("href", loader25);
}
else if(percentage >= 0){
$("link[rel='icon']").attr("href", loader0);
$("link[rel='shortcut icon']").attr("href", loader0);
}
else {
$("link[rel='icon']").attr("href", siteFavIconHref);
$("link[rel='shortcut icon']").attr("href", siteFavIconHref);
}
};
fExt.enqueuePopup = function(msg) {
fExt.popupQueue.push(msg);
};
fExt.dequeuePopup = function() {
var rv;
if(fExt.popupQueue.length > 0){
rv = fExt.popupQueue[0];
fExt.popupQueue.splice(0, 1);
}
return rv;
};
fExt.createStyle = function(newClass) {
$( "<style>" + newClass + "</style>" ).appendTo("head");
};
fExt.center = function (element,w) {
if(w !== undefined)
element.width(w);
element.css("position", "fixed")
.css("top", (($(window).height() - $(element).outerHeight()) / 2) + "px")
.css("left", (($(window).width() - $(element).outerWidth()) / 2) + "px");
};
fExt.popup = function(msg) {
if(msg !== undefined){
if(fExt.popping){
fExt.enqueuePopup(msg);
return;
}
fExt.popping = true;
fExt.fExtPopup.text(msg);
fExt.show(fExt.fExtPopup);
var showTime = msg.length * 50;
if(showTime < fExt.settings.minShowTime)
showTime = fExt.settings.minShowTime;
else if(showTime > fExt.settings.maxShowTime)
shotTime = fExt.settings.maxShowTime;
setTimeout(function(){
fExt.hide(fExt.fExtPopup);
setTimeout(function(){
fExt.fExtPopup.text('');
fExt.popping = false;
fExt.popup(fExt.dequeuePopup());
},fExt.settings.animationSpeed);
},showTime);
}
};
fExt.clipboard = function(action, text) {
var clipboardCopy, retVal;
action = action.toLowerCase();
clipboardCopy = document.createElement('textarea');
var jqClipboardCopy = $(clipboardCopy);
document.body.appendChild(clipboardCopy);
clipboardCopy.value = text;
clipboardCopy.select();
var msg;
try {
var successful = document.execCommand(action);
if(successful)
msg = action + " successful";
else
msg = "Could not perform Clipboard-Action " + action;
if(clipboardCopy.value.length <= 50)
msg += " (" + clipboardCopy.value + ")";
document.fExt.popup(msg);
}
catch (err) {
document.fExt.popup("Error on Clipboard-Action " + action +": " + err);
}
if(clipboardCopy)
clipboardCopy.remove();
};
fExt.message = function(msg) {
if(msg !== undefined && msg.length > 0){
fExt.fExtMessageText.text(msg);
if(!fExt.fExtMessage.is(":visible"))
fExt.show(fExt.fExtMessage);
fExt.center($("div#fExtMessage"));
}
else {
fExt.hide(fExt.fExtMessage);
setTimeout(function(){
fExt.fExtMessageText.text('');
},fExt.settings.animationSpeed);
}
};
fExt.rotate = function(element, rotation) {
var jqEl = $(element);
var degree = jqEl.data('rotation');
if (!degree)
degree = 0;
degree += rotation;
jqEl.css('-webkit-transform','rotate(' + degree + 'deg)');
jqEl.data('rotation',degree);
};
fExt.zoom = function(element, zoom){
var jqEl = $(element);
var zValue = parseFloat(jqEl.css('zoom'));
var zAdd = parseFloat(zoom) / 100;
if(!zValue)
zValue = 1.0;
var zNew = zValue + zAdd;
jqEl.css('zoom', zNew);
};
fExt.zoomIn = function(element, zoom){
fExt.zoom(element, zoom);
};
fExt.zoomOut = function(element, zoom){
fExt.zoom(element, zoom * -1);
};
switch(fExt.settings.animationType) {
case "fade":
fExt.show = function(element, speed) {
if(speed === undefined)
speed = fExt.settings.animationSpeed;
$(element).fadeIn(speed);
};
fExt.hide = function(element, speed) {
if(speed === undefined)
speed = fExt.settings.animationSpeed;
$(element).fadeOut(speed);
};
break;
case "slide":
fExt.show = function(element, speed) {
if(speed === undefined)
speed = fExt.settings.animationSpeed;
$(element).slideDown(speed);
};
fExt.hide = function(element, speed) {
if(speed === undefined)
speed = fExt.settings.animationSpeed;
$(element).slideUp(speed);
};
break;
//case "none":
default:
fExt.show = function(element, speed) {
if(speed === undefined)
speed = fExt.settings.animationSpeed;
$(element).show(speed);
};
fExt.hide = function(element, speed) {
if(speed === undefined)
speed = fExt.settings.animationSpeed;
$(element).hide(speed);
};
break;
}
switch(fExt.settings.position) {
case "TopRight":
fExt.fExtPopup.attr("style", fExt.fExtPopup.attr("style") + "top:50px; right:50px;");
break;
case "TopLeft":
fExt.fExtPopup.attr("style", fExt.fExtPopup.attr("style") + "top:50px; left:50px;");
break;
case "BottomRight":
fExt.fExtPopup.attr("style", fExt.fExtPopup.attr("style") + "bottom:50px; right:50px;");
break;
case "BottomLeft":
fExt.fExtPopup.attr("style", fExt.fExtPopup.attr("style") + "bottom:50px; left:50px;");
break;
default:
break;
}
var siteFavIcon = $("link[rel='icon']");
if(siteFavIcon === undefined) {
$("head").append('<link id="favIcon" rel="icon" href="');
siteFavIcon = $("link[rel='icon']");
}
else {
siteFavIcon.attr("id","favIcon");
}
var siteFavIconHref = siteFavIcon.attr("href");
fExt.getSelection = function() {
return window.getSelection().toString();
};
fExt.getSource = function(element) {
var retVal;
var jqTarget = $(element);
var ucTagName = element.tagName.toUpperCase();
switch(ucTagName) {
case "IMG":
retVal = element.src;
break;
case "A":
retVal = element.href;
break;
case "VIDEO":
var videoSource = jqTarget.find("source");
retVal = (videoSource !== undefined) ? videoSource.get(0).src : undefined;
break;
case "INPUT":
return jqTarget.val();
default:
jqTarget = $(element).closest("a");
if(jqTarget.length > 0)
retVal = jqTarget.attr("href");
break;
}
if(retVal) {
if(!retVal.match("http.*")) {
retVal = window.location.origin + retVal;
}
}
else {
retVal = $(element).text();
}
return retVal;
};
fExt.createStyle(".fExtLoader { background:url('" + loader50 + "') #EFF7FF no-repeat top center; }");
// ContextMenu
fExt.ctxMenu = [];
// Styles
fExt.createStyle("#fExtContextMenu,#fExtContextMenu * { text-align: left !important; text-decoration: none !important; color: #fff; }");
fExt.createStyle("#fExtContextMenu { position: fixed; z-index: 10000; }");
fExt.createStyle("#fExtContextMenu,.ctxSubList { font-size: 14px; background-color: #263238; width: 300px; height: auto; padding: 0;}");
fExt.createStyle("#fExtContextMenu a,#fExtContextMenu hr,#fExtContextMenu li { width: 100%;}");
fExt.createStyle("#fExtContextMenu .ctxElement { float: left; clear: left;}");
fExt.createStyle("#fExtContextMenu li>a,#fExtContextMenu li>div,.ctxSubList li>a,.ctxSubList li>div { padding: 8px;}");
fExt.createStyle("#fExtContextMenu li hr { margin: 0; border-style: solid; border-color: #666; border-width: 1px 0 0 0;}");
fExt.createStyle("#fExtContextMenu li.ctxSub { cursor: default; font-weight: bold;}");
fExt.createStyle("#fExtContextMenu li.ctxSub * { cursor: pointer; font-weight: normal;}");
fExt.createStyle("#fExtContextMenu li { list-style-type: none; margin: 0 !important;}");
fExt.createStyle("#fExtContextMenu li.ctxSub:hover>div.ctxSubLabel,");
fExt.createStyle("#fExtContextMenu li.ctxSub:hover>div.ctxArrow {}");
fExt.createStyle("#fExtContextMenu li.ctxItem { padding-left: 5px;}");
fExt.createStyle("#fExtContextMenu li.ctxItem { border: none; position: initial; box-sizing: border-box; transition: all 250ms ease; }");
fExt.createStyle("#fExtContextMenu li.ctxItem:hover { background: #3a7999; color: #3a7999; box-shadow: inset 0 0 0 3px #3a7999; }");
fExt.createStyle("#fExtContextMenu li.ctxSub.disabled div.ctxSubLabel:hover,");
fExt.createStyle("#fExtContextMenu li.ctxItem.disabled:hover a,#fExtContextMenu li.disabled div.ctxSubLabel,#fExtContextMenu li.disabled a { opacity: 0.70; font-style: italic; padding-left: 7px !important;}");
fExt.createStyle("#fExtContextMenu li.ctxSeparator { display: inline-block;}");
fExt.createStyle("#fExtContextMenu li.ctxSub div.ctxSubLabel { padding-left: 10px !important; float: left; clear: left;}");
fExt.createStyle("#fExtContextMenu li.ctxSub div.ctxArrow { float: right; clear: right;}");
fExt.createStyle("#fExtContextMenu li.ctxSub ul.ctxSubList { position: absolute; height: auto;}");
// Default variables
fExt.ctxMenu.uniqueID = 1;
fExt.ctxMenu.allItems = [];
fExt.ctxMenu.actor = undefined;
fExt.ctxMenu.html = $('<ul id="fExtContextMenu" class="ctxElement" style="display: none;"></ul>');
fExt.ctxMenu.html.appendTo("body");
// Private Methods
ctxCtor = function(ret, sub) {
ret.ItemText = function(value){
if(value === undefined)
return this.item.text();
this.item.text(value);
};
ret.Attribute = function(attribute, value){
if(value === undefined)
return this.item.attr(attribute);
else
this.item.attr(attribute, value);
};
ret.Toggle = function(enabled){
if(enabled === true || (enabled === undefined && this.hasClass("disabled")))
this.removeClass("disabled");
else
this.addClass("disabled");
};
ret.IsDisabled = function(){
return this.hasClass("disabled");
};
ret.ID = function() {
return this.data("Item-ID");
};
};
getParentFrom = function(sub){
if(sub !== undefined)
return $(sub).find("ul:first");
else
return fExt.ctxMenu.html;
};
setZIndex = function(item, parent){
var parentzIndex = parent.css("z-index");
parentzIndex++;
item.css("z-index", parentzIndex);
item.children().each(function(){ $(this).css("z-index", parentzIndex); });
};
// Public Methods
fExt.ctxMenu.addItem = function(label, unused, sub){
var ret = $('<li class="ctxItem ctxElement"></li>');
var retObj = $('<a href="#" class="ctxElement"></a>');
retObj.appendTo(ret);
ret.item = retObj;
var parentMenu = getParentFrom(sub);
ret.appendTo(parentMenu);
setZIndex(ret, parentMenu);
ctxCtor(ret, sub);
ret.Action = undefined;
ret.ItemText(label);
fExt.ctxMenu.assignID(ret);
return ret;
};
fExt.ctxMenu.addSub = function(label, orientation, sub){
var ret = $('<li class="ctxElement ctxSub"><div class="ctxElement ctxArrow">></div><ul class="ctxElement ctxSubList" style="display: none"></ul></li>');
var item = $('<div class="ctxElement ctxSubLabel">' + label + ' </div>');
item.appendTo(ret);
ctxCtor(ret, sub);
var parentMenu = getParentFrom(sub);
if(orientation === 'bottom' || orientation === undefined || $(parentMenu).children().length === 0) {
ret.appendTo(parentMenu);
}
else {
ret.insertBefore($(parentMenu).children("li:first"));
}
setZIndex(ret, parentMenu);
return ret;
};
fExt.ctxMenu.addSeparator = function(sub){
var ret = $('<li class="ctxElement ctxSeparator"><hr/></li>');
var parentMenu = getParentFrom(sub);
ret.appendTo(parentMenu);
setZIndex(ret, parentMenu);
return ret;
};
fExt.ctxMenu.assignID = function(item){
if(item.ID() !== undefined)
console.log("Item already has an ID: " + item.ID());
item.data("Item-ID", fExt.ctxMenu.uniqueID);
fExt.ctxMenu.uniqueID++;
fExt.ctxMenu.allItems.push(item);
};
fExt.ctxMenu.getItem = function(id) {
if(id !== undefined) {
for(i = 0;i < fExt.ctxMenu.allItems.length; i++){
var item = fExt.ctxMenu.allItems[i];
if(id === item.ID())
return item;
}
}
return undefined;
};
// Events
$("#fExtContextMenu").on("click", "li", function( event ) {
var sender = fExt.ctxMenu.getItem($(this).data("Item-ID"));
if(sender === undefined || sender.Action === undefined)
return true;
event.preventDefault();
if(!sender.IsDisabled())
sender.Action(event, sender, fExt.ctxMenu.actor);
fExt.hide(fExt.ctxMenu.html);
return false;
});
$("#fExtContextMenu").on("mouseenter", "li.ctxSub", function(event){
var sub = $(this);
var subOffs = sub.offset();
var ul = sub.children("ul.ctxSubList:first");
if(ul.children("li.ctxItem, li.ctxSub").length === 0)
return;
fExt.show(ul, 0);
var height = ul.height();
var width = ul.width();
var y = subOffs.top;
var x = subOffs.left + sub.width();
if ((x + width) >= window.screen.availWidth)
x = subOffs.left - ul.width();
if ((y + height + 100 - window.scrollY) >= window.screen.availHeight)
y = y - height;
if ((y - window.scrollY) < 0)
y = window.scrollY;
if ((x - window.scrollX) < 0)
x = window.scrollX;
ul.offset({ top: y, left: x});
});
$("#fExtContextMenu").on("mouseleave", "li.ctxSub", function(event){
var ul = $(this).children("ul.ctxSubList:first");
fExt.hide(ul);
});
$("#fExtContextMenu").on("mouseenter", "li.ctxSub", function(event){
var ul = $(this).children("ul.ctxSubList:first");
ul.stop();
if($(this).find("li.ctxItem").length > 0)
ul.show(0);
});
fExt.customContextMenuHandler = function (event){
event.preventDefault();
fExt.ctxMenu.actor = $(event.target);
var y = event.clientY - fExt.settings.toleranceY,
x = event.clientX - fExt.settings.toleranceX;
if ((x + fExt.ctxMenu.html.width()) >= window.screen.availWidth)
x = x - fExt.ctxMenu.html.width() + (fExt.settings.toleranceX * 1.5);
if ((y + fExt.ctxMenu.html.height() + 100) >= window.screen.availHeight)
y = y - fExt.ctxMenu.html.height() + (fExt.settings.toleranceY * 1.5);
if (y < 0)
y = 0;
if (x < 0)
x = 0;
fExt.ctxMenu.html.css({
top: y,
left: x
});
$("#fExtContextMenu").trigger("fExtContextMenuOpening", [fExt.ctxMenu.actor]);
fExt.show(fExt.ctxMenu.html, 0);
return false;
};
$("#fExtContextMenu").on("fExtContextMenuOpening", function(event, actor){
});
if(fExt.settings.hideContextMenuOnLeave) {
$("#fExtContextMenu").mouseleave(function(){
fExt.hide(fExt.ctxMenu.html);
});
$("#fExtContextMenu").mouseenter(function(){
fExt.ctxMenu.html.stop();
fExt.ctxMenu.html.show(0);
});
}
else {
$("body").click(function(e){
if(!$(this).hasClass("ctxElement") && fExt.ctxMenu.html.is(":visible"))
fExt.hide(fExt.ctxMenu.html,0);
});
}
$(document).on('contextmenu', function(event){
if(fExt.settings.customContextMenu) {
if(!$(event.target).hasClass("ctxElement"))
fExt.customContextMenuHandler(event);
else
fExt.ctxMenu.html.hide();
}
});
$(document).on('keyup', function(event){
if((event.which === 17 && event.altKey) || (event.which === 18 && event.ctrlKey)) {
fExt.settings.customContextMenu = !fExt.settings.customContextMenu;
fExt.popup("Context-Menu has been toggled " + (fExt.settings.customContextMenu ? "on" : "off"));
}
});
// Finalize
document.fExt = fExt;
fExt.fExtPopup.click(function(e){
fExt.popupQueue = [];
fExt.hide($(this));
});
fExt.fExtMessage.click(function(e){
fExt.hide($(this));
});
console.log("frisch's UserScript Extender initialized!");
}