CrikeyCleanCommentPreview

Fix Comment Preview box on crikey blogs

// ==UserScript==
// @name           CrikeyCleanCommentPreview
// @author         Musrum
// @namespace      2
// @description    Fix Comment Preview box on crikey blogs
// @include        https://blogs.crikey.com.au/*
// @include        https://www.crikey.com.au/*
// @include        http://blogs.crikey.com.au/*
// @include        http://www.crikey.com.au/*
// @exclude        http://www.crikey.com.au/*.gif
// @exclude        https://www.crikey.com.au/*.gif
// @require 	   https://greasyfork.org/scripts/1884-gm-config/code/GM_config.js?version=4836
// @grant          none
// @version        5.51
// ==/UserScript==
var scriptVer = '5.51';
// Ver 5.51
// Fixed issue with single page comment lists
// Ver 5.50
// Fixed Help Screen, Fixed Emoji support, Added check for uneven blockquotes
// Ver 5.49
// Fixed Basic UNICODE Emoji support not allowing quotes
// Ver 5.48
// Basic UNICODE Emoji support
////////////////////////////////////////////////////////////////////////////////
/*jslint browser: true */
/*global GM_config, GM_registerMenuCommand */
////////////////////////////////////////////////////////////////////////////////
//don't run in iframes
if (window.top !== window.self) {return;}
////////////////////////////////////////////////////////////////////////////////
// Config settings dialog
GM_config.storage = 'Crikey Clean Comment Preview';
GM_config.init('Crikey Clean Comment Preview - Ver ' + scriptVer,
    {
        cleanVertSpace: {
            label: 'Clean Vertical Space',
            type: 'checkbox',
            'default': true
        },
        addCommentNumbers: {
            label: 'Add Comment Numbers',
            type: 'checkbox',
            'default': true
        },
        betterCommentNavigation: {
            label: 'Better Comment Numbers',
            type: 'checkbox',
            'default': true
        },
        commentsPerPage: {
            label: 'Comments Per Page',
            type: 'text',
            'default': '50'
        },
        wordsFinder: {
            label: 'Word/phrase Finder (use the format: \'Username|NBN\')',
            type: 'text',
            'default': '',
            size:50
		}
    },
    {
        open: function() {
            GM_config.addBorder(); // add a fancy border
            GM_config.resizeFrame('200px','300px'); // resize the config window
        }
    },
    {
        save: function () { location.reload(); } // reload the page when configuration was changed
    }
    );
////////////////////////////////////////////////////////////////////////////////
function showConfigCCCP() {GM_config.open();}
////////////////////////////////////////////////////////////////////////////////
//  Declare Global Hashtable
var gh = [];
////////////////////////////////////////////////////////////////////////////////
// Run if DOM is ready, otherwise add a listerner to wait
if (document.readyState == "complete" || document.readyState == "loaded" || document.readyState == "interactive") {main();}
else {window.addEventListener('DOMContentLoaded',function(e){main();});}
////////////////////////////////////////////////////////////////////////////////
//  Init Global Hash Variables
function initGlobalHash() {
    console.log('cccp: initGlobalHash()');
    // RegExpr
    gh["rxp.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');
    // Images
    gh["img.quotes.png"]    = '';
    // Avatar cosplay
    gh["avt.william bowe~"] = '';
    gh["avt.musrum"]        = '';
    //  URLS
    gh["url.Firefox"]       = "http://www.mozilla.com/en-US/firefox/personal.html";
    gh["url.Greasemonkey"]  = "https://addons.mozilla.org/en-US/firefox/addon/748";
    gh["url.Google Chrome"] = "http://www.google.com/chrome";
    gh["url.Tampermonkey"]  = "https://chrome.google.com/webstore/detail/dhdgffkkebhmkfjojejmpbldmpobfkfo";
    gh["url.cccp"]          = "https://greasyfork.org/en/scripts/18677-crikeycleancommentpreview";
    // Filters
    gh["flt.words"]         = GM_config.get('wordsFinder').toLowerCase();
    // Elements
    gh["emt.nav"]           = '<nav id="comment-nav-above" class="navigation comment-navigation" role="navigation"><h2 class="screen-reader-text">Comment navigation</h2><div class="nav-links"></div><!-- .nav-links --></nav>';
}
////////////////////////////////////////////////////////////////////////////////
//    If we are logged in, perform additional steps related to commenting
function main() {
    console.log('cccp: main()');
    initGlobalHash();
    if ( loginStatus() === 1 ) {
        mainFormating();
        mainCommenting();
     } else {
        mainFormating();
    }
}
////////////////////////////////////////////////////////////////////////////////
//    If we are logged in, perform additional steps related to commenting
function loginStatus() {
    var ul = document.getElementById("menu-top_right_logged_out_view");
    var ls = (! ul) ? 1 : 0;
    console.log('cccp: loginStatus() = ' + ls);
    return ls;
}
////////////////////////////////////////////////////////////////////////////////
function loginStatusOld2() {
    var li = document.getElementById("menu-item-543624");
    var ls = (li.firstChild.innerHTML === "Sign out") ? 1 : 0;
    console.log('cccp: loginStatus() = ' + ls);
    return ls;
}
////////////////////////////////////////////////////////////////////////////////
function loginStatusOld() {
    var pg = document.getElementsByClassName("page")[0];
    var p  = pg.getElementsByTagName("p");
    var ls = 0;
    for (var i = 0; i < p.length; i++ ) {
        if ( p[i].parentNode === pg && p[i].innerHTML.match(/User login status : /) ) {
            ls = parseInt(p[i].innerHTML.replace(/User login status : /,''));
            break;
        }
    }
    return ls;
}
////////////////////////////////////////////////////////////////////////////////
function mainFormating() {
    console.log('cccp: mainFormating()');
//    ensureCommentOrder();
    getPageNos();
    tweakAvatars();
    cleanVerticalSpace();
    fixDuplicateNavigationIds();
    addCommentNumbers();
    betterCommentNavigation();
    cccpNavBar();
    wordsFinder();
}
////////////////////////////////////////////////////////////////////////////////
function mainCommenting() {
    console.log('cccp: mainCommenting()');
    if (document.getElementById("comments")) {
        addCCCPLinks();
    }
    if (document.getElementById("respond")) {
        if (document.getElementsByClassName("comment-respond").length > 1 ) {
            removeExtraCommentBox();
        } else {
            moveCommentBox();
        }
//      removeReplyButtons();
        addSubmitButton();
        addPreviewButton();
        addQuoteButtons();
        cloneTextBox();
    }
}
////////////////////////////////////////////////////////////////////////////////
function fixDuplicateNavigationIds() {
    console.log('cccp: fixDuplicateNavigationIds()');
    var cna = document.querySelectorAll("#comment-nav-above")
    console.log('cccp: cna.length = ' + cna.length);
    if ( cna.length === 2 ) {
        cna[1].id = 'comment-nav-below';
    }
}
////////////////////////////////////////////////////////////////////////////////
function removeExtraCommentBox() {
    console.log('cccp: removeExtraCommentBox()');
    var resp = document.querySelectorAll("#respond")
    console.log('cccp: respond.length = ' + resp.length);
    if ( resp.length < 2 ) {return;}
    resp[0].parentNode.removeChild(resp[0]);
}
////////////////////////////////////////////////////////////////////////////////
function ensureCommentOrder() {
    var ol = document.getElementsByClassName("comment-list")[0];
    var cl = ol.getElementsByTagName("li");
    if (cl.length < 2 ) {return;}
    if ( parseInt(cl[0].id.replace('comment-','')) < parseInt(cl[1].id.replace('comment-','')) ) {return;}
    // List out of order: reverse order now
    for ( var i = cl.length-2; i >= 0; i-- ) {
        ol.appendChild(cl[i]);
    }
}
////////////////////////////////////////////////////////////////////////////////
function getPageNos() {
    console.log('cccp: getPageNos()');
    var nc = parseInt(document.getElementById("block_disqus").getElementsByClassName("block__label")[0].getElementsByClassName("info")[0].getElementsByClassName("info__title")[0].innerHTML.replace(/ comments/,''));
    console.log('cccp: nc = ' + nc);
    gh["pg.cno"] = nc;
    var cpp = parseInt(GM_config.get('commentsPerPage'));
    gh["pg.max"] = Math.ceil( parseInt(nc) / cpp );
    if ( window.location.href.match(/comment-page/) ) {
            gh["pg.cur"] = parseInt( window.location.href.replace(/.*\/comment-page-([0-9]+).*/,'$1') );
    } else {
            gh["pg.cur"] =  1; //gh["pg.max"];
    }
}
////////////////////////////////////////////////////////////////////////////////
function addCommentNumbers() {
    console.log('cccp: addCommentNumbers()');
    if (GM_config.get('addCommentNumbers') !== true) {return;}
    if (document.getElementsByClassName("comment-list").length === 0) {return;}

    var ol = document.getElementsByClassName("comment-list");
    var cpp = parseInt(GM_config.get('commentsPerPage'));
    var cno = gh["pg.cno"];
    var cur = gh["pg.cur"];
    var max = gh["pg.max"];
    
    var cl = ol[0].getElementsByTagName("li");
    console.log('cccp: cl.length = ' + cl.length);
    console.log('cccp: pg.cno = ' + cno);
    console.log('cccp: pg.cur = ' + cur);
    console.log('cccp: pg.max = ' + max);
    
    if (cl.length === 0 ) {return;}
//    var cn = cno - ((max-cur)*cpp);
    var cn = 1 + ((cur-1)*cpp);
    for ( var i = 0; i < cl.length; i++ ) {
        var a = cl[i].getElementsByClassName("commentmetadata")[0].getElementsByTagName("a");
        if ( a.length === 0 ) {continue;}
        if (a[0].firstChild) {
            var sp = document.createElement("span");
            sp.innerHTML = '#' + (cn+i) + ' ';
            a[0].insertBefore(sp,a[0].firstChild);
        } else {
            a[0].innerHTML  = '#' + (cn+i) + ' ' + a[0].innerHTML.replace(/^\s*/,'');
        }
    }
}
////////////////////////////////////////////////////////////////////////////////
function wordsFinder() {
    console.log('cccp: wordsFinder()');
    var wfon = ( gh["flt.words"  ] !== null && gh["flt.words"  ].replace(/\s*/,'').length > 0 );
    if ( ! wfon ) {return;}
	if ( document.getElementsByClassName("comment-list").length === 0 ) {return;}

    var wf = gh["flt.words"];
    var ol = document.getElementsByClassName("comment-list")[0];
    var cl = ol.getElementsByTagName("li");
    if (cl.length === 0 ) {return;}

    var ci = [];
    for ( var i = 0; i < cl.length; i++) {
	    if ( cl[i].getElementsByClassName('comment-content')[0].innerHTML.toLowerCase().match(wf) ) {
           ci.push(i);
        }
    }

    if (ci.length === 0 ) {return;}
    console.log('ci.length: ' + ci.length);
    // Create the link to the first found comment
    if (document.getElementById("comment-nav-below")) {
        wordsFinderTag('match',cl,ci,0,document.getElementById("comment-nav-below").getElementsByClassName("comment-notes")[0]);
    }
    if (document.getElementById("comment-nav-above")) {
        wordsFinderTag('match',cl,ci,0,document.getElementById("comment-nav-above").getElementsByClassName("comment-notes")[0]);
    }
    if (ci.length === 1 ) {return;}
    // Create additional links if required
    for ( i = 0; i < ci.length; i++ ) {
        var j   = ( i === ci.length-1 ) ? 0 : (i+1);
        var txt = ( j === 0 ) ? 'first' : 'next';
        wordsFinderTag(txt,cl,ci,j,cl[ci[i]].getElementsByClassName("comment-notes")[0]);
    }

}
//----------------------------------------------------------------------------//
function wordsFinderTag(txt,cl,ci,i,e) {
    var a = document.createElement('a');
    a.href = cl[ci[i]].getElementsByClassName("comment-meta")[0].getElementsByTagName("a")[0].href;
//  console.log(JSON.stringify(cc[ci[i]].parentNode.getElementsByClassName("comment-metadata")[0].getElementsByTagName("a")[0].href));
    var text = txt + ': ' + (i+1) + '/' + ci.length + ' ' + getAuthor(cl[ci[i]].firstChild);
    a.innerHTML = text;
    e.appendChild(a);
}
////////////////////////////////////////////////////////////////////////////////
function  getAuthor(e) {
    var author;
    var a = e.parentNode.getElementsByClassName('fn')[0];
    if ( a.firstChild.innerHTML === undefined ) {
        author = a.innerHTML;
    } else {
        author = a.firstChild.innerHTML;
    }
    return author.toLowerCase();
}
////////////////////////////////////////////////////////////////////////////////
//    Clean out wasted vertical space
function cleanVerticalSpace() {
    console.log('cccp: cleanVerticalSpace()');
    if (GM_config.get('cleanVertSpace') !== true) {return;}
    cleanAll("article-body__share");
    cleanAll("author");
    cleanAll("block_outlined-desktop");
    cleanAll("footer");
}
////////////////////////////////////////////////////////////////////////////////
function cleanAll(className) {
    var e = document.getElementsByClassName(className);
    if ( ! e ) {return;}
    for ( var i = 0; i < e.length; i++ ) {
        e[i].style.display = 'none';
    }
}
////////////////////////////////////////////////////////////////////////////////
//    Better Comment Navigation
function betterCommentNavigation() {
    console.log('cccp: betterCommentNavigation()');
//    if (GM_config.get('betterCommentNavigation') !== true) {return;}
    if ( ! document.getElementById("comment-nav-above") ) {return;}
//    var cpp = parseInt(GM_config.get('commentsPerPage'));

    //  Add Right-Side div for comment-notes (wordFinder) to bottom Nav Bar heading
    addRightSideDiv(document.getElementById("comment-nav-below").getElementsByTagName("h2")[0],"comment-notes");
    addRightSideDiv(document.getElementById("comment-nav-above").getElementsByTagName("h2")[0],"comment-notes");
    return;
    
    var nc = document.getElementsByClassName("info_linear-mobile")[0].getElementsByTagName("a")[1].innerHTML.replace(/ .*/,'');
    var pg = Math.ceil( parseInt(nc) / cpp );
    var href = window.location.href.replace(/\/comment-page-[0-9]+/,'').replace(/\/#comment?s/,'').replace(/\/#comment-[0-9]+/,'').replace(/\/$/,'');;
    var cp = pg;
    if ( window.location.href.match(/comment-page/) ) {
        cp = parseInt( window.location.href.replace(/.*\/comment-page-([0-9]+).*/,'$1') );
    }

    var nav = ["above","below"];
    for (var i = 0; i < nav.length; i++) {
        var nl = document.getElementById("comment-nav-" + nav[i]).getElementsByClassName("nav-links")[0];
        nl.innerHTML = '';
        for (var j = 1; j <= pg; j++) {
         var div = document.createElement('div');
         var a = document.createElement('a');
         div.setAttribute("class","nav-next");
         div.appendChild(a);
         a.href = href + '/comment-page-' + j + '/#comments';
         var d = document.createElement('div');
         a.appendChild(d);
         if ( j == cp ) {
             d.style.backgroundColor = "lightblue";
             a.href="javascript:location.reload();";
         }
         d.innerHTML = j;
         nl.appendChild(div);
        }
    }
}
////////////////////////////////////////////////////////////////////////////////
function moveCommentBox() {
    var e = document.getElementById("respond");
    var n = document.getElementById("cccp_nav");
    n.parentNode.insertBefore(e, n);
}
////////////////////////////////////////////////////////////////////////////////
function removeReplyButtons() {
    if (GM_config.get('removeReplyBtns') === false) {return;}
    var replys = document.getElementsByClassName('reply');
    for ( var i = 0; i < replys.length; i++ ) {
        replys[i].setAttribute("style","display: none");
    }
}
////////////////////////////////////////////////////////////////////////////////
function copyCCCPTextBox() {
    var comment = document.getElementById("comment");
    if (! comment ) {return;}
    var ctbx = document.getElementById("cccp-comment");
    if (! ctbx ) {return;}
    var text = ctbx.value;
//    console.log(text);

    // Add balancing open/close square brackets if required.
    var cOpen = (text.match(/\[/g) || []).length;
    var cClse = (text.match(/\]/g) || []).length;
    var pre = (cOpen < cClse) ? "[" : "";
    var pst = (cOpen > cClse) ? "]" : "";
    for ( var i = 0; i <= Math.abs(cOpen-cClse) ; i++ ) {
        text = pre + text + pst;
    }

    text = text.replace(/\[/g,'<blockquote>').replace(/\]/g,'</blockquote>');
    text = text.replace(/<p>[\s]*<p>/g,'<p>').replace(/<\/p>[\s]*<\/p>/g,'</p>');

    text = text.replace(/:smile:/g,'?');
    text = text.replace(/:grin:/g,'?');
    text = text.replace(/:sad:/g,'?');
    text = text.replace(/:eek:/g,'?');
    text = text.replace(/:shock:/g,'?');
    text = text.replace(/:\?\?\?:/g,'?');
    text = text.replace(/:cool:/g,'?');
    text = text.replace(/:mad:/g,'?');
    text = text.replace(/:razz:/g,'?');
    text = text.replace(/:neutral:/g,'?');
    text = text.replace(/:wink:/g,'?');
    text = text.replace(/:lol:/g,'?');

    console.log(text);
    comment.value = text;
}
////////////////////////////////////////////////////////////////////////////////
function finalCCCPTextBox() {
    var comment = document.getElementById("comment");
    if (! comment ) {return;}
    var text = comment.value;
    console.log(text);
    comment.value = text;
}
////////////////////////////////////////////////////////////////////////////////
function unicodeEscape(str) {
  for (var result = '', index = 0, charCode; !isNaN(charCode = str.charCodeAt(index++));) {
    result += '&#' + ('0000' + charCode.toString(16)).slice(-4) + ";";
  }
  return result;
}
////////////////////////////////////////////////////////////////////////////////
function cccpSubmit() {
    copyCCCPTextBox();
    finalCCCPTextBox();
    document.getElementById("submit").click();
}
////////////////////////////////////////////////////////////////////////////////
function addSubmitButton() {
    // Add the cccp Submit Button (hide original)
    var fs = respond.getElementsByClassName("form-submit")[0];
    fs.firstChild.setAttribute("style","display: none");
    var prv = document.createElement("input");
    fs.insertBefore(prv, fs.firstChild);
    prv.id    = "cccp-submit";
    prv.setAttribute("class", "submit");
    prv.setAttribute("type", "button");
    prv.value = "Submit";
    prv.addEventListener("click", cccpSubmit, false);
}
////////////////////////////////////////////////////////////////////////////////
function addPreviewButton() {
    // Add the Preview Button
    var fs = respond.getElementsByClassName("form-submit")[0];
    var sp = document.createElement("span");
    sp.innerHTML = ' ';
    fs.insertBefore(sp, fs.firstChild);
    var prv = document.createElement("input");
    fs.insertBefore(prv, fs.firstChild);
    prv.id    = "preview";
    prv.setAttribute("class", "submit");
    prv.setAttribute("type", "button");
    prv.value = "Preview";
    prv.addEventListener("click", previewCCCP, false);
}
////////////////////////////////////////////////////////////////////////////////
function addCCCPLinks() {

    // Fix for Help Box
    var cnb = document.getElementById("comment-nav-below");
    if ( cnb ) {
        cnb.style.position = 'relative';
        cnb.style.zIndex = 5;
    }

    var cccp_nav = document.getElementById("cccp_nav");

    if (! cccp_nav) {return;}

    cccp_nav.style.position = 'relative';
    cccp_nav.style.zIndex = 5;

    // Add the Recommend CCCP Link
    var rec = document.createElement("a");

    document.getElementById("cccp_rec").appendChild(rec);
    rec.innerHTML = "Recommend CCCP";
    rec.addEventListener("click", recommendCCCP, false);

    // Add the CCCP Help Link
    setupHelpCCCP();

}
////////////////////////////////////////////////////////////////////////////////
//  Create a new cccp_nav <div> with space for help/settings/recommend
function cccpNavBar() {
    var comments = document.getElementById("comments");
    if (! comments) {return;}

    var cccp_nav = document.createElement('div');
    cccp_nav.id = "cccp_nav";
    comments.appendChild(cccp_nav);

    var nav = ["hlp","set","rec"];
    var alg = ["left","center","right"];

    var tbl = document.createElement('table');
    cccp_nav.appendChild(tbl);
    var row = document.createElement('tr');
    tbl.appendChild(row);

    for (var i = 0; i < nav.length; i++) {
        var td = document.createElement('td');
        td.width = '33%';
        row.appendChild(td);
        var dv = document.createElement('div');
        dv.id = 'cccp_' + nav[i];
        dv.style = "text-align: " +  alg[i] + ";";
        td.appendChild(dv);
    }
    // Add the CCCP Settings Link
    var set = document.createElement("a");
    document.getElementById("cccp_set").appendChild(set);
    set.innerHTML = "CCCP Settings";
    set.addEventListener("click", showConfigCCCP, false);
}
////////////////////////////////////////////////////////////////////////////////
function recommendCCCP() {
    var comment = document.getElementById("cccp-comment");
    comment.value += '\nTo use the Crikey Clear Comment Preview script, install in order:\n';
    comment.value += '<a href="' + gh["url.Firefox"] + '">Firefox</a>\n';
    comment.value += '<a href="' + gh["url.Greasemonkey"] + '">Greasemonkey</a>\n';
    comment.value += '<a href="' + gh["url.cccp"] + '">cccp</a>\n';
    comment.value += 'or:\n';
    comment.value += '<a href="' + gh["url.Google Chrome"] + '">Google Chrome</a>\n';
    comment.value += '<a href="' + gh["url.Tampermonkey"] + '">Tampermonkey</a>\n';
    comment.value += '<a href="' + gh["url.cccp"] + '">cccp</a>\n';
}
////////////////////////////////////////////////////////////////////////////////
function setupHelpCCCP() {
    var styleEl = document.createElement('style'), styleSheet;
    document.head.appendChild(styleEl);
    styleSheet = styleEl.sheet;
    styleSheet.insertRule(".cccp-modal {display: none;position: fixed;z-index: 1000;padding-top: 100px;left: 0;top: 0;width: 100%;height: 100%;overflow: auto;background-color: rgb(0,0,0);background-color: rgba(0,0,0,0.4);}", 0);
    styleSheet.insertRule(".cccp-modal-content {background-color: #fefefe;margin: auto;padding: 20px;border: 1px solid #888;width: 80%;}", 0);
    styleSheet.insertRule(".cccp-close {color: #aaaaaa;float: right;font-size: 28px;font-weight: bold;}", 0);
    styleSheet.insertRule(".cccp-close:hover,.cccp-close:focus {color: #000;text-decoration: none;cursor: pointer;}", 0);

    // Add the CCCP Help Link
    var hlp = document.createElement("a");
    document.getElementById("cccp_hlp").appendChild(hlp);
    hlp.innerHTML = "CCCP Help";
    hlp.addEventListener("click", helpCCCP, false);

    var cccp_help = document.createElement('div');
    cccp_help.id = "cccp-help";
    document.getElementById("cccp_hlp").appendChild(cccp_help);
    cccp_help.setAttribute("class", "cccp-modal");
    var div = document.createElement('div');
    div.setAttribute("class", "cccp-modal-content");
    cccp_help.appendChild(div);
    var span = document.createElement('span');
    span.setAttribute("class", "cccp-close");
    span.innerHTML = '&times;';
    div.appendChild(span);
    var p = document.createElement('p');
    div.appendChild(p);
    // When the user clicks on <span> (x), close the modal
    span.onclick = function() {
        cccp_help.style.display = "none";
    }
    // When the user clicks anywhere outside of the modal, close it
    window.onclick = function(event) {
        if (event.target == cccp_help) {
            cccp_help.style.display = "none";
        }
    }

    var text = '';
    text += '<b>Emoticons:</b></br>';
    text += '<table>';
    text += '<tr><td>:smile:</td><td>?</td>';
    text += '<td>:grin:</td><td>?</td>';
    text += '<td>:sad:</td><td>?</td>';
    text += '<td>:eek:</td><td>?</td></tr>';
    text += '<tr><td>:shock:</td><td>?</td>';
    text += '<td>:???:</td><td>?</td>';
    text += '<td>:cool:</td><td>?</td>';
    text += '<td>:mad:</td><td>?</td></tr>';
    text += '<tr><td>:razz:</td><td>?</td>';
    text += '<td>:neutral:</td><td>?</td>';
    text += '<td>:wink:</td><td>?</td>';
    text += '<td>:lol:</td><td>?</td></tr>';
    text += '</table>';

    p.innerHTML = text;

}
////////////////////////////////////////////////////////////////////////////////
function helpCCCP() {
    document.getElementById('cccp-help').style.display = "block";
    return;


}
////////////////////////////////////////////////////////////////////////////////
function previewCCCP() {
    var comments = document.getElementById("comments");
    if (! comments ) {return;}
    var respond = document.getElementById("respond");
    if (! respond ) {return;}
    var comment = document.getElementById("comment");
    if (! comment ) {return;}
    copyCCCPTextBox();
    var preview;
    preview = document.getElementById("cccp_preview");
    if (! preview ) {
        var ol = document.createElement('ol');
        ol.setAttribute("class", "comment-list");
        comments.insertBefore(ol, document.getElementById("cccp_nav"));
        var li = document.createElement('li');
        li.setAttribute("class", "comment byuser even thread-even depth-1");
        ol.appendChild(li);
        var ar = document.createElement('article');
        ar.setAttribute("class", "comment-body");
        li.appendChild(ar);
        preview = document.createElement('div');
        preview.id = "cccp_preview";
        preview.setAttribute("class", "comment-content");
        ar.appendChild(preview);
    }
    preview.innerHTML = comment.value.replace(/\n/g,'<br>');
}
////////////////////////////////////////////////////////////////////////////////
function tweakAvatars() {
    console.log('cccp: tweakAvatars()');
    if ( document.getElementsByClassName("comment-list").length === 0) {return;}
    var cli = document.getElementsByClassName("comment-list")[0].getElementsByTagName("li");
    console.log('cccp: cli.length = ' + cli.length);

    var sar = GM_config.get('shiftAvatarRight');

    console.log('cccp: sar = ' + sar);

    for ( var i = 0; i < cli.length; i++ ) {
        var cc = cli[i].getElementsByClassName("comment-content-right")[0];
        var div = document.createElement('div');
        div.setAttribute("style", "width: 100%; display: inline-block;");
        div.setAttribute("class", "comment-content-right-top");
        cc.insertBefore(div,cc.firstChild);
        var tbl1 = document.createElement('table');
        div.appendChild(tbl1);
        tbl1.setAttribute("style", "margin-left: auto; margin-right: auto");
        var tr1 = document.createElement('tr');
        tr1.setAttribute("style", "width: 100%; padding: 0px;");
        tbl1.appendChild(tr1);
        var c11 = document.createElement('td');
        c11.setAttribute("style", "width: 91px; padding: 0px;");
        var c12 = document.createElement('td');
        tr1.appendChild(c11);
        tr1.appendChild(c12);

        var ca = cli[i].getElementsByClassName("comment-author")[0];
        c11.appendChild(ca);

        var tbl2 = document.createElement('table');
        c12.appendChild(tbl2);
        tbl2.setAttribute("style", "margin-left: auto; margin-right: auto");
        var tr2 = document.createElement('tr');
        tr2.setAttribute("style", "width: 100%; padding: 0px;");
        tbl2.appendChild(tr2);
        var c21 = document.createElement('td');
        c21.setAttribute("style", "width: 60%; padding: 0px;");
        var c22 = document.createElement('td');
        c22.setAttribute("style", "width: 40%; padding: 0px;");
        tr2.appendChild(c21);
        tr2.appendChild(c22);

        var fn = cli[i].getElementsByClassName("fn")[0];
        c21.appendChild(fn);
        var cm = cli[i].getElementsByClassName("comment-meta")[0];
        c21.appendChild(cm);

        var div2 = document.createElement('div');
        c22.appendChild(div2);
        div2.setAttribute("style", "text-align:right");
        div2.setAttribute("class", "comment-notes");
        
        var author = getAuthor(cli[i].firstChild);
//      console.log(author);
        if ( gh['avt.' + author] !== undefined ) {
            var img = cli[i].getElementsByClassName('avatar')[0];
            img.src = gh['avt.' + author];
        }
    }
}
//----------------------------------------------------------------------------//
function addRightSideDiv(e,className) {
    var div = document.createElement('div');
    div.setAttribute("style", "width: 100%; display: inline-block;");
    e.parentNode.insertBefore(div,e);
    var tbl = document.createElement('table');
    div.appendChild(tbl);
    var tr = document.createElement('tr');
    tr.setAttribute("style", "width: 100%; padding: 0px;");
    tbl.appendChild(tr);
    var c1 = document.createElement('td');
    c1.setAttribute("style", "width: 50%; padding: 0px;");
    tr.appendChild(c1);
    c1.appendChild(e);
    var c2 = document.createElement('td');
    tr.appendChild(c2);
    c2.setAttribute("style", "width: 50%; padding: 0px; text-align: right;");
    var div = document.createElement('div');
    c2.appendChild(div);
    div.setAttribute("style", "width: 100%; text-align:right");
    div.setAttribute("class", className);
}
////////////////////////////////////////////////////////////////////////////////
function addQuoteButtons() {
   var cb = document.getElementsByClassName('comment-body');
   for ( var i = 0; i < cb.length; i++ ) {
       (function (i) {
           var e = cb[i].getElementsByClassName('fn')[0];
           var img = document.createElement('img');
           img.setAttribute("src", gh["img.quotes.png"]);
           var a = document.createElement('a');
           a.appendChild(img);
           e.parentNode.insertBefore(a,e.nextSibling);
           var cid = cb[i].id;
//           console.log(cid);
           a.addEventListener("click", function() {getQuote(cid);} , false);
       })(i);
   }
}
////////////////////////////////////////////////////////////////////////////////
function getQuote(cid) {
    var comment = document.getElementById("cccp-comment");
    if (! comment ) {return;}
//    console.log(cid);
    var cmt = document.getElementById(cid);
    var ctn = cmt.getElementsByClassName("comment-content")[0];
    var author = getAuthor(cmt);
    var link = cmt.getElementsByClassName("commentmetadata")[0];
    comment.value += '<p><b>'+ author + '</b> @ ' + link.innerHTML.replace(/^\s*/,'').replace(/\s*$/,'').replace(/<br>/g,'').replace(/\n/g,'').replace(/[\s]+/g,' ') + '</p>\n';
    comment.value += '[';
    var quote = ctn.innerHTML.replace(/^[\s]*/,'').replace(/[\s]*$/,'');
    comment.value += quote;
    comment.value += ']\n';
}
////////////////////////////////////////////////////////////////////////////////
//  To enable easy quotes we need to hide the comment text box and submit button
//  And then create clones of these
function cloneTextBox() {
    var comment = document.getElementById("comment");
    if (! comment ) {return;}
    var ctbx = comment.cloneNode(true);
    ctbx.id = 'cccp-comment';
    comment.setAttribute("style","display: none");
    comment.parentNode.insertBefore(ctbx, comment);
}