Musrum / CrikeyCleanCommentPreview

// ==UserScript==
// @name           CrikeyCleanCommentPreview
// @author         Musrum
// @namespace      2
// @description    Fix Comment Preview box on crikey blogs
// @include        http://blogs.crikey.com.au/*
// @include        http://www.crikey.com.au/*
// @exclude        http://www.crikey.com.au/*.gif
// @require 	   https://raw.github.com/sizzlemctwizzle/GM_config/master/gm_config.js
// @grant          none
// @version        4.15
// ==/UserScript==
// PRD https://openuserjs.org/scripts/Musrum/CrikeyCleanCommentPreview
// Ver 4.00
// New major version to deal with Crikey fixing their numbering mess up
// Ver 4.01
// Only one comment box
// Redirects back to current page (instead of first page) after posting.
// Ver 4.02
// Clean up for JSLint
// Ver 4.03 - 4.05 
// Bug Fixes
// Ver 4.06
// Revert the Redirects fix (Crikey has fixed their bug)
// Ver 4.07 
// Bug Fix (Multiple QQ Bubbles)
// Ver 4.08
// Stop loading in iframes
// Auto Login
// Ver 4.09
// Version Number in cccp setings
// Ver 4.11
// Fixed whitespace in blockquotes
// Ver 4.12
// Fixed issue with STFU hidden comments reappearing.
// Ver 4.13
// Moved to openuserjs.org
// Ver 4.14
// Hide OUTBRAIN clickbait
// Ver 4.15
// Comment out GM_registerMenuCommand (breaks Firefox)
4////////////////////////////////////////////////////////////////////////////////
/*jslint browser: true */
/* global GM_config, GM_registerMenuCommand */
////////////////////////////////////////////////////////////////////////////////
//don't run in iframes
if (window.top !== window.self) {
    return;
}
////////////////////////////////////////////////////////////////////////////////
// Auto login
window.addEventListener('load', function () {
    if ( document.getElementById("wp-submit-login") && document.getElementById("user_login").value !== "" ) {
        var wp_submit_login = document.getElementById("wp-submit-login")
        var login = wp_submit_login.parentNode.parentNode;
        login.action = login.action.replace(/redirect_to=.*/, 'redirect_to=' + encodeURIComponent(location.href));
        wp_submit_login.click();
    }
});
////////////////////////////////////////////////////////////////////////////////
// Config settings dialog
GM_config.storage = 'BetterBludging';
GM_config.init('Better Bludging',
    {
        oldfont: {
            section: ['cccp Ver 4.15'],
            label: 'Use the old Crikey font',
            type: 'checkbox',
            'default': true
        },
        stopfloat: {
            label: 'Stop the login bar from floating',
            type: 'checkbox',
            'default': true
        },
        hideavatars: {
            label: 'Hide avatars',
            type: 'checkbox',
            'default': false
        }
    },
    {
        open: function() {
            GM_config.addBorder(); // add a fancy border
            GM_config.resizeFrame('300px','400px'); // resize the config window
        }
    },
    {
        save: function () { location.reload(); } // reload the page when configuration was changed
    }
    );
////////////////////////////////////////////////////////////////////////////////
function showConfigCCCP() {
    GM_config.open();
}
//GM_registerMenuCommand('cccp Settings', showConfigCCCP); /* Appears to break in Firefox Ver 4.15*/
////////////////////////////////////////////////////////////////////////////////
// First Thing - Get the next page if required
////////////////////////////////////////////////////////////////////////////////
var comments;
var uls;
var ul;
var req_dom;
var comment;
var submit;
var acp_preview;
var cccp_comment;
var cccp_submit;
var cccp_acp_preview;
var cccp_comment_dom;
var removeHTML;
var em;
var respond;
var respond_on;
var comments_list = document.getElementById("comments-list");
var comments_on = (comments_list !== null);
if (comments_on) {
    comments = document.getElementById("comments");
    uls = comments_list.getElementsByTagName("ul");
    ul = uls[uls.length - 1];
    req_dom = document.createElement('div');
}
// Remove color from <a> elements
// Revert to old font
if (document.styleSheets !== undefined
        && document.styleSheets.length > 2) {
    if (document.styleSheets[2].cssRules !== undefined
            && document.styleSheets[2].cssRules.length > 4) {
        document.styleSheets[2].cssRules[4].style.color = '';
        if (GM_config.get('oldfont') === true) {
            document.body.style.cssText += ';font:13px verdana,clean,sans-serif;';
        }
    } else if (document.styleSheets[2].rules !== undefined
            && document.styleSheets[2].rules.length > 4) {
        document.styleSheets[2].rules[4].style.color = '';
        if (GM_config.get('oldfont') === true) {
            document.body.style.font = "13px verdana,clean,sans-serif";
        }
    }
}
////////////////////////////////////////////////////////////////////////////////
////////////////////////////////////////////////////////////////////////////////
function setupEmoticons() {
    var em = [];
    em["quotes.png"] = '';
    em["exclaim.png"] = '';
    em["face-angel.png"] = '';
    em["face-angry.png"] = '';
    em["face-cool.png"] = '';
    em["face-crying.png"] = '';
    em["face-devilish.png"] = '';
    em["face-embarrassed.png"] = '';
    em["face-kiss.png"] = '';
    em["face-laugh.png"] = '';
    em["face-monkey.png"] = '';
    em["face-plain.png"] = '';
    em["face-raspberry.png"] = '';
    em["face-sad.png"] = '';
    em["face-smile-big.png"] = '';
    em["face-smile.png"] = '';
    em["face-smirk.png"] = '';
    em["face-surprise.png"] = '';
    em["face-uncertain.png"] = '';
    em["face-wink.png"] = '';
    em["face-worried.png"] = '';
    em["idea.png"] = '';
    em["question.png"] = '';
    em["stock_right.png"] = '';
    return em;
}
////////////////////////////////////////////////////////////////////////////////
function insertAfter(el, rf) {
    if (rf.nextSibling !== null) {
        rf.parentNode.insertBefore(el, rf.nextSibling);
    } else {
        rf.parentNode.appendChild(el);
    }
}
////////////////////////////////////////////////////////////////////////////////
//  Allow users to reommend cccp with the links required
function cccpRecommend() {
    cccp_comment.value = cccp_comment.value + '\nTo use the Crikey Clear Comment Preview script, install in order:\n<a href="http://www.mozilla.com/en-US/firefox/personal.html">Firefox</a>\n<a href="https://addons.mozilla.org/en-US/firefox/addon/748">Greasemonkey</a>\n<a href="https://openuserjs.org/scripts/Musrum/CrikeyCleanCommentPreview">cccp</a>\nor:\n<a href="http://www.google.com/chrome">Google Chrome</a>\n<a href="https://chrome.google.com/webstore/detail/dhdgffkkebhmkfjojejmpbldmpobfkfo">Tampermonkey</a>\n<a href="https://openuserjs.org/scripts/Musrum/CrikeyCleanCommentPreview">cccp</a>';
}
////////////////////////////////////////////////////////////////////////////////
//  Used to Hide/Show the help section
function helpToggle() {
    var cccp_helptoggle     = document.getElementById("cccp-helptoggle");
    var cccp_helpcheatsheet = document.getElementById("cccp-htmlcheatsheet");
    if (cccp_helpcheatsheet.style.display === 'none') {
        cccp_helpcheatsheet.style.display = 'block';
        cccp_helptoggle.innerHTML = 'Hide Help';
    } else {
        cccp_helpcheatsheet.style.display = 'none';
        cccp_helptoggle.innerHTML = 'Show Help';
    }
}
////////////////////////////////////////////////////////////////////////////////
// Replace space with &nbsp; in code blocks
function mungeCommentCode() {
    var codeTags, i;
    codeTags = cccp_comment_dom.getElementsByTagName("code");
    for (i = 0; i < codeTags.length; i++) {
        codeTags[i].innerHTML = codeTags[i].innerHTML.replace(/ /g, '&nbsp;');
    }
}
//  Munge quote URLs into a pretty form
function mungeCommentQuoteURLs() {
    "use strict";
    var qURLs, qId, qLi, i, j;
    qURLs = cccp_comment_dom.innerHTML.match(new RegExp('(?![">])(http:\/\/blogs.crikey.com.au\/.*#comment-[0-9]+)\\b(?!["<])', 'g'));
    if (qURLs === null) {
        return;
    }
    qLi = document.createElement('li');
    for (i = 0; i < qURLs.length; i++) {
        qId = qURLs[i].match(/comment-[0-9]+/);
        //  Get the quoted comment as an <li> node
        if (document.getElementById(qId)) {
            qLi = document.getElementById(qId).cloneNode(true);
            qLi.setAttribute("title", '');
        } else {
            var req = new XMLHttpRequest();
            req.open('GET', qURLs[i], false);
            req.send(null);
            if ((req.readyState === 4) && (req.status === 200)) {
                req_dom.innerHTML = req.responseText.replace(/id="/mig, 'id="req-');
                var qComments = req_dom.getElementsByClassName('byuser');
                for (j = 0; j < qComments.length; j++) {
                    if (qComments[j] !== undefined && qComments[j] !== null && qComments[j].getAttribute('id') === 'req-' + qId) {
                        qLi = qComments[j].cloneNode(true);
                        break;
                    }
                }
                var titleElementDoc = document.getElementsByTagName('title')[0].innerHTML;
                var titleElementReq = req_dom.getElementsByTagName('title')[0].innerHTML;
                if (titleElementDoc !== titleElementReq) {
                    qLi.setAttribute("title", ' on ' + titleElementReq);
                } else {
                    qLi.setAttribute("title", '');
                }
            }
        }
        //
        var qAuthor = qLi.getElementsByClassName('fn')[0].innerHTML.replace(/<a[\w\W]*>([\w\W]*)<\/a>/, '$1');
        var qNumber = qLi.getElementsByClassName('comment-number')[0].innerHTML;
        var qThread = qLi.getAttribute("title");
        //  Remove header info from quoted comment
        for (j = 0; j < 6; j++) {
            qLi.removeChild(qLi.childNodes[0]);
        }
        var qText =  '<blockquote><i>' + qLi.innerHTML.replace(/^\s+|[\s\n]+$/g, '').replace(new RegExp('<[/]p>', 'g'), '\n').replace(removeHTML, '').replace(/\n$/, '')  + '</i></blockquote>';
        cccp_comment_dom.innerHTML = cccp_comment_dom.innerHTML.replace(qURLs[i], '<a href=\"' + qURLs[i] + '\">' + qAuthor + '@' + qNumber + qThread + '</a>' + qText);
    }
}
////////////////////////////////////////////////////////////////////////////////
// Munge comment text before preview or submit
function mungeComment() {
    mungeCommentQuoteURLs();
    // Previous munges are copied back to cccp_comment;
    cccp_comment.value = cccp_comment_dom.innerHTML;
    mungeCommentCode();
}
////////////////////////////////////////////////////////////////////////////////
// Very Easy Quotes
function veryEasyQuotes() {
    cccp_comment_dom.innerHTML += location.href.replace(/\/#.*/, '') + '/' + this.id;
    mungeComment();
}
////////////////////////////////////////////////////////////////////////////////
// Preview comment (copy to DOM; munge; copy to hidden textarea; preview)
function previewComment() {
    cccp_comment_dom.innerHTML = cccp_comment.value;
    mungeComment();
    comment.value = cccp_comment_dom.innerHTML;
    acp_preview.click();
}
////////////////////////////////////////////////////////////////////////////////
// Submit comment (copy to DOM; munge; copy to hidden textarea; submit)
function submitComment() {
    cccp_comment_dom.innerHTML = cccp_comment.value;
    mungeComment();
    comment.value = cccp_comment_dom.innerHTML;
    submit.click();
}
////////////////////////////////////////////////////////////////////////////////
function externalLinks()
{
    if (!document.getElementsByTagName) return;
    var anchors = document.getElementsByTagName("a");
    for ( var i=0; i < anchors.length; i++ )
    {
        var anchor = anchors[i];
        if ( anchor.getAttribute("href") && anchor.getAttribute("rel") == "external" )
        {
            if ( anchor.getAttribute("className") )
            {
                anchor.target = anchor.getAttribute("className");
            }
            else
            {
                anchor.target = "_blank";
            }
        }
    }
}
////////////////////////////////////////////////////////////////////////////////
function removeOutbrain()
{
    var ob = document.getElementsByClassName("outbrain-frame")[0];
    if (ob === undefined || ob === null) return;
    ob.style.display = 'none';
    
}
////////////////////////////////////////////////////////////////////////////////
//  Main
////////////////////////////////////////////////////////////////////////////////
//  Init Global Variables
removeHTML = new RegExp('<[/]?([psu]|address|applet|area|base|basefont|bdo|big|body|br|button|caption|center|col|colgroup|dd|dfn|dir|div|dl|dt|fieldset|font|form|frame|frameset|head|h[1-6r]|html|iframe|img|input|ins|kbd|label|legend|li|link|map|menu|meta|noframes|noscript|object|ol|optgroup|option|param|pre|samp|script|select|small|span|style|sub|sup|table|tbody|td|textarea|tfoot|th|thead|title|tr|tt|ul|var:\w+)[^>]*?>', 'gi');
em = setupEmoticons();
var quotes_img = document.createElement('img');
quotes_img.setAttribute("src", em["quotes.png"]);
var quotes_lnk = document.createElement('a');
quotes_lnk.appendChild(quotes_img);
var jj;
externalLinks();
removeOutbrain();
if (comments_on) {
    // Comments Box at the Bottom
    respond = document.getElementById("respond");
    respond_on = (respond !== undefined);
    if (respond_on) {
        var send_comments = document.getElementsByClassName('send-comment');
        for (jj = 1; jj < send_comments.length; jj++) {
            send_comments[jj].parentNode.parentNode.parentNode.removeChild(send_comments[jj].parentNode.parentNode);
        }
        comments.appendChild(respond);
        var clear = document.createElement('div');
        clear.setAttribute("class", "clear");
        comments.appendChild(clear);
        // var respNew = respond.cloneNode(1);
        // comments.appendChild(respNew); 
        // Create a new: comment <textarea>; submit <input>; and acp-preview <input>. Hide the originals
        comment = document.getElementById("comment");
        submit = document.getElementById("submit");
        acp_preview = document.getElementById("acp-preview");
        cccp_comment = comment.cloneNode(0);
        cccp_submit = submit.cloneNode(0);
        cccp_acp_preview = acp_preview.cloneNode(0);
        // Fix redirection issue
//        var redirect_to = document.createElement('input');
//        redirect_to.setAttribute("id", "redirect_to");
//        redirect_to.setAttribute("name", "redirect_to");
//        redirect_to.setAttribute("type", "hidden");
//        redirect_to.setAttribute("value", location.href.replace(/#comment.*/, ''));
//        submit.parentNode.appendChild(redirect_to);
        cccp_comment.setAttribute("id", "cccp_comment");
        cccp_comment.setAttribute("name", "cccp_comment");
        cccp_submit.setAttribute("id", "cccp_submit");
        cccp_submit.setAttribute("name", "cccp_submit");
        cccp_acp_preview.setAttribute("id", "cccp_acp_preview");
        cccp_acp_preview.setAttribute("name", "cccp_acp_preview");
        comment.parentNode.insertBefore(cccp_comment, comment);
        submit.parentNode.insertBefore(cccp_submit, submit);
        acp_preview.parentNode.insertBefore(cccp_acp_preview, acp_preview);
        comment.style.display = "none";
        submit.style.display = "none";
        acp_preview.style.display = "none";
        cccp_submit.addEventListener("click", submitComment, true);
        cccp_acp_preview.addEventListener("click", previewComment, true);
        // Hide the horrible Crickey Help
        var send_help = submit.parentNode.childNodes[1];
        send_help.style.display = "none";
        //  Create a new cccp_comment_dom <div>
        cccp_comment_dom = document.createElement('div');
        //  Create a new cccp-option <div>
        var cccp_option = document.createElement('div');
        cccp_option.setAttribute("id", "cccp-option");
        comments.appendChild(cccp_option);
        comments.appendChild(clear.cloneNode(0));
        // Preview Box Next
        var ajax_comment_preview = document.getElementById("ajax-comment-preview");
        ajax_comment_preview.innerHTML = "<div align=\"center\">Click the \"Preview\" button to preview your comment here.</div>";
        ajax_comment_preview.style.width = "auto";
        ajax_comment_preview.style.margin = "auto";
        comments.appendChild(ajax_comment_preview);
        //  Create the cccp helper section
        var cccp_helper = document.createElement('div');
        cccp_option.appendChild(cccp_helper);
        cccp_option.appendChild(document.createElement('p'));
        cccp_helper.setAttribute("id", "cccp-helper");
        cccp_helper.style.textAlign = 'center';
        cccp_helper.innerHTML = '<span style="float:left;clear:left;"><a id="cccp-helptoggle" style="color:blue">Show Help</a></span><span style="clear:both;"><a id="cccp-showconfig" style="color:blue">cccp Settings</a></span><span style="float:right;clear:right;"><a id="cccp-recommend">Recommend cccp</a></span><ul class="commentslist" list-style="none outside none" id="cccp-help"><li id="cccp-htmlcheatsheet" style="display:none"><p><big><big><b>HTML Cheatsheet</b></big></big><br />&lt;b&gt;<b>Bold</b>&lt;/b&gt;\t&lt;i&gt;<i>Italics</i>&lt;/i&gt;\t&lt;del&gt;<del>Deleted</del>&lt;/del&gt;<br />&lt;a href="http://google.com.au"&gt;<a href="http://google.com.au">URL Link</a>&lt;/a&gt;<br />&lt;abbr title="Abbreviation"&gt;<abbr title="Abbreviation">Abbr.</abbr>&lt;/abbr&gt; &lt;- Mouse-over to see text<br />&lt;code&gt;<code>Code text</code>&lt;/code&gt;<br />&lt;blockquote&gt;<blockquote style="blockquote">Block Quote</blockquote>&lt;/blockquote&gt;<br /></p><p><big><big><b>Very Easy Quotes</b></big></big><br />Click on the Quote Icon: <img src="' + em['quotes.png'] + '" /> of each comment.</p><p><big><big><b>Easy Quotes</b></big></big><br />Right-Click on the <href>Permalink</href> link of the post you wish to quote.<br />Select <b>Copy Link Address</b> from the drop-down menu.<br />Paste the link into the Comment Box.<br />Click on the <b>Preview</b> button to get the Post text, and formatted link.<br /></p><big><big><b>Emoticons</b></big></big><br /><p id="cccp-emoticons"></p></li></ul>';
        var cccp_helptoggle = document.getElementById("cccp-helptoggle");
        var cccp_showconfig = document.getElementById("cccp-showconfig");
        var cccp_recommend  = document.getElementById("cccp-recommend");
        cccp_recommend.setAttribute("style", "color:blue");
        cccp_helptoggle.addEventListener("click", helpToggle, false);
        cccp_showconfig.addEventListener("click", showConfigCCCP, false);
        cccp_recommend.addEventListener("click", cccpRecommend, false);
        var cccp_emoticons = document.getElementById("cccp-emoticons");
        cccp_emoticons.innerHTML = '<img src="' + em['face-uncertain.png'] + '" /> :-/<br /><img src="' + em['face-monkey.png'] + '" /> :monkey:<br /><img src="' + em['face-devilish.png'] + '" /> :devil:<br /><img src="' + em['face-angel.png'] + '" /> :angel:<br /><img src="' + em['face-smirk.png'] + '" /> :smirk:<br /><img src="' + em['face-kiss.png'] + '" /> :kiss:<br /><img src="' + em['face-plain.png'] + '" /> :neutral:<br /><img src="' + em['face-surprise.png'] + '" /> :shock:<br /><img src="' + em['face-smile.png'] + '" /> :smile:<br /><img src="' + em['face-worried.png'] + '" /> :???:<br /><img src="' + em['face-cool.png'] + '" /> :cool:<br /><img src="' + em['face-smile-big.png'] + '" /> :grin:<br /><img src="' + em['face-embarrassed.png'] + '" /> :oops:<br /><img src="' + em['face-raspberry.png'] + '" /> :razz:<br /><img src="' + em['face-wink.png'] + '" /> :wink:<br /><img src="' + em['face-crying.png'] + '" /> :cry:<br /><img src="' + em['face-surprise.png'] + '" /> :eek:<br /><img src="' + em['face-laugh.png'] + '" /> :lol:<br /><img src="' + em['face-angry.png'] + '" /> :mad:<br /><img src="' + em['face-sad.png'] + '" /> :sad:<br /><img src="' + em['exclaim.png'] + '" /> :!:<br /><img src="' + em['question.png'] + '" /> :?:<br /><img src="' + em['stock_right.png'] + '" /> :arrow:<br /><img src="' + em['idea.png'] + '" /> :idea:<br />';
    } // Respond section
    // Make sure every blockquote has italics
    Array.filter(document.getElementById('comments-list').getElementsByTagName('li'), function (elem) {
        elem.innerHTML = elem.innerHTML.replace(/([^i][^>])(<blockquote>)/g, '$1<i><blockquote>');
        elem.innerHTML = elem.innerHTML.replace(/(<\/blockquote>)([^<][^\/][^i])/g, '</blockquote></i>$2');
    });
    // Reorder comments; clean up italics run-on; and add VEQ icons
    var i = ul.childNodes.length;
    while (i--) {
        if (ul.childNodes[i] !== null && ul.childNodes[i].innerHTML !== null) {
            var li = document.getElementById(ul.childNodes[i].id);
            if (li !== null) {
                if (li.className !== null) {
                    var style = li.getAttribute("style");
                    if (li.className.indexOf("bypostauthor") > 0) {
                        li.setAttribute("style", "background: #ffb; " + style);
                    } else if (i % 2 === 0) {
                        li.setAttribute("style", "background: #F5F6FC; " + style);
                    }
                }
                if (li.childNodes.length === 1) {
                    var italics = (li.childNodes[0].childNodes.length === 1) ? li.childNodes[0].childNodes[0] : li.childNodes[0];
                    for (jj = 0; jj < italics.childNodes.length; jj++) {
                        li.appendChild(italics.childNodes[jj].cloneNode(1));
                    }
                    li.removeChild(li.childNodes[0]);
//                    li.setAttribute("style", "display: block; ");
                }
                if (respond_on && ! document.getElementById("id", '#' + li.id)) {
                    var a = quotes_lnk.cloneNode(1);
                    a.setAttribute("id", '#' + li.id);
                    a.addEventListener("click", veryEasyQuotes, false);
                    li.getElementsByClassName("comment-meta")[0].appendChild(a);
                }
            }
        }
    }
} // Comments section
// Hide avatars
if (GM_config.get('hideavatars') === true) {
    Array.filter(document.getElementsByClassName('avatar'), function (elem) {
        elem.height = 0;
    });
}
// Stop login nav bar from floating down the page
if (GM_config.get('stopfloat') === true) {
    var topNav = document.getElementById("top-nav");
    if (topNav !== null) {
        topNav.style.position = 'static';
    }
}