Last active
December 16, 2015 17:39
-
-
Save h-collector/5471519 to your computer and use it in GitHub Desktop.
Script for 2chan.net futaba board, to use with Opera Scripter extension .
Moved to https://github.com/h-collector/2chan-utils (after 1.0.6)
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// ==UserScript== | |
// @author h-collector <[email protected]> | |
// @name 2chan-utils | |
// @namespace https://gist.github.com/h-collector/ | |
// @description Script for 2chan.net futaba board and archived threads on yakumo-family.com adding useful functions | |
// @include http://*.2chan.net/* | |
// @exclude http://*.2chan.net/*/src/* | |
// @include http://yakumo-family.com/fdat/* | |
// @include http://yakumo-family.com/f*dat/* | |
// @include http://www.yakumo-family.com/fdat/* | |
// @include http://www.yakumo-family.com/f*dat/* | |
// @require https://ajax.googleapis.com/ajax/libs/jquery/1/jquery.min.js | |
// @homepageURL https://github.com/h-collector/2chan-utils | |
// @update https://raw.github.com/h-collector/2chan-utils/master/src/includes/2chan-utils.js | |
// @history 1.0.6 add cached links count, some refactoring, added constriction on max image height/width | |
// @history 1.0.5 fix: userjs @include/exclude/require declarations (overlay on image page, no www on yakumo-family) | |
// @history 1.0.4 fix: clicking on image while loading open link, a little code reformat, added more icons | |
// @history 1.0.3 minor changes | |
// @history 1.0.2 fixed and improved sidebar, added goto link, fixed autoscroll | |
// @history 1.0.1 partially fix sideeffect of reverse node traversal on sidebar | |
// @history 1.0 initial release | |
// @version 1.0.6 | |
// @date 2013-05-26 | |
// @license GPL | |
// ==/UserScript== | |
// Features: | |
// - inline image expansion, | |
// - inline thread expansion, | |
// - expose mailto hidden messages | |
// - single post anchoring and post highlight , | |
// - futalog and axfc uploader autolinking with highlight and unique links in sidebar | |
// - page autorefresh on new content, | |
// - removing ads. | |
// To use with eg. opera scripter (tested), or using converter to oex on opera (tested) | |
// Should be used in domready event, didn't really try on greasemonkey but should work | |
(function(){ | |
if (window.document.readyState == 'complete'){ | |
init(); | |
} else { | |
window.addEventListener('DOMContentLoaded', init, false); | |
} | |
function init(){ | |
//just in odd case | |
if (!("console" in window)) { | |
var names = ["log", "error", "time", "timeEnd"]; | |
window.console = {}; | |
for (var i = 0, len = names.length; i < len; ++i) | |
window.console[names[i]] = function(){} | |
} | |
// Add jQuery if not avalible | |
if (typeof window.jQuery == 'undefined') { | |
var head = document.getElementsByTagName('head')[0] || document.documentElement, | |
script = document.createElement('script'); | |
script.src = 'http://ajax.googleapis.com/ajax/libs/jquery/1/jquery.min.js'; | |
script.type = 'text/javascript'; | |
script.onload = function(){ | |
run(window.jQuery) | |
}; | |
head.appendChild(script); | |
} else { | |
run(window.jQuery) | |
} | |
} | |
function run($){ | |
var timeit = true, | |
optConstrictWidth = true, | |
optConstrictHeight = true, | |
optLinkifyAxfc = true, | |
optLinkifyFutaba = true, | |
optLinkifyPosts = true; | |
timeit && console.time("2chan-utils"); | |
console.log('jQuery version: ' + $().jquery + ' Script version: 1.0.6'); | |
var windowWidth, | |
windowHeight, | |
$window = $(window); | |
$window.resize(function() { | |
windowWidth = $window.width(), | |
windowHeight = $window.height(); | |
}).triggerHandler('resize'); | |
//simple counter class | |
function Counter(options){ | |
this.count = options.count || 60; | |
this.tick = options.count || 60; | |
this.interval = options.interval || 1000; | |
this.ontick = options.ontick || function(){}; | |
this.oncomplete = options.oncomplete || function(){}; | |
} | |
Counter.prototype = { | |
id : undefined, | |
isRunning: function(){ return this.id !== undefined }, | |
stop : function(){ this.id = clearInterval(this.id) }, | |
start : function(ticktock){ | |
var self = this; | |
this.id = setInterval(ticktock || function() { | |
self.ontick(--self.tick); | |
if (self.tick <= 0) { | |
self.tick = self.count; | |
self.oncomplete(self.count) | |
} | |
}, this.interval) | |
} | |
} | |
//highlighter | |
$.fn.highlight = function() { | |
return $(this).each(function () { | |
var $el = $(this); | |
$('<div/>', {'class': 'highlight'}) | |
.width( $el.outerWidth()) | |
.height($el.outerHeight()) | |
.css({ | |
left: $el.offset().left, | |
top: $el.offset().top | |
}) | |
.appendTo('body') | |
.fadeOut(1000) | |
.queue(function () { $(this).remove() }) | |
}) | |
} | |
String.prototype.replaceArray = function(find, replace) { | |
var replaceString = this; | |
for (var i = 0, len = find.length; i < len; i++) | |
replaceString = replaceString.replace(find[i], replace[i]); | |
return replaceString | |
}; | |
//search and replace text content | |
$.fn.searchAndReplace = function(pattern, replacement) { | |
if (this.length === 0 || !pattern || !replacement) | |
return this; | |
var isArray = $.isArray(pattern), | |
tempHolder = document.createElement('span'); | |
var innerSearchAndReplace = function(node, pattern, replacement) { | |
if (node.nodeType === 3) { | |
var parent = node.parentNode, | |
oldText = node.nodeValue; | |
tempHolder.innerHTML = isArray | |
? oldText.replaceArray(pattern,replacement) | |
: oldText.replace(pattern,replacement); | |
if (oldText === tempHolder.innerHTML) | |
return; | |
var len = tempHolder.childNodes.length; | |
if (len > 1){ | |
while (len--) | |
parent.insertBefore(tempHolder.firstChild, node); | |
parent.removeChild(node); | |
} else { | |
parent.replaceChild(tempHolder.firstChild, node); | |
} | |
} else if (node.nodeType === 1 && node.childNodes && !/(script|style)/i.test(node.tagName)) { | |
for (var i = node.childNodes.length - 1; i >= 0; --i) | |
innerSearchAndReplace(node.childNodes[i], pattern, replacement) | |
} | |
}; | |
return this.each(function() { | |
innerSearchAndReplace(this, pattern, replacement); | |
}); | |
}; | |
//add styles | |
$('<style type="text/css">' | |
+'a.del { color: #222}' | |
+'a.del:hover { color: #888}' | |
+'td { border-radius: 10px}' | |
+'hr { clear:both;}' | |
+'.expand { background: url() center left no-repeat; }' | |
+'.expanding { background: url() left center no-repeat; }' | |
+'td.highlight { background: #eba; }' | |
+'div.highlight{ background: #ff9; position: absolute; opacity: 0.7; z-index: 1000; }' | |
+'a.postanchor { margin: 0 2px; }' | |
+'a.axfc { background: #eba; }' | |
+'a.futalog { background: #0e0; }' | |
+'a.axfc,' | |
+'a.futalog,' | |
+'#sidebar a { display:inline; display:inline-block; padding: 0 4px; ' | |
+'text-decoration:none; border-radius:4px;}' | |
+'#sidebar { padding: 4px; position: fixed; right: 0; ' | |
+'overflow: auto; ' | |
+'border: 1px solid #a08070; }' | |
+'#sidebar div { text-align:center; font-weight:bold; }' | |
+'#sidebar ul { padding:0; margin:0; text-align:left; }' | |
+'#sidebar li,' | |
+'.collapse { background: url() center left no-repeat; }' | |
+'#sidebar li,' | |
+'.expand, .expanding, .collapse ' | |
+'{ padding:0; margin:0; padding-left: 20px; cursor:pointer; }' | |
+'#sidebar a { display:block; height: 22px;}' | |
+'#sidebar li:hover a, a.highlight ' | |
+'{ background: #ff0;}' | |
+'#stickynav { background: #eba; text-align:center; ' | |
+'position: fixed; top: 50px; right: 10px; }' | |
+'.pointer { background: #a00; padding:1px; margin:1px; cursor:pointer; ' | |
+'display:inline-block; width:14px; height:14px; ' | |
+'text-decoration:none; color:#fff; font-size:14px; font-weight:900; ' | |
+'border:1px solid #000; border-radius:4px; }' | |
+'.active { color:#f00; }' | |
+'.resizeable { width:auto; height:auto; }' | |
+'.loading { opacity: 0.5; }' | |
+'.overlay-parent { position: relative; display:block; float:left; }' | |
+'.overlay { position:absolute; z-index:1000; left:20px; opacity: 0.5;' | |
+'background: #00f url() center center no-repeat;}' | |
+'.olinfo { padding: 4px; background: #00f; color: #fff}' | |
+'.fullimg { border: 1px solid #f00; }' | |
+'.loaded { border: 1px dashed #a08070; }' | |
+'.secret { border: 1px dashed #a08070; }' | |
+'#autoscroll { display:block; margin:0 2px; width: 34px; ' | |
+'border: 1px solid #a08070; }' | |
+'</style>').appendTo('head'); | |
//utalog links | |
var futalog = { | |
su : 'nijibox5.com/futabafiles/tubu/src/', /* 12 */ | |
sa : 'nijibox6.com/futabafiles/001/src/', /* 12 */ | |
ss : 'nijibox5.com/futabafiles/kobin/src/',/* 24 */ | |
sq : 'nijibox6.com/futabafiles/mid/src/auth.redirect.php?', /* 48 key */ | |
sp : 'nijibox2.com/futabafiles/003/src/' /* 60 */ | |
}, futaAlt = 's[uaspq]'; | |
//axfc uploader links | |
var axfc = {//didn't use full names but | |
Sc: "Scandium", He: "Helium", Ne: "Neon", H: "Hydrogen", | |
Li: "Lithium", N: "Nitrogen", Si: "Silicon", C: "Carbon", | |
O: "Oxygen", Al: "Aluminium", S: "Sulphur", P: "Phosphorus", | |
Ar: "Argon", B: "Boron", K: "Potassium", F: "Fluorine", | |
Be: "Beryllium", Na: "Sodium", Ca: "Calcium", Mg: "Magnesium", | |
Cl: "Chlorine" | |
}, axfcAlt = '[FKOP]|C[al]?|N[ae]?|S[ci]?|A[lr]|Be?|He?|Li|Mg'; | |
////////////////// | |
var emailReg = /^([\w-\.]+@([\w-]+\.)+[\w-]{2,4})?$/, | |
aId = 0, | |
sidebar = {}, | |
$placeholder = $('<ul/>'), | |
addToSidebar = function(m, aId, attributes){ | |
if(sidebar[m]){ | |
var $li = $('#'+m, $placeholder); | |
if( $li !== $placeholder.first()) | |
$li.prependTo($placeholder); | |
$li.data('anchorId').push(aId) | |
} else{//var id = m.replace('.','') | |
sidebar[m] = true; | |
$('<li id="'+m+'"><a'+attributes+'>'+m+'</a></li>') | |
.prependTo($placeholder) | |
.data('anchorId',[aId]) | |
} | |
return attributes; | |
}; | |
//process contexted form | |
$.fn.processDoc = function(url){ | |
timeit && console.time("processing: " + url); | |
var $context = $(this), | |
pattern = [], | |
replace = []; | |
if(optLinkifyPosts) {/* post number */ | |
pattern.push(/\bNo\.(\d+)\b/g); | |
replace.push('<a href="'+url+'#delcheck$1" class="postanchor">$&</a>'); | |
} | |
if(optLinkifyAxfc){ /* axfc links */ | |
pattern.push(new RegExp("(" + axfcAlt + ")_([0-9]{4,8})",'g')); | |
replace.push(function(m, pre, num){ | |
return '<a id="a'+(++aId)+'"' | |
+ addToSidebar(m, aId, ' href="http://www1.axfc.net/uploader/'+pre+'/so/'+num+'" class="axfc"') | |
+ '>'+m+'</a>' | |
}); | |
} | |
if(optLinkifyFutaba){/* futalog links */ | |
pattern.push(new RegExp("(" + futaAlt + ")[0-9]{5,7}",'g')); | |
replace.push(function(m, pre){ | |
return '<a id="a'+(++aId)+'"' | |
+ addToSidebar(m, aId, ' href="http://www.'+futalog[pre]+m+'" class="futalog"') | |
+ '>'+m+'</a>' | |
}); | |
} | |
//move to offscreen rendering//hide() | |
$context.css({display: 'none'}); | |
//remove right ad | |
$context.find('#rightad').remove(); | |
//expose mailto hidden messages | |
$context.find('a').each(function(){ | |
var href = this.href;// $(this).attr('href') or decodeURI | |
//faster .filter('[href^=mailto]') | |
if(!href || href.indexOf('mailto') !== 0) return true; | |
//extract fragment after mailto: | |
var text = href.substr(7); | |
//don't expose sage or valid emails (don't need them) | |
if(text === 'sage' || emailReg.test(text)) | |
return true; | |
$(this).after($('<span/>',{html: decodeURI(text), 'class':'secret'})) | |
}); | |
////add post anchor link and axfc uploader links | |
try { | |
$context.searchAndReplace(pattern, replace); | |
} catch(err){ | |
console.error(err.stack); | |
var | |
txt="There was an error in script.\n\n"; | |
txt+="Error description: " + err.message + "\n\n"; | |
txt+="Click OK to continue.\n\n"; | |
alert(txt); | |
} | |
//display content //show() | |
$context.css({display: 'block'}); | |
//add found links to futalog or axfc uploader to sidebar and recalculate height | |
if($placeholder.children().length > 0){ | |
//if($placeholder.parent('body').length === 0) | |
var $sidebar = $('#sidebar'); | |
if( $sidebar.length === 0){ | |
$sidebar = $('<div/>', {id:'sidebar'}) | |
.append($('<div/>', {text:'Links '}) | |
.append($('<span/>', {id:'dlcount'})) | |
) | |
.append($placeholder) | |
.appendTo('body'); | |
//add links highlight | |
$placeholder | |
.on('mouseenter','li',function(){ | |
$.each($(this).data('anchorId'), function(idx, val){ | |
$('#a'+val).addClass('highlight') | |
}) | |
}) | |
.on('mouseleave','li',function(){ | |
$.each($(this).data('anchorId'), function(idx, val){ | |
$('#a'+val).removeClass('highlight') | |
}) | |
}) | |
.on('click','a',function(event){ | |
event.stopPropagation(); | |
}) | |
.on('click','li',function(){ | |
$('html, body').scrollTop( | |
$('#a'+ $(this) | |
.data('anchorId') | |
.slice(-1)[0] | |
) | |
.closest('td') | |
.highlight() | |
.offset().top | |
); | |
}); | |
} | |
$('#dlcount').text( | |
'('+(function(s){var i=0,x;for(x in s)++i;return i})(sidebar) + '/' + aId +')' | |
); | |
var hHeight = $placeholder.prev().height(), | |
wHeight = $(window).height(), | |
pHeight = Math.min($placeholder.height() + hHeight,wHeight); | |
$sidebar.css({ | |
top : Math.max(0, ((wHeight - pHeight) / 2) ), | |
height: pHeight | |
}); | |
} | |
timeit && console.timeEnd("processing: " + url); | |
return $context; | |
}; | |
//remove ads (alt set display none i css), comment if you like them :D | |
$('td.chui > div, iframe').remove(); | |
//#ufm + div, hr + b, | |
$('#ufm').next('div').remove(); | |
$('hr').next('b').remove(); | |
///inital parse | |
var basehref = window.location.href.split('#')[0], | |
$contentForm = $('form').eq(1); | |
if( $contentForm.length === 0)//for yakumo-family.com | |
$contentForm = $('body');/*.wrapInner($('<div/>')).first();*/ | |
$contentForm.processDoc(basehref); | |
//add post highlight | |
var $highlight = $(); | |
$contentForm.on('click', 'a.postanchor', function(){ | |
$highlight.removeClass('highlight'); | |
$highlight = $(this.hash).closest('td').addClass('highlight'); | |
}); | |
if(window.location.hash)//don't need to fire click | |
$highlight = $(window.location.hash).closest('td').addClass('highlight'); | |
//add image expansion on click | |
$contentForm.on('click', /*a>*/'img', function(e) { | |
e.stopPropagation(); | |
e.preventDefault(); | |
var $img = $(this), | |
data = $img.data(); | |
if(!data.srcfull){//or use one()? | |
var href = $img.parent().attr('href'); | |
data.srcfull = href; | |
data.srcalt = href; | |
$img.addClass('resizeable') | |
.bind('load', function() { | |
//$img.prev('div.overlay').remove(); | |
var $img = $(this), | |
$prev = $img.prev('div.overlay'), | |
//width = $img.width(), | |
//height = $img.height(), | |
nWidth = $img[0].naturalWidth, | |
nHeight = $img[0].naturalHeight; | |
if(!$prev.hasClass('olinfo')) | |
$prev.css({ | |
width: 'auto', | |
height:'auto' | |
}) | |
.addClass('olinfo'); | |
$prev.text(nWidth + ' x ' + nHeight); | |
// $prev.html( | |
// width + ' x ' + height | |
// + (width === nWidth ? '' : '<br />width : ' + nWidth) | |
// + (height === nHeight ? '' : '<br />height: ' + nHeight) | |
// ); | |
if (($img.attr('src') === $img.data('srcfull'))) { | |
$img.addClass('fullimg'); | |
} else | |
$img.removeClass('fullimg'); | |
}) | |
.bind('error', function() { | |
$(this).prev('div.overlay').hide() | |
}) | |
.before($('<div/>',{'class': 'overlay'}) | |
.click(function(){ return false }) | |
.width( $img.outerWidth()) | |
.height($img.outerHeight()) | |
.show() | |
) | |
.parent() | |
.addClass('overlay-parent') | |
} | |
var src = data.srcalt; | |
data.srcalt = $img.attr('src'); | |
if(optConstrictWidth) | |
$img.css('max-width', windowWidth); | |
if(optConstrictHeight) | |
$img.css('max-height', windowHeight); | |
$img.attr('src', src) | |
.prev('div.overlay') | |
.show() | |
}); | |
//add inline thread expansion | |
$contentForm.find('font').filter('[color=#707070]').filter(":contains('レス')") | |
.addClass('expand') | |
.one('click',function() { | |
var $self = $(this); | |
var prev = $self.prevUntil("a:contains('返信'), small").last().prev().get(0); | |
if( prev.tagName === 'small') | |
return;//how come? | |
$self.toggleClass('expand expanding').next('br').remove(); | |
$.get(prev.href, {}, function(data) { | |
$self.toggleClass('expanding collapse') | |
.data('expanded',true); | |
//$self.nextUntil('hr').remove();//without slicing | |
$(data).filter('form') | |
.eq(1) | |
.find('table') | |
.slice(0, -10) /* can take more than is needed*/ | |
.processDoc(prev.href) | |
.insertAfter($self) | |
.wrapAll('<div class="loaded"/>') | |
}) | |
}) | |
.click(function(){ | |
var $self = $(this); | |
if( $self.data('expanded')) { | |
$self.toggleClass('collapse expand') | |
.next('div.loaded') | |
.toggle(); | |
} | |
}); | |
//add top/bottom sticky nav and autoscroll | |
var autoscroll = new Counter({interval:200}); | |
$('body') | |
.attr('id','top') | |
.append($('<div/>',{id:'btm'})) | |
.append($('<div/>',{id:'stickynav'}) | |
.append($('<a/>',{text:'⬆'/*▲*/, href:basehref+'#top','class':'pointer', title:'Top'})) | |
.append($('<a/>',{text:'⬇'/*▼*/, href:basehref+'#btm','class':'pointer', title:'Bottom'})) | |
.append($('<input/>',{id:'autoscroll', type:'text', value:$('body').height(), title: 'Speed'})) | |
.on('click', 'a.pointer', function(e){ | |
e.preventDefault(); | |
autoscroll.stop(); | |
var $self = $(this); | |
if( $self.hasClass('active')){ | |
$self.removeClass('active'); | |
return | |
} | |
var hash = $self | |
.siblings() | |
.removeClass('active') | |
.end() | |
.addClass('active') | |
.get(0).hash, | |
speed = parseInt($('#autoscroll').val()), | |
$body = $('html, body'), | |
offset, lastOffset; | |
if( hash === '#top') | |
speed *= -1; | |
autoscroll.start(function(){ | |
offset = $body.scrollTop() + speed; | |
if(offset === lastOffset) | |
return $self.click(); | |
$body.scrollTop(lastOffset = offset); | |
}) | |
}) | |
); | |
//add autorefresh and counter | |
var $contres = $('#contres a'); | |
if( $contres.length){ | |
var cookie = "AutoRefresh="; | |
var $timer = $('<span> 60s</span>').appendTo($('#contres')); | |
var counter = new Counter({ | |
count : 60, | |
interval : 1000, | |
ontick : function(tick){ $timer.text(' ' + tick + 's') }, | |
oncomplete: function() { $contres.click() } | |
}); | |
var chBox = $('<input/>', {type:'checkbox'}) | |
.appendTo($('<label/>', {text:'[Auto]'}).insertBefore($timer)) | |
.change(function(){ | |
if(counter.isRunning()) {//stop autorefresh | |
counter.stop(); | |
document.cookie = cookie + ":; expires = Thu, 01-Jan-70 00:00:01 GMT; path=/;" | |
} else {//start autorefresh | |
counter.start(); | |
document.cookie = cookie + "1; path=/;" | |
} | |
}) | |
//restore state of checkbox on page refresh | |
var ca = document.cookie.split(';'); | |
for (var i = 0, len = ca.length; i < len; i++) { | |
var c = ca[i]; | |
while (c.charAt(0) == ' ') c = c.substring(1, c.length); | |
if (c.indexOf(cookie) == 0) { | |
chBox.click(); | |
break; | |
} | |
} | |
} | |
timeit && console.timeEnd("2chan-utils"); | |
} | |
})(); |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment