This repository has been archived on 2022-03-04. You can view files and clone it, but cannot push or open issues or pull requests.
dotfiles/qutebrowser/.local/share/qutebrowser/greasemonkey/82915.user.js

7802 lines
295 KiB
JavaScript

// ==UserScript==
// @name Reddit Enhancement Suite
// @namespace http://reddit.honestbleeps.com/
// @description A suite of tools to enhance reddit...
// @author honestbleeps
// @include http://reddit.honestbleeps.com/*
// @include http://reddit.com/*
// @include https://reddit.com/*
// @include http://*.reddit.com/*
// @include https://*.reddit.com/*
// ==/UserScript==
var RESVersion = 2.1;
// array compare utility function for keyCode arrays
function keyArrayCompare(fromArr, toArr) {
// if we've passed in a number, fix that and make it an array with alt, shift and ctrl set to false.
if (typeof(toArr) == 'number') {
toArr = Array(toArr,false,false,false);
} else if (toArr.length == 4) {
toArr.push(false);
}
if (fromArr.length != toArr.length) return false;
for (var i = 0; i < toArr.length; i++) {
if (fromArr[i].compare) {
if (!fromArr[i].compare(toArr[i])) return false;
}
if (fromArr[i] !== toArr[i]) return false;
}
return true;
}
function operaUpdateCallback(obj) {
RESUtils.compareVersion(obj);
}
function operaForcedUpdateCallback(obj) {
RESUtils.compareVersion(obj, true);
}
function hasClass(ele,cls) {
if ((typeof(ele) == 'undefined') || (ele == null)) {
console.log(arguments.callee.caller);
return false;
}
return ele.className.match(new RegExp('(\\s|^)'+cls+'(\\s|$)'));
}
function addClass(ele,cls) {
if (!hasClass(ele,cls)) ele.className += " "+cls;
}
function removeClass(ele,cls) {
if (hasClass(ele,cls)) {
var reg = new RegExp('(\\s|^)'+cls+'(\\s|$)');
ele.className=ele.className.replace(reg,' ');
}
}
function post_to_url(path, params, method) {
method = method || "post"; // Set method to post by default, if not specified.
// The rest of this code assumes you are not using a library.
// It can be made less wordy if you use one.
var form = document.createElement("form");
form.setAttribute("method", method);
form.setAttribute("action", path);
for(var key in params) {
var hiddenField = document.createElement("input");
hiddenField.setAttribute("type", "hidden");
hiddenField.setAttribute("name", key);
hiddenField.setAttribute("value", params[key]);
form.appendChild(hiddenField);
}
document.body.appendChild(form);
form.submit();
}
function safariMessageHandler(msgEvent) {
switch (msgEvent.name) {
case 'compareVersion':
var forceUpdate = false;
if (typeof(msgEvent.message.forceUpdate) != 'undefined') forceUpdate = true;
RESUtils.compareVersion(msgEvent.message, forceUpdate);
break;
default:
// console.log('unknown event type in safariMessageHandler');
break;
}
}
if (typeof(safari) != 'undefined') {
safari.self.addEventListener("message", safariMessageHandler, false);
}
// GreaseMonkey API compatibility for Chrome
// @copyright 2009, 2010 James Campos
// @modified 2010 Steve Sobel - added some missing gm_* functions
// @license cc-by-3.0; http://creativecommons.org/licenses/by/3.0/
if (typeof GM_deleteValue == 'undefined') {
GM_addStyle = function(css) {
var style = document.createElement('style');
style.textContent = css;
var head = document.getElementsByTagName('head')[0];
if (head) {
head.appendChild(style);
}
}
GM_deleteValue = function(name) {
localStorage.removeItem(name);
}
GM_getValue = function(name, defaultValue) {
var value = localStorage.getItem(name);
if (!value)
return defaultValue;
var type = value[0];
value = value.substring(1);
switch (type) {
case 'b':
return value == 'true';
case 'n':
return Number(value);
default:
return value;
}
}
GM_log = function(message) {
console.log(message);
}
GM_registerMenuCommand = function(name, funk) {
//todo
}
GM_setValue = function(name, value) {
value = (typeof value)[0] + value;
localStorage.setItem(name, value);
}
GM_xmlhttpRequest=function(obj) {
var request=new XMLHttpRequest();
request.onreadystatechange=function() { if(obj.onreadystatechange) { obj.onreadystatechange(request); }; if(request.readyState==4 && obj.onload) { obj.onload(request); } }
request.onerror=function() { if(obj.onerror) { obj.onerror(request); } }
try { request.open(obj.method,obj.url,true); } catch(e) { if(obj.onerror) { obj.onerror( {readyState:4,responseHeaders:'',responseText:'',responseXML:'',status:403,statusText:'Forbidden'} ); }; return; }
if(obj.headers) { for(name in obj.headers) { request.setRequestHeader(name,obj.headers[name]); } }
request.send(obj.data); return request;
}
}
var RESConsoleContainer = '';
var modalOverlay = '';
var RESMenuItems = new Array();
var RESConsolePanels = new Array();
var modules = new Array();
// define common RESUtils
var RESUtils = {
// imgur API key
imgurAPIKey: 'fe266bc9466fe69aa1cf0904e7298eda',
// A cache variable to store CSS that will be applied at the end of execution...
css: '',
addCSS: function(css) {
this.css += css;
},
// checks if script should run on current URL using exclude / include.
isMatchURL: function (moduleID) {
var currURL = location.href;
// get includes and excludes...
var excludes = modules[moduleID].exclude;
var includes = modules[moduleID].include;
// first check excludes...
if (typeof(excludes) != 'undefined') {
for (i=0, len = excludes.length; i<len; i++) {
// console.log(moduleID + ' -- ' + excludes[i] + ' - excl test - ' + currURL + ' - result: ' + excludes[i].test(currURL));
if (excludes[i].test(currURL)) {
return false;
}
}
}
// then check includes...
for (i=0, len=includes.length; i<len; i++) {
// console.log(moduleID + ' -- ' + includes[i] + ' - incl test - ' + currURL + ' - result: ' + includes[i].test(currURL));
if (includes[i].test(currURL)) {
return true;
}
}
return false;
},
// gets options for a module...
getOptions: function(moduleID) {
var thisOptions = localStorage.getItem('RESoptions.' + moduleID);
var currentTime = new Date();
if ((thisOptions) && (thisOptions != 'undefined') && (thisOptions != null)) {
// merge options (in case new ones were added via code) and if anything has changed, update to localStorage
storedOptions = JSON.parse(thisOptions);
codeOptions = modules[moduleID].options;
for (attrname in codeOptions) {
if (typeof(storedOptions[attrname]) == 'undefined') {
storedOptions[attrname] = codeOptions[attrname];
}
}
modules[moduleID].options = storedOptions;
localStorage.setItem('RESoptions.' + moduleID, JSON.stringify(modules[moduleID].options));
} else {
// nothing in localStorage, let's set the defaults...
localStorage.setItem('RESoptions.' + moduleID, JSON.stringify(modules[moduleID].options));
}
return modules[moduleID].options;
},
setOption: function(moduleID, optionName, optionValue) {
// console.log('called with: moduleid:' + moduleID + ' - optionname: ' + optionName + ' - optionvalue: ' + optionValue + ' -- type: ' + isNaN(optionValue));
var thisOptions = this.getOptions(moduleID);
if ((isNaN(optionValue)) || (typeof(optionValue) == 'boolean') || (typeof(optionValue) == 'object')) {
saveOptionValue = optionValue;
} else if (optionValue.indexOf('.')) {
saveOptionValue = parseFloat(optionValue);
} else {
saveOptionValue = parseInt(optionValue);
}
thisOptions[optionName].value = saveOptionValue;
// save it to the object...
modules[moduleID].options = thisOptions;
// save it to localStorage...
localStorage.setItem('RESoptions.' + moduleID, JSON.stringify(modules[moduleID].options));
return true;
},
click: function(obj, button) {
var button = button || 0;
var evt = document.createEvent('MouseEvents');
evt.initMouseEvent('click', true, true, window, 0, 1, 1, 1, 1, false, false, false, false, button, null); obj.dispatchEvent(evt);
},
loggedInUser: function() {
var userLink = document.querySelector('#header-bottom-right > span.user > a');
if (userLink != null) {
return userLink.innerHTML
} else {
return null;
}
},
pageType: function() {
var pageType = '';
var currURL = location.href;
var commentsRegex = new RegExp(/http:\/\/www.reddit.com\/[-\w\.\/]*comments\/[-\w\.\/]*/i);
var inboxRegex = new RegExp(/http:\/\/www.reddit.com\/message\/[-\w\.\/]*/i);
var submitRegex = new RegExp(/http:\/\/www.reddit.com\/[-\w\.\/]*\/submit\/?/i);
if (commentsRegex.test(currURL)) {
pageType = 'comments'
} else if (inboxRegex.test(currURL)) {
pageType = 'inbox';
} else if (submitRegex.test(currURL)) {
pageType = 'submit';
} else {
pageType = 'linklist';
}
return pageType;
},
currentSubReddit: function() {
var match = location.href.match(/http:\/\/www.reddit.com\/r\/([\w]+).*/i);
if (match != null) {
return match[1];
} else {
return null;
}
},
getXYpos: function (obj) {
var topValue= 0,leftValue= 0;
while(obj){
leftValue+= obj.offsetLeft;
topValue+= obj.offsetTop;
obj= obj.offsetParent;
}
finalvalue = { 'x': leftValue, 'y': topValue };
return finalvalue;
},
elementInViewport: function (obj) {
var top = obj.offsetTop;
var left = obj.offsetLeft;
var width = obj.offsetWidth;
var height = obj.offsetHeight;
while(obj.offsetParent) {
obj = obj.offsetParent;
top += obj.offsetTop;
left += obj.offsetLeft;
}
return (
top >= window.pageYOffset &&
left >= window.pageXOffset &&
(top + height) <= (window.pageYOffset + window.innerHeight) &&
(left + width) <= (window.pageXOffset + window.innerWidth)
);
},
setSelectValue: function(obj, value) {
for (var i=0, len=obj.length; i < len; i++) {
if (obj[i].value == value) {
obj[i].selected = true;
}
}
},
stripHTML: function(str) {
var regExp = /<\/?[^>]+>/gi;
str = str.replace(regExp,"");
return str;
},
fadeElementOut: function(obj, speed) {
if (obj.getAttribute('isfading') == 'in') {
return false;
}
obj.setAttribute('isfading','out');
speed = speed || 0.1;
if (obj.style.opacity <= 0) {
obj.style.display = 'none';
obj.setAttribute('isfading',false);
return true;
} else {
var newOpacity = parseFloat(obj.style.opacity) - parseFloat(speed);
if (newOpacity < 0) newOpacity = 0;
obj.style.opacity = newOpacity;
setTimeout(function() { RESUtils.fadeElementOut(obj, speed) }, 100);
}
},
fadeElementIn: function(obj, speed) {
if (obj.getAttribute('isfading') == 'out') {
return false;
}
obj.setAttribute('isfading','in');
speed = speed || 0.1;
if ((obj.style.display == 'none') || (obj.style.display == '')) {
obj.style.opacity = 0;
obj.style.display = 'block';
}
if (obj.style.opacity >= 1) {
obj.setAttribute('isfading',false);
return true;
} else {
var newOpacity = parseFloat(obj.style.opacity) + parseFloat(speed);
if (newOpacity > 1) newOpacity = 1;
obj.style.opacity = newOpacity;
setTimeout(function() { RESUtils.fadeElementIn(obj, speed) }, 100);
}
},
checkForUpdate: function(forceUpdate) {
var now = new Date();
var lastCheck = parseInt(localStorage.getItem('RESLastUpdateCheck')) || 0;
// if we haven't checked for an update in 24 hours, check for one now!
if (((now.getTime() - lastCheck) > 86400000) || (RESVersion > localStorage.getItem('RESlatestVersion')) || ((localStorage.getItem('RESoutdated') == 'true') && (RESVersion == localStorage.getItem('RESlatestVersion')))|| forceUpdate) {
var jsonURL = 'http://reddit.honestbleeps.com/update.json?v=' + RESVersion;
// mark off that we've checked for an update...
localStorage.setItem('RESLastUpdateCheck',now.getTime());
var outdated = false;
if (typeof(chrome) != 'undefined') {
// we've got chrome, so we need to hit up the background page to do cross domain XHR
thisJSON = {
requestType: 'compareVersion',
url: jsonURL
}
chrome.extension.sendRequest(thisJSON, function(response) {
// send message to background.html to open new tabs...
outdated = RESUtils.compareVersion(response, forceUpdate);
});
} else if (typeof(safari) != 'undefined') {
// we've got safari, so we need to hit up the background page to do cross domain XHR
thisJSON = {
requestType: 'compareVersion',
url: jsonURL,
forceUpdate: forceUpdate
}
safari.self.tab.dispatchMessage("compareVersion", thisJSON);
} else if (typeof(opera) != 'undefined') {
var jsonScript = createElementWithID('script','updateJSON');
var callback = forceUpdate ? 'operaForcedUpdateCallback' : 'operaUpdateCallback';
// This will call operaUpdateCallback(), or -- if this is
// an update check initiated by the user -- operaForcedUpdateCallback().
jsonScript.setAttribute('src',jsonURL.replace('json','php') + '&callback=' + callback);
document.body.appendChild(jsonScript);
// it will call jsonCallback();
} else {
// we've got greasemonkey, so we can do cross domain XHR.
GM_xmlhttpRequest({
method: "GET",
url: jsonURL,
onload: function(response) {
outdated = RESUtils.compareVersion(JSON.parse(response.responseText), forceUpdate);
}
});
}
}
},
compareVersion: function(response, forceUpdate) {
if (RESVersion < response.latestVersion) {
localStorage.setItem('RESoutdated','true');
localStorage.setItem('RESlatestVersion',response.latestVersion);
localStorage.setItem('RESmessage',response.message);
if (forceUpdate) {
RESConsole.RESCheckUpdateButton.innerHTML = 'You are out of date! <a target="_blank" href="http://reddit.honestbleeps.com/download">[click to update]</a>';
}
return true;
} else {
localStorage.setItem('RESlatestVersion',response.latestVersion);
localStorage.setItem('RESoutdated','false');
if (forceUpdate) {
RESConsole.RESCheckUpdateButton.innerHTML = 'You are up to date!';
}
return false;
}
},
niceKeyCode: function(charCode) {
keyComboString = '';
if (typeof(charCode) == 'string') {
var tempArray = charCode.split(',');
if (tempArray.length) {
if (tempArray[1] == 'true') keyComboString += 'alt-';
if (tempArray[2] == 'true') keyComboString += 'ctrl-';
if (tempArray[3] == 'true') keyComboString += 'shift-';
if (tempArray[4] == 'true') keyComboString += 'command-';
}
testCode = parseInt(charCode);
} else if (typeof(charCode) == 'object') {
testCode = parseInt(charCode[0]);
if (charCode[1]) keyComboString += 'alt-';
if (charCode[2]) keyComboString += 'ctrl-';
if (charCode[3]) keyComboString += 'shift-';
if (charCode[4]) keyComboString += 'command-';
}
switch(testCode) {
case 8:
niceString = "backspace"; // backspace
break;
case 9:
niceString = "tab"; // tab
break;
case 13:
niceString = "enter"; // enter
break;
case 16:
niceString = "shift"; // shift
break;
case 17:
niceString = "ctrl"; // ctrl
break;
case 18:
niceString = "alt"; // alt
break;
case 19:
niceString = "pause/break"; // pause/break
break;
case 20:
niceString = "caps lock"; // caps lock
break;
case 27:
niceString = "escape"; // escape
break;
case 33:
niceString = "page up"; // page up, to avoid displaying alternate character and confusing people
break;
case 34:
niceString = "page down"; // page down
break;
case 35:
niceString = "end"; // end
break;
case 36:
niceString = "home"; // home
break;
case 37:
niceString = "left arrow"; // left arrow
break;
case 38:
niceString = "up arrow"; // up arrow
break;
case 39:
niceString = "right arrow"; // right arrow
break;
case 40:
niceString = "down arrow"; // down arrow
break;
case 45:
niceString = "insert"; // insert
break;
case 46:
niceString = "delete"; // delete
break;
case 91:
niceString = "left window"; // left window
break;
case 92:
niceString = "right window"; // right window
break;
case 93:
niceString = "select key"; // select key
break;
case 96:
niceString = "numpad 0"; // numpad 0
break;
case 97:
niceString = "numpad 1"; // numpad 1
break;
case 98:
niceString = "numpad 2"; // numpad 2
break;
case 99:
niceString = "numpad 3"; // numpad 3
break;
case 100:
niceString = "numpad 4"; // numpad 4
break;
case 101:
niceString = "numpad 5"; // numpad 5
break;
case 102:
niceString = "numpad 6"; // numpad 6
break;
case 103:
niceString = "numpad 7"; // numpad 7
break;
case 104:
niceString = "numpad 8"; // numpad 8
break;
case 105:
niceString = "numpad 9"; // numpad 9
break;
case 106:
niceString = "multiply"; // multiply
break;
case 107:
niceString = "add"; // add
break;
case 109:
niceString = "subtract"; // subtract
break;
case 110:
niceString = "decimal point"; // decimal point
break;
case 111:
niceString = "divide"; // divide
break;
case 112:
niceString = "F1"; // F1
break;
case 113:
niceString = "F2"; // F2
break;
case 114:
niceString = "F3"; // F3
break;
case 115:
niceString = "F4"; // F4
break;
case 116:
niceString = "F5"; // F5
break;
case 117:
niceString = "F6"; // F6
break;
case 118:
niceString = "F7"; // F7
break;
case 119:
niceString = "F8"; // F8
break;
case 120:
niceString = "F9"; // F9
break;
case 121:
niceString = "F10"; // F10
break;
case 122:
niceString = "F11"; // F11
break;
case 123:
niceString = "F12"; // F12
break;
case 144:
niceString = "num lock"; // num lock
break;
case 145:
niceString = "scroll lock"; // scroll lock
break;
case 186:
niceString = ";"; // semi-colon
break;
case 187:
niceString = "="; // equal-sign
break;
case 188:
niceString = ","; // comma
break;
case 189:
niceString = "-"; // dash
break;
case 190:
niceString = "."; // period
break;
case 191:
niceString = "/"; // forward slash
break;
case 192:
niceString = "`"; // grave accent
break;
case 219:
niceString = "["; // open bracket
break;
case 220:
niceString = "\\"; // back slash
break;
case 221:
niceString = "]"; // close bracket
break;
case 222:
niceString = "'"; // single quote
break;
default:
niceString = String.fromCharCode(testCode);
break;
}
return keyComboString + niceString;
},
niceDate: function(d, usformat) {
var year = d.getFullYear();
var month = (d.getMonth() + 1);
(month < 10) ? month = '0'+month : month=month;
var day = d.getDate();
(day < 10) ? day = '0'+day : day=day;
var fullString = year+'-'+month+'-'+day;
if (usformat) {
fullString = month+'-'+day+'-'+year;
}
return fullString;
},
niceDateDiff: function(origdate, newdate) {
// Enter the month, day, and year below you want to use as
// the starting point for the date calculation
var amonth = origdate.getMonth()+1;
var aday = origdate.getDate();
var ayear = origdate.getFullYear();
if (newdate == null) newdate = new Date();
var dyear;
var dmonth;
var dday;
var tyear = newdate.getFullYear();
var tmonth = newdate.getMonth()+1;
var tday = newdate.getDate();
var y=1;
var mm=1;
var d=1;
var a2=0;
var a1=0;
var f=28;
if ((tyear/4)-parseInt(tyear/4)==0) {
f=29;
}
m = new Array(31, f, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31);
dyear = tyear-(ayear);
dmonth = tmonth-amonth;
if (dmonth<0) {
dmonth = dmonth+12;
dyear--;
}
dday = tday-aday
if (dday<0) {
var ma = amonth+tmonth;
// console.log('amonth: ' + amonth + ' -- tmonth: ' +tmonth);
if (ma>12) {ma = ma-12}
if (ma=0) {ma = ma+12}
dday = dday+m[ma];
dmonth--;
if (dmonth < 0) {
dyear--;
dmonth = dmonth+12;
}
}
var returnString = '';
if (dyear==0) {y=0}
if (dmonth==0) {mm=0}
if (dday==0) {d=0}
if ((y==1) && (mm==1)) {a1=1}
if ((y==1) && (d==1)) {a1=1}
if ((mm==1) && (d==1)) {a2=1}
if (y==1){
if (dyear == 1) {
returnString += dyear + " year";
} else {
returnString += dyear + " years";
}
}
if ((a1==1) && (a2==0)) { returnString += " and "; }
if ((a1==1) && (a2==1)) { returnString += ", "; }
if (mm==1){
if (dmonth == 1) {
returnString += dmonth + " month";
} else {
returnString += dmonth + " months";
}
}
if (a2==1) { returnString += " and "; }
if (d==1){
if (dday == 1) {
returnString += dday + " day";
} else {
returnString += dday + " days";
}
}
return returnString;
},
checkIfSubmitting: function() {
this.checkedIfSubmitting = true;
if ((location.href.match(/\/r\/[\w]+\/submit\/?/i)) || (location.href.match(/reddit.com\/submit\/?/i))) {
var thisSubRedditInput = document.getElementById('sr-autocomplete');
if (thisSubRedditInput) {
var thisSubReddit = thisSubRedditInput.value;
var title = document.querySelector('textarea[name=title]');
if (typeof(this.thisSubRedditInputListener) == 'undefined') {
this.thisSubRedditInputListener = true;
thisSubRedditInput.addEventListener('change', function(e) {
RESUtils.checkIfSubmitting();
}, false);
}
if (thisSubReddit.toLowerCase() == 'enhancement') {
var textDesc = document.getElementById('text-desc');
this.submittingToEnhancement = createElementWithID('div','submittingToEnhancement');
this.submittingToEnhancement.innerHTML = "<b>Hey there!</b><br><br>It looks like you are submitting to /r/Enhancement. Are you submitting a bug report? \
If so, please consider the following:<br> \
<ol> \
<li>Have you searched /r/Enhancement to see if someone else has reported it?</li> \
<li>Have you checked the <a target=\"_blank\" href=\"http://reddit.honestbleeps.com/faq\">FAQ</a> yet?</li> \
<li>Are you sure it's a bug with RES specifically? Do you have any other Reddit-related userscripts/extensions running?</li> \
<li>Okay - if you've answered \"yes\" to all of the above, go ahead and report it. Please be sure to specify your browser, operating system, anything showing up in your Javascript error console, and any special settings you may have that might help debug.</li> \
</ol> \
";
insertAfter(textDesc, this.submittingToEnhancement);
if (title.value == '') title.value = 'Submitting a bug? Please read the box above...';
RESUtils.addCSS('#submittingToEnhancement { font-size: 12px; border: 1px solid blue; padding: 5px; margin-top: 10px; } #submittingToEnhancement ol { margin-left: 10px; margin-top: 15px; list-style-type: decimal; } #submittingToEnhancement li { margin-left: 25px; }');
} else if (typeof(this.submittingToEnhancement) != 'undefined') {
this.submittingToEnhancement.parentNode.removeChild(this.submittingToEnhancement);
if (title.value == 'Submitting a bug? Please read the box above...') {
title.value = '';
}
}
}
}
}
}
RESUtils.addCSS(' \
#RESConsole { \
display: none; \
font-size: 12px; \
z-index: 1000; \
position: absolute; \
top: 50px; \
left: 50px; \
width: 740px; \
height: 500px; \
border-radius: 10px 10px 10px 10px; \
-moz-border-radius: 10px 10px 10px 10px; \
-webkit-border-radius: 10px 10px 10px 10px; \
padding: 10px; \
border: 4px solid #CCCCCC; \
background-color: #ffffff; \
} \
#modalOverlay { \
display: none; \
z-index: 999; \
position: absolute; \
top: 0px; \
left: 0px; \
width: 100%; \
background-color: #333333; \
opacity: 0.6; \
} \
#RESConsoleHeader { \
width: 100%; \
} \
#RESLogo { \
margin-right: 5px; \
float: left; \
} \
#RESConsoleTopBar { \
width: 100%; \
height: 35px; \
margin-bottom: 10px; \
padding: 3px; \
background-color: #F0F3FC; \
float: left; \
} \
#RESConsoleTopBar h1 { \
float: left; \
margin-top: 5px; \
padding: 0px; \
font-size: 14px; \
font-weight: bold; \
} \
#RESConsoleSubredditLink { \
margin-left: 15px; \
font-size: 11px; \
} \
#RESClose { \
background-image: url("http://thumbs.reddit.com/t5_2s10b_0.png"); \
background-position: 0px -120px; \
width: 16px; \
height: 16px; \
float: right; \
cursor: pointer; \
} \
#RESHelp { \
background-image: url("http://thumbs.reddit.com/t5_2s10b_0.png"); \
background-position: -16px -120px; \
margin-right: 8px; \
width: 16px; \
height: 16px; \
float: right; \
cursor: pointer; \
} \
#RESMenu li { \
float: left; \
text-align: center; \
min-width: 80px; \
margin-right: 15px; \
border: 1px solid black; \
border-radius: 5px 5px 5px 5px; \
-moz-border-radius: 5px 5px 5px 5px; \
-webkit-border-radius: 5px 5px 5px 5px; \
padding: 3px; \
cursor: pointer; \
} \
#RSSConsoleContent { \
clear: both; \
margin-top: 40px; \
padding: 6px; \
height: 410px; \
border: 1px solid #DDDDDD; \
overflow: auto; \
} \
#RESConfigPanelOptions { \
margin-top: 15px; \
display: block; \
width: 100%; \
} \
.RESPanel { \
display: none; \
} \
.clear { \
clear: both; \
} \
#keyCodeModal { \
display: none; \
width: 200px; \
height: 40px; \
position: absolute; \
z-index: 1000; \
background-color: #FFFFFF; \
padding: 4px; \
border: 2px solid #CCCCCC; \
} \
p.moduleListing { \
padding-left: 5px; \
padding-right: 5px; \
padding-top: 5px; \
padding-bottom: 15px; \
border: 1px solid #BBBBBB; \
-moz-box-shadow: 3px 3px 3px #BBB; \
-webkit-box-shadow: 3px 3px 3px #BBB; \
} \
#RESConsoleModulesPanel label { \
float: left; \
width: 140px; \
padding-top: 6px; \
} \
#RESConsoleModulesPanel input[type=checkbox] { \
float: left; \
margin-left: 10px; \
} \
#RESConsoleModulesPanel input[type=button] { \
float: right; \
padding: 3px; \
margin-left: 20px; \
font-size: 12px; \
border: 1px solid #DDDDDD; \
-moz-box-shadow: 3px 3px 3px #BBB; \
-webkit-box-shadow: 3px 3px 3px #BBB; \
background-color: #F0F3FC; \
margin-bottom: 10px; \
} \
#RESConsoleModulesPanel p { \
overflow: auto; \
clear: both; \
margin-bottom: 10px; \
} \
.moduleDescription { \
float: left; \
width: 500px; \
margin-left: 10px; \
padding-top: 6px; \
} \
.optionContainer { \
width: 700px; \
overflow: auto; \
padding: 5px; \
border: 1px dashed #AAAAAA; \
} \
.optionContainer table { \
width: 650px; \
margin-top: 20px; \
} \
.optionContainer label { \
float: left; \
width: 140px; \
} \
.optionContainer input[type=text] { \
float: left; \
width: 100px; \
} \
.optionDescription { \
float: left; \
width: 430px; \
margin-left: 10px; \
} \
#RESCheckForUpdate { \
float: left; \
font-size: 10px; \
cursor: pointer; \
color: f0f3fc; \
margin-left: 10px; \
} \
#moduleOptionsSave { \
float: right; \
padding: 3px; \
margin-top: 6px; \
margin-left: 20px; \
font-size: 12px; \
border: 1px solid #DDDDDD; \
-moz-box-shadow: 3px 3px 3px #BBB; \
-webkit-box-shadow: 3px 3px 3px #BBB; \
background-color: #F0F3FC; \
margin-bottom: 10px; \
} \
#moduleOptionsSaveStatus { \
display: none; \
margin-top: 10px; \
padding: 5px; \
text-align: center; \
background-color: #FFFACD; \
} \
#RESConsoleAboutPanel p { \
margin-bottom: 10px; \
} \
#RESConsoleAboutPanel ul { \
margin-bottom: 10px; \
margin-top: 10px; \
} \
#RESConsoleAboutPanel li { \
list-style-type: disc; \
margin-left: 15px; \
} \
.outdated { \
float: right; \
font-size: 11px; \
margin-right: 15px; \
margin-top: 5px; \
} \
');
// define the RESConsole class
var RESConsole = {
// make the modules panel accessible to this class for updating (i.e. when preferences change, so we can redraw it)
RESConsoleModulesPanel: createElementWithID('div', 'RESConsoleModulesPanel', 'RESPanel'),
RESConsoleConfigPanel: createElementWithID('div', 'RESConsoleConfigPanel', 'RESPanel'),
addConsoleLink: function() {
var userMenu = document.querySelector('#header-bottom-right');
if (userMenu) {
var preferencesUL = userMenu.querySelector('UL');
var separator = document.createElement('span');
separator.setAttribute('class','separator');
separator.innerHTML = '|';
this.RESPrefsLink = document.createElement('a');
this.RESPrefsLink.setAttribute('id','openRESPrefs');
this.RESPrefsLink.setAttribute('href','javascript:void(0)');
if (localStorage.getItem('RESoutdated') == 'true') {
this.RESPrefsLink.innerHTML = '[RES](u)';
} else {
this.RESPrefsLink.innerHTML = '[RES]';
}
this.RESPrefsLink.addEventListener('click', function(e) {
e.preventDefault();
RESConsole.open();
}, true);
insertAfter(preferencesUL, this.RESPrefsLink);
insertAfter(preferencesUL, separator);
}
},
resetModulePrefs: function() {
prefs = {
'userTagger': true,
'commentsLinker': true,
'singleClick': true,
'subRedditTagger': true,
'uppersAndDowners': true,
'keyboardNav': true,
'commentPreview': true,
'showImages': true,
'showKarma': true,
'usernameHider': false,
'accountSwitcher': true,
'styleTweaks': true
};
this.setModulePrefs(prefs);
return prefs;
},
getAllModulePrefs: function() {
// get the stored preferences out first.
if (localStorage.getItem('RES.modulePrefs') != null) {
var storedPrefs = JSON.parse(localStorage.getItem('RES.modulePrefs'));
} else if (localStorage.getItem('modulePrefs') != null) {
// Clean up old moduleprefs.
var storedPrefs = JSON.parse(localStorage.getItem('modulePrefs'));
localStorage.removeItem('modulePrefs');
this.setModulePrefs(storedPrefs);
} else {
// looks like this is the first time RES has been run - set prefs to defaults...
storedPrefs = this.resetModulePrefs();
}
if (storedPrefs == null) {
storedPrefs = {};
}
// create a new JSON object that we'll use to return all preferences. This is just in case we add a module, and there's no pref stored for it.
var prefs = {};
// for any stored prefs, drop them in our prefs JSON object.
for (i in modules) {
if (storedPrefs[i]) {
prefs[i] = storedPrefs[i];
} else if (storedPrefs[i] == null) {
// looks like a new module, or no preferences. We'll default it to on.
prefs[i] = true;
} else {
prefs[i] = false;
}
}
if ((typeof(prefs) != 'undefined') && (prefs != 'undefined') && (prefs)) {
return prefs;
}
},
getModulePrefs: function(moduleID) {
if (moduleID) {
var prefs = this.getAllModulePrefs();
return prefs[moduleID];
} else {
alert('no module name specified for getModulePrefs');
}
},
setModulePrefs: function(prefs) {
if (prefs != null) {
localStorage.setItem('RES.modulePrefs', JSON.stringify(prefs));
this.drawModulesPanel();
return prefs;
} else {
alert('error - no prefs specified');
}
},
container: RESConsoleContainer,
create: function() {
// create the console container
RESConsoleContainer = createElementWithID('div', 'RESConsole');
// create a modal overlay
modalOverlay = createElementWithID('div', 'modalOverlay');
modalOverlay.addEventListener('click',function(e) {
e.preventDefault();
return false;
}, true);
document.body.appendChild(modalOverlay);
// create the header
RESConsoleHeader = createElementWithID('div', 'RESConsoleHeader');
// create the top bar and place it in the header
RESConsoleTopBar = createElementWithID('div', 'RESConsoleTopBar');
this.logo = '';
this.loader = '';
RESConsoleTopBar.innerHTML = '<img id="RESLogo" src="'+this.logo+'"><h1>Reddit Enhancement Suite Console</h1>';
RESConsoleHeader.appendChild(RESConsoleTopBar);
this.RESCheckUpdateButton = createElementWithID('div','RESCheckForUpdate');
this.RESCheckUpdateButton.innerHTML = 'v' + RESVersion + ' [check for update]';
this.RESCheckUpdateButton.addEventListener('click',function(e) {
RESUtils.checkForUpdate(true);
}, true);
RESConsoleTopBar.appendChild(this.RESCheckUpdateButton);
RESSubredditLink = createElementWithID('a','RESConsoleSubredditLink');
RESSubredditLink.innerHTML = '/r/Enhancement';
RESSubredditLink.setAttribute('href','http://reddit.com/r/Enhancement');
RESSubredditLink.setAttribute('alt','The RES Subreddit');
RESConsoleTopBar.appendChild(RESSubredditLink);
// create the close button and place it in the header
RESClose = createElementWithID('span', 'RESClose');
RESClose.innerHTML = '&nbsp;';
RESClose.addEventListener('click',function(e) {
e.preventDefault();
RESConsole.close();
}, true);
RESConsoleTopBar.appendChild(RESClose);
// create the help button and place it in the header
RESHelp = createElementWithID('span', 'RESHelp');
RESHelp.innerHTML = '&nbsp;';
RESHelp.addEventListener('click',function(e) {
e.preventDefault();
modules['RESTips'].randomTip();
}, true);
RESConsoleTopBar.appendChild(RESHelp);
if (localStorage.getItem('RESoutdated') == 'true') {
var RESOutdated = document.createElement('div');
RESOutdated.setAttribute('class','outdated');
RESOutdated.innerHTML = 'There is a new version of RES! <a target="_blank" href="http://reddit.honestbleeps.com/">click to grab it</a>';
RESConsoleTopBar.appendChild(RESOutdated);
}
// create the menu
var menuItems = new Array('Enable Modules','Configure Modules','About RES');
if (localStorage.getItem('RESPro')) {
menuItems[menuItems.length] = 'RES Pro';
}
RESMenu = createElementWithID('ul', 'RESMenu');
for (i in menuItems) {
thisMenuItem = document.createElement('li');
thisMenuItem.innerHTML = menuItems[i];
thisMenuItem.setAttribute('id', 'Menu-' + menuItems[i]);
thisMenuItem.addEventListener('click', function(e) {
e.preventDefault();
RESConsole.menuClick(this);
}, true);
RESMenu.appendChild(thisMenuItem);
}
RESConsoleHeader.appendChild(RESMenu);
RESConsoleContainer.appendChild(RESConsoleHeader);
// Store the menu items in a global variable for easy access by the menu selector function.
RESMenuItems = RESMenu.querySelectorAll('li');
// Create a container for each management panel
RESConsoleContent = createElementWithID('div', 'RSSConsoleContent');
RESConsoleContainer.appendChild(RESConsoleContent);
this.drawModulesPanel();
// Draw the config panel
this.drawConfigPanel();
// Draw the about panel
this.drawAboutPanel();
if (localStorage.getItem('RESPro')) {
// Draw the Pro panel
this.drawProPanel();
}
// Set an easily accessible array of the panels so we can show/hide them as necessary.
RESConsolePanels = RESConsoleContent.querySelectorAll('div');
// Okay, the console is done. Add it to the document body.
document.body.appendChild(RESConsoleContainer);
},
drawModulesPanel: function() {
// Create the module management panel (toggle modules on/off)
RESConsoleModulesPanel = this.RESConsoleModulesPanel;
RESConsoleModulesPanel.innerHTML = '';
var prefs = this.getAllModulePrefs();
var modulesPanelHTML = '';
for (i in modules) {
(prefs[i]) ? thisChecked = 'CHECKED' : thisChecked = '';
if (typeof(modules[i]) != 'undefined') {
thisDesc = modules[i].description;
modulesPanelHTML += '<p class="moduleListing"><label for="'+i+'">' + modules[i].moduleName + ':</label> <input type="checkbox" name="'+i+'" '+thisChecked+' value="true"> <span class="moduleDescription">'+thisDesc+'</span></p>';
}
}
RESConsoleModulesPanel.innerHTML = modulesPanelHTML;
var RESConsoleModulesPanelButtons = createElementWithID('span','RESConsoleModulesPanelButtons');
var RESSavePrefsButton = createElementWithID('input','savePrefs');
RESSavePrefsButton.setAttribute('type','button');
RESSavePrefsButton.setAttribute('name','savePrefs');
RESSavePrefsButton.setAttribute('value','save');
RESSavePrefsButton.addEventListener('click', function(e) {
e.preventDefault();
var modulePrefsCheckboxes = RESConsole.RESConsoleModulesPanel.querySelectorAll('input[type=checkbox]');
var prefs = {};
for (i=0, len=modulePrefsCheckboxes.length;i<len;i++) {
var thisName = modulePrefsCheckboxes[i].getAttribute('name');
var thisChecked = modulePrefsCheckboxes[i].checked;
prefs[thisName] = thisChecked;
}
RESConsole.setModulePrefs(prefs);
RESConsole.close();
}, true);
RESConsoleModulesPanelButtons.appendChild(RESSavePrefsButton);
var RESResetPrefsButton = createElementWithID('input','resetPrefs');
RESResetPrefsButton.setAttribute('type','button');
RESResetPrefsButton.setAttribute('name','resetPrefs');
RESResetPrefsButton.setAttribute('value','reset to default');
RESConsoleModulesPanelButtons.appendChild(RESResetPrefsButton);
RESResetPrefsButton.addEventListener('click', function(e) {
e.preventDefault();
RESConsole.resetModulePrefs();
}, true);
RESConsoleModulesPanel.appendChild(RESConsoleModulesPanelButtons);
var clearDiv = document.createElement('p');
clearDiv.setAttribute('class','clear');
clearDiv.style.display = 'block';
RESConsoleModulesPanel.appendChild(clearDiv);
RESConsoleContent.appendChild(RESConsoleModulesPanel);
},
drawConfigPanel: function() {
RESConsoleConfigPanel = createElementWithID('div', 'RESConsoleConfigPanel', 'RESPanel');
RESConfigPanelSelectorLabel = document.createElement('label');
RESConfigPanelSelectorLabel.setAttribute('for','RESConfigPanelSelector');
RESConfigPanelSelectorLabel.innerHTML = 'Configure module: ';
RESConsoleConfigPanel.appendChild(RESConfigPanelSelectorLabel);
this.RESConfigPanelSelector = createElementWithID('select', 'RESConfigPanelSelector');
thisOption = document.createElement('option');
thisOption.setAttribute('value','');
thisOption.innerHTML = 'Select Module';
this.RESConfigPanelSelector.appendChild(thisOption);
for (i in modules) {
thisOption = document.createElement('option');
thisOption.value = modules[i].moduleID;
thisOption.innerHTML = modules[i].moduleName;
this.RESConfigPanelSelector.appendChild(thisOption);
}
this.RESConfigPanelSelector.addEventListener('change', function(e) {
thisModule = this.options[this.selectedIndex].value;
if (thisModule != '') {
RESConsole.drawConfigOptions(thisModule);
}
}, true);
RESConsoleConfigPanel.appendChild(this.RESConfigPanelSelector);
RESConfigPanelOptions = createElementWithID('div', 'RESConfigPanelOptions');
RESConsoleConfigPanel.appendChild(RESConfigPanelOptions);
RESConsoleContent.appendChild(RESConsoleConfigPanel);
},
drawOptionInput: function(moduleID, optionName, optionObject, isTable) {
switch(optionObject.type) {
case 'text':
// text...
var thisOptionFormEle = createElementWithID('input', optionName);
thisOptionFormEle.setAttribute('type','text');
thisOptionFormEle.setAttribute('moduleID',moduleID);
thisOptionFormEle.setAttribute('value',optionObject.value);
break;
case 'password':
// password...
var thisOptionFormEle = createElementWithID('input', optionName);
thisOptionFormEle.setAttribute('type','password');
thisOptionFormEle.setAttribute('moduleID',moduleID);
thisOptionFormEle.setAttribute('value',optionObject.value);
break;
case 'boolean':
// checkbox
var thisOptionFormEle = createElementWithID('input', optionName);
thisOptionFormEle.setAttribute('type','checkbox');
thisOptionFormEle.setAttribute('moduleID',moduleID);
thisOptionFormEle.setAttribute('value',optionObject.value);
if (optionObject.value) {
thisOptionFormEle.setAttribute('checked',true);
}
break;
case 'enum':
// radio buttons
if (typeof(optionObject.values) == 'undefined') {
alert('misconfigured enum option in module: ' + moduleID);
} else {
var thisOptionFormEle = createElementWithID('div', optionName);
thisOptionFormEle.setAttribute('class','enum');
for (var j=0;j<optionObject.values.length;j++) {
var thisDisplay = optionObject.values[j].display;
var thisValue = optionObject.values[j].value;
thisOptionFormSubEle = createElementWithID('input', optionName+'-'+j);
thisOptionFormSubEle.setAttribute('type','radio');
thisOptionFormSubEle.setAttribute('name',optionName);
thisOptionFormSubEle.setAttribute('moduleID',moduleID);
thisOptionFormSubEle.setAttribute('value',optionObject.values[j].value);
if (optionObject.value == optionObject.values[j].value) {
thisOptionFormSubEle.setAttribute('checked','checked');
}
var thisOptionFormSubEleText = document.createTextNode(optionObject.values[j].name + ' ');
thisOptionFormEle.appendChild(thisOptionFormSubEle);
thisOptionFormEle.appendChild(thisOptionFormSubEleText);
}
}
break;
case 'keycode':
// keycode - shows a key value, but stores a keycode and possibly shift/alt/ctrl combo.
var thisOptionFormEle = createElementWithID('input', optionName);
thisOptionFormEle.setAttribute('type','text');
thisOptionFormEle.setAttribute('class','keycode');
thisOptionFormEle.setAttribute('moduleID',moduleID);
thisOptionFormEle.setAttribute('value',optionObject.value);
break;
default:
console.log('misconfigured option in module: ' + moduleID);
break;
}
if (isTable) {
thisOptionFormEle.setAttribute('tableOption','true');
}
return thisOptionFormEle;
},
drawConfigOptions: function(moduleID) {
var thisOptions = RESUtils.getOptions(moduleID);
var optCount = 0;
RESConfigPanelOptions.setAttribute('style','display: block;');
RESConfigPanelOptions.innerHTML = '';
for (var i in thisOptions) {
if (!(thisOptions[i].noconfig)) {
optCount++;
var thisOptionContainer = createElementWithID('div', null, 'optionContainer');
var thisLabel = document.createElement('label');
thisLabel.setAttribute('for',i);
thisLabel.innerHTML = i;
thisOptionDescription = createElementWithID('div', null, 'optionDescription');
thisOptionDescription.innerHTML = thisOptions[i].description;
thisOptionContainer.appendChild(thisLabel);
if (thisOptions[i].type == 'table') {
// table - has a list of fields (headers of table), users can add/remove rows...
if (typeof(thisOptions[i].fields) == 'undefined') {
alert('misconfigured table option in module: ' + moduleID);
} else {
// get field names...
var fieldNames = new Array();
// now that we know the field names, get table rows...
var thisTable = document.createElement('table');
thisTable.setAttribute('moduleID',moduleID);
thisTable.setAttribute('optionName',i);
thisTable.setAttribute('class','optionsTable');
var thisThead = document.createElement('thead');
var thisTableHeader = document.createElement('tr');
thisTable.appendChild(thisThead);
for (var j=0;j<thisOptions[i].fields.length;j++) {
fieldNames[j] = thisOptions[i].fields[j].name;
var thisTH = document.createElement('th');
thisTH.innerHTML = thisOptions[i].fields[j].name;
thisTableHeader.appendChild(thisTH);
}
thisThead.appendChild(thisTableHeader);
thisTable.appendChild(thisThead);
var thisTbody = document.createElement('tbody');
thisTbody.setAttribute('id','tbody_'+i);
for (var j=0;j<thisOptions[i].value.length;j++) {
var thisTR = document.createElement('tr');
for (var k=0;k<thisOptions[i].fields.length;k++) {
var thisTD = document.createElement('td');
thisOpt = thisOptions[i].fields[k];
thisOpt.value = thisOptions[i].value[j][k];
var thisOptInputName = thisOpt.name + '_' + j;
var thisTableEle = this.drawOptionInput(moduleID, thisOptInputName, thisOpt, true);
thisTD.appendChild(thisTableEle);
thisTR.appendChild(thisTD);
}
thisTbody.appendChild(thisTR);
}
thisTable.appendChild(thisTbody);
var thisOptionFormEle = thisTable;
}
thisOptionContainer.appendChild(thisOptionDescription);
thisOptionContainer.appendChild(thisOptionFormEle);
// Create an "add row" button...
var addRowButton = document.createElement('input');
addRowButton.setAttribute('type','button');
addRowButton.setAttribute('value','Add Row');
addRowButton.setAttribute('optionName',i);
addRowButton.setAttribute('moduleID',moduleID);
addRowButton.addEventListener('click',function() {
var optionName = this.getAttribute('optionName');
var thisTbodyName = 'tbody_' + optionName;
var thisTbody = document.getElementById(thisTbodyName);
var newRow = document.createElement('tr');
for (var i=0, len=modules[moduleID].options[optionName].fields.length;i<len;i++) {
var newCell = document.createElement('td');
var thisOpt = modules[moduleID].options[optionName].fields[i];
thisOpt.value = '';
var thisInput = RESConsole.drawOptionInput(moduleID, optionName, thisOpt, true);
newCell.appendChild(thisInput);
newRow.appendChild(newCell);
}
thisTbody.appendChild(newRow);
}, true);
thisOptionContainer.appendChild(addRowButton);
} else {
var thisOptionFormEle = this.drawOptionInput(moduleID, i, thisOptions[i]);
thisOptionContainer.appendChild(thisOptionFormEle);
thisOptionContainer.appendChild(thisOptionDescription);
}
RESConfigPanelOptions.appendChild(thisOptionContainer);
}
}
// run through any keycode options and mask them for input...
var keyCodeInputs = RESConfigPanelOptions.querySelectorAll('.keycode');
if (keyCodeInputs.length > 0) {
this.keyCodeModal = createElementWithID('div','keyCodeModal');
this.keyCodeModal.innerHTML = 'Press a key (or combination with shift, alt and/or ctrl) to assign this action.';
document.body.appendChild(this.keyCodeModal);
for (var i=0, len=keyCodeInputs.length;i<len;i++) {
keyCodeInputs[i].style.border = '1px solid red';
keyCodeInputs[i].style.display = 'none';
thisKeyCodeDisplay = createElementWithID('input',keyCodeInputs[i].getAttribute('id')+'-display');
thisKeyCodeDisplay.setAttribute('type','text');
thisKeyCodeDisplay.setAttribute('capturefor',keyCodeInputs[i].getAttribute('id'));
thisKeyCodeDisplay.setAttribute('displayonly','true');
thisKeyCodeDisplay.setAttribute('value',RESUtils.niceKeyCode(keyCodeInputs[i].value.toString()));
// thisKeyCodeDisplay.disabled = true;
thisKeyCodeDisplay.addEventListener('focus',function(e) {
window.addEventListener('keydown', function(e) {
if ((RESConsole.captureKey) && (e.keyCode != 16) && (e.keyCode != 17) && (e.keyCode != 18)) {
// capture the key, display something nice for it, and then close the popup...
e.preventDefault();
document.getElementById(RESConsole.captureKeyID).value = e.keyCode + ',' + e.altKey + ',' + e.ctrlKey + ',' + e.shiftKey + ',' + e.metaKey;
var keyArray = Array(e.keyCode, e.altKey, e.ctrlKey, e.shiftKey, e.metaKey);
document.getElementById(RESConsole.captureKeyID+'-display').value = RESUtils.niceKeyCode(keyArray);
RESConsole.keyCodeModal.style.display = 'none';
RESConsole.captureKey = false;
}
}, true);
thisXY=RESUtils.getXYpos(this);
RESConsole.keyCodeModal.setAttribute('style', 'display: block; top: ' + thisXY.y + 'px; left: ' + thisXY.x + 'px;');
// show dialog box to grab keycode, but display something nice...
RESConsole.keyCodeModal.style.display = 'block';
RESConsole.captureKey = true;
RESConsole.captureKeyID = this.getAttribute('capturefor');
}, true);
insertAfter(keyCodeInputs[i], thisKeyCodeDisplay);
}
}
var thisSaveButton = createElementWithID('input','moduleOptionsSave');
thisSaveButton.setAttribute('type','button');
thisSaveButton.setAttribute('value','save');
thisSaveButton.addEventListener('click',function(e) {
e.preventDefault();
var panelOptionsDiv = document.getElementById('RESConfigPanelOptions');
// first, go through inputs that aren't a part of a "table of options"...
var inputs = panelOptionsDiv.querySelectorAll('input');
for (var i=0, len=inputs.length;i<len;i++) {
// save values of any inputs onscreen, but skip ones with 'capturefor' - those are display only.
if ((inputs[i].getAttribute('type') != 'button') && (inputs[i].getAttribute('displayonly') != 'true') && (inputs[i].getAttribute('tableOption') != 'true')) {
// get the option name out of the input field id - unless it's a radio button...
if (inputs[i].getAttribute('type') == 'radio') {
var optionName = inputs[i].getAttribute('name');
} else {
var optionName = inputs[i].getAttribute('id');
}
// get the module name out of the input's moduleid attribute
var moduleID = inputs[i].getAttribute('moduleID');
if (inputs[i].getAttribute('type') == 'checkbox') {
(inputs[i].checked) ? optionValue = true : optionValue = false;
} else if (inputs[i].getAttribute('type') == 'radio') {
if (inputs[i].checked) {
var optionValue = inputs[i].value;
}
} else {
// check if it's a keycode, in which case we need to parse it into an array...
if ((inputs[i].getAttribute('class')) && (inputs[i].getAttribute('class').indexOf('keycode') >= 0)) {
var tempArray = inputs[i].value.split(',');
// convert the internal values of this array into their respective types (int, bool, bool, bool)
var optionValue = Array(parseInt(tempArray[0]), (tempArray[1] == 'true'), (tempArray[2] == 'true'), (tempArray[3] == 'true'), (tempArray[4] == 'true'));
} else {
var optionValue = inputs[i].value;
}
}
if (typeof(optionValue) != 'undefined') {
RESUtils.setOption(moduleID, optionName, optionValue);
}
}
}
// Check if there are any tables of options on this panel...
var optionsTables = panelOptionsDiv.querySelectorAll('.optionsTable');
if (typeof(optionsTables) != 'undefined') {
// For each table, we need to go through each row in the tbody, and then go through each option and make a multidimensional array.
// For example, something like: [['foo','bar','baz'],['pants','warez','cats']]
for (i=0, len=optionsTables.length;i<len;i++) {
var moduleID = optionsTables[i].getAttribute('moduleID');
var optionName = optionsTables[i].getAttribute('optionName');
var thisTBODY = optionsTables[i].querySelector('tbody');
var thisRows = thisTBODY.querySelectorAll('tr');
// check if there are any rows...
if (typeof(thisRows) != 'undefined') {
// go through each row, and get all of the inputs...
var optionMulti = Array();
for (j=0;j<thisRows.length;j++) {
var optionRow = Array();
var inputs = thisRows[j].querySelectorAll('input');
var notAllBlank = false;
for (k=0;k<inputs.length;k++) {
// get the module name out of the input's moduleid attribute
var moduleID = inputs[k].getAttribute('moduleID');
if (inputs[k].getAttribute('type') == 'checkbox') {
(inputs[k].checked) ? optionValue = true : optionValue = false;
} else if (inputs[k].getAttribute('type') == 'radio') {
if (inputs[k].checked) {
var optionValue = inputs[k].value;
}
} else {
// check if it's a keycode, in which case we need to parse it into an array...
if ((inputs[k].getAttribute('class')) && (inputs[k].getAttribute('class').indexOf('keycode') >= 0)) {
var tempArray = inputs[k].value.split(',');
// convert the internal values of this array into their respective types (int, bool, bool, bool)
var optionValue = Array(parseInt(tempArray[0]), (tempArray[1] == 'true'), (tempArray[2] == 'true'), (tempArray[3] == 'true'));
} else {
var optionValue = inputs[k].value;
}
}
if (optionValue != '') {
notAllBlank = true;
}
optionRow[k] = optionValue;
}
if (notAllBlank) {
optionMulti[j] = optionRow;
}
}
if (optionMulti == null) {
optionMulti = [];
}
// ok, we've got all the rows... set the option.
if (typeof(optionValue) != 'undefined') {
RESUtils.setOption(moduleID, optionName, optionMulti);
}
}
}
}
var statusEle = document.getElementById('moduleOptionsSaveStatus');
statusEle.innerHTML = 'Options have been saved...';
statusEle.setAttribute('style','display: block; opacity: 1');
RESUtils.fadeElementOut(statusEle, 0.1)
}, true);
RESConfigPanelOptions.appendChild(thisSaveButton);
var thisSaveStatus = createElementWithID('div','moduleOptionsSaveStatus','saveStatus');
RESConfigPanelOptions.appendChild(thisSaveStatus);
if (optCount == 0) {
RESConfigPanelOptions.innerHTML = 'There are no configurable options for this module';
}
},
drawAboutPanel: function() {
RESConsoleAboutPanel = createElementWithID('div', 'RESConsoleAboutPanel', 'RESPanel');
var AboutPanelHTML = ' \
<p>Author: <a target="_blank" href="http://reddit.honestbleeps.com/">honestbleeps</a><br></p> \
<p>Description: Reddit Enhancement Suite is a collection of modules that make browsing reddit a bit nicer.</p> \
<p>It\'s built with <a target="_blank" href="http://reddit.honestbleeps.com/api">an API</a> that allows you to contribute and include your own modules!</p> \
<p>If you\'ve got bug reports, you\'d like to discuss RES, or you\'d like to converse with other users, please see the <a target="_blank" href="http://www.reddit.com/r/Enhancement/">Enhancement subreddit.</a> </p> \
<p>If you want to contact me directly with suggestions, bug reports or just want to say you appreciate the work, an <a href="mailto:steve@honestbleeps.com">email</a> would be great.</p> \
<p>If you really, really like the work, a donation would be much appreciated.</p> \
<p> \
<form action="https://www.paypal.com/cgi-bin/webscr" method="post"><input type="hidden" name="cmd" value="_s-xclick"><input type="hidden" name="hosted_button_id" value="S7TAR7QU39H22"><input type="image" src="https://www.paypal.com/en_US/i/btn/btn_donateCC_LG.gif" border="0" name="submit" alt="PayPal - The safer, easier way to pay online!"><img alt="" border="0" src="https://www.paypal.com/en_US/i/scr/pixel.gif" width="1" height="1"></form> \
</p> \
<p><strong>Note:</strong> Reddit Enhancement Suite will check, at most once a day, to see if a new version is available. No data about you is sent to me nor is it stored.</p> \
'
RESConsoleAboutPanel.innerHTML = AboutPanelHTML;
RESConsoleContent.appendChild(RESConsoleAboutPanel);
},
drawProPanel: function() {
RESConsoleProPanel = createElementWithID('div', 'RESConsoleProPanel', 'RESPanel');
var ProPanelHTML = ' \
Pro stuff will go here. \
';
RESConsoleProPanel.innerHTML = ProPanelHTML;
RESConsoleContent.appendChild(RESConsoleProPanel);
},
open: function() {
// hide the ad-frame div in case it's flash, because then it covers up the settings console and makes it impossible to see the save button!
var adFrame = document.getElementById('ad-frame');
if ((typeof(adFrame) != 'undefined') && (adFrame != null)) {
adFrame.style.display = 'none';
}
var leftCentered = Math.floor((window.innerWidth - 720) / 2);
modalOverlay.setAttribute('style','display: block; height: ' + document.documentElement.scrollHeight + 'px');
RESConsoleContainer.setAttribute('style','display: block; left: ' + leftCentered + 'px');
RESConsole.menuClick(RESMenuItems[0]);
},
close: function() {
// Let's be nice to reddit and put their ad frame back now...
var adFrame = document.getElementById('ad-frame');
if ((typeof(adFrame) != 'undefined') && (adFrame != null)) {
adFrame.style.display = 'block';
}
modalOverlay.setAttribute('style','display: none;');
RESConsoleContainer.setAttribute('style','display: none;');
// just in case the user was in the middle of setting a key and decided to close the dialog, clean that up.
if (typeof(RESConsole.keyCodeModal) != 'undefined') {
RESConsole.keyCodeModal.style.display = 'none';
RESConsole.captureKey = false;
}
},
menuClick: function(obj) {
objID = obj.getAttribute('id')
// make all menu items look unselected
for (i in RESMenuItems) {
if (i == 'length') break;
RESMenuItems[i].setAttribute('style', 'background-color: #ffffff');
}
// make selected menu item look selected
obj.setAttribute('style', 'background-color: #dddddd');
// hide all console panels
for (i in RESConsolePanels) {
// bug in chrome, barfs on for i in loops with queryselectorall...
if (i == 'length') break;
RESConsolePanels[i].setAttribute('style', 'display: none');
}
switch(objID) {
case 'Menu-Enable Modules':
// show the modules panel
RESConsoleModulesPanel.setAttribute('style', 'display: block');
break;
case 'Menu-Configure Modules':
// show the config panel
this.RESConfigPanelSelector.selectedIndex = 0;
RESConsoleConfigPanel.setAttribute('style', 'display: block');
break;
case 'Menu-About RES':
// show the about panel
RESConsoleAboutPanel.setAttribute('style', 'display: block');
break;
case 'Menu-RES Pro':
// show the pro panel
RESConsoleProPanel.setAttribute('style', 'display: block');
break;
default:
alert('err, what just happened here? I do not recognize that menu item.');
break;
}
},
};
// DOM utility functions
function insertAfter( referenceNode, newNode ) {
if ((typeof(referenceNode) == 'undefined') || (referenceNode == null)) {
console.log(arguments.callee.caller);
} else if ((typeof(referenceNode.parentNode) != 'undefined') && (typeof(referenceNode.nextSibling) != 'undefined')) {
referenceNode.parentNode.insertBefore( newNode, referenceNode.nextSibling );
}
};
function createElementWithID(elementType, id, classname) {
obj = document.createElement(elementType);
if (id != null) {
obj.setAttribute('id', id);
}
if ((typeof(classname) != 'undefined') && (classname != '')) {
obj.setAttribute('class', classname);
}
return obj;
};
/************************************************************************************************************
Creating your own module:
Modules must have the following format, with required functions:
- moduleID - the name of the module, i.e. myModule
- moduleName - a "nice name" for your module...
- description - for the config panel, explains what the module is
- isEnabled - should always return RESConsole.getModulePrefs('moduleID') - where moduleID is your module name.
- isMatchURL - should always return RESUtils.isMatchURL('moduleID') - checks your include and exclude URL matches.
- include - an array of regexes to match against location.href (basically like include in GM)
- exclude (optional) - an array of regexes to exclude against location.href
- go - always checks both if isEnabled() and if RESUtils.isMatchURL(), and if so, runs your main code.
modules['myModule'] = {
moduleID: 'myModule',
moduleName: 'my module',
options: {
// any configurable options you have go here...
// options must have a type and a value..
// valid types are: text, boolean (if boolean, value must be true or false)
// for example:
defaultMessage: {
type: 'text',
value: 'this is default text',
description: 'explanation of what this option is for'
},
doSpecialStuff: {
type: 'boolean',
value: false,
description: 'explanation of what this option is for'
}
},
description: 'This is my module!',
isEnabled: function() {
return RESConsole.getModulePrefs(this.moduleID);
},
include: Array(
/http:\/\/www.reddit.com\/user\/[-\w\.]+/i,
/http:\/\/www.reddit.com\/message\/comments\/[-\w\.]+/i
),
isMatchURL: function() {
return RESUtils.isMatchURL(this.moduleID);
},
go: function() {
if ((this.isEnabled()) && (this.isMatchURL())) {
// get this module's options...
RESUtils.getOptions(this.moduleID);
// do stuff now!
// this is where your code goes...
}
}
}; // note: you NEED this semicolon at the end!
************************************************************************************************************/
modules['subRedditTagger'] = {
moduleID: 'subRedditTagger',
moduleName: 'Subreddit Tagger',
options: {
subReddits: {
type: 'table',
fields: [
{ name: 'subreddit', type: 'text' },
{ name: 'doesntContain', type: 'text' },
{ name: 'tag', type: 'text' }
],
value: [
/*
['somebodymakethis','SMT','[SMT]'],
['pics','pic','[pic]']
*/
],
description: 'Set your subreddits below. For that subreddit, if the title of the post doesn\'t contain what you place in the "doesn\'t contain" field, the subreddit will be tagged with whatever you specify.'
}
},
description: 'Adds tags to posts on subreddits (i.e. [SMT] on SomebodyMakeThis when the user leaves it out)',
isEnabled: function() {
return RESConsole.getModulePrefs(this.moduleID);
},
include: new Array(
/http:\/\/www.reddit.com\/[\?]*/i
),
isMatchURL: function() {
return RESUtils.isMatchURL(this.moduleID);
},
go: function() {
if ((this.isEnabled()) && (this.isMatchURL())) {
// get this module's options...
RESUtils.getOptions(this.moduleID);
// do stuff now!
// this is where your code goes...
this.checkForOldSettings();
this.SRTDoesntContain = new Array();
this.SRTTagWith = new Array();
this.loadSRTRules();
document.body.addEventListener('DOMNodeInserted', function(event) {
if ((event.target.tagName == 'DIV') && (event.target.getAttribute('id') == 'siteTable')) {
modules['subRedditTagger'].scanTitles();
}
}, true);
this.scanTitles();
}
},
loadSRTRules: function () {
var subReddits = this.options.subReddits.value;
for (var i=0, len=subReddits.length; i<len; i++) {
thisGetArray = subReddits[i];
this.SRTDoesntContain[thisGetArray[0].toLowerCase()] = thisGetArray[1];
this.SRTTagWith[thisGetArray[0].toLowerCase()] = thisGetArray[2];
}
},
scanTitles: function() {
var entries = document.querySelectorAll('#siteTable > .thing > DIV.entry');
for (var i=0, len=entries.length; i<len;i++) {
// bug in chrome, barfs on for i in loops with queryselectorall...
if (i == 'length') break;
thisSubRedditEle = entries[i].querySelector('A.subreddit');
if ((typeof(thisSubRedditEle) != 'undefined') && (thisSubRedditEle != null)) {
thisSubReddit = thisSubRedditEle.innerHTML.toLowerCase();
if (typeof this.SRTDoesntContain[thisSubReddit] != 'undefined') {
thisTitle = entries[i].querySelector('A.title');
if (!(hasClass(thisTitle, 'srTagged'))) {
addClass(thisTitle, 'srTagged');
thisRegEx = this.SRTDoesntContain[thisSubReddit];
thisTagWith = this.SRTTagWith[thisSubReddit];
if (thisTitle.text.search(thisRegEx) == -1) {
thisTitle.innerHTML = thisTagWith + ' ' + thisTitle.innerHTML;
}
}
}
}
}
},
checkForOldSettings: function() {
var settingsCopy = Array();
var subRedditCount = 0;
while (localStorage.getItem('subreddit_' + subRedditCount)) {
var thisGet = localStorage.getItem('subreddit_' + subRedditCount).replace(/\"/g,"");
var thisGetArray = thisGet.split("|");
settingsCopy[subRedditCount] = thisGetArray;
localStorage.removeItem('subreddit_' + subRedditCount);
subRedditCount++;
}
if (subRedditCount > 0) {
RESUtils.setOption('subRedditTagger', 'subReddits', settingsCopy);
}
}
};
modules['uppersAndDowners'] = {
moduleID: 'uppersAndDowners',
moduleName: 'Uppers and Downers Enhanced',
options: {
showSigns: {
type: 'boolean',
value: false,
description: 'Show +/- signs next to upvote/downvote tallies.'
}
},
description: 'Displays up/down vote counts on comments.',
isEnabled: function() {
return RESConsole.getModulePrefs(this.moduleID);
},
include: Array(
/http:\/\/www.reddit.com\/user\/[-\w\.]+/i,
/http:\/\/www.reddit.com\/[-\w\.\/]+\/comments\/[-\w\.]+/i
),
isMatchURL: function() {
return RESUtils.isMatchURL(this.moduleID);
},
go: function() {
if ((this.isEnabled()) && (this.isMatchURL())) {
this.commentsWithMoos = Array();
this.moreCommentsIDs = Array();
this.applyUppersAndDowners();
/* having trouble figuring out what to do with top level comments. this will have to wait.
var moreComments = document.querySelectorAll('.morecomments');
for (var i=0, len=moreComments.length; i<len; i++) {
moreComments[i].addEventListener('click', this.addParentListener, false);
}
*/
}
},
addParentListener: function (event) {
var moreCommentsParent = event.target;
var i=0;
var isDeepComment = true;
while (i<7) {
if (typeof(moreCommentsParent.parentNode) != 'undefined') {
moreCommentsParent = moreCommentsParent.parentNode;
} else {
i=7;
isDeepComment = false;
}
i++;
}
if (isDeepComment) {
moreCommentsParent.addEventListener('DOMNodeInserted', modules['uppersAndDowners'].processMoreComments, true);
} else {
document.body.addEventListener('DOMNodeInserted', modules['uppersAndDowners'].processMoreCommentsTopLevel, true);
}
},
processMoreCommentsTopLevel: function (event) {
modules['uppersAndDowners'].applyUppersAndDowners(location.href);
},
processMoreComments: function (event) {
if ((event.target.tagName == 'DIV') && (hasClass(event.target, 'thing'))) {
var getID = /id-([\w])+\_([\w]+)/i;
var IDMatch = getID.exec(event.currentTarget.getAttribute('class'));
if (IDMatch) {
var thisID = IDMatch[2];
if (typeof(modules['uppersAndDowners'].moreCommentsIDs[thisID]) == 'undefined') {
modules['uppersAndDowners'].moreCommentsIDs[thisID] = true;
var thisHREF = location.href + thisID;
event.currentTarget.removeEventListener('DOMNodeInserted', this.processMoreComments, true);
modules['uppersAndDowners'].applyUppersAndDowners(thisHREF);
}
}
}
},
applyUppersAndDowners: function(href) {
/*
The Reddit Uppers/Downers module is included as a convenience, but I did not write it.
Original credits are below.
*/
/*
This code is provided as is, with no warranty of any kind.
I hacked it together in one night for my own use, and have not tested it extensively.
The script can slow down comment page load time; if the lag is noticeable, you may want
to change your preferences to load fewer comments per page.
Note that this runs once and does not install any persistent code on the page. So any votes
you cast will not affect the numbers displayed until you reload.
Also note that the ups and downs will not always add up to the score displayed on reddit.
I think this is because of caching on reddit's part. It's usually within one or two points though.
Code contributors: Allan Bogh - http://www.opencodeproject.com
brasso - https://userscripts-mirror.org/scripts/show/56641
savetheclocktower - http://gist.github.com/174069
skeww (jslint, fragment, chunking) - http://kaioa.com
*/
// get this module's options...
RESUtils.getOptions(this.moduleID);
// do stuff now!
// this is where your code goes...
RESUtils.addCSS(".moo_ups { color:rgb(255, 139, 36); font-weight:bold; } .moo_downs { color:rgb(148,148,255); font-weight:bold; }");
var loc, jsonURL, voteTable, onloadJSON, displayVotes, processTree, isComment, processChildren, processReplies, chunker;
//Get the URL for the JSON details of this comments page
if (href) {
loc = href;
} else {
loc = "" + window.location;
}
jsonURL = loc + "/.json";
if (loc.indexOf("?") !== -1) {
jsonURL = loc.replace("?", "/.json?");
}
voteTable = {};
onloadJSON = function (response) {
var jsonText = response.responseText, data;
try {
data = JSON.parse(jsonText);
} catch (e) {
if (window.console) {
window.console.error(e);
}
}
//Load the vote table by processing the tree
processTree(data); //this takes up no time (4ms on 4000 records)
//Display the loaded votes
displayVotes();
};
// spend up to 50msec a time with a task, wait for 25msec and continue if necessary
// changed to 100/100
chunker = function (items, process) {
var todo = items.concat();
setTimeout(function () {
var start = Date.now();
do {
process(todo.shift());
} while (todo.length && Date.now() - start < 100); // was 50
if (todo.length) {
setTimeout(arguments.callee, 100); // was 25
}
}, 100); // was 25
};
displayVotes = function () {
//Add the style sheets for up and down ratings
var taglines,
commentID = null,
toArray;
toArray = function (col) {
var a = [], i, len;
for (i = 0, len = col.length; i < len; i++) {
a[i] = col[i];
}
return a;
};
taglines = toArray(document.getElementsByClassName("tagline"));
chunker(taglines, function (item) {
var votes, openparen, mooups, pipe, moodowns, voteDowns, voteUps, closeparen, frag;
if (item.nextSibling.nodeName === "FORM") { //the first item is the title of the post
commentID = item.nextSibling.firstChild.value;
if ((voteTable[commentID]) && (typeof(modules['uppersAndDowners'].commentsWithMoos[commentID]) == 'undefined')) {
modules['uppersAndDowners'].commentsWithMoos[commentID] = true;
frag = document.createDocumentFragment(); //using a fragment speeds this up by a factor of about 2
votes = voteTable[commentID];
if (modules['uppersAndDowners'].options.showSigns.value) {
votes.ups = '+'+votes.ups;
votes.downs = '-'+votes.downs;
}
openparen = document.createTextNode(" (");
frag.appendChild(openparen);
mooups = document.createElement("span");
mooups.className = "moo_ups";
voteUps = document.createTextNode(votes.ups);
mooups.appendChild(voteUps);
frag.appendChild(mooups);
pipe = document.createTextNode("|");
item.appendChild(pipe);
moodowns = document.createElement("span");
moodowns.className = "moo_downs";
voteDowns = document.createTextNode(votes.downs);
moodowns.appendChild(voteDowns);
frag.appendChild(moodowns);
closeparen = document.createTextNode(")");
frag.appendChild(closeparen);
frag.appendChild(openparen);
frag.appendChild(mooups);
frag.appendChild(pipe);
frag.appendChild(moodowns);
frag.appendChild(closeparen);
item.appendChild(frag);
}
}
});
};
//Recursively process the comment tree
processTree = function (obj) {
var i, il, data, name;
if (obj instanceof Array) {
for (i = 0, il = obj.length; i < il; i++) {
processTree(obj[i]);
}
}
data = obj.data;
if (data) { //Data found
if (isComment(obj) && data.author !== "[deleted]") {
name = data.name;
if (name) { //Store the votes in the vote table
voteTable[name] = {
downs: data.downs || 0,
ups: data.ups || 0
};
}
}
//Process any subtrees
processChildren(data);
processReplies(data);
}
};
isComment = function (obj) {
return obj.kind === "t1";
};
processChildren = function (data) {
var children = data.children, i, il;
if (children) {
for (i = 0, il = children.length; i < il; i++) {
processTree(children[i]);
}
}
};
processReplies = function (data) {
var replies = data.replies;
if (replies) {
processTree(replies);
}
};
//load the JSON
if (jsonURL.indexOf('/comscore-iframe/') === -1) {
GM_xmlhttpRequest({
method: "GET",
url: jsonURL,
onload: onloadJSON
});
}
}
};
modules['keyboardNav'] = {
moduleID: 'keyboardNav',
moduleName: 'Keyboard Navigation',
options: {
// any configurable options you have go here...
// options must have a type and a value..
// valid types are: text, boolean (if boolean, value must be true or false)
// for example:
focusBorder: {
type: 'text',
value: '1px dashed #888888',
description: 'Border style of focused element'
},
focusBGColor: {
type: 'text',
value: '#F0F3FC',
description: 'Background color of focused element'
},
toggleHelp: {
type: 'keycode',
value: [191,false,false,true], // ? (note the true in the shift slot)
description: 'Show help'
},
hide: {
type: 'keycode',
value: [72,false,false,false], // h
description: 'Hide link'
},
moveUp: {
type: 'keycode',
value: [75,false,false,false], // k
description: 'Move up (previous link or comment)'
},
moveDown: {
type: 'keycode',
value: [74,false,false,false], // j
description: 'Move down (next link or comment)'
},
moveTop: {
type: 'keycode',
value: [75,false,false,true], // shift-k
description: 'Move to top of list (on link pages)'
},
moveBottom: {
type: 'keycode',
value: [74,false,false,true], // shift-j
description: 'Move to bottom of list (on link pages)'
},
moveUpSibling: {
type: 'keycode',
value: [75,false,false,true], // shift-k
description: 'Move to previous sibling (in comments) - skips to previous sibling at the same depth.'
},
moveDownSibling: {
type: 'keycode',
value: [74,false,false,true], // shift-j
description: 'Move to next sibling (in comments) - skips to next sibling at the same depth.'
},
followLink: {
type: 'keycode',
value: [13,false,false,false], // enter
description: 'Follow link (hold shift to open it in a new tab) (link pages only)'
},
followLinkNewTab: {
type: 'keycode',
value: [13,false,false,true], // shift-enter
description: 'Follow link in new tab (link pages only)'
},
toggleChildren: {
type: 'keycode',
value: [13,false,false,false], // enter
description: 'Expand/collapse comments (comments pages only)'
},
followComments: {
type: 'keycode',
value: [67,false,false,false], // c
description: 'View comments for link (shift opens them in a new tab)'
},
followCommentsNewTab: {
type: 'keycode',
value: [67,false,false,true], // shift-c
description: 'View comments for link in a new tab'
},
followLinkAndCommentsNewTab: {
type: 'keycode',
value: [76,false,false,false], // l
description: 'View link and comments in new tabs'
},
followLinkAndCommentsNewTabBG: {
type: 'keycode',
value: [76,false,false,true], // l
description: 'View link and comments in new background tabs'
},
upVote: {
type: 'keycode',
value: [65,false,false,false], // a
description: 'Upvote selected link or comment'
},
downVote: {
type: 'keycode',
value: [90,false,false,false], // z
description: 'Downvote selected link or comment'
},
save: {
type: 'keycode',
value: [83,false,false,false], // s
description: 'Save the current link'
},
reply: {
type: 'keycode',
value: [82,false,false,false], // r
description: 'Reply to current comment (comment pages only)'
},
followSubreddit: {
type: 'keycode',
value: [82,false,false,false], // r
description: 'Go to subreddit of selected link (link pages only)'
},
inbox: {
type: 'keycode',
value: [73,false,false,false], // i
description: 'Go to inbox'
},
frontPage: {
type: 'keycode',
value: [70,false,false,false], // f
description: 'Go to front page'
},
nextPage: {
type: 'keycode',
value: [78,false,false,false], // n
description: 'Go to next page (link list pages only)'
},
prevPage: {
type: 'keycode',
value: [80,false,false,false], // p
description: 'Go to prev page (link list pages only)'
},
link1: {
type: 'keycode',
value: [49,false,false,false], // 1
description: 'Open first link within comment.',
noconfig: true
},
link2: {
type: 'keycode',
value: [50,false,false,false], // 2
description: 'Open link #2 within comment.',
noconfig: true
},
link3: {
type: 'keycode',
value: [51,false,false,false], // 3
description: 'Open link #3 within comment.',
noconfig: true
},
link4: {
type: 'keycode',
value: [52,false,false,false], // 4
description: 'Open link #4 within comment.',
noconfig: true
},
link5: {
type: 'keycode',
value: [53,false,false,false], // 5
description: 'Open link #5 within comment.',
noconfig: true
},
link6: {
type: 'keycode',
value: [54,false,false,false], // 6
description: 'Open link #6 within comment.',
noconfig: true
},
link7: {
type: 'keycode',
value: [55,false,false,false], // 7
description: 'Open link #7 within comment.',
noconfig: true
},
link8: {
type: 'keycode',
value: [56,false,false,false], // 8
description: 'Open link #8 within comment.',
noconfig: true
},
link9: {
type: 'keycode',
value: [57,false,false,false], // 9
description: 'Open link #9 within comment.',
noconfig: true
},
link10: {
type: 'keycode',
value: [48,false,false,false], // 0
description: 'Open link #10 within comment.',
noconfig: true
},
commentsLinkNumbers: {
type: 'boolean',
value: true,
description: 'Assign number keys to links within selected comment'
},
commentsLinkNewTab: {
type: 'boolean',
value: true,
description: 'Open number key links in a new tab'
},
clickFocus: {
type: 'boolean',
value: true,
description: 'Move keyboard focus to a link or comment when clicked with the mouse'
}
},
description: 'Keyboard navigation for reddit!',
isEnabled: function() {
return RESConsole.getModulePrefs(this.moduleID);
},
include: Array(
/http:\/\/www.reddit.com\/[-\w\.\/]*/i,
/https:\/\/www.reddit.com\/[-\w\.\/]*/i
),
isMatchURL: function() {
return RESUtils.isMatchURL(this.moduleID);
},
go: function() {
if ((this.isEnabled()) && (this.isMatchURL())) {
// get this module's options...
RESUtils.getOptions(this.moduleID);
// do stuff now!
// this is where your code goes...
if (typeof(this.options.focusBorder) == 'undefined') {
focusBorder = '2px dashed #888888';
} else {
focusBorder = this.options.focusBorder.value;
}
if (typeof(this.options.focusBGColor) == 'undefined') {
focusBGColor = '#F0F3FC';
} else {
focusBGColor = this.options.focusBGColor.value;
}
RESUtils.addCSS(' \
.keyHighlight { border: '+focusBorder+'; background-color: '+focusBGColor+'; } \
#keyHelp { display: none; position: fixed; right: 20px; top: 20px; z-index: 1000; border: 2px solid #AAAAAA; border-radius: 5px 5px 5px 5px; -moz-border-radius: 5px 5px 5px 5px; -webkit-border-radius: 5px 5px 5px 5px; width: 300px; padding: 5px; background-color: #ffffff; } \
#keyHelp th { font-weight: bold; padding: 2px; border-bottom: 1px dashed #dddddd; } \
#keyHelp td { padding: 2px; border-bottom: 1px dashed #dddddd; } \
#keyHelp td:first-child { width: 70px; } \
');
this.drawHelp();
//GM_addStyle(css);
window.addEventListener('keydown', function(e) {
// console.log(e.keyCode);
modules['keyboardNav'].handleKeyPress(e);
}, true);
this.scanPageForKeyboardLinks();
// listen for new DOM nodes so that modules like autopager, never ending reddit, etc still get keyboard nav.
document.body.addEventListener('DOMNodeInserted', function(event) {
if ((event.target.tagName == 'DIV') && ((event.target.getAttribute('id') == 'siteTable') || (hasClass(event.target,'child')))) {
modules['keyboardNav'].scanPageForKeyboardLinks(true);
}
}, true);
}
},
scanPageForKeyboardLinks: function(isNew) {
if (typeof(isNew) == 'undefined') {
isNew = false;
}
// check if we're on a link listing (regular page, subreddit page, etc) or comments listing...
this.pageType = RESUtils.pageType();
switch(this.pageType) {
case 'linklist':
// get all links into an array...
var siteTable = document.querySelector('#siteTable');
var stMultiCheck = document.querySelectorAll('#siteTable');
// stupid sponsored links create a second div with ID of sitetable (bad reddit! you should never have 2 IDs with the same name! naughty, naughty reddit!)
if (stMultiCheck.length == 2) {
siteTable = stMultiCheck[1];
}
if (siteTable) {
this.keyboardLinks = document.body.querySelectorAll('div.linklisting .entry');
if (!isNew) {
if (localStorage.getItem('RESmodules.keyboardNavLastIndex.'+location.href) > 0) {
this.activeIndex = localStorage.getItem('RESmodules.keyboardNavLastIndex.'+location.href);
} else {
this.activeIndex = 0;
}
if (localStorage.getItem('RESmodules.keyboardNavLastIndex.'+location.href) >= this.keyboardLinks.length) {
this.activeIndex = 0;
}
this.keyFocus(this.keyboardLinks[this.activeIndex]);
}
}
break;
case 'comments':
// get all links into an array...
this.keyboardLinks = document.body.querySelectorAll('#siteTable .entry, div.content > div.commentarea .entry');
this.activeIndex = 0;
this.keyFocus(this.keyboardLinks[0]);
break;
case 'inbox':
var siteTable = document.querySelector('#siteTable');
if (siteTable) {
this.keyboardLinks = siteTable.querySelectorAll('.entry');
this.activeIndex = 0;
this.keyFocus(this.keyboardLinks[this.activeIndex]);
}
break;
}
// wire up keyboard links for mouse clicky selecty goodness...
if ((typeof(this.keyboardLinks) != 'undefined') && (this.options.clickFocus.value)) {
for (var i=0, len=this.keyboardLinks.length;i<len;i++) {
this.keyboardLinks[i].setAttribute('keyIndex', i);
this.keyboardLinks[i].addEventListener('click', function(e) {
var thisIndex = this.getAttribute('keyIndex');
if (modules['keyboardNav'].activeIndex != thisIndex) {
modules['keyboardNav'].keyUnfocus(modules['keyboardNav'].keyboardLinks[modules['keyboardNav'].activeIndex]);
modules['keyboardNav'].activeIndex = thisIndex;
modules['keyboardNav'].keyFocus(modules['keyboardNav'].keyboardLinks[modules['keyboardNav'].activeIndex]);
}
}, true);
}
}
},
keyFocus: function(obj) {
if ((typeof(obj) != 'undefined') && (hasClass(obj, 'keyHighlight'))) {
return false;
} else if (typeof(obj) != 'undefined') {
addClass(obj, 'keyHighlight');
if (this.pageType == 'linklist') localStorage.setItem('RESmodules.keyboardNavLastIndex.'+location.href, this.activeIndex);
if ((this.pageType == 'comments') && (this.options.commentsLinkNumbers.value)) {
var links = obj.querySelectorAll('div.md a');
var annotationCount = 0;
for (var i=0, len=links.length; i<len; i++) {
if ((!(hasClass(links[i],'madeVisible'))) && (!(hasClass(links[i],'toggleImage')))) {
var annotation = document.createElement('span');
annotationCount++;
annotation.innerHTML = '['+annotationCount+'] ';
addClass(annotation,'keyNavAnnotation');
if (!(hasClass(links[i],'hasListener'))) {
addClass(links[i],'hasListener');
links[i].addEventListener('click',function(e) {
e.preventDefault();
if (modules['keyboardNav'].options.commentsLinkNewTab.value) {
if (typeof(chrome) != 'undefined') {
thisJSON = {
requestType: 'keyboardNav',
linkURL: this.getAttribute('href')
}
chrome.extension.sendRequest(thisJSON, function(response) {
// send message to background.html to open new tabs...
return true;
});
} else if (typeof(safari) != 'undefined') {
thisJSON = {
requestType: 'keyboardNav',
linkURL: this.getAttribute('href')
}
safari.self.tab.dispatchMessage("keyboardNav", thisJSON);
} else {
window.open(this.getAttribute('href'));
}
} else {
location.href = this.getAttribute('href');
}
}, true);
}
links[i].parentNode.insertBefore(annotation, links[i]);
}
}
}
}
},
keyUnfocus: function(obj) {
removeClass(obj, 'keyHighlight');
if (this.pageType == 'comments') {
var annotations = obj.querySelectorAll('div.md .keyNavAnnotation');
for (var i=0, len=annotations.length; i<len; i++) {
annotations[i].parentNode.removeChild(annotations[i]);
}
}
},
drawHelp: function() {
var thisHelp = createElementWithID('div','keyHelp');
var helpTable = document.createElement('table');
thisHelp.appendChild(helpTable);
var helpTableHeader = document.createElement('thead');
var helpTableHeaderRow = document.createElement('tr');
var helpTableHeaderKey = document.createElement('th');
helpTableHeaderKey.innerHTML = 'Key';
helpTableHeaderRow.appendChild(helpTableHeaderKey);
var helpTableHeaderFunction = document.createElement('th');
helpTableHeaderFunction.innerHTML = 'Function';
helpTableHeaderRow.appendChild(helpTableHeaderFunction);
helpTableHeader.appendChild(helpTableHeaderRow);
helpTable.appendChild(helpTableHeader);
helpTableBody = document.createElement('tbody');
for (i in this.options) {
var isLink = new RegExp(/^link[\d]+$/i);
if ((this.options[i].type == 'keycode') && (!isLink.test(i))) {
var thisRow = document.createElement('tr');
var thisRowKey = document.createElement('td');
var keyCodeArray = this.options[i].value;
if (typeof(keyCodeArray) == 'string') {
keyCodeAarray = parseInt(keyCodeArray);
}
if (typeof(keyCodeArray) == 'number') {
keyCodeArray = Array(keyCodeArray, false, false, false, false);
}
thisRowKey.innerHTML = RESUtils.niceKeyCode(keyCodeArray);
thisRow.appendChild(thisRowKey);
var thisRowDesc = document.createElement('td');
thisRowDesc.innerHTML = this.options[i].description;
thisRow.appendChild(thisRowDesc);
helpTableBody.appendChild(thisRow);
}
}
helpTable.appendChild(helpTableBody);
document.body.appendChild(thisHelp);
},
handleKeyPress: function(e) {
if (document.activeElement.tagName == 'BODY') {
// comments page, or link list?
keyArray = Array(e.keyCode, e.altKey, e.ctrlKey, e.shiftKey, e.metaKey);
switch(this.pageType) {
case 'linklist':
switch(true) {
case keyArrayCompare(keyArray, this.options.moveUp.value):
this.moveUp();
break;
case keyArrayCompare(keyArray, this.options.moveDown.value):
this.moveDown();
break;
case keyArrayCompare(keyArray, this.options.moveTop.value):
this.moveTop();
break;
case keyArrayCompare(keyArray, this.options.moveBottom.value):
this.moveBottom();
break;
case keyArrayCompare(keyArray, this.options.followLink.value):
this.followLink();
break;
case keyArrayCompare(keyArray, this.options.followLinkNewTab.value):
this.followLink(true);
break;
case keyArrayCompare(keyArray, this.options.followComments.value):
this.followComments();
break;
case keyArrayCompare(keyArray, this.options.followCommentsNewTab.value):
this.followComments(true);
break;
case keyArrayCompare(keyArray, this.options.followLinkAndCommentsNewTab.value):
e.preventDefault();
this.followLinkAndComments();
break;
case keyArrayCompare(keyArray, this.options.followLinkAndCommentsNewTabBG.value):
e.preventDefault();
this.followLinkAndComments(true);
break;
case keyArrayCompare(keyArray, this.options.upVote.value):
this.upVote();
break;
case keyArrayCompare(keyArray, this.options.downVote.value):
this.downVote();
break;
case keyArrayCompare(keyArray, this.options.save.value):
this.saveLink();
break;
case keyArrayCompare(keyArray, this.options.inbox.value):
e.preventDefault();
this.inbox();
break;
case keyArrayCompare(keyArray, this.options.frontPage.value):
e.preventDefault();
this.frontPage();
break;
case keyArrayCompare(keyArray, this.options.nextPage.value):
e.preventDefault();
this.nextPage();
break;
case keyArrayCompare(keyArray, this.options.prevPage.value):
e.preventDefault();
this.prevPage();
break;
case keyArrayCompare(keyArray, this.options.toggleHelp.value):
this.toggleHelp();
break;
case keyArrayCompare(keyArray, this.options.hide.value):
this.hide();
break;
case keyArrayCompare(keyArray, this.options.followSubreddit.value):
this.followSubreddit();
break;
default:
// do nothing. unrecognized key.
break;
}
break;
case 'comments':
switch(true) {
case keyArrayCompare(keyArray, this.options.toggleHelp.value):
this.toggleHelp();
break;
case keyArrayCompare(keyArray, this.options.moveUp.value):
this.moveUp();
break;
case keyArrayCompare(keyArray, this.options.moveDown.value):
this.moveDown();
break;
case keyArrayCompare(keyArray, this.options.moveUpSibling.value):
this.moveUpSibling();
break;
case keyArrayCompare(keyArray, this.options.moveDownSibling.value):
this.moveDownSibling();
break;
case keyArrayCompare(keyArray, this.options.toggleChildren.value):
this.toggleChildren();
break;
case keyArrayCompare(keyArray, this.options.upVote.value):
this.upVote();
break;
case keyArrayCompare(keyArray, this.options.downVote.value):
this.downVote();
break;
case keyArrayCompare(keyArray, this.options.reply.value):
e.preventDefault();
this.reply();
break;
case keyArrayCompare(keyArray, this.options.inbox.value):
e.preventDefault();
this.inbox();
break;
case keyArrayCompare(keyArray, this.options.frontPage.value):
e.preventDefault();
this.frontPage();
break;
case keyArrayCompare(keyArray, this.options.link1.value):
e.preventDefault();
this.commentLink(0);
break;
case keyArrayCompare(keyArray, this.options.link2.value):
e.preventDefault();
this.commentLink(1);
break;
case keyArrayCompare(keyArray, this.options.link3.value):
e.preventDefault();
this.commentLink(2);
break;
case keyArrayCompare(keyArray, this.options.link4.value):
e.preventDefault();
this.commentLink(3);
break;
case keyArrayCompare(keyArray, this.options.link5.value):
e.preventDefault();
this.commentLink(4);
break;
case keyArrayCompare(keyArray, this.options.link6.value):
e.preventDefault();
this.commentLink(5);
break;
case keyArrayCompare(keyArray, this.options.link7.value):
e.preventDefault();
this.commentLink(6);
break;
case keyArrayCompare(keyArray, this.options.link8.value):
e.preventDefault();
this.commentLink(7);
break;
case keyArrayCompare(keyArray, this.options.link9.value):
e.preventDefault();
this.commentLink(8);
break;
case keyArrayCompare(keyArray, this.options.link10.value):
e.preventDefault();
this.commentLink(9);
break;
default:
// do nothing. unrecognized key.
break;
}
break;
case 'inbox':
switch(true) {
case keyArrayCompare(keyArray, this.options.toggleHelp.value):
this.toggleHelp();
break;
case keyArrayCompare(keyArray, this.options.moveUp.value):
this.moveUp();
break;
case keyArrayCompare(keyArray, this.options.moveDown.value):
this.moveDown();
break;
case keyArrayCompare(keyArray, this.options.toggleChildren.value):
this.toggleChildren();
break;
case keyArrayCompare(keyArray, this.options.upVote.value):
this.upVote();
break;
case keyArrayCompare(keyArray, this.options.downVote.value):
this.downVote();
break;
case keyArrayCompare(keyArray, this.options.reply.value):
e.preventDefault();
this.reply();
break;
case keyArrayCompare(keyArray, this.options.frontPage.value):
e.preventDefault();
this.frontPage();
break;
default:
// do nothing. unrecognized key.
break;
}
break;
}
} else {
// console.log('ignored keypress');
}
},
toggleHelp: function() {
(document.getElementById('keyHelp').style.display == 'block') ? this.hideHelp() : this.showHelp();
},
showHelp: function() {
// show help!
RESUtils.fadeElementIn(document.getElementById('keyHelp'), 0.3);
},
hideHelp: function() {
// show help!
RESUtils.fadeElementOut(document.getElementById('keyHelp'), 0.3);
},
hide: function() {
// find the hide link and click it...
var hideLink = this.keyboardLinks[this.activeIndex].querySelector('form.hide-button > span > a');
RESUtils.click(hideLink);
},
followSubreddit: function() {
// find the subreddit link and click it...
var srLink = this.keyboardLinks[this.activeIndex].querySelector('A.subreddit');
var thisHREF = srLink.getAttribute('href');
location.href = thisHREF;
},
moveUp: function() {
if (this.activeIndex > 0) {
this.keyUnfocus(this.keyboardLinks[this.activeIndex]);
this.activeIndex--;
thisXY=RESUtils.getXYpos(this.keyboardLinks[this.activeIndex]);
// skip over hidden elements...
while ((thisXY.x == 0) && (thisXY.y == 0) && (this.activeIndex > 0)) {
this.activeIndex--;
thisXY=RESUtils.getXYpos(this.keyboardLinks[this.activeIndex]);
}
this.keyFocus(this.keyboardLinks[this.activeIndex]);
if (!(RESUtils.elementInViewport(this.keyboardLinks[this.activeIndex]))) {
window.scrollTo(0,thisXY.y);
}
}
},
moveDown: function() {
if (this.activeIndex < this.keyboardLinks.length-1) {
this.keyUnfocus(this.keyboardLinks[this.activeIndex]);
this.activeIndex++;
thisXY=RESUtils.getXYpos(this.keyboardLinks[this.activeIndex]);
// skip over hidden elements...
while ((thisXY.x == 0) && (thisXY.y == 0) && (this.activeIndex < this.keyboardLinks.length-1)) {
this.activeIndex++;
thisXY=RESUtils.getXYpos(this.keyboardLinks[this.activeIndex]);
}
this.keyFocus(this.keyboardLinks[this.activeIndex]);
// console.log('xy: ' + RESUtils.getXYpos(this.keyboardLinks[this.activeIndex]).toSource());
if (!(RESUtils.elementInViewport(this.keyboardLinks[this.activeIndex]))) {
window.scrollTo(0,thisXY.y);
}
}
},
moveTop: function() {
this.keyUnfocus(this.keyboardLinks[this.activeIndex]);
this.activeIndex = 0;
this.keyFocus(this.keyboardLinks[this.activeIndex]);
thisXY=RESUtils.getXYpos(this.keyboardLinks[this.activeIndex]);
if (!(RESUtils.elementInViewport(this.keyboardLinks[this.activeIndex]))) {
window.scrollTo(0,thisXY.y);
}
},
moveBottom: function() {
this.keyUnfocus(this.keyboardLinks[this.activeIndex]);
this.activeIndex = this.keyboardLinks.length-1;
this.keyFocus(this.keyboardLinks[this.activeIndex]);
thisXY=RESUtils.getXYpos(this.keyboardLinks[this.activeIndex]);
if (!(RESUtils.elementInViewport(this.keyboardLinks[this.activeIndex]))) {
window.scrollTo(0,thisXY.y);
}
},
moveDownSibling: function() {
if (this.activeIndex < this.keyboardLinks.length-1) {
this.keyUnfocus(this.keyboardLinks[this.activeIndex]);
var thisParent = this.keyboardLinks[this.activeIndex].parentNode;
var childCount = thisParent.querySelectorAll('.entry').length;
this.activeIndex += childCount;
thisXY=RESUtils.getXYpos(this.keyboardLinks[this.activeIndex]);
// skip over hidden elements...
while ((thisXY.x == 0) && (thisXY.y == 0) && (this.activeIndex < this.keyboardLinks.length-1)) {
this.activeIndex++;
thisXY=RESUtils.getXYpos(this.keyboardLinks[this.activeIndex]);
}
if (this.pageType == 'linklist') localStorage.setItem('RESmodules.keyboardNavLastIndex.'+location.href, this.activeIndex);
this.keyFocus(this.keyboardLinks[this.activeIndex]);
if (!(RESUtils.elementInViewport(this.keyboardLinks[this.activeIndex]))) {
window.scrollTo(0,thisXY.y);
}
}
},
moveUpSibling: function() {
if (this.activeIndex < this.keyboardLinks.length-1) {
this.keyUnfocus(this.keyboardLinks[this.activeIndex]);
var thisParent = this.keyboardLinks[this.activeIndex].parentNode;
if (thisParent.previousSibling != null) {
var childCount = thisParent.previousSibling.previousSibling.querySelectorAll('.entry').length;
} else {
var childCount = 1;
}
this.activeIndex -= childCount;
thisXY=RESUtils.getXYpos(this.keyboardLinks[this.activeIndex]);
while ((thisXY.x == 0) && (thisXY.y == 0) && (this.activeIndex < this.keyboardLinks.length-1)) {
this.activeIndex++;
thisXY=RESUtils.getXYpos(this.keyboardLinks[this.activeIndex]);
}
if (this.pageType == 'linklist') localStorage.setItem('RESmodules.keyboardNavLastIndex.'+location.href, this.activeIndex);
this.keyFocus(this.keyboardLinks[this.activeIndex]);
if (!(RESUtils.elementInViewport(this.keyboardLinks[this.activeIndex]))) {
window.scrollTo(0,thisXY.y);
}
}
},
toggleChildren: function() {
if (this.activeIndex == 0) {
// Ahh, we're not in a comment, but in the main story... that key should follow the link.
this.followLink();
} else {
// find out if this is a collapsed or uncollapsed view...
var thisCollapsed = this.keyboardLinks[this.activeIndex].querySelector('div.collapsed');
var thisNonCollapsed = this.keyboardLinks[this.activeIndex].querySelector('div.noncollapsed');
if (thisCollapsed.style.display != 'none') {
thisToggle = thisCollapsed.querySelector('a.expand');
} else {
// check if this is a "show more comments" box, or just contracted content...
moreComments = thisNonCollapsed.querySelector('span.morecomments > a');
if (moreComments) {
thisToggle = moreComments;
} else {
thisToggle = thisNonCollapsed.querySelector('a.expand');
}
}
RESUtils.click(thisToggle);
}
},
followLink: function(newWindow) {
var thisA = this.keyboardLinks[this.activeIndex].querySelector('a.title');
var thisHREF = thisA.getAttribute('href');
// console.log(thisA);
if (newWindow) {
if (typeof(chrome) != 'undefined') {
thisJSON = {
requestType: 'keyboardNav',
linkURL: thisHREF
}
chrome.extension.sendRequest(thisJSON, function(response) {
// send message to background.html to open new tabs...
return true;
});
} else if (typeof(safari) != 'undefined') {
thisJSON = {
requestType: 'keyboardNav',
linkURL: thisHREF
}
safari.self.tab.dispatchMessage("keyboardNav", thisJSON);
} else {
window.open(thisHREF);
}
} else {
location.href = thisHREF;
}
},
followComments: function(newWindow) {
var thisA = this.keyboardLinks[this.activeIndex].querySelector('a.comments');
var thisHREF = thisA.getAttribute('href');
if (newWindow) {
if (typeof(chrome) != 'undefined') {
thisJSON = {
requestType: 'keyboardNav',
linkURL: thisHREF
}
chrome.extension.sendRequest(thisJSON, function(response) {
// send message to background.html to open new tabs...
return true;
});
} else if (typeof(safari) != 'undefined') {
thisJSON = {
requestType: 'keyboardNav',
linkURL: thisHREF
}
safari.self.tab.dispatchMessage("keyboardNav", thisJSON);
} else {
window.open(thisHREF);
}
} else {
location.href = thisHREF;
}
},
followLinkAndComments: function(background) {
// find the [l+c] link and click it...
var lcLink = this.keyboardLinks[this.activeIndex].querySelector('.redditSingleClick');
RESUtils.click(lcLink, background);
},
upVote: function() {
if (this.keyboardLinks[this.activeIndex].previousSibling.tagName == 'A') {
var upVoteButton = this.keyboardLinks[this.activeIndex].previousSibling.previousSibling.querySelector('div.up') || this.keyboardLinks[this.activeIndex].previousSibling.previousSibling.querySelector('div.upmod');
} else {
var upVoteButton = this.keyboardLinks[this.activeIndex].previousSibling.querySelector('div.up') || this.keyboardLinks[this.activeIndex].previousSibling.querySelector('div.upmod');
}
RESUtils.click(upVoteButton);
},
downVote: function() {
if (this.keyboardLinks[this.activeIndex].previousSibling.tagName == 'A') {
var downVoteButton = this.keyboardLinks[this.activeIndex].previousSibling.previousSibling.querySelector('div.down') || this.keyboardLinks[this.activeIndex].previousSibling.previousSibling.querySelector('div.downmod');
} else {
var downVoteButton = this.keyboardLinks[this.activeIndex].previousSibling.querySelector('div.down') || this.keyboardLinks[this.activeIndex].previousSibling.querySelector('div.downmod');
}
RESUtils.click(downVoteButton);
},
saveLink: function() {
var saveLink = this.keyboardLinks[this.activeIndex].querySelector('form.save-button > span > a');
RESUtils.click(saveLink);
},
reply: function() {
// activeIndex = 0 means we're at the original post, not a comment
if ((this.activeIndex > 0) || (RESUtils.pageType() != 'comments')) {
var commentButtons = this.keyboardLinks[this.activeIndex].querySelectorAll('ul.buttons > li > a');
for (var i=0, len=commentButtons.length;i<len;i++) {
if (commentButtons[i].innerHTML == 'reply') {
RESUtils.click(commentButtons[i]);
}
}
} else {
infoBar = document.body.querySelector('.infobar');
// We're on the original post, so shift keyboard focus to the comment reply box.
if (infoBar) {
// uh oh, we must be in a subpage, there is no first comment box. The user probably wants to reply to the OP. Let's take them to the comments page.
var commentButton = this.keyboardLinks[this.activeIndex].querySelector('ul.buttons > li > a.comments');
location.href = commentButton.getAttribute('href');
} else {
var firstCommentBox = document.querySelector('.commentarea textarea[name=text]');
firstCommentBox.focus();
}
}
},
inbox: function() {
location.href = 'http://www.reddit.com/message/inbox/';
},
frontPage: function() {
location.href = 'http://www.reddit.com/';
},
nextPage: function() {
// if Never Ending Reddit is enabled, just scroll to the bottom. Otherwise, click the 'next' link.
if (modules['neverEndingReddit'].isEnabled()) {
this.moveBottom();
} else {
// get the first link to the next page of reddit...
var nextPrevLinks = document.body.querySelectorAll('.content .nextprev a');
if (nextPrevLinks.length > 0) {
var nextLink = nextPrevLinks[nextPrevLinks.length-1];
// RESUtils.click(nextLink);
location.href = nextLink.getAttribute('href');
}
}
},
prevPage: function() {
// if Never Ending Reddit is enabled, do nothing. Otherwise, click the 'prev' link.
if (modules['neverEndingReddit'].isEnabled()) {
return false;
} else {
// get the first link to the next page of reddit...
var nextPrevLinks = document.body.querySelectorAll('.content .nextprev a');
if (nextPrevLinks.length > 0) {
var prevLink = nextPrevLinks[0];
// RESUtils.click(prevLink);
location.href = prevLink.getAttribute('href');
}
}
},
commentLink: function(num) {
if (this.options.commentsLinkNumbers.value) {
var links = this.keyboardLinks[this.activeIndex].querySelectorAll('div.md a');
if (typeof(links[num]) != 'undefined') {
RESUtils.click(links[num]);
}
}
}
};
// user tagger functions
modules['userTagger'] = {
moduleID: 'userTagger',
moduleName: 'User Tagger',
options: {
defaultMark: {
type: 'text',
value: '_',
description: 'clickable mark for users with no tag'
},
hardIgnore: {
type: 'boolean',
value: false,
description: 'If "hard ignore" is off, only post titles and comment text is hidden. If it is on, the entire block is hidden - note that this would make it difficult/impossible for you to unignore a person.'
},
colorUser: {
type: 'boolean',
value: true,
description: 'Color users based on cumulative upvotes / downvotes'
},
hoverInfo: {
type: 'boolean',
value: true,
description: 'Show information on user (karma, how long they\'ve been a redditor) on hover.'
},
hoverDelay: {
type: 'text',
value: 400,
description: 'Delay, in milliseconds, before hover tooltip loads. Default is 400.'
},
fadeDelay: {
type: 'text',
value: 200,
description: 'Delay, in milliseconds, before hover tooltip fades away. Default is 200.'
},
USDateFormat: {
type: 'boolean',
value: false,
description: 'Show date (redditor since...) in US format (i.e. 08-31-2010)'
}
},
description: 'Adds a great deal of customization around users - tagging them, ignoring them, and more.',
isEnabled: function() {
return RESConsole.getModulePrefs(this.moduleID);
},
isMatchURL: function() {
return RESUtils.isMatchURL(this.moduleID);
},
include: new Array(
/^https?:\/\/([-\w\.]+\.)?reddit\.com\/[-\w\.]*/i
),
exclude: new Array(
/^http:\/\/www\.reddit\.com\/prefs\//i
),
go: function() {
if ((this.isEnabled()) && (this.isMatchURL())) {
// get this module's options...
RESUtils.getOptions(this.moduleID);
// set up an array to cache user data
this.authorInfoCache = Array();
if (this.options.colorUser.value) {
// add an event listener to vote buttons...
var voteButtons = document.body.querySelectorAll('.arrow');
for (var i=0, len=voteButtons.length;i<len;i++) {
voteButtons[i].addEventListener('click', function(e) {
var thisAuthorA = this.parentNode.nextSibling.querySelector('p.tagline a.author');
var thisVWobj = this.parentNode.nextSibling.querySelector('.voteWeight');
var thisAuthor = thisAuthorA.text;
var votes = parseInt(localStorage.getItem('reddituser.votes.'+thisAuthor)) || 0;
if ((hasClass(this, 'upmod')) || (hasClass(this, 'down'))) {
// upmod = upvote. down = undo of downvote.
votes = votes + 1;
} else if ((hasClass(this, 'downmod')) || (hasClass(this, 'up'))) {
// downmod = downvote. up = undo of downvote.
votes = votes - 1;
}
localStorage.setItem('reddituser.votes.'+thisAuthor, votes);
modules['userTagger'].colorUser(thisVWobj, thisAuthor, votes);
}, true);
}
}
// add tooltip to document body...
var css = '#userTaggerToolTip { display: none; position: absolute; padding: 5px; background-color: #CCCCCC; border: 3px solid #999999; border-radius: 5px 5px 5px 5px; -moz-border-radius: 5px 5px 5px 5px; -webkit-border-radius: 5px 5px 5px 5px; width: 200px; font-size: 14px; color: #000000; }';
css += '#userTaggerToolTip label { margin-top: 5px; clear: both; float: left; width: 40px; }';
css += '#userTaggerToolTip input, #userTaggerToolTip select { margin-top: 5px; float: left; width: 80px; }';
css += '.ignoredUserComment { color: #CACACA; padding: 3px; font-size: 10px; }';
css += '.ignoredUserPost { color: #CACACA; padding: 3px; font-size: 10px; }';
css += 'a.voteWeight { text-decoration: none; color: #336699; }';
css += 'a.voteWeight:hover { text-decoration: none; }';
css += '#authorInfoToolTip { display: none; position: absolute; padding: 5px; background-color: #EFF7FF; border: 3px solid #CEE3F8; border-radius: 5px 5px 5px 5px; -moz-border-radius: 5px 5px 5px 5px; -webkit-border-radius: 5px 5px 5px 5px; width: 180px; font-size: 11px; color: #000000; }';
css += '#benefits { width: 200px; margin-left: 0px; }';
RESUtils.addCSS(css);
//GM_addStyle(css);
this.userTaggerToolTip = createElementWithID('div','userTaggerToolTip');
thisHTML = '<input type="hidden" id="userTaggerName" value="">';
thisHTML += '<label for="userTaggerTag">Tag</label> <input type="text" id="userTaggerTag" value="">';
thisHTML += '<label for="userTaggerColor">Color</label> <select id="userTaggerColor">';
for (color in this.bgToTextColorMap) {
thisValue = color;
if (thisValue == 'none') thisValue = '';
thisHTML += '<option style="background-color: '+color+'; color: '+this.bgToTextColorMap[color]+'" value="'+thisValue+'">'+color+'</option>';
}
thisHTML += '</select><br>';
thisHTML += '<label for="userTaggerTag">Ignore</label> <input type="checkbox" id="userTaggerIgnore" value="true">';
this.userTaggerToolTip.innerHTML = thisHTML;
userTaggerSave = createElementWithID('input', 'userTaggerSave');
userTaggerSave.setAttribute('type','button');
userTaggerSave.setAttribute('value','Save');
this.userTaggerToolTip.appendChild(userTaggerSave);
userTaggerSave.addEventListener('click',function(e) {
var thisName = document.getElementById('userTaggerName').value;
var thisTag = document.getElementById('userTaggerTag').value;
var thisColor = document.getElementById('userTaggerColor').value;
var thisIgnore = document.getElementById('userTaggerIgnore').checked;
modules['userTagger'].setUserTag(thisName, thisTag, thisColor, thisIgnore);
}, true);
document.body.appendChild(this.userTaggerToolTip);
if (this.options.hoverInfo.value) {
this.authorInfoToolTip = createElementWithID('div','authorInfoToolTip');
this.authorInfoToolTip.addEventListener('mouseover', function(e) {
if (typeof(modules['userTagger'].hideTimer) != 'undefined') {
clearTimeout(modules['userTagger'].hideTimer);
}
}, false);
this.authorInfoToolTip.addEventListener('mouseout', function(e) {
if (e.target.getAttribute('class') != 'hoverAuthor') {
modules['userTagger'].hideTimer = setTimeout(function() {
modules['userTagger'].hideAuthorInfo();
}, modules['userTagger'].options.fadeDelay.value);
}
}, false);
document.body.appendChild(this.authorInfoToolTip);
}
document.getElementById('userTaggerTag').addEventListener('keydown', function(e) {
if (e.keyCode == 27) {
// close prompt.
modules['userTagger'].closeUserTagPrompt();
}
}, true);
//console.log('before applytags: ' + Date());
this.applyTags();
//console.log('after applytags: ' + Date());
// listen for new DOM nodes so that modules like autopager, river of reddit, etc still get user tags
document.body.addEventListener('DOMNodeInserted', function(event) {
if ((event.target.tagName == 'DIV') && (event.target.getAttribute('id') == 'siteTable')) {
modules['userTagger'].applyTags();
}
}, true);
var userpagere = new RegExp(/http:\/\/www.reddit.com\/user\/[-\w\.]+\/?/i);
if (userpagere.test(location.href)) {
var friendButton = document.querySelector('.titlebox .fancy-toggle-button');
if ((typeof(friendButton) != 'undefined') && (friendButton != null)) {
var firstAuthor = document.querySelector('a.author');
if (typeof(firstAuthor) != 'undefined') {
var thisFriendComment = firstAuthor.getAttribute('title');
(thisFriendComment != null) ? thisFriendComment = thisFriendComment.substring(8,thisFriendComment.length-1) : thisFriendComment = '';
} else {
var thisFriendComment = '';
}
var benefitsForm = document.createElement('div');
var thisUser = document.querySelector('.titlebox > h1').innerHTML;
benefitsForm.innerHTML = '<form action="/post/friendnote" id="friendnote-r9_2vt1" method="post" class="pretty-form medium-text friend-note" onsubmit="return post_form(this, \'friendnote\');"><input type="hidden" name="name" value="'+thisUser+'"><input type="text" maxlength="300" name="note" id="benefits" class="tiny" onfocus="$(this).parent().addClass(\'edited\')" value="'+thisFriendComment+'"><button onclick="$(this).parent().removeClass(\'edited\')" type="submit">submit</button><span class="status"></span></form>';
insertAfter( friendButton, benefitsForm );
}
}
}
},
bgToTextColorMap: {
'none':'black',
'aqua':'black',
'black':'white',
'blue':'white',
'fuchsia':'white',
'gray':'white',
'green':'white',
'lime':'black',
'maroon':'white',
'navy':'white',
'olive':'black',
'orange':'black',
'purple':'white',
'red':'black',
'silver':'black',
'teal':'white',
'white':'black',
'yellow':'black'
},
openUserTagPrompt: function(obj, username) {
thisXY=RESUtils.getXYpos(obj);
this.clickedTag = obj;
document.getElementById('userTaggerName').value = username;
document.getElementById('userTaggerTag').value = localStorage.getItem('reddituser.tag.'+username);
if (localStorage.getItem('reddituser.ignore.'+username)) {
document.getElementById('userTaggerIgnore').checked = true;
}
RESUtils.setSelectValue(document.getElementById('userTaggerColor'), localStorage.getItem('reddituser.color.'+username));
this.userTaggerToolTip.setAttribute('style', 'display: block; top: ' + thisXY.y + 'px; left: ' + thisXY.x + 'px;');
document.getElementById('userTaggerTag').focus();
return false;
},
closeUserTagPrompt: function() {
this.userTaggerToolTip.setAttribute('style','display: none');
},
setUserTag: function(username, tag, color, ignore) {
if ((tag != null) && (tag != '')) {
localStorage.setItem('reddituser.tag.'+username, tag);
localStorage.setItem('reddituser.color.'+username, color);
this.clickedTag.setAttribute('style', 'background-color: '+color+'; color: ' + this.bgToTextColorMap[color]);
this.clickedTag.innerHTML = tag;
} else {
localStorage.removeItem('reddituser.tag.'+username);
localStorage.removeItem('reddituser.color.'+username);
this.clickedTag.setAttribute('style', 'background-color: none');
this.clickedTag.innerHTML = this.options.defaultMark.value;
}
if (ignore) {
localStorage.setItem('reddituser.ignore.'+username, true);
} else {
localStorage.removeItem('reddituser.ignore.'+username, true);
}
this.closeUserTagPrompt();
},
applyTags: function() {
this.authors = document.querySelectorAll('.noncollapsed a.author, p.tagline a.author');
this.authorCount = this.authors.length;
this.authori = 0;
(function(){
modules['userTagger'].authorCount;
var chunkLength = Math.min((modules['userTagger'].authorCount - modules['userTagger'].authori), 15);
for (var i=0;i<chunkLength;i++) {
modules['userTagger'].applyTagToAuthor(modules['userTagger'].authori);
modules['userTagger'].authori++;
}
if (modules['userTagger'].authori < modules['userTagger'].authorCount) {
setTimeout(arguments.callee, 500);
}
})();
},
applyTagToAuthor: function(authorNum) {
var userObject = new Array();
var thisAuthorObj = this.authors[authorNum];
if (!(hasClass(thisAuthorObj,'userTagged'))) {
if (this.options.hoverInfo.value) {
// add event listener to hover, so we can grab user data on hover...
thisAuthorObj.addEventListener('mouseover', function(e) {
modules['userTagger'].showTimer = setTimeout(function() {
modules['userTagger'].showAuthorInfo(thisAuthorObj);
}, modules['userTagger'].options.hoverDelay.value);
}, false);
thisAuthorObj.addEventListener('mouseout', function(e) {
clearTimeout(modules['userTagger'].showTimer);
}, false);
}
var thisAuthor = thisAuthorObj.text;
addClass(thisAuthorObj, 'userTagged');
if (typeof(userObject[thisAuthor]) == 'undefined') {
var thisVotes = parseInt(localStorage.getItem('reddituser.votes.'+thisAuthor)) || 0;
userObject[thisAuthor] = {
tag: localStorage.getItem('reddituser.tag.'+thisAuthor),
color: localStorage.getItem('reddituser.color.'+thisAuthor),
ignore: localStorage.getItem('reddituser.ignore.'+thisAuthor),
votes: thisVotes
}
}
var thisTag = userObject[thisAuthor].tag;
var thisColor = userObject[thisAuthor].color;
if (!(thisTag)) thisTag = this.options.defaultMark.value;
var userTagFrag = document.createDocumentFragment();
var userTagLink = document.createElement('a');
userTagLink.setAttribute('class','userTagLink');
userTagLink.innerHTML = thisTag;
if (thisColor) {
userTagLink.setAttribute('style','background-color: '+thisColor+'; color: '+this.bgToTextColorMap[thisColor]);
}
userTagLink.setAttribute('username',thisAuthor);
userTagLink.setAttribute('title','set a tag');
userTagLink.setAttribute('href','javascript:void(0)');
userTagLink.addEventListener('click', function(e) {
modules['userTagger'].openUserTagPrompt(this, this.getAttribute('username'));
}, true);
var userTag = document.createElement('span');
var lp = document.createTextNode(' (');
var rp = document.createTextNode(')');
userTag.appendChild(userTagLink);
userTagFrag.appendChild(lp);
userTagFrag.appendChild(userTag);
userTagFrag.appendChild(rp);
if (this.options.colorUser.value) {
var userVoteFrag = document.createDocumentFragment();
var spacer = document.createTextNode(' ');
userVoteFrag.appendChild(spacer);
var userVoteWeight = document.createElement('a');
userVoteWeight.setAttribute('href','javascript:void(0)');
userVoteWeight.setAttribute('class','voteWeight');
userVoteWeight.innerHTML = '[vw]';
this.colorUser(userVoteWeight, thisAuthor, userObject[thisAuthor].votes);
userVoteFrag.appendChild(userVoteWeight);
userTagFrag.appendChild(userVoteFrag);
}
insertAfter( thisAuthorObj, userTagFrag );
// thisAuthorObj.innerHTML += userTagFrag.innerHTML;
thisIgnore = userObject[thisAuthor].ignore;
if (thisIgnore) {
if (this.options.hardIgnore.value) {
var commentsre = new RegExp(/comments\/[-\w\.\/]/i);
if (commentsre.test(location.href)) {
var thisComment = thisAuthorObj.parentNode.parentNode;
// hide comment block first...
thisComment.style.display = 'none';
// hide associated voting block...
thisComment.previousSibling.style.display = 'none';
} else {
var thisPost = thisAuthorObj.parentNode.parentNode.parentNode;
// hide post block first...
thisPost.style.display = 'none';
// hide associated voting block...
thisPost.previousSibling.style.display = 'none';
}
} else {
var commentsre = new RegExp(/comments\/[-\w\.\/]/i);
if (commentsre.test(location.href)) {
var thisComment = thisAuthorObj.parentNode.parentNode.querySelector('.usertext');
thisComment.innerHTML = thisAuthor + ' is an ignored user';
addClass(thisComment, 'ignoredUserComment');
} else {
var thisPost = authors[i].parentNode.parentNode.parentNode.querySelector('p.title');
thisPost.innerHTML = thisAuthor + ' is an ignored user';
thisPost.setAttribute('class','ignoredUserPost');
}
}
}
}
},
colorUser: function(obj, author, votes) {
if (this.options.colorUser.value) {
var red = 255;
var green = 255;
var blue = 255;
var voteString = '+';
if (votes > 0) {
red = Math.max(0, (255-(8*votes)));
green = 255;
blue = Math.max(0, (255-(8*votes)));
} else if (votes < 0) {
red = 255;
green = Math.max(0, (255-Math.abs(8*votes)));
blue = Math.max(0, (255-Math.abs(8*votes)));
voteString = '';
}
voteString = voteString + votes;
var rgb='rgb('+red+','+green+','+blue+')';
if (obj != null) {
if (votes == 0) {
obj.style.display = 'none';
} else {
obj.style.display = 'inline';
obj.style.backgroundColor = rgb;
obj.setAttribute('title','your votes for '+author+': '+voteString);
}
}
}
},
showAuthorInfo: function(obj) {
thisXY=RESUtils.getXYpos(obj);
this.authorInfoToolTip.innerHTML = '<a class="hoverAuthor" href="/user/'+obj.text+'/">'+obj.text+'</a>:<br><img src="'+RESConsole.loader+'"> loading...';
this.authorInfoToolTip.setAttribute('style', 'top: ' + (thisXY.y - 10) + 'px; left: ' + (thisXY.x - 10) + 'px;');
RESUtils.fadeElementIn(this.authorInfoToolTip, 0.3);
var thisUserName = obj.text;
if (typeof(this.authorInfoCache[thisUserName]) != 'undefined') {
this.writeAuthorInfo(this.authorInfoCache[thisUserName]);
} else {
GM_xmlhttpRequest({
method: "GET",
url: "http://www.reddit.com/user/" + thisUserName + "/about.json",
onload: function(response) {
var thisResponse = JSON.parse(response.responseText);
modules['userTagger'].authorInfoCache[thisUserName] = thisResponse;
modules['userTagger'].writeAuthorInfo(thisResponse);
}
});
}
},
writeAuthorInfo: function(jsonData) {
var utctime = jsonData.data.created;
var d = new Date(utctime*1000);
var userHTML = '<a class="hoverAuthor" href="/user/'+jsonData.data.name+'/">'+jsonData.data.name+'</a>:';
userHTML += '<br>Link Karma: ' + jsonData.data.link_karma;
userHTML += '<br>Comment Karma: ' + jsonData.data.comment_karma;
userHTML += '<br>Redditor since: ' + RESUtils.niceDate(d, this.options.USDateFormat.value) + '<br>('+RESUtils.niceDateDiff(d)+')';
this.authorInfoToolTip.innerHTML = userHTML;
},
hideAuthorInfo: function(obj) {
// this.authorInfoToolTip.setAttribute('style', 'display: none');
RESUtils.fadeElementOut(this.authorInfoToolTip, 0.3);
}
};
// full comments linker
modules['commentsLinker'] = {
moduleID: 'commentsLinker',
moduleName: 'Full Comments Linker',
options: {
fullCommentsText: {
type: 'text',
value: 'full comments',
description: 'text of full comments link'
}
},
description: 'Adds a link to a post\'s full comments page beyond just \'context\', etc.',
isEnabled: function() {
return RESConsole.getModulePrefs(this.moduleID);
},
isMatchURL: function() {
return RESUtils.isMatchURL(this.moduleID);
},
include: Array(
/http:\/\/www.reddit.com\/user\/[-\w\.]+\/?/i,
/http:\/\/www.reddit.com\/message\/[-\w\.]*\/?/i,
/http:\/\/www.reddit.com\/message\/comments\/[-\w\.]+/i
),
go: function() {
if ((this.isEnabled()) && (this.isMatchURL())) {
// get this module's options...
RESUtils.getOptions(this.moduleID);
var entries = document.querySelectorAll('#siteTable .entry');
for (var i=0, len=entries.length; i<len;i++) {
// bug in chrome, barfs on for i in loops with queryselectorall...
if (i == 'length') break;
var linkEle = entries[i].querySelector('A.bylink');
var thisCommentsLink = '';
if ((typeof(linkEle) != 'undefined') && (linkEle != null)) {
thisCommentsLink = linkEle.getAttribute('href');
}
if (thisCommentsLink != '') {
thisCommentsSplit = thisCommentsLink.split("/");
thisCommentsSplit.pop();
thisCommentsLink = thisCommentsSplit.join("/");
linkList = entries[i].querySelector('.flat-list');
var fullCommentsLink = document.createElement('li');
fullCommentsLink.innerHTML = '<a class="redditFullComments" href="' + thisCommentsLink + '">'+ this.options.fullCommentsText.value +'</a>';
linkList.appendChild(fullCommentsLink);
}
}
}
}
};
modules['singleClick'] = {
moduleID: 'singleClick',
moduleName: 'Single Click Opener',
options: {
openOrder: {
type: 'enum',
values: [
{ name: 'open comments then link', value: 'commentsfirst' },
{ name: 'open link then comments', value: 'linkfirst' }
],
value: 'commentsfirst',
description: 'What order to open the link/comments in.'
},
hideLEC: {
type: 'boolean',
value: false,
description: 'Hide the [l=c] when the link is the same as the comments page'
}
},
description: 'Adds an [l+c] link that opens a link and the comments page in new tabs for you in one click.',
isEnabled: function() {
return RESConsole.getModulePrefs(this.moduleID);
},
isMatchURL: function() {
return RESUtils.isMatchURL(this.moduleID);
},
isURLMatch: function() {
},
include: new Array(
/http:\/\/www.reddit.com\/[\?]*/i,
/http:\/\/www.reddit.com\/r\/[-\w\._]*\//i
),
exclude: new Array(
/http:\/\/www.reddit.com\/r\/[-\w\._\/\?]*\/comments[-\w\._\/\?=]*/i
),
go: function() {
if ((this.isEnabled()) && (this.isMatchURL())) {
// get this module's options...
RESUtils.getOptions(this.moduleID);
// do stuff here!
this.applyLinks();
// listen for new DOM nodes so that modules like autopager, river of reddit, etc still get l+c links...
document.body.addEventListener('DOMNodeInserted', function(event) {
if ((event.target.tagName == 'DIV') && (event.target.getAttribute('id') == 'siteTable')) {
modules['singleClick'].applyLinks();
}
}, true);
}
},
applyLinks: function() {
var entries = document.querySelectorAll('#siteTable .entry');
for (var i=0, len=entries.length; i<len;i++) {
if ((typeof(entries[i]) != 'undefined') && (!(hasClass(entries[i],'lcTagged')))) {
// bug in chrome, barfs on for i in loops with queryselectorall...
if (i == 'length') break;
addClass(entries[i],'lcTagged')
thisLA = entries[i].querySelector('A.title');
if (thisLA != null) {
thisLink = thisLA.getAttribute('href');
thisComments = entries[i].querySelector('.comments');
if (!(thisLink.match(/^http/i))) {
thisLink = 'http://' + document.domain + thisLink;
}
thisUL = entries[i].querySelector('.flat-list');
var singleClickLI = document.createElement('li');
var singleClickLink = document.createElement('a');
singleClickLink.setAttribute('href',thisLink);
singleClickLink.setAttribute('class','redditSingleClick');
singleClickLink.setAttribute('thisLink',thisLink);
singleClickLink.setAttribute('thisComments',thisComments);
if (thisLink != thisComments) {
singleClickLink.innerHTML = '[l+c]';
} else if (!(this.options.hideLEC.value)) {
singleClickLink.innerHTML = '[l=c]';
}
singleClickLI.appendChild(singleClickLink);
thisUL.appendChild(singleClickLI);
singleClickLink.addEventListener('click', function(e) {
e.preventDefault();
// check if it's a relative link (no http://domain) because chrome barfs on these when creating a new tab...
var thisLink = this.getAttribute('thisLink')
if (typeof(chrome) != 'undefined') {
thisJSON = {
requestType: 'singleClick',
linkURL: thisLink,
openOrder: modules['singleClick'].options.openOrder.value,
commentsURL: this.getAttribute('thisComments'),
button: event.button
}
chrome.extension.sendRequest(thisJSON, function(response) {
// send message to background.html to open new tabs...
return true;
});
} else if (typeof(safari) != 'undefined') {
thisJSON = {
requestType: 'singleClick',
linkURL: thisLink,
openOrder: modules['singleClick'].options.openOrder.value,
commentsURL: this.getAttribute('thisComments'),
button: event.button
}
safari.self.tab.dispatchMessage("singleClick", thisJSON);
} else {
if (modules['singleClick'].options.openOrder.value == 'commentsfirst') {
if (this.getAttribute('thisLink') != this.getAttribute('thisComments')) {
window.open(this.getAttribute('thisComments'));
}
window.open(this.getAttribute('thisLink'));
} else {
window.open(this.getAttribute('thisLink'));
if (this.getAttribute('thisLink') != this.getAttribute('thisComments')) {
window.open(this.getAttribute('thisComments'));
}
}
}
}, true);
}
}
}
}
};
modules['commentPreview'] = {
moduleID: 'commentPreview',
moduleName: 'Live Comment Preview',
options: {
// any configurable options you have go here...
},
description: 'Provides a live preview of comments, as well as shortcuts for easier markdown.',
isEnabled: function() {
return RESConsole.getModulePrefs(this.moduleID);
},
include: Array(
/http:\/\/www.reddit.com\/[-\w\.\/]+\/comments\/[-\w\.]+/i,
/http:\/\/www.reddit.com\/message\/[-\w\.]*\/?[-\w\.]*/i,
/http:\/\/www.reddit.com\/r\/[-\w\.]*\/submit\/?/i,
/http:\/\/www.reddit.com\/submit\/?/i
),
isMatchURL: function() {
return RESUtils.isMatchURL(this.moduleID);
},
go: function() {
if ((this.isEnabled()) && (this.isMatchURL())) {
// get this module's options...
RESUtils.getOptions(this.moduleID);
// do stuff now!
// this is where your code goes...
/*
2009-08-30
- Fixed top comment box.
- Added markdown toolbar.
2009-02-05
- Fixed the preview clearing after clicking "comment". That was broken, too.
2009-02-04
- Fix because reddit broke it.
*/
//
// showdown.js -- A javascript port of Markdown.
//
// Copyright (c) 2007 John Fraser.
//
// Original Markdown Copyright (c) 2004-2005 John Gruber
// <http://daringfireball.net/projects/markdown/>
//
// Redistributable under a BSD-style open source license.
// See license.txt for more information.
//
// The full source distribution is at:
//
// A A L
// T C A
// T K B
//
// <http://www.attacklab.net/>
//
//
// Wherever possible, Showdown is a straight, line-by-line port
// of the Perl version of Markdown.
//
// This is not a normal parser design; it's basically just a
// series of string substitutions. It's hard to read and
// maintain this way, but keeping Showdown close to the original
// design makes it easier to port new features.
//
// More importantly, Showdown behaves like markdown.pl in most
// edge cases. So web applications can do client-side preview
// in Javascript, and then build identical HTML on the server.
//
// This port needs the new RegExp functionality of ECMA 262,
// 3rd Edition (i.e. Javascript 1.5). Most modern web browsers
// should do fine. Even with the new regular expression features,
// We do a lot of work to emulate Perl's regex functionality.
// The tricky changes in this file mostly have the "attacklab:"
// label. Major or self-explanatory changes don't.
//
// Smart diff tools like Araxis Merge will be able to match up
// this file with markdown.pl in a useful way. A little tweaking
// helps: in a copy of markdown.pl, replace "#" with "//" and
// replace "$text" with "text". Be sure to ignore whitespace
// and line endings.
//
//
// Showdown usage:
//
// var text = "Markdown *rocks*.";
//
// var converter = new Showdown.converter();
// var html = converter.makeHtml(text);
//
// alert(html);
//
// Note: move the sample code to the bottom of this
// file before uncommenting it.
//
//
// Showdown namespace
//
var Showdown = {};
//
// converter
//
// Wraps all "globals" so that the only thing
// exposed is makeHtml().
//
Showdown.converter = function() {
//
// Globals:
//
// Global hashes, used by various utility routines
var g_urls;
var g_titles;
var g_html_blocks;
// Used to track when we're inside an ordered or unordered list
// (see _ProcessListItems() for details):
var g_list_level = 0;
this.makeHtml = function(text) {
//
// Main function. The order in which other subs are called here is
// essential. Link and image substitutions need to happen before
// _EscapeSpecialCharsWithinTagAttributes(), so that any *'s or _'s in the <a>
// and <img> tags get encoded.
//
// Clear the global hashes. If we don't clear these, you get conflicts
// from other articles when generating a page which contains more than
// one article (e.g. an index page that shows the N most recent
// articles):
g_urls = new Array();
g_titles = new Array();
g_html_blocks = new Array();
// Replace < with &lt; and > with &gt;
// This lets us use tilde as an escape char to avoid md5 hashes
// The choice of character is arbitray; anything that isn't
// magic in Markdown will work.
text = text.replace(/</g,"&lt;");
text = text.replace(/>/g,"&gt;");
// attacklab: Replace ~ with ~T
// This lets us use tilde as an escape char to avoid md5 hashes
// The choice of character is arbitray; anything that isn't
// magic in Markdown will work.
text = text.replace(/~/g,"~T");
// attacklab: Replace $ with ~D
// RegExp interprets $ as a special character
// when it's in a replacement string
text = text.replace(/\$/g,"~D");
// Standardize line endings
text = text.replace(/\r\n/g,"\n"); // DOS to Unix
text = text.replace(/\r/g,"\n"); // Mac to Unix
// Make sure text begins and ends with a couple of newlines:
text = "\n\n" + text + "\n\n";
// Convert all tabs to spaces.
text = _Detab(text);
// Strip any lines consisting only of spaces and tabs.
// This makes subsequent regexen easier to write, because we can
// match consecutive blank lines with /\n+/ instead of something
// contorted like /[ \t]*\n+/ .
text = text.replace(/^[ \t]+$/mg,"");
// Turn block-level HTML blocks into hash entries
text = _HashHTMLBlocks(text);
// Strip link definitions, store in hashes.
text = _StripLinkDefinitions(text);
text = _RunBlockGamut(text);
text = _UnescapeSpecialChars(text);
// attacklab: Restore dollar signs
text = text.replace(/~D/g,"$$");
// attacklab: Restore tildes
text = text.replace(/~T/g,"~");
return text;
}
var _StripLinkDefinitions = function(text) {
//
// Strips link definitions from text, stores the URLs and titles in
// hash references.
//
// Link defs are in the form: ^[id]: url "optional title"
/*
var text = text.replace(/
^[ ]{0,3}\[(.+)\]: // id = $1 attacklab: g_tab_width - 1
[ \t]*
\n? // maybe *one* newline
[ \t]*
<?(\S+?)>? // url = $2
[ \t]*
\n? // maybe one newline
[ \t]*
(?:
(\n*) // any lines skipped = $3 attacklab: lookbehind removed
["(]
(.+?) // title = $4
[")]
[ \t]*
)? // title is optional
(?:\n+|$)
/gm,
function(){...});
*/
var text = text.replace(/^[ ]{0,3}\[(.+)\]:[ \t]*\n?[ \t]*<?(\S+?)>?[ \t]*\n?[ \t]*(?:(\n*)["(](.+?)[")][ \t]*)?(?:\n+|\Z)/gm,
function (wholeMatch,m1,m2,m3,m4) {
m1 = m1.toLowerCase();
g_urls[m1] = _EncodeAmpsAndAngles(m2); // Link IDs are case-insensitive
if (m3) {
// Oops, found blank lines, so it's not a title.
// Put back the parenthetical statement we stole.
return m3+m4;
} else if (m4) {
g_titles[m1] = m4.replace(/"/g,"&quot;");
}
// Completely remove the definition from the text
return "";
}
);
return text;
}
var _HashHTMLBlocks = function(text) {
// attacklab: Double up blank lines to reduce lookaround
text = text.replace(/\n/g,"\n\n");
// Hashify HTML blocks:
// We only want to do this for block-level HTML tags, such as headers,
// lists, and tables. That's because we still want to wrap <p>s around
// "paragraphs" that are wrapped in non-block-level tags, such as anchors,
// phrase emphasis, and spans. The list of tags we're looking for is
// hard-coded:
var block_tags_a = "p|div|h[1-6]|blockquote|pre|table|dl|ol|ul|script|noscript|form|fieldset|iframe|math|ins|del"
var block_tags_b = "p|div|h[1-6]|blockquote|pre|table|dl|ol|ul|script|noscript|form|fieldset|iframe|math"
// First, look for nested blocks, e.g.:
// <div>
// <div>
// tags for inner block must be indented.
// </div>
// </div>
//
// The outermost tags must start at the left margin for this to match, and
// the inner nested divs must be indented.
// We need to do this before the next, more liberal match, because the next
// match will start at the first `<div>` and stop at the first `</div>`.
// attacklab: This regex can be expensive when it fails.
/*
var text = text.replace(/
( // save in $1
^ // start of line (with /m)
<($block_tags_a) // start tag = $2
\b // word break
// attacklab: hack around khtml/pcre bug...
[^\r]*?\n // any number of lines, minimally matching
</\2> // the matching end tag
[ \t]* // trailing spaces/tabs
(?=\n+) // followed by a newline
) // attacklab: there are sentinel newlines at end of document
/gm,function(){...}};
*/
text = text.replace(/^(<(p|div|h[1-6]|blockquote|pre|table|dl|ol|ul|script|noscript|form|fieldset|iframe|math|ins|del)\b[^\r]*?\n<\/\2>[ \t]*(?=\n+))/gm,hashElement);
//
// Now match more liberally, simply from `\n<tag>` to `</tag>\n`
//
/*
var text = text.replace(/
( // save in $1
^ // start of line (with /m)
<($block_tags_b) // start tag = $2
\b // word break
// attacklab: hack around khtml/pcre bug...
[^\r]*? // any number of lines, minimally matching
.*</\2> // the matching end tag
[ \t]* // trailing spaces/tabs
(?=\n+) // followed by a newline
) // attacklab: there are sentinel newlines at end of document
/gm,function(){...}};
*/
text = text.replace(/^(<(p|div|h[1-6]|blockquote|pre|table|dl|ol|ul|script|noscript|form|fieldset|iframe|math)\b[^\r]*?.*<\/\2>[ \t]*(?=\n+)\n)/gm,hashElement);
// Special case just for <hr />. It was easier to make a special case than
// to make the other regex more complicated.
/*
text = text.replace(/
( // save in $1
\n\n // Starting after a blank line
[ ]{0,3}
(<(hr) // start tag = $2
\b // word break
([^<>])*? //
\/?>) // the matching end tag
[ \t]*
(?=\n{2,}) // followed by a blank line
)
/g,hashElement);
*/
text = text.replace(/(\n[ ]{0,3}(<(hr)\b([^<>])*?\/?>)[ \t]*(?=\n{2,}))/g,hashElement);
// Special case for standalone HTML comments:
/*
text = text.replace(/
( // save in $1
\n\n // Starting after a blank line
[ ]{0,3} // attacklab: g_tab_width - 1
<!
(--[^\r]*?--\s*)+
>
[ \t]*
(?=\n{2,}) // followed by a blank line
)
/g,hashElement);
*/
text = text.replace(/(\n\n[ ]{0,3}<!(--[^\r]*?--\s*)+>[ \t]*(?=\n{2,}))/g,hashElement);
// PHP and ASP-style processor instructions (<?...?> and <%...%>)
/*
text = text.replace(/
(?:
\n\n // Starting after a blank line
)
( // save in $1
[ ]{0,3} // attacklab: g_tab_width - 1
(?:
<([?%]) // $2
[^\r]*?
\2>
)
[ \t]*
(?=\n{2,}) // followed by a blank line
)
/g,hashElement);
*/
text = text.replace(/(?:\n\n)([ ]{0,3}(?:<([?%])[^\r]*?\2>)[ \t]*(?=\n{2,}))/g,hashElement);
// attacklab: Undo double lines (see comment at top of this function)
text = text.replace(/\n\n/g,"\n");
return text;
}
var hashElement = function(wholeMatch,m1) {
var blockText = m1;
// Undo double lines
blockText = blockText.replace(/\n\n/g,"\n");
blockText = blockText.replace(/^\n/,"");
// strip trailing blank lines
blockText = blockText.replace(/\n+$/g,"");
// Replace the element text with a marker ("~KxK" where x is its key)
blockText = "\n\n~K" + (g_html_blocks.push(blockText)-1) + "K\n\n";
return blockText;
};
var _RunBlockGamut = function(text) {
//
// These are all the transformations that form block-level
// tags like paragraphs, headers, and list items.
//
text = _DoHeaders(text);
// Do Horizontal Rules:
var key = hashBlock("<hr />");
text = text.replace(/^[ ]{0,2}([ ]?\*[ ]?){3,}[ \t]*$/gm,key);
text = text.replace(/^[ ]{0,2}([ ]?\-[ ]?){3,}[ \t]*$/gm,key);
text = text.replace(/^[ ]{0,2}([ ]?\_[ ]?){3,}[ \t]*$/gm,key);
text = _DoLists(text);
text = _DoCodeBlocks(text);
text = _DoBlockQuotes(text);
// We already ran _HashHTMLBlocks() before, in Markdown(), but that
// was to escape raw HTML in the original Markdown source. This time,
// we're escaping the markup we've just created, so that we don't wrap
// <p> tags around block-level tags.
text = _HashHTMLBlocks(text);
text = _FormParagraphs(text);
return text;
}
var _RunSpanGamut = function(text) {
//
// These are all the transformations that occur *within* block-level
// tags like paragraphs, headers, and list items.
//
text = _DoCodeSpans(text);
text = _EscapeSpecialCharsWithinTagAttributes(text);
text = _EncodeBackslashEscapes(text);
// Process anchor and image tags. Images must come first,
// because ![foo][f] looks like an anchor.
text = _DoImages(text);
text = _DoAnchors(text);
// Make links out of things like `<http://example.com/>`
// Must come after _DoAnchors(), because you can use < and >
// delimiters in inline links like [this](<url>).
text = _DoAutoLinks(text);
text = _EncodeAmpsAndAngles(text);
text = _DoItalicsAndBold(text);
// Do hard breaks:
text = text.replace(/ +\n/g," <br />\n");
return text;
}
var _EscapeSpecialCharsWithinTagAttributes = function(text) {
//
// Within tags -- meaning between < and > -- encode [\ ` * _] so they
// don't conflict with their use in Markdown for code, italics and strong.
//
// Build a regex to find HTML tags and comments. See Friedl's
// "Mastering Regular Expressions", 2nd Ed., pp. 200-201.
var regex = /(<[a-z\/!$]("[^"]*"|'[^']*'|[^'">])*>|<!(--.*?--\s*)+>)/gi;
text = text.replace(regex, function(wholeMatch) {
var tag = wholeMatch.replace(/(.)<\/?code>(?=.)/g,"$1`");
tag = escapeCharacters(tag,"\\`*_");
return tag;
});
return text;
}
var _DoAnchors = function(text) {
//
// Turn Markdown link shortcuts into XHTML <a> tags.
//
//
// First, handle reference-style links: [link text] [id]
//
/*
text = text.replace(/
( // wrap whole match in $1
\[
(
(?:
\[[^\]]*\] // allow brackets nested one level
|
[^\[] // or anything else
)*
)
\]
[ ]? // one optional space
(?:\n[ ]*)? // one optional newline followed by spaces
\[
(.*?) // id = $3
\]
)()()()() // pad remaining backreferences
/g,_DoAnchors_callback);
*/
text = text.replace(/(\[((?:\[[^\]]*\]|[^\[\]])*)\][ ]?(?:\n[ ]*)?\[(.*?)\])()()()()/g,writeAnchorTag);
//
// Next, inline-style links: [link text](url "optional title")
//
/*
text = text.replace(/
( // wrap whole match in $1
\[
(
(?:
\[[^\]]*\] // allow brackets nested one level
|
[^\[\]] // or anything else
)
)
\]
\( // literal paren
[ \t]*
() // no id, so leave $3 empty
<?(.*?)>? // href = $4
[ \t]*
( // $5
(['"]) // quote char = $6
(.*?) // Title = $7
\6 // matching quote
[ \t]* // ignore any spaces/tabs between closing quote and )
)? // title is optional
\)
)
/g,writeAnchorTag);
*/
text = text.replace(/(\[((?:\[[^\]]*\]|[^\[\]])*)\]\([ \t]*()<?(.*?)>?[ \t]*((['"])(.*?)\6[ \t]*)?\))/g,writeAnchorTag);
//
// Last, handle reference-style shortcuts: [link text]
// These must come last in case you've also got [link test][1]
// or [link test](/foo)
//
/*
text = text.replace(/
( // wrap whole match in $1
\[
([^\[\]]+) // link text = $2; can't contain '[' or ']'
\]
)()()()()() // pad rest of backreferences
/g, writeAnchorTag);
*/
text = text.replace(/(\[([^\[\]]+)\])()()()()()/g, writeAnchorTag);
return text;
}
var writeAnchorTag = function(wholeMatch,m1,m2,m3,m4,m5,m6,m7) {
if (m7 == undefined) m7 = "";
var whole_match = m1;
var link_text = m2;
var link_id = m3.toLowerCase();
var url = m4;
var title = m7;
if (url == "") {
if (link_id == "") {
// lower-case and turn embedded newlines into spaces
link_id = link_text.toLowerCase().replace(/ ?\n/g," ");
}
url = "#"+link_id;
if (g_urls[link_id] != undefined) {
url = g_urls[link_id];
if (g_titles[link_id] != undefined) {
title = g_titles[link_id];
}
}
else {
if (whole_match.search(/\(\s*\)$/m)>-1) {
// Special case for explicit empty url
url = "";
} else {
return whole_match;
}
}
}
url = escapeCharacters(url,"*_");
var result = "<a href=\"" + url + "\"";
if (title != "") {
title = title.replace(/"/g,"&quot;");
title = escapeCharacters(title,"*_");
result += " title=\"" + title + "\"";
}
result += ">" + link_text + "</a>";
return result;
}
var _DoImages = function(text) {
//
// Turn Markdown image shortcuts into <img> tags.
//
//
// First, handle reference-style labeled images: ![alt text][id]
//
/*
text = text.replace(/
( // wrap whole match in $1
!\[
(.*?) // alt text = $2
\]
[ ]? // one optional space
(?:\n[ ]*)? // one optional newline followed by spaces
\[
(.*?) // id = $3
\]
)()()()() // pad rest of backreferences
/g,writeImageTag);
*/
text = text.replace(/(!\[(.*?)\][ ]?(?:\n[ ]*)?\[(.*?)\])()()()()/g,writeImageTag);
//
// Next, handle inline images: ![alt text](url "optional title")
// Don't forget: encode * and _
/*
text = text.replace(/
( // wrap whole match in $1
!\[
(.*?) // alt text = $2
\]
\s? // One optional whitespace character
\( // literal paren
[ \t]*
() // no id, so leave $3 empty
<?(\S+?)>? // src url = $4
[ \t]*
( // $5
(['"]) // quote char = $6
(.*?) // title = $7
\6 // matching quote
[ \t]*
)? // title is optional
\)
)
/g,writeImageTag);
*/
text = text.replace(/(!\[(.*?)\]\s?\([ \t]*()<?(\S+?)>?[ \t]*((['"])(.*?)\6[ \t]*)?\))/g,writeImageTag);
return text;
}
var writeImageTag = function(wholeMatch,m1,m2,m3,m4,m5,m6,m7) {
var whole_match = m1;
var alt_text = m2;
var link_id = m3.toLowerCase();
var url = m4;
var title = m7;
if (!title) title = "";
if (url == "") {
if (link_id == "") {
// lower-case and turn embedded newlines into spaces
link_id = alt_text.toLowerCase().replace(/ ?\n/g," ");
}
url = "#"+link_id;
if (g_urls[link_id] != undefined) {
url = g_urls[link_id];
if (g_titles[link_id] != undefined) {
title = g_titles[link_id];
}
}
else {
return whole_match;
}
}
alt_text = alt_text.replace(/"/g,"&quot;");
url = escapeCharacters(url,"*_");
var result = "<img src=\"" + url + "\" alt=\"" + alt_text + "\"";
// attacklab: Markdown.pl adds empty title attributes to images.
// Replicate this bug.
//if (title != "") {
title = title.replace(/"/g,"&quot;");
title = escapeCharacters(title,"*_");
result += " title=\"" + title + "\"";
//}
result += " />";
return result;
}
var _DoHeaders = function(text) {
// Setext-style headers:
// Header 1
// ========
//
// Header 2
// --------
//
text = text.replace(/^(.+)[ \t]*\n=+[ \t]*\n+/gm,
function(wholeMatch,m1){return hashBlock("<h1>" + _RunSpanGamut(m1) + "</h1>");});
text = text.replace(/^(.+)[ \t]*\n-+[ \t]*\n+/gm,
function(matchFound,m1){return hashBlock("<h2>" + _RunSpanGamut(m1) + "</h2>");});
// atx-style headers:
// # Header 1
// ## Header 2
// ## Header 2 with closing hashes ##
// ...
// ###### Header 6
//
/*
text = text.replace(/
^(\#{1,6}) // $1 = string of #'s
[ \t]*
(.+?) // $2 = Header text
[ \t]*
\#* // optional closing #'s (not counted)
\n+
/gm, function() {...});
*/
text = text.replace(/^(\#{1,6})[ \t]*(.+?)[ \t]*\#*\n+/gm,
function(wholeMatch,m1,m2) {
var h_level = m1.length;
return hashBlock("<h" + h_level + ">" + _RunSpanGamut(m2) + "</h" + h_level + ">");
});
return text;
}
// This declaration keeps Dojo compressor from outputting garbage:
var _ProcessListItems;
var _DoLists = function(text) {
//
// Form HTML ordered (numbered) and unordered (bulleted) lists.
//
// attacklab: add sentinel to hack around khtml/safari bug:
// http://bugs.webkit.org/show_bug.cgi?id=11231
text += "~0";
// Re-usable pattern to match any entirel ul or ol list:
/*
var whole_list = /
( // $1 = whole list
( // $2
[ ]{0,3} // attacklab: g_tab_width - 1
([*+-]|\d+[.]) // $3 = first list item marker
[ \t]+
)
[^\r]+?
( // $4
~0 // sentinel for workaround; should be $
|
\n{2,}
(?=\S)
(?! // Negative lookahead for another list item marker
[ \t]*
(?:[*+-]|\d+[.])[ \t]+
)
)
)/g
*/
var whole_list = /^(([ ]{0,3}([*+-]|\d+[.])[ \t]+)[^\r]+?(~0|\n{2,}(?=\S)(?![ \t]*(?:[*+-]|\d+[.])[ \t]+)))/gm;
if (g_list_level) {
text = text.replace(whole_list,function(wholeMatch,m1,m2) {
var list = m1;
var list_type = (m2.search(/[*+-]/g)>-1) ? "ul" : "ol";
// Turn double returns into triple returns, so that we can make a
// paragraph for the last item in a list, if necessary:
list = list.replace(/\n{2,}/g,"\n\n\n");;
var result = _ProcessListItems(list);
// Trim any trailing whitespace, to put the closing `</$list_type>`
// up on the preceding line, to get it past the current stupid
// HTML block parser. This is a hack to work around the terrible
// hack that is the HTML block parser.
result = result.replace(/\s+$/,"");
result = "<"+list_type+">" + result + "</"+list_type+">\n";
return result;
});
} else {
whole_list = /(\n\n|^\n?)(([ ]{0,3}([*+-]|\d+[.])[ \t]+)[^\r]+?(~0|\n{2,}(?=\S)(?![ \t]*(?:[*+-]|\d+[.])[ \t]+)))/g;
text = text.replace(whole_list,function(wholeMatch,m1,m2,m3) {
var runup = m1;
var list = m2;
var list_type = (m3.search(/[*+-]/g)>-1) ? "ul" : "ol";
// Turn double returns into triple returns, so that we can make a
// paragraph for the last item in a list, if necessary:
var list = list.replace(/\n{2,}/g,"\n\n\n");;
var result = _ProcessListItems(list);
result = runup + "<"+list_type+">\n" + result + "</"+list_type+">\n";
return result;
});
}
// attacklab: strip sentinel
text = text.replace(/~0/,"");
return text;
}
_ProcessListItems = function(list_str) {
//
// Process the contents of a single ordered or unordered list, splitting it
// into individual list items.
//
// The $g_list_level global keeps track of when we're inside a list.
// Each time we enter a list, we increment it; when we leave a list,
// we decrement. If it's zero, we're not in a list anymore.
//
// We do this because when we're not inside a list, we want to treat
// something like this:
//
// I recommend upgrading to version
// 8. Oops, now this line is treated
// as a sub-list.
//
// As a single paragraph, despite the fact that the second line starts
// with a digit-period-space sequence.
//
// Whereas when we're inside a list (or sub-list), that line will be
// treated as the start of a sub-list. What a kludge, huh? This is
// an aspect of Markdown's syntax that's hard to parse perfectly
// without resorting to mind-reading. Perhaps the solution is to
// change the syntax rules such that sub-lists must start with a
// starting cardinal number; e.g. "1." or "a.".
g_list_level++;
// trim trailing blank lines:
list_str = list_str.replace(/\n{2,}$/,"\n");
// attacklab: add sentinel to emulate \z
list_str += "~0";
/*
list_str = list_str.replace(/
(\n)? // leading line = $1
(^[ \t]*) // leading whitespace = $2
([*+-]|\d+[.]) [ \t]+ // list marker = $3
([^\r]+? // list item text = $4
(\n{1,2}))
(?= \n* (~0 | \2 ([*+-]|\d+[.]) [ \t]+))
/gm, function(){...});
*/
list_str = list_str.replace(/(\n)?(^[ \t]*)([*+-]|\d+[.])[ \t]+([^\r]+?(\n{1,2}))(?=\n*(~0|\2([*+-]|\d+[.])[ \t]+))/gm,
function(wholeMatch,m1,m2,m3,m4){
var item = m4;
var leading_line = m1;
var leading_space = m2;
if (leading_line || (item.search(/\n{2,}/)>-1)) {
item = _RunBlockGamut(_Outdent(item));
}
else {
// Recursion for sub-lists:
item = _DoLists(_Outdent(item));
item = item.replace(/\n$/,""); // chomp(item)
item = _RunSpanGamut(item);
}
return "<li>" + item + "</li>\n";
}
);
// attacklab: strip sentinel
list_str = list_str.replace(/~0/g,"");
g_list_level--;
return list_str;
}
var _DoCodeBlocks = function(text) {
//
// Process Markdown `<pre><code>` blocks.
//
/*
text = text.replace(text,
/(?:\n\n|^)
( // $1 = the code block -- one or more lines, starting with a space/tab
(?:
(?:[ ]{4}|\t) // Lines must start with a tab or a tab-width of spaces - attacklab: g_tab_width
.*\n+
)+
)
(\n*[ ]{0,3}[^ \t\n]|(?=~0)) // attacklab: g_tab_width
/g,function(){...});
*/
// attacklab: sentinel workarounds for lack of \A and \Z, safari\khtml bug
text += "~0";
text = text.replace(/(?:\n\n|^)((?:(?:[ ]{4}|\t).*\n+)+)(\n*[ ]{0,3}[^ \t\n]|(?=~0))/g,
function(wholeMatch,m1,m2) {
var codeblock = m1;
var nextChar = m2;
codeblock = _EncodeCode( _Outdent(codeblock));
codeblock = _Detab(codeblock);
codeblock = codeblock.replace(/^\n+/g,""); // trim leading newlines
codeblock = codeblock.replace(/\n+$/g,""); // trim trailing whitespace
codeblock = "<pre><code>" + codeblock + "\n</code></pre>";
return hashBlock(codeblock) + nextChar;
}
);
// attacklab: strip sentinel
text = text.replace(/~0/,"");
return text;
}
var hashBlock = function(text) {
text = text.replace(/(^\n+|\n+$)/g,"");
return "\n\n~K" + (g_html_blocks.push(text)-1) + "K\n\n";
}
var _DoCodeSpans = function(text) {
//
// * Backtick quotes are used for <code></code> spans.
//
// * You can use multiple backticks as the delimiters if you want to
// include literal backticks in the code span. So, this input:
//
// Just type ``foo `bar` baz`` at the prompt.
//
// Will translate to:
//
// <p>Just type <code>foo `bar` baz</code> at the prompt.</p>
//
// There's no arbitrary limit to the number of backticks you
// can use as delimters. If you need three consecutive backticks
// in your code, use four for delimiters, etc.
//
// * You can use spaces to get literal backticks at the edges:
//
// ... type `` `bar` `` ...
//
// Turns to:
//
// ... type <code>`bar`</code> ...
//
/*
text = text.replace(/
(^|[^\\]) // Character before opening ` can't be a backslash
(`+) // $2 = Opening run of `
( // $3 = The code block
[^\r]*?
[^`] // attacklab: work around lack of lookbehind
)
\2 // Matching closer
(?!`)
/gm, function(){...});
*/
text = text.replace(/(^|[^\\])(`+)([^\r]*?[^`])\2(?!`)/gm,
function(wholeMatch,m1,m2,m3,m4) {
var c = m3;
c = c.replace(/^([ \t]*)/g,""); // leading whitespace
c = c.replace(/[ \t]*$/g,""); // trailing whitespace
c = _EncodeCode(c);
return m1+"<code>"+c+"</code>";
});
return text;
}
var _EncodeCode = function(text) {
//
// Encode/escape certain characters inside Markdown code runs.
// The point is that in code, these characters are literals,
// and lose their special Markdown meanings.
//
// Encode all ampersands; HTML entities are not
// entities within a Markdown code span.
text = text.replace(/&/g,"&amp;");
// Do the angle bracket song and dance:
text = text.replace(/</g,"&lt;");
text = text.replace(/>/g,"&gt;");
// Now, escape characters that are magic in Markdown:
text = escapeCharacters(text,"\*_{}[]\\",false);
// jj the line above breaks this:
//---
//* Item
// 1. Subitem
// special char: *
//---
return text;
}
var _DoItalicsAndBold = function(text) {
// <strong> must go first:
text = text.replace(/(\*\*|__)(?=\S)([^\r]*?\S[*_]*)\1/g,
"<strong>$2</strong>");
text = text.replace(/(\*|_)(?=\S)([^\r]*?\S)\1/g,
"<em>$2</em>");
return text;
}
var _DoBlockQuotes = function(text) {
/*
text = text.replace(/
( // Wrap whole match in $1
(
^[ \t]*>[ \t]? // '>' at the start of a line
.+\n // rest of the first line
(.+\n)* // subsequent consecutive lines
\n* // blanks
)+
)
/gm, function(){...});
*/
text = text.replace(/((^[ \t]*>[ \t]?.+\n(.+\n)*\n*)+)/gm,
function(wholeMatch,m1) {
var bq = m1;
// attacklab: hack around Konqueror 3.5.4 bug:
// "----------bug".replace(/^-/g,"") == "bug"
bq = bq.replace(/^[ \t]*>[ \t]?/gm,"~0"); // trim one level of quoting
// attacklab: clean up hack
bq = bq.replace(/~0/g,"");
bq = bq.replace(/^[ \t]+$/gm,""); // trim whitespace-only lines
bq = _RunBlockGamut(bq); // recurse
bq = bq.replace(/(^|\n)/g,"$1 ");
// These leading spaces screw with <pre> content, so we need to fix that:
bq = bq.replace(
/(\s*<pre>[^\r]+?<\/pre>)/gm,
function(wholeMatch,m1) {
var pre = m1;
// attacklab: hack around Konqueror 3.5.4 bug:
pre = pre.replace(/^ /mg,"~0");
pre = pre.replace(/~0/g,"");
return pre;
});
return hashBlock("<blockquote>\n" + bq + "\n</blockquote>");
});
return text;
}
var _FormParagraphs = function(text) {
//
// Params:
// $text - string to process with html <p> tags
//
// Strip leading and trailing lines:
text = text.replace(/^\n+/g,"");
text = text.replace(/\n+$/g,"");
var grafs = text.split(/\n{2,}/g);
var grafsOut = new Array();
//
// Wrap <p> tags.
//
var end = grafs.length;
for (var i=0; i<end; i++) {
var str = grafs[i];
// if this is an HTML marker, copy it
if (str.search(/~K(\d+)K/g) >= 0) {
grafsOut.push(str);
}
else if (str.search(/\S/) >= 0) {
str = _RunSpanGamut(str);
str = str.replace(/^([ \t]*)/g,"<p>");
str += "</p>"
grafsOut.push(str);
}
}
//
// Unhashify HTML blocks
//
end = grafsOut.length;
for (var i=0; i<end; i++) {
// if this is a marker for an html block...
while (grafsOut[i].search(/~K(\d+)K/) >= 0) {
var blockText = g_html_blocks[RegExp.$1];
blockText = blockText.replace(/\$/g,"$$$$"); // Escape any dollar signs
grafsOut[i] = grafsOut[i].replace(/~K\d+K/,blockText);
}
}
return grafsOut.join("\n\n");
}
var _EncodeAmpsAndAngles = function(text) {
// Smart processing for ampersands and angle brackets that need to be encoded.
// Ampersand-encoding based entirely on Nat Irons's Amputator MT plugin:
// http://bumppo.net/projects/amputator/
text = text.replace(/&(?!#?[xX]?(?:[0-9a-fA-F]+|\w+);)/g,"&amp;");
// Encode naked <'s
text = text.replace(/<(?![a-z\/?\$!])/gi,"&lt;");
return text;
}
var _EncodeBackslashEscapes = function(text) {
//
// Parameter: String.
// Returns: The string, with after processing the following backslash
// escape sequences.
//
// attacklab: The polite way to do this is with the new
// escapeCharacters() function:
//
// text = escapeCharacters(text,"\\",true);
// text = escapeCharacters(text,"`*_{}[]()>#+-.!",true);
//
// ...but we're sidestepping its use of the (slow) RegExp constructor
// as an optimization for Firefox. This function gets called a LOT.
text = text.replace(/\\(\\)/g,escapeCharacters_callback);
text = text.replace(/\\([`*_{}\[\]()>#+-.!])/g,escapeCharacters_callback);
return text;
}
var _DoAutoLinks = function(text) {
text = text.replace(/<((https?|ftp|dict):[^'">\s]+)>/gi,"<a href=\"$1\">$1</a>");
// Email addresses: <address@domain.foo>
/*
text = text.replace(/
<
(?:mailto:)?
(
[-.\w]+
\@
[-a-z0-9]+(\.[-a-z0-9]+)*\.[a-z]+
)
>
/gi, _DoAutoLinks_callback());
*/
text = text.replace(/<(?:mailto:)?([-.\w]+\@[-a-z0-9]+(\.[-a-z0-9]+)*\.[a-z]+)>/gi,
function(wholeMatch,m1) {
return _EncodeEmailAddress( _UnescapeSpecialChars(m1) );
}
);
return text;
}
var _EncodeEmailAddress = function(addr) {
//
// Input: an email address, e.g. "foo@example.com"
//
// Output: the email address as a mailto link, with each character
// of the address encoded as either a decimal or hex entity, in
// the hopes of foiling most address harvesting spam bots. E.g.:
//
// <a href="&#x6D;&#97;&#105;&#108;&#x74;&#111;:&#102;&#111;&#111;&#64;&#101;
// x&#x61;&#109;&#x70;&#108;&#x65;&#x2E;&#99;&#111;&#109;">&#102;&#111;&#111;
// &#64;&#101;x&#x61;&#109;&#x70;&#108;&#x65;&#x2E;&#99;&#111;&#109;</a>
//
// Based on a filter by Matthew Wickline, posted to the BBEdit-Talk
// mailing list: <http://tinyurl.com/yu7ue>
//
// attacklab: why can't javascript speak hex?
function char2hex(ch) {
var hexDigits = '0123456789ABCDEF';
var dec = ch.charCodeAt(0);
return(hexDigits.charAt(dec>>4) + hexDigits.charAt(dec&15));
}
var encode = [
function(ch){return "&#"+ch.charCodeAt(0)+";";},
function(ch){return "&#x"+char2hex(ch)+";";},
function(ch){return ch;}
];
addr = "mailto:" + addr;
addr = addr.replace(/./g, function(ch) {
if (ch == "@") {
// this *must* be encoded. I insist.
ch = encode[Math.floor(Math.random()*2)](ch);
} else if (ch !=":") {
// leave ':' alone (to spot mailto: later)
var r = Math.random();
// roughly 10% raw, 45% hex, 45% dec
ch = (
r > .9 ? encode[2](ch) :
r > .45 ? encode[1](ch) :
encode[0](ch)
);
}
return ch;
});
addr = "<a href=\"" + addr + "\">" + addr + "</a>";
addr = addr.replace(/">.+:/g,"\">"); // strip the mailto: from the visible part"
return addr;
}
var _UnescapeSpecialChars = function(text) {
//
// Swap back in all the special characters we've hidden.
//
text = text.replace(/~E(\d+)E/g,
function(wholeMatch,m1) {
var charCodeToReplace = parseInt(m1);
return String.fromCharCode(charCodeToReplace);
}
);
return text;
}
var _Outdent = function(text) {
//
// Remove one level of line-leading tabs or spaces
//
// attacklab: hack around Konqueror 3.5.4 bug:
// "----------bug".replace(/^-/g,"") == "bug"
text = text.replace(/^(\t|[ ]{1,4})/gm,"~0"); // attacklab: g_tab_width
// attacklab: clean up hack
text = text.replace(/~0/g,"")
return text;
}
var _Detab = function(text) {
// attacklab: Detab's completely rewritten for speed.
// In perl we could fix it by anchoring the regexp with \G.
// In javascript we're less fortunate.
// expand first n-1 tabs
text = text.replace(/\t(?=\t)/g," "); // attacklab: g_tab_width
// replace the nth with two sentinels
text = text.replace(/\t/g,"~A~B");
// use the sentinel to anchor our regex so it doesn't explode
text = text.replace(/~B(.+?)~A/g,
function(wholeMatch,m1,m2) {
var leadingText = m1;
var numSpaces = 4 - leadingText.length % 4; // attacklab: g_tab_width
// there *must* be a better way to do this:
for (var i=0; i<numSpaces; i++) leadingText+=" ";
return leadingText;
}
);
// clean up sentinels
text = text.replace(/~A/g," "); // attacklab: g_tab_width
text = text.replace(/~B/g,"");
return text;
}
//
// attacklab: Utility functions
//
var escapeCharacters = function(text, charsToEscape, afterBackslash) {
// First we have to escape the escape characters so that
// we can build a character class out of them
var regexString = "([" + charsToEscape.replace(/([\[\]\\])/g,"\\$1") + "])";
if (afterBackslash) {
regexString = "\\\\" + regexString;
}
var regex = new RegExp(regexString,"g");
text = text.replace(regex,escapeCharacters_callback);
return text;
}
var escapeCharacters_callback = function(wholeMatch,m1) {
var charCodeToEscape = m1.charCodeAt(0);
return "~E"+charCodeToEscape+"E";
}
} // end of Showdown.converter
// ###########################################################################
// Start user script
// ###########################################################################
var converter = new Showdown.converter();
function wireupNewCommentEditors( parent )
{
if (!parent.getElementsByTagName) return;
if ( parent.tagName == 'FORM' )
{
/*HACK: I don't get it! When I click Reply to a comment, a live
preview already exists! Even before our handler for DOMNodeInserted
fires. (I tested this with a timeout.) Plus, the existing live preview
will already contain whatever preview is in the main comment preview
at the time, but it is not linked to the main comment text area, so it
never updates. The crazy thig is, the form for the comment reply
doesn't even exist in the DOM before we click the Reply link. So where
the hell is this preview coming from? I'm just brute-forcing it away
for now. Same issue with the editor*/
removeExistingPreview( parent );
removeExistingEditor( parent );
var preview = addPreviewToParent( parent );
addMarkdownEditorToForm( parent, preview );
}
else
{
var forms = parent.getElementsByTagName('form');
for ( var i=0, form=null; form=forms[i]; i++ ) {
if ( form.getAttribute('id') &&
(
(form.getAttribute('id').match(/^commentreply_./)) //||
// (form.getAttribute('id').match(/^form-./))
)
)
{
var preview = addPreviewToParent(form);
addMarkdownEditorToForm( form, preview );
}
}
}
}
function wireupExistingCommentEditors()
{
var editDivs = document.evaluate(
"//div[@class='usertext-edit']",
document,
null,
XPathResult.UNORDERED_NODE_SNAPSHOT_TYPE,
null
);
//First one is not an edit form.
for ( var i = 0; i < editDivs.snapshotLength; i++)
{
var editDiv = editDivs.snapshotItem(i);
var preview = addPreviewToParent( editDiv );
addMarkdownEditorToForm( editDiv, preview );
refreshPreview( preview, preview.textArea )
}
}
function addPreviewToParent( parent )
{
var set=document.createElement('fieldset');
set.setAttribute('class', 'liveComment');
var legend=document.createElement('legend');
legend.textContent='Live Preview';
var preview=document.createElement('div');
preview.setAttribute('class', 'md');
set.appendChild(legend);
set.appendChild(preview);
// modification: hide this thing until someone types...
preview.parentNode.style.display = 'none';
parent.appendChild(set);
var textAreas = parent.getElementsByTagName('textarea');
if ( textAreas[0] )
{
var targetTextArea = textAreas[0];
var timer=null;
targetTextArea.addEventListener(
'keyup',
function()
{
if (timer) clearTimeout(timer);
timer = setTimeout(
function()
{
refreshPreview( preview, targetTextArea );
},
100
);
},
false
);
preview.textArea = targetTextArea;
addPreviewClearOnCommentSubmit( parent, preview );
}
return preview;
}
//Find any fieldsets with a class of liveComment as children of this element and remove them.
function removeExistingPreview( parent )
{
var fieldSets = parent.getElementsByTagName('fieldset');
for (var i = 0, fieldSet = null; fieldSet = fieldSets[i]; i++)
{
if ( fieldSet.getAttribute( 'class' ) == 'liveComment' )
{
fieldSet.parentNode.removeChild( fieldSet );
break;
}
}
}
function removeExistingEditor( parent )
{
var divs = parent.getElementsByTagName('div');
for (var i = 0, div = null; div = divs[i]; i++)
{
if ( div.getAttribute( 'class' ) == 'markdownEditor' )
{
div.parentNode.removeChild( div );
break;
}
}
}
function addPreviewClearOnCommentSubmit( parent, preview )
{
var buttons = parent.getElementsByTagName('button');
for (var i = 0, button = null; button = buttons[i]; i++)
{
if ( button.getAttribute('class') == "save" )
{
button.addEventListener(
'click',
function()
{
preview.innerHTML='';
},
false
);
}
}
}
function refreshPreview( preview, targetTextArea )
{
// modification: hide this thing if it's empty...
if (targetTextArea.value == '') {
preview.parentNode.style.display = 'none';
} else {
preview.parentNode.style.display = 'block';
}
preview.innerHTML = converter.makeHtml( targetTextArea.value );
}
function addMarkdownEditorToForm( parent, preview )
{
var textAreas = parent.getElementsByTagName('textarea');
if ( !textAreas[0] ) return;
var targetTextArea = textAreas[0];
var controlBox = document.createElement( 'div' );
controlBox.setAttribute('class', 'markdownEditor');
parent.insertBefore( controlBox, parent.firstChild );
var bold = new EditControl(
'<b>Bold</b>',
function()
{
tagSelection( targetTextArea, '**', '**' );
refreshPreview( preview, targetTextArea );
}
);
var italics = new EditControl(
'<i>Italic</i>',
function()
{
tagSelection( targetTextArea, '*', '*' );
refreshPreview( preview, targetTextArea );
}
);
var link = new EditControl(
'Link',
function()
{
linkSelection( targetTextArea );
refreshPreview( preview, targetTextArea );
}
);
var quote = new EditControl(
'|Quote',
function()
{
prefixSelectionLines( targetTextArea, '>' );
refreshPreview( preview, targetTextArea );
}
);
var code = new EditControl(
'<span style="font-family: Courier New;">Code</span>',
function()
{
prefixSelectionLines( targetTextArea, ' ' );
refreshPreview( preview, targetTextArea );
}
);
var bullets = new EditControl(
'&bull;Bullets',
function()
{
prefixSelectionLines( targetTextArea, '* ' );
refreshPreview( preview, targetTextArea );
}
);
var numbers = new EditControl(
'1.Numbers',
function()
{
prefixSelectionLines( targetTextArea, '1. ' );
refreshPreview( preview, targetTextArea );
}
);
var disapproval = new EditControl(
'&#3232;\_&#3232;',
function() {
prefixSelectionLines( targetTextArea, '&#3232;_&#3232;' );
refreshPreview( preview, targetTextArea );
}
);
var promoteRES = new EditControl(
'[Promote]',
function() {
prefixSelectionLines( targetTextArea, '[Reddit Enhancement Suite](http://reddit.honestbleeps.com?referral='+RESUtils.loggedInUser()+')' );
refreshPreview( preview, targetTextArea );
}
);
controlBox.appendChild( bold.create() );
controlBox.appendChild( italics.create() );
controlBox.appendChild( link.create() );
controlBox.appendChild( quote.create() );
controlBox.appendChild( code.create() );
controlBox.appendChild( bullets.create() );
controlBox.appendChild( numbers.create() );
controlBox.appendChild( disapproval.create() );
controlBox.appendChild( promoteRES.create() );
}
function EditControl( label, editFunction )
{
this.create = function()
{
var link = document.createElement('a');
link.innerHTML = label;
link.href = 'javascript:;';
link.setAttribute('style','Margin-Right: 15px; text-decoration: none;');
link.execute = editFunction;
addEvent( link, 'click', 'execute' );
return link;
}
}
function tagSelection( targetTextArea, tagOpen, tagClose, textEscapeFunction )
{
//record scroll top to restore it later.
var scrollTop = targetTextArea.scrollTop;
//We will restore the selection later, so record the current selection.
var selectionStart = targetTextArea.selectionStart;
var selectionEnd = targetTextArea.selectionEnd;
var selectedText = targetTextArea.value.substring( selectionStart, selectionEnd );
//Markdown doesn't like it when you tag a word like **this **. The space messes it up. So we'll account for that because Firefox selects the word, and the followign space when you double click a word.
var potentialTrailingSpace = '';
if( selectedText[ selectedText.length - 1 ] == ' ' )
{
potentialTrailingSpace = ' ';
selectedText = selectedText.substring( 0, selectedText.length - 1 );
}
if ( textEscapeFunction )
{
selectedText = textEscapeFunction( selectedText );
}
targetTextArea.value =
targetTextArea.value.substring( 0, selectionStart ) + //text leading up to the selection start
tagOpen +
selectedText +
tagClose +
potentialTrailingSpace +
targetTextArea.value.substring( selectionEnd ); //text after the selection end
targetTextArea.selectionStart = selectionStart + tagOpen.length;
targetTextArea.selectionEnd = selectionEnd + tagOpen.length;
targetTextArea.scrollTop = scrollTop;
}
function linkSelection( targetTextArea )
{
var url = prompt( "Enter the URL:", "" );
if ( url != null )
{
tagSelection(
targetTextArea,
'[',
'](' + url.replace( /\(/, '\\(' ).replace( /\)/, '\\)' ) + ')', //escape parens in url
function( text )
{
return text.replace( /\[/, '\\[' ).replace( /\]/, '\\]' ).replace( /\(/, '\\(' ).replace( /\)/, '\\)' ); //escape brackets and parens in text
}
);
}
}
function prefixSelectionLines( targetTextArea, prefix )
{
var scrollTop = targetTextArea.scrollTop;
var selectionStart = targetTextArea.selectionStart;
var selectionEnd = targetTextArea.selectionEnd;
var selectedText = targetTextArea.value.substring( selectionStart, selectionEnd );
var lines = selectedText.split( '\n' );
var newValue = '';
for( var i = 0; i < lines.length; i++ )
{
newValue += prefix + lines[i] + '\n';
}
targetTextArea.value =
targetTextArea.value.substring( 0, selectionStart ) + //text leading up to the selection start
newValue +
targetTextArea.value.substring( selectionEnd ); //text after the selection end
targetTextArea.scrollTop = scrollTop;
}
//Delegated event wire-up utitlity. Using this allows you to use the "this" keyword in a delegated function.
function addEvent( target, eventName, handlerName )
{
target.addEventListener(eventName, function(e){target[handlerName](e);}, false);
}
// Make the stuff we add look pretty!
RESUtils.addCSS('fieldset.liveComment, fieldset.liveComment legend { border: 1px solid black; border-radius: 1em; -moz-border-radius: 1em; -webkit-border-radius: 1em; }'+
'fieldset.liveComment { padding: 1ex; margin: 1em 0; }'+
'fieldset.liveComment legend { padding: 0 1ex; background-color: #E9E9E9; }');
// Bootstrap with the top-level comment always in the page, and editors for your existing comments.
wireupExistingCommentEditors();
// Watch for any future 'reply' forms.
document.body.addEventListener(
'DOMNodeInserted',
function( event )
{
wireupNewCommentEditors( event.target );
},
false
);
}
}
};
modules['usernameHider'] = {
moduleID: 'usernameHider',
moduleName: 'Username Hider',
options: {
displayText: {
type: 'text',
value: '~anonymous~',
description: 'What to replace your username with, default is ~anonymous~'
}
},
description: 'This module hides your real username when you\'re logged in to reddit.',
isEnabled: function() {
return RESConsole.getModulePrefs(this.moduleID);
},
include: Array(
/http:\/\/www.reddit.com\/[-\w\.\/]*/i,
/http:\/\/reddit.com\/[-\w\.\/]*/i
),
isMatchURL: function() {
return RESUtils.isMatchURL(this.moduleID);
},
go: function() {
if ((this.isEnabled()) && (this.isMatchURL())) {
// get this module's options...
RESUtils.getOptions(this.moduleID);
var userNameEle = document.querySelector('#header-bottom-right > span > a');
var thisUserName = userNameEle.text;
userNameEle.textContent = this.options.displayText.value;
var authors = document.querySelectorAll('.author');
for (var i=0, len=authors.length; i<len;i++) {
if (authors[i].textContent == thisUserName) {
authors[i].textContent = this.options.displayText.value;
}
}
}
}
};
modules['showImages'] = {
moduleID: 'showImages',
moduleName: 'Inline Image Viewer',
options: {
maxWidth: {
type: 'text',
value: '640',
description: 'Max width of image displayed onscreen'
},
maxHeight: {
type: 'text',
value: '480',
description: 'Max height of image displayed onscreen'
},
openInNewWindow: {
type: 'boolean',
value: true,
description: 'Open images in a new tab/window when clicked?'
},
hideNSFW: {
type: 'boolean',
value: false,
description: 'If checked, do not show images with NSFW in the title.'
},
useSmallImages: {
type: 'boolean',
value: false,
description: 'If checked, use smaller images from imgur to save bandwidth.'
}
},
description: 'Opens images inline in your browser with the click of a button. Also has configuration options, check it out!',
isEnabled: function() {
return RESConsole.getModulePrefs(this.moduleID);
},
include: Array(
/http:\/\/www.reddit.com\/[-\w\.\_\?=]*/i
),
exclude: Array(
/http:\/\/www.reddit.com\/ads\/[-\w\.\_\?=]*/i,
/http:\/\/www.reddit.com\/[-\w\.\/]*\/submit\/?/i
),
isMatchURL: function() {
return RESUtils.isMatchURL(this.moduleID);
},
go: function() {
if ((this.isEnabled()) && (this.isMatchURL())) {
// get this module's options...
RESUtils.getOptions(this.moduleID);
// Show Images - source originally from Richard Lainchbury - https://userscripts-mirror.org/scripts/show/67729
// Source modified to work as a module in RES, and improved slightly..
RESUtils.addCSS(".expando-button.image { float: left; width: 23px; height: 23px; max-width: 23px; max-height: 23px; display: inline-block; background-image: url('http://thumbs.reddit.com/t5_2s10b_0.png'); margin-right: 6px; cursor: pointer; padding: 0px; }");
RESUtils.addCSS(".expando-button.image.commentImg { float: none; margin-left: 4px; } ");
RESUtils.addCSS(".expando-button.image.collapsed { background-position: 0px 0px; } ");
RESUtils.addCSS(".expando-button.image.collapsed:hover { background-position: 0px -24px; } ");
RESUtils.addCSS(".expando-button.image.expanded { background-position: 0px -48px; } ");
RESUtils.addCSS(".expando-button.image.expanded:hover { background-position: 0px -72px; } ");
//GM_addStyle(css);
this.imageList = Array();
this.imagesRevealed = Array();
//this.flickrImages = Array();
document.body.addEventListener('DOMNodeInserted', function(event) {
if ((event.target.tagName == 'DIV') && (event.target.getAttribute('id') == 'siteTable')) {
modules['showImages'].findAllImages(event.target);
if (modules['showImages'].imagesVisible) {
this.waitForScan = setInterval(function() {
if (!(modules['showImages'].scanningForImages)) {
modules['showImages'].showImages(modules['showImages'].gw, true);
clearInterval(modules['showImages'].waitForScan);
}
}, 100);
}
}
}, true);
// this.imguReddit();
this.createImageButtons();
this.findAllImages();
}
},
createImageButtons: function() {
var mainmenuUL = document.body.querySelector('#header-bottom-left ul.tabmenu');
if (mainmenuUL) {
var li = document.createElement('li');
var a = document.createElement('a');
var text = document.createTextNode('scanning for images...');
this.scanningForImages = true;
a.setAttribute('href','javascript:void(0);');
a.setAttribute('id','viewImagesButton');
a.addEventListener('click', function(e) {
e.preventDefault();
if (!(modules['showImages'].scanningForImages)) {
modules['showImages'].showImages();
}
}, true);
a.appendChild(text);
li.appendChild(a);
mainmenuUL.appendChild(li);
this.viewButton = a;
this.gw = '';
var commentsre = new RegExp(/comments\/[-\w\.\/]/i);
if (!(commentsre.test(location.href)) && (window.location.href.indexOf('gonewild')>=0)){
var li = document.createElement('li');
var a = document.createElement('a');
var text = document.createTextNode('[m]');
a.setAttribute('href','javascript:void(0);');
a.addEventListener('click', function(e) {
e.preventDefault();
modules['showImages'].gw = 'm';
modules['showImages'].showImages('m');
}, true);
a.appendChild(text);
li.appendChild(a);
mainmenuUL.appendChild(li);
var li = document.createElement('li');
var a = document.createElement('a');
var text = document.createTextNode('[f]');
a.setAttribute('href','javascript:void(0);');
a.addEventListener('click', function(e) {
e.preventDefault();
modules['showImages'].gw = 'f';
modules['showImages'].showImages('f');
}, true);
a.appendChild(text);
li.appendChild(a);
mainmenuUL.appendChild(li);
}
}
},
updateImageButtons: function(imgCount) {
if ((typeof(this.viewButton) != 'undefined')) {
if (this.imagesVisible) {
this.viewButton.innerHTML = 'hide images ('+imgCount+')';
} else {
this.viewButton.innerHTML = 'view images ('+imgCount+')';
}
}
},
findImages: function(gonewild, showmore) {
switch (gonewild) {
case 'f':
re = new RegExp(/[\[\{\<\(](f|fem|female)[\]\}\>\)]/i);
break;
case 'm':
re = new RegExp(/[\[\{\<\(](m|man|male)[\]\}\>\)]/i);
break;
}
if (this.options.hideNSFW.value) {
re = new RegExp(/nsfw/i);
}
for(var i=0, len=this.imageList.length;i<len;i++) {
var href= this.imageList[i].getAttribute("href");
var title_text=this.imageList[i].text;
(gonewild) ? titleMatch = re.test(title_text) : titleMatch = false;
var NSFW = false;
if (this.options.hideNSFW.value) {
NSFW = re.test(title_text);
}
if (href && (gonewild == '' || titleMatch) && (!NSFW) && (href.indexOf('imgur.')>=0 || href.indexOf('.jpeg')>=0 || href.indexOf('.jpg')>=0 || href.indexOf('.gif')>=0)) {
if (hasClass(this.imageList[i].parentNode,'title')) {
var targetImage = this.imageList[i].parentNode.nextSibling
} else {
var targetImage = this.imageList[i].nextSibling
}
this.revealImage(targetImage, showmore);
}
}
},
imgurType: function(url) {
// Detect the type of imgur link
// Direct image link? http://i.imgur.com/0ZxQF.jpg
// imgur "page" link? http://imgur.com/0ZxQF
// imgur "gallery"? ??????????
var urlPieces = url.split('?');
var cleanURL = urlPieces[0];
var directImg = /i.imgur.com\/[\w]+\.[\w]+/gi;
var imgPage = /imgur.com\.[\w+]$/gi;
},
findAllImages: function(ele) {
this.scanningForImages = true;
if (ele == null) {
ele = document.body;
}
// get elements common across all pages first...
// if we're on a comments page, get those elements too...
var commentsre = new RegExp(/comments\/[-\w\.\/]/i);
var userre = new RegExp(/user\/[-\w\.\/]/i);
if ((commentsre.test(location.href)) || (userre.test(location.href))) {
this.allElements = ele.querySelectorAll('#siteTable A.title, .usertext-body > div.md a');
} else {
this.allElements = ele.querySelectorAll('#siteTable A.title');
}
// make an array to store any links we've made calls to for the imgur API so we don't do any multiple hits to it.
this.imgurCalls = new Array();
// this.allElements contains all link elements on the page - now let's filter it for images...
this.hashRe = /^http:\/\/[i.]*imgur.com\/([\w]+)(\..+)?$/i;
var groups = Array();
this.allElementsCount=this.allElements.length;
this.allElementsi = 0;
(function(){
// scan 15 links at a time...
var chunkLength = Math.min((modules['showImages'].allElementsCount - modules['showImages'].allElementsi), 10);
for (var i=0;i<chunkLength;i++) {
modules['showImages'].checkElementForImage(modules['showImages'].allElementsi);
modules['showImages'].allElementsi++;
}
if (modules['showImages'].allElementsi < modules['showImages'].allElementsCount) {
setTimeout(arguments.callee, 300);
} else {
modules['showImages'].scanningForImages = false;
modules['showImages'].updateImageButtons(modules['showImages'].imageList.length);
}
})();
},
checkElementForImage: function(index) {
ele = this.allElements[index];
var href = ele.getAttribute('href');
if ((!(hasClass(ele,'imgScanned'))) && (typeof(this.imagesRevealed[href]) == 'undefined') && (href != null)) {
addClass(ele,'imgScanned');
this.imagesRevealed[href] = true;
isImgur = (href.indexOf('imgur.com')>=0);
//isFlickr = (href.indexOf('flickr.com')>=0);
if (!(ele.getAttribute('scanned') == 'true') && (isImgur || href.indexOf('.jpeg')>=0 || href.indexOf('.jpg')>=0 || href.indexOf('.gif')>=0)) {
if (isImgur) {
// if it's not a full (direct) imgur link, get the relevant data and append it... otherwise, go now!
// first, kill any URL parameters that screw with the parser, like ?full.
var splithref = href.split('?');
href = splithref[0];
if (this.options.useSmallImages.value) {
splithref = href.split('.');
if ((splithref[splithref.length-1] == 'jpg') || (splithref[splithref.length-1] == 'jpeg') || (splithref[splithref.length-1] == 'png') || (splithref[splithref.length-1] == 'gif')) {
splithref[splithref.length-2] += 'h';
href = splithref.join('.');
}
}
ele.setAttribute('href',href);
// now, strip the hash off of it so we can make an API call if need be
groups = this.hashRe.exec(href);
// opera doesn't support cross domain XHR, so this part won't work in opera. boo.
// if we got a match, but we don't have a file extension, hit the imgur API for that info
if ((groups && !groups[2]) && (typeof(opera) == 'undefined')) {
var apiURL = 'http://api.imgur.com/2/image/'+groups[1]+'.xml';
// avoid making duplicate calls from the same page... want to minimize hits to imgur API
if (typeof(this.imgurCalls[apiURL]) == 'undefined') {
// store the object we want to modify when the xml request is finished...
this.imgurCalls[apiURL] = ele;
GM_xmlhttpRequest({
method: 'GET',
url: apiURL,
onload: function(response) {
var parser = new DOMParser();
var xml = parser.parseFromString(response.responseText, "application/xml");
if (xml.getElementsByTagName('original')[0]) {
modules['showImages'].imgurCalls[apiURL].setAttribute('href',xml.getElementsByTagName('original')[0].textContent);
}
}
});
}
}
if (groups) this.createImageExpando(ele, true);
} else {
this.createImageExpando(ele);
}
}
} else if (!(hasClass(ele,'imgFound'))) {
if (!(ele.getAttribute('scanned') == 'true') && (href.indexOf('imgur.')>=0 || href.indexOf('.jpeg')>=0 || href.indexOf('.jpg')>=0 || href.indexOf('.gif')>=0)) {
var textFrag = document.createTextNode(' [duplicate image ignored.]');
insertAfter(ele, textFrag);
}
}
},
createImageExpando: function(obj, asynch) {
addClass(obj,'imgFound');
obj.setAttribute('scanned','true');
this.imageList.push(obj);
var thisExpandLink = document.createElement('a');
thisExpandLink.setAttribute('class','toggleImage expando-button image');
// thisExpandLink.innerHTML = '[show img]';
thisExpandLink.innerHTML = '&nbsp;';
removeClass(thisExpandLink, 'expanded');
addClass(thisExpandLink, 'collapsed');
thisExpandLink.addEventListener('click', function(e) {
e.preventDefault();
modules['showImages'].revealImage(e.target);
}, true);
if (hasClass(obj.parentNode,'title')) {
var nodeToInsertAfter = obj.parentNode;
addClass(thisExpandLink, 'linkImg');
} else {
var nodeToInsertAfter = obj;
addClass(thisExpandLink, 'commentImg');
}
insertAfter(nodeToInsertAfter, thisExpandLink);
},
revealImage: function(showLink, showhide) {
// showhide = false means hide, true means show!
if (hasClass(showLink, 'commentImg')) {
var thisImageLink = showLink.previousSibling;
} else {
var thisImageLink = showLink.previousSibling.firstChild;
}
var imageCheck = showLink.nextSibling;
// Check if the next sibling is an image. If so, we've already revealed that image.
if ((typeof(imageCheck) != 'undefined') && (imageCheck != null) && (typeof(imageCheck.tagName) != 'undefined') && (hasClass(imageCheck, 'madeVisible'))) {
if ((showhide != true) && (imageCheck.style.display != 'none')) {
imageCheck.style.display = 'none';
// showLink.innerHTML = '[show img]';
removeClass(showLink, 'expanded');
addClass(showLink, 'collapsed');
} else {
imageCheck.style.display = 'block';
// showLink.innerHTML = '[hide img]';
removeClass(showLink, 'collapsed');
addClass(showLink, 'expanded');
}
} else {
// we haven't revealed this image before. Load it in.
var href = thisImageLink.getAttribute('href');
var ext = (href.indexOf('imgur.')>=0 && href.indexOf('.jpg')<0 && href.indexOf('.png')<0 && href.indexOf('.gif')<0) ? '.jpg' :'';
var img = document.createElement('a');
if (this.options.openInNewWindow.value) {
img.setAttribute('target','_blank');
}
img.setAttribute('href',href);
img.setAttribute('class','madeVisible');
img.setAttribute('style','display: block;max-width:'+this.options.maxWidth.value+'px;max-height:'+this.options.maxHeight.value+'px; clear: both;');
/* Chrome doesn't seem happy with cross domain XHR to flickr... tabling this idea for now.
if (href.indexOf('flickr.com') >= 0) {
this.flickrImages[href] = img;
GM_xmlhttpRequest({
method: "GET",
url: href,
onload: function(response) {
console.log('response: ');
console.log(response);
var tempDiv = document.createElement(tempDiv);
tempDiv.innerHTML = thisHTML.replace(/<script(.|\s)*?\/script>/g, '');
// check for new mail
var flickrImg = tempDiv.querySelector('#photo > .photo-div > IMG');
modules['showImages'].flickrImages[this.url].innerHTML = flickrImg.getAttribute('src');
}
});
// img.innerHTML = flickrImg;
} else {
img.innerHTML = '<img style="display:block;max-width:'+this.options.maxWidth.value+'px;max-height:'+this.options.maxHeight.value+'px;" src="' + href + ext + '" />';
}
*/
img.innerHTML = '<img style="display:block;max-width:'+this.options.maxWidth.value+'px;max-height:'+this.options.maxHeight.value+'px;" src="' + href + ext + '" />';
insertAfter(showLink, img);
// showLink.innerHTML = '[hide img]';
removeClass(showLink, 'collapsed');
addClass(showLink, 'expanded');
}
},
showImages: function(gonewild, showmore) {
if ((this.imagesVisible) && (!(showmore))) {
this.imagesVisible = false;
// images are already visible. Hide them.
var imageList = document.body.querySelectorAll('.madeVisible');
for (var i=0, len=this.imageList.length;i<len;i++) {
this.revealImage(imageList[i].previousSibling, false);
}
this.viewButton.innerHTML = 'view images ('+this.imageList.length+')';
return false;
} else {
this.imagesVisible = true;
this.viewButton.innerHTML = 'hide images ('+this.imageList.length+')';
var gw = gonewild || '';
this.findImages(gw, showmore);
}
}
}; // note: you NEED this semicolon at the end!
modules['showKarma'] = {
moduleID: 'showKarma',
moduleName: 'Show Comment Karma',
options: {
separator: {
type: 'text',
value: '\u00b7',
description: 'Separator character between post/comment karma'
}
},
description: 'Shows your comment karma next to your link karma.',
isEnabled: function() {
return RESConsole.getModulePrefs(this.moduleID);
},
include: Array(
/http:\/\/www.reddit.com\/.*/i
),
isMatchURL: function() {
return RESUtils.isMatchURL(this.moduleID);
},
go: function() {
if ((this.isEnabled()) && (this.isMatchURL())) {
// get this module's options...
RESUtils.getOptions(this.moduleID);
// do stuff now!
// this is where your code goes...
var thisUserName = RESUtils.userName;
GM_xmlhttpRequest({
method: "GET",
url: "http://www.reddit.com/user/" + thisUserName + "/about.json",
onload: function(response) {
var thisResponse = JSON.parse(response.responseText);
var karmaDiv = document.querySelector("#header-bottom-right > .user b");
if ((typeof(karmaDiv) != 'undefined') && (karmaDiv != null)) {
karmaDiv.innerHTML += " " + modules['showKarma'].options.separator.value + " " + thisResponse.data.comment_karma;
}
}
});
}
}
};
modules['hideChildComments'] = {
moduleID: 'hideChildComments',
moduleName: 'Hide All Child Comments',
options: {
// any configurable options you have go here...
// options must have a type and a value..
// valid types are: text, boolean (if boolean, value must be true or false)
// for example:
automatic: {
type: 'boolean',
value: false,
description: 'Automatically hide all but parent comments, or provide a link to hide them all?'
}
},
description: 'Allows you to hide all comments except for replies to the OP for easier reading.',
isEnabled: function() {
return RESConsole.getModulePrefs(this.moduleID);
},
include: Array(
/http:\/\/www.reddit.com\/[-\w\.\/]+\/comments\/[-\w\.]+/i
),
isMatchURL: function() {
return RESUtils.isMatchURL(this.moduleID);
},
go: function() {
if ((this.isEnabled()) && (this.isMatchURL())) {
// get this module's options...
RESUtils.getOptions(this.moduleID);
// do stuff now!
// this is where your code goes...
var toggleButton = document.createElement('li');
this.toggleAllLink = document.createElement('a');
this.toggleAllLink.innerHTML = 'hide all child comments';
this.toggleAllLink.setAttribute('action','hide');
this.toggleAllLink.setAttribute('href','javascript:void(0);');
this.toggleAllLink.setAttribute('title','Show only replies to original poster.');
this.toggleAllLink.addEventListener('click', function() {
modules['hideChildComments'].toggleComments(this.getAttribute('action'));
if (this.getAttribute('action') == 'hide') {
this.setAttribute('action','show');
this.setAttribute('title','Show all comments.');
this.innerHTML = 'show all child comments';
} else {
this.setAttribute('action','hide');
this.setAttribute('title','Show only replies to original poster.');
this.innerHTML = 'hide all child comments';
}
}, true);
toggleButton.appendChild(this.toggleAllLink);
var commentMenu = document.querySelector('ul.buttons');
if (commentMenu) {
commentMenu.appendChild(toggleButton);
var rootComments = document.querySelectorAll('div.commentarea > div.sitetable > div.thing > div.child > div.listing');
for (var i=0, len=rootComments.length; i<len; i++) {
var toggleButton = document.createElement('li');
var toggleLink = document.createElement('a');
toggleLink.innerHTML = 'hide child comments';
toggleLink.setAttribute('action','hide');
toggleLink.setAttribute('href','javascript:void(0);');
toggleLink.setAttribute('class','toggleChildren');
toggleLink.setAttribute('title','Hide child comments.');
toggleLink.addEventListener('click', function(e) {
modules['hideChildComments'].toggleComments(this.getAttribute('action'), this);
if (this.getAttribute('action') == 'hide') {
this.setAttribute('action','show');
this.setAttribute('title','show child comments.');
this.innerHTML = 'show child comments';
} else {
this.setAttribute('action','hide');
this.setAttribute('title','hide child comments.');
this.innerHTML = 'hide child comments';
}
}, true);
toggleButton.appendChild(toggleLink);
var sib = rootComments[i].parentNode.previousSibling;
if (typeof(sib) != 'undefined') {
var sibMenu = sib.querySelector('ul.buttons');
sibMenu.appendChild(toggleButton);
}
}
if (this.options.automatic.value) {
RESUtils.click(this.toggleAllLink);
}
}
}
},
toggleComments: function(action, obj) {
if (obj) {
var thisChildren = obj.parentNode.parentNode.parentNode.parentNode.nextSibling.firstChild;
(action == 'hide') ? thisChildren.style.display = 'none' : thisChildren.style.display = 'block';
} else {
// toggle all comments
var commentContainers = document.querySelectorAll('div.commentarea > div.sitetable > div.thing');
for (var i=0, len=commentContainers.length; i<len; i++) {
var thisChildren = commentContainers[i].querySelector('div.child > div.sitetable');
var thisToggleLink = commentContainers[i].querySelector('a.toggleChildren');
if (thisToggleLink != null) {
if (action == 'hide') {
if (thisChildren != null) {
thisChildren.style.display = 'none'
}
thisToggleLink.innerHTML = 'show child comments';
thisToggleLink.setAttribute('title','show child comments');
thisToggleLink.setAttribute('action','show');
} else {
if (thisChildren != null) {
thisChildren.style.display = 'block';
}
thisToggleLink.innerHTML = 'hide child comments';
thisToggleLink.setAttribute('title','hide child comments');
thisToggleLink.setAttribute('action','hide');
}
}
}
}
}
};
modules['showParent'] = {
moduleID: 'showParent',
moduleName: 'Show Parent on Hover',
options: {
},
description: 'Shows parent comment when hovering over the "parent" link of a comment.',
isEnabled: function() {
return RESConsole.getModulePrefs(this.moduleID);
},
include: Array(
/http:\/\/www.reddit.com\/[-\w\.\/]+\/comments\/[-\w\.]+/i
),
isMatchURL: function() {
return RESUtils.isMatchURL(this.moduleID);
},
go: function() {
if ((this.isEnabled()) && (this.isMatchURL())) {
// get this module's options...
RESUtils.getOptions(this.moduleID);
// do stuff now!
// this is where your code goes...
// code included from https://userscripts-mirror.org/scripts/show/34362
// author: lazyttrick - https://userscripts-mirror.org/users/20871
function show(evt)
{
var id = evt.target.getAttribute('href').replace(/\#/,"");
var top = parseInt(evt.pageY,10)+10,
left = parseInt(evt.pageX,10)+10;
try{
var div = createElement('div', {id:"parentComment"+id, style:"padding:5px; border: 1px solid #666666; overflow:auto; background:#F8F8F8; position:absolute; top:"+top+"px; left:"+left+"px;"}, null );
var query = "//div[contains(@class, 'id-t1_"+id+"')]/div[starts-with(@class,'entry')]";
div.innerHTML = xp(query)[0].innerHTML.replace(/\<ul\s+class[\s\S]+\<\/ul\>/,"").replace(/\<a[^\>]+>\[-\]\<\/a\>/,'');
getTag('body')[0].appendChild(div);
}catch(e){
}
}
function hide(evt)
{
var id = evt.target.getAttribute('href').replace(/\#/,"");
try{
var div = getId("parentComment"+id);
div.parentNode.removeChild(div);
}catch(e){
}
}
function xp(p, context) {
if (!context)
context = document;
var i, arr = [], xpr = document.evaluate(p, context, null, XPathResult.UNORDERED_NODE_SNAPSHOT_TYPE, null);
for (i = 0; item = xpr.snapshotItem(i); i++)
arr.push(item);
return arr;
}
function createElement(type, attrArray, evtListener, html)
{
var node = document.createElement(type);
for (var attr in attrArray) if (attrArray.hasOwnProperty(attr)){
node.setAttribute(attr, attrArray[attr]);
}
if(evtListener){
var a = evtListener.split(' ');
node.addEventListener(a[0], eval(a[1]), eval(a[2]));
}
if(html)
node.innerHTML = html;
return node;
}
function getId(id, parent){
if(!parent)
return document.getElementById(id);
return parent.getElementById(id);
}
function getTag(name, parent){
if(!parent)
return document.getElementsByTagName(name);
return parent.getElementsByTagName(name);
}
//TODO: print debugObj with added variables
function debug(str)
{
var d = document.getElementById('debuggg');
if(!d) {
d = document.createElement('textarea');
d.setAttribute('id','debuggg');
d.setAttribute('style',"height:200px; width:50%; position:fixed; bottom:0px; right:0px;");
document.body.appendChild(d);
}
d.innerHTML += '\n' + str;
d.scrollTop = d.scrollHeight;
}
/*
xp('//div[starts-with(@class,"entry")]//ul[@class="flat-list buttons"]/li[2]/a').forEach(function(a){
a.addEventListener('mouseover', show, false);
a.addEventListener('mouseout', hide, false);
});
*/
this.parentLinks = document.querySelectorAll('.child ul.flat-list > li:nth-child(2) > a');
this.parentLinksCount = this.parentLinks.length;
this.parentLinksi = 0;
(function(){
// add 15 event listeners at a time...
var chunkLength = Math.min((modules['showParent'].parentLinksCount - modules['showParent'].parentLinksi), 15);
for (var i=0;i<chunkLength;i++) {
modules['showParent'].parentLinks[modules['showParent'].parentLinksi].addEventListener('mouseover', show, false);
modules['showParent'].parentLinks[modules['showParent'].parentLinksi].addEventListener('mouseout', hide, false);
modules['showParent'].parentLinksi++;
}
if (modules['showParent'].parentLinksi < modules['showParent'].parentLinksCount) {
setTimeout(arguments.callee, 1000);
}
})();
}
}
};
modules['neverEndingReddit'] = {
moduleID: 'neverEndingReddit',
moduleName: 'Never Ending Reddit',
options: {
// any configurable options you have go here...
// options must have a type and a value..
},
description: 'Inspired by modules like River of Reddit and Auto Pager - gives you a never ending stream of reddit goodness.',
isEnabled: function() {
return RESConsole.getModulePrefs(this.moduleID);
},
include: Array(
/http:\/\/www.reddit.com\/[-\w\.\_\?=]*/i
),
exclude: Array(
/http:\/\/www.reddit.com\/saved\//i
),
isMatchURL: function() {
return RESUtils.isMatchURL(this.moduleID);
},
go: function() {
if ((this.isEnabled()) && (this.isMatchURL())) {
// get this module's options...
RESUtils.getOptions(this.moduleID);
RESUtils.addCSS('#NERModal { display: none; z-index: 999; position: absolute; top: 0px; left: 0px; width: 100%; height: 100%; background-color: #333333; opacity: 0.6; }');
RESUtils.addCSS('#NERContent { display: none; position: absolute; top: 40px; z-index: 1000; width: 720px; background-color: #FFFFFF; color: #000000; padding: 10px; font-size: 12px; }');
//GM_addStyle(css);
// code inspired by River of Reddit, but rewritten from scratch to work across multiple browsers...
// Original River of Reddit author: reddy kapil
// Original link to Chrome extension: https://chrome.google.com/extensions/detail/bjiggjllfebckflfdjbimogjieeghcpp
// set the style for our little loader widget
RESUtils.addCSS('#progressIndicator { width: 95%; height: 30px; font-size: 14px; border: 1px solid #999999; border-radius: 10px 10px 10px 10px; -moz-border-radius: 10px 10px 10px 10px; -webkit-border-radius: 10px 10px 10px 10px; padding: 5px; text-align: center; bgcolor: #f0f3fc; } ');
//GM_addStyle(css);
// store access to the siteTable div since that's where we'll append new data...
this.siteTable = document.body.querySelector('#siteTable');
var stMultiCheck = document.querySelectorAll('#siteTable');
// stupid sponsored links create a second div with ID of sitetable (bad reddit! you should never have 2 IDs with the same name! naughty, naughty reddit!)
if (stMultiCheck.length == 2) {
// console.log('skipped first sitetable, stupid reddit.');
this.siteTable = stMultiCheck[1];
}
// get the first link to the next page of reddit...
var nextPrevLinks = document.body.querySelectorAll('.content .nextprev a');
if (nextPrevLinks.length > 0) {
var nextLink = nextPrevLinks[nextPrevLinks.length-1];
if (nextLink) {
this.nextPageURL = nextLink.getAttribute('href');
var nextXY=RESUtils.getXYpos(nextLink);
this.nextPageScrollY = nextXY.y;
}
this.attachLoaderWidget();
this.attachModalWidget();
// Set the current page to page 1...
this.currPage = 1;
// If there's a page=# value in location.hash, then update the currPage...
var currPageRe = /#page=([0-9]+)/i;
var backButtonPageNumber = currPageRe.exec(location.href);
if (backButtonPageNumber) {
this.currPage = backButtonPageNumber[1];
this.loadNewPage(true);
}
// watch for the user scrolling to the bottom of the page. If they do it, load a new page.
window.addEventListener('scroll', function(e) {
if ((RESUtils.elementInViewport(modules['neverEndingReddit'].progressIndicator)) && (modules['neverEndingReddit'].fromBackButton != true)) {
modules['neverEndingReddit'].loadNewPage();
}
if ((typeof(modules['neverEndingReddit'].navMail) != 'undefined') && (modules['neverEndingReddit'].navMail != null) && (!(RESUtils.elementInViewport(modules['neverEndingReddit'].navMail)))) {
modules['neverEndingReddit'].showNewMail(true);
} else {
modules['neverEndingReddit'].showNewMail(false);
}
}, false);
}
// hide any next/prev page indicators
var nextprev = document.body.querySelectorAll('.content .nextprev');
for (var i=0, len=nextprev.length;i<len;i++) {
nextprev[i].style.display = 'none';
}
// check if the user has new mail...
this.navMail = document.body.querySelector('#mail');
this.NREMail = createElementWithID('a','NREMail');
this.NREMail.style.position = 'fixed';
this.NREMail.style.top = '10px';
this.NREMail.style.right = '10px';
this.NREMail.style.display = 'none';
document.body.appendChild(this.NREMail);
var hasNew = false;
if ((typeof(this.navMail) != 'undefined') && (this.navMail != null)) {
hasNew = hasClass(this.navMail,'havemail');
}
this.setMailIcon();
}
},
setMailIcon: function(newmail) {
if (newmail) {
modules['neverEndingReddit'].hasNewMail = true;
addClass(this.NREMail, 'havemail');
this.NREMail.setAttribute('href','http://www.reddit.com/message/unread/');
this.NREMail.setAttribute('alt','new mail!');
this.NREMail.innerHTML = '<img src="/static/mail.png" alt="messages">';
} else {
modules['neverEndingReddit'].hasNewMail = false;
addClass(this.NREMail, 'nohavemail');
this.NREMail.setAttribute('href','http://www.reddit.com/message/inbox/');
this.NREMail.setAttribute('alt','no new mail');
this.NREMail.innerHTML = '<img src="/static/mailgray.png" alt="messages">';
}
},
attachModalWidget: function() {
this.modalWidget = createElementWithID('div','NERModal');
this.modalWidget.innerHTML = '&nbsp;';
this.modalContent = createElementWithID('div','NERContent');
this.modalContent.innerHTML = 'Never Ending Reddit has detected that you are returning from a page that it loaded. Please give us a moment while we reload that content and return you to where you left off.<br><img src="'+RESConsole.loader+'">';
document.body.appendChild(this.modalWidget);
document.body.appendChild(this.modalContent);
},
attachLoaderWidget: function() {
// add a widget at the bottom that will be used to detect that we've scrolled to the bottom, and will also serve as a "loading" bar...
this.progressIndicator = document.createElement('p');
this.progressIndicator.innerHTML = 'Never Ending Reddit...';
this.progressIndicator.id = 'progressIndicator';
this.progressIndicator.className = 'neverEndingReddit';
insertAfter(this.siteTable, this.progressIndicator);
},
loadNewPage: function(fromBackButton) {
if (fromBackButton) {
window.scrollTo(0,0)
this.fromBackButton = true;
this.nextPageURL = localStorage.getItem('RESmodules.neverEndingReddit.lastPage');
var leftCentered = Math.floor((window.innerWidth - 720) / 2);
this.modalWidget.style.display = 'block';
this.modalContent.style.display = 'block';
this.modalContent.style.left = leftCentered + 'px';
// remove the progress indicator early, as we don't want the user to scroll past it on accident, loading more content.
this.progressIndicator.parentNode.removeChild(modules['neverEndingReddit'].progressIndicator);
} else {
this.fromBackButton = false;
}
if (this.isLoading != true) {
this.progressIndicator.innerHTML = '<img src="'+RESConsole.loader+'"> Loading next page...';
this.isLoading = true;
GM_xmlhttpRequest({
method: "GET",
url: this.nextPageURL,
onload: function(response) {
if (!(modules['neverEndingReddit'].fromBackButton)) {
modules['neverEndingReddit'].currPage++;
localStorage.setItem('RESmodules.neverEndingReddit.lastPage', modules['neverEndingReddit'].nextPageURL);
location.hash = 'page='+modules['neverEndingReddit'].currPage;
}
if ((typeof(modules['neverEndingReddit'].progressIndicator.parentNode) != 'undefined') && (modules['neverEndingReddit'].progressIndicator.parentNode != null)) {
modules['neverEndingReddit'].progressIndicator.parentNode.removeChild(modules['neverEndingReddit'].progressIndicator);
}
// drop the HTML we got back into a div...
var thisHTML = response.responseText;
var tempDiv = document.createElement('div');
// clear out any javascript so we don't render it again...
tempDiv.innerHTML = thisHTML.replace(/<script(.|\s)*?\/script>/g, '');
// check for new mail
var hasNewMail = tempDiv.querySelector('#mail');
if ((typeof('hasNewMail') != 'undefined') && (hasClass(hasNewMail,'havemail'))) {
modules['neverEndingReddit'].setMailIcon(true);
} else {
modules['neverEndingReddit'].setMailIcon(false);
}
// grab the siteTable out of there...
var newHTML = tempDiv.querySelector('#siteTable');
var stMultiCheck = tempDiv.querySelectorAll('#siteTable');
// stupid sponsored links create a second div with ID of sitetable (bad reddit! you should never have 2 IDs with the same name! naughty, naughty reddit!)
if (stMultiCheck.length == 2) {
// console.log('skipped first sitetable, stupid reddit.');
newHTML = stMultiCheck[1];
}
// load up uppers and downers, if enabled...
if (modules['uppersAndDowners'].isEnabled()) {
modules['uppersAndDowners'].applyUppersAndDowners(modules['neverEndingReddit'].nextPageURL);
}
// get the new nextLink value for the next page...
var nextPrevLinks = tempDiv.querySelectorAll('.content .nextprev a');
var nextLink = nextPrevLinks[nextPrevLinks.length-1];
modules['neverEndingReddit'].siteTable.appendChild(newHTML);
modules['neverEndingReddit'].isLoading = false;
if (nextLink) {
modules['neverEndingReddit'].nextPageURL = nextLink.getAttribute('href');
modules['neverEndingReddit'].attachLoaderWidget();
}
if (fromBackButton) {
modules['neverEndingReddit'].modalWidget.style.display = 'none';
modules['neverEndingReddit'].modalContent.style.display = 'none';
// window.scrollTo(0,0)
window.scrollTo(0,modules['neverEndingReddit'].nextPageScrollY);
modules['neverEndingReddit'].fromBackButton = false;
}
}
});
} else {
// console.log('load new page ignored');
}
},
showNewMail: function(show) {
if (show) {
this.NREMail.style.display = 'block';
} else {
this.NREMail.style.display = 'none';
}
}
};
modules['saveComments'] = {
moduleID: 'saveComments',
moduleName: 'Save Comments',
options: {
// any configurable options you have go here...
// options must have a type and a value..
// valid types are: text, boolean (if boolean, value must be true or false)
// for example:
},
description: 'Save Comments allows you to save comments, since reddit doesn\'t!',
isEnabled: function() {
return RESConsole.getModulePrefs(this.moduleID);
},
include: Array(
/http:\/\/www.reddit.com\/[-\w\.\/]*/i,
/https:\/\/www.reddit.com\/[-\w\.\/]*/i
),
exclude: Array(
/http:\/\/www.reddit.com\/user\/[-\w\.\/]*\/?/i,
/http:\/\/www.reddit.com\/[-\w\.\/]*\/submit\/?/i,
/http:\/\/www.reddit.com\/submit\/?/i
),
isMatchURL: function() {
return RESUtils.isMatchURL(this.moduleID);
},
go: function() {
if ((this.isEnabled()) && (this.isMatchURL())) {
// get this module's options...
RESUtils.getOptions(this.moduleID);
// do stuff now!
// this is where your code goes...
var currURL = location.href;
var commentsRegex = new RegExp(/http:\/\/www.reddit.com\/[-\w\.\/]*comments\/[-\w\.\/]*/i);
var savedRegex = new RegExp(/http:\/\/www.reddit.com\/saved\/?/i);
if (commentsRegex.test(currURL)) {
// load already-saved comments into memory...
this.loadSavedComments();
this.addSaveLinks();
} else if (savedRegex.test(currURL)) {
// load already-saved comments into memory...
this.loadSavedComments();
this.addSavedCommentsTab();
this.drawSavedComments();
if (location.hash == '#comments') {
this.showSavedTab('comments');
}
} else {
this.addSavedCommentsTab();
}
}
},
addSaveLinks: function() {
this.allComments = document.querySelectorAll('div.commentarea > div.sitetable > div.thing div.entry div.noncollapsed');
this.allCommentsCount = this.allComments.length;
this.allCommentsi = 0;
(function(){
// add 15 save links at a time...
var chunkLength = Math.min((modules['saveComments'].allCommentsCount - modules['saveComments'].allCommentsi), 15);
for (var i=0;i<chunkLength;i++) {
modules['saveComments'].addSaveLinkToComment(modules['saveComments'].allCommentsi);
modules['saveComments'].allCommentsi++;
}
if (modules['saveComments'].allCommentsi < modules['saveComments'].allCommentsCount) {
setTimeout(arguments.callee, 1000);
}
})();
},
addSaveLinkToComment: function(num) {
var commentObj = this.allComments[num];
var commentsUL = commentObj.querySelector('ul.flat-list');
var permaLink = commentsUL.querySelector('li.first a.bylink');
if (permaLink != null) {
var userLink = commentObj.querySelector('a.author');
if (userLink != null) {
var saveUser = userLink.text;
var saveHREF = permaLink.getAttribute('href')
var saveLink = document.createElement('li');
if (typeof(this.savedCommentsArray[saveHREF]) != 'undefined') {
saveLink.innerHTML = '<a href="/saved#comments">saved</a>';
} else {
saveLink.innerHTML = '<a href="javascript:void(0);" class="saveComments">save</a>';
saveLink.setAttribute('saveLink',saveHREF);
saveLink.setAttribute('saveUser',saveUser);
saveLink.addEventListener('click', function(e) {
e.preventDefault();
modules['saveComments'].saveComment(this, this.getAttribute('saveLink'), this.getAttribute('saveUser'));
}, true);
}
commentsUL.insertBefore(saveLink, commentsUL.lastChild);
}
}
},
loadSavedComments: function() {
// get existing comments first...
var thisComments = localStorage.getItem('RESmodules.saveComments.savedComments');
if (thisComments == null) {
this.storedComments = new Array();
} else {
this.storedComments = JSON.parse(thisComments);
}
this.savedCommentsArray = new Array();
for (var i=0, len=this.storedComments.length;i<len;i++) {
this.savedCommentsArray[this.storedComments[i].href] = 1;
}
},
saveComment: function(obj, href, username, comment) {
// loop through comments and make sure we haven't already saved this one...
if (typeof(this.savedCommentsArray[href]) != 'undefined') {
alert('comment already saved!');
} else {
var comment = obj.parentNode.parentNode.querySelector('div.usertext-body > div.md');
if (comment != null) {
commentHTML = comment.innerHTML;
var savedComment = {
href: href,
username: username,
comment: commentHTML,
timeSaved: Date()
};
this.storedComments.push(savedComment);
obj.innerHTML = '<a href="/saved/#comments">saved</a>';
}
}
localStorage.setItem('RESmodules.saveComments.savedComments', JSON.stringify(this.storedComments));
},
addSavedCommentsTab: function() {
var mainmenuUL = document.body.querySelector('#header-bottom-left ul.tabmenu');
if (mainmenuUL) {
var menuItems = mainmenuUL.querySelectorAll('li');
for (var i=0, len=menuItems.length;i<len;i++) {
var savedLink = menuItems[i].querySelector('a');
if (hasClass(menuItems[i], 'selected')) {
menuItems[i].addEventListener('click', function(e) {
e.preventDefault();
modules['saveComments'].showSavedTab('links');
}, true);
}
if (savedLink.href == 'http://www.reddit.com/saved/') {
this.savedLinksTab = menuItems[i];
savedLink.innerHTML = 'saved links';
}
}
this.savedCommentsTab = document.createElement('li');
this.savedCommentsTab.innerHTML = '<a href="javascript:void(0);">saved comments</a>';
var savedRegex = new RegExp(/http:\/\/www.reddit.com\/saved\/?/i);
if (savedRegex.test(location.href)) {
this.savedCommentsTab.addEventListener('click', function(e) {
e.preventDefault();
modules['saveComments'].showSavedTab('comments');
}, true);
} else {
this.savedCommentsTab.addEventListener('click', function(e) {
e.preventDefault();
location.href = 'http://www.reddit.com/saved/#comments';
}, true);
}
if (this.savedLinksTab != null) {
insertAfter(this.savedLinksTab, this.savedCommentsTab);
}
}
},
showSavedTab: function(tab) {
switch(tab) {
case 'links':
location.hash = 'links';
this.savedLinksContent.style.display = 'block';
this.savedCommentsContent.style.display = 'none';
addClass(this.savedLinksTab, 'selected');
removeClass(this.savedCommentsTab, 'selected');
break;
case 'comments':
location.hash = 'comments';
this.savedLinksContent.style.display = 'none';
this.savedCommentsContent.style.display = 'block';
removeClass(this.savedLinksTab, 'selected');
addClass(this.savedCommentsTab, 'selected');
break;
}
},
drawSavedComments: function() {
RESUtils.addCSS('.savedComment { float: left; padding: 5px; font-size: 12px; margin-bottom: 20px; margin-left: 40px; border: 1px solid #CCCCCC; border-radius: 10px 10px 10px 10px; -moz-border-radius: 10px 10px 10px 10px; -webkit-border-radius: 10px 10px 10px 10px; width: auto; } ');
RESUtils.addCSS('.savedCommentHeader { margin-bottom: 8px; }');
RESUtils.addCSS('.savedCommentBody { margin-bottom: 8px; }');
// css += '.savedCommentFooter { }';
//GM_addStyle(css);
this.savedLinksContent = document.body.querySelector('#siteTable');
this.savedCommentsContent = createElementWithID('ul', 'savedLinksList');
this.savedCommentsContent.style.display = 'none';
for (var i=0, len=this.storedComments.length;i<len;i++) {
var thisComment = document.createElement('li');
addClass(thisComment, 'savedComment');
addClass(thisComment, 'entry');
thisComment.innerHTML = '<div class="savedCommentHeader">Comment by user: ' + this.storedComments[i].username + ' saved on ' + this.storedComments[i].timeSaved + '</div><div class="savedCommentBody">' + this.storedComments[i].comment + '</div>';
thisComment.innerHTML += '<div class="savedCommentFooter"><ul class="flat-list buttons"><li><a class="unsaveComment" href="javascript:void(0);">unsave</a></li><li><a href="' + this.storedComments[i].href + '">view original</a></li></ul></div>';
var unsaveLink = thisComment.querySelector('.unsaveComment');
unsaveLink.setAttribute('unsaveLink', this.storedComments[i].href);
unsaveLink.addEventListener('click', function(e) {
e.preventDefault();
modules['saveComments'].unsaveComment(this.getAttribute('unsaveLink'));
}, true);
this.savedCommentsContent.appendChild(thisComment);
}
if (this.storedComments.length == 0) {
this.savedCommentsContent.innerHTML = '<li>You have not yet saved any comments.</li>';
}
insertAfter(this.savedLinksContent, this.savedCommentsContent);
},
unsaveComment: function(href) {
var newStoredComments = Array();
for (var i=0, len=this.storedComments.length;i<len;i++) {
if (this.storedComments[i].href != href) {
newStoredComments.push(this.storedComments[i]);
} else {
// console.log('found match. deleted comment');
}
}
this.storedComments = newStoredComments;
localStorage.setItem('RESmodules.saveComments.savedComments', JSON.stringify(this.storedComments));
this.savedCommentsContent.parentNode.removeChild(this.savedCommentsContent);
this.drawSavedComments();
this.showSavedTab('comments');
}
};
modules['userHighlight'] = {
moduleID: 'userHighlight',
moduleName: 'User Highlighter',
description: 'Highlights certain users in comment threads: OP, Admin, Friends, Mod - contributed by MrDerk',
options: {
highlightOP: {
type: 'boolean',
value: true,
description: 'Highlight OP\'s comments'
},
OPColor: {
type: 'text',
value: '#0055DF',
description: 'Color to use to highlight OP. Defaults to original text color'
},
OPColorHover: {
type: 'text',
value: '#4E7EAB',
description: 'Color used to highlight OP on hover.'
},
highlightAdmin: {
type: 'boolean',
value: true,
description: 'Highlight Admin\'s comments'
},
adminColor: {
type: 'text',
value: '#FF0011',
description: 'Color to use to highlight Admins. Defaults to original text color'
},
adminColorHover: {
type: 'text',
value: '#B3000C',
description: 'Color used to highlight Admins on hover.'
},
highlightFriend: {
type: 'boolean',
value: true,
description: 'Highlight Friends\' comments'
},
friendColor: {
type: 'text',
value: '#FF4500',
description: 'Color to use to highlight Friends. Defaults to original text color'
},
friendColorHover: {
type: 'text',
value: '#B33000',
description: 'Color used to highlight Friends on hover.'
},
highlightMod: {
type: 'boolean',
value: true,
description: 'Highlight Mod\'s comments'
},
modColor: {
type: 'text',
value: '#228822',
description: 'Color to use to highlight Mods. Defaults to original text color'
},
modColorHover: {
type: 'text',
value: '#134913',
description: 'Color used to highlight Mods on hover. Defaults to gray.'
},
fontColor: {
type: 'text',
value: 'white',
description: 'Color for highlighted text.',
}
},
isEnabled: function() {
return RESConsole.getModulePrefs(this.moduleID);
},
include: Array(
/http:\/\/www.reddit.com\/[\?]*/i
),
isMatchURL: function() {
return RESUtils.isMatchURL(this.moduleID);
},
go: function() {
if ((this.isEnabled()) && (this.isMatchURL())) {
// get this module's options...
RESUtils.getOptions(this.moduleID);
if (this.options.highlightFriend.value) {
var name = 'friend';
var color = this.options.friendColor.value;
var hoverColor = this.options.friendColorHover.value;
this.doHighlight(name,color,hoverColor);
}
if (this.options.highlightMod.value) {
var name = 'moderator';
var color = this.options.modColor.value;
var hoverColor = this.options.modColorHover.value;
this.doHighlight(name,color,hoverColor);
}
if (this.options.highlightAdmin.value) {
var name = 'admin';
var color = this.options.adminColor.value;
var hoverColor = this.options.adminColorHover.value;
this.doHighlight(name,color,hoverColor);
}
if (this.options.highlightOP.value) {
var name = 'submitter';
var color = this.options.OPColor.value;
var hoverColor = this.options.OPColorHover.value;
this.doHighlight(name,color,hoverColor);
}
}
},
doHighlight: function(name,color,hoverColor) {
// First look for .noncollapsed members. If they're there, we have comments
// If we skip the noncollapsed, we can pick up the gray, collapsed versions
// If that's the case, you'll end up with gray as your 'default' color
var firstComment = document.querySelector('.noncollapsed .' + name);
// This kicks in if a friend/admin/mod has made a post but not a comment,
// allowing them to be highlighted at the top of the submission
if (firstComment === null) {
firstComment = document.querySelector('.' + name);
}
if (firstComment != null) {
if (color === 'default') {
color = this.getStyle(firstComment, 'color');
}
if (hoverColor === 'default') {
hoverColor = "#AAA";
}
if(typeof(color) != "undefined" && color != 'rgb(255, 255, 255)') {
RESUtils.addCSS("\
.author." + name + " { \
color: " + this.options.fontColor.value + " !important; \
font-weight: bold; \
padding: 0 2px 0 2px; \
border-radius: 3px; \
-moz-border-radius: 3px; \
-webkit-border-radius: 3px; \
background-color:" + color + " !important} \
.collapsed .author." + name + " { \
color: white !important; \
background-color: #AAA !important}\
.author." + name + ":hover {\
background-color: " + hoverColor + " !important; \
text-decoration: none !important}");
// this.addCSS(css);
}
}
},
/*addCSS: function(css) {
// Add CSS Style
var heads = document.getElementsByTagName("head");
if (heads.length > 0) {
var node = document.createElement("style");
node.type = "text/css";
node.appendChild(document.createTextNode(css));
heads[0].appendChild(node);
}
},*/
getStyle: function(oElm, strCssRule){
var strValue = "";
if(document.defaultView && document.defaultView.getComputedStyle){
strValue = document.defaultView.getComputedStyle(oElm, "").getPropertyValue(strCssRule);
}
else if(oElm.currentStyle){
strCssRule = strCssRule.replace(/\-(\w)/g, function (strMatch, p1){
return p1.toUpperCase();
});
strValue = oElm.currentStyle[strCssRule];
}
return strValue;
}
};
modules['styleTweaks'] = {
moduleID: 'styleTweaks',
moduleName: 'Style Tweaks',
description: 'Provides a number of style tweaks to the Reddit interface',
options: {
navTop: {
type: 'boolean',
value: true,
description: 'Moves the username navbar to the top (great on netbooks!)'
},
commentBoxes: {
type: 'boolean',
value: true,
description: 'Highlights comment boxes for easier reading / placefinding in large threads.'
},
commentBoxShadows: {
type: 'boolean',
value: false,
description: 'Drop shadows on comment boxes (turn off for faster performance)'
},
commentRounded: {
type: 'boolean',
value: true,
description: 'Round corners of comment boxes'
},
commentBoxHover: {
type: 'boolean',
value: true,
description: 'Highlight comment box hierarchy on hover (turn off for faster performance)'
},
commentIndent: {
type: 'text',
value: 10,
description: 'Indent comments by [x] pixels (only enter the number, no \'px\')'
},
continuity: {
type: 'boolean',
value: false,
description: 'Show comment continuity lines'
},
lightSwitch: {
type: 'boolean',
value: true,
description: 'Enable lightswitch (toggle between light / dark reddit)'
},
lightOrDark: {
type: 'enum',
values: [
{ name: 'Light', value: 'light' },
{ name: 'Dark', value: 'dark' }
],
value: 'light',
description: 'Light, or dark?'
},
showExpandos: {
type: 'boolean',
value: true,
description: 'Bring back video and text expando buttons for users with compressed link display'
}
},
isEnabled: function() {
return RESConsole.getModulePrefs(this.moduleID);
},
include: Array(
/http:\/\/www.reddit.com\/[-\w\.\/]*/i,
/https:\/\/www.reddit.com\/[-\w\.\/]*/i
),
isMatchURL: function() {
return RESUtils.isMatchURL(this.moduleID);
},
go: function() {
if ((this.isEnabled()) && (this.isMatchURL())) {
// get this module's options...
RESUtils.getOptions(this.moduleID);
if (this.options.navTop.value) {
this.navTop();
}
var commentsRegex = new RegExp(/http:\/\/www.reddit.com\/[-\w\.\/]*comments\/[-\w\.\/]*/i);
if ((this.options.commentBoxes.value) && (commentsRegex.test(location.href))) {
this.commentBoxes();
}
if (this.options.lightSwitch.value) {
this.lightSwitch();
}
if (this.options.lightOrDark.value == 'dark') {
this.redditDark();
}
if (this.options.showExpandos.value) {
RESUtils.addCSS('.compressed .expando-button { display: block !important; }');
//GM_addStyle(css);
}
this.userbarHider();
this.subredditStyles();
}
},
navTop: function() {
RESUtils.addCSS('#header-bottom-right { top: 20px; border-radius: 0px 0px 0px 7px; -moz-border-radius: 0px 0px 0px 7px; -webkit-border-radius: 0px 0px 0px 7px; bottom: auto; }');
//GM_addStyle(css);
},
userbarHider: function() {
RESUtils.addCSS("#userbarToggle { width: 7px; height: 11px; display: inline-block; background-image: url('http://thumbs.reddit.com/t5_2s10b_0.png'); cursor: pointer; margin-right: 4px; background-repeat: no-repeat; }");
RESUtils.addCSS(".userbarHide { background-position: 0px -137px; }");
RESUtils.addCSS(".userbarShow { background-position: -8px -137px; }");
//GM_addStyle(css);
var userbar = document.getElementById('header-bottom-right');
if (userbar) {
this.userbarToggle = createElementWithID('div','userbarToggle');
this.userbarToggle.setAttribute('title','Toggle Userbar');
addClass(this.userbarToggle, 'userbarHide');
this.userbarToggle.addEventListener('click', function(e) {
modules['styleTweaks'].toggleUserBar();
}, false);
userbar.insertBefore(this.userbarToggle, userbar.firstChild);
if (localStorage.getItem('RESModules.styleTweaks.userbarState') == 'hidden') {
this.toggleUserBar();
}
}
},
toggleUserBar: function() {
var nextEle = this.userbarToggle.nextSibling;
if (hasClass(this.userbarToggle,'userbarHide')) {
removeClass(this.userbarToggle,'userbarHide');
addClass(this.userbarToggle,'userbarShow');
localStorage.setItem('RESModules.styleTweaks.userbarState', 'hidden');
while ((typeof(nextEle) != 'undefined') && (nextEle != null)) {
nextEle.style.display = 'none';
nextEle = nextEle.nextSibling;
}
} else {
removeClass(this.userbarToggle,'userbarShow');
addClass(this.userbarToggle,'userbarHide');
localStorage.setItem('RESModules.styleTweaks.userbarState', 'visible');
while ((typeof(nextEle) != 'undefined') && (nextEle != null)) {
nextEle.style.display = 'inline-block';
nextEle = nextEle.nextSibling;
}
}
},
commentBoxes: function() {
// replaced with a less intensive method... adapted from Reddit Comment Boxes via:
// @description Updated version of Tiby312's Reddit Comment Boxes script (https://userscripts-mirror.org/scripts/show/63628)
// @author flatluigi
RESUtils.addCSS(".comment{");
if (this.options.commentRounded.value) {
RESUtils.addCSS(" -moz-border-radius:7px !important;"+
" -webkit-border-radius:7px !important;"+
" border-radius:7px !important;");
}
RESUtils.addCSS(" margin-left:"+this.options.commentIndent.value+"px !important;"+
" margin-right:8px!important;"+
" margin-top:0px!important;"+
" margin-bottom:8px!important;"+
" background-color:#ffffff !important;"+
" border:1px solid #bbbcbf !important;"+
" padding-left:5px!important;"+
" padding-top:5px!important;"+
" padding-right:8px!important;"+
" padding-bottom:0px!important;"+
"}");
if (this.options.commentBoxShadows.value) {
RESUtils.addCSS('.comment {-webkit-box-shadow: #BBB 2px 2px 0px inset; -moz-box-shadow: #BBB 2px 2px 0px inset; box-shadow: #BBB 2px 2px 0px inset; border:1px solid #DDD; padding-bottom:0px; padding: 4px 5px 4px 4px; margin: 0px 0px 3px; }');
}
if (this.options.continuity.value) {
RESUtils.addCSS('.comment div.child { border-left: 1px dotted #555555 !important; } ');
} else {
RESUtils.addCSS('.comment div.child { border-left: none !important; } ');
}
RESUtils.addCSS(".comment .comment{"+
" margin-right:0px!important;"+
" background-color:#F7F7F8 !important;"+
"}"+
".comment .comment .comment{"+
" background-color:#ffffff !important;"+
"}"+
".comment .comment .comment .comment{"+
" background-color:#F7F7F8 !important;"+
"}"+
".comment .comment .comment .comment .comment{"+
" background-color:#ffffff !important;"+
"}"+
".comment .comment .comment .comment .comment .comment{"+
" background-color:#F7F7F8 !important;"+
"}"+
".comment .comment .comment .comment .comment .comment .comment{"+
" background-color:#ffffff !important;"+
"}"+
".comment .comment .comment .comment .comment .comment .comment .comment{"+
" background-color:#F7F7F8 !important;"+
"}"+
".comment .comment .comment .comment .comment .comment .comment .comment .comment{"+
" background-color:#ffffff !important;"+
"}"+
".comment .comment .comment .comment .comment .comment .comment .comment .comment .comment{"+
" background-color:#F7F7F8 !important;"+
"}"+
".commentarea, .link, .comment {"+
" overflow:hidden; !important;"+
"}"+
"body > .content {"+
" padding-right:0px; !important;"+
"}");
if (this.options.commentBoxHover.value) {
RESUtils.addCSS(" .comment:hover {border: 1px solid #99AAEE !important; }");
}
//GM_addStyle(css);
},
lightSwitch: function() {
RESUtils.addCSS("#lightSwitch { width: 24px; height: 11px; display: inline-block; background-image: url('http://thumbs.reddit.com/t5_2s10b_0.png'); cursor: pointer; }");
RESUtils.addCSS(".lightOn { background-position: 0px -96px; } ");
RESUtils.addCSS(".lightOff { background-position: 0px -108px; } ");
//GM_addStyle(css);
var thisFrag = document.createDocumentFragment();
var separator = document.createElement('span');
addClass(separator,'separator');
separator.innerHTML = '|';
this.lightSwitch = document.createElement('span');
this.lightSwitch.setAttribute('title',"Toggle night and day");
this.lightSwitch.addEventListener('click',function(e) {
e.preventDefault();
if (hasClass(modules['styleTweaks'].lightSwitch, 'lightOn')) {
removeClass(modules['styleTweaks'].lightSwitch, 'lightOn');
addClass(modules['styleTweaks'].lightSwitch, 'lightOff');
RESUtils.setOption('styleTweaks','lightOrDark','dark');
modules['styleTweaks'].redditDark();
} else {
removeClass(modules['styleTweaks'].lightSwitch, 'lightOff');
addClass(modules['styleTweaks'].lightSwitch, 'lightOn');
RESUtils.setOption('styleTweaks','lightOrDark','light');
modules['styleTweaks'].redditDark(true);
}
}, true);
this.lightSwitch.setAttribute('id','lightSwitch');
this.lightSwitch.innerHTML = '&nbsp;';
(this.options.lightOrDark.value == 'dark') ? addClass(this.lightSwitch, 'lightOff') : addClass(this.lightSwitch, 'lightOn');
thisFrag.appendChild(separator);
thisFrag.appendChild(this.lightSwitch);
insertAfter(RESConsole.RESPrefsLink, thisFrag);
},
subredditStyles: function() {
this.ignoredSubReddits = JSON.parse(localStorage.getItem('RESmodules.styleTweaks.ignoredSubredditStyles'));
if (this.ignoredSubReddits == null) {
this.ignoredSubReddits = Array();
}
this.head = document.getElementsByTagName("head")[0];
var subredditTitle = document.querySelector('.titlebox h1');
var styleToggle = document.createElement('div');
var thisLabel = document.createElement('label');
addClass(styleToggle,'styleToggle');
thisLabel.setAttribute('for','subRedditStyleCheckbox');
thisLabel.innerHTML = 'Use subreddit style ';
styleToggle.appendChild(thisLabel);
this.styleToggleCheckbox = document.createElement('input');
this.styleToggleCheckbox.setAttribute('type','checkbox');
this.styleToggleCheckbox.setAttribute('name','subRedditStyleCheckbox');
this.curSubReddit = RESUtils.currentSubReddit();
if ((this.curSubReddit != null) && (subredditTitle != null)) {
var idx = this.ignoredSubReddits.indexOf(this.curSubReddit);
if (idx == -1) {
this.styleToggleCheckbox.checked = true;
} else {
this.toggleSubredditStyle(false);
}
this.styleToggleCheckbox.addEventListener('change', function(e) {
modules['styleTweaks'].toggleSubredditStyle(this.checked);
}, false);
styleToggle.appendChild(this.styleToggleCheckbox);
insertAfter(subredditTitle, styleToggle);
}
},
toggleSubredditStyle: function(toggle) {
if (toggle) {
var idx = this.ignoredSubReddits.indexOf(this.curSubReddit);
if (idx != -1) this.ignoredSubReddits.splice(idx, 1); // Remove it if found...
var subredditStyleSheet = document.createElement('link');
subredditStyleSheet.setAttribute('title','applied_subreddit_stylesheet');
subredditStyleSheet.setAttribute('rel','stylesheet');
subredditStyleSheet.setAttribute('href','http://www.reddit.com/r/'+this.curSubReddit+'/stylesheet.css');
this.head.appendChild(subredditStyleSheet);
} else {
var idx = this.ignoredSubReddits.indexOf(this.curSubReddit); // Find the index
if (idx==-1) this.ignoredSubReddits[this.ignoredSubReddits.length] = this.curSubReddit;
var subredditStyleSheet = this.head.querySelector('link[title=applied_subreddit_stylesheet]');
if (subredditStyleSheet) {
subredditStyleSheet.parentNode.removeChild(subredditStyleSheet);
}
}
localStorage.setItem('RESmodules.styleTweaks.ignoredSubredditStyles',JSON.stringify(this.ignoredSubReddits));
},
redditDark: function(off) {
if (off) {
if (typeof(this.darkStyle) != 'undefined') {
this.darkStyle.parentNode.removeChild(this.darkStyle);
}
} else {
var css = "div[class=\"organic-listing\"] ul[class=\"tabmenu \"], div[id=\"header-bottom-left\"]] {background-color: #666 !important; } ::-moz-selection { background:orangered; }";
css += "body { background-color: #222 !important; } .infobar { background-color:#222 !important; color:black !important; }";
css += ".side { background:none !important; } h2, .tagline a, .content a, .footer a, .wired a, .side a, .subredditbox li a { color:#8AD !important; }";
css += ".rank .star { color:orangered !important; } .content { color:#CCC !important; } .thing .title.loggedin, .link .title { color:#DFDFDF !important; }";
css += ".link .midcol, .linkcompressed .midcol, .comment .midcol { background:none !important; margin-right:6px !important; margin-top:4px !important; }";
css += ".link .midcol { width:29px !important; } .link .midcol .arrow { margin-left:7px !important; margin-right:7px !important; }";
css += ".arrow { height:14px !important; margin-top:0 !important; width:15px !important; }";
css += ".arrow.up { background:url(http://thumbs.reddit.com/t5_2qlyl_0.png?v=zs9q49wxah08x4kpv2tu5x4nbda7kmcpgkbj) -15px 0 no-repeat !important; }";
css += ".arrow.down { background:url(http://thumbs.reddit.com/t5_2qlyl_0.png?v=10999ad3mtco31oaf6rrggme3t9jdztmxtg6) -15px -14px no-repeat !important; }";
css += ".arrow.up:hover { background:url(http://thumbs.reddit.com/t5_2qlyl_0.png?v=9oeida688vtqjpb4k0uy93oongrzuv5j7vcj) -30px 0 no-repeat !important; }";
css += ".arrow.down:hover { background:url(http://thumbs.reddit.com/t5_2qlyl_0.png?v=cmsw4qrin2rivequ0x1wnmn8ltd7ke328yqs) -30px -14px no-repeat !important; }";
css += ".arrow.upmod { background:url(http://thumbs.reddit.com/t5_2qlyl_0.png?v=8oarqkcswl255wrw3q1kyd74xrty50a7wr3z) 0 0 no-repeat !important; }";
css += ".arrow.downmod { background:url(http://thumbs.reddit.com/t5_2qlyl_0.png?v=90eauq018nf41z3vr0u249gv2q6651xyzrkh) 0 -14px no-repeat !important; }";
css += ".link .score.likes, .linkcompressed .score.likes { color:orangered !important; }";
css += ".link .score.dislikes, .linkcompressed .score.dislikes { color:#8AD !important; }";
css += ".linkcompressed .entry .buttons li a, .link .usertext .md, .thing .compressed, organic-listing .link, .organic-listing .link.promotedlink, .link.promotedlink.promoted { background:none !important; }";
css += ".subredditbox li a:before { content:\"#\" !important; } .subredditbox li { font-weight:bold !important; text-transform:lowercase !important; }";
css += ".side h3:after { content:\" (#reddit on freenode)\" !important; font-size:85% !important; font-weight:normal !important; }";
css += "#subscribe a { color:#8AD !important; } .dropdown.lightdrop .drop-choices { background-color:#333 !important; }";
css += ".dropdown.lightdrop a.choice:hover { background-color:#111 !important; } .side { background:none !important; color:#fff !important; margin-left:10px !important; }";
css += ".side h4, .side h3 { color:#ddd !important; } .side h5 { color:#aaa !important; margin-top:5px !important; } .side p { margin-top:5px !important; }";
css += ".sidebox, .subredditbox, .subreddit-info, .raisedbox, .login-form-side { background-color:#393939 !important; border:2px solid #151515 !important; color:#aaa !important; border-radius:8px !important; -moz-border-radius:8px !important; -webkit-border-radius:8px !important; }";
css += ".login-form-side { background:#e8690a !important; border-bottom:0 !important; border-color:#e8690a !important; padding-bottom:1px !important; position:relative !important; }";
css += ".login-form-side input { width:125px !important; } .login-form-side label { color:#111 !important; } .login-form-side a { color:#FFFFFF !important; font-size:11px !important; }";
css += ".login-form-side .error { color:#660000 !important; } .subreddit-info .label { color:#aaa !important; } .subreddit-info { padding:10px !important; }";
css += ".subreddit-info .spacer a { background-color:#222; border:none !important; margin-right:3px !important; }";
css += ".subredditbox ul { padding:10px 0px 10px 3px !important; width:140px !important; } .subredditbox ul a:hover { text-decoration:underline !important; } .morelink { background:none !important; border:0 !important; border-radius-bottomleft:6px !important; -moz-border-radius-bottomleft:6px !important; -webkit-border-radius-bottomleft:6px !important; -moz-border-radius-topright:6px !important; -webkit-border-radius-bottom-left-radius:6px !important; -webkit-border-radius-top-right-radius:6px !important; }";
css += ".morelink.blah:hover { background:none !important; color:#369 !important; } .morelink.blah { background:none !important; border:0 !important; color:#369 !important; }";
css += ".morelink:hover { border:0 !important; color:white !important; } .sidebox { padding-left:60px !important; }";
css += ".sidebox.submit { background:#393939 url(http://thumbs.reddit.com/t5_2qlyl_2.png?v=0s1s9iul2umpm0bx46cioc7yjwbkprt7r2qr) no-repeat 6px 50% !important; }";
css += ".sidebox .spacer, .linkinfo {background-color:#393939 !important; } .nub {background-color: transparent !important;}";
css += ".sidebox.create { background:#393939 url(http://thumbs.reddit.com/t5_2qlyl_1.png?v=gl82ywfldj630zod4iaq56cidjud4n79wqw8) no-repeat 6px 50% !important; }";
css += ".sidebox .subtitle { color:#aaa !important; } h1 { border-bottom:1px solid #444 !important; }";
css += "button.btn { background:none !important; border:2px solid black !important; color:black !important; position:relative !important; width:auto !important; }";
css += ".commentreply .buttons button { margin-left:0 !important; margin-top:5px !important; } .commentreply .textarea { color:black !important; }";
css += ".menuarea { margin-right:315px !important; } .permamessage { background-image:url(http://thumbs.reddit.com/t5_2qlyl_3.png?v=uza2aq80cb2x2e90ojhdqooj1wazax4jjzfc) !important; border-color:#369 !important; }";
css += ".commentbody.border { background-color:#369 !important; } .commentreply .help tr { background:none !important; } .commentreply table.help { margin:2px !important; }";
css += "#newlink th { padding-top:5px !important; vertical-align:top !important; } .pretty-form.long-text input[type=\"text\"], .pretty-form.long-text textarea, .pretty-form.long-text input[type=\"password\"], .commentreply textarea { background-color:#333 !important; border:2px solid black !important; color:#CCC !important; padding:4px !important; }";
css += "input#title { height:5em !important; } .spam, .reported { background:none !important; border:2px dotted !important; padding:4px !important; }";
css += ".spam { border-color:orangered !important; } .reported { border-color:goldenrod !important; } .organic-listing .linkcompressed { background:none !important; }";
css += ".organic-listing .nextprev img { opacity:.7 !important; } .organic-listing .nextprev img:hover { opacity:.85 !important; }";
css += "#search input[type=\"text\"] { background-color:#222 !important; color:gray !important; } #search input[type=\"text\"]:focus { color:white !important; }";
css += "#sr-header-area, #sr-more-link { background:#c2d2e2 !important; } #sr-header-area { border-bottom:none !important; }";
css += "#header-bottom-left .tabmenu .selected a { border-bottom:none !important; padding-bottom:0 !important; } #ad-frame { opacity:.8 !important; }";
css += ".comment.unread { background-color:#4A473B !important; } .raisedbox .flat-list a { background-color:#222 !important; -moz-border-radius:2px !important; -webkit-border-radius:2px !important; }";
css += ".raisedbox .flat-list a:hover { background-color:#336699 !important; color:white !important; } .instructions { background:white !important; padding:10px !important; }";
css += ".instructions .preftable th, .instructions .pretty-form { color:black !important; } #feedback { padding:10px !important; } span[class=\"hover pagename redditname\"] a {font-size: 1.7em !important;}";
css += ".thing .title.loggedin:visited, .link .title:visited {color: #666666 !important;} legend {background-color: black !important;}";
css += "a.author.moderator, a.moderator {color:#3F4 !important; } a.author.friend, a.friend {color:rgb(255, 139, 36) !important; } a.submitter {color: #36F !important; }";
css += "a.author.admin, a.admin{color: #f22 !important; } a.author.submitter { } table[class=\"markhelp md\"] tr td { background-color: #555 !important; }";
css += "div.infobar { color: #ccc !important; } table[class=\"markhelp md\"] tr[style=\"background-color: rgb(255, 255, 153); text-align: center;\"] td { background-color: #36c !important; }";
css += "form[class=\"usertext border\"] div.usertext-body { background-color: transparent !important; border-width: 2px !important; border-style: solid !important; border-color: #999 !important; }";
css += "div[class=\"midcol likes\"], div[class=\"midcol dislikes\"], div[class=\"midcol unvoted\"] {padding: 0px 7px 0px 0px !important;}";
css += "form[class=\"usertext border\"] div.usertext-body div.md { background-color: transparent !important; } form#form-t1_c0b71p54yc div {color: black !important;}";
css += "a[rel=\"tag\"], a.dsq-help {color: #8AD !important; } div[class=\"post-body entry-content\"], div.dsq-auth-header { color: #ccc !important; }";
css += "div#siteTable div[onclick=\"click_thing(this)\"] {background-color: #222 !important;} .md p {color: #ddd !important; } .mail .havemail img, .mail .nohavemail img { visibility: hidden; }";
css += ".havemail { background: url('http://i.imgur.com/2Anoz.gif') bottom left no-repeat; } .mail .nohavemail { background: url('http://imgur.com/6WV6Il.gif') bottom left no-repeat; }";
css += "#header-bottom-right { background-color: #BBBBBB !important; }";
css += '.expando-button.image {background: none !important; background-image: url(http://thumbs.reddit.com/t5_2s10b_2.png) !important;}';
css += '.expando-button.image.collapsed {background-position: 0px 0px !important;}';
css += '.expando-button.image.collapsed:hover {background-position: 0px -24px !important;}';
css += '.expando-button.image.expanded, .eb-se { margin-bottom:5px; background-position: 0px -48px !important;}';
css += '.expando-button.image.expanded:hover, .eb-seh {background-position: 0px -72px !important;}';
css += '.expando-button.selftext {background: none !important; background-image: url(http://thumbs.reddit.com/t5_2s10b_2.png) !important;}';
css += '.expando-button.selftext.collapsed {background-position: 0px -96px !important;}';
css += '.expando-button.selftext.collapsed:hover {background-position: 0px -120px !important;}';
css += '.expando-button.selftext.expanded, .eb-se { margin-bottom:5px; background-position: 0px -144px !important;}';
css += '.expando-button.selftext.expanded:hover, .eb-seh {background-position: 0px -168px !important;}';
css += '.expando-button.video {background: none !important; background-image: url(http://thumbs.reddit.com/t5_2s10b_2.png) !important;}';
css += '.expando-button.video.collapsed {background-position: 0px -192px !important;}';
css += '.expando-button.video.collapsed:hover {background-position: 0px -216px !important;}';
css += '.expando-button.video.expanded, .eb-se { margin-bottom:5px; background-position: 0px -240px !important;}';
css += '.expando-button.video.expanded:hover, .eb-seh {background-position: 0px -264px !important;}';
css += '.expando-button { background-color:transparent!important; }';
css += ".keyHighlight, .keyHighlight div.md { background-color: #666666 !important; } .keyHighlight .title.loggedin:visited { color: #dfdfdf !important; } .nub {background: none !important;}";
css += ".comment{"+
" background-color:#444444 !important;"+
"}"+
".comment .comment{"+
" background-color:#111111 !important;"+
"}"+
".comment .comment .comment{"+
" background-color:#444444 !important;"+
"}"+
".comment .comment .comment .comment{"+
" background-color:#111111 !important;"+
"}"+
".comment .comment .comment .comment .comment{"+
" background-color:#444444 !important;"+
"}"+
".comment .comment .comment .comment .comment .comment{"+
" background-color:#111111 !important;"+
"}"+
".comment .comment .comment .comment .comment .comment .comment{"+
" background-color:#444444 !important;"+
"}"+
".comment .comment .comment .comment .comment .comment .comment .comment{"+
" background-color:#111111 !important;"+
"}"+
".comment .comment .comment .comment .comment .comment .comment .comment .comment{"+
" background-color:#444444 !important;"+
"}"+
".comment .comment .comment .comment .comment .comment .comment .comment .comment .comment{"+
" background-color:#111111 !important;"+
"}";
css += '.organic-listing .link { background-color: #333333 !important; } .sidecontentbox { background-color: #111111; } .side { background: none !important; background-color: #000000 !important; }';
if (this.options.continuity.value) {
css += '.comment div.child { border-left: 1px dotted #555555 !important; } ';
} else {
css += '.comment div.child { border-left: none !important; } ';
}
css += '.roundfield {background-color: #111111 !important;}';
css += '#authorInfoToolTip { background-color: #666666 !important; color: #cccccc !important; border-color: #888888 !important; } #authorInfoToolTip a { color: #88AADD !important; } ';
css += '.new-comment .usertext-body { background-color: #444444 !important; border: none !important; margin:-1px 0; }';
css += '.thing { border: 1px solid #666666 !important; } ';
css += '.usertext-edit textarea { background-color: #666666 !important; color: #CCCCCC !important; } ';
this.darkStyle = createElementWithID('style', 'darkStyle');
this.darkStyle.innerHTML = css;
document.body.appendChild(this.darkStyle);
}
// GM_addStyle(css);
}
};
modules['accountSwitcher'] = {
moduleID: 'accountSwitcher',
moduleName: 'Account Switcher',
options: {
accounts: {
type: 'table',
fields: [
{ name: 'username', type: 'text' },
{ name: 'password', type: 'password' }
],
value: [
/*
['somebodymakethis','SMT','[SMT]'],
['pics','pic','[pic]']
*/
],
description: 'Set your usernames and passwords below. They are only stored in RES preferences.'
}
},
description: 'Store username/password pairs and switch accounts instantly while browsing Reddit!',
isEnabled: function() {
return RESConsole.getModulePrefs(this.moduleID);
},
include: Array(
/http:\/\/www.reddit.com\/[-\w\.\/]*/i,
/https:\/\/www.reddit.com\/[-\w\.\/]*/i
),
isMatchURL: function() {
return RESUtils.isMatchURL(this.moduleID);
},
go: function() {
if ((this.isEnabled()) && (this.isMatchURL())) {
// get this module's options...
RESUtils.getOptions(this.moduleID);
// do stuff now!
// this is where your code goes...
this.userLink = document.querySelector('#header-bottom-right > span.user > a');
// this.loggedInUser = userLink.innerHTML;
this.loggedInUser = RESUtils.loggedInUser();
var downArrowIMG = '';
var downArrow = document.createElement('img');
downArrow.setAttribute('src', downArrowIMG);
downArrow.style.cursor = 'pointer';
downArrow.style.marginLeft = '3px';
downArrow.addEventListener('click',function(e) {
e.preventDefault();
modules['accountSwitcher'].toggleAccountMenu();
}, true);
insertAfter(this.userLink, downArrow);
this.accountMenu = createElementWithID('UL','accountSwitcherMenu');
this.accountMenu.style.display = 'none';
RESUtils.addCSS('#accountSwitcherMenu { position: absolute; z-index: 999; display: none; padding: 3px; background-color: #ffffff; }');
RESUtils.addCSS('.accountName { padding: 2px; border-bottom: 1px solid #AAAAAA; border-left: 1px solid #AAAAAA; border-right: 1px solid #AAAAAA; }');
RESUtils.addCSS('.accountName:first-child { padding: 2px; border-top: 1px solid #AAAAAA; }');
RESUtils.addCSS('.accountName:hover { background-color: #F3FAFF; }');
// GM_addStyle(css);
var accounts = this.options.accounts.value;
if (accounts != null) {
var accountCount = 0;
for (var i=0, len=accounts.length; i<len; i++) {
thisPair = accounts[i];
if (thisPair[0] != this.loggedInUser) {
accountCount++;
var thisLI = document.createElement('LI');
addClass(thisLI, 'accountName');
thisLI.innerHTML = thisPair[0];
thisLI.style.cursor = 'pointer';
thisLI.addEventListener('click', function(e) {
e.preventDefault();
modules['accountSwitcher'].toggleAccountMenu();
modules['accountSwitcher'].switchTo(e.target.innerHTML);
}, true);
this.accountMenu.appendChild(thisLI);
}
}
var thisLI = document.createElement('LI');
addClass(thisLI, 'accountName');
thisLI.innerHTML = '+ add account';
thisLI.style.cursor = 'pointer';
thisLI.addEventListener('click', function(e) {
e.preventDefault();
modules['accountSwitcher'].toggleAccountMenu();
modules['accountSwitcher'].manageAccounts();
}, true);
this.accountMenu.appendChild(thisLI);
}
document.body.appendChild(this.accountMenu);
}
},
toggleAccountMenu: function() {
if (this.accountMenu.style.display == 'none') {
thisXY=RESUtils.getXYpos(this.userLink);
this.accountMenu.style.top = (thisXY.y + 12) + 'px';
this.accountMenu.style.left = (thisXY.x - 10) + 'px';
this.accountMenu.style.display = 'block';
} else {
this.accountMenu.style.display = 'none';
}
},
switchTo: function(username) {
var accounts = this.options.accounts.value;
var password = '';
for (var i=0, len=accounts.length; i<len; i++) {
thisPair = accounts[i];
if (thisPair[0] == username) {
password = thisPair[1];
}
}
// console.log('request with user: ' +username+ ' -- passwd: ' + password);
GM_xmlhttpRequest({
method: "POST",
url: '/api/login',
data: 'user='+username+'&passwd='+password,
headers: {
"Content-Type": "application/x-www-form-urlencoded"
},
onload: function(response) {
var data = JSON.parse(response.responseText);
var errorCheck = data.jquery[10][3][0];
var error = /error/;
if (error.test(errorCheck)) {
alert('Incorrect login and/or password. Please check your configuration.');
} else {
location.reload();
}
}
});
},
manageAccounts: function() {
RESConsole.open();
RESConsole.menuClick(document.getElementById('Menu-Configure Modules'));
RESConsole.drawConfigOptions('accountSwitcher');
}
};
modules['RESTips'] = {
moduleID: 'RESTips',
moduleName: 'RES Tips and Tricks',
options: {
// any configurable options you have go here...
// options must have a type and a value..
// valid types are: text, boolean (if boolean, value must be true or false)
// for example:
dailyTip: {
type: 'boolean',
value: true,
description: 'Show a random tip once every 24 hours.'
}
},
description: 'Adds tips/tricks help to RES console',
isEnabled: function() {
return RESConsole.getModulePrefs(this.moduleID);
},
include: Array(
/http:\/\/www.reddit.com\/[\?]*/i
),
isMatchURL: function() {
return RESUtils.isMatchURL(this.moduleID);
},
go: function() {
if ((this.isEnabled()) && (this.isMatchURL())) {
// get this module's options...
RESUtils.getOptions(this.moduleID);
// do stuff now!
// this is where your code goes...
this.init();
if (this.options.dailyTip.value) {
this.dailyTip();
}
}
},
init: function() {
thisXY=RESUtils.getXYpos(RESConsole.RESPrefsLink);
RESUtils.addCSS("#tipsAndTricksToolTip { position: absolute; background-color: #F3FCFF; width: 300px; border: 2px solid #cccccc; padding: 6px; font-size: 12px; }");
RESUtils.addCSS("#tipsAndTricksToolTip h2 { float: left; }");
RESUtils.addCSS("#closeTipsAndTricksToolTip { float: right; cursor: pointer; } ");
RESUtils.addCSS("#tipsAndTricksText { clear: both; padding-top: 3px; padding-bottom: 3px; } ");
RESUtils.addCSS("#toolTipNextPrev { border-top: 1px solid #cccccc; }");
RESUtils.addCSS("#toolTipPrev, #toolTipNext { color: #7799ff; cursor: pointer; }");
RESUtils.addCSS("#toolTipPrev { float: left; }");
RESUtils.addCSS("#toolTipNext { float: right; }");
//GM_addStyle(css);
this.toolTip = createElementWithID('div','tipsAndTricksToolTip');
var toolTipHeader = document.createElement('h2');
toolTipHeader.innerHTML = 'RES Tips and Tricks:';
this.toolTip.appendChild(toolTipHeader);
var toolTipCloseButton = createElementWithID('div','closeTipsAndTricksToolTip');
toolTipCloseButton.innerHTML = 'x';
toolTipCloseButton.addEventListener('click',function(e) {
e.preventDefault();
RESUtils.fadeElementOut(modules['RESTips'].toolTip, 0.3);
}, false);
this.toolTip.appendChild(toolTipCloseButton);
this.toolTip.setAttribute('style', 'top: ' + (thisXY.y + 20) + 'px; left: ' + (thisXY.x - 305) + 'px;');
this.toolTip.style.display = 'none';
this.toolTip.style.zIndex = 1001;
this.toolTipText = createElementWithID('div','tipsAndTricksText');
this.toolTipText.innerHTML = '';
this.toolTip.appendChild(this.toolTipText);
var toolTipNextPrev = createElementWithID('div', 'toolTipNextPrev');
var toolTipPrev = createElementWithID('span','toolTipPrev');
toolTipPrev.innerHTML = 'Prev';
toolTipPrev.addEventListener('click', function(e) {
e.preventDefault();
modules['RESTips'].nextPrevTip(-1);
}, false);
this.toolTip.appendChild(toolTipPrev);
var toolTipNext = createElementWithID('span','toolTipNext');
toolTipNext.innerHTML = 'Next';
toolTipNext.addEventListener('click', function(e) {
e.preventDefault();
modules['RESTips'].nextPrevTip(1);
}, false);
this.toolTip.appendChild(toolTipNext);
document.body.appendChild(this.toolTip);
},
dailyTip: function() {
var lastCheck = parseInt(localStorage.getItem('RESLastToolTip')) || 0;
var now = new Date();
// 86400000 = 1 day
if ((now.getTime() - lastCheck) > 86400000) {
// mark off that we've displayed a new tooltip
localStorage.setItem('RESLastToolTip',now.getTime());
if (lastCheck == 0) {
var thisTip = 'Welcome to RES. You can turn modules on and off, and configure settings for the modules using the [RES] link above. For feature requests, bug reports, etc - head over to <a href="http://reddit.com/r/Enhancement">/r/Enhancement</a>.<br>Do you keep seeing this message? <a target=\"_blank\" href=\"http://reddit.honestbleeps.com/faq\">see the FAQ</a> about BetterPrivacy and similar addons.';
this.showTip(thisTip);
} else {
this.randomTip();
}
}
},
randomTip: function() {
this.currTip = Math.floor(Math.random()*this.tips.length);
var thisTip = this.tips[this.currTip];
this.showTip(thisTip);
},
nextPrevTip: function(idx) {
if (typeof(this.currTip) == 'undefined') this.currTip = 0;
this.currTip += idx;
if (this.currTip < 0) {
this.currTip = this.tips.length;
} else if (this.currTip >= this.tips.length) {
this.currTip = 0;
}
var thisTip = this.tips[this.currTip];
this.showTip(thisTip);
},
tips: Array(
"Don't forget to subscribe to <a href=\"http://reddit.com/r/Enhancement\">/r/Enhancement</a> to keep up to date on the latest versions of RES, report bugs, or suggest features!",
"Don't want to see posts containing certain keywords? Want to filter out certain subreddits from /r/all? Try the filteReddit module!",
"RES checks for new versions every 24 hours, but if you want to check manually, open up the RES console and click [check for update]",
"Did you know you can configure the appearance of a number of things in RES? For example: Keyboard navigation lets you configure the look of the 'selected' box, and commentBoxes lets you configure the borders / shadows.",
"Do you subscribe to a ton of reddits? Give the subreddit tagger a try, it makes your homepage a bit more readable.",
"If you haven't tried it yet, Keyboard Navigation is great. Just hit ? while browsing for instructions.",
"Click the (_) next to a user to tag that user with any name you like - you can also color code the tag.",
"Roll over a user's name to get information about them such as their karma, and how long they've been a reddit user.",
"Hover over the 'parent' link in comments pages to see the text of the parent being referred to.",
"You can configure the color and style of MrDerk's User Highlighter module if you want to change how the highlights look.",
"Not a fan of how comments pages look? You can change the appearance in the Style Tweaks module",
"Sick of seeing these tips? They only show up once every 24 hours, but you can disable that in the RES Tips and Tricks preferences.",
"Don't like the night/day icon for switching between light/dark styles? Just disable the 'lightSwitch' option under Style Tweaks",
"See that little [vw] next to users you've voted on? That's their vote weight - it moves up and down as you vote the same user up / down."
),
showTip: function(message) {
this.toolTipText.innerHTML = message;
RESUtils.fadeElementIn(this.toolTip, 0.2);
}
};
modules['filteReddit'] = {
moduleID: 'filteReddit',
moduleName: 'filteReddit',
options: {
// any configurable options you have go here...
// options must have a type and a value..
// valid types are: text, boolean (if boolean, value must be true or false)
// for example:
NSFWfilter: {
type: 'boolean',
value: false,
description: 'Filters all links labelled NSFW'
},
keywords: {
type: 'table',
fields: [
{ name: 'keyword', type: 'text' }
],
value: [
],
description: 'Type in title keywords you want to ignore if they show up in a title'
},
subreddits: {
type: 'table',
fields: [
{ name: 'subreddit', type: 'text' }
],
value: [
],
description: 'Type in a subreddit you want to ignore (only applies to /r/all)'
},
domains: {
type: 'table',
fields: [
{ name: 'domain', type: 'text' }
],
value: [
],
description: 'Type in domain keywords you want to ignore. Note that "reddit" would ignore "reddit.com" and "fooredditbar.com"'
}
},
description: 'Filter out NSFW content, or links by keyword, domain (use User Tagger to ignore by user) or subreddit (for /r/all).',
isEnabled: function() {
return RESConsole.getModulePrefs(this.moduleID);
},
include: new Array(
/http:\/\/www.reddit.com\/?/i,
/http:\/\/www.reddit.com\/r\/[\w]+\/?$/i
),
isMatchURL: function() {
return RESUtils.isMatchURL(this.moduleID);
},
go: function() {
if ((this.isEnabled()) && (this.isMatchURL())) {
// get this module's options...
RESUtils.getOptions(this.moduleID);
// do stuff now!
// this is where your code goes...
if (this.options.NSFWfilter.value) {
this.filterNSFW();
}
document.body.addEventListener('DOMNodeInserted', function(event) {
if ((event.target.tagName == 'DIV') && (event.target.getAttribute('id') == 'siteTable')) {
modules['filteReddit'].scanEntries(event.target);
}
}, true);
this.scanEntries();
}
},
scanEntries: function(ele) {
if (ele == null) {
var entries = document.querySelectorAll('#siteTable div.thing.link');
} else {
var entries = ele.querySelectorAll('div.thing.link');
}
var RALLre = /\/r\/all\/?$/i;
var onRALL = RALLre.exec(location.href);
for (var i=0, len=entries.length; i<len;i++) {
var postTitle = entries[i].querySelector('.entry a.title').innerHTML;
var postDomain = entries[i].querySelector('.entry span.domain > a').innerHTML;
var thisSubreddit = entries[i].querySelector('.entry a.subreddit');
if (thisSubreddit != null) {
var postSubreddit = thisSubreddit.innerHTML;
} else {
var postSubreddit = false;
}
var filtered = false;
filtered = this.filterTitle(postTitle);
if (!filtered) filtered = this.filterDomain(postDomain);
if ((!filtered) && (onRALL) && (postSubreddit)) {
filtered = this.filterSubreddit(postSubreddit);
}
if (filtered) {
// console.log('filtered this entry: ' + postTitle);
entries[i].style.display = 'none';
}
}
},
filterNSFW: function() {
RESUtils.addCSS('.over18 { display: none; !important }');
//GM_addStyle(css);
},
filterTitle: function(title) {
return this.arrayContainsSubstring(this.options.keywords.value, title.toLowerCase());
},
filterDomain: function(domain) {
return this.arrayContainsSubstring(this.options.domains.value, domain.toLowerCase());
},
filterSubreddit: function(subreddit) {
return this.arrayContainsSubstring(this.options.subreddits.value, subreddit.toLowerCase());
},
arrayContainsSubstring: function(obj, substring) {
var i = obj.length;
while (i--) {
if ((obj[i] != null) && (substring.indexOf(obj[i].toString().toLowerCase()) != -1)) {
return true;
}
}
return false;
}
};
modules['newCommentCount'] = {
moduleID: 'newCommentCount',
moduleName: 'New Comment Count',
options: {
// any configurable options you have go here...
// options must have a type and a value..
// valid types are: text, boolean (if boolean, value must be true or false)
// for example:
cleanComments: {
type: 'text',
value: 7,
description: 'Clean out cached comment counts of pages you haven\t visited in [x] days - enter a number here only!'
}
},
description: 'Shows how many new comments there are since your last visit.',
isEnabled: function() {
return RESConsole.getModulePrefs(this.moduleID);
},
include: Array(
/http:\/\/www.reddit.com\/?/i,
/http:\/\/www.reddit.com\/r\/[\w]+\/?$/i,
/http:\/\/www.reddit.com\/[-\w\.\/]+\/comments\/[-\w\.]+/i
),
isMatchURL: function() {
return RESUtils.isMatchURL(this.moduleID);
},
go: function() {
if ((this.isEnabled()) && (this.isMatchURL())) {
// get this module's options...
RESUtils.getOptions(this.moduleID);
// go!
var counts = localStorage.getItem('RESmodules.newCommentCount.counts');
if (counts == null) {
counts = '{}';
}
this.commentCounts = JSON.parse(counts);
if (RESUtils.pageType() == 'comments') {
this.updateCommentCount();
document.body.addEventListener('DOMNodeInserted', function(event) {
if ((event.target.tagName == 'DIV') && (hasClass(event.target,'thing'))) {
modules['newCommentCount'].updateCommentCount();
}
}, true);
} else {
this.processCommentCounts();
}
RESUtils.addCSS('.newComments { display: inline; color: #aaaadd; }');
}
},
processCommentCounts: function() {
var lastClean = localStorage.getItem('RESmodules.newCommentCount.lastClean');
var now = new Date();
if (lastClean == null) {
localStorage.setItem('RESmodules.newCommentCount.lastClean', now.getTime());
}
//if ((now.getTime() - lastClean) > 86400000) {
if ((now.getTime() - lastClean) > 5) {
this.cleanCache();
}
var IDre = /\/r\/[\w]+\/comments\/([\w]+)\//i;
var commentsLinks = document.querySelectorAll('#siteTable div.thing.link a.comments');
for (var i=0, len=commentsLinks.length; i<len;i++) {
var href = commentsLinks[i].getAttribute('href');
var thisCount = commentsLinks[i].innerHTML;
var split = thisCount.split(' ');
thisCount = split[0];
var matches = IDre.exec(href);
if (matches) {
var thisID = matches[1];
if (typeof(this.commentCounts[thisID]) != 'undefined') {
var diff = thisCount - this.commentCounts[thisID].count;
if (diff > 0) {
commentsLinks[i].innerHTML += ' ('+diff+' new)';
}
}
}
}
},
updateCommentCount: function() {
var IDre = /\/r\/[\w]+\/comments\/([\w]+)\//i;
var matches = IDre.exec(location.href);
if (matches) {
if (!this.currentCommentCount) {
this.currentCommentID = matches[1];
var thisCountText = document.querySelector('#siteTable a.comments').innerHTML;
var split = thisCountText.split(' ');
this.currentCommentCount = split[0];
} else {
this.currentCommentCount++;
}
}
var now = new Date();
if (typeof(this.commentCounts) == 'undefined') {
this.commentCounts = {};
}
this.commentCounts[this.currentCommentID] = {
count: this.currentCommentCount,
updateTime: now.getTime()
}
localStorage.setItem('RESmodules.newCommentCount.counts', JSON.stringify(this.commentCounts));
},
cleanCache: function() {
var now = new Date();
for(i in this.commentCounts) {
if ((now.getTime() - this.commentCounts[i].updateTime) > (86400000 * this.options.cleanComments.value)) {
this.commentCounts[i] = null;
}
}
localStorage.setItem('RESmodules.newCommentCount.lastClean', now.getTime());
}
};
/* eh, not for now...
modules['tweetit'] = {
moduleID: 'tweetit',
moduleName: 'tweetit',
options: {
// any configurable options you have go here...
// options must have a type and a value..
// valid types are: text, boolean (if boolean, value must be true or false)
// for example:
defaultMessage: {
type: 'text',
value: 'this is default text',
description: 'explanation of what this option is for'
},
doSpecialStuff: {
type: 'boolean',
value: false,
description: 'explanation of what this option is for'
}
},
description: 'Adds twitter functionality to let you tweet comments and posts.',
isEnabled: function() {
return RESConsole.getModulePrefs(this.moduleID);
},
include: Array(
// NEED TO FIX * / when uncommenting
// /http:\/\/www.reddit.com\/[-\w\.\/]* /i,
// /https:\/\/www.reddit.com\/[-\w\.\/]* /i
),
isMatchURL: function() {
return RESUtils.isMatchURL(this.moduleID);
},
go: function() {
if ((this.isEnabled()) && (this.isMatchURL())) {
// get this module's options...
RESUtils.getOptions(this.moduleID);
// do stuff now!
// this is where your code goes...
var css = '#tweetBoxContainer { position: absolute; width: 520px; height: 170px; left: 0px; top: 0px; z-index: 1010; background-color: #FAFAFA; padding: 4px; }';
css += '#tweetBoxContainerClose { float: right; width: 16px; height: 16px; background-position: 0px -120px; background-image: url("http://thumbs.reddit.com/t5_2s10b_0.png"); cursor: pointer; } #tweetBox { z-index: 1011; }';
css += '.tweetButton { cursor: pointer; }';
GM_addStyle(css);
this.injectAnywhere();
// TODO: this will need to be adjusted for comments vs. links pages...
var allTaglines = document.body.querySelectorAll('.tagline');
for (var i=0; i<allTaglines.length; i++) {
var tweetButton = document.createElement('span');
addClass(tweetButton, 'tweetButton');
tweetButton.innerHTML = '[tweet]';
tweetButton.setAttribute('title', 'tweet this!');
tweetButton.addEventListener('click', function(e) {
thisXY=RESUtils.getXYpos(this);
// modules['tweetit'].tweetBoxContainer.setAttribute('style', 'display: block; top: ' + thisXY.y + 'px; left: ' + thisXY.x + 'px;');
var commentText = ''; // username: will go here.
commentText += ' ' + RESUtils.stripHTML(this.parentNode.parentNode.querySelector('div.md').innerHTML);
modules['tweetit'].updateTweetBox(commentText);
modules['tweetit'].tweetBox.style.display = 'block';
// modules['tweetit'].tweetBoxContainer.style.display = 'block';
RESUtils.fadeElementIn(modules['tweetit'].tweetBoxContainer, 0.3);
modules['tweetit'].tweetBoxContainer.style.top = thisXY.y + 'px';
modules['tweetit'].tweetBoxContainer.style.left = thisXY.x + 'px';
}, false);
allTaglines[i].appendChild(tweetButton);
}
}
},
updateTweetBox: function(defaultText) {
defaultText = defaultText.replace(/"/g,"&quot;").replace(/\n/g," ");
this.tweetBox.innerHTML = '';
var tweetitScript = document.createElement('script');
tweetitScript.setAttribute('type','text/javascript');
tweetitScript.innerHTML = 'twttr.anywhere(function (T) { T("#tweetBox").tweetBox({defaultContent: "'+defaultText+'"}); })';
this.head.appendChild(tweetitScript);
},
injectAnywhere: function() {
this.head = document.getElementsByTagName("head")[0];
this.TwitterAnywhereAPIKEY = 'QVX2iC3kQIAvHfX4Wsjhw';
var tAnywhereAPI = document.createElement('script');
tAnywhereAPI.setAttribute('src','http://platform.twitter.com/anywhere.js?id='+this.TwitterAnywhereAPIKEY+'&v=1');
this.head.appendChild(tAnywhereAPI);
this.tweetBoxContainer = createElementWithID('div','tweetBoxContainer');
this.tweetBoxContainer.style.display = 'none';
this.tweetBoxContainerCloseButton = createElementWithID('div','tweetBoxContainerClose');
this.tweetBoxContainerCloseButton.addEventListener('click',function(e) {
RESUtils.fadeElementOut(modules['tweetit'].tweetBoxContainer, 0.3);
}, false);
this.tweetBoxContainer.appendChild(this.tweetBoxContainerCloseButton);
this.tweetBox = createElementWithID('div','tweetBox');
this.tweetBoxContainer.appendChild(this.tweetBox);
document.body.appendChild(this.tweetBoxContainer);
}
};
*/
/*
* Konami-JS ~
* :: Now with support for touch events and multiple instances for
* :: those situations that call for multiple easter eggs!
* Code: http://konami-js.googlecode.com/
* Examples: http://www.snaptortoise.com/konami-js
* Copyright (c) 2009 George Mandis (georgemandis.com, snaptortoise.com)
* Version: 1.3.2 (7/02/2010)
* Licensed under the GNU General Public License v3
* http://www.gnu.org/copyleft/gpl.html
* Tested in: Safari 4+, Google Chrome 4+, Firefox 3+, IE7+ and Mobile Safari 2.2.1
*/
var Konami = function() {
var konami= {
addEvent:function ( obj, type, fn, ref_obj )
{
if (obj.addEventListener)
obj.addEventListener( type, fn, false );
else if (obj.attachEvent)
{
// IE
obj["e"+type+fn] = fn;
obj[type+fn] = function() { obj["e"+type+fn]( window.event,ref_obj ); }
obj.attachEvent( "on"+type, obj[type+fn] );
}
},
input:"",
pattern:"3838404037393739666513",
load: function(link) {
this.addEvent(document,"keydown", function(e,ref_obj) {
if (ref_obj) konami = ref_obj; // IE
konami.input+= e ? e.keyCode : event.keyCode;
if (konami.input.length > konami.pattern.length) konami.input = konami.input.substr((konami.input.length - konami.pattern.length));
if (konami.input == konami.pattern) {
konami.code(link);
konami.input="";
return;
}
},this);
this.iphone.load(link)
},
code: function(link) { window.location=link},
iphone:{
start_x:0,
start_y:0,
stop_x:0,
stop_y:0,
tap:false,
capture:false,
orig_keys:"",
keys:["UP","UP","DOWN","DOWN","LEFT","RIGHT","LEFT","RIGHT","TAP","TAP","TAP"],
code: function(link) { konami.code(link);},
load: function(link){
orig_keys = this.keys;
konami.addEvent(document,"touchmove",function(e){
if(e.touches.length == 1 && konami.iphone.capture==true){
var touch = e.touches[0];
konami.iphone.stop_x = touch.pageX;
konami.iphone.stop_y = touch.pageY;
konami.iphone.tap = false;
konami.iphone.capture=false;
konami.iphone.check_direction();
}
});
konami.addEvent(document,"touchend",function(evt){
if (konami.iphone.tap==true) konami.iphone.check_direction(link);
},false);
konami.addEvent(document,"touchstart", function(evt){
konami.iphone.start_x = evt.changedTouches[0].pageX
konami.iphone.start_y = evt.changedTouches[0].pageY
konami.iphone.tap = true
konami.iphone.capture = true
});
},
check_direction: function(link){
x_magnitude = Math.abs(this.start_x-this.stop_x)
y_magnitude = Math.abs(this.start_y-this.stop_y)
x = ((this.start_x-this.stop_x) < 0) ? "RIGHT" : "LEFT";
y = ((this.start_y-this.stop_y) < 0) ? "DOWN" : "UP";
result = (x_magnitude > y_magnitude) ? x : y;
result = (this.tap==true) ? "TAP" : result;
if (result==this.keys[0]) this.keys = this.keys.slice(1,this.keys.length)
if (this.keys.length==0) {
this.keys=this.orig_keys;
this.code(link)
}
}
}
}
return konami;
};
(function(u) {
// Don't fire the script on the iframe. This annoyingly fires this whole thing twice. Yuck.
// Also don't fire it on static.reddit or thumbs.reddit, as those are just images.
if ((location.href.match(/comscore-iframe/i)) || (location.href.match(/static\.reddit/i)) || (location.href.match(/thumbs\.reddit/i))) {
// do nothing.
return false;
}
// added this if statement because some people's Greasemonkey "include" lines are getting borked or ignored, so they're calling RES on non-reddit pages.
if (location.href.match(/reddit\.com/i)) {
RESUtils.checkForUpdate();
// add the config console link...
RESConsole.create();
RESConsole.addConsoleLink();
var userNameEle = document.querySelector('#header-bottom-right span a');
if ((typeof(userNameEle) != 'undefined') && (userNameEle != null)) {
RESUtils.userName = userNameEle.text;
}
RESUtils.checkIfSubmitting();
// go through each module and run it
for (i in modules) {
thisModuleID = i;
if (typeof(modules[thisModuleID]) == 'object') {
//console.log(thisModuleID + ' start: ' + Date());
//perfTest(thisModuleID+' start');
modules[thisModuleID].go();
//perfTest(thisModuleID+' end');
//console.log(thisModuleID + ' end: ' + Date());
}
}
/*
for (var i = 0; i < localStorage.length; i++){
console.log('key: ' + localStorage.key(i));
console.log('value: ' + localStorage.getItem(localStorage.key(i)));
}
*/
GM_addStyle(RESUtils.css);
// console.log('end: ' + Date());
}
if (location.href.match(/reddit.honestbleeps.com\/download/)) {
var installLinks = document.body.querySelectorAll('.install');
for (var i=0, len=installLinks.length;i<len;i++) {
addClass(installLinks[i], 'update');
removeClass(installLinks[i], 'install');
}
}
konami = new Konami();
konami.code = function() {
location.href = 'http://reddit.com/r/Enhancement/?I+AM+AWESOME!';
}
konami.load();
})();
var lastPerf = 0;
function perfTest(name) {
var d = new Date();
var diff = d.getTime() - lastPerf;
console.log(name+' executed. Diff since last: ' + diff +'ms');
lastPerf=d.getTime();
}