NOTICE: By continued use of this site you understand and agree to the binding Terms of Service and Privacy Policy.
// ==UserScript== // @name Fix BanniNation // @description fixes up various parts of the bn ui // @version 25 // @namespace http://www.bannination.com/fixbn // @include http://www.bannination.com/* // @include http://bannination.com/* // @grant GM_getValue // @grant GM_setValue // @grant GM_log // @grant GM_addStyle // @grant GM_getResourceText // @require https://ajax.aspnetcdn.com/ajax/jquery/jquery-1.11.0.min.js // @require https://ajax.aspnetcdn.com/ajax/jquery.migrate/jquery-migrate-1.2.1.min.js // @require https://ajax.aspnetcdn.com/ajax/jquery.ui/1.10.3/jquery-ui.min.js // @require https://raw.github.com/markitup/1.x/master/markitup/jquery.markitup.js // @require https://raw.github.com/sizzlemctwizzle/GM_config/master/gm_config.js // @require https://raw.github.com/medialize/URI.js/gh-pages/src/URI.min.js // @require https://raw.github.com/accursoft/caret/master/jquery.caret.js // @require https://raw.github.com/dimsemenov/Magnific-Popup/master/dist/jquery.magnific-popup.js // @require https://raw.github.com/needim/noty/master/js/noty/packaged/jquery.noty.packaged.min.js // @require https://raw.github.com/bgrins/spectrum/master/spectrum.js // @require https://raw.github.com/ksylvest/jquery-age/master/javascripts/jquery.age.js // @require https://raw.github.com/artificeren/jqSmartTag/master/site/script/jquery.smartTag.js // @require https://raw.github.com/silvestreh/onScreen/master/jquery.onscreen.js // @resource juipepper https://ajax.aspnetcdn.com/ajax/jquery.ui/1.8.10/themes/pepper-grinder/jquery-ui.css // @resource magnificcss https://raw.github.com/dimsemenov/Magnific-Popup/master/dist/magnific-popup.css // @resource spectrumcss https://raw.github.com/bgrins/spectrum/master/spectrum.css // ==/UserScript== /* jshint browser: true*/ /* global $, jQuery, URI, GM_config, GM_addStyle, GM_getResourceText, GM_configStruct, GM_getValue, GM_setValue, noty */ // main code var __fixbn = null; try { (function ($) { "use strict"; var FixbN; var $bind = function (fn, me) { return function () { return fn.apply(me, arguments); }; }; var $root = null; FixbN = (function () { // ctor function FixbN() { this.fix = $bind(this.fix, this); this.fixAllPages = $bind(this.fixAllPages, this); this.fixHeadlinesPages = $bind(this.fixHeadlinesPages, this); this.fixCommentsPage = $bind(this.fixCommentsPage, this); this.fixQueuePage = $bind(this.fixQueuePage, this); this.fixTaggers = $bind(this.fixTaggers, this); this.createDateUrl = $bind(this.createDateUrl, this); var mainConfigFrame = $("<div style='display:none;' />")[0]; document.body.appendChild(mainConfigFrame); GM_config.init({ 'id': 'FixbNconfig', 'title': 'Fix bN Global Settings', 'fields': { 'switchColumns': { 'label': 'Switch Headline and Comment Columns', 'type': 'checkbox', 'default': true }, 'hideModThreads': { 'label': 'Hide all Moderated Threads', 'type': 'checkbox', 'default': false }, 'fixedHeader': { 'label': "Keep Header at Top of Window", 'type': 'checkbox', 'default': true }, 'scrollArrow': { 'label': 'Show a Floating Arrow for Scrolling', 'type': 'checkbox', 'default': true }, 'stickyTagger': { 'label': "Show a Floating Tagger in Comment Threads", 'type': 'checkbox', 'default': true }, 'tagNotyDuration': { 'label': 'Tag Notification Duration', 'type': 'int', 'min': 3, 'max': 30, 'default': 6 }, 'repliesOveride': { 'label': "Override the masterbn replies notification", 'type': 'checkbox', 'default': true }, 'ignoreReplies': { 'label': "Ignoring a User Also Ignores All Replies To That User", 'type': 'checkbox', 'default': true }, 'showRepliesToMe': { 'label': "Show a List of Comments That Are Replies To Me", 'type': 'checkbox', 'default': true }, 'nabbitVisibility': { 'label': "Enable Nabbit User Images (Can Be Overridden Per-User)", 'type': 'checkbox', 'default': true }, 'highScoreThreshold': { 'label': "Mark Posts With Scores Over:", 'type': 'int', 'default': '4' }, 'autoLinkComments': { 'label': "Auto-linkify pasted URLs", 'type': 'checkbox', 'default': false } }, 'frame': mainConfigFrame }); GM_config.onSave = function () { if (GM_config.isOpen) { GM_config.close(); window.location.reload(); } }; GM_config.onOpen = function (doc, win, frame) { var notyDuration = $(frame).find("input[id$='tagNotyDuration']"); notyDuration.css("width", "3em"); $(frame).find("input[id$='highScoreThreshold']").css("width", "3em"); var wrapper = $(frame).find("div#FixbNconfig_wrapper"); var extras = $("<div style='float:right; text-align:right; font-weight: bold; margin-top:40px; margin-right:10px;'></div>"); wrapper.prepend(extras); $("<p><a style='font-size:14px; color:black;' href='http://www.bannination.com/comments/5182401/'>Bannination.com Feedback Thread</a></p>").appendTo(extras); $("<p><a style='font-size:14px; color:black;' href='https://github.com/artificeren/FixbN/issues?state=open'>Fix bN GitHub Project Feedback</a></p>").appendTo(extras); var bitCoinDonate = $('<div style="font-size:14px;margin:0 auto;width:300px" class="blockchain-btn" data-address="1Fe2d6giUvTMJniufMyivybg6v6HQGpeXG" data-shared="false"> <div class="blockchain stage-begin"><p style="text-decoration:underline;cursor:pointer;"> Donate Bitcoin to Fix bN </p></div> <div class="blockchain stage-loading" style="text-align:center"> <img src=""/> </div> <div class="blockchain stage-ready"> <p>Bitcoin Address: <span style="font-size:12px;">[[address]]</span></p> <p class="qr-code"></p> </div> <div class="blockchain stage-paid"> <p>Donation of <b>[[value]] BTC</b> Received. Thank You.</p> </div> <div class="blockchain stage-error"> <font color="red">[[error]]</font> </div> </div>'); bitCoinDonate.appendTo(extras).bitcoin(); $(frame).find("#FixbNconfig_saveBtn").text(" OK "); $(frame).find("#FixbNconfig_closeBtn").text("Cancel"); }; } FixbN.prototype.fix = function () { this.$root = $("html, body"); try { var firstMenuItem = $("div#menu ul:first li:first"); $("<li><span title='Fix bN Version {0}' style='text-decoration:underline;cursor:pointer;'>fix bN</span></li>".fex(GM_info.script.version)) .insertAfter(firstMenuItem) .click(function () { GM_config.open(); }) .attr("title", "Settings for the Fix bN addon"); firstMenuItem.after("\r\n\r\n"); } catch (ex) { console.error("FixbN failed adding options menu item", ex); } try { this.fixAllPages(); } catch (ex) { console.error("FixbN failed the allPages fixup", ex); } try { this.fixHeadlinesPages(); } catch (ex) { console.error("FixbN failed the Headlines page fixup", ex); } try { this.fixCommentsPage(); } catch (ex) { console.error("FixbN failed the Comments page fixup", ex); } try { this.fixQueuePage(); } catch (ex) { console.error("FixbN failed the Queue page fixup", ex); } try { this.fixTaggers(); } catch (ex) { console.error("FixbN failed the Tagging fixup", ex); } }; FixbN.prototype.fixAllPages = function () { // there was a bug where jsessionid caused problems, not sure if still needed if ($("#banner a").first().attr("href").indexOf("jsessionid") > 0) { document.location.reload(); } // optionally keep the header fixed to the top of the window if (GM_config.get("fixedHeader")) { var header = $("div#header"); header.addClass("fixedHeader"); $("div#main").addClass("fixedHeader"); var $style = $("<style type='text/css'>").appendTo('head'); var bindTopMargin = function() { $("body").css("margin-top", $("div#header").height()); var $style = $("<style type='text/css'>").appendTo('head'); var adjustment = $("div#header").height() + 10; $style.html("div#main.fixedHeader a:target { padding-top: " + adjustment + "px; margin-top: -" + adjustment + "px; }"); }; $(window).resize(bindTopMargin); bindTopMargin(); window.setTimeout(bindTopMargin, 200); } else { $("<style type='text/css'> div#main a:target { padding-top: 50px; margin-top: -50px; } </style>").appendTo('head'); } // scroll To Bottom/Top buttons if (GM_config.get("scrollArrow")) { var scrollToBottom = $("<div class='fbnScrollToBottom' style='width:64px; height:64px;'><img title='Scroll To Bottom' alt='Scroll To Bottom' src='' /></div>"); scrollToBottom .prependTo($("div#main")) .click((function () { this.$root.animate({ scrollTop: $(document).height() - $(window).height() }); }).bind(this)); var scrollToTop = $("<div class='fbnScrollToTop' style='display: none; width:64px; height:64px;'><img title='Scroll To Top' alt='Scroll To Top' src='' /></div>"); scrollToTop .prependTo($("div#main")) .click((function () { this.$root.animate({ scrollTop: 0 }); }).bind(this)); $("div#footer").onScreen({ doIn: function () { $("div.fbnScrollToBottom").hide(); $("div.fbnScrollToTop").show(); }, doOut: function () { $("div.fbnScrollToTop").hide(); $("div.fbnScrollToBottom").show(); } }); } // split the menu var leftMenu = $("<ul class='leftMenu'></ul>"); leftMenu .append($("div#menu a[href$='/queue']").closest("li")) .append("<li><a href='{0}'>today</a></li>".fex(this.createDateUrl(0))) .append("<li><a href='{0}'>yesterday</a></li>".fex(this.createDateUrl(-1))) .append("<li><a href='{0}' title='Damn You, Dictionary! This Is A Great Word!'>ereyesterday</a>".fex(this.createDateUrl(-2))) .append($("div#menu a[href^='http://wiki']").closest("li")) .append("<li><a href='/comments/1000' class='fbnBgLink " + $("div#welcome a[href='/comments/1000']").attr("class") + "'>beer garden</a></li>") .prependTo("div#menu"); if ((new Date()).getHours() < 8 && $("div#menu a:contains('dangerous mode')").length > 0) { leftMenu.append("<li class='nsfw'><a href='/comments/901' class='nsfw'>strip club</a></li>"); } // consolidate the profile links try { var profilePopup = $("<div class='profilePopup' style='min-width:200px;margin:0px;padding:10px;color:white;position:absolute;display:none;right:0px;background:#373737;z-index:1001;' />"); if (GM_config.get("fixedHeader")) { profilePopup.css("position", "fixed"); } profilePopup.appendTo("body"); var selfItem = $("div#menu li:contains('logged in as')"); if (selfItem.length !== 0) { var selfLink = selfItem.find("a"); selfLink.remove(); selfItem.empty().append(selfLink).append(" "); var originalLogoutLink = selfLink.closest("li").next("li").find("a"); originalLogoutLink.closest("li").hide(); var list = selfLink.closest("ul"); originalLogoutLink.appendTo(profilePopup); var profileLink = selfLink.clone(); profileLink.text("profile"); profilePopup.prepend(profileLink); profileLink.css("float", "right").addClass("topLink"); originalLogoutLink.css("float", "left").addClass("topLink"); var recentCommentWrapper = $("<div class='recentCommentWrapper' style='clear:both;margin-top:5px;' />"); recentCommentWrapper.appendTo(profilePopup); var loadComments = function () { var userPageUrl = profileLink.attr("href"); $.ajax({ url: userPageUrl, dataType: "text", success: function (data, textStatus, jqXHR) { var comments = data.substring(data.indexOf("<h2>Recent Comments</h2>")); comments = comments.substring(0, comments.indexOf("<h2>Account Settings</h2>") - 1); recentCommentWrapper.css("overflow-y", "scroll").css("height", $(window).height() - 200); recentCommentWrapper.html(comments); profilePopup.css("width", $("div#menu").width() / 1.5).position({ my: "right top", at: "right bottom", of: $("div#menu"), collision: "none" }); recentCommentWrapper.find("h2").css({ "cursor": "pointer" }).click(loadComments); }, error: function () { window.location.href = userPageUrl; } }); }; selfLink.click(function (event) { event.preventDefault(); try { profilePopup.toggle(); if (profilePopup.is(":visible")) { profilePopup.position({ my: "right top", at: "right bottom", of: $("div#menu"), collision: "fit" }); loadComments(); } } catch (ex) { console.error(ex); } }); } } catch (ex) { console.error("FixbN Failed consolidating profile links", ex); } // fix the masterbn replies box if (GM_config.get("repliesOveride")) { $("div#replies").hide().data("replies", $("div#replies").html()); var replyNoty = null; window.setInterval(function () { var mbnReplies = null; try { var replies = $("div#replies"); if (replies.html() === replies.data("replies")) { return; } replies.data("replies", replies.html()); mbnReplies = []; var mbnRepliesConfigsPromises = []; replies.contents().filter(function () { return this.nodeType === 3; //Node.TEXT_NODE }).each(function () { try { var words = this; var rplyUsername = words.nodeValue.trim(); if (rplyUsername.indexOf("has replied to your comment in thread") > 0) { rplyUsername = rplyUsername.substring(0, rplyUsername.indexOf("has replied to your comment in thread")).trim(); var link = $(words.nextSibling); mbnReplies.push( { username: rplyUsername, url: link.attr("href"), threadTitle: link.text() }); mbnRepliesConfigsPromises.push(__userConfig.getConfigPromise(rplyUsername, null)); } } catch (ex) { console.error("FixbN Failed creating masterbn reply info", ex); } }); $.when.apply(this, mbnRepliesConfigsPromises).then(function () { try { var mbnConfigs = {}; for (var x = 0; x < arguments.length; x++) { var replierConfig = arguments[x]; mbnConfigs[replierConfig.bnUsername] = replierConfig; replierConfig.onSave.callbacks.add(function () { $("div#replies").data("replies", "force refresh"); }); } var mbnRepliesHtml = []; for (var i = 0; i < mbnReplies.length; i++) { var replyInfo = mbnReplies[i]; if (mbnConfigs[replyInfo.username].get("visibility") !== "Ignore") { mbnRepliesHtml.push("<span class='notyReply'><a href='{url}' title='{threadTitle}'>{username}</a></span>".fex(replyInfo)); } } if (mbnRepliesHtml.length > 0) { if (replyNoty === null) { replyNoty = new noty({ layout: 'topCenter', text: mbnRepliesHtml.join(" <span class='notyReplyDivider'> | </span> "), type: "information", timeout: false, force: true, callback: { onClose: function (theNoty) { replyNoty = null; $.ajax("/post?action=clear_notifcations", { dataType: "text", global: false }); } } }); } else { replyNoty.text = mbnRepliesHtml.join(" <span class='notyReplyDivider'> | </span> "); } } } catch (ex) { console.error("FixbN Failed creating noty for masterbn replies", ex); } }); } catch (ex) { console.error("FixbN Failed masterbn reply override", ex); } }, 500); } }; FixbN.prototype.fixHeadlinesPages = function () { if (!bnurl.isHeadlinesPage()) { return; } // fuck the welcome $("div#welcome").css("display", "none"); //hide moderated threads if (GM_config.get("hideModThreads")) { $("div.moder").css("display", "none"); } // fix bydate stories if (bnurl.isHeadlinesByDate()) { var stories = $("table#stories tr"); if (stories.length) { // the bydate table has an incorrect number of columns in the header stories[0].removeChild(stories[0].cells[1]); // move headline after the discuss link if (GM_config.get("switchColumns")) { stories.each(function () { $(this).append(this.cells[0]); }); } } // stories should show nsfw tags $("table#stories tr.nsfw a.storylink").css({ "padding-right": "85px", "background": "url('http://www.bannination.com/img/nsfw.jpg') no-repeat scroll right center transparent" }); } // fix by-score headlines if (bnurl.isHeadlinesByScore()) { // fix up the tag cloud $("div#tag_cloud").css("width", "90%"); if (GM_config.get("fixedHeader")) { $("div#tag_cloud span").click((function (evt) { var c = "tag_" + $(evt.target).text().replace(/[^a-z]/g, ""); window.setTimeout(function () { try { this.$root.animate({ scrollTop: $("." + c + ":first").offset().top - $("div#header").height() - 10 }, 50); } catch (ex) { console.error("Fix bN Failed correcting tag scroll", ex); } }.bind(this), 600); }).bind(this)); } // move the welcome title to the top of the tag cloud var bnTitleTagline = $("div#welcome p").first().text(); $("div#tag_cloud center").first().text(bnTitleTagline); if ($("div#tag_cloud center").last().text() === "[tag cloud hidden]") { $("div#tag_cloud").html("<center>" + bnTitleTagline + "</center>"); } // queue link can get extra info from the main page welcome var queueEntries = $("div#welcome a[href='/queue']").text().replace(/[^\d\.]/g, ''); if (queueEntries && queueEntries !== "") { $("div#menu a[href$='/queue']").append(" (" + queueEntries + ")"); } // move headline after the discuss link if (GM_config.get("switchColumns")) { $("div.storytable_row").each(function () { var storyRow = $(this); storyRow.append(storyRow.children("div.headline_cell")); }); } } }; FixbN.prototype.fixCommentsPage = function () { if (!bnurl.isCommentsPage()) { return; } var threadId = $("h1").first().attr("id").substring(1); if (!threadId) { return; } // add calendar to beer garden var beerGardenHeader = $("h1#s1000"); if (beerGardenHeader.length > 0) { var bgDatePicker = $("<input type='hidden' id='bgDatePicker' />"); bgDatePicker.prependTo("h1#s1000").datepicker({ showOn: "button", buttonImage: "https://cdn1.iconfinder.com/data/icons/Pretty_office_icon_part_2/16/event.png", buttonImageOnly: true, changeMonth: true, changeYear: true, showAnim: 'fold', onSelect: function (dateText, cal) { var selectedDate = new Date(dateText); window.location.href = "/date/{year}/{month}/{day}/1000".fex({ year: selectedDate.getFullYear(), month: selectedDate.getMonth() + 1, day: selectedDate.getDate() }); } }); } // store the userid and username and score in each header data $("div.ch").each(function () { var header = $(this); header.data("uname", header.find("span.ui").text()); var uidSpan = header.find("span.uid"); if (uidSpan.length === 0) { header.data("uid", 0); } else { header.data("uid", uidSpan.text()); } var score = header.find("span.score span").text(); header.data("score", score); }); // clear out tag instructions var tagCloud = $("td.current_tags"); if (tagCloud.text().indexOf("add relevant short tags using the box to the left") > 0) { tagCloud.empty(); } if (tagCloud.text().indexOf("[tag cloud hidden]") > 0) { $("div#tag_story").css("display", "none"); } // more clickable refresh link $("a.commentslink") .last() .clone() .empty() .removeClass("commentslink") .addClass("commentsLinkButton") .append("<img title='check for more comments' src='' />") .prependTo("div#comment_form form"); // score/sunlight $.fn.sunlight.defaults.handler = function (sunlight) { this.attr("title", sunlight.blame).text("score (" + sunlight.cool + "|" + sunlight.uncool + ")"); this.closest("div.ch").data("score", parseInt(sunlight.cool) - parseInt(sunlight.uncool)); this.closest("div.ch").find("span.uid").userDecoration("update"); }; $.fn.sunlight.defaults.threadId = threadId; // auto-load my own comment sunlight $("div.uc span.score").css("cursor", "pointer").sunlight(); // click to load/refresh all comment sunlight $("div.ch span.score").css("cursor", "pointer").sunlight({ event: 'click' }); // nabbit and pretty dates $.fn.nabbit.defaults.userId = function () { return this.closest("div.ch").data("uid"); }; $.fn.nabbit.defaults.username = function () { return this.closest("div.ch").data("uname"); }; $("div.ch span.uid").nabbit({ imgSize: 'small', imgClass: 'nabbitSmall', imgTitle: $.fn.nabbit.defaults.userId }); // pretty dates var timeSpans = $("div.ch span.time"); timeSpans.each(function () { var bigSpot = $(this); var postTime = bigSpot.text(); var cleanTime = (postTime || "").replace(/-/g, "/").replace(/\.0/g, " PST"); // HACK! bigSpot.attr("datetime", cleanTime); bigSpot.attr("title", postTime); bigSpot.age({ suffixes: { past: "ago", future: "from now" } }); }); // nabbit big images timeSpans.nabbit({ imgSize: 'large', imgClass: 'nabbitBig', imgTitle: function () { return this.text(); }, loaded: function (img) { this.next("span.time").remove(); this.after($("<span class='time'></span>").append(img)); } }); // anon $("div.ch.u0").find("span.ui").html(function (index, text) { return "~ " + text.substring(30); }).after("<span class='uid'><img title='settings' src='https://cdn1.iconfinder.com/data/icons/hamburg/32/settings.png' style='height:1em; width:1em;' /></span>"); // add out-of-page quotes to the style /* jshint -W064 */ GM_addStyle("div.cb a[href^='/comments/{0}'] { color:black; }".fex(threadId)); /* jshint +W064 */ // add user decoration to userid span $("div.ch span.uid").userDecoration({ 'threadId': threadId, onImageReplaced: function () { $(this).magnificPopup({ type: 'image', verticalFit: true, closeOnContentClick: true, showCloseBtn: false }); } }); // html comment editor $('textarea').markItUp({ /* jshint ignore:start */ resizeHandle: false, markupSet: [ { name: 'Bold', key: 'B', openWith: '<b>', closeWith: '</b>', className: 'miuBold' }, { name: 'Italic', key: 'I', openWith: '<i>', closeWith: '</i>', className: 'miuItalic' }, { name: 'Stroke through', key: 'S', openWith: '<strike>', closeWith: '</strike>', className: 'miuStrike' }, { name: 'Colors', className: 'palette', dropMenu: [ { name: 'Yellow', openWith: '<font color="#FCE94F">', closeWith: '</font>', className: "col1-1" }, { name: 'Yellow', openWith: '<font color="#EDD400">', closeWith: '</font>', className: "col1-2" }, { name: 'Yellow', openWith: '<font color="#C4A000">', closeWith: '</font>', className: "col1-3" }, { name: 'Orange', openWith: '<font color="#FCAF3E">', closeWith: '</font>', className: "col2-1" }, { name: 'Orange', openWith: '<font color="#F57900">', closeWith: '</font>', className: "col2-2" }, { name: 'Orange', openWith: '<font color="#CE5C00">', closeWith: '</font>', className: "col2-3" }, { name: 'Brown', openWith: '<font color="#E9B96E">', closeWith: '</font>', className: "col3-1" }, { name: 'Brown', openWith: '<font color="#C17D11">', closeWith: '</font>', className: "col3-2" }, { name: 'Brown', openWith: '<font color="#8F5902">', closeWith: '</font>', className: "col3-3" }, { name: 'Green', openWith: '<font color="#8AE234">', closeWith: '</font>', className: "col4-1" }, { name: 'Green', openWith: '<font color="#73D216">', closeWith: '</font>', className: "col4-2" }, { name: 'Green', openWith: '<font color="#4E9A06">', closeWith: '</font>', className: "col4-3" }, { name: 'Blue', openWith: '<font color="#729FCF">', closeWith: '</font>', className: "col5-1" }, { name: 'Blue', openWith: '<font color="#3465A4">', closeWith: '</font>', className: "col5-2" }, { name: 'Blue', openWith: '<font color="#204A87">', closeWith: '</font>', className: "col5-3" }, { name: 'Purple', openWith: '<font color="#AD7FA8">', closeWith: '</font>', className: "col6-1" }, { name: 'Purple', openWith: '<font color="#75507B">', closeWith: '</font>', className: "col6-2" }, { name: 'Purple', openWith: '<font color="#5C3566">', closeWith: '</font>', className: "col6-3" }, { name: 'Red', openWith: '<font color="#EF2929">', closeWith: '</font>', className: "col7-1" }, { name: 'Red', openWith: '<font color="#CC0000">', closeWith: '</font>', className: "col7-2" }, { name: 'Red', openWith: '<font color="#A40000">', closeWith: '</font>', className: "col7-3" }, { name: 'Gray', openWith: '<font color="#FFFFFF">', closeWith: '</font>', className: "col8-1" }, { name: 'Gray', openWith: '<font color="#D3D7CF">', closeWith: '</font>', className: "col8-2" }, { name: 'Gray', openWith: '<font color="#BABDB">', closeWith: '</font>6', className: "col8-3" }, { name: 'Gray', openWith: '<font color="#888A85">', closeWith: '</font>', className: "col9-1" }, { name: 'Gray', openWith: '<font color="#555753">', closeWith: '</font>', className: "col9-2" }, { name: 'Gray', openWith: '<font color="#000000">', closeWith: '</font>', className: "col9-3" } ] }, { name: 'Pre', className: 'miuPre', openWith: '<pre>', closeWith: '</pre>' }, { separator: '---------------' }, { name: 'Ul', openWith: '<ul>\n', closeWith: '</ul>\n', className: 'miuUList' }, { name: 'Ol', openWith: '<ol>\n', closeWith: '</ol>\n', className: 'miuOList' }, { name: 'Li', openWith: '<li>', closeWith: '</li>', className: 'miuListitem' }, { separator: '---------------' }, { name: 'Picture', key: 'P', replaceWith: '<img src="[![Source:!:http://]!]" />', className: 'miuImage' }, { name: 'Link', key: 'L', openWith: '<a href="[![Link:!:http://]!]"(!( title="[![Title]!]")!)>', closeWith: '</a>', placeHolder: 'Your text to link...', className: 'miuLink' }, { separator: '---------------' }, { name: 'Clean', replaceWith: function (markitup) { return markitup.selection.replace(/<(.*?)>/g, ""); }, className: 'miuClean' }, { name: 'Symbols', className: 'miuSymbols', dropMenu: [ { name: 'Tags', openWith: '<', closeWith: '>' }, { name: '&', openWith: '&' } ] } ] /* jshint ignore:end */ }); // intercept comment pastes $('textarea').each(function(){ $(this).data("oldVal", $(this).val()).data("editing", false); }).bind('input propertychange', function (evt) { if (!GM_config.get("autoLinkComments")) { return; } try { var commentBox = $(this); var oldVal = commentBox.data("oldVal"); var newVal = commentBox.val(); commentBox.data("oldVal", newVal); if (commentBox.data("editing")) { console.log("leaving: editing"); return; } if (newVal.length > oldVal.length) { commentBox.data("editing", true); try { var change = diff(oldVal, newVal); if (oldVal.indexOf(change) > 0 || change.length <= 1) { return; } var url = new URI(change); if (url.is("url") && url.protocol().indexOf("http") === 0) { var tag = ""; switch (url.suffix()) { case "gif": case "jpg": case "jpeg": case "png": tag = "<img src='{0}' />".fex(change); break; default: tag = "<a href='{0}'>{0}</a>".fex(change); break; } var parts = newVal.split(change); var result = parts[0] + tag + parts[1]; commentBox.val(result); } } catch (ex) { console.error(ex); } finally { commentBox.data("editing", false); } } } catch (ex) { console.log("FixbN Failed intercepting comment changes", ex); } }); // any selective reply child quotes have font tags around them, let's style them /* jshint -W014 */ $("div.cb font").filter(function () { var quoteChildren = $(this).contents(); return ( (quoteChildren.length === 3 && quoteChildren[0].tagName === "A" && quoteChildren[2].tagName === "I") || (quoteChildren.length === 2 && quoteChildren[0].tagName === "A" && quoteChildren[1].tagName === "I") ); }).addClass("commentquote"); /* jshint +W014 */ // any selective reply images have been converted to links, let's style them $("div.cb a").filter(function () { var text = $(this).text(); return text === "[quoted image removed]" || text === "image removed"; }).magnificPopup({ type: 'image', verticalFit: true, closeOnContentClick: true, showCloseBtn: false }); // comment warning var commentForm = $("div#comment_form form"); var warning = $("div#comment_warning"); warning.css('display', 'none'); if (warning.text().length > 0) { if (warning.text().indexOf("Not Safe") > 0) { commentForm.append("<h2 style='display:inline;color:red;'>Not Safe For Work Allowed</h2>"); } else { commentForm.append("<h2 style='display:inline;color:green;'>Keep Thread Safe For Work</h2>"); } var modindex = warning.text().indexOf("thread."); //Add moderated thread text back in *wushupork 05/03/2014 if (modindex > 0) { var modtext = warning.text().substr((modindex+7)); commentForm.append("<br /><br />This is a moderated thread. "+modtext); } commentForm.css('width:90%;'); } // comment preview $("div#comment_form form :submit[value='preview']").click(function (event) { event.preventDefault(); if ($("textarea#comment").val().trim() === "") { return; } var values = $("div#comment_form form").serializeArray(); values.push({ name: 'button', value: 'preview' }); $.ajax({ url: $("div#comment_form form").attr("action"), type: "POST", dataType: "text", data: values, success: function (data) { var commentHtml = data.substring(data.indexOf('<div class="cb" id="c0">')); commentHtml = commentHtml.substring(0, commentHtml.indexOf('<div id="comment_form">')); var previewComment = $("<div class='ch uc'><span class='ui'>Comment Preview</span></div>" + $.trim(commentHtml)); previewComment.find("font").filter(function () { var quoteChildren = $(this).contents(); /* jshint -W014 */ return ( (quoteChildren.length === 3 && quoteChildren[0].tagName === "A" && quoteChildren[2].tagName === "I") || (quoteChildren.length === 2 && quoteChildren[0].tagName === "A" && quoteChildren[1].tagName === "I") ); /* jshint +W014 */ }).addClass("commentquote"); $.magnificPopup.open({ showCloseBtn: false, items: { src: previewComment, type: 'inline' } }); } }); }); // "replies to me" that I can see if (GM_config.get("showRepliesToMe")) { var myUserName = $("div#comment_form form input#username").val(); if (myUserName) { var repliesToMeList = $("<ul></ul>"); var repliesToMe = $("<div class='fbnRepliesToMe'><span class='label'>Replies To Me</span><span class='replyToMeListHolder' /></div>"); repliesToMe.find("span.replyToMeListHolder").append(repliesToMeList); var visibleCommentLinks = $("div.cb:visible:not(.fbnIgnored) a[href^='#'], div.cb:visible:not(.fbnIgnored) a[href^='/comments']"); visibleCommentLinks.filter(function () { return $(this).text() === myUserName; }).each(function () { var replyToMeHeader = $(this).closest("div.cb").prev("div.ch"); if (replyToMeHeader.data("uname") !== myUserName) { repliesToMeList.append("<li><a href='#{0}'>{1}</a></li>".fex(replyToMeHeader.attr("id").substring(1), replyToMeHeader.data("uname"))); } }); if (repliesToMeList.find("li").length > 0) { repliesToMe.insertAfter($("div#main h1").first()); repliesToMeList.parent().hide(); repliesToMe.find("span.label").click(function () { repliesToMeList.parent().toggle(250); }); } } } // add comment body smarttags/menu var contextMenu = $("<div id='selectionMenu' style='display:none;position:absolute;'><div id='selectionMenuQuote'>Quote this</div><hr /><div id='selectionMenuTag'>Tag this</div></div>"); $("body").append(contextMenu); contextMenu.click(function (event) { var menu = $(this); try { var item = $(event.target); var text = menu.data("selectionText"); var commentInfo = __fixbn.findCommentInfo(menu.closest("div.cb").find("div.reply a")); var cleanHtml = __fixbn.cleanSelection(text, commentInfo); switch (event.target.id) { case "selectionMenuQuote": var commentBox = $("#comment"); if (commentBox.length) { $("html, body").animate({ scrollTop: $(document).height() - $(window).height() }); var originalComment = commentBox.val(); if (originalComment !== "") { originalComment = originalComment + "\n"; } try { commentBox.data("editing", true); commentBox.val(originalComment + cleanHtml + "\n\n").focus().scrollTop(commentBox[0].scrollHeight).caret(-1); commentBox.data("oldVal", commentBox.val()); } catch (ex) { console.error(ex); } finally { commentBox.data("editing", false); console.log("set editing to " + commentBox.data("editing")); } } else { alert("You must log in to comment"); } break; case "selectionMenuTag": if (GM_config.get("stickyTagger")) { var stickyTaggerInput = $("div.stickyTagger input"); stickyTaggerInput.show(250); stickyTaggerInput.val(text); } else { alert("Enable the Floating Tagger in options to use selection tagging."); } break; } } catch (ex) { console.error(ex); } menu.fadeOut(); }); $("div.cb").smartTag({ tagWidth: "32px", tagHeight: "32px", onClick: function (event) { try { var tag = $(this); var container = $(this).closest("div.cb"); var menu = $("#selectionMenu"); menu.data("selectionText", event.data); container.append(menu); menu.css({ top: parseInt(tag.css("top")) + parseInt(tag.css("height")), left: tag.css("left") }).fadeIn(); } catch (ex) { console.error(ex); } }, //onShowing onHiding: function () { var menu = $("#selectionMenu"); menu.fadeOut(); $("body").append(menu); } }); // make sure the standard reply scrolls the comment textarea if ($("#comment").length) { $("div.reply a").click((function () { //window.setTimeout(function () { try { $("html, body").animate({ scrollTop: $(document).height() - $(window).height() }); $("#comment").focus().scrollTop($("#comment")[0].scrollHeight).caret(-1); } catch (ex) { console.error("FixbN failed scroll comment box", ex); } //}, 500); })); } else { $("div.reply").hide(); } }; FixbN.prototype.fixQueuePage = function () { if (!bnurl.isQueuePage()) { return; } $("input.submitform[name='headline']").attr("maxlength", "254"); }; FixbN.prototype.fixTaggers = function () { var notification = null; $.fn.tagn.defaults.beforeSubmit = function (tagSet) { var tagInput = $(this); tagInput.val(""); }; $.fn.tagn.defaults.taggingComplete = function (tagSet) { try { var accepted = 0; var result = $("<ul></ul>"); for (var i = 0; i < tagSet.tags.length; i++) { var tag = tagSet.tags[i]; var status = tagSet.tags[i].status; switch (status) { case "accepted": accepted = Math.max(accepted, parseInt(tag.message)); result.append("<li>{0} accepted</li>".fex(tag.value)); break; case "matched": result.append("<li>{0} matched with {1}</li>".fex(tag.value, tag.message)); break; default: result.append("<li>{0} rejected: {1}</li>".fex(tag.value, tag.message)); break; } } result.prepend("<li>{0} total tags accepted for this thread".fex(accepted.toString())); new noty({ 'timeout': GM_config.get("tagNotyDuration") * 1000, 'type': 'information', 'killer': true, 'text': result.html() }); } catch (ex) { console.error("Fix bN Failed to notify of tagging status", ex); } }; // heh, i couldn't disable the other submit handlers, so I just hide and copy that fucker if (bnurl.isCommentsPage()) { var storyId = $("form input[name='storyid']").val(); var oldForm = $("form input[id^='tag']").closest("form"); if (oldForm.length > 0) { var newForm = oldForm.clone(false); newForm.find("label[id^='tagsaccepted']").attr("id", "tagnResult").addClass("tagnResult"); newForm.addClass("tagnForm").insertAfter(oldForm); oldForm.css("display", "none"); newForm.find("input[id^='tag']").tagn({ "storyId": storyId }); if (GM_config.get("stickyTagger")) { var stickyTagger = $("<div class='stickyTagger'><form><label for='stickyTaggerInput'>Tag</label><input id='stickyTaggerInput' /></form></div>"); $("div#main").append(stickyTagger); stickyTagger.find("label").click(function () { var tagger = $("#" + $(this).attr("for")); tagger.toggle(250); }); stickyTagger.find("input").tagn({ "storyId": storyId }); } } } if (bnurl.isQueuePage()) { $(".storyentry form input[id^='tag']").each(function () { var queueStoryId = this.id.substring(3); var that = $(this); var oldQueueForm = that.closest("form"); var newQueueForm = oldQueueForm.clone(false).addClass("tagnForm").insertAfter(oldQueueForm); newQueueForm.find("label[id^='tagsaccepted']").addClass("tagnResult"); newQueueForm.find("input[id^='tag']").tagn({ "storyId": queueStoryId }); oldQueueForm.css("display", "none"); }); } }; FixbN.prototype.createDateUrl = function (relativeDays) { var linkDate = new Date(); linkDate.setDate(linkDate.getDate() + relativeDays); return "/date/" + linkDate.getFullYear() + "/" + (linkDate.getMonth() + 1) + "/" + linkDate.getDate(); }; FixbN.prototype.findCommentInfo = function (replyLink) { return eval($(replyLink).attr("href").replace("javascript:reply", "(function(c,u){return { 'commentId': c, 'username': u };})") + ";"); // jshint ignore:line }; FixbN.prototype.cleanSelection = function (selectionHtml, commentInfo) { var quote = $("<quote />").append(selectionHtml); quote.find("img").replaceWith(function () { return "<a href='" + $(this).attr("src") + "'>[quoted image removed]</a>"; }); quote.find("br").remove(); var quotesToWrap = []; var quoteContents = quote.contents(); /* jshint -W014 */ quoteContents.each(function (index) { if (this.nodeType !== 3 && this.tagName === "A" && quoteContents.length > index + 2 && quoteContents[index + 1].nodeType === 3 && quoteContents[index + 2].tagName === "I" ) { quotesToWrap.push(index); } }); /* jshint +W014 */ var i; for (i = 0; i < quotesToWrap.length; i++) { var indexOfA = quotesToWrap[i]; $(quoteContents[indexOfA]).add(quoteContents[indexOfA + 1]).add(quoteContents[indexOfA + 2]).wrapAll("<font class='commentquote' ></font>"); } var result = quote.html(); result = result.replace("</i></font>\n\n", "</i></font>\n").replace(/^[\n\r\t ]+/, "").replace(/[\n\r\t ]+$/, ""); // wrap in a quote result = '<quote user="' + commentInfo.username + '" cid="' + commentInfo.commentId + '">' + result + "</quote>"; return result; }; return FixbN; })(); __fixbn = new FixbN(); })(jQuery); } catch (ex) { console.error("FixbN Failed declaring __fixbn", ex); } // user config class var __userConfig = null; try { __userConfig = (function ($) { "use strict"; var _configPromiseCache = {}; var _idStore = null; function _getConfigPromise(username, userid) { username = _cleanUsername(username); var result = _configPromiseCache[username]; if (!result) { var dfd = $.Deferred(); if (typeof userid !== "undefined" && userid !== null && !isNaN(userid)) { _saveUserId(username, userid); dfd.resolve(_createConfig(username, userid)); } else { var userIdPromise = _getUserIdPromise(username); userIdPromise.always(function (result, evt) { if (typeof result !== "undefined" && result !== null && typeof result.userId !== "undefined") { dfd.resolve(_createConfig(username, result.userId)); } else { // get userid from ajax console.log("FixbN Looking up userId on bannination.com/users/{0}".fex(username)); var userPageUrl = "http://www.bannination.com/users/{0}".fex(username); $.ajax({ url: userPageUrl, dataType: "text", success: function (data, textStatus, jqXHR) { // <h1>The user bleh was not found</h1> // <h1> Profile for artificeren (757)</h1> if (data.indexOf("Profile for") > 0) { //lolregex var match = data.match("(?:<h1> Profile for [^\(]* )(.*)(?:</h1>)"); // jshint ignore:line var result = match[1].replace(/\(/g, '').replace(/\)/g, ''); _saveUserId(username, result); dfd.resolve(_createConfig(username, result)); } else { console.error("FixbN Error retrieving userid for user {0}".fex(username)); dfd.reject(null); } }, error: function () { console.error("FixbN Error retrieving userid for user {0}".fex(username)); dfd.reject(null); } }); } }); } _configPromiseCache[username] = dfd.promise(); } return _configPromiseCache[username]; } function _cleanUsername(username) { return (username.indexOf("~") !== 0 && username.indexOf("someone who may or may not be") !== 0) ? username : "~Anonymous"; } function _createConfig(username, userId) { var configId = 'FixbNUser' + userId; var configKey = username; var userConfigFrame = $("<div style='display:none;' />")[0]; document.body.appendChild(userConfigFrame); var config = new GM_configStruct(); config.init({ 'id': configId, 'title': 'Fix bN Settings for User: {0}'.fex(configKey), 'fields': { 'visibility': { 'label': 'User Comment Visibility', 'type': 'radio', 'options': ['Normal', 'Ignore'], 'default': 'Normal' }, 'ignoreReplies': { 'label': 'User Being Quoted Visibility', 'type': 'radio', 'options': ['Shown', 'Default', 'Hidden'], 'default': 'Default' }, 'blockImages': { 'label': 'User Comment Image Visibility', 'type': 'radio', 'options': ['Normal', 'Removed'], 'default': 'Normal' }, 'nabbitVisibility': { 'label': 'User Nabbit Image Visibility', 'type': 'radio', 'options': ['Shown', 'Default', 'Hidden'], 'default': 'Default' }, 'overrideIgnoreReplies': { 'label': "Show This User's Replies to Ignored Users", 'type': 'checkbox', 'default': false }, 'headColor': { 'label': 'Comment Header Font Color', 'type': 'text', 'size': 25, 'default': '' }, 'headBackColor': { 'label': 'Comment Header Background Color', 'type': 'text', 'size': 25, 'default': '' } }, 'frame': userConfigFrame }); var openCallbacks = $.Callbacks(); openCallbacks.fire.callbacks = openCallbacks; config.onOpen = openCallbacks.fire; var saveCallbacks = $.Callbacks(); saveCallbacks.fire.callbacks = saveCallbacks; config.onSave = saveCallbacks.fire; var closeCallbacks = $.Callbacks(); closeCallbacks.fire.callbacks = closeCallbacks; config.onClose = closeCallbacks.fire; var resetCallbacks = $.Callbacks(); resetCallbacks.fire.callbacks = resetCallbacks; config.onReset = resetCallbacks.fire; config.onSave.callbacks.add((function () { if (this.isOpen) { this.close(); } }).bind(config)); config.onOpen.callbacks.add(function (doc, win, frame) { $(frame).find("input[id$='_field_headColor'], input[id$='_field_headBackColor']").spectrum({ clickoutFiresChange: true, preferredFormat: "hex6", showInput: true, showButtons: true, allowEmpty: true, showPalette: true, showSelectionPalette: true, palette: [ "#EEEEEE", "#807373", "#955050", "#373737" ], localStorageKey: "spectrum.fixbn.bannination" }); $(frame).find(".saveclose_buttons[id$='_saveBtn']").text(" OK "); $(frame).find(".saveclose_buttons[id$='_closeBtn']").text("Cancel"); }); config.bnUsername = username; return config; } function _getUserIdPromise(username) { return __fixbndb.idStore().get(username); } function _saveUserId(username, userId) { try { $.when(__fixbndb.idStore().get(username)).then( function (item, evt) { if ( typeof item !== "undefined" && item !== null ) { if (item.userId !== userId) { __fixbndb.idStore().put({ "username": username, "userId": userId }, username); } } else { __fixbndb.idStore().add({ "username": username, "userId": userId }, username); } }, function (error, evt) { __fixbndb.idStore().add({ "username": username, "userId": userId }, username); } ); } catch (ex) { console.error("FixbN Failed saving userId to local storage", ex); } } return { getConfigPromise: _getConfigPromise, getCleanUsername: _cleanUsername }; })(jQuery); } catch (ex) { console.error("FixbN Failed declaring __userConfig", ex); } // local database var __fixbndb = null; try { __fixbndb = (function ($) { function _idStore() { return $.indexedDB("FixbN").objectStore("Users", { "keypath": "username", "autoincrement": false }); } return { idStore: _idStore }; })(jQuery); } catch (ex) { console.error("FixbN Failed declaring __fixbndb", ex); } // bannination.com url tests var bnurl = (function () { var url = new URI(); function _isHeadlinesPage() { result = (url.path() === "/" || url.path() === "/pages/main/index.vm" || (url.segment().length > 0 && url.segment(0).toLowerCase() === "date" && url.segment().length < 5)); return result; } function _isHeadlinesByScore() { result = _isHeadlinesPage && ( url.path() === "/" || url.path() === "/pages/main/index.vm" ); return result; } function _isHeadlinesByDate() { result = _isHeadlinesPage() && url.segment().length > 0 && url.segment(0).toLowerCase() === "date"; return result; } function _isCommentsPage() { result = url.segment().length > 0 && (url.segment(0).toLowerCase() === "comments" || (url.segment(0).toLowerCase() === "date" && url.segment().length >= 5)); return result; } function _isCommentsPageTagable() { result = _isCommentsPage() && (url.segment(url.segment.length - 1) !== "1000"); return result; } function _isQueuePage() { result = url.segment.length > 0 && url.segment(0).toLowerCase() === "queue"; return result; } return { isHeadlinesPage: _isHeadlinesPage, isHeadlinesByScore: _isHeadlinesByScore, isHeadlinesByDate: _isHeadlinesByDate, isCommentsPage: _isCommentsPage, isCommentsPageTagable: _isCommentsPageTagable, isQueuePage: _isQueuePage }; })(); // nabbit jquery plugin (function ($) { "use strict"; var Nabbit; var $bind = function (fn, me) { return function () { return fn.apply(me, arguments); }; }; Nabbit = (function () { // ctor function Nabbit($el, settings) { if (settings === null) { settings = {}; } this.uid = ""; this.uname = ""; this.config = null; this.attach = $bind(this.attach, this); this.update = $bind(this.update, this); this.$el = $el; this.settings = $.extend({}, $.fn.nabbit.defaults, settings); if (this.settings.userId) { if (typeof (this.settings.userId) === "function") { this.uid = $bind(this.settings.userId, this.$el)(); } else { this.uid = this.settings.userId.toString(); } } else { this.uid = this.$el.text(); } if (this.settings.username) { if (typeof (this.settings.username) === "function") { this.uname = $bind(this.settings.username, this.$el)(); } else { this.uname = this.settings.username.toString(); } } $.when(__userConfig.getConfigPromise(this.uname, this.uid)).then(this.attach); } Nabbit.prototype.attach = function (userConfig) { this.config = userConfig; this.config.onSave.callbacks.add(this.update); this.update(); }; Nabbit.prototype.update = function () { var img = null; var imgsrc = ""; var title = ""; if (this.settings.imgTitle) { if ($.type(this.settings.imgTitle) === "function") { title = $bind(this.settings.imgTitle, this.$el)(); } else if (this.settings.imgTitle !== null) { title = this.settings.imgTitle.toString(); } } var imgHtml = "<img alt='{title}' title='{title}' {size} />"; if (title !== "") { imgHtml = imgHtml.fex({ 'title': title }); } else { imgHtml = imgHtml.fex({ 'title': "" }); } switch (this.settings.imgSize) { case "small": img = $(imgHtml.fex({ 'size': "width='58' height='18'" })); imgsrc = "http://webmonkees.com/naBBits/" + this.uid + ".gif"; break; case "large": img = $(imgHtml.fex({ 'size': "width='180' height='18' align='top' border='0'" })); imgsrc = "http://webmonkees.com/naBBits/m" + this.uid + ".png"; break; default: } var globalNabbitVisibility = GM_config.get("nabbitVisibility"); var userNabbitVisibility = this.config.get("nabbitVisibility"); var visClass = ""; switch (userNabbitVisibility) { case "Shown": visClass = "nabbitForceOn"; break; case "Hidden": visClass = "nabbitOff"; break; case "Default": visClass = globalNabbitVisibility ? "nabbitOn" : "nabbitOff"; break; } img.addClass(visClass); if ($.type(this.settings.imgClass) === "string" && this.settings.imgClass !== "") { img.addClass(this.settings.imgClass); } img.load($bind(function (evt) { if ($.type(this.settings.loaded) === 'function') { $bind(this.settings.loaded, this.$el)(evt.target); } else { this.$el.empty().append(evt.target); } }, this)).attr("src", imgsrc); }; return Nabbit; })(); $.fn.extend({ nabbit: function (options) { if (options === null) { options = {}; } return this.each(function () { return new Nabbit($(this), options); }); } }); // settings default $.fn.nabbit.defaults = { userId: null, // string or function that returns the nabbit userId username: null, // string or function that returns the username imgTitle: null, // string of function that returns the title for the created img loaded: null, // function that handles the img loaded event, recieves the loaded image imgClass: '', // string which is added to the img css class imgSize: 'small' // choices: small, large }; })(jQuery); // bN Sunlight jquery plugin (function ($) { "use strict"; var Sunlight; var $bind = function (fn, me) { return function () { return fn.apply(me, arguments); }; }; Sunlight = (function () { // ctor function Sunlight($el, settings) { if (settings === null) { settings = {}; } this.attach = $bind(this.attach, this); this.load = $bind(this.load, this); this.$el = $el; this.settings = $.extend({}, $.fn.sunlight.defaults, settings); this.attach(); } Sunlight.prototype.attach = function () { if (this.settings && this.settings.event && this.settings.event !== "") { this.$el.on(this.settings.event, this.load); } else { this.load(); } }; Sunlight.prototype.load = function () { var commentId = this.$el.closest("div.ch").attr("id").substring(1); //http://www.bannination.com/comments/5181856/sunlight/5087531 var sunlightUrl = "http://www.bannination.com/comments/" + this.settings.threadId + "/sunlight/" + commentId; this.$el.data("originalcursor", this.$el.css("cursor")); this.$el.css("cursor", "progress"); $.ajax({ url: sunlightUrl, dataType: "text", success: $bind(function (data) { var sunlight = { cool: 0, uncool: 0, blame: "" }; var cools = data.match(new RegExp(" Cool from (.*)\n", "g")); var uncools = data.match(new RegExp(" Uncool from (.*)\n", "g")); var i; if (cools && cools.length) { sunlight.cool = cools.length; for (i = 0; i < cools.length; i++) { sunlight.blame += "* " + cools[i]; } } if (uncools && uncools.length) { sunlight.uncool = uncools.length; for (i = 0; i < uncools.length; i++) { sunlight.blame += "* " + uncools[i]; } } if (sunlight.blame.length === 0) { sunlight.blame = "No Votes"; } if (this.settings && this.settings.handler) { ($bind(this.settings.handler, this.$el))(sunlight); } }, this), complete: $bind(function () { this.$el.css("cursor", this.$el.data("originalcursor")); }, this) }); }; return Sunlight; })(); $.fn.extend({ sunlight: function (options) { if (options === null) { options = {}; } return this.each(function () { return new Sunlight($(this), options); }); } }); // settings default $.fn.sunlight.defaults = { threadId: '0', handler: null, event: '' }; })(jQuery); // bN User Decoration jquery plugin try { (function ($) { "use strict"; var UserDecoration; var $bind = function (fn, me) { return function () { return fn.apply(me, arguments); }; }; UserDecoration = (function () { // ctor function UserDecoration($el, settings) { if (settings === null) { settings = {}; } this.userConfig = null; this.childConfigs = {}; this.userId = "0"; this.userName = ""; this.attach = $bind(this.attach, this); this.showUI = $bind(this.showUI, this); this.updateAll = $bind(this.updateAll, this); this.update = $bind(this.update, this); this.onImageReplaced = $.Callbacks(); this.$el = $el; this.$el.data("userDecoration", this); this.settings = $.extend({}, $.fn.userDecoration.defaults, settings); if (typeof this.settings.onImageReplaced === "function") { this.onImageReplaced.add(this.settings.onImageReplaced); } this.attach(); } // attach to element UserDecoration.prototype.attach = function () { var header = this.$el.closest("div.ch"); this.userId = header.data("uid"); this.userName = header.data("uname"); var body = $("div.cb.u" + this.userId + "[id$='" + header.attr("id").substring(1) + "']"); var peep = $("<span class='peep' >ಠ_ಠ</span>"); peep.click( $bind(function () { this.$el.closest("div.ch").removeClass("fbnIgnored"); var body = $("div.cb.u" + this.userId + "[id$='" + header.attr("id").substring(1) + "']"); body.slideDown('fast').removeClass("fbnIgnored"); }, this)); header.prepend(peep); var me = this; // Get the configs for every quoted user and create a promise array from them var quotedConfigPromises = []; var threadId = $("form input[name='storyid']").val(); var quoteLinks = body.find("a[href^='#'], a[href^='/comments/{0}']".fex(threadId)); quoteLinks.each(function () { var link = $(this); var quotedUsername = link.text().replace("someone who may or may not be", "~"); var quotePromise = __userConfig.getConfigPromise(quotedUsername); quotePromise.done(function (quotedConfig) { me.childConfigs[quotedConfig.bnUsername] = quotedConfig; quotedConfigPromises.push(quotePromise); }); }); // when I have the main config and then all child configs, // attach to all of their save events for updating, and then update to current settings $.when(__userConfig.getConfigPromise(me.userName, me.userId)) .then(function (config) { $.when.apply(this, quotedConfigPromises).then(function () { me.userConfig = config; me.userConfig.onSave.callbacks.add(me.update); $.each(me.childConfigs, function (index, value) { value.onSave.callbacks.add(me.update); }); me.$el.css("cursor", "pointer").click(me.showUI); me.update(); }); }); }; UserDecoration.prototype.update = function () { if (this.userConfig === null) { return; } var header = this.$el.closest("div.ch.u" + this.userId); var body = $("div.cb.u" + this.userId + "[id$='" + header.attr("id").substring(1) + "']"); var me = this; // score hilighting var score = header.data("score"); if (score && score > parseInt(GM_config.get("highScoreThreshold"))) { header.addClass("highScore"); } else { header.removeClass("highScore"); } // image blocking var blockImages = this.userConfig.get("blockImages"); switch (blockImages) { case "Normal": body.find("a.fbnBlockedImage").replaceWith(function () { var blockedImage = $(this); var unblockedImage = $("<img src='" + blockedImage.attr("href") + "' />"); var height = blockedImage.data("height"); var width = blockedImage.data("width"); if (height) { unblockedImage.attr("height", height).css("height", height); } if (width) { unblockedImage.attr("width", width).css("width", width); } return unblockedImage; }); break; case "Removed": body.find("img").replaceWith(function () { var unBlockedImage = $(this); var blockedImageLink = $("<a class='fbnBlockedImage' href='" + unBlockedImage.attr("src") + "'>[image blocked]</a>"); blockedImageLink.data("width", unBlockedImage.attr("width")); blockedImageLink.data("height", unBlockedImage.attr("height")); me.onImageReplaced.fireWith(blockedImageLink, blockedImageLink); return blockedImageLink; }); break; } // header colors header .css("background-color", this.userConfig.get("headBackColor")) .css("color", this.userConfig.get("headColor")) .find("a").css("color", this.userConfig.get("headColor")) ; // quote style and ignored reply visibility var visibility = this.userConfig.get("visibility"); var threadId = $("form input[name='storyid']").val(); var quoteLinks = body.find("a[href^='#'], a[href^='/comments/{0}']".fex(threadId)); quoteLinks.each(function () { try { var link = $(this); link.text(link.text().replace("someone who may or may not be", "~")); var quotedUsername = link.text(); var quotedConfig = me.childConfigs[ __userConfig.getCleanUsername( quotedUsername )]; if (typeof quotedConfig === "undefined") { console.error("FixbN Failed acquiring config for quoted user {0} on a comment by {1}".fex(quotedUsername, me.userName)); return; } // styling quotes link.css({ "background-color": quotedConfig.get("headBackColor"), "color": quotedConfig.get("headColor") }); // ignoring replies as needed try { var replyVisibile = true; // first only bother checking if the quoted user is on ignore at all var quoteIgnore = (quotedConfig.get("visibility") === "Ignore"); if (quoteIgnore) { // since they are on Ignore, then determine if they have a reply override on them var gIgnore = GM_config.get("ignoreReplies"); var uIgnore = quotedConfig.get("ignoreReplies"); switch (uIgnore) { case "Shown": quoteIgnore = false; break; case "Hidden": quoteIgnore = true; break; default: quoteIgnore = gIgnore; break; } if (quoteIgnore) { //ok, this person is set to be reply-ignored, but prefered user settings can override that if (!(me.userConfig.get("overrideIgnoreReplies"))) { // BAM! Ignored Reply! replyVisibile = false; } } } visibility = ( visibility === "Normal" ) && replyVisibile ? "Normal" : "Ignore"; } catch (ex) { console.error("FixbN Failed determining quoted user's visibility", ex); } } catch (ex) { console.error(ex); } }); switch (visibility) { case 'Normal': body.filter(".fbnIgnored").slideDown('fast').removeClass("fbnIgnored"); header.removeClass("fbnIgnored"); break; case 'Ignore': body.filter(":visible").addClass("fbnIgnored").slideUp('fast'); header.addClass("fbnIgnored"); break; } }; UserDecoration.prototype.showUI = function (evt) { if (evt) { evt.preventDefault(); } this.userConfig.open(); }; return UserDecoration; })(); $.fn.extend({ userDecoration: function (options) { if (typeof options === "string") { switch (options) { case "update": try { $(this).data("userDecoration").update(); } catch (ex) { console.error(ex); console.log(this); console.error($(this).data("userDecoration")); } break; } return $(this); } else { if (options === null) { options = {}; } return this.each(function () { return new UserDecoration($(this), options); }); } } }); // settings default $.fn.userDecoration.defaults = { threadId: 0, onImageReplaced: function () { } }; })(jQuery); } catch (ex) { console.error("FixbN Failed declaring UserDecoration", ex); } // bN taggination replacement (function ($) { "use strict"; var Tagn; var $bind = function (fn, me) { return function () { return fn.apply(me, arguments); }; }; var TAG_ACCEPTED = "1"; var INVALID_CHARACTER_IN_TAG = "2"; var TAG_TOO_LONG = "3"; var TOO_MANY_REPEATED_WORDS = "4"; var TAG_MATCH = "5"; Tagn = (function () { /* * @constructor */ function Tagn($el, settings) { if (settings === null) { settings = {}; } this.attach = $bind(this.attach, this); this.formSubmit = $bind(this.formSubmit, this); this.submitTags = $bind(this.submitTags, this); this.tagSubmitSuccess = $bind(this.tagSubmitSuccess, this); this.tagSubmitError = $bind(this.tagSubmitError, this); this.$el = $el; this.settings = $.extend({}, $.fn.tagn.defaults, settings); this.tagSet = {}; this.attach(); } Tagn.prototype.attach = function () { var theForm = this.$el.closest("form"); theForm.get(0).submit = null; theForm .unbind("submit") .off("submit") .on("submit", this.formSubmit); }; Tagn.prototype.formSubmit = function (evt) { evt.preventDefault(); this.createTagSet(this.$el.val()); if (typeof this.settings.beforeSubmit === "function") { this.settings.beforeSubmit.bind(this.$el)(this.tagSet); } this.submitTags().done(function () { this.settings.taggingComplete.bind(this.$el)(this.tagSet); }.bind(this)); return false; }; Tagn.prototype.submitTags = function() { var promises = []; try { var url = ""; if (bnurl.isQueuePage()) { url = this.settings.queueTagUrl; } else { url = this.settings.articleTagUrl; } url = url.fex({ "storyId": this.settings.storyId }); var me = this; $.each(this.tagSet.tags, function (index, tag) { var def = new $.Deferred(); if (tag.status === "valid") { // call ajax $.ajax({ url: url.fex({ "tag": tag.value }), type: "GET", dataType: "text", context: me, beforeSend: function (jqXHR, settings) { jqXHR.stamp = me.tagSet.stamp; jqXHR.tag = tag; }, success: me.tagSubmitSuccess, error: me.tagSubmitError, complete: function (jqXHR, textStatus) { def.resolve(); } }); } else { $bind(me.settings.tagUpdated, me.$el)(me.tagSet); def.resolve(); } promises.push(def); }); } catch (ex) { console.error("Fix bN Failed sending tags", ex); } return $.when.apply(undefined, promises).promise(); }; Tagn.prototype.createTagSet = function (text) { // Split the tags separated by commas into an array. var initialTags = text.split(","); var cleaningRegex = /[^A-Za-z0-9$\.&'\s\-_]/g; this.tagSet = { "stamp": $.now(), "tags": [], }; // Clean the tags and only attempt to submit valid ones. for (var i = 0; i < initialTags.length; ++i) { var currentTag = initialTags[i].trim().replace(cleaningRegex, "").trim(); var tag = { 'value': currentTag, 'status': "", 'message': "" }; if (2 <= currentTag.length && currentTag.length <= 30) { tag.status = "valid"; } else { tag.status = "invalid"; tag.message = (currentTag.length > 5) ? "Tag too long" : "Tag too short"; } this.tagSet.tags.push(tag); } }; Tagn.prototype.tagSubmitSuccess = function (data, textStatus, jqXHR) { if (jqXHR.stamp !== this.tagSet.stamp) { return; } var tag = jqXHR.tag; var responseCode = data.substring(0, 1); switch (responseCode) { case TAG_ACCEPTED: tag.status = "accepted"; tag.message = data.substring(2, 99).trim(); break; case TAG_MATCH: var user = data.substring(2, 99).trim(); tag.status = "matched"; tag.message = (user.length === 0) ? "the auto-tagger" : user; break; case INVALID_CHARACTER_IN_TAG: tag.status = "rejected"; tag.message = "Invalid character in tag"; break; case TAG_TOO_LONG: tag.status = "rejected"; tag.message = "Tag too long"; break; case TOO_MANY_REPEATED_WORDS: tag.status = "rejected"; tag.message = "Too many repeated words"; break; default: if (data.indexOf("You must be logged in to do that") > 0) { tag.status = "rejected"; tag.message = "You must be logged in to do that"; } else { tag.status = "unknown"; tag.message = "unknown response from server"; } break; } $bind(this.settings.tagUpdated, this.$el)(this.tagSet); }; Tagn.prototype.tagSubmitError = function (jqXHR, textStatus, errorThrown) { if (jqXHR.stamp !== this.tagSet.stamp) { return; } var tag = jqXHR.tag; tag.status = String(textStatus); tag.message = String(errorThrown); $bind(this.settings.tagUpdated, this.$el)(this.tagSet); }; return Tagn; })(); $.fn.extend({ tagn: function (options) { if (options === null) { options = {}; } return this.each(function () { return new Tagn($(this), options); }); } }); $.fn.tagn.defaults = { storyId: "", articleTagUrl: "/post?action=addtag&storyid={storyId}&tag={tag}", queueTagUrl: "/post?action=addqueuetag&channel_id=1&story_queue_id={storyId}&tag={tag}", beforeSubmit: function () { }, tagUpdated: function () { }, taggingComplete: function () { } }; })(jQuery); // bitcoin donate jquery plugin try { (function ($) { "use strict"; var $bind = function (fn, me) { return function () { return fn.apply(me, arguments); }; }; var root = "https://blockchain.info/"; var Bitcoin = (function () { // @constructor function Bitcoin($el, settings) { if (settings === null) { settings = {}; } this.attach = $bind(this.attach, this); this.$el = $el; this.settings = $.extend({}, $.fn.tagn.defaults, settings); this.attach(); } Bitcoin.prototype.attach = function () { var button = this.$el; button.find(".blockchain").hide(); button.find('.stage-begin').trigger('show').show(); button.click(function () { var receivers_address = $(this).data('address'); var shared = $(this).data('shared'); var test = $(this).data('test'); if (!shared) { shared = false; } var callback_url = $(this).data('callback'); if (!callback_url) { callback_url = ''; } button.find('.blockchain').hide(); button.find('.stage-loading').trigger('show').show(); $.ajax({ type: "GET", dataType: 'json', url: root + 'api/receive', data: { method: 'create', address: encodeURIComponent(receivers_address), shared: shared, callback: callback_url }, success: function (response) { button.find('.qr-code').empty(); button.find('.blockchain').hide(); if (!response || !response.input_address) { button.find('.stage-error').trigger('show').show().html(button.find('.stage-error').html().replace('[[error]]', 'Unknown Error')); return; } function checkBalance() { $.ajax({ type: "GET", url: root + 'q/getreceivedbyaddress/' + response.input_address, data: { format: 'plain' }, success: function (response) { if (!response) { return; } var value = parseInt(response); if (value > 0 || test) { button.find('.blockchain').hide(); button.find('.stage-paid').trigger('show').show().html(button.find('.stage-paid').html().replace('[[value]]', value / 100000000)); } else { setTimeout(checkBalance, 5000); } } }); } try { var ws = new WebSocket('ws://ws.blockchain.info/inv'); if (!ws) { return; } ws.onmessage = function (msg) { try { var obj = $.parseJSON(msg.data); var result = 0; if (obj.op == 'utx') { var tx = obj.x; for (var i = 0; i < tx.out.length; i++) { var output = tx.out[i]; if (output.addr == response.input_address) { result += parseInt(output.value); } } } button.find('.blockchain').hide(); button.find('.stage-paid').trigger('show').show().html(button.find('.stage-paid').html().replace('[[value]]', result / 100000000)); ws.close(); } catch (ex) { console.error(ex); console.error(ex.data); } }; ws.onopen = function () { ws.send('{"op":"addr_sub", "addr":"' + response.input_address + '"}'); }; } catch (ex) { console.error(ex); } button.find('.stage-ready').trigger('show').show().html(button.find('.stage-ready').html().replace('[[address]]', response.input_address)); button.find('.qr-code').html('<img style="margin:5px" src="' + root + 'qr?data=' + response.input_address + '&size=125">'); button.unbind(); ///Check for incoming payment setTimeout(checkBalance, 5000); }, error: function (ex) { button.find('.blockchain').hide(); button.find('.stage-error').show().trigger('show').html(button.find('.stage-error').html().replace('[[error]]', ex.responseText)); } }); }); }; return Bitcoin; })(); $.fn.extend({ bitcoin: function (options) { if (options === null) { options = {}; } return this.each(function () { return new Bitcoin($(this), options); }); } }); })(jQuery); } catch (ex) { console.error("FixbN Failed declaring bitcoin", ex); } /* string format function attached to the string prototype */ if ("function" !== typeof "".fex) { // add fex() if one does not exist already String.prototype.fex = (function () { // closure to store regex "use strict"; var rxp = /\{\{|\}\}|\{(\w+)\}/g; /* * String formatting function. Input string can use either indexed or named placeholders {0} or {foo}, use {{ }} to escape placeholders. Parameters can either be a series of strings, an array, or simple object with name/stringvalue pairs * @example * "foo {0} {1}".fex('bar', 'zap'); // returns "foo bar zap" * "foo {0} {1}".fex(['bar', 'zap']); // returns "foo bar zap" * "foo {second} {third}".fex({'second':'bar', 'third':'zap'}); // returns "foo bar zap" * @param {(...string|string[]|object} replacements - the replacement value(s) to be inserted into the string * @this String * @returns {string} The formatted string */ return function () { // the fex function var args = arguments; if (!args || args.length === 0) { return this.toString(); } var col = {}; if (args.length === 1) { var arg = args[0]; if (arg === null) { arg = "null"; } switch ((typeof arg)) { case "object": col = arg; break; case "string": col[0] = arg; break; default: return this.toString(); } } if (args.length > 1) { col = Array.prototype.slice.call(args); } return this.replace(rxp, function (match, property) { if (property === null || typeof (property) === "undefined") { return match; } if (match === "{{") { return "{"; } if (match === "}}") { return "}"; } return (typeof col[property] !== "undefined") ? col[property] : match; }); }; }()); } /* * string trim function * attached to the string prototype to support chaining */ if ("function" !== typeof "".trim) { String.prototype.trim = function () { return $.trim(this); }; } // simple diff that assumes one addition to "n" function diff(o, n) { var i = 0; for (i = 0; i < o.length; i++) { if (o[i] !== n[i]) { break; } } o = o.substring(i); n = n.substring(i); var l = 0; for (i = 0; i < n.length; i++) { if (n.substring(i) === o) { break; } } return n.substring(0, i + 1); } /* jshint ignore:start */ try { (function ($, undefined) { 'use strict'; var indexedDB = window.indexedDB || window.mozIndexedDB || window.webkitIndexedDB || window.msIndexedDB; var IDBKeyRange = window.IDBKeyRange || window.webkitIDBKeyRange; var IDBCursor = window.IDBCursor || window.webkitIDBCursor || {}; if (typeof IDBCursor.PREV === "undefined") { IDBCursor.PREV = "prev"; } if (typeof IDBCursor.NEXT === "undefined") { IDBCursor.NEXT = "next"; } /** * Best to use the constant IDBTransaction since older version support numeric types while the latest spec * supports strings */ var IDBTransaction = window.IDBTransaction || window.webkitIDBTransaction; function getDefaultTransaction(mode) { var result = null; switch (mode) { case 0: case 1: case "readwrite": case "readonly": result = mode; break; default: result = IDBTransaction.READ_WRITE || "readwrite"; } return result; } $.extend({ /** * The IndexedDB object used to open databases * @param {Object} dbName - name of the database * @param {Object} config - version, onupgradeneeded, onversionchange, schema */ "indexedDB": function (dbName, config) { if (config) { // Parse the config argument if (typeof config === "number") config = { "version": config }; var version = config.version; if (config.schema && !version) { var max = -1; for (var key in config.schema) { max = max > key ? max : key; } version = config.version || max; } } var wrap = { "request": function (req, args) { return $.Deferred(function (dfd) { try { var idbRequest = typeof req === "function" ? req(args) : req; idbRequest.onsuccess = function (e) { dfd.resolveWith(idbRequest, [idbRequest.result, e]); }; idbRequest.onerror = function (e) { dfd.rejectWith(idbRequest, [idbRequest.error, e]); }; if (typeof idbRequest.onblocked !== "undefined" && idbRequest.onblocked === null) { idbRequest.onblocked = function (e) { var res; try { res = idbRequest.result; } catch (e) { res = null; // Required for Older Chrome versions, accessing result causes error } dfd.notifyWith(idbRequest, [res, e]); }; } if (typeof idbRequest.onupgradeneeded !== "undefined" && idbRequest.onupgradeneeded === null) { idbRequest.onupgradeneeded = function (e) { dfd.notifyWith(idbRequest, [idbRequest.result, e]); }; } } catch (e) { e.name = "exception"; dfd.rejectWith(idbRequest, ["exception", e]); } }); }, // Wraps the IDBTransaction to return promises, and other dependent methods "transaction": function (idbTransaction) { return { "objectStore": function (storeName) { try { return wrap.objectStore(idbTransaction.objectStore(storeName)); } catch (e) { idbTransaction.readyState !== idbTransaction.DONE && idbTransaction.abort(); return wrap.objectStore(null); } }, "createObjectStore": function (storeName, storeParams) { try { return wrap.objectStore(idbTransaction.db.createObjectStore(storeName, storeParams)); } catch (e) { idbTransaction.readyState !== idbTransaction.DONE && idbTransaction.abort(); } }, "deleteObjectStore": function (storeName) { try { idbTransaction.db.deleteObjectStore(storeName); } catch (e) { idbTransaction.readyState !== idbTransaction.DONE && idbTransaction.abort(); } }, "abort": function () { idbTransaction.abort(); } }; }, "objectStore": function (idbObjectStore) { var result = {}; // Define CRUD operations var crudOps = ["add", "put", "get", "delete", "clear", "count"]; for (var i = 0; i < crudOps.length; i++) { result[crudOps[i]] = (function (op) { return function () { return wrap.request(function (args) { return idbObjectStore[op].apply(idbObjectStore, args); }, arguments); }; })(crudOps[i]); } result.each = function (callback, range, direction) { return wrap.cursor(function () { if (direction) { return idbObjectStore.openCursor(wrap.range(range), direction); } else { return idbObjectStore.openCursor(wrap.range(range)); } }, callback); }; result.index = function (name) { return wrap.index(function () { return idbObjectStore.index(name); }); }; result.createIndex = function (prop, options, indexName) { if (arguments.length === 2 && typeof options === "string") { indexName = arguments[1]; options = null; } if (!indexName) { indexName = prop; } return wrap.index(function () { return idbObjectStore.createIndex(indexName, prop, options); }); }; result.deleteIndex = function (indexName) { return idbObjectStore.deleteIndex(indexName); }; return result; }, "range": function (r) { if ($.isArray(r)) { if (r.length === 1) { return IDBKeyRange.only(r[0]); } else { return IDBKeyRange.bound(r[0], r[1], (typeof r[2] === 'undefined') ? false : r[2], (typeof r[3] === 'undefined') ? false : r[3]); } } else if (typeof r === "undefined") { return null; } else { return r; } }, "cursor": function (idbCursor, callback) { return $.Deferred(function (dfd) { try { var cursorReq = typeof idbCursor === "function" ? idbCursor() : idbCursor; cursorReq.onsuccess = function (e) { if (!cursorReq.result) { dfd.resolveWith(cursorReq, [null, e]); return; } var elem = { // Delete, update do not move "delete": function () { return wrap.request(function () { return cursorReq.result["delete"](); }); }, "update": function (data) { return wrap.request(function () { return cursorReq.result["update"](data); }); }, "next": function (key) { this.data = key; }, "key": cursorReq.result.key, "value": cursorReq.result.value }; dfd.notifyWith(cursorReq, [elem, e]); var result = callback.apply(cursorReq, [elem]); try { if (result === false) { dfd.resolveWith(cursorReq, [null, e]); } else if (typeof result === "number") { cursorReq.result["advance"].apply(cursorReq.result, [result]); } else { if (elem.data) cursorReq.result["continue"].apply(cursorReq.result, [elem.data]); else cursorReq.result["continue"](); } } catch (e) { dfd.rejectWith(cursorReq, [cursorReq.result, e]); } }; cursorReq.onerror = function (e) { dfd.rejectWith(cursorReq, [cursorReq.result, e]); }; } catch (e) { e.type = "exception"; dfd.rejectWith(cursorReq, [null, e]); } }); }, "index": function (index) { try { var idbIndex = (typeof index === "function" ? index() : index); } catch (e) { idbIndex = null; } return { "each": function (callback, range, direction) { return wrap.cursor(function () { if (direction) { return idbIndex.openCursor(wrap.range(range), direction); } else { return idbIndex.openCursor(wrap.range(range)); } }, callback); }, "eachKey": function (callback, range, direction) { return wrap.cursor(function () { if (direction) { return idbIndex.openKeyCursor(wrap.range(range), direction); } else { return idbIndex.openKeyCursor(wrap.range(range)); } }, callback); }, "get": function (key) { if (typeof idbIndex.get === "function") { return wrap.request(idbIndex.get(key)); } else { return idbIndex.openCursor(wrap.range(key)); } }, "count": function () { if (typeof idbIndex.count === "function") { return wrap.request(idbIndex.count()); } else { throw "Count not implemented for cursors"; } }, "getKey": function (key) { if (typeof idbIndex.getKey === "function") { return wrap.request(idbIndex.getKey(key)); } else { return idbIndex.openKeyCursor(wrap.range(key)); } } }; } }; // Start with opening the database var dbPromise = wrap.request(function () { return version ? indexedDB.open(dbName, parseInt(version)) : indexedDB.open(dbName); }); dbPromise.then(function (db, e) { db.onversionchange = function () { // Try to automatically close the database if there is a version change request if (!(config && config.onversionchange && config.onversionchange() !== false)) { db.close(); } }; }, function (error, e) { // Nothing much to do if an error occurs }, function (db, e) { if (e && e.type === "upgradeneeded") { if (config && config.schema) { // Assuming that version is always an integer for (var i = e.oldVersion + 1; i <= e.newVersion; i++) { typeof config.schema[i] === "function" && config.schema[i].call(this, wrap.transaction(this.transaction)); } } if (config && typeof config.upgrade === "function") { config.upgrade.call(this, wrap.transaction(this.transaction)); } } }); return $.extend(dbPromise, { "cmp": function (key1, key2) { return indexedDB.cmp(key1, key2); }, "deleteDatabase": function () { // Kinda looks ugly coz DB is opened before it needs to be deleted. // Blame it on the API return $.Deferred(function (dfd) { dbPromise.then(function (db, e) { db.close(); wrap.request(function () { return indexedDB.deleteDatabase(dbName); }).then(function (result, e) { dfd.resolveWith(this, [result, e]); }, function (error, e) { dfd.rejectWith(this, [error, e]); }, function (db, e) { dfd.notifyWith(this, [db, e]); }); }, function (error, e) { dfd.rejectWith(this, [error, e]); }, function (db, e) { dfd.notifyWith(this, [db, e]); }); }); }, "transaction": function (storeNames, mode) { !$.isArray(storeNames) && (storeNames = [storeNames]); mode = getDefaultTransaction(mode); return $.Deferred(function (dfd) { dbPromise.then(function (db, e) { var idbTransaction; try { idbTransaction = db.transaction(storeNames, mode); idbTransaction.onabort = idbTransaction.onerror = function (e) { dfd.rejectWith(idbTransaction, [e]); }; idbTransaction.oncomplete = function (e) { dfd.resolveWith(idbTransaction, [e]); }; } catch (e) { e.type = "exception"; dfd.rejectWith(this, [e]); return; } try { dfd.notifyWith(idbTransaction, [wrap.transaction(idbTransaction)]); } catch (e) { e.type = "exception"; dfd.rejectWith(this, [e]); } }, function (err, e) { dfd.rejectWith(this, [e, err]); }, function (res, e) { //dfd.notifyWith(this, ["", e]); }); }); }, "objectStore": function (storeName, mode) { var me = this, result = {}; function op(callback) { return $.Deferred(function (dfd) { function onTransactionProgress(trans, callback) { try { callback(trans.objectStore(storeName)).then(function (result, e) { dfd.resolveWith(this, [result, e]); }, function (err, e) { dfd.rejectWith(this, [err, e]); }); } catch (e) { e.name = "exception"; dfd.rejectWith(trans, [e, e]); } } me.transaction(storeName, getDefaultTransaction(mode)).then(function () { // Nothing to do when transaction is complete }, function (err, e) { // If transaction fails, CrudOp fails if (err.code === err.NOT_FOUND_ERR && (mode === true || typeof mode === "object")) { var db = this.result; db.close(); dbPromise = wrap.request(function () { return indexedDB.open(dbName, (parseInt(db.version, 10) || 1) + 1); }); dbPromise.then(function (db, e) { db.onversionchange = function () { // Try to automatically close the database if there is a version change request if (!(config && config.onversionchange && config.onversionchange() !== false)) { db.close(); } }; me.transaction(storeName, getDefaultTransaction(mode)).then(function () { // Nothing much to do }, function (err, e) { dfd.rejectWith(this, [err, e]); }, function (trans, e) { onTransactionProgress(trans, callback); }); }, function (err, e) { dfd.rejectWith(this, [err, e]); }, function (db, e) { if (e.type === "upgradeneeded") { try { db.createObjectStore(storeName, mode === true ? { "autoIncrement": true } : mode); } catch (ex) { dfd.rejectWith(this, [ex, e]); } } }); } else { dfd.rejectWith(this, [err, e]); } }, function (trans) { onTransactionProgress(trans, callback); }); }); } function crudOp(opName, args) { return op(function (wrappedObjectStore) { return wrappedObjectStore[opName].apply(wrappedObjectStore, args); }); } function indexOp(opName, indexName, args) { return op(function (wrappedObjectStore) { var index = wrappedObjectStore.index(indexName); return index[opName].apply(index[opName], args); }); } var crud = ["add", "delete", "get", "put", "clear", "count", "each"]; for (var i = 0; i < crud.length; i++) { result[crud[i]] = (function (op) { return function () { return crudOp(op, arguments); }; })(crud[i]); } result.index = function (indexName) { return { "each": function (callback, range, direction) { return indexOp("each", indexName, [callback, range, direction]); }, "eachKey": function (callback, range, direction) { return indexOp("eachKey", indexName, [callback, range, direction]); }, "get": function (key) { return indexOp("get", indexName, [key]); }, "count": function () { return indexOp("count", indexName, []); }, "getKey": function (key) { return indexOp("getKey", indexName, [key]); } }; }; return result; } }); } }); $.indexedDB.IDBCursor = IDBCursor; $.indexedDB.IDBTransaction = IDBTransaction; $.idb = $.indexedDB; })(jQuery); } catch (ex) { console.error("FixbN Failed Declaring indexedDB", ex); } /* jshint ignore:end */ // The kickoff function $(document).ready(function () { "use strict"; // general style changes /* jshint -W064 */ GM_addStyle((function () { return [ "div#selectionMenu { box-shadow:0px 0px 5px black; background:#eeeeee; border:4px solid black; border-radius:5px; padding:5px; padding-left:8px; }", "div#selectionMenu div { cursor:pointer; }", "div#selectionMenu hr { margin:3px; padding:0px; }", "div#menu { vertical-align:middle; }", "div#menu ul { font-size: 0.8em; white-space: nowrap; clear:both; padding:0px; margin-bottom:5px; }", "div#menu ul li img.fbnIcon { display:inline;float:none;margin:0px;padding:0px; height:20px;}", "div#menu ul.leftMenu { float:left; padding:0px; }", "div#menu ul.leftMenu li { margin:0px; padding:0px 0.5em; border:none; border-right:1px solid white; }", "div#menu ul.leftMenu li:last-child { border:none; }", "div.profilePopup a { color: white; }", "div.profilePopup a.topLink { padding-bottom:5px; }", "div.recentCommentWrapper { font-size:12px; padding-top:5px; }", "div.recentCommentWrapper a:visited { color:#aaa; }", "div.fbnRepliesToMe { float:left; background:#807373; border: 1px solid black; margin-left:10px; margin-right:10px; margin-bottom:10px; font-size:0.75em; }", "div.fbnRepliesToMe span { color:white; cursor:pointer; padding-left:0.5em; padding-right:0.5em; }", "div.fbnRepliesToMe ul { display: inline; }", "div.fbnRepliesToMe ul li { display: inline-block; white-space:nowrap; border-left: 1px solid white; padding-left: 0.5em; padding-right:0.5em; margin:0em; }", "div.fbnRepliesToMe ul a { color:white; }", "div.fbnRepliesToMe ul a:visited { color:#bbb; }", ".notyReply a { color: white; }", "div.fbnScrollToBottom { margin:0px; padding:0px; border:none; position:fixed; top:-5px; left:auto; right:-3px; z-index:1001; }", "div.fbnScrollToBottom img { margin:0px; padding:0px; border:none; cursor:pointer; }", "div.fbnScrollToTop { margin:0px; padding:0px; border:none; position:fixed; top:-5px; left:auto; right:-3px; z-index:1001; }", "div.fbnScrollToTop img { margin:0px; padding:0px; border:none; cursor:pointer; }", "div.mfp-content div.ch { margin-left: 5em; margin-right:5em; border:10px solid white; border-bottom-width:0px; }", "div.mfp-content div.cb { margin-left: 5em; margin-right:5em; border:10px solid white; border-top-width:0px;}", "div.stickyTagger {margin:0.25em; padding:0em; background:#807373; position: fixed; left: auto; right: 0; bottom: 0; text-align: right; border:1px solid black; box-shadow: 0px 0px 5px black;}", "div.stickyTagger label { color:white; cursor:pointer; padding:2em 1em 2em 1em; font-size:1.5em; }", "div.stickyTagger input { margin-right:1em; display:none; font-size:1.25em; margin-bottom:3px; width: 30em; }", "div#header.fixedHeader { position:fixed; top:0px; box-shadow: 0px 0px 10px black; z-index:1000; }", "div.form { width:98%; min-width:511px; }", "div#tag_cloud span { cursor:pointer; }", "div#footer { color: #CCCCCC; font-size: 0.5em; }", "div#footer a { font-size: 1.5em; }", "a.commentsLinkButton { float:right; margin-top:-15px; margin-right:-10px; }", "div.ch.fbnIgnored img { opacity: 0; }", "div.ch.highScore { padding-left: 25px; background-image: url(); background-repeat: no-repeat; background-attachment: scroll; background-position:left center; }", "div.ch img.nabbitOff { opacity: 0; }", "div.ch img.nabbitForceOn { opacity: 1; }", "div.ch span.peep { display: none; float: left; }", "div.ch.fbnIgnored span.peep { display: inline; float: left; cursor:pointer; font-weight:bold; padding: 0px 0.5em 0px 0.5em; margin: 0.25em 0px 0.5em 0px; }", "div.cb ul, div.cb pre { padding:0px; margin:0px; } div.cb li { padding:0px; margin:0px 0px 0px 1em; }", "div.cb .commentquote { display: block; padding:0px 0px 0px 0.5em; margin:1em 0px 0px 0.5em; border-left: 2px solid #807373; font-size: 0.75em; }", "div.cb a[href^='#'] { color:black; }", "div#menu a.a1{padding-right:25px;background:url(../img/bncombined.png) no-repeat top;background-position:right -360px;}", "div#menu a.a1{padding-right:25px;background:url(../img/bncombined.png) no-repeat top;background-position:right -360px;}", "div#menu a.a2{padding-right:25px;background:url(../img/bncombined.png) no-repeat top;background-position:right -432px;}", "div#menu a.a3{padding-right:25px;background:url(../img/bncombined.png) no-repeat center;background-position:right -504px;}", "div#menu a.a4{padding-right:25px;background:url(../img/bncombined.png) no-repeat center;background-position:right -576px;}", "div#menu a.a5{padding-right:25px;background:url(../img/bncombined.png) no-repeat center;background-position:right -648px;}", "div#menu a.a1{padding-right:25px;background:url(../img/bncombined.png) no-repeat top;background-position:right -360px;}" ].join('\n') + '\n'; })()); // gm_config styling GM_addStyle((function () { return [ "div[id$='_wrapper'] * { font-family: calibri,arial,tahoma,myriad pro,sans-serif; }", "div[id$='_wrapper'] { background: #FFF; padding: 0px; height: 99%; background-repeat: no-repeat; background-position: top right; background-size: contain; background-image:url(); }", "div[id$='_wrapper'] .block { display: block; }", "div[id$='_wrapper'] .center { text-align: center; }", "div[id$='_wrapper'] .indent40 { margin-left: 40%; }", "div[id$='_wrapper'] .config_header { font-size: 2em; margin: -5px -5px 0px -5px; background-color: #807373; color:white; white-space: nowrap;}", "div[id$='_wrapper'] .config_desc, div[id$='_wrapper'] .section_desc, div[id$='_wrapper'] .reset { font-size: 0.75em; }", "div[id$='_wrapper'] .config_var { margin: 0.5em; padding: 0.5em; }", "div[id$='_wrapper'] .config_var input[type='checkbox'] { float:left; position:relative; top:2px; margin-right:5px; }", "div[id$='_wrapper'] input[type='radio'] { margin-left: 0.5em; margin-top: 0.5em; }", "div[id$='_wrapper'] .field_label { font-size: 1em; font-weight: bold; margin-right: 1em; }", "div[id$='_wrapper'] .radio_label { font-size: 1em; margin-right: 1em; }", "div[id$='_wrapper'] .section_header_holder { margin-top: 1em; }", "div[id$='_wrapper'] .section_header { background: #414141; border: 1px solid #000; color: #FFF; font-size: 13pt; margin: 0; }", "div[id$='_wrapper'] .section_desc { background: #EFEFEF; border: 1px solid #CCC; color: #575757; font-size: 9pt; margin: 0 0 6px; }", "div[id$='_wrapper'] div[id$='_buttons_holder'] { position:absolute; bottom:0px; right:0px; left:0px; color: #000; text-align: right; padding-right:0.5em; padding-bottom:0.5em; }", "div[id$='_wrapper'] div.saveclose_buttons { margin:0em 2em 2em 0em; }", "div[id$='_wrapper'] button[id$='_saveBtn'] { font-weight:bold; }", "div[id$='_wrapper'] button[id$='_closeBtn'] { font-weight:bold; }" ].join('\n') + '\n'; })()); /* jshint +W064 */ // add markItUp! style information (including base64 images). // converted by Stephen Cronin /* jshint ignore:start */ GM_addStyle((function () { return "\ .markItUp .miuBold a { background-image:url(); } \ .markItUp .miuItalic a { background-image:url(); } \ .markItUp .miuItalic a { background-image:url(); } \ .markItUp .miuStrike a { background-image:url(); } \ .markItUp .miuPara a { background-image:url(); } \ .markItUp .miuQuote a { background-image:url(); } \ .markItUp .miuUList a { background-image:url(); } \ .markItUp .miuOList a { background-image:url(); } \ .markItUp .miuListitem a { background-image:url(); } \ .markItUp .miuImage a { background-image:url(); } \ .markItUp .miuLink a { background-image:url(); } \ .markItUp .miuPre a { background-image:url(); } \ .markItUp .miuClean a { background-image:url(); } \ .markItUp .miuSymbols a { background-image:url(); } \ .markItUp * { margin:0px; padding:0px; outline:none; } \ .markItUp a:link, .markItUp a:visited { color:#000; text-decoration:none; } \ .markItUp { width:100% !important; margin:5px 0 5px 0; } \ .markItUpContainer { border: 2px solid #807373; padding:5px 10px 0px 0px; font:1em Calibri, Verdana, Arial, Helvetica, sans-serif; color:white; } \ .markItUpEditor { font:1em Calibri, Verdana, Arial, Helvetica, sans-serif; background-color: #eeeeee; padding:5px !important; width:100% !important; border:none; margin:0px; height:320px; clear:both; display:block; line-height:18px; overflow:auto; } \ .markItUpPreviewFrame { overflow:auto; background-color:#FFFFFF; border:1px solid #3C769D; width:99.9%; height:300px; margin:5px 0; } \ .markItUpFooter { width:100%; cursor:n-resize; } \ .markItUpResizeHandle { overflow:hidden; width:22px; height:5px; margin-left:auto; margin-right:auto; cursor:n-resize; background-image:url(); } \ .markItUpHeader ul { padding-left:3px; } \ .markItUpHeader ul li { list-style:none; float:left; position:relative; } \ .markItUpHeader ul li ul{ display:none; } \ .markItUpHeader ul li:hover > ul{ display:block; } \ .markItUpHeader ul .markItUpDropMenu { margin-right:5px; background:transparent url() no-repeat 115% 50%; } \ .markItUpHeader ul .markItUpDropMenu li { margin-right:0px; } \ .markItUpHeader ul .markItUpSeparator { margin:0 10px; width:1px; height:16px; overflow:hidden; background-color:#CCC; } \ .markItUpHeader ul ul .markItUpSeparator { width:auto; height:1px; margin:0px; } \ .markItUpHeader ul ul { display:none; position:absolute; top:18px; left:0px; background:#F5F5F5; border:1px solid #3C769D; height:inherit; } \ .markItUpHeader ul ul li { float:none; border-bottom:1px solid #3C769D; } \ .markItUpHeader ul ul .markItUpDropMenu { background:#F5F5F5 url() no-repeat 100% 50%; } \ .markItUpHeader ul ul ul { position:absolute; top:-1px; left:150px; } \ .markItUpHeader ul ul ul li { float:none; } \ .markItUpHeader ul a { display:block; width:16px; height:16px; text-indent:-10000px; background-repeat:no-repeat; padding:3px; margin:0px; } \ .markItUpHeader ul ul a { display:block; padding-left:0px; text-indent:0; width:120px; padding:5px 5px 5px 25px; background-position:2px 50%; } \ .markItUpHeader ul ul a:hover { color:#FFF; background-color:#3C769D; } \ .markItUp .palette a { background-image:url(); } \ .markItUp .palette ul { width:81px; padding:1px; } \ .markItUp .palette li { border:1px solid white; width:25px; height:25px; overflow:hidden; padding:0px; margin:0px; float:left; } \ .markItUp .palette ul a { width:25px; height:25px; } \ .markItUp .palette ul a:hover { background-color:transparent; } \ .markItUp .palette .col1-1 a { background:#FCE94F; } \ .markItUp .palette .col1-2 a { background:#EDD400; } \ .markItUp .palette .col1-3 a { background:#C4A000; } \ .markItUp .palette .col2-1 a { background:#FCAF3E; } \ .markItUp .palette .col2-2 a { background:#F57900; } \ .markItUp .palette .col2-3 a { background:#CE5C00; } \ .markItUp .palette .col3-1 a { background:#E9B96E; } \ .markItUp .palette .col3-2 a { background:#C17D11; } \ .markItUp .palette .col3-3 a { background:#8F5902; } \ .markItUp .palette .col4-1 a { background:#8AE234; } \ .markItUp .palette .col4-2 a { background:#73D216; } \ .markItUp .palette .col4-3 a { background:#4E9A06; } \ .markItUp .palette .col5-1 a { background:#729FCF; } \ .markItUp .palette .col5-2 a { background:#3465A4; } \ .markItUp .palette .col5-3 a { background:#204A87; } \ .markItUp .palette .col6-1 a { background:#AD7FA8; } \ .markItUp .palette .col6-2 a { background:#75507B; } \ .markItUp .palette .col6-3 a { background:#5C3566; } \ .markItUp .palette .col7-1 a { background:#EF2929; } \ .markItUp .palette .col7-2 a { background:#CC0000; } \ .markItUp .palette .col7-3 a { background:#A40000; } \ .markItUp .palette .col8-1 a { background:#FFFFFF; } \ .markItUp .palette .col8-2 a { background:#D3D7CF; } \ .markItUp .palette .col8-3 a { background:#BABDB6; } \ .markItUp .palette .col9-1 a { background:#888A85; } \ .markItUp .palette .col9-2 a { background:#555753; } \ .markItUp .palette .col9-3 a { background:#000000; } \ "; })()); /* jshint ignore:end */ /* jshint -W064 */ try { GM_addStyle(GM_getResourceText('magnificcss')); GM_addStyle(GM_getResourceText('juipepper')); GM_addStyle(GM_getResourceText('spectrumcss')); } catch (ex) { console.error("Fix bN Failed applying css resources", ex); } /* jshint +W064 */ __fixbn.fix(); }); console.info("Fix bN v" + GM_info.script.version);