4chan/js/janitor-unminified.js
2025-04-17 14:46:47 -05:00

1244 lines
30 KiB
JavaScript

/**
* Janitor Extension
*/
(function() {
/**
* Admin tools
*/
var AdminTools = {
cacheTTL: 60000,
autoRefreshDelay: 120000,
autoRefreshTimeout: null
};
// FIXME, put it as a helper in extension.js
AdminTools.initVisibilityAPI = function() {
this.hidden = 'hidden';
this.visibilitychange = 'visibilitychange';
if (typeof document.hidden === 'undefined') {
if ('mozHidden' in document) {
this.hidden = 'mozHidden';
this.visibilitychange = 'mozvisibilitychange';
}
else if ('webkitHidden' in document) {
this.hidden = 'webkitHidden';
this.visibilitychange = 'webkitvisibilitychange';
}
else if ('msHidden' in document) {
this.hidden = 'msHidden';
this.visibilitychange = 'msvisibilitychange';
}
}
document.addEventListener(this.visibilitychange, this.onVisibilityChange, false);
};
AdminTools.init = function() {
var cnt, html;
AdminTools.initVisibilityAPI();
cnt = document.createElement('div');
cnt.className = 'extPanel reply';
cnt.id = 'adminToolbox';
cnt.setAttribute('data-trackpos', 'AT-position');
if( Config['AT-position'] ) {
cnt.style.cssText = Config['AT-position'];
} else {
cnt.style.right = '10px';
cnt.style.top = '380px';
}
cnt.style.position = Config.fixedAdminToolbox ? 'fixed' : '';
html = '<div class="drag" id="atHeader">Janitor Tools'
+ '<img alt="Refresh" title="Refresh" src="' + Main.icons.refresh
+ '" id="atRefresh" data-cmd="at-refresh" class="pointer right"></div>'
+ '<h4><a href="https://' + J.reportsSubDomain + '.4chan.org/" target="_blank">Reports</a>: '
+ '<span title="Total" id="at-total">?</span> ('
+ '<span title="Illegal" id="at-illegal">?</span>)</h4>'
+ '<h4 id="at-msg-cnt"><a data-cmd="at-msg" href="https://' + J.reportsSubDomain
+ '.4chan.org/?action=staffmessages" target="_blank">Messages</a>: <span id="at-msg">?</span></h4>';
cnt.innerHTML = html;
document.body.appendChild(cnt);
AdminTools.refreshReportCount();
Draggable.set($.id('atHeader'));
};
AdminTools.onVisibilityChange = function() {
var self;
self = AdminTools;
if (document[AdminTools.hidden]) {
clearInterval(self.autoRefreshTimeout);
self.autoRefreshTimeout = null;
}
else {
self.refreshReportCount();
self.autoRefreshTimeout = setInterval(self.refreshReportCount, self.autoRefreshDelay);
}
};
AdminTools.refreshReportCount = function(force) {
var xhr, cache;
if (force !== true && (cache = localStorage.getItem('4chan-cache-rc'))) {
cache = JSON.parse(cache);
if (cache.ts > Date.now() - AdminTools.cacheTTL) {
$.id('at-total').textContent = cache.data[0];
$.id('at-illegal').textContent = cache.data[1];
$.id('at-msg-cnt').style.display = cache.data[2] ? 'block' : '';
$.id('at-msg').textContent = cache.data[2] || 0;
return;
}
}
xhr = new XMLHttpRequest();
xhr.open('GET', 'https://' + J.reportsSubDomain + '.4chan.org/H429f6uIsUqU.php', true);
xhr.withCredentials = true;
xhr.onload = function() {
var cache, resp, data, msg_count;
if (this.status == 200) {
try {
resp = JSON.parse(this.responseText);
}
catch (e) {
console.log(e);
return;
}
if (resp.status !== 'success') {
console.log(resp.message); // FIXME, use global message
return;
}
data = resp.data;
msg_count = data.msg || 0;
$.id('at-msg-cnt').style.display = msg_count ? 'block' : '';
$.id('at-msg').textContent = msg_count;
$.id('at-total').textContent = data.total;
$.id('at-illegal').textContent = data.illegal;
cache = {
ts: Date.now(),
data: [ data.total, data.illegal, msg_count ]
};
cache = JSON.stringify(cache);
localStorage.setItem('4chan-cache-rc', cache);
document.dispatchEvent(new CustomEvent('4chanATUpdated'));
}
else {
this.onerror();
}
};
xhr.onerror = function() {
console.log('Error while refreshing the report count (Status: ' + this.status + ').');
};
xhr.onloadend = function() {
$.id('atRefresh').src = Main.icons.refresh;
};
$.id('atRefresh').src = Main.icons.rotate;
xhr.send(null);
};
AdminTools.resetMsgCount = function() {
var cache;
$.id('at-msg').textContent = 0;
if (cache = localStorage.getItem('4chan-cache-rc')) {
cache = JSON.parse(cache);
cache.data[2] = 0;
cache = JSON.stringify(cache);
localStorage.setItem('4chan-cache-rc', cache);
}
};
var J = {
nextChunkIndex: 0,
nextChunk: null,
chunkSize: 100,
reportsSubDomain: 'reports'
};
J.initIconsCatalog = function() {
var key, paths, url;
Main.icons = {
up: 'arrow_up.png',
down: 'arrow_down.png',
right: 'arrow_right.png',
download: 'arrow_down2.png',
refresh: 'refresh.png',
cross: 'cross.png',
gis: 'gis.png',
iqdb: 'iqdb.png',
minus: 'post_expand_minus.png',
plus: 'post_expand_plus.png',
rotate: 'post_expand_rotate.gif',
quote: 'quote.png',
report: 'report.png',
notwatched: 'watch_thread_off.png',
watched: 'watch_thread_on.png',
help: 'question.png'
};
paths = {
yotsuba_new: 'futaba/',
futaba_new: 'futaba/',
yotsuba_b_new: 'burichan/',
burichan_new: 'burichan/',
tomorrow: 'tomorrow/',
photon: 'photon/'
};
url = '//s.4cdn.org/image/';
if (window.devicePixelRatio >= 2) {
for (key in Main.icons) {
Main.icons[key] = Main.icons[key].replace('.', '@2x.');
}
}
url += 'buttons/' + paths[Main.stylesheet];
for (key in Main.icons) {
Main.icons[key] = url + Main.icons[key];
}
};
J.apiUrlFilter = function(url) {
return url + '?' + Math.round(Date.now() / 1000 / 3);
};
J.openDeletePrompt = function(id) {
var html, cnt;
id = id.getAttribute('data-id');
html = '<div class="extPanel reply"><div class="panelHeader">Delete Post No.' + id
+ '<span class="panelCtrl"><img alt="Close" title="Close" class="pointer" data-cmd="close-delete-prompt" src="'
+ Main.icons.cross + '"></a>'
+ '</span></div><span id="delete-prompt-inner">'
+ '<input type="button" value="Delete Post" tabindex="-1" data-cmd="delete-post" data-id="' + id + '"> '
+ '<input type="button" value="Delete Image Only" data-cmd="delete-image" data-id="' + id + '">';
cnt = document.createElement('div');
cnt.className = 'UIPanel';
cnt.id = 'delete-prompt';
cnt.innerHTML = html;
cnt.addEventListener('click', J.closeDeletePrompt, false);
document.body.appendChild(cnt);
$.id('delete-prompt-inner').firstElementChild.focus();
};
J.closeDeletePrompt = function(e) {
var prompt;
if (!e || e.target.id == 'delete-prompt') {
if (prompt = $.id('delete-prompt')) {
prompt.removeEventListener('click', J.closeDeletePrompt, false);
document.body.removeChild(prompt);
}
}
};
J.deletePost = function(btn, imageOnly) {
var id, xhr, form, msg, el, url, mode, del, isOp;
id = btn.getAttribute('data-id');
isOp = $.id('t' + id);
form = new FormData();
msg = 'Delete Post No.';
url = 'https://sys.' + $L.d(Main.board) + '/' + Main.board + '/post';
mode = window.thread_archived ? 'arcdel' : 'usrdel';
if(imageOnly) {
msg = 'Delete Image No.';
form.append('onlyimgdel', 'on');
}
form.append(id, 'delete');
form.append('mode', mode);
form.append('pwd', 'janitorise');
(del = $.id('delete-prompt-inner')).textContent = 'Deleting...';
xhr = new XMLHttpRequest();
xhr.open('POST', url);
xhr.withCredentials = true;
xhr.onload = function() {
btn.src = Main.icons.cross;
if (this.status == 200) {
if (this.responseText.indexOf('Updating') != -1) {
if (!imageOnly) {
if (id == Main.tid) {
location.href = '//boards.' + $L.d(Main.board) + '/' + Main.board + '/';
return;
}
else {
if (isOp) {
el = isOp.parentNode;
el.removeChild(isOp.nextElementSibling);
el.removeChild(isOp);
}
else {
el = $.id('pc' + id);
el.parentNode.removeChild(el);
}
}
}
else {
el = $.id('f' + id);
el.innerHTML = '<span class="fileThumb"><img alt="File deleted."'
+ ' src="//s.4cdn.org/image/filedeleted' + (isOp ? '' : '-res') + '.gif"></span>';
}
J.closeDeletePrompt();
}
else {
del.textContent = 'Error: Post might have already been deleted, or is a sticky.';
}
}
else {
del.textContent = 'Error: Wrong status while deleting No.' + id + ' (Status: ' + this.status + ').';
}
};
xhr.onerror = function() {
del.textContent = 'Error: Error while deleting No.' + id + ' (Status: ' + this.status + ').';
};
xhr.send(form);
};
J.openBanReqWindow = function(btn)
{
var id;
id = btn.getAttribute('data-id');
window.open('https://sys.' + $L.d(Main.board) + '/' + Main.board + '/admin?mode=admin&admin=banreq&id=' + id, '_blank', 'scrollBars=yes,resizable=no,toolbar=no,menubar=no,location=no,directories=no,width=400,height=245');
};
J.openBanReqFrame = function(btn) {
var id;
if (this.banReqCnt) {
this.close();
}
id = btn.getAttribute('data-id');
this.banReqCnt = document.createElement('div');
this.banReqCnt.id = 'banReq';
this.banReqCnt.className = 'extPanel reply';
this.banReqCnt.setAttribute('data-trackpos', 'banReq-position');
if (Config['banReq-position']) {
this.banReqCnt.style.cssText = Config['banReq-position'];
}
else {
this.banReqCnt.style.right = '0px';
this.banReqCnt.style.top = '50px';
}
this.banReqCnt.innerHTML =
'<div id="banReqHeader" class="drag postblock">Ban Request No.' + id
+ '<img alt="X" src="' + Main.icons.cross + '" id="banReqClose" '
+ 'class="extButton" title="Close Window"></div>'
+ '<iframe src="https://sys.' + $L.d(Main.board) + '/'
+ Main.board + '/admin?mode=admin&admin=banreq&id=' + id
+ '&noheader=true" width="400" height="230" frameborder="0"></iframe>';
document.body.appendChild(this.banReqCnt);
window.addEventListener('message', J.onMessage, false);
document.addEventListener('keydown', J.onKeyDown, false);
$.id('banReqClose').addEventListener('click', J.closeBanReqFrame, false);
Draggable.set($.id('banReqHeader'));
};
J.closeBanReqFrame = function() {
window.removeEventListener('message', J.onMessage, false);
document.removeEventListener('keydown', J.onKeyDown, false);
Draggable.unset($.id('banReqHeader'));
$.id('banReqClose').removeEventListener('click', J.closeBanReqFrame, false);
document.body.removeChild(J.banReqCnt);
J.banReqCnt = null;
};
J.processMessage = function(data) {
if (!data) {
return {};
}
data = data.split('-');
return {
cmd: data[0],
type: data[1],
id: data.slice(2).join('-')
};
};
J.onKeyDown = function(e) {
if (e.keyCode == 27 && !e.ctrlKey && !e.altKey && !e.shiftKey && !e.metaKey) {
J.closeBanReqFrame();
}
};
J.onMessage = function(e) {
var msg;
if (e.origin !== 'https://sys.' + $L.d(Main.board)) {
return;
}
msg = J.processMessage(e.data);
if (msg.type !== 'ban') {
return;
}
if (msg.cmd === 'done' || msg.cmd === 'cancel') {
J.closeBanReqFrame();
}
};
/**
* Click handler
*/
J.onClick = function(e) {
var t, cmd;
if ((t = e.target) == document) {
return;
}
if (cmd = t.getAttribute('data-cmd')) {
switch (cmd) {
case 'at-refresh':
AdminTools.refreshReportCount(true);
break;
case 'delete-post':
case 'delete-image':
J.deletePost(t, (cmd === 'delete-image'));
break;
case 'open-delete-prompt':
J.openDeletePrompt(t);
break;
case 'close-delete-prompt':
J.closeDeletePrompt();
break;
case 'open-banreq-prompt':
if (Config.inlinePopups) {
J.openBanReqFrame(t);
}
else {
J.openBanReqWindow(t);
}
break;
case 'at-msg':
AdminTools.resetMsgCount();
break;
case 'toggle-file-spoiler':
J.setFileSpoiler(t);
break;
case 'prompt-spoiler':
if (confirm('Toggle spoiler?')) {
J.setFileSpoiler(t);
}
break;
case 'boardlist-open':
J.openBoardList();
break;
case 'boardlist-close':
J.closeBoardList();
break;
case 'boardlist-save':
J.saveBoardList();
J.closeBoardList();
break;
}
}
};
J.onScroll = function() {
var end;
while (J.nextChunk.offsetTop < (document.documentElement.clientHeight + window.scrollY)) {
end = J.nextChunkIndex + J.chunkSize;
if (end >= J.postCount) {
J.parseRange(J.nextChunkIndex, J.postCount);
window.removeEventListener('scroll', J.onScroll, false);
return false;
}
else {
J.parseRange(J.nextChunkIndex, end);
}
}
return true;
};
J.parseRange = function(start, end) {
var i, j, posts;
posts = document.getElementById('t' + Main.tid).getElementsByClassName('postInfo');
for (i = start; i < end; ++i) {
j = posts[i];
if (!j) {
break;
}
J.parsePost(j);
}
J.nextChunkIndex = i;
J.nextChunk = posts[i];
};
J.onParsingDone = function(e) {
var i, tid, offset, limit, posts;
if (Config.useIconButtons) {
if (e) {
tid = e.detail.threadId;
offset = e.detail.offset;
limit = e.detail.limit;
posts = document.getElementById('t' + tid).getElementsByClassName('postInfo');
}
else {
offset = 0;
posts = document.getElementsByClassName('postInfo');
limit = posts.length;
}
for (i = offset; i < limit; ++i) {
J.parsePost(posts[i]);
}
}
};
J.onPostMenuReady = function(e) {
var el, pid, menu, flag;
pid = e.detail.postId;
menu = e.detail.node;
el = document.createElement('li');
el.className = 'dd-admin';
el.setAttribute('data-cmd', 'open-delete-prompt');
el.setAttribute('data-id', pid);
el.textContent = 'Delete';
menu.appendChild(el);
if (window.spoilers && (el = $.id('fT' + pid))) {
flag = $.cls('imgspoiler', el.parentNode)[0] ? 0 : 1;
el = document.createElement('li');
el.className = 'dd-admin';
el.setAttribute('data-cmd', 'toggle-file-spoiler');
el.setAttribute('data-id', pid);
el.setAttribute('data-flag', flag);
el.textContent = (flag ? 'Set' : 'Unset') + ' Spoiler';
menu.appendChild(el);
}
if (window.thread_archived) {
return;
}
el = document.createElement('li');
el.className = 'dd-admin';
el.setAttribute('data-cmd', 'open-banreq-prompt');
el.setAttribute('data-id', pid);
el.textContent = 'Ban request';
menu.appendChild(el);
};
J.parsePost = function(postInfo) {
var pid, html, cnt, tail;
pid = postInfo.id.slice(2);
html = '<img class="extButton" alt="X" data-cmd="open-delete-prompt" data-id="'
+ pid + '" src="' + Main.icons.cross
+ '" title="Delete">';
if (window.spoilers && (el = $.id('fT' + pid))) {
html += '<img class="extButton" alt="S" data-cmd="prompt-spoiler" data-id="'
+ pid + '" src="' + J.icons.spoiler
+ '" title="Toggle Spoiler">';
}
if (!window.thread_archived) {
html += '<img class="extButton" alt="B" data-cmd="open-banreq-prompt" data-id="'
+ pid + '" src="' + J.icons.ban
+ '" title="Ban Request">';
}
cnt = document.createElement('div');
cnt.className = 'extControls';
cnt.innerHTML = html;
tail = postInfo.getElementsByClassName('postMenuBtn')[0];
postInfo.insertBefore(cnt, tail);
};
J.displayJCount = function(jLink, jLinkBot, no, delta) {
var msg;
$.addClass(jLink, 'j-newposts');
$.addClass(jLinkBot, 'j-newposts');
jLink.setAttribute('data-no', no);
jLinkBot.setAttribute('data-no', no);
jLink.textContent = jLinkBot.textContent = 'j +' + delta;
msg = delta + ' new post' + (delta > 1 ? 's' : '');
Main.addTooltip(jLink, msg, 'j-tooltip');
Main.addTooltip(jLinkBot, msg, 'j-tooltip-bot');
};
J.refreshJCount = function() {
var stored, jLink, jLinkBot, xhr;
jLink = $.id('j-link');
jLinkBot = $.id('j-link-bot');
if (!jLink || !jLinkBot) {
return;
}
jLink = jLink.firstElementChild;
jLinkBot = jLinkBot.firstElementChild;
if (stored = localStorage.getItem('4chan-j-count')) {
stored = JSON.parse(stored);
}
if (!stored || (Date.now() - stored.time) >= 10000) {
xhr = new XMLHttpRequest();
xhr.open('GET', 'https://sys.4chan.org/j/1mcQTXbjW5WO.php?&' + Date.now());
xhr.withCredentials = true;
xhr.onloadend = function() {
var data, obj, delta;
if (this.status == 200 || this.status == 304) {
data = JSON.parse(this.responseText);
if (!stored || Main.board == 'j') {
obj = { time: Date.now(), no: data.no };
}
else if (data.no > stored.no) {
delta = data.no - stored.no;
J.displayJCount(jLink, jLinkBot, data.no, delta);
obj = { time: Date.now(), no: stored.no, delta: delta };
}
if (obj) {
localStorage.setItem('4chan-j-count', JSON.stringify(obj));
}
}
else {
console.log('Error: Could not load /j/ post count (Status: ' + this.status + ').');
}
};
xhr.send(null);
}
else if (stored.delta) {
J.displayJCount(jLink, jLinkBot, stored.no, stored.delta);
}
};
J.clearJCount = function() {
var obj, no, tt, ttbot;
tt = $.id('j-tooltip');
ttbot = $.id('j-tooltip-bot');
if (!tt) {
return;
}
no = this.getAttribute('data-no');
obj = { time: Date.now(), no: no };
localStorage.setItem('4chan-j-count', JSON.stringify(obj));
tt.parentNode.removeChild(tt);
ttbot.parentNode.removeChild(ttbot);
setTimeout(function() {
var nodes = $.cls('j-newposts');
if (nodes[0]) {
nodes[0].textContent = 'j';
$.removeClass(nodes[0], 'j-newposts');
nodes[0].textContent = 'j';
$.removeClass(nodes[0], 'j-newposts');
}
}, 10);
};
J.icons = {
ban: 'ban.png',
spoiler: 's.png'
};
J.initIcons = function() {
var key, paths, url;
paths = {
yotsuba_new: 'futaba/',
futaba_new: 'futaba/',
yotsuba_b_new: 'burichan/',
burichan_new: 'burichan/',
tomorrow: 'tomorrow/',
photon: 'photon/'
};
url = '//s.4cdn.org/image/buttons/' + paths[Main.stylesheet];
if (window.devicePixelRatio >= 2) {
for (key in J.icons) {
J.icons[key] = J.icons[key].replace('.', '@2x.');
}
}
for (key in J.icons) {
J.icons[key] = url + J.icons[key];
}
};
J.initNavLinks = function() {
var el, nav, navbot;
nav = $.id('navtopright');
navbot = $.id('navbotright');
// [j] link
el = document.createElement('span');
el.id = 'j-link';
el.innerHTML = '[<a href="https://sys.4chan.org/j/" title="Janitor &amp; Moderator Discussion">j</a>]';
el.firstElementChild.addEventListener('mouseup', J.clearJCount, false);
nav.parentNode.insertBefore(el, nav);
// [j] bottom link
el = el.cloneNode(true);
el.id = 'j-link-bot';
el.firstElementChild.addEventListener('mouseup', J.clearJCount, false);
navbot.parentNode.insertBefore(el, navbot);
J.refreshJCount();
};
J.openBoardList = function() {
var cnt;
if ($.id('boardList')) {
return;
}
cnt = document.createElement('div');
cnt.id = 'boardList';
cnt.className = 'UIPanel';
cnt.setAttribute('data-cmd', 'boardlist-close');
cnt.innerHTML = '\
<div class="extPanel reply"><div class="panelHeader">Boards\
<span class="panelCtrl"><img alt="Close" title="Close" class="pointer" data-cmd="boardlist-close" src="'
+ Main.icons.cross + '"></a></span></div>\
<input placeholder="Example: jp tg mu or all" id="boardListBox" type="text" value="'
+ (localStorage.getItem('4chan-boardlist') || '') + '">\
<div class="center"><button id="boardListSave" data-cmd="boardlist-save">Save</button></div>\
</td></tr></tfoot></table></div>';
document.body.appendChild(cnt);
cnt.addEventListener('click', this.onClick, false);
};
J.saveBoardList = function() {
var input;
if (input = $.id('boardListBox')) {
localStorage.setItem('4chan-boardlist', input.value);
}
};
J.closeBoardList = function() {
var cnt;
if (cnt = $.id('boardList')) {
cnt.removeEventListener('click', this.onClick, false);
document.body.removeChild(cnt);
}
};
J.setFileSpoiler = function(t) {
var xhr, pid, flag, el;
pid = t.getAttribute('data-id');
flag = t.getAttribute('data-flag');
if (!pid) {
return;
}
el = $.id('f' + pid);
if (!flag) {
flag = $.cls('imgspoiler', el.parentNode)[0] ? 0 : 1;
}
if (!el || el.hasAttribute('data-processing')) {
return;
}
xhr = new XMLHttpRequest();
xhr.open('GET', 'https://sys.' + $L.d(Main.board) + '/' + Main.board
+ '/admin.php?admin=spoiler&pid=' + pid + '&flag=' + flag, true);
xhr.withCredentials = true;
xhr.onload = J.onFileSpoilerLoad;
xhr.onerror = J.onFileSpoilerError;
xhr._pid = +pid;
xhr._flag = +flag;
Feedback.notify('Processing...', null);
el.setAttribute('data-processing', '1');
xhr.send(null);
};
J.onFileSpoilerLoad = function() {
var el, el2;
Feedback.hideMessage();
if (this.responseText !== '1') {
if (this.responseText === '-1') {
Feedback.error('You are not logged in');
}
else {
Feedback.error("Couldn't set spoiler flag for post No." + this._pid);
}
return;
}
if (!(el = $.id('f' + this._pid))) {
return;
}
el.removeAttribute('data-processing');
if (!(el = $.cls('fileThumb', el)[0])) {
return;
}
if (this._flag) {
$.addClass(el, 'imgspoiler');
el2 = el.previousElementSibling;
el2.setAttribute('title', el2.firstElementChild.textContent);
if (!Config.revealSpoilers) {
el = $.tag('img', el)[0];
el.style.width = el.style.height = '100px';
el.src = '//s.4cdn.org/image/spoiler-' + Main.board + '.png';
}
}
else {
if (!Config.revealSpoilers) {
Parser.revealImageSpoiler(el);
}
$.removeClass(el, 'imgspoiler');
}
};
J.onFileSpoilerError = function() {
var el;
if (!(el = $.id('f' + this._pid))) {
return;
}
el.removeAttribute('data-processing');
Feedback.error("Couldn't update the spoiler flag for post No." + this.pid);
};
J.initCatalog = function() {
var storage;
window.Main = {
board: location.pathname.split(/\//)[1]
};
window.Main.addTooltip = function(link, message, id) {
var el, pos;
el = document.createElement('div');
el.className = 'click-me';
if (id) {
el.id = id;
}
el.innerHTML = message || 'Change your settings';
link.parentNode.appendChild(el);
pos = (link.offsetWidth - el.offsetWidth + link.offsetLeft - el.offsetLeft) / 2;
el.style.marginLeft = pos + 'px';
return el;
};
if (J.stylesheet = J.getCookie(window.style_group)) {
J.stylesheet = J.stylesheet.toLowerCase().replace(/ /g, '_');
}
else {
J.stylesheet =
style_group == 'nws_style' ? 'yotsuba_new' : 'yotsuba_b_new';
}
Main.stylesheet = J.stylesheet;
J.initIconsCatalog();
J.addCss(); // fixme
document.addEventListener('click', J.onClick, false);
J.runCatalog();
};
J.runCatalog = function () {
var threads;
//J.addCss(); // fixme
//document.removeEventListener('4chanMainInit', J.runCatalog, false);
J.initNavLinks();
if (!FC.hasMobileLayout) {
AdminTools.init();
}
threads = $.id('threads');
$.on(threads, 'mouseover', J.onThreadMouseOver);
//$.on(threads, 'mouseout', J.onThreadMouseOut);
};
J.init = function() {
var ts;
Config.boardList = true;
SettingsMenu.options['Janitor'] = {
boardList: [ 'Janitor Boards [<a href="javascript:;" data-cmd="boardlist-open">Select</a>]', 'Select boards to enable janitor buttons on', true ],
useIconButtons: [ 'Use icon buttons', 'Display old-style buttons instead of using drop-down' ],
changeUpdateDelay: [ 'Reduce auto-update delay', 'Reduce the thread updater delay', true ],
fixedAdminToolbox: [ 'Pin Janitor Tools to the page', 'Janitor Tools will scroll with you' ],
inlinePopups: [ 'Inline ban request panel', 'Open ban request panel in browser window, instead of a popup' ],
disableMngExt: [ 'Disable janitor extension', 'Completely disable the janitor extension (overrides any checked boxes)', true ]
};
if (Config.disableMngExt) {
return;
}
J.addCss();
if (Config.useIconButtons) {
J.initIcons();
}
QR.noCooldown = QR.noCaptcha = true;
document.addEventListener('click', J.onClick, false);
document.addEventListener('DOMContentLoaded', J.run, false);
};
J.run = function() {
var boards, posts, nav, el;
document.removeEventListener('DOMContentLoaded', J.run, false);
J.initNavLinks();
if (!Main.hasMobileLayout) {
AdminTools.init();
}
if (Config.revealSpoilers) {
$.addClass(document.body, 'reveal-img-spoilers');
}
if (Config.threadUpdater && Main.tid) {
if (Config.changeUpdateDelay) {
ThreadUpdater.delayIdHidden = 3;
ThreadUpdater.delayRange = [ 5, 10, 15, 20, 30, 60 ];
ThreadUpdater.apiUrlFilter = J.apiUrlFilter;
}
}
boards = localStorage.getItem('4chan-boardlist') || '';
if (Main.board != 'j' && (boards == 'all' || boards.split(/[, ]+/).indexOf(Main.board) != -1)) {
if (Config.useIconButtons && !Main.hasMobileLayout) {
if (Main.tid) {
posts = document.getElementById('t' + Main.tid).getElementsByClassName('postInfo');
J.postCount = posts.length;
if (J.postCount > J.chunkSize) {
J.nextChunk = posts[0];
window.addEventListener('scroll', J.onScroll, false);
J.onScroll();
}
else {
J.onParsingDone();
}
}
else {
J.onParsingDone();
}
document.addEventListener('4chanParsingDone', J.onParsingDone, false);
}
document.addEventListener('4chanPostMenuReady', J.onPostMenuReady, false);
}
if (nav = $.id('boardSelectMobile')) {
el = document.createElement('option');
el.value = 'j';
el.textContent = '/j/ - Janitors & Moderators';
nav.insertBefore(el, nav.firstElementChild);
}
};
J.getCookie = function(name) {
var i, c, ca, key;
key = name + "=";
ca = document.cookie.split(';');
for (i = 0; c = ca[i]; ++i) {
while (c.charAt(0) == ' ') {
c = c.substring(1, c.length);
}
if (c.indexOf(key) === 0) {
return decodeURIComponent(c.substring(key.length, c.length));
}
}
return null;
};
J.addCss = function() {
var style, css;
css = '\
#adminToolbox {\
max-width: 256px;\
display: block;\
position: absolute;\
padding: 3px;\
}\
#adminToolbox h4 {\
font-size: 12px;\
margin: 2px 0 0;\
padding: 0;\
font-weight: normal;\
}\
#adminToolbox li {\
list-style: none;\
}\
#adminToolbox ul {\
padding: 0;\
margin: 2px 0 0 10px;\
}\
#atHeader {\
height: 17px;\
font-weight: bold;\
padding-bottom: 2px;\
}\
#atRefresh {\
margin: -1px 0 0 3px;\
}\
.post-ip {\
margin-left: 5px;\
}\
#delete-prompt > div {\
text-align: center;\
}\
#watchList li:first-child {\
margin-top: 3px;\
padding-top: 2px;\
border-top: 1px solid rgba(0, 0, 0, 0.20);\
}\
.photon #atHeader {\
border-bottom: 1px solid #ccc;\
}\
.yotsuba_new #atHeader {\
border-bottom: 1px solid #d9bfb7;\
}\
.yotsuba_b_new #atHeader {\
border-bottom: 1px solid #b7c5d9;\
}\
.tomorrow #atHeader {\
border-bottom: 1px solid #111;\
}\
#at-illegal {\
color: red;\
}\
#at-msg-cnt {\
display: none;\
}\
.j-newposts {\
font-weight: bold !important;\
}\
#j-link,\
#j-link-bot {\
margin-right: 3px;\
display: inline-block;\
margin-left: 3px;\
}\
#boardList input {\
width: 385px;\
margin: auto;\
display: block;\
}\
#boardListSave {\
margin-top: 5px;\
}\
#banReqClose {\
float: right;\
}\
#banReq iframe {\
overflow: hidden;\
}\
#banReq {\
display: block;\
position: fixed;\
padding: 2px;\
font-size: 10pt;\
height: 250px;\
}\
#banReqHeader {\
text-align: center;\
margin-bottom: 1px;\
padding: 0;\
height: 18px;\
line-height: 18px;\
}\
#captchaFormPart {\
display: none;\
}\
.mobileExtControls {\
float: right;\
font-size: 11px;\
margin-bottom: 3px;\
}\
.ws .mobileExtControls {\
color: #34345C;\
}\
.nws .mobileExtControls {\
color: #0000EE;\
}\
.reply .mobileExtControls {\
margin-right: 5px;\
}\
.mobileExtControls span {\
margin-left: 10px;\
}\
.mobileExtControls span:after {\
content: "]";\
}\
.mobileExtControls span:before {\
content: "[";\
}\
.nws .mobileExtControls span:after {\
color: #800000;\
}\
.nws .mobileExtControls span:before {\
color: #800000;\
}\
.ws .mobileExtControls span:after {\
color: #000;\
}\
.ws .mobileExtControls span:before {\
color: #000;\
}\
.m-dark .mobileExtControls,\
.m-dark .mobileExtControls span:after,\
.m-dark .mobileExtControls span:before {\
color: #707070 !important;\
}\
.dd-admin {\
text-indent: 5px;\
}\
.dd-admin:before {\
color: #FF0000;\
content: "•";\
left: -3px;\
position: absolute;\
}\
.extPanel {\
border: 1px solid rgba(0, 0, 0, 0.2);\
}\
.extPanel img.pointer { width: 18px; height: 18px }\
.drag {\
-moz-user-select: none !important;\
cursor: move !important;\
}\
.reveal-img-spoilers .imgspoiler::before {\
content: " ";\
width:0.75em;\
height:0.75em;\
border-radius: 0.5em;\
position: absolute;\
display: block;\
background: red;\
margin-top: 1px;\
margin-left: 1px;\
pointer-events: none;\
}\
.reveal-img-spoilers.is_catalog .imgspoiler::before { margin-top: 4px; margin-left: 12px;}\
.reveal-img-spoilers .imgspoiler:hover::before { background: #fff; }';
style = document.createElement('style');
style.setAttribute('type', 'text/css');
style.textContent = css;
document.head.appendChild(style);
};
if (/https?:\/\/boards\.(?:4chan|4channel)\.org\/[a-z0-9]+\/catalog($|#.*$)/.test(location.href)) {
J.initCatalog();
}
else {
J.init();
//J.run();
}
})();