// ==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://greasyfork.org/scripts/1884-gm-config/code/GM_config.js?version=4836
// @grant none
// @version 5.12
// ==/UserScript==
// Ver 5.00
// New major version to deal with New, souped-up Crikey
// Ver 5.01
// Added Recommend CCCP
// Ver 5.02
// Added Basic Preview
// Ver 5.03
// Show Avatars Option
// Ver 5.04
// Set Avatars Size Option
// Ver 5.06
// Added Quote Buttons
// Ver 5.07
// Block Quote now working
// Ver 5.08
// Revolution!
// Ver 5.09
// Fixed the error with new thread (no navigation)
// Ver 5.10
// Fixed the Reccommend CCCP not working
// Ver 5.11
// Hilight current page in comment nav.
// Ver 5.12
// Changed to DOMContentLoaded; testing Avatars loaded
var scriptVer = '5.12';
////////////////////////////////////////////////////////////////////////////////
/*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
},
showAvatars: {
label: 'Show Avatars',
type: 'checkbox',
'default': true
},
avatarSize: {
label: 'Avatar Size',
type: 'text',
'default': '48'
}
},
{
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() {
// 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"] = '';
gh["imp.che.png"] = '';
// 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";
}
////////////////////////////////////////////////////////////////////////////////
// If we are logged in, perform additional steps related to commenting
function main() {
initGlobalHash();
if ( document.getElementById("menu-item-543649") ) {
mainFormating();
mainCommenting();
} else {
mainFormating();
}
}
////////////////////////////////////////////////////////////////////////////////
function mainFormating() {
console.log('mainFormating()');
showAvatars();
cleanVerticalSapce();
nicerCommentNavigation();
cccpNavBar();
}
////////////////////////////////////////////////////////////////////////////////
// Clean out wasted vertical space
function cleanVerticalSapce() {
if (GM_config.get('cleanVertSpace') !== true) {return;}
document.getElementsByClassName("article-body__share")[0].style.display = 'none';
document.getElementsByClassName("author ")[0].style.display = 'none';
document.getElementsByClassName("author ")[1].style.display = 'none';
document.getElementsByClassName("article-body__actions")[0].style.display = 'none';
document.getElementsByClassName("block_outlined-desktop")[0].style.display = 'none';
document.getElementsByClassName("block_outlined-desktop")[1].style.display = 'none';
document.getElementsByClassName("footer")[0].style.display = 'none';
}
////////////////////////////////////////////////////////////////////////////////
// Nicer Comment Navigation
function nicerCommentNavigation() {
if ( ! document.getElementById("comment-nav-above") ) {return;}
var nc = document.getElementsByClassName("info_linear-mobile")[0].getElementsByTagName("a")[1].innerHTML.replace(/ .*/,'');
var pg = Math.ceil( parseInt(nc) / 50 );
var href = window.location.href.replace(/\/comment-page-[0-9]+/,'').replace(/\/#comments/,'');
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";
}
d.innerHTML = j;
nl.appendChild(div);
}
}
}
////////////////////////////////////////////////////////////////////////////////
function mainCommenting() {
var comments = document.getElementById("comments");
if (! comments) {return;}
addCCCPLinks();
var respond = document.getElementById("respond");
if (! respond) {return;}
addSubmitButton();
addPreviewButton();
addQuoteButtons();
cloneTextBox();
}
////////////////////////////////////////////////////////////////////////////////
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);
text = text.replace(/\[/g,'<blockquote><p>').replace(/\]/g,'</blockquote></p>');
text = text.replace(/<p>[\s]*<p>/g,'<p>').replace(/<\/p>[\s]*<\/p>/g,'</p>');
console.log(text);
comment.value = text;
}
////////////////////////////////////////////////////////////////////////////////
function cccpSubmit() {
copyCCCPTextBox();
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() {
// 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
var hlp = document.createElement("a");
document.getElementById("cccp_hlp").appendChild(hlp);
hlp.innerHTML = "CCCP Help";
hlp.addEventListener("click", helpCCCP, false);
}
////////////////////////////////////////////////////////////////////////////////
// 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 helpCCCP() {
alert('No Help for you! (TODO)');
}
////////////////////////////////////////////////////////////////////////////////
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 showAvatars() {
if (GM_config.get('showAvatars') === true) {
var sz = parseInt(GM_config.get('avatarSize'));
Array.filter(document.getElementsByClassName('avatar'), function (e) {
if ( e.complete ) {
showAvatar(e, sz);
} else {
e.onload = function() { showAvatar(this, sz); };
}
});
}
}
//----------------------------------------------------------------------------//
function showAvatar(img, sz) {
img.height = sz;
img.width = sz;
img.setAttribute("style","display: inline");
if (img.src.match(/Author-William-Bowe/)) {
img.src = gh["img.che.png"];
}
}
////////////////////////////////////////////////////////////////////////////////
function addQuoteButtons() {
Array.filter(document.getElementsByClassName('vcard'), function (e) {
var img = document.createElement('img');
img.setAttribute("src", gh["img.quotes.png"]);
var a = document.createElement('a');
a.appendChild(img);
e.appendChild(a);
var id = e.parentNode.parentNode.parentNode.id;
a.addEventListener("click", function() {getQuote(id);} , false);
});
}
////////////////////////////////////////////////////////////////////////////////
function getQuote(id) {
var comment = document.getElementById("cccp-comment");
if (! comment ) {return;}
var cid = document.getElementById(id);
var ctn = cid.getElementsByClassName("comment-content")[0];
var author = cid.getElementsByClassName("fn")[0].innerHTML;
var link = cid.getElementsByClassName("comment-metadata")[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 += '[';
comment.value += ctn.innerHTML.replace(/^[\s]*/,'').replace(/[\s]*$/,'');
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);
}