5551 lines
176 KiB
JavaScript
5551 lines
176 KiB
JavaScript
// ==UserScript==
|
||
// @name 4chan x
|
||
// @version 2.33.8
|
||
// @namespace aeosynth
|
||
// @description Adds various features.
|
||
// @copyright 2009-2011 James Campos <james.r.campos@gmail.com>
|
||
// @copyright 2012 Nicolas Stepien <stepien.nicolas@gmail.com>
|
||
// @license MIT; http://en.wikipedia.org/wiki/Mit_license
|
||
// @include http://boards.4chan.org/*
|
||
// @include https://boards.4chan.org/*
|
||
// @include http://images.4chan.org/*
|
||
// @include https://images.4chan.org/*
|
||
// @include http://sys.4chan.org/*
|
||
// @include https://sys.4chan.org/*
|
||
// @run-at document-start
|
||
// @updateURL https://github.com/MayhemYDG/4chan-x/raw/stable/4chan_x.user.js
|
||
// @downloadURL https://github.com/MayhemYDG/4chan-x/raw/stable/4chan_x.user.js
|
||
// @icon http://mayhemydg.github.com/4chan-x/favicon.gif
|
||
// ==/UserScript==
|
||
|
||
/* LICENSE
|
||
*
|
||
* Copyright (c) 2009-2011 James Campos <james.r.campos@gmail.com>
|
||
* Copyright (c) 2012 Nicolas Stepien <stepien.nicolas@gmail.com>
|
||
* http://mayhemydg.github.com/4chan-x/
|
||
* 4chan X 2.33.8
|
||
*
|
||
* Permission is hereby granted, free of charge, to any person
|
||
* obtaining a copy of this software and associated documentation
|
||
* files (the "Software"), to deal in the Software without
|
||
* restriction, including without limitation the rights to use,
|
||
* copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||
* copies of the Software, and to permit persons to whom the
|
||
* Software is furnished to do so, subject to the following
|
||
* conditions:
|
||
*
|
||
* The above copyright notice and this permission notice shall be
|
||
* included in all copies or substantial portions of the Software.
|
||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
|
||
* OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
|
||
* HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
||
* WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
||
* OTHER DEALINGS IN THE SOFTWARE.
|
||
*
|
||
* HACKING
|
||
*
|
||
* 4chan X is written in CoffeeScript[1], and developed on GitHub[2].
|
||
*
|
||
* [1]: http://coffeescript.org/
|
||
* [2]: https://github.com/MayhemYDG/4chan-x
|
||
*
|
||
* CONTRIBUTORS
|
||
*
|
||
* noface - unique ID fixes
|
||
* desuwa - Firefox filename upload fix
|
||
* seaweed - bottom padding for image hover
|
||
* e000 - cooldown sanity check
|
||
* ahodesuka - scroll back when unexpanding images, file info formatting
|
||
* Shou- - pentadactyl fixes
|
||
* ferongr - new favicons
|
||
* xat- - new favicons
|
||
* Zixaphir - fix qr textarea - captcha-image gap
|
||
* Ongpot - sfw favicon
|
||
* thisisanon - nsfw + 404 favicons
|
||
* Anonymous - empty favicon
|
||
* Seiba - chrome quick reply focusing
|
||
* herpaderpderp - recaptcha fixes
|
||
* WakiMiko - recaptcha tab order http://userscripts.org/scripts/show/82657
|
||
* btmcsweeney - allow users to specify text for sauce links
|
||
*
|
||
* All the people who've taken the time to write bug reports.
|
||
*
|
||
* Thank you.
|
||
*/
|
||
|
||
(function() {
|
||
var $, $$, Anonymize, ArchiveLink, AutoGif, Conf, Config, DeleteLink, DownloadLink, ExpandComment, ExpandThread, Favicon, FileInfo, Filter, Get, ImageExpand, ImageHover, Keybinds, Main, Menu, Nav, Options, QR, QuoteBacklink, QuoteCT, QuoteInline, QuoteOP, QuotePreview, Quotify, Redirect, ReplyHiding, ReportLink, RevealSpoilers, Sauce, StrikethroughQuotes, ThreadHiding, ThreadStats, Time, TitlePost, UI, Unread, Updater, Watcher, d, g, _base;
|
||
|
||
Config = {
|
||
main: {
|
||
Enhancing: {
|
||
'404 Redirect': [true, 'Redirect dead threads and images'],
|
||
'Keybinds': [true, 'Binds actions to keys'],
|
||
'Time Formatting': [true, 'Arbitrarily formatted timestamps, using your local time'],
|
||
'File Info Formatting': [true, 'Reformats the file information'],
|
||
'Comment Expansion': [true, 'Expand too long comments'],
|
||
'Thread Expansion': [true, 'View all replies'],
|
||
'Index Navigation': [true, 'Navigate to previous / next thread'],
|
||
'Reply Navigation': [false, 'Navigate to top / bottom of thread'],
|
||
'Check for Updates': [true, 'Check for updated versions of 4chan X']
|
||
},
|
||
Filtering: {
|
||
'Anonymize': [false, 'Make everybody anonymous'],
|
||
'Filter': [true, 'Self-moderation placebo'],
|
||
'Recursive Filtering': [true, 'Filter replies of filtered posts, recursively'],
|
||
'Reply Hiding': [true, 'Hide single replies'],
|
||
'Thread Hiding': [true, 'Hide entire threads'],
|
||
'Show Stubs': [true, 'Of hidden threads / replies']
|
||
},
|
||
Imaging: {
|
||
'Image Auto-Gif': [false, 'Animate gif thumbnails'],
|
||
'Image Expansion': [true, 'Expand images'],
|
||
'Image Hover': [false, 'Show full image on mouseover'],
|
||
'Sauce': [true, 'Add sauce to images'],
|
||
'Reveal Spoilers': [false, 'Replace spoiler thumbnails by the original thumbnail'],
|
||
'Expand From Current': [false, 'Expand images from current position to thread end.']
|
||
},
|
||
Menu: {
|
||
'Menu': [true, 'Add a drop-down menu in posts.'],
|
||
'Report Link': [true, 'Add a report link to the menu.'],
|
||
'Delete Link': [true, 'Add a delete link to the menu.'],
|
||
'Download Link': [true, 'Add a download with original filename link to the menu. Chrome-only currently.'],
|
||
'Archive Link': [true, 'Add an archive link to the menu.']
|
||
},
|
||
Monitoring: {
|
||
'Thread Updater': [true, 'Update threads. Has more options in its own dialog.'],
|
||
'Unread Count': [true, 'Show unread post count in tab title'],
|
||
'Unread Favicon': [true, 'Show a different favicon when there are unread posts'],
|
||
'Post in Title': [true, 'Show the op\'s post in the tab title'],
|
||
'Thread Stats': [true, 'Display reply and image count'],
|
||
'Thread Watcher': [true, 'Bookmark threads'],
|
||
'Auto Watch': [true, 'Automatically watch threads that you start'],
|
||
'Auto Watch Reply': [false, 'Automatically watch threads that you reply to']
|
||
},
|
||
Posting: {
|
||
'Quick Reply': [true, 'Reply without leaving the page.'],
|
||
'Cooldown': [true, 'Prevent "flood detected" errors.'],
|
||
'Persistent QR': [false, 'The Quick reply won\'t disappear after posting.'],
|
||
'Auto Hide QR': [true, 'Automatically hide the quick reply when posting.'],
|
||
'Open Reply in New Tab': [false, 'Open replies in a new tab that are made from the main board.'],
|
||
'Remember QR size': [false, 'Remember the size of the Quick reply (Firefox only).'],
|
||
'Remember Subject': [false, 'Remember the subject field, instead of resetting after posting.'],
|
||
'Remember Spoiler': [false, 'Remember the spoiler state, instead of resetting after posting.'],
|
||
'Hide Original Post Form': [true, 'Replace the normal post form with a shortcut to open the QR.']
|
||
},
|
||
Quoting: {
|
||
'Quote Backlinks': [true, 'Add quote backlinks'],
|
||
'OP Backlinks': [false, 'Add backlinks to the OP'],
|
||
'Quote Highlighting': [true, 'Highlight the previewed post'],
|
||
'Quote Inline': [true, 'Show quoted post inline on quote click'],
|
||
'Quote Preview': [true, 'Show quote content on hover'],
|
||
'Resurrect Quotes': [true, 'Linkify dead quotes to archives'],
|
||
'Indicate OP quote': [true, 'Add \'(OP)\' to OP quotes'],
|
||
'Indicate Cross-thread Quotes': [true, 'Add \'(Cross-thread)\' to cross-threads quotes'],
|
||
'Forward Hiding': [true, 'Hide original posts of inlined backlinks']
|
||
}
|
||
},
|
||
filter: {
|
||
name: ['# Filter any namefags:', '#/^(?!Anonymous$)/'].join('\n'),
|
||
uniqueid: ['# Filter a specific ID:', '#/Txhvk1Tl/'].join('\n'),
|
||
tripcode: ['# Filter any tripfags', '#/^!/'].join('\n'),
|
||
mod: ['# Set a custom class for mods:', '#/Mod$/;highlight:mod;op:yes', '# Set a custom class for moot:', '#/Admin$/;highlight:moot;op:yes'].join('\n'),
|
||
email: ['# Filter any e-mails that are not `sage` on /a/ and /jp/:', '#/^(?!sage$)/;boards:a,jp'].join('\n'),
|
||
subject: ['# Filter Generals on /v/:', '#/general/i;boards:v;op:only'].join('\n'),
|
||
comment: ['# Filter Stallman copypasta on /g/:', '#/what you\'re refer+ing to as linux/i;boards:g'].join('\n'),
|
||
country: [''].join('\n'),
|
||
filename: [''].join('\n'),
|
||
dimensions: ['# Highlight potential wallpapers:', '#/1920x1080/;op:yes;highlight;top:no;boards:w,wg'].join('\n'),
|
||
filesize: [''].join('\n'),
|
||
md5: [''].join('\n')
|
||
},
|
||
sauces: ['http://iqdb.org/?url=$1', 'http://www.google.com/searchbyimage?image_url=$1', '#http://tineye.com/search?url=$1', '#http://saucenao.com/search.php?db=999&url=$1', '#http://3d.iqdb.org/?url=$1', '#http://regex.info/exif.cgi?imgurl=$2', '# uploaders:', '#http://imgur.com/upload?url=$2;text:Upload to imgur', '#http://omploader.org/upload?url1=$2;text:Upload to omploader', '# "View Same" in archives:', '#http://archive.foolz.us/search/image/$3/;text:View same on foolz', '#http://archive.foolz.us/$4/search/image/$3/;text:View same on foolz /$4/', '#https://archive.installgentoo.net/$4/image/$3;text:View same on installgentoo /$4/'].join('\n'),
|
||
time: '%m/%d/%y(%a)%H:%M',
|
||
backlink: '>>%id',
|
||
fileInfo: '%l (%p%s, %r)',
|
||
favicon: 'ferongr',
|
||
hotkeys: {
|
||
openQR: ['i', 'Open QR with post number inserted'],
|
||
openEmptyQR: ['I', 'Open QR without post number inserted'],
|
||
openOptions: ['ctrl+o', 'Open Options'],
|
||
close: ['Esc', 'Close Options or QR'],
|
||
spoiler: ['ctrl+s', 'Quick spoiler tags'],
|
||
code: ['alt+c', 'Quick code tags'],
|
||
submit: ['alt+s', 'Submit post'],
|
||
watch: ['w', 'Watch thread'],
|
||
update: ['u', 'Update now'],
|
||
unreadCountTo0: ['z', 'Reset unread status'],
|
||
expandImage: ['m', 'Expand selected image'],
|
||
expandAllImages: ['M', 'Expand all images'],
|
||
zero: ['0', 'Jump to page 0'],
|
||
nextPage: ['L', 'Jump to the next page'],
|
||
previousPage: ['H', 'Jump to the previous page'],
|
||
nextThread: ['n', 'See next thread'],
|
||
previousThread: ['p', 'See previous thread'],
|
||
expandThread: ['e', 'Expand thread'],
|
||
openThreadTab: ['o', 'Open thread in current tab'],
|
||
openThread: ['O', 'Open thread in new tab'],
|
||
nextReply: ['J', 'Select next reply'],
|
||
previousReply: ['K', 'Select previous reply'],
|
||
hide: ['x', 'Hide thread']
|
||
},
|
||
updater: {
|
||
checkbox: {
|
||
'Scrolling': [false, 'Scroll updated posts into view. Only enabled at bottom of page.'],
|
||
'Scroll BG': [false, 'Scroll background tabs'],
|
||
'Verbose': [true, 'Show countdown timer, new post count'],
|
||
'Auto Update': [true, 'Automatically fetch new posts']
|
||
},
|
||
'Interval': 30
|
||
}
|
||
};
|
||
|
||
Conf = {};
|
||
|
||
d = document;
|
||
|
||
g = {};
|
||
|
||
UI = {
|
||
dialog: function(id, position, html) {
|
||
var el;
|
||
el = d.createElement('div');
|
||
el.className = 'reply dialog';
|
||
el.innerHTML = html;
|
||
el.id = id;
|
||
el.style.cssText = localStorage.getItem("" + Main.namespace + id + ".position") || position;
|
||
el.querySelector('.move').addEventListener('mousedown', UI.dragstart, false);
|
||
return el;
|
||
},
|
||
dragstart: function(e) {
|
||
var el, rect;
|
||
e.preventDefault();
|
||
UI.el = el = this.parentNode;
|
||
d.addEventListener('mousemove', UI.drag, false);
|
||
d.addEventListener('mouseup', UI.dragend, false);
|
||
rect = el.getBoundingClientRect();
|
||
UI.dx = e.clientX - rect.left;
|
||
UI.dy = e.clientY - rect.top;
|
||
UI.width = d.documentElement.clientWidth - rect.width;
|
||
return UI.height = d.documentElement.clientHeight - rect.height;
|
||
},
|
||
drag: function(e) {
|
||
var left, style, top;
|
||
left = e.clientX - UI.dx;
|
||
top = e.clientY - UI.dy;
|
||
left = left < 10 ? '0px' : UI.width - left < 10 ? null : left + 'px';
|
||
top = top < 10 ? '0px' : UI.height - top < 10 ? null : top + 'px';
|
||
style = UI.el.style;
|
||
style.left = left;
|
||
style.top = top;
|
||
style.right = left === null ? '0px' : null;
|
||
return style.bottom = top === null ? '0px' : null;
|
||
},
|
||
dragend: function() {
|
||
localStorage.setItem("" + Main.namespace + UI.el.id + ".position", UI.el.style.cssText);
|
||
d.removeEventListener('mousemove', UI.drag, false);
|
||
d.removeEventListener('mouseup', UI.dragend, false);
|
||
return delete UI.el;
|
||
},
|
||
hover: function(e) {
|
||
var clientHeight, clientWidth, clientX, clientY, height, style, top, _ref;
|
||
clientX = e.clientX, clientY = e.clientY;
|
||
style = UI.el.style;
|
||
_ref = d.documentElement, clientHeight = _ref.clientHeight, clientWidth = _ref.clientWidth;
|
||
height = UI.el.offsetHeight;
|
||
top = clientY - 120;
|
||
style.top = clientHeight <= height || top <= 0 ? '0px' : top + height >= clientHeight ? clientHeight - height + 'px' : top + 'px';
|
||
if (clientX <= clientWidth - 400) {
|
||
style.left = clientX + 45 + 'px';
|
||
return style.right = null;
|
||
} else {
|
||
style.left = null;
|
||
return style.right = clientWidth - clientX + 45 + 'px';
|
||
}
|
||
},
|
||
hoverend: function() {
|
||
$.rm(UI.el);
|
||
return delete UI.el;
|
||
}
|
||
};
|
||
|
||
/*
|
||
loosely follows the jquery api:
|
||
http://api.jquery.com/
|
||
not chainable
|
||
*/
|
||
|
||
|
||
$ = function(selector, root) {
|
||
if (root == null) {
|
||
root = d.body;
|
||
}
|
||
return root.querySelector(selector);
|
||
};
|
||
|
||
$.extend = function(object, properties) {
|
||
var key, val;
|
||
for (key in properties) {
|
||
val = properties[key];
|
||
object[key] = val;
|
||
}
|
||
};
|
||
|
||
$.extend($, {
|
||
SECOND: 1000,
|
||
MINUTE: 1000 * 60,
|
||
HOUR: 1000 * 60 * 60,
|
||
DAY: 1000 * 60 * 60 * 24,
|
||
log: typeof (_base = console.log).bind === "function" ? _base.bind(console) : void 0,
|
||
engine: /WebKit|Presto|Gecko/.exec(navigator.userAgent)[0].toLowerCase(),
|
||
ready: function(fc) {
|
||
var cb;
|
||
if (/interactive|complete/.test(d.readyState)) {
|
||
return setTimeout(fc);
|
||
}
|
||
cb = function() {
|
||
$.off(d, 'DOMContentLoaded', cb);
|
||
return fc();
|
||
};
|
||
return $.on(d, 'DOMContentLoaded', cb);
|
||
},
|
||
sync: function(key, cb) {
|
||
return $.on(window, 'storage', function(e) {
|
||
if (e.key === ("" + Main.namespace + key)) {
|
||
return cb(JSON.parse(e.newValue));
|
||
}
|
||
});
|
||
},
|
||
id: function(id) {
|
||
return d.getElementById(id);
|
||
},
|
||
formData: function(arg) {
|
||
var fd, key, val;
|
||
if (arg instanceof HTMLFormElement) {
|
||
fd = new FormData(arg);
|
||
} else {
|
||
fd = new FormData();
|
||
for (key in arg) {
|
||
val = arg[key];
|
||
if (val) {
|
||
fd.append(key, val);
|
||
}
|
||
}
|
||
}
|
||
return fd;
|
||
},
|
||
ajax: function(url, callbacks, opts) {
|
||
var form, headers, key, r, type, upCallbacks, val;
|
||
if (opts == null) {
|
||
opts = {};
|
||
}
|
||
type = opts.type, headers = opts.headers, upCallbacks = opts.upCallbacks, form = opts.form;
|
||
r = new XMLHttpRequest();
|
||
type || (type = form && 'post' || 'get');
|
||
r.open(type, url, true);
|
||
for (key in headers) {
|
||
val = headers[key];
|
||
r.setRequestHeader(key, val);
|
||
}
|
||
$.extend(r, callbacks);
|
||
$.extend(r.upload, upCallbacks);
|
||
r.send(form);
|
||
return r;
|
||
},
|
||
cache: function(url, cb) {
|
||
var req;
|
||
if (req = $.cache.requests[url]) {
|
||
if (req.readyState === 4) {
|
||
return cb.call(req);
|
||
} else {
|
||
return req.callbacks.push(cb);
|
||
}
|
||
} else {
|
||
req = $.ajax(url, {
|
||
onload: function() {
|
||
var _i, _len, _ref, _results;
|
||
_ref = this.callbacks;
|
||
_results = [];
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
cb = _ref[_i];
|
||
_results.push(cb.call(this));
|
||
}
|
||
return _results;
|
||
},
|
||
onabort: function() {
|
||
return delete $.cache.requests[url];
|
||
},
|
||
onerror: function() {
|
||
return delete $.cache.requests[url];
|
||
}
|
||
});
|
||
req.callbacks = [cb];
|
||
return $.cache.requests[url] = req;
|
||
}
|
||
},
|
||
cb: {
|
||
checked: function() {
|
||
$.set(this.name, this.checked);
|
||
return Conf[this.name] = this.checked;
|
||
},
|
||
value: function() {
|
||
$.set(this.name, this.value.trim());
|
||
return Conf[this.name] = this.value;
|
||
}
|
||
},
|
||
addStyle: function(css) {
|
||
var style;
|
||
style = $.el('style', {
|
||
textContent: css
|
||
});
|
||
$.add(d.head, style);
|
||
return style;
|
||
},
|
||
x: function(path, root) {
|
||
if (root == null) {
|
||
root = d.body;
|
||
}
|
||
return d.evaluate(path, root, null, 8, null).singleNodeValue;
|
||
},
|
||
addClass: function(el, className) {
|
||
return el.classList.add(className);
|
||
},
|
||
rmClass: function(el, className) {
|
||
return el.classList.remove(className);
|
||
},
|
||
rm: function(el) {
|
||
return el.parentNode.removeChild(el);
|
||
},
|
||
tn: function(s) {
|
||
return d.createTextNode(s);
|
||
},
|
||
nodes: function(nodes) {
|
||
var frag, node, _i, _len;
|
||
if (!(nodes instanceof Array)) {
|
||
return nodes;
|
||
}
|
||
frag = d.createDocumentFragment();
|
||
for (_i = 0, _len = nodes.length; _i < _len; _i++) {
|
||
node = nodes[_i];
|
||
frag.appendChild(node);
|
||
}
|
||
return frag;
|
||
},
|
||
add: function(parent, children) {
|
||
return parent.appendChild($.nodes(children));
|
||
},
|
||
prepend: function(parent, children) {
|
||
return parent.insertBefore($.nodes(children), parent.firstChild);
|
||
},
|
||
after: function(root, el) {
|
||
return root.parentNode.insertBefore($.nodes(el), root.nextSibling);
|
||
},
|
||
before: function(root, el) {
|
||
return root.parentNode.insertBefore($.nodes(el), root);
|
||
},
|
||
replace: function(root, el) {
|
||
return root.parentNode.replaceChild($.nodes(el), root);
|
||
},
|
||
el: function(tag, properties) {
|
||
var el;
|
||
el = d.createElement(tag);
|
||
if (properties) {
|
||
$.extend(el, properties);
|
||
}
|
||
return el;
|
||
},
|
||
on: function(el, events, handler) {
|
||
var event, _i, _len, _ref;
|
||
_ref = events.split(' ');
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
event = _ref[_i];
|
||
el.addEventListener(event, handler, false);
|
||
}
|
||
},
|
||
off: function(el, events, handler) {
|
||
var event, _i, _len, _ref;
|
||
_ref = events.split(' ');
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
event = _ref[_i];
|
||
el.removeEventListener(event, handler, false);
|
||
}
|
||
},
|
||
open: function(url) {
|
||
return (GM_openInTab || window.open)(location.protocol + url, '_blank');
|
||
},
|
||
event: function(el, e) {
|
||
return el.dispatchEvent(e);
|
||
},
|
||
globalEval: function(code) {
|
||
var script;
|
||
script = $.el('script', {
|
||
textContent: code
|
||
});
|
||
$.add(d.head, script);
|
||
return $.rm(script);
|
||
},
|
||
bytesToString: function(size) {
|
||
var unit;
|
||
unit = 0;
|
||
while (size >= 1024) {
|
||
size /= 1024;
|
||
unit++;
|
||
}
|
||
size = unit > 1 ? Math.round(size * 100) / 100 : Math.round(size);
|
||
return "" + size + " " + ['B', 'KB', 'MB', 'GB'][unit];
|
||
}
|
||
});
|
||
|
||
$.cache.requests = {};
|
||
|
||
$.extend($, typeof GM_deleteValue !== "undefined" && GM_deleteValue !== null ? {
|
||
"delete": function(name) {
|
||
name = Main.namespace + name;
|
||
return GM_deleteValue(name);
|
||
},
|
||
get: function(name, defaultValue) {
|
||
var value;
|
||
name = Main.namespace + name;
|
||
if (value = GM_getValue(name)) {
|
||
return JSON.parse(value);
|
||
} else {
|
||
return defaultValue;
|
||
}
|
||
},
|
||
set: function(name, value) {
|
||
name = Main.namespace + name;
|
||
localStorage.setItem(name, JSON.stringify(value));
|
||
return GM_setValue(name, JSON.stringify(value));
|
||
}
|
||
} : {
|
||
"delete": function(name) {
|
||
return localStorage.removeItem(Main.namespace + name);
|
||
},
|
||
get: function(name, defaultValue) {
|
||
var value;
|
||
if (value = localStorage.getItem(Main.namespace + name)) {
|
||
return JSON.parse(value);
|
||
} else {
|
||
return defaultValue;
|
||
}
|
||
},
|
||
set: function(name, value) {
|
||
return localStorage.setItem(Main.namespace + name, JSON.stringify(value));
|
||
}
|
||
});
|
||
|
||
$$ = function(selector, root) {
|
||
if (root == null) {
|
||
root = d.body;
|
||
}
|
||
return Array.prototype.slice.call(root.querySelectorAll(selector));
|
||
};
|
||
|
||
Filter = {
|
||
filters: {},
|
||
init: function() {
|
||
var boards, filter, hl, key, op, regexp, stub, top, _i, _len, _ref, _ref1, _ref2, _ref3, _ref4;
|
||
for (key in Config.filter) {
|
||
this.filters[key] = [];
|
||
_ref = Conf[key].split('\n');
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
filter = _ref[_i];
|
||
if (filter[0] === '#') {
|
||
continue;
|
||
}
|
||
if (!(regexp = filter.match(/\/(.+)\/(\w*)/))) {
|
||
continue;
|
||
}
|
||
filter = filter.replace(regexp[0], '');
|
||
boards = ((_ref1 = filter.match(/boards:([^;]+)/)) != null ? _ref1[1].toLowerCase() : void 0) || 'global';
|
||
if (boards !== 'global' && boards.split(',').indexOf(g.BOARD) === -1) {
|
||
continue;
|
||
}
|
||
try {
|
||
if (key === 'md5') {
|
||
regexp = regexp[1];
|
||
} else {
|
||
regexp = RegExp(regexp[1], regexp[2]);
|
||
}
|
||
} catch (e) {
|
||
alert(e.message);
|
||
continue;
|
||
}
|
||
op = ((_ref2 = filter.match(/[^t]op:(yes|no|only)/)) != null ? _ref2[1] : void 0) || 'no';
|
||
stub = (function() {
|
||
var _ref3;
|
||
switch ((_ref3 = filter.match(/stub:(yes|no)/)) != null ? _ref3[1] : void 0) {
|
||
case 'yes':
|
||
return true;
|
||
case 'no':
|
||
return false;
|
||
default:
|
||
return Conf['Show Stubs'];
|
||
}
|
||
})();
|
||
if (hl = /highlight/.test(filter)) {
|
||
hl = ((_ref3 = filter.match(/highlight:(\w+)/)) != null ? _ref3[1] : void 0) || 'filter_highlight';
|
||
top = ((_ref4 = filter.match(/top:(yes|no)/)) != null ? _ref4[1] : void 0) || 'yes';
|
||
top = top === 'yes';
|
||
}
|
||
this.filters[key].push(this.createFilter(regexp, op, stub, hl, top));
|
||
}
|
||
if (!this.filters[key].length) {
|
||
delete this.filters[key];
|
||
}
|
||
}
|
||
if (Object.keys(this.filters).length) {
|
||
return Main.callbacks.push(this.node);
|
||
}
|
||
},
|
||
createFilter: function(regexp, op, stub, hl, top) {
|
||
var settings, test;
|
||
test = typeof regexp === 'string' ? function(value) {
|
||
return regexp === value;
|
||
} : function(value) {
|
||
return regexp.test(value);
|
||
};
|
||
settings = {
|
||
hide: !hl,
|
||
stub: stub,
|
||
"class": hl,
|
||
top: top
|
||
};
|
||
return function(value, isOP) {
|
||
if (isOP && op === 'no' || !isOP && op === 'only') {
|
||
return false;
|
||
}
|
||
if (!test(value)) {
|
||
return false;
|
||
}
|
||
return settings;
|
||
};
|
||
},
|
||
node: function(post) {
|
||
var filter, firstThread, isOP, key, result, root, thisThread, value, _i, _len, _ref;
|
||
if (post.isInlined) {
|
||
return;
|
||
}
|
||
isOP = post.ID === post.threadID;
|
||
root = post.root;
|
||
for (key in Filter.filters) {
|
||
value = Filter[key](post);
|
||
if (value === false) {
|
||
continue;
|
||
}
|
||
_ref = Filter.filters[key];
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
filter = _ref[_i];
|
||
if (!(result = filter(value, isOP))) {
|
||
continue;
|
||
}
|
||
if (result.hide) {
|
||
if (isOP) {
|
||
if (!g.REPLY) {
|
||
ThreadHiding.hide(root.parentNode, result.stub);
|
||
} else {
|
||
continue;
|
||
}
|
||
} else {
|
||
ReplyHiding.hide(root, result.stub);
|
||
}
|
||
return;
|
||
}
|
||
$.addClass(root, result["class"]);
|
||
if (isOP && result.top && !g.REPLY) {
|
||
thisThread = root.parentNode;
|
||
if (firstThread = $('div[class="postContainer opContainer"]').parentNode) {
|
||
$.before(firstThread, [thisThread, thisThread.nextElementSibling]);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
},
|
||
name: function(post) {
|
||
return $('.name', post.el).textContent;
|
||
},
|
||
uniqueid: function(post) {
|
||
var uid;
|
||
if (uid = $('.posteruid', post.el)) {
|
||
return uid.textContent.slice(5, -1);
|
||
}
|
||
return false;
|
||
},
|
||
tripcode: function(post) {
|
||
var trip;
|
||
if (trip = $('.postertrip', post.el)) {
|
||
return trip.textContent;
|
||
}
|
||
return false;
|
||
},
|
||
mod: function(post) {
|
||
var mod;
|
||
if (mod = $('.capcode', post.el)) {
|
||
return mod.textContent;
|
||
}
|
||
return false;
|
||
},
|
||
email: function(post) {
|
||
var mail;
|
||
if (mail = $('.useremail', post.el)) {
|
||
return decodeURIComponent(mail.href.slice(7));
|
||
}
|
||
return false;
|
||
},
|
||
subject: function(post) {
|
||
return $('.postInfo .subject', post.el).textContent || false;
|
||
},
|
||
comment: function(post) {
|
||
var data, i, nodes, text, _i, _ref;
|
||
text = [];
|
||
nodes = d.evaluate('.//br|.//text()', post.blockquote, null, 7, null);
|
||
for (i = _i = 0, _ref = nodes.snapshotLength; 0 <= _ref ? _i < _ref : _i > _ref; i = 0 <= _ref ? ++_i : --_i) {
|
||
text.push((data = nodes.snapshotItem(i).data) ? data : '\n');
|
||
}
|
||
return text.join('');
|
||
},
|
||
country: function(post) {
|
||
var flag;
|
||
if (flag = $('.countryFlag', post.el)) {
|
||
return flag.title.replace('Country: ', '');
|
||
}
|
||
return false;
|
||
},
|
||
filename: function(post) {
|
||
var file, fileInfo;
|
||
fileInfo = post.fileInfo;
|
||
if (fileInfo) {
|
||
if (file = $('.fileText > span', fileInfo)) {
|
||
return file.title;
|
||
} else {
|
||
return fileInfo.firstElementChild.dataset.filename;
|
||
}
|
||
}
|
||
return false;
|
||
},
|
||
dimensions: function(post) {
|
||
var fileInfo, match;
|
||
fileInfo = post.fileInfo;
|
||
if (fileInfo && (match = fileInfo.textContent.match(/\d+x\d+/))) {
|
||
return match[0];
|
||
}
|
||
return false;
|
||
},
|
||
filesize: function(post) {
|
||
var img;
|
||
img = post.img;
|
||
if (img) {
|
||
return img.alt;
|
||
}
|
||
return false;
|
||
},
|
||
md5: function(post) {
|
||
var img;
|
||
img = post.img;
|
||
if (img) {
|
||
return img.dataset.md5;
|
||
}
|
||
return false;
|
||
},
|
||
menuInit: function() {
|
||
var div, entry, type, _i, _len, _ref;
|
||
div = $.el('div', {
|
||
textContent: 'Filter'
|
||
});
|
||
entry = {
|
||
el: div,
|
||
open: function() {
|
||
return true;
|
||
},
|
||
children: []
|
||
};
|
||
_ref = [['Name', 'name'], ['Unique ID', 'uniqueid'], ['Tripcode', 'tripcode'], ['Admin/Mod', 'mod'], ['E-mail', 'email'], ['Subject', 'subject'], ['Comment', 'comment'], ['Country', 'country'], ['Filename', 'filename'], ['Image dimensions', 'dimensions'], ['Filesize', 'filesize'], ['Image MD5', 'md5']];
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
type = _ref[_i];
|
||
entry.children.push(Filter.createSubEntry(type[0], type[1]));
|
||
}
|
||
return Menu.addEntry(entry);
|
||
},
|
||
createSubEntry: function(text, type) {
|
||
var el, onclick, open;
|
||
el = $.el('a', {
|
||
href: 'javascript:;',
|
||
textContent: text
|
||
});
|
||
onclick = null;
|
||
open = function(post) {
|
||
var value;
|
||
value = Filter[type](post);
|
||
if (value === false) {
|
||
return false;
|
||
}
|
||
$.off(el, 'click', onclick);
|
||
onclick = function() {
|
||
var re, save, select, ta, tl;
|
||
re = type === 'md5' ? value : value.replace(/\/|\\|\^|\$|\n|\.|\(|\)|\{|\}|\[|\]|\?|\*|\+|\|/g, function(c) {
|
||
if (c === '\n') {
|
||
return '\\n';
|
||
} else if (c === '\\') {
|
||
return '\\\\';
|
||
} else {
|
||
return "\\" + c;
|
||
}
|
||
});
|
||
re = type === 'md5' ? "/" + value + "/" : "/^" + re + "$/";
|
||
if (/\bop\b/.test(post["class"])) {
|
||
re += ';op:yes';
|
||
}
|
||
save = (save = $.get(type, '')) ? "" + save + "\n" + re : re;
|
||
$.set(type, save);
|
||
Options.dialog();
|
||
select = $('select[name=filter]', $.id('options'));
|
||
select.value = type;
|
||
$.event(select, new Event('change'));
|
||
$.id('filter_tab').checked = true;
|
||
ta = select.nextElementSibling;
|
||
tl = ta.textLength;
|
||
ta.setSelectionRange(tl, tl);
|
||
return ta.focus();
|
||
};
|
||
$.on(el, 'click', onclick);
|
||
return true;
|
||
};
|
||
return {
|
||
el: el,
|
||
open: open
|
||
};
|
||
}
|
||
};
|
||
|
||
StrikethroughQuotes = {
|
||
init: function() {
|
||
return Main.callbacks.push(this.node);
|
||
},
|
||
node: function(post) {
|
||
var el, quote, show_stub, _i, _len, _ref;
|
||
if (post.isInlined) {
|
||
return;
|
||
}
|
||
_ref = post.quotes;
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
quote = _ref[_i];
|
||
if ((el = $.id(quote.hash.slice(1))) && el.hidden) {
|
||
$.addClass(quote, 'filtered');
|
||
if (Conf['Recursive Filtering']) {
|
||
show_stub = !!$.x('preceding-sibling::div[contains(@class,"stub")]', el);
|
||
ReplyHiding.hide(post.root, show_stub);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
};
|
||
|
||
ExpandComment = {
|
||
init: function() {
|
||
var a, _i, _len, _ref;
|
||
_ref = $$('.abbr');
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
a = _ref[_i];
|
||
$.on(a.firstElementChild, 'click', ExpandComment.expand);
|
||
}
|
||
},
|
||
expand: function(e) {
|
||
var a, replyID, threadID, _, _ref;
|
||
e.preventDefault();
|
||
_ref = this.href.match(/(\d+)#p(\d+)/), _ = _ref[0], threadID = _ref[1], replyID = _ref[2];
|
||
this.textContent = "Loading " + replyID + "...";
|
||
a = this;
|
||
return $.cache(this.pathname, function() {
|
||
return ExpandComment.parse(this, a, threadID, replyID);
|
||
});
|
||
},
|
||
parse: function(req, a, threadID, replyID) {
|
||
var doc, href, node, post, quote, quotes, _i, _len;
|
||
if (req.status !== 200) {
|
||
a.textContent = "" + req.status + " " + req.statusText;
|
||
return;
|
||
}
|
||
doc = d.implementation.createHTMLDocument('');
|
||
doc.documentElement.innerHTML = req.response;
|
||
node = d.importNode(doc.getElementById("m" + replyID), true);
|
||
quotes = node.getElementsByClassName('quotelink');
|
||
for (_i = 0, _len = quotes.length; _i < _len; _i++) {
|
||
quote = quotes[_i];
|
||
href = quote.getAttribute('href');
|
||
if (href[0] === '/') {
|
||
continue;
|
||
}
|
||
quote.href = "res/" + href;
|
||
}
|
||
post = {
|
||
blockquote: node,
|
||
threadID: threadID,
|
||
quotes: quotes,
|
||
backlinks: []
|
||
};
|
||
if (Conf['Resurrect Quotes']) {
|
||
Quotify.node(post);
|
||
}
|
||
if (Conf['Quote Preview']) {
|
||
QuotePreview.node(post);
|
||
}
|
||
if (Conf['Quote Inline']) {
|
||
QuoteInline.node(post);
|
||
}
|
||
if (Conf['Indicate OP quote']) {
|
||
QuoteOP.node(post);
|
||
}
|
||
if (Conf['Indicate Cross-thread Quotes']) {
|
||
QuoteCT.node(post);
|
||
}
|
||
$.replace(a.parentNode.parentNode, node);
|
||
return Main.prettify(node);
|
||
}
|
||
};
|
||
|
||
ExpandThread = {
|
||
init: function() {
|
||
var a, span, _i, _len, _ref, _results;
|
||
_ref = $$('.summary');
|
||
_results = [];
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
span = _ref[_i];
|
||
a = $.el('a', {
|
||
textContent: "+ " + span.textContent,
|
||
className: 'summary desktop',
|
||
href: 'javascript:;'
|
||
});
|
||
$.on(a, 'click', function() {
|
||
return ExpandThread.toggle(this.parentNode);
|
||
});
|
||
_results.push($.replace(span, a));
|
||
}
|
||
return _results;
|
||
},
|
||
toggle: function(thread) {
|
||
var a, num, pathname, replies, reply, _i, _len;
|
||
pathname = "/" + g.BOARD + "/res/" + thread.id.slice(1);
|
||
a = $('.summary', thread);
|
||
switch (a.textContent[0]) {
|
||
case '+':
|
||
a.textContent = a.textContent.replace('+', '× Loading...');
|
||
$.cache(pathname, function() {
|
||
return ExpandThread.parse(this, thread, a);
|
||
});
|
||
break;
|
||
case '×':
|
||
a.textContent = a.textContent.replace('× Loading...', '+');
|
||
$.cache.requests[pathname].abort();
|
||
break;
|
||
case '-':
|
||
a.textContent = a.textContent.replace('-', '+');
|
||
num = (function() {
|
||
switch (g.BOARD) {
|
||
case 'b':
|
||
case 'vg':
|
||
return 3;
|
||
case 't':
|
||
return 1;
|
||
default:
|
||
return 5;
|
||
}
|
||
})();
|
||
replies = $$('.replyContainer', thread);
|
||
replies.splice(replies.length - num, num);
|
||
for (_i = 0, _len = replies.length; _i < _len; _i++) {
|
||
reply = replies[_i];
|
||
$.rm(reply);
|
||
}
|
||
}
|
||
},
|
||
parse: function(req, thread, a) {
|
||
var backlink, doc, href, id, link, nodes, post, quote, reply, threadID, _i, _j, _k, _l, _len, _len1, _len2, _len3, _ref, _ref1, _ref2, _ref3;
|
||
if (req.status !== 200) {
|
||
a.textContent = "" + req.status + " " + req.statusText;
|
||
$.off(a, 'click', ExpandThread.cb.toggle);
|
||
return;
|
||
}
|
||
a.textContent = a.textContent.replace('× Loading...', '-');
|
||
doc = d.implementation.createHTMLDocument('');
|
||
doc.documentElement.innerHTML = req.response;
|
||
threadID = thread.id.slice(1);
|
||
nodes = [];
|
||
_ref = $$('.replyContainer', doc);
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
reply = _ref[_i];
|
||
reply = d.importNode(reply, true);
|
||
_ref1 = $$('.quotelink', reply);
|
||
for (_j = 0, _len1 = _ref1.length; _j < _len1; _j++) {
|
||
quote = _ref1[_j];
|
||
href = quote.getAttribute('href');
|
||
if (href[0] === '/') {
|
||
continue;
|
||
}
|
||
quote.href = "res/" + href;
|
||
}
|
||
id = reply.id.slice(2);
|
||
link = $('.postNum > a[title="Highlight this post"]', reply);
|
||
link.href = "res/" + threadID + "#p" + id;
|
||
link.nextSibling.href = "res/" + threadID + "#q" + id;
|
||
nodes.push(reply);
|
||
}
|
||
_ref2 = $$('.summary ~ .replyContainer', a.parentNode);
|
||
for (_k = 0, _len2 = _ref2.length; _k < _len2; _k++) {
|
||
post = _ref2[_k];
|
||
$.rm(post);
|
||
}
|
||
_ref3 = $$('.backlink', a.previousElementSibling);
|
||
for (_l = 0, _len3 = _ref3.length; _l < _len3; _l++) {
|
||
backlink = _ref3[_l];
|
||
if (!$.id(backlink.hash.slice(1))) {
|
||
$.rm(backlink);
|
||
}
|
||
}
|
||
return $.after(a, nodes);
|
||
}
|
||
};
|
||
|
||
ThreadHiding = {
|
||
init: function() {
|
||
var a, hiddenThreads, thread, _i, _len, _ref;
|
||
hiddenThreads = $.get("hiddenThreads/" + g.BOARD + "/", {});
|
||
_ref = $$('.thread');
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
thread = _ref[_i];
|
||
a = $.el('a', {
|
||
className: 'hide_thread_button',
|
||
innerHTML: '<span>[ - ]</span>',
|
||
href: 'javascript:;'
|
||
});
|
||
$.on(a, 'click', ThreadHiding.cb);
|
||
$.prepend(thread, a);
|
||
if (thread.id.slice(1) in hiddenThreads) {
|
||
ThreadHiding.hide(thread);
|
||
}
|
||
}
|
||
},
|
||
cb: function() {
|
||
return ThreadHiding.toggle($.x('ancestor::div[parent::div[@class="board"]]', this));
|
||
},
|
||
toggle: function(thread) {
|
||
var hiddenThreads, id;
|
||
hiddenThreads = $.get("hiddenThreads/" + g.BOARD + "/", {});
|
||
id = thread.id.slice(1);
|
||
if (thread.hidden || /\bhidden_thread\b/.test(thread.firstChild.className)) {
|
||
ThreadHiding.show(thread);
|
||
delete hiddenThreads[id];
|
||
} else {
|
||
ThreadHiding.hide(thread);
|
||
hiddenThreads[id] = Date.now();
|
||
}
|
||
return $.set("hiddenThreads/" + g.BOARD + "/", hiddenThreads);
|
||
},
|
||
hide: function(thread, show_stub) {
|
||
var a, menuButton, num, opInfo, span, stub, text;
|
||
if (show_stub == null) {
|
||
show_stub = Conf['Show Stubs'];
|
||
}
|
||
if (!show_stub) {
|
||
thread.hidden = true;
|
||
thread.nextElementSibling.hidden = true;
|
||
return;
|
||
}
|
||
if (/\bhidden_thread\b/.test(thread.firstChild.className)) {
|
||
return;
|
||
}
|
||
num = 0;
|
||
if (span = $('.summary', thread)) {
|
||
num = Number(span.textContent.match(/\d+/));
|
||
}
|
||
num += $$('.opContainer ~ .replyContainer', thread).length;
|
||
text = num === 1 ? '1 reply' : "" + num + " replies";
|
||
opInfo = $('.op > .postInfo > .nameBlock', thread).textContent;
|
||
stub = $.el('div', {
|
||
className: 'hide_thread_button hidden_thread',
|
||
innerHTML: '<a href="javascript:;"><span>[ + ]</span> </a>'
|
||
});
|
||
a = stub.firstChild;
|
||
$.on(a, 'click', ThreadHiding.cb);
|
||
$.add(a, $.tn("" + opInfo + " (" + text + ")"));
|
||
if (Conf['Menu']) {
|
||
menuButton = Menu.a.cloneNode(true);
|
||
$.on(menuButton, 'click', Menu.toggle);
|
||
$.add(stub, [$.tn(' '), menuButton]);
|
||
}
|
||
return $.prepend(thread, stub);
|
||
},
|
||
show: function(thread) {
|
||
var stub;
|
||
if (stub = $('.hidden_thread', thread)) {
|
||
$.rm(stub);
|
||
}
|
||
thread.hidden = false;
|
||
return thread.nextElementSibling.hidden = false;
|
||
}
|
||
};
|
||
|
||
ReplyHiding = {
|
||
init: function() {
|
||
return Main.callbacks.push(this.node);
|
||
},
|
||
node: function(post) {
|
||
var side;
|
||
if (post.isInlined || post.ID === post.threadID) {
|
||
return;
|
||
}
|
||
side = $('.sideArrows', post.root);
|
||
$.addClass(side, 'hide_reply_button');
|
||
side.innerHTML = '<a href="javascript:;"><span>[ - ]</span></a>';
|
||
$.on(side.firstChild, 'click', ReplyHiding.toggle);
|
||
if (post.ID in g.hiddenReplies) {
|
||
return ReplyHiding.hide(post.root);
|
||
}
|
||
},
|
||
toggle: function() {
|
||
var button, id, quote, quotes, root, _i, _j, _len, _len1;
|
||
button = this.parentNode;
|
||
root = button.parentNode;
|
||
id = root.id.slice(2);
|
||
quotes = $$(".quotelink[href$='#p" + id + "'], .backlink[href$='#p" + id + "']");
|
||
if (/\bstub\b/.test(button.className)) {
|
||
ReplyHiding.show(root);
|
||
for (_i = 0, _len = quotes.length; _i < _len; _i++) {
|
||
quote = quotes[_i];
|
||
$.rmClass(quote, 'filtered');
|
||
}
|
||
delete g.hiddenReplies[id];
|
||
} else {
|
||
ReplyHiding.hide(root);
|
||
for (_j = 0, _len1 = quotes.length; _j < _len1; _j++) {
|
||
quote = quotes[_j];
|
||
$.addClass(quote, 'filtered');
|
||
}
|
||
g.hiddenReplies[id] = Date.now();
|
||
}
|
||
return $.set("hiddenReplies/" + g.BOARD + "/", g.hiddenReplies);
|
||
},
|
||
hide: function(root, show_stub) {
|
||
var a, el, menuButton, side, stub;
|
||
if (show_stub == null) {
|
||
show_stub = Conf['Show Stubs'];
|
||
}
|
||
side = $('.sideArrows', root);
|
||
if (side.hidden) {
|
||
return;
|
||
}
|
||
side.hidden = true;
|
||
el = side.nextElementSibling;
|
||
el.hidden = true;
|
||
if (!show_stub) {
|
||
return;
|
||
}
|
||
stub = $.el('div', {
|
||
className: 'hide_reply_button stub',
|
||
innerHTML: '<a href="javascript:;"><span>[ + ]</span> </a>'
|
||
});
|
||
a = stub.firstChild;
|
||
$.on(a, 'click', ReplyHiding.toggle);
|
||
$.add(a, $.tn($('.nameBlock', el).textContent));
|
||
if (Conf['Menu']) {
|
||
menuButton = Menu.a.cloneNode(true);
|
||
$.on(menuButton, 'click', Menu.toggle);
|
||
$.add(stub, [$.tn(' '), menuButton]);
|
||
}
|
||
return $.prepend(root, stub);
|
||
},
|
||
show: function(root) {
|
||
var stub;
|
||
if (stub = $('.stub', root)) {
|
||
$.rm(stub);
|
||
}
|
||
$('.sideArrows', root).hidden = false;
|
||
return $('.post', root).hidden = false;
|
||
}
|
||
};
|
||
|
||
Menu = {
|
||
entries: [],
|
||
init: function() {
|
||
this.a = $.el('a', {
|
||
className: 'menu_button',
|
||
href: 'javascript:;',
|
||
innerHTML: '[<span></span>]'
|
||
});
|
||
this.el = $.el('div', {
|
||
className: 'reply dialog',
|
||
id: 'menu',
|
||
tabIndex: 0
|
||
});
|
||
$.on(this.el, 'click', function(e) {
|
||
return e.stopPropagation();
|
||
});
|
||
$.on(this.el, 'keydown', this.keybinds);
|
||
$.on(d, 'AddMenuEntry', function(e) {
|
||
return Menu.addEntry(e.detail);
|
||
});
|
||
return Main.callbacks.push(this.node);
|
||
},
|
||
node: function(post) {
|
||
var a;
|
||
if (post.isInlined && !post.isCrosspost) {
|
||
a = $('.menu_button', post.el);
|
||
} else {
|
||
a = Menu.a.cloneNode(true);
|
||
$.add($('.postInfo', post.el), a);
|
||
}
|
||
return $.on(a, 'click', Menu.toggle);
|
||
},
|
||
toggle: function(e) {
|
||
var lastOpener, post;
|
||
e.preventDefault();
|
||
e.stopPropagation();
|
||
if (Menu.el.parentNode) {
|
||
lastOpener = Menu.lastOpener;
|
||
Menu.close();
|
||
if (lastOpener === this) {
|
||
return;
|
||
}
|
||
}
|
||
Menu.lastOpener = this;
|
||
post = /\bhidden_thread\b/.test(this.parentNode.className) ? $.x('ancestor::div[parent::div[@class="board"]]/child::div[contains(@class,"opContainer")]', this) : $.x('ancestor::div[contains(@class,"postContainer")][1]', this);
|
||
return Menu.open(this, Main.preParse(post));
|
||
},
|
||
open: function(button, post) {
|
||
var bLeft, bRect, bTop, el, entry, funk, mRect, _i, _len, _ref;
|
||
el = Menu.el;
|
||
el.setAttribute('data-id', post.ID);
|
||
el.setAttribute('data-rootid', post.root.id);
|
||
funk = function(entry, parent) {
|
||
var child, children, subMenu, _i, _len;
|
||
children = entry.children;
|
||
if (!entry.open(post)) {
|
||
return;
|
||
}
|
||
$.add(parent, entry.el);
|
||
if (!children) {
|
||
return;
|
||
}
|
||
if (subMenu = $('.subMenu', entry.el)) {
|
||
$.rm(subMenu);
|
||
}
|
||
subMenu = $.el('div', {
|
||
className: 'reply dialog subMenu'
|
||
});
|
||
$.add(entry.el, subMenu);
|
||
for (_i = 0, _len = children.length; _i < _len; _i++) {
|
||
child = children[_i];
|
||
funk(child, subMenu);
|
||
}
|
||
};
|
||
_ref = Menu.entries;
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
entry = _ref[_i];
|
||
funk(entry, el);
|
||
}
|
||
Menu.focus($('.entry', Menu.el));
|
||
$.on(d, 'click', Menu.close);
|
||
$.add(d.body, el);
|
||
mRect = el.getBoundingClientRect();
|
||
bRect = button.getBoundingClientRect();
|
||
bTop = d.documentElement.scrollTop + d.body.scrollTop + bRect.top;
|
||
bLeft = d.documentElement.scrollLeft + d.body.scrollLeft + bRect.left;
|
||
el.style.top = bRect.top + bRect.height + mRect.height < d.documentElement.clientHeight ? bTop + bRect.height + 2 + 'px' : bTop - mRect.height - 2 + 'px';
|
||
el.style.left = bRect.left + mRect.width < d.documentElement.clientWidth ? bLeft + 'px' : bLeft + bRect.width - mRect.width + 'px';
|
||
return el.focus();
|
||
},
|
||
close: function() {
|
||
var el, focused, _i, _len, _ref;
|
||
el = Menu.el;
|
||
$.rm(el);
|
||
_ref = $$('.focused.entry', el);
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
focused = _ref[_i];
|
||
$.rmClass(focused, 'focused');
|
||
}
|
||
el.innerHTML = null;
|
||
el.removeAttribute('style');
|
||
delete Menu.lastOpener;
|
||
delete Menu.focusedEntry;
|
||
return $.off(d, 'click', Menu.close);
|
||
},
|
||
keybinds: function(e) {
|
||
var el, next, subMenu;
|
||
el = Menu.focusedEntry;
|
||
switch (Keybinds.keyCode(e) || e.keyCode) {
|
||
case 'Esc':
|
||
Menu.lastOpener.focus();
|
||
Menu.close();
|
||
break;
|
||
case 13:
|
||
case 32:
|
||
el.click();
|
||
break;
|
||
case 'Up':
|
||
if (next = el.previousElementSibling) {
|
||
Menu.focus(next);
|
||
}
|
||
break;
|
||
case 'Down':
|
||
if (next = el.nextElementSibling) {
|
||
Menu.focus(next);
|
||
}
|
||
break;
|
||
case 'Right':
|
||
if ((subMenu = $('.subMenu', el)) && (next = subMenu.firstElementChild)) {
|
||
Menu.focus(next);
|
||
}
|
||
break;
|
||
case 'Left':
|
||
if (next = $.x('parent::*[contains(@class,"subMenu")]/parent::*', el)) {
|
||
Menu.focus(next);
|
||
}
|
||
break;
|
||
default:
|
||
return;
|
||
}
|
||
e.preventDefault();
|
||
return e.stopPropagation();
|
||
},
|
||
focus: function(el) {
|
||
var focused, _i, _len, _ref;
|
||
if (focused = $.x('parent::*/child::*[contains(@class,"focused")]', el)) {
|
||
$.rmClass(focused, 'focused');
|
||
}
|
||
_ref = $$('.focused', el);
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
focused = _ref[_i];
|
||
$.rmClass(focused, 'focused');
|
||
}
|
||
Menu.focusedEntry = el;
|
||
return $.addClass(el, 'focused');
|
||
},
|
||
addEntry: function(entry) {
|
||
var funk;
|
||
funk = function(entry) {
|
||
var child, children, el, _i, _len;
|
||
el = entry.el, children = entry.children;
|
||
$.addClass(el, 'entry');
|
||
$.on(el, 'focus mouseover', function(e) {
|
||
e.stopPropagation();
|
||
return Menu.focus(this);
|
||
});
|
||
if (!children) {
|
||
return;
|
||
}
|
||
$.addClass(el, 'hasSubMenu');
|
||
for (_i = 0, _len = children.length; _i < _len; _i++) {
|
||
child = children[_i];
|
||
funk(child);
|
||
}
|
||
};
|
||
funk(entry);
|
||
return Menu.entries.push(entry);
|
||
}
|
||
};
|
||
|
||
Keybinds = {
|
||
init: function() {
|
||
var node, _i, _len, _ref;
|
||
_ref = $$('[accesskey]');
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
node = _ref[_i];
|
||
node.removeAttribute('accesskey');
|
||
}
|
||
return $.on(d, 'keydown', Keybinds.keydown);
|
||
},
|
||
keydown: function(e) {
|
||
var key, link, o, target, thread;
|
||
if (!(key = Keybinds.keyCode(e))) {
|
||
return;
|
||
}
|
||
target = e.target;
|
||
if (/TEXTAREA|INPUT/.test(target.nodeName)) {
|
||
if (!((key === 'Esc') || (/\+/.test(key)))) {
|
||
return;
|
||
}
|
||
}
|
||
thread = Nav.getThread();
|
||
switch (key) {
|
||
case Conf.openQR:
|
||
Keybinds.qr(thread, true);
|
||
break;
|
||
case Conf.openEmptyQR:
|
||
Keybinds.qr(thread);
|
||
break;
|
||
case Conf.openOptions:
|
||
if (!$.id('overlay')) {
|
||
Options.dialog();
|
||
}
|
||
break;
|
||
case Conf.close:
|
||
if (o = $.id('overlay')) {
|
||
Options.close.call(o);
|
||
} else if (QR.el) {
|
||
QR.close();
|
||
}
|
||
break;
|
||
case Conf.submit:
|
||
if (QR.el && !QR.status()) {
|
||
QR.submit();
|
||
}
|
||
break;
|
||
case Conf.spoiler:
|
||
if (target.nodeName !== 'TEXTAREA') {
|
||
return;
|
||
}
|
||
Keybinds.tags('spoiler', target);
|
||
break;
|
||
case Conf.code:
|
||
if (target.nodeName !== 'TEXTAREA') {
|
||
return;
|
||
}
|
||
Keybinds.tags('code', target);
|
||
break;
|
||
case Conf.watch:
|
||
Watcher.toggle(thread);
|
||
break;
|
||
case Conf.update:
|
||
Updater.update();
|
||
break;
|
||
case Conf.unreadCountTo0:
|
||
Unread.replies = [];
|
||
Unread.update(true);
|
||
break;
|
||
case Conf.expandImage:
|
||
Keybinds.img(thread);
|
||
break;
|
||
case Conf.expandAllImages:
|
||
Keybinds.img(thread, true);
|
||
break;
|
||
case Conf.zero:
|
||
window.location = "/" + g.BOARD + "/0#delform";
|
||
break;
|
||
case Conf.nextPage:
|
||
if (link = $('link[rel=next]', d.head)) {
|
||
window.location = link.href;
|
||
}
|
||
break;
|
||
case Conf.previousPage:
|
||
if (link = $('link[rel=prev]', d.head)) {
|
||
window.location.href = link.href;
|
||
}
|
||
break;
|
||
case Conf.nextThread:
|
||
if (g.REPLY) {
|
||
return;
|
||
}
|
||
Nav.scroll(+1);
|
||
break;
|
||
case Conf.previousThread:
|
||
if (g.REPLY) {
|
||
return;
|
||
}
|
||
Nav.scroll(-1);
|
||
break;
|
||
case Conf.expandThread:
|
||
ExpandThread.toggle(thread);
|
||
break;
|
||
case Conf.openThread:
|
||
Keybinds.open(thread);
|
||
break;
|
||
case Conf.openThreadTab:
|
||
Keybinds.open(thread, true);
|
||
break;
|
||
case Conf.nextReply:
|
||
Keybinds.hl(+1, thread);
|
||
break;
|
||
case Conf.previousReply:
|
||
Keybinds.hl(-1, thread);
|
||
break;
|
||
case Conf.hide:
|
||
if (/\bthread\b/.test(thread.className)) {
|
||
ThreadHiding.toggle(thread);
|
||
}
|
||
break;
|
||
default:
|
||
return;
|
||
}
|
||
return e.preventDefault();
|
||
},
|
||
keyCode: function(e) {
|
||
var c, kc, key;
|
||
key = (function() {
|
||
switch (kc = e.keyCode) {
|
||
case 8:
|
||
return '';
|
||
case 27:
|
||
return 'Esc';
|
||
case 37:
|
||
return 'Left';
|
||
case 38:
|
||
return 'Up';
|
||
case 39:
|
||
return 'Right';
|
||
case 40:
|
||
return 'Down';
|
||
case 48:
|
||
case 49:
|
||
case 50:
|
||
case 51:
|
||
case 52:
|
||
case 53:
|
||
case 54:
|
||
case 55:
|
||
case 56:
|
||
case 57:
|
||
case 65:
|
||
case 66:
|
||
case 67:
|
||
case 68:
|
||
case 69:
|
||
case 70:
|
||
case 71:
|
||
case 72:
|
||
case 73:
|
||
case 74:
|
||
case 75:
|
||
case 76:
|
||
case 77:
|
||
case 78:
|
||
case 79:
|
||
case 80:
|
||
case 81:
|
||
case 82:
|
||
case 83:
|
||
case 84:
|
||
case 85:
|
||
case 86:
|
||
case 87:
|
||
case 88:
|
||
case 89:
|
||
case 90:
|
||
c = String.fromCharCode(kc);
|
||
if (e.shiftKey) {
|
||
return c;
|
||
} else {
|
||
return c.toLowerCase();
|
||
}
|
||
break;
|
||
default:
|
||
return null;
|
||
}
|
||
})();
|
||
if (key) {
|
||
if (e.altKey) {
|
||
key = 'alt+' + key;
|
||
}
|
||
if (e.ctrlKey) {
|
||
key = 'ctrl+' + key;
|
||
}
|
||
if (e.metaKey) {
|
||
key = 'meta+' + key;
|
||
}
|
||
}
|
||
return key;
|
||
},
|
||
tags: function(tag, ta) {
|
||
var range, selEnd, selStart, value;
|
||
value = ta.value;
|
||
selStart = ta.selectionStart;
|
||
selEnd = ta.selectionEnd;
|
||
ta.value = value.slice(0, selStart) + ("[" + tag + "]") + value.slice(selStart, selEnd) + ("[/" + tag + "]") + value.slice(selEnd);
|
||
range = ("[" + tag + "]").length + selEnd;
|
||
ta.setSelectionRange(range, range);
|
||
return $.event(ta, new Event('input'));
|
||
},
|
||
img: function(thread, all) {
|
||
var thumb;
|
||
if (all) {
|
||
return $.id('imageExpand').click();
|
||
} else {
|
||
thumb = $('img[data-md5]', $('.post.highlight', thread) || thread);
|
||
return ImageExpand.toggle(thumb.parentNode);
|
||
}
|
||
},
|
||
qr: function(thread, quote) {
|
||
if (quote) {
|
||
QR.quote.call($('.postNum > a[title="Quote this post"]', $('.post.highlight', thread) || thread));
|
||
} else {
|
||
QR.open();
|
||
}
|
||
return $('textarea', QR.el).focus();
|
||
},
|
||
open: function(thread, tab) {
|
||
var id, url;
|
||
id = thread.id.slice(1);
|
||
url = "//boards.4chan.org/" + g.BOARD + "/res/" + id;
|
||
if (tab) {
|
||
return $.open(url);
|
||
} else {
|
||
return location.href = url;
|
||
}
|
||
},
|
||
hl: function(delta, thread) {
|
||
var next, post, rect, replies, reply, _i, _len;
|
||
if (post = $('.reply.highlight', thread)) {
|
||
$.rmClass(post, 'highlight');
|
||
post.removeAttribute('tabindex');
|
||
rect = post.getBoundingClientRect();
|
||
if (rect.bottom >= 0 && rect.top <= d.documentElement.clientHeight) {
|
||
next = $.x('child::div[contains(@class,"post reply")]', delta === +1 ? post.parentNode.nextElementSibling : post.parentNode.previousElementSibling);
|
||
if (!next) {
|
||
this.focus(post);
|
||
return;
|
||
}
|
||
if (!(g.REPLY || $.x('ancestor::div[parent::div[@class="board"]]', next) === thread)) {
|
||
return;
|
||
}
|
||
rect = next.getBoundingClientRect();
|
||
if (rect.top < 0 || rect.bottom > d.documentElement.clientHeight) {
|
||
next.scrollIntoView(delta === -1);
|
||
}
|
||
this.focus(next);
|
||
return;
|
||
}
|
||
}
|
||
replies = $$('.reply', thread);
|
||
if (delta === -1) {
|
||
replies.reverse();
|
||
}
|
||
for (_i = 0, _len = replies.length; _i < _len; _i++) {
|
||
reply = replies[_i];
|
||
rect = reply.getBoundingClientRect();
|
||
if (delta === +1 && rect.top >= 0 || delta === -1 && rect.bottom <= d.documentElement.clientHeight) {
|
||
this.focus(reply);
|
||
return;
|
||
}
|
||
}
|
||
},
|
||
focus: function(post) {
|
||
$.addClass(post, 'highlight');
|
||
post.tabIndex = 0;
|
||
return post.focus();
|
||
}
|
||
};
|
||
|
||
Nav = {
|
||
init: function() {
|
||
var next, prev, span;
|
||
span = $.el('span', {
|
||
id: 'navlinks'
|
||
});
|
||
prev = $.el('a', {
|
||
textContent: '▲',
|
||
href: 'javascript:;'
|
||
});
|
||
next = $.el('a', {
|
||
textContent: '▼',
|
||
href: 'javascript:;'
|
||
});
|
||
$.on(prev, 'click', this.prev);
|
||
$.on(next, 'click', this.next);
|
||
$.add(span, [prev, $.tn(' '), next]);
|
||
return $.add(d.body, span);
|
||
},
|
||
prev: function() {
|
||
if (g.REPLY) {
|
||
return window.scrollTo(0, 0);
|
||
} else {
|
||
return Nav.scroll(-1);
|
||
}
|
||
},
|
||
next: function() {
|
||
if (g.REPLY) {
|
||
return window.scrollTo(0, d.body.scrollHeight);
|
||
} else {
|
||
return Nav.scroll(+1);
|
||
}
|
||
},
|
||
getThread: function(full) {
|
||
var bottom, i, rect, thread, _i, _len, _ref;
|
||
Nav.threads = $$('.thread:not([hidden])');
|
||
_ref = Nav.threads;
|
||
for (i = _i = 0, _len = _ref.length; _i < _len; i = ++_i) {
|
||
thread = _ref[i];
|
||
rect = thread.getBoundingClientRect();
|
||
bottom = rect.bottom;
|
||
if (bottom > 0) {
|
||
if (full) {
|
||
return [thread, i, rect];
|
||
}
|
||
return thread;
|
||
}
|
||
}
|
||
return $('.board');
|
||
},
|
||
scroll: function(delta) {
|
||
var i, rect, thread, top, _ref, _ref1;
|
||
_ref = Nav.getThread(true), thread = _ref[0], i = _ref[1], rect = _ref[2];
|
||
top = rect.top;
|
||
if (!((delta === -1 && Math.ceil(top) < 0) || (delta === +1 && top > 1))) {
|
||
i += delta;
|
||
}
|
||
top = (_ref1 = Nav.threads[i]) != null ? _ref1.getBoundingClientRect().top : void 0;
|
||
return window.scrollBy(0, top);
|
||
}
|
||
};
|
||
|
||
QR = {
|
||
init: function() {
|
||
if (!$.id('postForm')) {
|
||
return;
|
||
}
|
||
Main.callbacks.push(this.node);
|
||
return setTimeout(this.asyncInit);
|
||
},
|
||
asyncInit: function() {
|
||
var link;
|
||
if (Conf['Hide Original Post Form']) {
|
||
link = $.el('h1', {
|
||
innerHTML: "<a href=javascript:;>" + (g.REPLY ? 'Reply to Thread' : 'Start a Thread') + "</a>"
|
||
});
|
||
$.on(link.firstChild, 'click', function() {
|
||
QR.open();
|
||
if (!g.REPLY) {
|
||
$('select', QR.el).value = 'new';
|
||
}
|
||
return $('textarea', QR.el).focus();
|
||
});
|
||
$.before($.id('postForm'), link);
|
||
}
|
||
if (Conf['Persistent QR']) {
|
||
QR.dialog();
|
||
if (Conf['Auto Hide QR']) {
|
||
QR.hide();
|
||
}
|
||
}
|
||
$.on(d, 'dragover', QR.dragOver);
|
||
$.on(d, 'drop', QR.dropFile);
|
||
return $.on(d, 'dragstart dragend', QR.drag);
|
||
},
|
||
node: function(post) {
|
||
return $.on($('.postNum > a[title="Quote this post"]', post.el), 'click', QR.quote);
|
||
},
|
||
open: function() {
|
||
if (QR.el) {
|
||
QR.el.hidden = false;
|
||
return QR.unhide();
|
||
} else {
|
||
return QR.dialog();
|
||
}
|
||
},
|
||
close: function() {
|
||
var i, spoiler, _i, _len, _ref;
|
||
QR.el.hidden = true;
|
||
QR.abort();
|
||
d.activeElement.blur();
|
||
$.rmClass(QR.el, 'dump');
|
||
_ref = QR.replies;
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
i = _ref[_i];
|
||
QR.replies[0].rm();
|
||
}
|
||
QR.cooldown.auto = false;
|
||
QR.status();
|
||
QR.resetFileInput();
|
||
if (!Conf['Remember Spoiler'] && (spoiler = $.id('spoiler')).checked) {
|
||
spoiler.click();
|
||
}
|
||
return QR.cleanError();
|
||
},
|
||
hide: function() {
|
||
d.activeElement.blur();
|
||
$.addClass(QR.el, 'autohide');
|
||
return $.id('autohide').checked = true;
|
||
},
|
||
unhide: function() {
|
||
$.rmClass(QR.el, 'autohide');
|
||
return $.id('autohide').checked = false;
|
||
},
|
||
toggleHide: function() {
|
||
return this.checked && QR.hide() || QR.unhide();
|
||
},
|
||
error: function(err) {
|
||
var el;
|
||
el = $('.warning', QR.el);
|
||
if (typeof err === 'string') {
|
||
el.textContent = err;
|
||
} else {
|
||
el.innerHTML = null;
|
||
$.add(el, err);
|
||
}
|
||
QR.open();
|
||
if (/captcha|verification/i.test(el.textContent)) {
|
||
$('[autocomplete]', QR.el).focus();
|
||
}
|
||
if (d.hidden || d.oHidden || d.mozHidden || d.webkitHidden) {
|
||
return alert(el.textContent);
|
||
}
|
||
},
|
||
cleanError: function() {
|
||
return $('.warning', QR.el).textContent = null;
|
||
},
|
||
status: function(data) {
|
||
var disabled, input, value;
|
||
if (data == null) {
|
||
data = {};
|
||
}
|
||
if (!QR.el) {
|
||
return;
|
||
}
|
||
if (g.dead) {
|
||
value = 404;
|
||
disabled = true;
|
||
QR.cooldown.auto = false;
|
||
}
|
||
value = QR.cooldown.seconds || data.progress || value;
|
||
input = QR.status.input;
|
||
input.value = QR.cooldown.auto && Conf['Cooldown'] ? value ? "Auto " + value : 'Auto' : value || 'Submit';
|
||
return input.disabled = disabled || false;
|
||
},
|
||
cooldown: {
|
||
init: function() {
|
||
if (!Conf['Cooldown']) {
|
||
return;
|
||
}
|
||
QR.cooldown.start($.get("/" + g.BOARD + "/cooldown", 0));
|
||
return $.sync("/" + g.BOARD + "/cooldown", QR.cooldown.start);
|
||
},
|
||
start: function(timeout) {
|
||
var seconds;
|
||
seconds = Math.floor((timeout - Date.now()) / 1000);
|
||
return QR.cooldown.count(seconds);
|
||
},
|
||
set: function(seconds) {
|
||
if (!Conf['Cooldown']) {
|
||
return;
|
||
}
|
||
QR.cooldown.count(seconds);
|
||
return $.set("/" + g.BOARD + "/cooldown", Date.now() + seconds * $.SECOND);
|
||
},
|
||
count: function(seconds) {
|
||
if (!((0 <= seconds && seconds <= 60))) {
|
||
return;
|
||
}
|
||
setTimeout(QR.cooldown.count, 1000, seconds - 1);
|
||
QR.cooldown.seconds = seconds;
|
||
if (seconds === 0) {
|
||
$["delete"]("/" + g.BOARD + "/cooldown");
|
||
if (QR.cooldown.auto) {
|
||
QR.submit();
|
||
}
|
||
}
|
||
return QR.status();
|
||
}
|
||
},
|
||
quote: function(e) {
|
||
var caretPos, id, range, s, sel, ta, text, _ref;
|
||
if (e != null) {
|
||
e.preventDefault();
|
||
}
|
||
QR.open();
|
||
if (!g.REPLY) {
|
||
$('select', QR.el).value = $.x('ancestor::div[parent::div[@class="board"]]', this).id.slice(1);
|
||
}
|
||
id = this.previousSibling.hash.slice(2);
|
||
text = ">>" + id + "\n";
|
||
sel = window.getSelection();
|
||
if ((s = sel.toString()) && id === ((_ref = $.x('ancestor-or-self::blockquote', sel.anchorNode)) != null ? _ref.id.match(/\d+$/)[0] : void 0)) {
|
||
if ($.engine === 'presto') {
|
||
s = d.getSelection();
|
||
}
|
||
s = s.replace(/\n/g, '\n>');
|
||
text += ">" + s + "\n";
|
||
}
|
||
ta = $('textarea', QR.el);
|
||
caretPos = ta.selectionStart;
|
||
ta.value = ta.value.slice(0, caretPos) + text + ta.value.slice(ta.selectionEnd);
|
||
ta.focus();
|
||
range = caretPos + text.length;
|
||
if ($.engine === 'presto') {
|
||
range += text.match(/\n/g).length;
|
||
}
|
||
ta.setSelectionRange(range, range);
|
||
return $.event(ta, new Event('input'));
|
||
},
|
||
characterCount: function() {
|
||
var count, counter;
|
||
counter = QR.charaCounter;
|
||
count = this.textLength;
|
||
counter.textContent = count;
|
||
counter.hidden = count < 1000;
|
||
return (count > 1500 ? $.addClass : $.rmClass)(counter, 'warning');
|
||
},
|
||
drag: function(e) {
|
||
var i;
|
||
i = e.type === 'dragstart' ? 'off' : 'on';
|
||
$[i](d, 'dragover', QR.dragOver);
|
||
return $[i](d, 'drop', QR.dropFile);
|
||
},
|
||
dragOver: function(e) {
|
||
e.preventDefault();
|
||
return e.dataTransfer.dropEffect = 'copy';
|
||
},
|
||
dropFile: function(e) {
|
||
if (!e.dataTransfer.files.length) {
|
||
return;
|
||
}
|
||
e.preventDefault();
|
||
QR.open();
|
||
QR.fileInput.call(e.dataTransfer);
|
||
return $.addClass(QR.el, 'dump');
|
||
},
|
||
fileInput: function() {
|
||
var file, _i, _len, _ref;
|
||
QR.cleanError();
|
||
if (this.files.length === 1) {
|
||
file = this.files[0];
|
||
if (file.size > this.max) {
|
||
QR.error('File too large.');
|
||
QR.resetFileInput();
|
||
} else if (-1 === QR.mimeTypes.indexOf(file.type)) {
|
||
QR.error('Unsupported file type.');
|
||
QR.resetFileInput();
|
||
} else {
|
||
QR.selected.setFile(file);
|
||
}
|
||
return;
|
||
}
|
||
_ref = this.files;
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
file = _ref[_i];
|
||
if (file.size > this.max) {
|
||
QR.error("File " + file.name + " is too large.");
|
||
break;
|
||
} else if (-1 === QR.mimeTypes.indexOf(file.type)) {
|
||
QR.error("" + file.name + ": Unsupported file type.");
|
||
break;
|
||
}
|
||
if (!QR.replies[QR.replies.length - 1].file) {
|
||
QR.replies[QR.replies.length - 1].setFile(file);
|
||
} else {
|
||
new QR.reply().setFile(file);
|
||
}
|
||
}
|
||
$.addClass(QR.el, 'dump');
|
||
return QR.resetFileInput();
|
||
},
|
||
resetFileInput: function() {
|
||
var clone, input;
|
||
input = $('[type=file]', QR.el);
|
||
input.value = null;
|
||
if ($.engine !== 'presto') {
|
||
return;
|
||
}
|
||
clone = $.el('input', {
|
||
type: 'file',
|
||
accept: input.accept,
|
||
max: input.max,
|
||
multiple: input.multiple,
|
||
size: input.size,
|
||
title: input.title
|
||
});
|
||
$.on(clone, 'change', QR.fileInput);
|
||
$.on(clone, 'click', function(e) {
|
||
if (e.shiftKey) {
|
||
return QR.selected.rmFile() || e.preventDefault();
|
||
}
|
||
});
|
||
return $.replace(input, clone);
|
||
},
|
||
replies: [],
|
||
reply: (function() {
|
||
|
||
function _Class() {
|
||
var persona, prev,
|
||
_this = this;
|
||
prev = QR.replies[QR.replies.length - 1];
|
||
persona = $.get('QR.persona', {});
|
||
this.name = prev ? prev.name : persona.name || null;
|
||
this.email = prev && !/^sage$/.test(prev.email) ? prev.email : persona.email || null;
|
||
this.sub = prev && Conf['Remember Subject'] ? prev.sub : Conf['Remember Subject'] ? persona.sub : null;
|
||
this.spoiler = prev && Conf['Remember Spoiler'] ? prev.spoiler : false;
|
||
this.com = null;
|
||
this.el = $.el('a', {
|
||
className: 'thumbnail',
|
||
draggable: true,
|
||
href: 'javascript:;',
|
||
innerHTML: '<a class=remove>×</a><label hidden><input type=checkbox> Spoiler</label><span></span>'
|
||
});
|
||
$('input', this.el).checked = this.spoiler;
|
||
$.on(this.el, 'click', function() {
|
||
return _this.select();
|
||
});
|
||
$.on($('.remove', this.el), 'click', function(e) {
|
||
e.stopPropagation();
|
||
return _this.rm();
|
||
});
|
||
$.on($('label', this.el), 'click', function(e) {
|
||
return e.stopPropagation();
|
||
});
|
||
$.on($('input', this.el), 'change', function(e) {
|
||
_this.spoiler = e.target.checked;
|
||
if (_this.el.id === 'selected') {
|
||
return $.id('spoiler').checked = _this.spoiler;
|
||
}
|
||
});
|
||
$.before($('#addReply', QR.el), this.el);
|
||
$.on(this.el, 'dragstart', this.dragStart);
|
||
$.on(this.el, 'dragenter', this.dragEnter);
|
||
$.on(this.el, 'dragleave', this.dragLeave);
|
||
$.on(this.el, 'dragover', this.dragOver);
|
||
$.on(this.el, 'dragend', this.dragEnd);
|
||
$.on(this.el, 'drop', this.drop);
|
||
QR.replies.push(this);
|
||
}
|
||
|
||
_Class.prototype.setFile = function(file) {
|
||
var fileUrl, img, url,
|
||
_this = this;
|
||
this.file = file;
|
||
this.el.title = "" + file.name + " (" + ($.bytesToString(file.size)) + ")";
|
||
if (QR.spoiler) {
|
||
$('label', this.el).hidden = false;
|
||
}
|
||
if (!/^image/.test(file.type)) {
|
||
this.el.style.backgroundImage = null;
|
||
return;
|
||
}
|
||
url = window.URL || window.webkitURL;
|
||
if (typeof url.revokeObjectURL === "function") {
|
||
url.revokeObjectURL(this.url);
|
||
}
|
||
fileUrl = url.createObjectURL(file);
|
||
img = $.el('img');
|
||
$.on(img, 'load', function() {
|
||
var bb, c, data, i, l, s, ui8a, _i;
|
||
s = 90 * 3;
|
||
if (img.height < s || img.width < s) {
|
||
_this.url = fileUrl;
|
||
_this.el.style.backgroundImage = "url(" + _this.url + ")";
|
||
return;
|
||
}
|
||
if (img.height <= img.width) {
|
||
img.width = s / img.height * img.width;
|
||
img.height = s;
|
||
} else {
|
||
img.height = s / img.width * img.height;
|
||
img.width = s;
|
||
}
|
||
c = $.el('canvas');
|
||
c.height = img.height;
|
||
c.width = img.width;
|
||
c.getContext('2d').drawImage(img, 0, 0, img.width, img.height);
|
||
data = atob(c.toDataURL().split(',')[1]);
|
||
l = data.length;
|
||
ui8a = new Uint8Array(l);
|
||
for (i = _i = 0; 0 <= l ? _i < l : _i > l; i = 0 <= l ? ++_i : --_i) {
|
||
ui8a[i] = data.charCodeAt(i);
|
||
}
|
||
bb = new (window.MozBlobBuilder || window.WebKitBlobBuilder)();
|
||
bb.append(ui8a.buffer);
|
||
_this.url = url.createObjectURL(bb.getBlob('image/png'));
|
||
_this.el.style.backgroundImage = "url(" + _this.url + ")";
|
||
return typeof url.revokeObjectURL === "function" ? url.revokeObjectURL(fileUrl) : void 0;
|
||
});
|
||
return img.src = fileUrl;
|
||
};
|
||
|
||
_Class.prototype.rmFile = function() {
|
||
var _base1;
|
||
QR.resetFileInput();
|
||
delete this.file;
|
||
this.el.title = null;
|
||
this.el.style.backgroundImage = null;
|
||
if (QR.spoiler) {
|
||
$('label', this.el).hidden = true;
|
||
}
|
||
return typeof (_base1 = window.URL || window.webkitURL).revokeObjectURL === "function" ? _base1.revokeObjectURL(this.url) : void 0;
|
||
};
|
||
|
||
_Class.prototype.select = function() {
|
||
var data, rectEl, rectList, _i, _len, _ref, _ref1;
|
||
if ((_ref = QR.selected) != null) {
|
||
_ref.el.id = null;
|
||
}
|
||
QR.selected = this;
|
||
this.el.id = 'selected';
|
||
rectEl = this.el.getBoundingClientRect();
|
||
rectList = this.el.parentNode.getBoundingClientRect();
|
||
this.el.parentNode.scrollLeft += rectEl.left + rectEl.width / 2 - rectList.left - rectList.width / 2;
|
||
_ref1 = ['name', 'email', 'sub', 'com'];
|
||
for (_i = 0, _len = _ref1.length; _i < _len; _i++) {
|
||
data = _ref1[_i];
|
||
$("[name=" + data + "]", QR.el).value = this[data];
|
||
}
|
||
QR.characterCount.call($('textarea', QR.el));
|
||
return $('#spoiler', QR.el).checked = this.spoiler;
|
||
};
|
||
|
||
_Class.prototype.dragStart = function() {
|
||
return $.addClass(this, 'drag');
|
||
};
|
||
|
||
_Class.prototype.dragEnter = function() {
|
||
return $.addClass(this, 'over');
|
||
};
|
||
|
||
_Class.prototype.dragLeave = function() {
|
||
return $.rmClass(this, 'over');
|
||
};
|
||
|
||
_Class.prototype.dragOver = function(e) {
|
||
e.preventDefault();
|
||
return e.dataTransfer.dropEffect = 'move';
|
||
};
|
||
|
||
_Class.prototype.drop = function() {
|
||
var el, index, newIndex, oldIndex, reply;
|
||
el = $('.drag', this.parentNode);
|
||
index = function(el) {
|
||
return Array.prototype.slice.call(el.parentNode.children).indexOf(el);
|
||
};
|
||
oldIndex = index(el);
|
||
newIndex = index(this);
|
||
if (oldIndex < newIndex) {
|
||
$.after(this, el);
|
||
} else {
|
||
$.before(this, el);
|
||
}
|
||
reply = QR.replies.splice(oldIndex, 1)[0];
|
||
return QR.replies.splice(newIndex, 0, reply);
|
||
};
|
||
|
||
_Class.prototype.dragEnd = function() {
|
||
var el;
|
||
$.rmClass(this, 'drag');
|
||
if (el = $('.over', this.parentNode)) {
|
||
return $.rmClass(el, 'over');
|
||
}
|
||
};
|
||
|
||
_Class.prototype.rm = function() {
|
||
var index, _base1;
|
||
QR.resetFileInput();
|
||
$.rm(this.el);
|
||
index = QR.replies.indexOf(this);
|
||
if (QR.replies.length === 1) {
|
||
new QR.reply().select();
|
||
} else if (this.el.id === 'selected') {
|
||
(QR.replies[index - 1] || QR.replies[index + 1]).select();
|
||
}
|
||
QR.replies.splice(index, 1);
|
||
if (typeof (_base1 = window.URL || window.webkitURL).revokeObjectURL === "function") {
|
||
_base1.revokeObjectURL(this.url);
|
||
}
|
||
return delete this;
|
||
};
|
||
|
||
return _Class;
|
||
|
||
})(),
|
||
captcha: {
|
||
init: function() {
|
||
var _this = this;
|
||
if (!(QR.captchaIsEnabled = !!$.id('captchaFormPart'))) {
|
||
return;
|
||
}
|
||
if ($.id('recaptcha_challenge_field_holder')) {
|
||
return this.ready();
|
||
} else {
|
||
this.onready = function() {
|
||
return _this.ready();
|
||
};
|
||
return $.on($.id('recaptcha_widget_div'), 'DOMNodeInserted', this.onready);
|
||
}
|
||
},
|
||
ready: function() {
|
||
var _this = this;
|
||
if (this.challenge = $.id('recaptcha_challenge_field_holder')) {
|
||
$.off($.id('recaptcha_widget_div'), 'DOMNodeInserted', this.onready);
|
||
delete this.onready;
|
||
} else {
|
||
return;
|
||
}
|
||
$.after($('.textarea', QR.el), $.el('div', {
|
||
className: 'captchaimg',
|
||
title: 'Reload',
|
||
innerHTML: '<img>'
|
||
}));
|
||
$.after($('.captchaimg', QR.el), $.el('div', {
|
||
className: 'captchainput',
|
||
innerHTML: '<input title=Verification class=field autocomplete=off size=1>'
|
||
}));
|
||
this.img = $('.captchaimg > img', QR.el);
|
||
this.input = $('.captchainput > input', QR.el);
|
||
$.on(this.img.parentNode, 'click', this.reload);
|
||
$.on(this.input, 'keydown', this.keydown);
|
||
$.on(this.challenge, 'DOMNodeInserted', function() {
|
||
return _this.load();
|
||
});
|
||
$.sync('captchas', function(arr) {
|
||
return _this.count(arr.length);
|
||
});
|
||
this.count($.get('captchas', []).length);
|
||
return this.reload();
|
||
},
|
||
save: function() {
|
||
var captcha, captchas, response;
|
||
if (!(response = this.input.value)) {
|
||
return;
|
||
}
|
||
captchas = $.get('captchas', []);
|
||
while ((captcha = captchas[0]) && captcha.time < Date.now()) {
|
||
captchas.shift();
|
||
}
|
||
captchas.push({
|
||
challenge: this.challenge.firstChild.value,
|
||
response: response,
|
||
time: this.timeout
|
||
});
|
||
$.set('captchas', captchas);
|
||
this.count(captchas.length);
|
||
return this.reload();
|
||
},
|
||
load: function() {
|
||
var challenge;
|
||
this.timeout = Date.now() + 4 * $.MINUTE;
|
||
challenge = this.challenge.firstChild.value;
|
||
this.img.alt = challenge;
|
||
this.img.src = "//www.google.com/recaptcha/api/image?c=" + challenge;
|
||
return this.input.value = null;
|
||
},
|
||
count: function(count) {
|
||
this.input.placeholder = (function() {
|
||
switch (count) {
|
||
case 0:
|
||
return 'Verification (Shift + Enter to cache)';
|
||
case 1:
|
||
return 'Verification (1 cached captcha)';
|
||
default:
|
||
return "Verification (" + count + " cached captchas)";
|
||
}
|
||
})();
|
||
return this.input.alt = count;
|
||
},
|
||
reload: function(focus) {
|
||
window.location = 'javascript:Recaptcha.reload("t")';
|
||
if (focus) {
|
||
return QR.captcha.input.focus();
|
||
}
|
||
},
|
||
keydown: function(e) {
|
||
var c;
|
||
c = QR.captcha;
|
||
if (e.keyCode === 8 && !c.input.value) {
|
||
c.reload();
|
||
} else if (e.keyCode === 13 && e.shiftKey) {
|
||
c.save();
|
||
} else {
|
||
return;
|
||
}
|
||
return e.preventDefault();
|
||
}
|
||
},
|
||
dialog: function() {
|
||
var fileInput, id, mimeTypes, name, spoiler, ta, thread, threads, _i, _j, _len, _len1, _ref, _ref1;
|
||
QR.el = UI.dialog('qr', 'top:0;right:0;', '\
|
||
<div class=move>\
|
||
Quick Reply <input type=checkbox id=autohide title=Auto-hide>\
|
||
<span> <a class=close title=Close>×</a></span>\
|
||
</div>\
|
||
<form>\
|
||
<div><input id=dump type=button title="Dump list" value=+ class=field><input name=name title=Name placeholder=Name class=field size=1><input name=email title=E-mail placeholder=E-mail class=field size=1><input name=sub title=Subject placeholder=Subject class=field size=1></div>\
|
||
<div id=replies><div><a id=addReply href=javascript:; title="Add a reply">+</a></div></div>\
|
||
<div class=textarea><textarea name=com title=Comment placeholder=Comment class=field></textarea><span id=charCount></span></div>\
|
||
<div><input type=file title="Shift+Click to remove the selected file." multiple size=16><input type=submit></div>\
|
||
<label id=spoilerLabel><input type=checkbox id=spoiler> Spoiler Image</label>\
|
||
<div class=warning></div>\
|
||
</form>');
|
||
if (Conf['Remember QR size'] && $.engine === 'gecko') {
|
||
$.on(ta = $('textarea', QR.el), 'mouseup', function() {
|
||
return $.set('QR.size', this.style.cssText);
|
||
});
|
||
ta.style.cssText = $.get('QR.size', '');
|
||
}
|
||
mimeTypes = $('ul.rules').firstElementChild.textContent.trim().match(/: (.+)/)[1].toLowerCase().replace(/\w+/g, function(type) {
|
||
switch (type) {
|
||
case 'jpg':
|
||
return 'image/jpeg';
|
||
case 'pdf':
|
||
return 'application/pdf';
|
||
case 'swf':
|
||
return 'application/x-shockwave-flash';
|
||
default:
|
||
return "image/" + type;
|
||
}
|
||
});
|
||
QR.mimeTypes = mimeTypes.split(', ');
|
||
QR.mimeTypes.push('');
|
||
fileInput = $('input[type=file]', QR.el);
|
||
fileInput.max = $('input[name=MAX_FILE_SIZE]').value;
|
||
if ($.engine !== 'presto') {
|
||
fileInput.accept = mimeTypes;
|
||
}
|
||
QR.spoiler = !!$('input[name=spoiler]');
|
||
spoiler = $('#spoilerLabel', QR.el);
|
||
spoiler.hidden = !QR.spoiler;
|
||
QR.charaCounter = $('#charCount', QR.el);
|
||
ta = $('textarea', QR.el);
|
||
if (!g.REPLY) {
|
||
threads = '<option value=new>New thread</option>';
|
||
_ref = $$('.thread');
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
thread = _ref[_i];
|
||
id = thread.id.slice(1);
|
||
threads += "<option value=" + id + ">Thread " + id + "</option>";
|
||
}
|
||
$.prepend($('.move > span', QR.el), $.el('select', {
|
||
innerHTML: threads,
|
||
title: 'Create a new thread / Reply to a thread'
|
||
}));
|
||
$.on($('select', QR.el), 'mousedown', function(e) {
|
||
return e.stopPropagation();
|
||
});
|
||
}
|
||
$.on($('#autohide', QR.el), 'change', QR.toggleHide);
|
||
$.on($('.close', QR.el), 'click', QR.close);
|
||
$.on($('#dump', QR.el), 'click', function() {
|
||
return QR.el.classList.toggle('dump');
|
||
});
|
||
$.on($('#addReply', QR.el), 'click', function() {
|
||
return new QR.reply().select();
|
||
});
|
||
$.on($('form', QR.el), 'submit', QR.submit);
|
||
$.on(ta, 'input', function() {
|
||
return QR.selected.el.lastChild.textContent = this.value;
|
||
});
|
||
$.on(ta, 'input', QR.characterCount);
|
||
$.on(fileInput, 'change', QR.fileInput);
|
||
$.on(fileInput, 'click', function(e) {
|
||
if (e.shiftKey) {
|
||
return QR.selected.rmFile() || e.preventDefault();
|
||
}
|
||
});
|
||
$.on(spoiler.firstChild, 'change', function() {
|
||
return $('input', QR.selected.el).click();
|
||
});
|
||
$.on($('.warning', QR.el), 'click', QR.cleanError);
|
||
new QR.reply().select();
|
||
_ref1 = ['name', 'email', 'sub', 'com'];
|
||
for (_j = 0, _len1 = _ref1.length; _j < _len1; _j++) {
|
||
name = _ref1[_j];
|
||
$.on($("[name=" + name + "]", QR.el), 'input', function() {
|
||
var _ref2;
|
||
QR.selected[this.name] = this.value;
|
||
if (QR.cooldown.auto && QR.selected === QR.replies[0] && (0 < (_ref2 = QR.cooldown.seconds) && _ref2 < 6)) {
|
||
return QR.cooldown.auto = false;
|
||
}
|
||
});
|
||
}
|
||
QR.status.input = $('input[type=submit]', QR.el);
|
||
QR.status();
|
||
QR.cooldown.init();
|
||
QR.captcha.init();
|
||
$.add(d.body, QR.el);
|
||
return $.event(QR.el, new CustomEvent('QRDialogCreation', {
|
||
bubbles: true
|
||
}));
|
||
},
|
||
submit: function(e) {
|
||
var callbacks, captcha, captchas, challenge, err, m, opts, post, reply, response, threadID;
|
||
if (e != null) {
|
||
e.preventDefault();
|
||
}
|
||
if (QR.cooldown.seconds) {
|
||
QR.cooldown.auto = !QR.cooldown.auto;
|
||
QR.status();
|
||
return;
|
||
}
|
||
QR.abort();
|
||
reply = QR.replies[0];
|
||
threadID = g.THREAD_ID || $('select', QR.el).value;
|
||
if (!(threadID === 'new' && reply.file || threadID !== 'new' && (reply.com || reply.file))) {
|
||
err = 'No file selected.';
|
||
} else if (QR.captchaIsEnabled) {
|
||
captchas = $.get('captchas', []);
|
||
while ((captcha = captchas[0]) && captcha.time < Date.now()) {
|
||
captchas.shift();
|
||
}
|
||
if (captcha = captchas.shift()) {
|
||
challenge = captcha.challenge;
|
||
response = captcha.response;
|
||
} else {
|
||
challenge = QR.captcha.img.alt;
|
||
if (response = QR.captcha.input.value) {
|
||
QR.captcha.reload();
|
||
}
|
||
}
|
||
$.set('captchas', captchas);
|
||
QR.captcha.count(captchas.length);
|
||
if (!response) {
|
||
err = 'No valid captcha.';
|
||
}
|
||
}
|
||
if (err) {
|
||
QR.cooldown.auto = false;
|
||
QR.status();
|
||
QR.error(err);
|
||
return;
|
||
}
|
||
QR.cleanError();
|
||
QR.cooldown.auto = QR.replies.length > 1;
|
||
if (Conf['Auto Hide QR'] && !QR.cooldown.auto) {
|
||
QR.hide();
|
||
}
|
||
if (Conf['Thread Watcher'] && Conf['Auto Watch Reply'] && threadID !== 'new') {
|
||
Watcher.watch(threadID);
|
||
}
|
||
if (!QR.cooldown.auto && $.x('ancestor::div[@id="qr"]', d.activeElement)) {
|
||
d.activeElement.blur();
|
||
}
|
||
QR.status({
|
||
progress: '...'
|
||
});
|
||
post = {
|
||
resto: threadID,
|
||
name: reply.name,
|
||
email: reply.email,
|
||
sub: reply.sub,
|
||
com: reply.com,
|
||
upfile: reply.file,
|
||
spoiler: reply.spoiler,
|
||
mode: 'regist',
|
||
pwd: (m = d.cookie.match(/4chan_pass=([^;]+)/)) ? decodeURIComponent(m[1]) : $('input[name=pwd]').value,
|
||
recaptcha_challenge_field: challenge,
|
||
recaptcha_response_field: response + ' '
|
||
};
|
||
callbacks = {
|
||
onload: function() {
|
||
return QR.response(this.response);
|
||
},
|
||
onerror: function() {
|
||
QR.status();
|
||
return QR.error($.el('a', {
|
||
href: '//www.4chan.org/banned',
|
||
target: '_blank',
|
||
textContent: 'Connection error, or you are banned.'
|
||
}));
|
||
}
|
||
};
|
||
opts = {
|
||
form: $.formData(post),
|
||
upCallbacks: {
|
||
onload: function() {
|
||
return QR.status({
|
||
progress: '...'
|
||
});
|
||
},
|
||
onprogress: function(e) {
|
||
return QR.status({
|
||
progress: "" + (Math.round(e.loaded / e.total * 100)) + "%"
|
||
});
|
||
}
|
||
}
|
||
};
|
||
return QR.ajax = $.ajax($.id('postForm').parentNode.action, callbacks, opts);
|
||
},
|
||
response: function(html) {
|
||
var bs, doc, err, msg, persona, postID, reply, threadID, _, _ref;
|
||
doc = d.implementation.createHTMLDocument('');
|
||
doc.documentElement.innerHTML = html;
|
||
if (doc.title === '4chan - Banned') {
|
||
bs = $$('b', doc);
|
||
err = $.el('span', {
|
||
innerHTML: /^You were issued a warning/.test($('.boxcontent', doc).textContent.trim()) ? "You were issued a warning on " + bs[0].innerHTML + " as " + bs[3].innerHTML + ".<br>Warning reason: " + bs[1].innerHTML : "You are banned! ;_;<br>Please click <a href=//www.4chan.org/banned target=_blank>HERE</a> to see the reason."
|
||
});
|
||
} else if (msg = doc.getElementById('errmsg')) {
|
||
err = msg.textContent;
|
||
if (msg.firstChild.tagName) {
|
||
err = msg.firstChild;
|
||
err.target = '_blank';
|
||
}
|
||
} else if (!(msg = $('b', doc))) {
|
||
err = 'Connection error with sys.4chan.org.';
|
||
}
|
||
if (err) {
|
||
if (/captcha|verification/i.test(err) || err === 'Connection error with sys.4chan.org.') {
|
||
QR.cooldown.auto = !!$.get('captchas', []).length;
|
||
QR.cooldown.set(2);
|
||
} else {
|
||
QR.cooldown.auto = false;
|
||
}
|
||
QR.status();
|
||
QR.error(err);
|
||
return;
|
||
}
|
||
reply = QR.replies[0];
|
||
persona = $.get('QR.persona', {});
|
||
persona = {
|
||
name: reply.name,
|
||
email: /^sage$/.test(reply.email) ? persona.email : reply.email,
|
||
sub: Conf['Remember Subject'] ? reply.sub : null
|
||
};
|
||
$.set('QR.persona', persona);
|
||
_ref = msg.lastChild.textContent.match(/thread:(\d+),no:(\d+)/), _ = _ref[0], threadID = _ref[1], postID = _ref[2];
|
||
$.event(QR.el, new CustomEvent('QRPostSuccessful', {
|
||
detail: {
|
||
threadID: threadID,
|
||
postID: postID
|
||
}
|
||
}));
|
||
if (threadID === '0') {
|
||
if (Conf['Thread Watcher'] && Conf['Auto Watch']) {
|
||
$.set('autoWatch', postID);
|
||
}
|
||
location.pathname = "/" + g.BOARD + "/res/" + postID;
|
||
} else {
|
||
QR.cooldown.auto = QR.replies.length > 1;
|
||
QR.cooldown.set(/sage/i.test(reply.email) ? 60 : 30);
|
||
if (Conf['Open Reply in New Tab'] && !g.REPLY && !QR.cooldown.auto) {
|
||
$.open("//boards.4chan.org/" + g.BOARD + "/res/" + threadID + "#p" + postID);
|
||
}
|
||
}
|
||
if (Conf['Persistent QR'] || QR.cooldown.auto) {
|
||
reply.rm();
|
||
} else {
|
||
QR.close();
|
||
}
|
||
if (g.REPLY && (Conf['Unread Count'] || Conf['Unread Favicon'])) {
|
||
Unread.foresee.push(postID);
|
||
}
|
||
if (g.REPLY && Conf['Thread Updater'] && Conf['Auto Update This']) {
|
||
Updater.update();
|
||
}
|
||
QR.status();
|
||
return QR.resetFileInput();
|
||
},
|
||
abort: function() {
|
||
var _ref;
|
||
if ((_ref = QR.ajax) != null) {
|
||
_ref.abort();
|
||
}
|
||
delete QR.ajax;
|
||
return QR.status();
|
||
}
|
||
};
|
||
|
||
Options = {
|
||
init: function() {
|
||
var a, home, _i, _len, _ref;
|
||
_ref = [$.id('navtopr'), $.id('navbotr')];
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
home = _ref[_i];
|
||
a = $.el('a', {
|
||
textContent: '4chan X Settings',
|
||
href: 'javascript:;'
|
||
});
|
||
$.on(a, 'click', Options.dialog);
|
||
$.replace(home.firstElementChild, a);
|
||
}
|
||
if (!$.get('firstrun')) {
|
||
if (!Favicon.el) {
|
||
Favicon.init();
|
||
}
|
||
$.set('firstrun', true);
|
||
return Options.dialog();
|
||
}
|
||
},
|
||
dialog: function() {
|
||
var arr, back, checked, description, dialog, favicon, fileInfo, filter, hiddenNum, hiddenThreads, indicator, indicators, input, key, li, obj, overlay, sauce, time, tr, ul, _i, _len, _ref, _ref1, _ref2;
|
||
dialog = $.el('div', {
|
||
id: 'options',
|
||
className: 'reply dialog',
|
||
innerHTML: '<div id=optionsbar>\
|
||
<div id=credits>\
|
||
<a target=_blank href=http://mayhemydg.github.com/4chan-x/>4chan X</a>\
|
||
| <a target=_blank href=https://raw.github.com/mayhemydg/4chan-x/master/changelog>' + Main.version + '</a>\
|
||
| <a target=_blank href=http://mayhemydg.github.com/4chan-x/#bug-report>Issues</a>\
|
||
</div>\
|
||
<div>\
|
||
<label for=main_tab>Main</label>\
|
||
| <label for=filter_tab>Filter</label>\
|
||
| <label for=sauces_tab>Sauce</label>\
|
||
| <label for=rice_tab>Rice</label>\
|
||
| <label for=keybinds_tab>Keybinds</label>\
|
||
</div>\
|
||
</div>\
|
||
<hr>\
|
||
<div id=content>\
|
||
<input type=radio name=tab hidden id=main_tab checked>\
|
||
<div></div>\
|
||
<input type=radio name=tab hidden id=sauces_tab>\
|
||
<div>\
|
||
<div class=warning><code>Sauce</code> is disabled.</div>\
|
||
Lines starting with a <code>#</code> will be ignored.<br>\
|
||
You can specify a certain display text by appending <code>;text:[text]</code> to the url.\
|
||
<ul>These parameters will be replaced by their corresponding values:\
|
||
<li>$1: Thumbnail url.</li>\
|
||
<li>$2: Full image url.</li>\
|
||
<li>$3: MD5 hash.</li>\
|
||
<li>$4: Current board.</li>\
|
||
</ul>\
|
||
<textarea name=sauces id=sauces class=field></textarea>\
|
||
</div>\
|
||
<input type=radio name=tab hidden id=filter_tab>\
|
||
<div>\
|
||
<div class=warning><code>Filter</code> is disabled.</div>\
|
||
<select name=filter>\
|
||
<option value=guide>Guide</option>\
|
||
<option value=name>Name</option>\
|
||
<option value=uniqueid>Unique ID</option>\
|
||
<option value=tripcode>Tripcode</option>\
|
||
<option value=mod>Admin/Mod</option>\
|
||
<option value=email>E-mail</option>\
|
||
<option value=subject>Subject</option>\
|
||
<option value=comment>Comment</option>\
|
||
<option value=country>Country</option>\
|
||
<option value=filename>Filename</option>\
|
||
<option value=dimensions>Image dimensions</option>\
|
||
<option value=filesize>Filesize</option>\
|
||
<option value=md5>Image MD5 (uses exact string matching, not regular expressions)</option>\
|
||
</select>\
|
||
</div>\
|
||
<input type=radio name=tab hidden id=rice_tab>\
|
||
<div>\
|
||
<div class=warning><code>Quote Backlinks</code> are disabled.</div>\
|
||
<ul>\
|
||
Backlink formatting\
|
||
<li><input name=backlink class=field> : <span id=backlinkPreview></span></li>\
|
||
</ul>\
|
||
<div class=warning><code>Time Formatting</code> is disabled.</div>\
|
||
<ul>\
|
||
Time formatting\
|
||
<li><input name=time class=field> : <span id=timePreview></span></li>\
|
||
<li>Supported <a href=http://en.wikipedia.org/wiki/Date_%28Unix%29#Formatting>format specifiers</a>:</li>\
|
||
<li>Day: %a, %A, %d, %e</li>\
|
||
<li>Month: %m, %b, %B</li>\
|
||
<li>Year: %y</li>\
|
||
<li>Hour: %k, %H, %l (lowercase L), %I (uppercase i), %p, %P</li>\
|
||
<li>Minutes: %M</li>\
|
||
<li>Seconds: %S</li>\
|
||
</ul>\
|
||
<div class=warning><code>File Info Formatting</code> is disabled.</div>\
|
||
<ul>\
|
||
File Info Formatting\
|
||
<li><input name=fileInfo class=field> : <span id=fileInfoPreview class=fileText></span></li>\
|
||
<li>Link (with original file name): %l (lowercase L, truncated), %L (untruncated)</li>\
|
||
<li>Original file name: %n (Truncated), %N (Untruncated)</li>\
|
||
<li>Spoiler indicator: %p</li>\
|
||
<li>Size: %B (Bytes), %K (KB), %M (MB), %s (4chan default)</li>\
|
||
<li>Resolution: %r (Displays PDF on /po/, for PDFs)</li>\
|
||
</ul>\
|
||
<div class=warning><code>Unread Favicon</code> is disabled.</div>\
|
||
Unread favicons<br>\
|
||
<select name=favicon>\
|
||
<option value=ferongr>ferongr</option>\
|
||
<option value=xat->xat-</option>\
|
||
<option value=Mayhem>Mayhem</option>\
|
||
<option value=Original>Original</option>\
|
||
</select>\
|
||
<span></span>\
|
||
</div>\
|
||
<input type=radio name=tab hidden id=keybinds_tab>\
|
||
<div>\
|
||
<div class=warning><code>Keybinds</code> are disabled.</div>\
|
||
<div>Allowed keys: Ctrl, Alt, Meta, a-z, A-Z, 0-9, Up, Down, Right, Left.</div>\
|
||
<table><tbody>\
|
||
<tr><th>Actions</th><th>Keybinds</th></tr>\
|
||
</tbody></table>\
|
||
</div>\
|
||
</div>'
|
||
});
|
||
_ref = Config.main;
|
||
for (key in _ref) {
|
||
obj = _ref[key];
|
||
ul = $.el('ul', {
|
||
textContent: key
|
||
});
|
||
for (key in obj) {
|
||
arr = obj[key];
|
||
checked = $.get(key, Conf[key]) ? 'checked' : '';
|
||
description = arr[1];
|
||
li = $.el('li', {
|
||
innerHTML: "<label><input type=checkbox name=\"" + key + "\" " + checked + ">" + key + "</label><span class=description>: " + description + "</span>"
|
||
});
|
||
$.on($('input', li), 'click', $.cb.checked);
|
||
$.add(ul, li);
|
||
}
|
||
$.add($('#main_tab + div', dialog), ul);
|
||
}
|
||
hiddenThreads = $.get("hiddenThreads/" + g.BOARD + "/", {});
|
||
hiddenNum = Object.keys(g.hiddenReplies).length + Object.keys(hiddenThreads).length;
|
||
li = $.el('li', {
|
||
innerHTML: "<button>hidden: " + hiddenNum + "</button> <span class=description>: Forget all hidden posts. Useful if you accidentally hide a post and have \"Show Stubs\" disabled."
|
||
});
|
||
$.on($('button', li), 'click', Options.clearHidden);
|
||
$.add($('ul:nth-child(2)', dialog), li);
|
||
filter = $('select[name=filter]', dialog);
|
||
$.on(filter, 'change', Options.filter);
|
||
sauce = $('#sauces', dialog);
|
||
sauce.value = $.get(sauce.name, Conf[sauce.name]);
|
||
$.on(sauce, 'change', $.cb.value);
|
||
(back = $('[name=backlink]', dialog)).value = $.get('backlink', Conf['backlink']);
|
||
(time = $('[name=time]', dialog)).value = $.get('time', Conf['time']);
|
||
(fileInfo = $('[name=fileInfo]', dialog)).value = $.get('fileInfo', Conf['fileInfo']);
|
||
$.on(back, 'input', $.cb.value);
|
||
$.on(back, 'input', Options.backlink);
|
||
$.on(time, 'input', $.cb.value);
|
||
$.on(time, 'input', Options.time);
|
||
$.on(fileInfo, 'input', $.cb.value);
|
||
$.on(fileInfo, 'input', Options.fileInfo);
|
||
favicon = $('select[name=favicon]', dialog);
|
||
favicon.value = $.get('favicon', Conf['favicon']);
|
||
$.on(favicon, 'change', $.cb.value);
|
||
$.on(favicon, 'change', Options.favicon);
|
||
_ref1 = Config.hotkeys;
|
||
for (key in _ref1) {
|
||
arr = _ref1[key];
|
||
tr = $.el('tr', {
|
||
innerHTML: "<td>" + arr[1] + "</td><td><input name=" + key + " class=field></td>"
|
||
});
|
||
input = $('input', tr);
|
||
input.value = $.get(key, Conf[key]);
|
||
$.on(input, 'keydown', Options.keybind);
|
||
$.add($('#keybinds_tab + div tbody', dialog), tr);
|
||
}
|
||
indicators = {};
|
||
_ref2 = $$('.warning', dialog);
|
||
for (_i = 0, _len = _ref2.length; _i < _len; _i++) {
|
||
indicator = _ref2[_i];
|
||
key = indicator.firstChild.textContent;
|
||
indicator.hidden = $.get(key, Conf[key]);
|
||
indicators[key] = indicator;
|
||
$.on($("[name='" + key + "']", dialog), 'click', function() {
|
||
return indicators[this.name].hidden = this.checked;
|
||
});
|
||
}
|
||
overlay = $.el('div', {
|
||
id: 'overlay'
|
||
});
|
||
$.on(overlay, 'click', Options.close);
|
||
$.on(dialog, 'click', function(e) {
|
||
return e.stopPropagation();
|
||
});
|
||
$.add(overlay, dialog);
|
||
$.add(d.body, overlay);
|
||
d.body.style.setProperty('width', "" + d.body.clientWidth + "px", null);
|
||
$.addClass(d.body, 'unscroll');
|
||
Options.filter.call(filter);
|
||
Options.backlink.call(back);
|
||
Options.time.call(time);
|
||
Options.fileInfo.call(fileInfo);
|
||
return Options.favicon.call(favicon);
|
||
},
|
||
close: function() {
|
||
$.rm(this);
|
||
d.body.style.removeProperty('width');
|
||
return $.rmClass(d.body, 'unscroll');
|
||
},
|
||
clearHidden: function() {
|
||
$["delete"]("hiddenReplies/" + g.BOARD + "/");
|
||
$["delete"]("hiddenThreads/" + g.BOARD + "/");
|
||
this.textContent = "hidden: 0";
|
||
return g.hiddenReplies = {};
|
||
},
|
||
keybind: function(e) {
|
||
var key;
|
||
if (e.keyCode === 9) {
|
||
return;
|
||
}
|
||
e.preventDefault();
|
||
e.stopPropagation();
|
||
if ((key = Keybinds.keyCode(e)) == null) {
|
||
return;
|
||
}
|
||
this.value = key;
|
||
return $.cb.value.call(this);
|
||
},
|
||
filter: function() {
|
||
var el, name, ta;
|
||
el = this.nextSibling;
|
||
if ((name = this.value) !== 'guide') {
|
||
ta = $.el('textarea', {
|
||
name: name,
|
||
className: 'field',
|
||
value: $.get(name, Conf[name])
|
||
});
|
||
$.on(ta, 'change', $.cb.value);
|
||
$.replace(el, ta);
|
||
return;
|
||
}
|
||
if (el) {
|
||
$.rm(el);
|
||
}
|
||
return $.after(this, $.el('article', {
|
||
innerHTML: '<p>Use <a href=https://developer.mozilla.org/en/JavaScript/Guide/Regular_Expressions>regular expressions</a>, one per line.<br>\
|
||
Lines starting with a <code>#</code> will be ignored.<br>\
|
||
For example, <code>/weeaboo/i</code> will filter posts containing the string `<code>weeaboo</code>`, case-insensitive.</p>\
|
||
<ul>You can use these settings with each regular expression, separate them with semicolons:\
|
||
<li>\
|
||
Per boards, separate them with commas. It is global if not specified.<br>\
|
||
For example: <code>boards:a,jp;</code>.\
|
||
</li>\
|
||
<li>\
|
||
Filter OPs only along with their threads (`only`), replies only (`no`, this is default), or both (`yes`).<br>\
|
||
For example: <code>op:only;</code>, <code>op:no;</code> or <code>op:yes;</code>.\
|
||
</li>\
|
||
<li>\
|
||
Overrule the `Show Stubs` setting if specified: create a stub (`yes`) or not (`no`).<br>\
|
||
For example: <code>stub:yes;</code> or <code>stub:no;</code>.\
|
||
</li>\
|
||
<li>\
|
||
Highlight instead of hiding. You can specify a class name to use with a userstyle.<br>\
|
||
For example: <code>highlight;</code> or <code>highlight:wallpaper;</code>.\
|
||
</li>\
|
||
<li>\
|
||
Highlighted OPs will have their threads put on top of board pages by default.<br>\
|
||
For example: <code>top:yes;</code> or <code>top:no;</code>.\
|
||
</li>\
|
||
</ul>'
|
||
}));
|
||
},
|
||
time: function() {
|
||
Time.foo();
|
||
Time.date = new Date();
|
||
return $.id('timePreview').textContent = Time.funk(Time);
|
||
},
|
||
backlink: function() {
|
||
return $.id('backlinkPreview').textContent = Conf['backlink'].replace(/%id/, '123456789');
|
||
},
|
||
fileInfo: function() {
|
||
FileInfo.data = {
|
||
link: 'javascript:;',
|
||
spoiler: true,
|
||
size: '276',
|
||
unit: 'KB',
|
||
resolution: '1280x720',
|
||
fullname: 'd9bb2efc98dd0df141a94399ff5880b7.jpg',
|
||
shortname: 'd9bb2efc98dd0df141a94399ff5880(...).jpg'
|
||
};
|
||
FileInfo.setFormats();
|
||
return $.id('fileInfoPreview').innerHTML = FileInfo.funk(FileInfo);
|
||
},
|
||
favicon: function() {
|
||
Favicon["switch"]();
|
||
Unread.update(true);
|
||
return this.nextElementSibling.innerHTML = "<img src=" + Favicon.unreadSFW + "> <img src=" + Favicon.unreadNSFW + "> <img src=" + Favicon.unreadDead + ">";
|
||
}
|
||
};
|
||
|
||
Updater = {
|
||
init: function() {
|
||
var checkbox, checked, dialog, html, input, name, title, _i, _len, _ref;
|
||
html = "<div class=move><span id=count></span> <span id=timer>-" + Conf['Interval'] + "</span></div>";
|
||
checkbox = Config.updater.checkbox;
|
||
for (name in checkbox) {
|
||
title = checkbox[name][1];
|
||
checked = Conf[name] ? 'checked' : '';
|
||
html += "<div><label title='" + title + "'>" + name + "<input name='" + name + "' type=checkbox " + checked + "></label></div>";
|
||
}
|
||
checked = Conf['Auto Update'] ? 'checked' : '';
|
||
html += " <div><label title='Controls whether *this* thread automatically updates or not'>Auto Update This<input name='Auto Update This' type=checkbox " + checked + "></label></div> <div><label>Interval (s)<input name=Interval value=" + Conf['Interval'] + " class=field size=4></label></div> <div><input value='Update Now' type=button></div>";
|
||
dialog = UI.dialog('updater', 'bottom: 0; right: 0;', html);
|
||
this.count = $('#count', dialog);
|
||
this.timer = $('#timer', dialog);
|
||
this.thread = $.id("t" + g.THREAD_ID);
|
||
_ref = $$('input', dialog);
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
input = _ref[_i];
|
||
if (input.type === 'checkbox') {
|
||
$.on(input, 'click', $.cb.checked);
|
||
if (input.name === 'Scroll BG') {
|
||
$.on(input, 'click', this.cb.scrollBG);
|
||
this.cb.scrollBG.call(input);
|
||
}
|
||
if (input.name === 'Verbose') {
|
||
$.on(input, 'click', this.cb.verbose);
|
||
this.cb.verbose.call(input);
|
||
} else if (input.name === 'Auto Update This') {
|
||
$.on(input, 'click', this.cb.autoUpdate);
|
||
this.cb.autoUpdate.call(input);
|
||
Conf[input.name] = input.checked;
|
||
}
|
||
} else if (input.name === 'Interval') {
|
||
$.on(input, 'input', this.cb.interval);
|
||
} else if (input.type === 'button') {
|
||
$.on(input, 'click', this.update);
|
||
}
|
||
}
|
||
$.add(d.body, dialog);
|
||
this.retryCoef = 10;
|
||
return this.lastModified = 0;
|
||
},
|
||
cb: {
|
||
interval: function() {
|
||
var val;
|
||
val = parseInt(this.value, 10);
|
||
this.value = val > 0 ? val : 1;
|
||
return $.cb.value.call(this);
|
||
},
|
||
verbose: function() {
|
||
if (Conf['Verbose']) {
|
||
Updater.count.textContent = '+0';
|
||
return Updater.timer.hidden = false;
|
||
} else {
|
||
$.extend(Updater.count, {
|
||
className: '',
|
||
textContent: 'Thread Updater'
|
||
});
|
||
return Updater.timer.hidden = true;
|
||
}
|
||
},
|
||
autoUpdate: function() {
|
||
if (this.checked) {
|
||
return Updater.timeoutID = setTimeout(Updater.timeout, 1000);
|
||
} else {
|
||
return clearTimeout(Updater.timeoutID);
|
||
}
|
||
},
|
||
scrollBG: function() {
|
||
return Updater.scrollBG = this.checked ? function() {
|
||
return true;
|
||
} : function() {
|
||
return !(d.hidden || d.oHidden || d.mozHidden || d.webkitHidden);
|
||
};
|
||
},
|
||
update: function() {
|
||
var count, doc, id, lastPost, nodes, reply, scroll, _i, _len, _ref, _ref1;
|
||
if (this.status === 404) {
|
||
Updater.timer.textContent = '';
|
||
Updater.count.textContent = 404;
|
||
Updater.count.className = 'warning';
|
||
clearTimeout(Updater.timeoutID);
|
||
g.dead = true;
|
||
if (Conf['Unread Count']) {
|
||
Unread.title = Unread.title.match(/^.+-/)[0] + ' 404';
|
||
} else {
|
||
d.title = d.title.match(/^.+-/)[0] + ' 404';
|
||
}
|
||
Unread.update(true);
|
||
QR.abort();
|
||
return;
|
||
}
|
||
if ((_ref = this.status) !== 0 && _ref !== 200 && _ref !== 304) {
|
||
Updater.retryCoef += 10 * (Updater.retryCoef < 120);
|
||
if (Conf['Verbose']) {
|
||
Updater.count.textContent = this.statusText;
|
||
Updater.count.className = 'warning';
|
||
}
|
||
return;
|
||
}
|
||
Updater.retryCoef = 10;
|
||
Updater.timer.textContent = "-" + Conf['Interval'];
|
||
/*
|
||
Status Code 304: Not modified
|
||
By sending the `If-Modified-Since` header we get a proper status code, and no response.
|
||
This saves bandwidth for both the user and the servers, avoid unnecessary computation,
|
||
and won't load images and scripts when parsing the response.
|
||
*/
|
||
|
||
if (this.status === 304) {
|
||
if (Conf['Verbose']) {
|
||
Updater.count.textContent = '+0';
|
||
Updater.count.className = null;
|
||
}
|
||
return;
|
||
}
|
||
Updater.lastModified = this.getResponseHeader('Last-Modified');
|
||
doc = d.implementation.createHTMLDocument('');
|
||
doc.documentElement.innerHTML = this.response;
|
||
lastPost = Updater.thread.lastElementChild;
|
||
id = lastPost.id.slice(2);
|
||
nodes = [];
|
||
_ref1 = $$('.replyContainer', doc).reverse();
|
||
for (_i = 0, _len = _ref1.length; _i < _len; _i++) {
|
||
reply = _ref1[_i];
|
||
if (reply.id.slice(2) <= id) {
|
||
break;
|
||
}
|
||
nodes.push(reply);
|
||
}
|
||
count = nodes.length;
|
||
scroll = Conf['Scrolling'] && Updater.scrollBG() && count && lastPost.getBoundingClientRect().bottom - d.documentElement.clientHeight < 25;
|
||
if (Conf['Verbose']) {
|
||
Updater.count.textContent = "+" + count;
|
||
Updater.count.className = count ? 'new' : null;
|
||
}
|
||
$.add(Updater.thread, nodes.reverse());
|
||
if (scroll) {
|
||
return nodes[0].scrollIntoView();
|
||
}
|
||
}
|
||
},
|
||
timeout: function() {
|
||
var n;
|
||
Updater.timeoutID = setTimeout(Updater.timeout, 1000);
|
||
n = 1 + Number(Updater.timer.textContent);
|
||
if (n === 0) {
|
||
return Updater.update();
|
||
} else if (n === Updater.retryCoef) {
|
||
Updater.retryCoef += 10 * (Updater.retryCoef < 120);
|
||
return Updater.retry();
|
||
} else {
|
||
return Updater.timer.textContent = n;
|
||
}
|
||
},
|
||
retry: function() {
|
||
this.count.textContent = 'Retry';
|
||
this.count.className = null;
|
||
return this.update();
|
||
},
|
||
update: function() {
|
||
var url, _ref;
|
||
Updater.timer.textContent = 0;
|
||
if ((_ref = Updater.request) != null) {
|
||
_ref.abort();
|
||
}
|
||
url = location.pathname + '?' + Date.now();
|
||
return Updater.request = $.ajax(url, {
|
||
onload: Updater.cb.update
|
||
}, {
|
||
headers: {
|
||
'If-Modified-Since': Updater.lastModified
|
||
}
|
||
});
|
||
}
|
||
};
|
||
|
||
Watcher = {
|
||
init: function() {
|
||
var favicon, html, input, _i, _len, _ref;
|
||
html = '<div class=move>Thread Watcher</div>';
|
||
this.dialog = UI.dialog('watcher', 'top: 50px; left: 0px;', html);
|
||
$.add(d.body, this.dialog);
|
||
_ref = $$('.op input');
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
input = _ref[_i];
|
||
favicon = $.el('img', {
|
||
className: 'favicon'
|
||
});
|
||
$.on(favicon, 'click', this.cb.toggle);
|
||
$.before(input, favicon);
|
||
}
|
||
if (g.THREAD_ID === $.get('autoWatch', 0)) {
|
||
this.watch(g.THREAD_ID);
|
||
$["delete"]('autoWatch');
|
||
} else {
|
||
this.refresh();
|
||
}
|
||
return $.sync('watched', this.refresh);
|
||
},
|
||
refresh: function(watched) {
|
||
var board, div, favicon, id, link, nodes, props, watchedBoard, x, _i, _j, _len, _len1, _ref, _ref1, _ref2;
|
||
watched || (watched = $.get('watched', {}));
|
||
nodes = [];
|
||
for (board in watched) {
|
||
_ref = watched[board];
|
||
for (id in _ref) {
|
||
props = _ref[id];
|
||
x = $.el('a', {
|
||
textContent: '×',
|
||
href: 'javascript:;'
|
||
});
|
||
$.on(x, 'click', Watcher.cb.x);
|
||
link = $.el('a', props);
|
||
link.title = link.textContent;
|
||
div = $.el('div');
|
||
$.add(div, [x, $.tn(' '), link]);
|
||
nodes.push(div);
|
||
}
|
||
}
|
||
_ref1 = $$('div:not(.move)', Watcher.dialog);
|
||
for (_i = 0, _len = _ref1.length; _i < _len; _i++) {
|
||
div = _ref1[_i];
|
||
$.rm(div);
|
||
}
|
||
$.add(Watcher.dialog, nodes);
|
||
watchedBoard = watched[g.BOARD] || {};
|
||
_ref2 = $$('.favicon');
|
||
for (_j = 0, _len1 = _ref2.length; _j < _len1; _j++) {
|
||
favicon = _ref2[_j];
|
||
id = favicon.nextSibling.name;
|
||
if (id in watchedBoard) {
|
||
favicon.src = Favicon["default"];
|
||
} else {
|
||
favicon.src = Favicon.empty;
|
||
}
|
||
}
|
||
},
|
||
cb: {
|
||
toggle: function() {
|
||
return Watcher.toggle(this.parentNode);
|
||
},
|
||
x: function() {
|
||
var thread;
|
||
thread = this.nextElementSibling.pathname.split('/');
|
||
return Watcher.unwatch(thread[3], thread[1]);
|
||
}
|
||
},
|
||
toggle: function(thread) {
|
||
var id;
|
||
id = $('.favicon + input', thread).name;
|
||
return Watcher.watch(id) || Watcher.unwatch(id, g.BOARD);
|
||
},
|
||
unwatch: function(id, board) {
|
||
var watched;
|
||
watched = $.get('watched', {});
|
||
delete watched[board][id];
|
||
$.set('watched', watched);
|
||
return Watcher.refresh();
|
||
},
|
||
watch: function(id) {
|
||
var thread, watched, _name;
|
||
thread = $.id("t" + id);
|
||
if ($('.favicon', thread).src === Favicon["default"]) {
|
||
return false;
|
||
}
|
||
watched = $.get('watched', {});
|
||
watched[_name = g.BOARD] || (watched[_name] = {});
|
||
watched[g.BOARD][id] = {
|
||
href: "/" + g.BOARD + "/res/" + id,
|
||
textContent: Get.title(thread)
|
||
};
|
||
$.set('watched', watched);
|
||
Watcher.refresh();
|
||
return true;
|
||
}
|
||
};
|
||
|
||
Anonymize = {
|
||
init: function() {
|
||
return Main.callbacks.push(this.node);
|
||
},
|
||
node: function(post) {
|
||
var name, parent, trip;
|
||
if (post.isInlined && !post.isCrosspost) {
|
||
return;
|
||
}
|
||
name = $('.postInfo .name', post.el);
|
||
name.textContent = 'Anonymous';
|
||
if ((trip = name.nextElementSibling) && trip.className === 'postertrip') {
|
||
$.rm(trip);
|
||
}
|
||
if ((parent = name.parentNode).className === 'useremail' && !/^mailto:sage$/i.test(parent.href)) {
|
||
return $.replace(parent, name);
|
||
}
|
||
}
|
||
};
|
||
|
||
Sauce = {
|
||
init: function() {
|
||
var link, _i, _len, _ref;
|
||
if (g.BOARD === 'f') {
|
||
return;
|
||
}
|
||
this.links = [];
|
||
_ref = Conf['sauces'].split('\n');
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
link = _ref[_i];
|
||
if (link[0] === '#') {
|
||
continue;
|
||
}
|
||
this.links.push(this.createSauceLink(link.trim()));
|
||
}
|
||
if (!this.links.length) {
|
||
return;
|
||
}
|
||
return Main.callbacks.push(this.node);
|
||
},
|
||
createSauceLink: function(link) {
|
||
var domain, el, href, m;
|
||
link = link.replace(/(\$\d)/g, function(parameter) {
|
||
switch (parameter) {
|
||
case '$1':
|
||
return "' + (isArchived ? img.firstChild.src : 'http://thumbs.4chan.org' + img.pathname.replace(/src(\\/\\d+).+$/, 'thumb$1s.jpg')) + '";
|
||
case '$2':
|
||
return "' + img.href + '";
|
||
case '$3':
|
||
return "' + encodeURIComponent(img.firstChild.dataset.md5) + '";
|
||
case '$4':
|
||
return g.BOARD;
|
||
default:
|
||
return parameter;
|
||
}
|
||
});
|
||
domain = (m = link.match(/;text:(.+)$/)) ? m[1] : link.match(/(\w+)\.\w+\//)[1];
|
||
href = link.replace(/;text:.+$/, '');
|
||
href = Function('img', 'isArchived', "return '" + href + "'");
|
||
el = $.el('a', {
|
||
target: '_blank',
|
||
textContent: domain
|
||
});
|
||
return function(img, isArchived) {
|
||
var a;
|
||
a = el.cloneNode(true);
|
||
a.href = href(img, isArchived);
|
||
return a;
|
||
};
|
||
},
|
||
node: function(post) {
|
||
var img, link, nodes, _i, _len, _ref;
|
||
img = post.img;
|
||
if (post.isInlined && !post.isCrosspost || !img) {
|
||
return;
|
||
}
|
||
img = img.parentNode;
|
||
nodes = [];
|
||
_ref = Sauce.links;
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
link = _ref[_i];
|
||
nodes.push($.tn('\u00A0'), link(img, post.isArchived));
|
||
}
|
||
return $.add(post.fileInfo, nodes);
|
||
}
|
||
};
|
||
|
||
RevealSpoilers = {
|
||
init: function() {
|
||
return Main.callbacks.push(this.node);
|
||
},
|
||
node: function(post) {
|
||
var img, s;
|
||
img = post.img;
|
||
if (!(img && /^Spoiler/.test(img.alt)) || post.isInlined && !post.isCrosspost || post.isArchived) {
|
||
return;
|
||
}
|
||
img.removeAttribute('style');
|
||
s = img.style;
|
||
s.maxHeight = s.maxWidth = /\bop\b/.test(post["class"]) ? '250px' : '125px';
|
||
return img.src = "//thumbs.4chan.org" + (img.parentNode.pathname.replace(/src(\/\d+).+$/, 'thumb$1s.jpg'));
|
||
}
|
||
};
|
||
|
||
Time = {
|
||
init: function() {
|
||
Time.foo();
|
||
return Main.callbacks.push(this.node);
|
||
},
|
||
node: function(post) {
|
||
var node;
|
||
if (post.isInlined && !post.isCrosspost) {
|
||
return;
|
||
}
|
||
node = $('.postInfo > .dateTime', post.el);
|
||
Time.date = new Date(node.dataset.utc * 1000);
|
||
return node.textContent = Time.funk(Time);
|
||
},
|
||
foo: function() {
|
||
var code;
|
||
code = Conf['time'].replace(/%([A-Za-z])/g, function(s, c) {
|
||
if (c in Time.formatters) {
|
||
return "' + Time.formatters." + c + "() + '";
|
||
} else {
|
||
return s;
|
||
}
|
||
});
|
||
return Time.funk = Function('Time', "return '" + code + "'");
|
||
},
|
||
day: ['Sunday', 'Monday', 'Tuesday', 'Wednesday', 'Thursday', 'Friday', 'Saturday'],
|
||
month: ['January', 'February', 'March', 'April', 'May', 'June', 'July', 'August', 'September', 'October', 'November', 'December'],
|
||
zeroPad: function(n) {
|
||
if (n < 10) {
|
||
return '0' + n;
|
||
} else {
|
||
return n;
|
||
}
|
||
},
|
||
formatters: {
|
||
a: function() {
|
||
return Time.day[Time.date.getDay()].slice(0, 3);
|
||
},
|
||
A: function() {
|
||
return Time.day[Time.date.getDay()];
|
||
},
|
||
b: function() {
|
||
return Time.month[Time.date.getMonth()].slice(0, 3);
|
||
},
|
||
B: function() {
|
||
return Time.month[Time.date.getMonth()];
|
||
},
|
||
d: function() {
|
||
return Time.zeroPad(Time.date.getDate());
|
||
},
|
||
e: function() {
|
||
return Time.date.getDate();
|
||
},
|
||
H: function() {
|
||
return Time.zeroPad(Time.date.getHours());
|
||
},
|
||
I: function() {
|
||
return Time.zeroPad(Time.date.getHours() % 12 || 12);
|
||
},
|
||
k: function() {
|
||
return Time.date.getHours();
|
||
},
|
||
l: function() {
|
||
return Time.date.getHours() % 12 || 12;
|
||
},
|
||
m: function() {
|
||
return Time.zeroPad(Time.date.getMonth() + 1);
|
||
},
|
||
M: function() {
|
||
return Time.zeroPad(Time.date.getMinutes());
|
||
},
|
||
p: function() {
|
||
if (Time.date.getHours() < 12) {
|
||
return 'AM';
|
||
} else {
|
||
return 'PM';
|
||
}
|
||
},
|
||
P: function() {
|
||
if (Time.date.getHours() < 12) {
|
||
return 'am';
|
||
} else {
|
||
return 'pm';
|
||
}
|
||
},
|
||
S: function() {
|
||
return Time.zeroPad(Time.date.getSeconds());
|
||
},
|
||
y: function() {
|
||
return Time.date.getFullYear() - 2000;
|
||
}
|
||
}
|
||
};
|
||
|
||
FileInfo = {
|
||
init: function() {
|
||
if (g.BOARD === 'f') {
|
||
return;
|
||
}
|
||
this.setFormats();
|
||
return Main.callbacks.push(this.node);
|
||
},
|
||
node: function(post) {
|
||
var alt, node, span;
|
||
if (post.isInlined && !post.isCrosspost || !post.fileInfo) {
|
||
return;
|
||
}
|
||
node = post.fileInfo.firstElementChild;
|
||
alt = post.img.alt;
|
||
span = $('span', node);
|
||
FileInfo.data = {
|
||
link: post.img.parentNode.href,
|
||
spoiler: /^Spoiler/.test(alt),
|
||
size: alt.match(/\d+\.?\d*/)[0],
|
||
unit: alt.match(/\w+$/)[0],
|
||
resolution: span.previousSibling.textContent.match(/\d+x\d+|PDF/)[0],
|
||
fullname: span.title,
|
||
shortname: span.textContent
|
||
};
|
||
node.setAttribute('data-filename', span.title);
|
||
return node.innerHTML = FileInfo.funk(FileInfo);
|
||
},
|
||
setFormats: function() {
|
||
var code;
|
||
code = Conf['fileInfo'].replace(/%([BKlLMnNprs])/g, function(s, c) {
|
||
if (c in FileInfo.formatters) {
|
||
return "' + f.formatters." + c + "() + '";
|
||
} else {
|
||
return s;
|
||
}
|
||
});
|
||
return this.funk = Function('f', "return '" + code + "'");
|
||
},
|
||
convertUnit: function(unitT) {
|
||
var i, size, unitF, units;
|
||
size = this.data.size;
|
||
unitF = this.data.unit;
|
||
if (unitF !== unitT) {
|
||
units = ['B', 'KB', 'MB'];
|
||
i = units.indexOf(unitF) - units.indexOf(unitT);
|
||
if (unitT === 'B') {
|
||
unitT = 'Bytes';
|
||
}
|
||
if (i > 0) {
|
||
while (i-- > 0) {
|
||
size *= 1024;
|
||
}
|
||
} else if (i < 0) {
|
||
while (i++ < 0) {
|
||
size /= 1024;
|
||
}
|
||
}
|
||
if (size < 1 && size.toString().length > size.toFixed(2).length) {
|
||
size = size.toFixed(2);
|
||
}
|
||
}
|
||
return "" + size + " " + unitT;
|
||
},
|
||
formatters: {
|
||
l: function() {
|
||
return "<a href=" + FileInfo.data.link + " target=_blank>" + (this.n()) + "</a>";
|
||
},
|
||
L: function() {
|
||
return "<a href=" + FileInfo.data.link + " target=_blank>" + (this.N()) + "</a>";
|
||
},
|
||
n: function() {
|
||
if (FileInfo.data.fullname === FileInfo.data.shortname) {
|
||
return FileInfo.data.fullname;
|
||
} else {
|
||
return "<span class=fntrunc>" + FileInfo.data.shortname + "</span><span class=fnfull>" + FileInfo.data.fullname + "</span>";
|
||
}
|
||
},
|
||
N: function() {
|
||
return FileInfo.data.fullname;
|
||
},
|
||
p: function() {
|
||
if (FileInfo.data.spoiler) {
|
||
return 'Spoiler, ';
|
||
} else {
|
||
return '';
|
||
}
|
||
},
|
||
s: function() {
|
||
return "" + FileInfo.data.size + " " + FileInfo.data.unit;
|
||
},
|
||
B: function() {
|
||
return FileInfo.convertUnit('B');
|
||
},
|
||
K: function() {
|
||
return FileInfo.convertUnit('KB');
|
||
},
|
||
M: function() {
|
||
return FileInfo.convertUnit('MB');
|
||
},
|
||
r: function() {
|
||
return FileInfo.data.resolution;
|
||
}
|
||
}
|
||
};
|
||
|
||
Get = {
|
||
post: function(board, threadID, postID, root, cb) {
|
||
var post, url;
|
||
if (board === g.BOARD && (post = $.id("pc" + postID))) {
|
||
$.add(root, Get.cleanPost(post.cloneNode(true)));
|
||
return;
|
||
}
|
||
root.textContent = "Loading post No." + postID + "...";
|
||
if (threadID) {
|
||
return $.cache("/" + board + "/res/" + threadID, function() {
|
||
return Get.parsePost(this, board, threadID, postID, root, cb);
|
||
});
|
||
} else if (url = Redirect.post(board, postID)) {
|
||
return $.cache(url, function() {
|
||
return Get.parseArchivedPost(this, board, postID, root, cb);
|
||
});
|
||
}
|
||
},
|
||
parsePost: function(req, board, threadID, postID, root, cb) {
|
||
var doc, href, link, pc, quote, status, url, _i, _len, _ref;
|
||
status = req.status;
|
||
if (status !== 200) {
|
||
if (url = Redirect.post(board, postID)) {
|
||
$.cache(url, function() {
|
||
return Get.parseArchivedPost(this, board, postID, root, cb);
|
||
});
|
||
} else {
|
||
root.textContent = status === 404 ? "Thread No." + threadID + " has not been found." : "Error " + req.status + ": " + req.statusText + ".";
|
||
}
|
||
return;
|
||
}
|
||
doc = d.implementation.createHTMLDocument('');
|
||
doc.documentElement.innerHTML = req.response;
|
||
if (!(pc = doc.getElementById("pc" + postID))) {
|
||
if (url = Redirect.post(board, postID)) {
|
||
$.cache(url, function() {
|
||
return Get.parseArchivedPost(this, board, postID, root, cb);
|
||
});
|
||
} else {
|
||
root.textContent = "Post No." + postID + " has not been found.";
|
||
}
|
||
return;
|
||
}
|
||
pc = Get.cleanPost(d.importNode(pc, true));
|
||
_ref = $$('.quotelink', pc);
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
quote = _ref[_i];
|
||
href = quote.getAttribute('href');
|
||
if (href[0] === '/') {
|
||
continue;
|
||
}
|
||
quote.href = "/" + board + "/res/" + href;
|
||
}
|
||
link = $('.postNum > a[title="Highlight this post"]', pc);
|
||
link.href = "/" + board + "/res/" + threadID + "#p" + postID;
|
||
link.nextSibling.href = "/" + board + "/res/" + threadID + "#q" + postID;
|
||
$.replace(root.firstChild, pc);
|
||
if (cb) {
|
||
return cb();
|
||
}
|
||
},
|
||
parseArchivedPost: function(req, board, postID, root, cb) {
|
||
var bq, br, capcode, data, email, file, filename, filesize, isOP, max, name, nameBlock, pc, pi, piM, span, spoiler, subject, threadID, thumb_src, timestamp, trip;
|
||
data = JSON.parse(req.response);
|
||
$.addClass(root, 'archivedPost');
|
||
if (data.error) {
|
||
root.textContent = data.error;
|
||
return;
|
||
}
|
||
threadID = data.thread_num;
|
||
isOP = postID === threadID;
|
||
name = data.name, trip = data.trip, timestamp = data.timestamp;
|
||
subject = data.title;
|
||
piM = $.el('div', {
|
||
id: "pim" + postID,
|
||
className: 'postInfoM mobile',
|
||
innerHTML: "<span class=nameBlock><span class=name></span><br><span class=subject></span></span><span class='dateTime postNum' data-utc=" + timestamp + ">" + data.fourchan_date + "<br><em></em><a href='/" + board + "/res/" + threadID + "#p" + postID + "' title='Highlight this post'>No.</a><a href='/" + board + "/res/" + threadID + "#q" + postID + "' title='Quote this post'>" + postID + "</a></span>"
|
||
});
|
||
$('.name', piM).textContent = name;
|
||
$('.subject', piM).textContent = subject;
|
||
br = $('br', piM);
|
||
if (trip) {
|
||
$.before(br, [
|
||
$.tn(' '), $.el('span', {
|
||
className: 'postertrip',
|
||
textContent: trip
|
||
})
|
||
]);
|
||
}
|
||
capcode = data.capcode;
|
||
if (capcode !== 'N') {
|
||
$.addClass(br.parentNode, capcode === 'A' ? 'capcodeAdmin' : 'capcodeMod');
|
||
$.before(br, [
|
||
$.tn(' '), $.el('strong', {
|
||
className: 'capcode',
|
||
textContent: capcode === 'A' ? '## Admin' : '## Mod'
|
||
}), $.tn(' '), $.el('img', {
|
||
src: capcode === 'A' ? '//static.4chan.org/image/adminicon.gif' : '//static.4chan.org/image/modicon.gif',
|
||
alt: capcode === 'A' ? 'This user is the 4chan Administrator.' : 'This user is a 4chan Moderator.',
|
||
title: capcode === 'A' ? 'This user is the 4chan Administrator.' : 'This user is a 4chan Moderator.',
|
||
className: 'identityIcon'
|
||
})
|
||
]);
|
||
}
|
||
pi = $.el('div', {
|
||
id: "pi" + postID,
|
||
className: 'postInfo desktop',
|
||
innerHTML: "<input type=checkbox name=" + postID + " value=delete> <span class=userInfo><span class=subject></span> <span class=nameBlock></span></span> <span class=dateTime data-utc=" + timestamp + ">data.fourchan_date</span> <span class='postNum desktop'><a href='/" + board + "/res/" + threadID + "#p" + postID + "' title='Highlight this post'>No.</a><a href='/" + board + "/res/" + threadID + "#q" + postID + "' title='Quote this post'>" + postID + "</a>" + (isOP ? ' ' : '') + "</span> "
|
||
});
|
||
$('.subject', pi).textContent = subject;
|
||
nameBlock = $('.nameBlock', pi);
|
||
if (data.email) {
|
||
email = $.el('a', {
|
||
className: 'useremail',
|
||
href: "mailto:" + data.email
|
||
});
|
||
$.add(nameBlock, email);
|
||
nameBlock = email;
|
||
}
|
||
$.add(nameBlock, $.el('span', {
|
||
className: 'name',
|
||
textContent: data.name
|
||
}));
|
||
if (trip) {
|
||
$.add(nameBlock, [
|
||
$.tn(' '), $.el('span', {
|
||
className: 'postertrip',
|
||
textContent: trip
|
||
})
|
||
]);
|
||
}
|
||
if (capcode !== 'N') {
|
||
$.add(nameBlock, [
|
||
$.tn(' '), $.el('strong', {
|
||
className: capcode === 'A' ? 'capcode capcodeAdmin' : 'capcode',
|
||
textContent: capcode === 'A' ? '## Admin' : '## Mod'
|
||
})
|
||
]);
|
||
nameBlock = $('.nameBlock', pi);
|
||
$.addClass(nameBlock, capcode === 'A' ? 'capcodeAdmin' : 'capcodeMod');
|
||
$.add(nameBlock, [
|
||
$.tn(' '), $.el('img', {
|
||
src: capcode === 'A' ? '//static.4chan.org/image/adminicon.gif' : '//static.4chan.org/image/modicon.gif',
|
||
alt: capcode === 'A' ? 'This user is the 4chan Administrator.' : 'This user is a 4chan Moderator.',
|
||
title: capcode === 'A' ? 'This user is the 4chan Administrator.' : 'This user is a 4chan Moderator.',
|
||
className: 'identityIcon'
|
||
})
|
||
]);
|
||
}
|
||
bq = $.el('blockquote', {
|
||
id: "m" + postID,
|
||
className: 'postMessage',
|
||
textContent: data.comment
|
||
});
|
||
bq.innerHTML = bq.innerHTML.replace(/\n|\[\/?b\]|\[\/?spoiler\]|\[\/?code\]|\[\/?moot\]|\[\/?banned\]/g, function(text) {
|
||
switch (text) {
|
||
case '\n':
|
||
return '<br>';
|
||
case '[b]':
|
||
return '<b>';
|
||
case '[/b]':
|
||
return '</b>';
|
||
case '[spoiler]':
|
||
return '<span class=spoiler>';
|
||
case '[/spoiler]':
|
||
return '</span>';
|
||
case '[code]':
|
||
return '<pre class=prettyprint>';
|
||
case '[/code]':
|
||
return '</pre>';
|
||
case '[moot]':
|
||
return '<div style="padding:5px;margin-left:.5em;border-color:#faa;border:2px dashed rgba(255,0,0,.1);border-radius:2px">';
|
||
case '[/moot]':
|
||
return '</div>';
|
||
case '[banned]':
|
||
return '<b style="color: red;">';
|
||
case '[/banned]':
|
||
return '</b>';
|
||
}
|
||
});
|
||
bq.innerHTML = bq.innerHTML.replace(/(^|>)(>[^<$]+)(<|$)/g, '$1<span class=quote>$2</span>$3');
|
||
pc = $.el('div', {
|
||
id: "pc" + postID,
|
||
className: "postContainer " + (isOP ? 'op' : 'reply') + "Container",
|
||
innerHTML: "<div id=p" + postID + " class='post " + (isOP ? 'op' : 'reply') + "'></div>"
|
||
});
|
||
$.add(pc.firstChild, [piM, pi, bq]);
|
||
if (filename = data.media_filename) {
|
||
file = $.el('div', {
|
||
id: "f" + postID,
|
||
className: 'file'
|
||
});
|
||
spoiler = data.spoiler === '1';
|
||
filesize = $.bytesToString(data.media_size);
|
||
$.add(file, $.el('div', {
|
||
className: 'fileInfo',
|
||
innerHTML: "<span id=fT" + postID + " class=fileText>File: <a href='" + (data.media_link || data.remote_media_link) + "' target=_blank>" + data.media_orig + "</a>-(" + (spoiler ? 'Spoiler Image, ' : '') + filesize + ", " + data.media_w + "x" + data.media_h + ", <span title></span>)</span>"
|
||
}));
|
||
span = $('span[title]', file);
|
||
span.title = filename;
|
||
max = isOP ? 40 : 30;
|
||
span.textContent = filename.replace(/\.\w+$/, '').length > max ? "" + filename.slice(0, max) + "(...)" + (filename.match(/\.\w+$/)) : filename;
|
||
thumb_src = data.media_status === 'available' ? "src=" + data.thumb_link : '';
|
||
$.add(file, $.el('a', {
|
||
className: spoiler ? 'fileThumb imgspoiler' : 'fileThumb',
|
||
href: data.media_link || data.remote_media_link,
|
||
target: '_blank',
|
||
innerHTML: "<img " + thumb_src + " alt='" + (data.media_status !== 'available' ? "Error: " + data.media_status + ", " : '') + (spoiler ? 'Spoiler Image, ' : '') + filesize + "' data-md5=" + data.media_hash + " style='height: " + data.preview_h + "px; width: " + data.preview_w + "px;'>"
|
||
}));
|
||
$.after((isOP ? piM : pi), file);
|
||
}
|
||
$.replace(root.firstChild, Get.cleanPost(pc));
|
||
if (cb) {
|
||
return cb();
|
||
}
|
||
},
|
||
cleanPost: function(root) {
|
||
var child, el, els, inline, inlined, now, post, _i, _j, _k, _l, _len, _len1, _len2, _len3, _ref, _ref1, _ref2;
|
||
post = $('.post', root);
|
||
_ref = Array.prototype.slice.call(root.childNodes);
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
child = _ref[_i];
|
||
if (child !== post) {
|
||
$.rm(child);
|
||
}
|
||
}
|
||
_ref1 = $$('.inline', post);
|
||
for (_j = 0, _len1 = _ref1.length; _j < _len1; _j++) {
|
||
inline = _ref1[_j];
|
||
$.rm(inline);
|
||
}
|
||
_ref2 = $$('.inlined', post);
|
||
for (_k = 0, _len2 = _ref2.length; _k < _len2; _k++) {
|
||
inlined = _ref2[_k];
|
||
$.rmClass(inlined, 'inlined');
|
||
}
|
||
now = Date.now();
|
||
els = $$('[id]', root);
|
||
els.push(root);
|
||
for (_l = 0, _len3 = els.length; _l < _len3; _l++) {
|
||
el = els[_l];
|
||
el.id = "" + now + "_" + el.id;
|
||
}
|
||
$.rmClass(root, 'forwarded');
|
||
$.rmClass(root, 'qphl');
|
||
$.rmClass(post, 'highlight');
|
||
$.rmClass(post, 'qphl');
|
||
root.hidden = post.hidden = false;
|
||
return root;
|
||
},
|
||
title: function(thread) {
|
||
var el, op, span;
|
||
op = $('.op', thread);
|
||
el = $('.subject', op);
|
||
if (!el.textContent) {
|
||
el = $('blockquote', op);
|
||
if (!el.textContent) {
|
||
el = $('.nameBlock', op);
|
||
}
|
||
}
|
||
span = $.el('span', {
|
||
innerHTML: el.innerHTML.replace(/<br>/g, ' ')
|
||
});
|
||
return "/" + g.BOARD + "/ - " + (span.textContent.trim());
|
||
}
|
||
};
|
||
|
||
TitlePost = {
|
||
init: function() {
|
||
return d.title = Get.title();
|
||
}
|
||
};
|
||
|
||
QuoteBacklink = {
|
||
init: function() {
|
||
var format;
|
||
format = Conf['backlink'].replace(/%id/g, "' + id + '");
|
||
this.funk = Function('id', "return '" + format + "'");
|
||
return Main.callbacks.push(this.node);
|
||
},
|
||
node: function(post) {
|
||
var a, container, el, link, qid, quote, quotes, _i, _len, _ref;
|
||
if (post.isInlined) {
|
||
return;
|
||
}
|
||
quotes = {};
|
||
_ref = post.quotes;
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
quote = _ref[_i];
|
||
if (qid = quote.hash.slice(2)) {
|
||
quotes[qid] = true;
|
||
}
|
||
}
|
||
a = $.el('a', {
|
||
href: "/" + g.BOARD + "/res/" + post.threadID + "#p" + post.ID,
|
||
className: post.el.hidden ? 'filtered backlink' : 'backlink',
|
||
textContent: QuoteBacklink.funk(post.ID)
|
||
});
|
||
for (qid in quotes) {
|
||
if (!(el = $.id("pi" + qid)) || !Conf['OP Backlinks'] && /\bop\b/.test(el.parentNode.className)) {
|
||
continue;
|
||
}
|
||
link = a.cloneNode(true);
|
||
if (Conf['Quote Preview']) {
|
||
$.on(link, 'mouseover', QuotePreview.mouseover);
|
||
}
|
||
if (Conf['Quote Inline']) {
|
||
$.on(link, 'click', QuoteInline.toggle);
|
||
} else {
|
||
link.setAttribute('onclick', "replyhl('" + post.ID + "');");
|
||
}
|
||
if (!(container = $.id("blc" + qid))) {
|
||
container = $.el('span', {
|
||
className: 'container',
|
||
id: "blc" + qid
|
||
});
|
||
$.add(el, container);
|
||
}
|
||
$.add(container, [$.tn(' '), link]);
|
||
}
|
||
}
|
||
};
|
||
|
||
QuoteInline = {
|
||
init: function() {
|
||
return Main.callbacks.push(this.node);
|
||
},
|
||
node: function(post) {
|
||
var quote, _i, _j, _len, _len1, _ref, _ref1;
|
||
_ref = post.quotes;
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
quote = _ref[_i];
|
||
if (!(quote.hash || /\bdeadlink\b/.test(quote.className))) {
|
||
continue;
|
||
}
|
||
quote.removeAttribute('onclick');
|
||
$.on(quote, 'click', QuoteInline.toggle);
|
||
}
|
||
_ref1 = post.backlinks;
|
||
for (_j = 0, _len1 = _ref1.length; _j < _len1; _j++) {
|
||
quote = _ref1[_j];
|
||
$.on(quote, 'click', QuoteInline.toggle);
|
||
}
|
||
},
|
||
toggle: function(e) {
|
||
var id;
|
||
if (e.shiftKey || e.altKey || e.ctrlKey || e.metaKey || e.button !== 0) {
|
||
return;
|
||
}
|
||
e.preventDefault();
|
||
id = this.dataset.id || this.hash.slice(2);
|
||
if (/\binlined\b/.test(this.className)) {
|
||
QuoteInline.rm(this, id);
|
||
} else {
|
||
if ($.x("ancestor::div[contains(@id,'p" + id + "')]", this)) {
|
||
return;
|
||
}
|
||
QuoteInline.add(this, id);
|
||
}
|
||
return this.classList.toggle('inlined');
|
||
},
|
||
add: function(q, id) {
|
||
var board, el, i, inline, isBacklink, path, postID, root, threadID;
|
||
if (q.host === 'boards.4chan.org') {
|
||
path = q.pathname.split('/');
|
||
board = path[1];
|
||
threadID = path[3];
|
||
postID = id;
|
||
} else {
|
||
board = q.dataset.board;
|
||
threadID = 0;
|
||
postID = q.dataset.id;
|
||
}
|
||
el = board === g.BOARD ? $.id("p" + postID) : false;
|
||
inline = $.el('div', {
|
||
id: "i" + postID,
|
||
className: el ? 'inline' : 'inline crosspost'
|
||
});
|
||
root = (isBacklink = /\bbacklink\b/.test(q.className)) ? q.parentNode : $.x('ancestor-or-self::*[parent::blockquote][1]', q);
|
||
$.after(root, inline);
|
||
Get.post(board, threadID, postID, inline);
|
||
if (!el) {
|
||
return;
|
||
}
|
||
if (isBacklink && Conf['Forward Hiding']) {
|
||
$.addClass(el.parentNode, 'forwarded');
|
||
++el.dataset.forwarded || (el.dataset.forwarded = 1);
|
||
}
|
||
if ((i = Unread.replies.indexOf(el)) !== -1) {
|
||
Unread.replies.splice(i, 1);
|
||
return Unread.update(true);
|
||
}
|
||
},
|
||
rm: function(q, id) {
|
||
var div, inlined, _i, _len, _ref;
|
||
div = $.x("following::div[@id='i" + id + "']", q);
|
||
$.rm(div);
|
||
if (!Conf['Forward Hiding']) {
|
||
return;
|
||
}
|
||
_ref = $$('.backlink.inlined', div);
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
inlined = _ref[_i];
|
||
div = $.id(inlined.hash.slice(1));
|
||
if (!--div.dataset.forwarded) {
|
||
$.rmClass(div.parentNode, 'forwarded');
|
||
}
|
||
}
|
||
if (/\bbacklink\b/.test(q.className)) {
|
||
div = $.id("p" + id);
|
||
if (!--div.dataset.forwarded) {
|
||
return $.rmClass(div.parentNode, 'forwarded');
|
||
}
|
||
}
|
||
}
|
||
};
|
||
|
||
QuotePreview = {
|
||
init: function() {
|
||
return Main.callbacks.push(this.node);
|
||
},
|
||
node: function(post) {
|
||
var quote, _i, _j, _len, _len1, _ref, _ref1;
|
||
_ref = post.quotes;
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
quote = _ref[_i];
|
||
if (quote.hash || /\bdeadlink\b/.test(quote.className)) {
|
||
$.on(quote, 'mouseover', QuotePreview.mouseover);
|
||
}
|
||
}
|
||
_ref1 = post.backlinks;
|
||
for (_j = 0, _len1 = _ref1.length; _j < _len1; _j++) {
|
||
quote = _ref1[_j];
|
||
$.on(quote, 'mouseover', QuotePreview.mouseover);
|
||
}
|
||
},
|
||
mouseover: function(e) {
|
||
var board, el, path, postID, qp, quote, quoterID, threadID, _i, _len, _ref;
|
||
if (/\binlined\b/.test(this.className)) {
|
||
return;
|
||
}
|
||
if (qp = $.id('qp')) {
|
||
if (qp === UI.el) {
|
||
delete UI.el;
|
||
}
|
||
$.rm(qp);
|
||
}
|
||
if (UI.el) {
|
||
return;
|
||
}
|
||
if (this.host === 'boards.4chan.org') {
|
||
path = this.pathname.split('/');
|
||
board = path[1];
|
||
threadID = path[3];
|
||
postID = this.hash.slice(2);
|
||
} else {
|
||
board = this.dataset.board;
|
||
threadID = 0;
|
||
postID = this.dataset.id;
|
||
}
|
||
qp = UI.el = $.el('div', {
|
||
id: 'qp',
|
||
className: 'reply dialog'
|
||
});
|
||
UI.hover(e);
|
||
$.add(d.body, qp);
|
||
if (board === g.BOARD) {
|
||
el = $.id("p" + postID);
|
||
}
|
||
Get.post(board, threadID, postID, qp, function() {
|
||
var bq, img, post;
|
||
bq = $('blockquote', qp);
|
||
Main.prettify(bq);
|
||
post = {
|
||
el: qp,
|
||
blockquote: bq,
|
||
isArchived: /\barchivedPost\b/.test(qp.className)
|
||
};
|
||
if (img = $('img[data-md5]', qp)) {
|
||
post.fileInfo = img.parentNode.previousElementSibling;
|
||
post.img = img;
|
||
}
|
||
if (Conf['Reveal Spoilers']) {
|
||
RevealSpoilers.node(post);
|
||
}
|
||
if (Conf['Image Auto-Gif']) {
|
||
AutoGif.node(post);
|
||
}
|
||
if (Conf['Time Formatting']) {
|
||
Time.node(post);
|
||
}
|
||
if (Conf['File Info Formatting']) {
|
||
FileInfo.node(post);
|
||
}
|
||
if (Conf['Resurrect Quotes']) {
|
||
return Quotify.node(post);
|
||
}
|
||
});
|
||
$.on(this, 'mousemove', UI.hover);
|
||
$.on(this, 'mouseout click', QuotePreview.mouseout);
|
||
if (!el) {
|
||
return;
|
||
}
|
||
if (Conf['Quote Highlighting']) {
|
||
if (/\bop\b/.test(el.className)) {
|
||
$.addClass(el.parentNode, 'qphl');
|
||
} else {
|
||
$.addClass(el, 'qphl');
|
||
}
|
||
}
|
||
quoterID = $.x('ancestor::*[@id][1]', this).id.match(/\d+$/)[0];
|
||
_ref = $$('.quotelink, .backlink', qp);
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
quote = _ref[_i];
|
||
if (quote.hash.slice(2) === quoterID) {
|
||
$.addClass(quote, 'forwardlink');
|
||
}
|
||
}
|
||
},
|
||
mouseout: function(e) {
|
||
var el;
|
||
UI.hoverend();
|
||
if (el = $.id(this.hash.slice(1))) {
|
||
$.rmClass(el, 'qphl');
|
||
$.rmClass(el.parentNode, 'qphl');
|
||
}
|
||
$.off(this, 'mousemove', UI.hover);
|
||
return $.off(this, 'mouseout click', QuotePreview.mouseout);
|
||
}
|
||
};
|
||
|
||
QuoteOP = {
|
||
init: function() {
|
||
return Main.callbacks.push(this.node);
|
||
},
|
||
node: function(post) {
|
||
var quote, _i, _len, _ref;
|
||
if (post.isInlined && !post.isCrosspost) {
|
||
return;
|
||
}
|
||
_ref = post.quotes;
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
quote = _ref[_i];
|
||
if (quote.hash.slice(2) === post.threadID) {
|
||
$.add(quote, $.tn('\u00A0(OP)'));
|
||
}
|
||
}
|
||
}
|
||
};
|
||
|
||
QuoteCT = {
|
||
init: function() {
|
||
return Main.callbacks.push(this.node);
|
||
},
|
||
node: function(post) {
|
||
var path, quote, _i, _len, _ref;
|
||
if (post.isInlined && !post.isCrosspost) {
|
||
return;
|
||
}
|
||
_ref = post.quotes;
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
quote = _ref[_i];
|
||
if (!quote.hash) {
|
||
continue;
|
||
}
|
||
path = quote.pathname.split('/');
|
||
if (path[1] === g.BOARD && path[3] !== post.threadID) {
|
||
$.add(quote, $.tn('\u00A0(Cross-thread)'));
|
||
}
|
||
}
|
||
}
|
||
};
|
||
|
||
Quotify = {
|
||
init: function() {
|
||
return Main.callbacks.push(this.node);
|
||
},
|
||
node: function(post) {
|
||
var a, board, data, i, id, index, m, node, nodes, quote, quotes, snapshot, text, _i, _j, _len, _ref;
|
||
if (post.isInlined && !post.isCrosspost) {
|
||
return;
|
||
}
|
||
snapshot = d.evaluate('.//text()[not(parent::a)]', post.blockquote, null, 6, null);
|
||
for (i = _i = 0, _ref = snapshot.snapshotLength; 0 <= _ref ? _i < _ref : _i > _ref; i = 0 <= _ref ? ++_i : --_i) {
|
||
node = snapshot.snapshotItem(i);
|
||
data = node.data;
|
||
if (!(quotes = data.match(/>>(>\/[a-z\d]+\/)?\d+/g))) {
|
||
continue;
|
||
}
|
||
nodes = [];
|
||
for (_j = 0, _len = quotes.length; _j < _len; _j++) {
|
||
quote = quotes[_j];
|
||
index = data.indexOf(quote);
|
||
if (text = data.slice(0, index)) {
|
||
nodes.push($.tn(text));
|
||
}
|
||
id = quote.match(/\d+$/)[0];
|
||
board = (m = quote.match(/^>>>\/([a-z\d]+)/)) ? m[1] : $('.postNum > a[title="Highlight this post"]', post.el).pathname.split('/')[1];
|
||
nodes.push(a = $.el('a', {
|
||
textContent: "" + quote + "\u00A0(Dead)"
|
||
}));
|
||
if (board === g.BOARD && $.id("p" + id)) {
|
||
a.href = "#p" + id;
|
||
a.className = 'quotelink';
|
||
a.setAttribute('onclick', "replyhl('" + id + "');");
|
||
} else {
|
||
a.href = Redirect.thread(board, 0, id);
|
||
a.className = 'deadlink';
|
||
a.target = '_blank';
|
||
if (Redirect.post(board, id)) {
|
||
$.addClass(a, 'quotelink');
|
||
a.setAttribute('data-board', board);
|
||
a.setAttribute('data-id', id);
|
||
}
|
||
}
|
||
data = data.slice(index + quote.length);
|
||
}
|
||
if (data) {
|
||
nodes.push($.tn(data));
|
||
}
|
||
$.replace(node, nodes);
|
||
}
|
||
}
|
||
};
|
||
|
||
DeleteLink = {
|
||
init: function() {
|
||
var a;
|
||
a = $.el('a', {
|
||
className: 'delete_link',
|
||
href: 'javascript:;'
|
||
});
|
||
return Menu.addEntry({
|
||
el: a,
|
||
open: function(post) {
|
||
if (post.isArchived) {
|
||
return false;
|
||
}
|
||
a.textContent = 'Delete this post';
|
||
$.on(a, 'click', DeleteLink["delete"]);
|
||
return true;
|
||
}
|
||
});
|
||
},
|
||
"delete": function() {
|
||
var board, form, id, m, pwd, self;
|
||
$.off(this, 'click', DeleteLink["delete"]);
|
||
this.textContent = 'Deleting...';
|
||
pwd = (m = d.cookie.match(/4chan_pass=([^;]+)/)) ? decodeURIComponent(m[1]) : $.id('delPassword').value;
|
||
id = this.parentNode.dataset.id;
|
||
board = $('.postNum > a[title="Highlight this post"]', $.id(this.parentNode.dataset.rootid)).pathname.split('/')[1];
|
||
self = this;
|
||
form = {
|
||
mode: 'usrdel',
|
||
pwd: pwd
|
||
};
|
||
form[id] = 'delete';
|
||
return $.ajax($.id('delform').action.replace("/" + g.BOARD + "/", "/" + board + "/"), {
|
||
onload: function() {
|
||
return DeleteLink.load(self, this.response);
|
||
},
|
||
onerror: function() {
|
||
return DeleteLink.error(self);
|
||
}
|
||
}, {
|
||
form: $.formData(form)
|
||
});
|
||
},
|
||
load: function(self, html) {
|
||
var doc, msg, s;
|
||
doc = d.implementation.createHTMLDocument('');
|
||
doc.documentElement.innerHTML = html;
|
||
if (doc.title === '4chan - Banned') {
|
||
s = 'Banned!';
|
||
} else if (msg = doc.getElementById('errmsg')) {
|
||
s = msg.textContent;
|
||
$.on(self, 'click', DeleteLink["delete"]);
|
||
} else {
|
||
s = 'Deleted';
|
||
}
|
||
return self.textContent = s;
|
||
},
|
||
error: function(self) {
|
||
self.textContent = 'Connection error, please retry.';
|
||
return $.on(self, 'click', DeleteLink["delete"]);
|
||
}
|
||
};
|
||
|
||
ReportLink = {
|
||
init: function() {
|
||
var a;
|
||
a = $.el('a', {
|
||
className: 'report_link',
|
||
href: 'javascript:;',
|
||
textContent: 'Report this post'
|
||
});
|
||
$.on(a, 'click', this.report);
|
||
return Menu.addEntry({
|
||
el: a,
|
||
open: function(post) {
|
||
return post.isArchived === false;
|
||
}
|
||
});
|
||
},
|
||
report: function() {
|
||
var a, id, set, url;
|
||
a = $('.postNum > a[title="Highlight this post"]', $.id(this.parentNode.dataset.rootid));
|
||
url = "//sys.4chan.org/" + (a.pathname.split('/')[1]) + "/imgboard.php?mode=report&no=" + this.parentNode.dataset.id;
|
||
id = Date.now();
|
||
set = "toolbar=0,scrollbars=0,location=0,status=1,menubar=0,resizable=1,width=685,height=200";
|
||
return window.open(url, id, set);
|
||
}
|
||
};
|
||
|
||
DownloadLink = {
|
||
init: function() {
|
||
var a;
|
||
if ($.el('a').download === void 0) {
|
||
return;
|
||
}
|
||
a = $.el('a', {
|
||
className: 'download_link',
|
||
textContent: 'Download file'
|
||
});
|
||
return Menu.addEntry({
|
||
el: a,
|
||
open: function(post) {
|
||
var fileText;
|
||
if (!post.img) {
|
||
return false;
|
||
}
|
||
a.href = post.img.parentNode.href;
|
||
fileText = post.fileInfo.firstElementChild;
|
||
a.download = Conf['File Info Formatting'] ? fileText.dataset.filename : $('span', fileText).title;
|
||
return true;
|
||
}
|
||
});
|
||
}
|
||
};
|
||
|
||
ArchiveLink = {
|
||
init: function() {
|
||
var a;
|
||
a = $.el('a', {
|
||
className: 'archive_link',
|
||
target: '_blank',
|
||
textContent: 'Archived post'
|
||
});
|
||
return Menu.addEntry({
|
||
el: a,
|
||
open: function(post) {
|
||
var href, path;
|
||
path = $('.postNum > a[title="Highlight this post"]', post.el).pathname.split('/');
|
||
if ((href = Redirect.thread(path[1], path[3], post.ID)) === ("//boards.4chan.org/" + path[1] + "/")) {
|
||
return false;
|
||
}
|
||
a.href = href;
|
||
return true;
|
||
}
|
||
});
|
||
}
|
||
};
|
||
|
||
ThreadStats = {
|
||
init: function() {
|
||
var dialog;
|
||
dialog = UI.dialog('stats', 'bottom: 0; left: 0;', '<div class=move><span id=postcount>0</span> / <span id=imagecount>0</span></div>');
|
||
dialog.className = 'dialog';
|
||
$.add(d.body, dialog);
|
||
this.posts = this.images = 0;
|
||
this.imgLimit = (function() {
|
||
switch (g.BOARD) {
|
||
case 'a':
|
||
case 'b':
|
||
case 'v':
|
||
case 'co':
|
||
case 'mlp':
|
||
return 251;
|
||
case 'vg':
|
||
return 501;
|
||
default:
|
||
return 151;
|
||
}
|
||
})();
|
||
return Main.callbacks.push(this.node);
|
||
},
|
||
node: function(post) {
|
||
var imgcount;
|
||
if (post.isInlined) {
|
||
return;
|
||
}
|
||
$.id('postcount').textContent = ++ThreadStats.posts;
|
||
if (!post.img) {
|
||
return;
|
||
}
|
||
imgcount = $.id('imagecount');
|
||
imgcount.textContent = ++ThreadStats.images;
|
||
if (ThreadStats.images > ThreadStats.imgLimit) {
|
||
return $.addClass(imgcount, 'warning');
|
||
}
|
||
}
|
||
};
|
||
|
||
Unread = {
|
||
init: function() {
|
||
this.title = d.title;
|
||
this.update();
|
||
$.on(window, 'scroll', Unread.scroll);
|
||
return Main.callbacks.push(this.node);
|
||
},
|
||
replies: [],
|
||
foresee: [],
|
||
node: function(post) {
|
||
var count, el, index;
|
||
if ((index = Unread.foresee.indexOf(post.ID)) !== -1) {
|
||
Unread.foresee.splice(index, 1);
|
||
return;
|
||
}
|
||
el = post.el;
|
||
if (el.hidden || /\bop\b/.test(post["class"]) || post.isInlined) {
|
||
return;
|
||
}
|
||
count = Unread.replies.push(el);
|
||
return Unread.update(count === 1);
|
||
},
|
||
scroll: function() {
|
||
var bottom, height, i, reply, _i, _len, _ref;
|
||
height = d.documentElement.clientHeight;
|
||
_ref = Unread.replies;
|
||
for (i = _i = 0, _len = _ref.length; _i < _len; i = ++_i) {
|
||
reply = _ref[i];
|
||
bottom = reply.getBoundingClientRect().bottom;
|
||
if (bottom > height) {
|
||
break;
|
||
}
|
||
}
|
||
if (i === 0) {
|
||
return;
|
||
}
|
||
Unread.replies = Unread.replies.slice(i);
|
||
return Unread.update(Unread.replies.length === 0);
|
||
},
|
||
setTitle: function(count) {
|
||
if (this.scheduled) {
|
||
clearTimeout(this.scheduled);
|
||
delete Unread.scheduled;
|
||
this.setTitle(count);
|
||
return;
|
||
}
|
||
return this.scheduled = setTimeout((function() {
|
||
return d.title = "(" + count + ") " + Unread.title;
|
||
}), 5);
|
||
},
|
||
update: function(updateFavicon) {
|
||
var count;
|
||
if (!g.REPLY) {
|
||
return;
|
||
}
|
||
count = this.replies.length;
|
||
if (Conf['Unread Count']) {
|
||
this.setTitle(count);
|
||
}
|
||
if (!(Conf['Unread Favicon'] && updateFavicon)) {
|
||
return;
|
||
}
|
||
if ($.engine === 'presto') {
|
||
$.rm(Favicon.el);
|
||
}
|
||
Favicon.el.href = g.dead ? count ? Favicon.unreadDead : Favicon.dead : count ? Favicon.unread : Favicon["default"];
|
||
if (g.dead) {
|
||
$.addClass(Favicon.el, 'dead');
|
||
} else {
|
||
$.rmClass(Favicon.el, 'dead');
|
||
}
|
||
if (count) {
|
||
$.addClass(Favicon.el, 'unread');
|
||
} else {
|
||
$.rmClass(Favicon.el, 'unread');
|
||
}
|
||
if ($.engine !== 'webkit') {
|
||
return $.add(d.head, Favicon.el);
|
||
}
|
||
}
|
||
};
|
||
|
||
Favicon = {
|
||
init: function() {
|
||
var href;
|
||
if (this.el) {
|
||
return;
|
||
}
|
||
this.el = $('link[rel="shortcut icon"]', d.head);
|
||
this.el.type = 'image/x-icon';
|
||
href = this.el.href;
|
||
this.SFW = /ws.ico$/.test(href);
|
||
this["default"] = href;
|
||
return this["switch"]();
|
||
},
|
||
"switch": function() {
|
||
switch (Conf['favicon']) {
|
||
case 'ferongr':
|
||
this.unreadDead = '';
|
||
this.unreadSFW = '';
|
||
this.unreadNSFW = '';
|
||
break;
|
||
case 'xat-':
|
||
this.unreadDead = '';
|
||
this.unreadSFW = '';
|
||
this.unreadNSFW = '';
|
||
break;
|
||
case 'Mayhem':
|
||
this.unreadDead = '';
|
||
this.unreadSFW = '';
|
||
this.unreadNSFW = '';
|
||
break;
|
||
case 'Original':
|
||
this.unreadDead = '';
|
||
this.unreadSFW = '';
|
||
this.unreadNSFW = '';
|
||
}
|
||
return this.unread = this.SFW ? this.unreadSFW : this.unreadNSFW;
|
||
},
|
||
empty: '',
|
||
dead: ''
|
||
};
|
||
|
||
Redirect = {
|
||
image: function(board, filename) {
|
||
switch (board) {
|
||
case 'a':
|
||
case 'jp':
|
||
case 'm':
|
||
case 'sp':
|
||
case 'tg':
|
||
case 'vg':
|
||
return "//archive.foolz.us/" + board + "/full_image/" + filename;
|
||
case 'u':
|
||
return "//nsfw.foolz.us/" + board + "/full_image/" + filename;
|
||
}
|
||
},
|
||
post: function(board, postID) {
|
||
switch (board) {
|
||
case 'a':
|
||
case 'co':
|
||
case 'jp':
|
||
case 'm':
|
||
case 'sp':
|
||
case 'tg':
|
||
case 'tv':
|
||
case 'v':
|
||
case 'vg':
|
||
case 'dev':
|
||
case 'foolz':
|
||
return "//archive.foolz.us/api/chan/post/board/" + board + "/num/" + postID + "/format/json";
|
||
case 'u':
|
||
case 'kuku':
|
||
return "//nsfw.foolz.us/api/chan/post/board/" + board + "/num/" + postID + "/format/json";
|
||
}
|
||
},
|
||
thread: function(board, threadID, postID) {
|
||
var path, url;
|
||
if (postID) {
|
||
postID = postID.match(/\d+/)[0];
|
||
}
|
||
path = threadID ? "" + board + "/thread/" + threadID : "" + board + "/post/" + postID;
|
||
switch (board) {
|
||
case 'a':
|
||
case 'co':
|
||
case 'jp':
|
||
case 'm':
|
||
case 'sp':
|
||
case 'tg':
|
||
case 'tv':
|
||
case 'v':
|
||
case 'vg':
|
||
case 'dev':
|
||
case 'foolz':
|
||
url = "//archive.foolz.us/" + path + "/";
|
||
if (threadID && postID) {
|
||
url += "#" + postID;
|
||
}
|
||
break;
|
||
case 'u':
|
||
case 'kuku':
|
||
url = "//nsfw.foolz.us/" + path + "/";
|
||
if (threadID && postID) {
|
||
url += "#" + postID;
|
||
}
|
||
break;
|
||
case 'ck':
|
||
case 'lit':
|
||
url = "//fuuka.warosu.org/" + path;
|
||
if (threadID && postID) {
|
||
url += "#p" + postID;
|
||
}
|
||
break;
|
||
case 'diy':
|
||
case 'g':
|
||
case 'k':
|
||
case 'sci':
|
||
url = "//archive.installgentoo.net/" + path;
|
||
if (threadID && postID) {
|
||
url += "#p" + postID;
|
||
}
|
||
break;
|
||
case 'cgl':
|
||
case 'mu':
|
||
case 'soc':
|
||
case 'w':
|
||
url = "//archive.rebeccablacktech.com/" + path;
|
||
if (threadID && postID) {
|
||
url += "#p" + postID;
|
||
}
|
||
break;
|
||
case 'an':
|
||
case 'r9k':
|
||
case 'toy':
|
||
case 'x':
|
||
url = "http://archive.maidlab.jp/" + path;
|
||
if (threadID && postID) {
|
||
url += "#p" + postID;
|
||
}
|
||
break;
|
||
case 'e':
|
||
url = "https://md401.homelinux.net/4chan/cgi-board.pl/" + path;
|
||
if (threadID && postID) {
|
||
url += "#p" + postID;
|
||
}
|
||
break;
|
||
default:
|
||
if (threadID) {
|
||
url = "//boards.4chan.org/" + board + "/";
|
||
}
|
||
}
|
||
return url || null;
|
||
}
|
||
};
|
||
|
||
ImageHover = {
|
||
init: function() {
|
||
return Main.callbacks.push(this.node);
|
||
},
|
||
node: function(post) {
|
||
if (!post.img) {
|
||
return;
|
||
}
|
||
return $.on(post.img, 'mouseover', ImageHover.mouseover);
|
||
},
|
||
mouseover: function() {
|
||
var el;
|
||
if (el = $.id('ihover')) {
|
||
if (el === UI.el) {
|
||
delete UI.el;
|
||
}
|
||
$.rm(el);
|
||
}
|
||
if (UI.el) {
|
||
return;
|
||
}
|
||
el = UI.el = $.el('img', {
|
||
id: 'ihover',
|
||
src: this.parentNode.href
|
||
});
|
||
$.add(d.body, el);
|
||
$.on(el, 'load', ImageHover.load);
|
||
$.on(el, 'error', ImageHover.error);
|
||
$.on(this, 'mousemove', UI.hover);
|
||
return $.on(this, 'mouseout', ImageHover.mouseout);
|
||
},
|
||
load: function() {
|
||
var style;
|
||
if (!this.parentNode) {
|
||
return;
|
||
}
|
||
style = this.style;
|
||
return UI.hover({
|
||
clientX: -45 + parseInt(style.left),
|
||
clientY: 120 + parseInt(style.top)
|
||
});
|
||
},
|
||
error: function() {
|
||
var src, timeoutID, url,
|
||
_this = this;
|
||
src = this.src.replace(/\?\d+$/, '').split('/');
|
||
if (!(src[2] === 'images.4chan.org' && (url = Redirect.image(src[3], src[5])))) {
|
||
if (g.dead) {
|
||
return;
|
||
}
|
||
url = "//images.4chan.org/" + src[3] + "/src/" + src[5] + "?" + (Date.now());
|
||
}
|
||
if ($.engine !== 'webkit' && url.split('/')[2] === 'images.4chan.org') {
|
||
return;
|
||
}
|
||
timeoutID = setTimeout((function() {
|
||
return _this.src = url;
|
||
}), 3000);
|
||
if ($.engine !== 'webkit' || url.split('/')[2] !== 'images.4chan.org') {
|
||
return;
|
||
}
|
||
return $.ajax(url, {
|
||
onreadystatechange: (function() {
|
||
if (this.status === 404) {
|
||
return clearTimeout(timeoutID);
|
||
}
|
||
})
|
||
}, {
|
||
type: 'head'
|
||
});
|
||
},
|
||
mouseout: function() {
|
||
UI.hoverend();
|
||
$.off(this, 'mousemove', UI.hover);
|
||
return $.off(this, 'mouseout', ImageHover.mouseout);
|
||
}
|
||
};
|
||
|
||
AutoGif = {
|
||
init: function() {
|
||
var _ref;
|
||
if ((_ref = g.BOARD) === 'gif' || _ref === 'wsg') {
|
||
return;
|
||
}
|
||
return Main.callbacks.push(this.node);
|
||
},
|
||
node: function(post) {
|
||
var gif, img, src;
|
||
img = post.img;
|
||
if (post.el.hidden || !img) {
|
||
return;
|
||
}
|
||
src = img.parentNode.href;
|
||
if (/gif$/.test(src) && !/spoiler/.test(img.src)) {
|
||
gif = $.el('img');
|
||
$.on(gif, 'load', function() {
|
||
return img.src = src;
|
||
});
|
||
return gif.src = src;
|
||
}
|
||
}
|
||
};
|
||
|
||
ImageExpand = {
|
||
init: function() {
|
||
Main.callbacks.push(this.node);
|
||
return this.dialog();
|
||
},
|
||
node: function(post) {
|
||
var a;
|
||
if (!post.img) {
|
||
return;
|
||
}
|
||
a = post.img.parentNode;
|
||
$.on(a, 'click', ImageExpand.cb.toggle);
|
||
if (ImageExpand.on && !post.el.hidden) {
|
||
return ImageExpand.expand(post.img);
|
||
}
|
||
},
|
||
cb: {
|
||
toggle: function(e) {
|
||
if (e.shiftKey || e.altKey || e.ctrlKey || e.metaKey || e.button !== 0) {
|
||
return;
|
||
}
|
||
e.preventDefault();
|
||
return ImageExpand.toggle(this);
|
||
},
|
||
all: function() {
|
||
var i, thumb, thumbs, _i, _j, _k, _len, _len1, _len2, _ref;
|
||
ImageExpand.on = this.checked;
|
||
if (ImageExpand.on) {
|
||
thumbs = $$('img[data-md5]');
|
||
if (Conf['Expand From Current']) {
|
||
for (i = _i = 0, _len = thumbs.length; _i < _len; i = ++_i) {
|
||
thumb = thumbs[i];
|
||
if (thumb.getBoundingClientRect().top > 0) {
|
||
break;
|
||
}
|
||
}
|
||
thumbs = thumbs.slice(i);
|
||
}
|
||
for (_j = 0, _len1 = thumbs.length; _j < _len1; _j++) {
|
||
thumb = thumbs[_j];
|
||
ImageExpand.expand(thumb);
|
||
}
|
||
} else {
|
||
_ref = $$('img[data-md5][hidden]');
|
||
for (_k = 0, _len2 = _ref.length; _k < _len2; _k++) {
|
||
thumb = _ref[_k];
|
||
ImageExpand.contract(thumb);
|
||
}
|
||
}
|
||
},
|
||
typeChange: function() {
|
||
var klass;
|
||
switch (this.value) {
|
||
case 'full':
|
||
klass = '';
|
||
break;
|
||
case 'fit width':
|
||
klass = 'fitwidth';
|
||
break;
|
||
case 'fit height':
|
||
klass = 'fitheight';
|
||
break;
|
||
case 'fit screen':
|
||
klass = 'fitwidth fitheight';
|
||
}
|
||
$.id('delform').className = klass;
|
||
if (/\bfitheight\b/.test(klass)) {
|
||
$.on(window, 'resize', ImageExpand.resize);
|
||
if (!ImageExpand.style) {
|
||
ImageExpand.style = $.addStyle('');
|
||
}
|
||
return ImageExpand.resize();
|
||
} else if (ImageExpand.style) {
|
||
return $.off(window, 'resize', ImageExpand.resize);
|
||
}
|
||
}
|
||
},
|
||
toggle: function(a) {
|
||
var rect, thumb;
|
||
thumb = a.firstChild;
|
||
if (thumb.hidden) {
|
||
rect = a.getBoundingClientRect();
|
||
if ($.engine === 'webkit') {
|
||
if (rect.top < 0) {
|
||
d.body.scrollTop += rect.top - 42;
|
||
}
|
||
if (rect.left < 0) {
|
||
d.body.scrollLeft += rect.left;
|
||
}
|
||
} else {
|
||
if (rect.top < 0) {
|
||
d.documentElement.scrollTop += rect.top - 42;
|
||
}
|
||
if (rect.left < 0) {
|
||
d.documentElement.scrollLeft += rect.left;
|
||
}
|
||
}
|
||
return ImageExpand.contract(thumb);
|
||
} else {
|
||
return ImageExpand.expand(thumb);
|
||
}
|
||
},
|
||
contract: function(thumb) {
|
||
thumb.hidden = false;
|
||
thumb.nextSibling.hidden = true;
|
||
return $.rmClass(thumb.parentNode.parentNode.parentNode, 'image_expanded');
|
||
},
|
||
expand: function(thumb, url) {
|
||
var a, img;
|
||
if ($.x('ancestor-or-self::*[@hidden]', thumb)) {
|
||
return;
|
||
}
|
||
thumb.hidden = true;
|
||
$.addClass(thumb.parentNode.parentNode.parentNode, 'image_expanded');
|
||
if (img = thumb.nextSibling) {
|
||
img.hidden = false;
|
||
return;
|
||
}
|
||
a = thumb.parentNode;
|
||
img = $.el('img', {
|
||
src: url || a.href
|
||
});
|
||
$.on(img, 'error', ImageExpand.error);
|
||
return $.add(a, img);
|
||
},
|
||
error: function() {
|
||
var src, thumb, timeoutID, url;
|
||
thumb = this.previousSibling;
|
||
ImageExpand.contract(thumb);
|
||
$.rm(this);
|
||
src = this.src.replace(/\?\d+$/, '').split('/');
|
||
if (!(src[2] === 'images.4chan.org' && (url = Redirect.image(src[3], src[5])))) {
|
||
if (g.dead) {
|
||
return;
|
||
}
|
||
url = "//images.4chan.org/" + src[3] + "/src/" + src[5] + "?" + (Date.now());
|
||
}
|
||
if ($.engine !== 'webkit' && url.split('/')[2] === 'images.4chan.org') {
|
||
return;
|
||
}
|
||
timeoutID = setTimeout(ImageExpand.expand, 10000, thumb, url);
|
||
if ($.engine !== 'webkit' || url.split('/')[2] !== 'images.4chan.org') {
|
||
return;
|
||
}
|
||
return $.ajax(url, {
|
||
onreadystatechange: (function() {
|
||
if (this.status === 404) {
|
||
return clearTimeout(timeoutID);
|
||
}
|
||
})
|
||
}, {
|
||
type: 'head'
|
||
});
|
||
},
|
||
dialog: function() {
|
||
var controls, imageType, select;
|
||
controls = $.el('div', {
|
||
id: 'imgControls',
|
||
innerHTML: "<select id=imageType name=imageType><option value=full>Full</option><option value='fit width'>Fit Width</option><option value='fit height'>Fit Height</option value='fit screen'><option value='fit screen'>Fit Screen</option></select><label>Expand Images<input type=checkbox id=imageExpand></label>"
|
||
});
|
||
imageType = $.get('imageType', 'full');
|
||
select = $('select', controls);
|
||
select.value = imageType;
|
||
ImageExpand.cb.typeChange.call(select);
|
||
$.on(select, 'change', $.cb.value);
|
||
$.on(select, 'change', ImageExpand.cb.typeChange);
|
||
$.on($('input', controls), 'click', ImageExpand.cb.all);
|
||
return $.prepend($.id('delform'), controls);
|
||
},
|
||
resize: function() {
|
||
return ImageExpand.style.textContent = ".fitheight img[data-md5] + img {max-height:" + d.documentElement.clientHeight + "px;}";
|
||
}
|
||
};
|
||
|
||
Main = {
|
||
init: function() {
|
||
var cutoff, hiddenThreads, id, key, now, path, pathname, temp, timestamp, val, _ref;
|
||
Main.flatten(null, Config);
|
||
path = location.pathname;
|
||
pathname = path.slice(1).split('/');
|
||
g.BOARD = pathname[0], temp = pathname[1];
|
||
if (temp === 'res') {
|
||
g.REPLY = true;
|
||
g.THREAD_ID = pathname[2];
|
||
}
|
||
for (key in Conf) {
|
||
val = Conf[key];
|
||
Conf[key] = $.get(key, val);
|
||
}
|
||
switch (location.hostname) {
|
||
case 'sys.4chan.org':
|
||
if (/report/.test(location.search)) {
|
||
$.ready(function() {
|
||
return $.on($.id('recaptcha_response_field'), 'keydown', function(e) {
|
||
if (e.keyCode === 8 && !e.target.value) {
|
||
return window.location = 'javascript:Recaptcha.reload()';
|
||
}
|
||
});
|
||
});
|
||
}
|
||
return;
|
||
case 'images.4chan.org':
|
||
$.ready(function() {
|
||
var url;
|
||
if (/^4chan - 404/.test(d.title) && Conf['404 Redirect']) {
|
||
path = location.pathname.split('/');
|
||
url = Redirect.image(path[1], path[3]);
|
||
if (url) {
|
||
return location.href = url;
|
||
}
|
||
}
|
||
});
|
||
return;
|
||
}
|
||
$.ready(Options.init);
|
||
if (Conf['Quick Reply'] && Conf['Hide Original Post Form']) {
|
||
Main.css += '#postForm { display: none; }';
|
||
}
|
||
Main.addStyle();
|
||
now = Date.now();
|
||
if (Conf['Check for Updates'] && $.get('lastUpdate', 0) < now - 6 * $.HOUR) {
|
||
$.ready(function() {
|
||
$.on(window, 'message', Main.message);
|
||
$.set('lastUpdate', now);
|
||
return $.add(d.head, $.el('script', {
|
||
src: 'https://github.com/MayhemYDG/4chan-x/raw/master/latest.js'
|
||
}));
|
||
});
|
||
}
|
||
g.hiddenReplies = $.get("hiddenReplies/" + g.BOARD + "/", {});
|
||
if ($.get('lastChecked', 0) < now - 1 * $.DAY) {
|
||
$.set('lastChecked', now);
|
||
cutoff = now - 7 * $.DAY;
|
||
hiddenThreads = $.get("hiddenThreads/" + g.BOARD + "/", {});
|
||
for (id in hiddenThreads) {
|
||
timestamp = hiddenThreads[id];
|
||
if (timestamp < cutoff) {
|
||
delete hiddenThreads[id];
|
||
}
|
||
}
|
||
_ref = g.hiddenReplies;
|
||
for (id in _ref) {
|
||
timestamp = _ref[id];
|
||
if (timestamp < cutoff) {
|
||
delete g.hiddenReplies[id];
|
||
}
|
||
}
|
||
$.set("hiddenThreads/" + g.BOARD + "/", hiddenThreads);
|
||
$.set("hiddenReplies/" + g.BOARD + "/", g.hiddenReplies);
|
||
}
|
||
if (Conf['Filter']) {
|
||
Filter.init();
|
||
}
|
||
if (Conf['Reply Hiding']) {
|
||
ReplyHiding.init();
|
||
}
|
||
if (Conf['Filter'] || Conf['Reply Hiding']) {
|
||
StrikethroughQuotes.init();
|
||
}
|
||
if (Conf['Anonymize']) {
|
||
Anonymize.init();
|
||
}
|
||
if (Conf['Time Formatting']) {
|
||
Time.init();
|
||
}
|
||
if (Conf['File Info Formatting']) {
|
||
FileInfo.init();
|
||
}
|
||
if (Conf['Sauce']) {
|
||
Sauce.init();
|
||
}
|
||
if (Conf['Reveal Spoilers']) {
|
||
RevealSpoilers.init();
|
||
}
|
||
if (Conf['Image Auto-Gif']) {
|
||
AutoGif.init();
|
||
}
|
||
if (Conf['Image Hover']) {
|
||
ImageHover.init();
|
||
}
|
||
if (Conf['Menu']) {
|
||
Menu.init();
|
||
if (Conf['Report Link']) {
|
||
ReportLink.init();
|
||
}
|
||
if (Conf['Delete Link']) {
|
||
DeleteLink.init();
|
||
}
|
||
if (Conf['Filter']) {
|
||
Filter.menuInit();
|
||
}
|
||
if (Conf['Download Link']) {
|
||
DownloadLink.init();
|
||
}
|
||
if (Conf['Archive Link']) {
|
||
ArchiveLink.init();
|
||
}
|
||
}
|
||
if (Conf['Resurrect Quotes']) {
|
||
Quotify.init();
|
||
}
|
||
if (Conf['Quote Inline']) {
|
||
QuoteInline.init();
|
||
}
|
||
if (Conf['Quote Preview']) {
|
||
QuotePreview.init();
|
||
}
|
||
if (Conf['Quote Backlinks']) {
|
||
QuoteBacklink.init();
|
||
}
|
||
if (Conf['Indicate OP quote']) {
|
||
QuoteOP.init();
|
||
}
|
||
if (Conf['Indicate Cross-thread Quotes']) {
|
||
QuoteCT.init();
|
||
}
|
||
return $.ready(Main.ready);
|
||
},
|
||
ready: function() {
|
||
var MutationObserver, a, board, nav, node, nodes, observer, _i, _j, _len, _len1, _ref, _ref1;
|
||
if (/^4chan - 404/.test(d.title)) {
|
||
if (Conf['404 Redirect'] && /^\d+$/.test(g.THREAD_ID)) {
|
||
location.href = Redirect.thread(g.BOARD, g.THREAD_ID, location.hash);
|
||
}
|
||
return;
|
||
}
|
||
if (!$.id('navtopr')) {
|
||
return;
|
||
}
|
||
$.addClass(d.body, $.engine);
|
||
$.addClass(d.body, 'fourchan_x');
|
||
_ref = ['boardNavDesktop', 'boardNavDesktopFoot'];
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
nav = _ref[_i];
|
||
if (a = $("a[href$='/" + g.BOARD + "/']", $.id(nav))) {
|
||
$.addClass(a, 'current');
|
||
}
|
||
}
|
||
Favicon.init();
|
||
if (Conf['Quick Reply']) {
|
||
QR.init();
|
||
}
|
||
if (Conf['Image Expansion']) {
|
||
ImageExpand.init();
|
||
}
|
||
if (Conf['Thread Watcher']) {
|
||
setTimeout(function() {
|
||
return Watcher.init();
|
||
});
|
||
}
|
||
if (Conf['Keybinds']) {
|
||
setTimeout(function() {
|
||
return Keybinds.init();
|
||
});
|
||
}
|
||
if (g.REPLY) {
|
||
if (Conf['Thread Updater']) {
|
||
setTimeout(function() {
|
||
return Updater.init();
|
||
});
|
||
}
|
||
if (Conf['Thread Stats']) {
|
||
ThreadStats.init();
|
||
}
|
||
if (Conf['Reply Navigation']) {
|
||
setTimeout(function() {
|
||
return Nav.init();
|
||
});
|
||
}
|
||
if (Conf['Post in Title']) {
|
||
TitlePost.init();
|
||
}
|
||
if (Conf['Unread Count'] || Conf['Unread Favicon']) {
|
||
Unread.init();
|
||
}
|
||
} else {
|
||
if (Conf['Thread Hiding']) {
|
||
ThreadHiding.init();
|
||
}
|
||
if (Conf['Thread Expansion']) {
|
||
setTimeout(function() {
|
||
return ExpandThread.init();
|
||
});
|
||
}
|
||
if (Conf['Comment Expansion']) {
|
||
setTimeout(function() {
|
||
return ExpandComment.init();
|
||
});
|
||
}
|
||
if (Conf['Index Navigation']) {
|
||
setTimeout(function() {
|
||
return Nav.init();
|
||
});
|
||
}
|
||
}
|
||
board = $('.board');
|
||
nodes = [];
|
||
_ref1 = $$('.postContainer', board);
|
||
for (_j = 0, _len1 = _ref1.length; _j < _len1; _j++) {
|
||
node = _ref1[_j];
|
||
nodes.push(Main.preParse(node));
|
||
}
|
||
Main.node(nodes, true);
|
||
Main.hasCodeTags = !!$('script[src="//static.4chan.org/js/prettify/prettify.js"]');
|
||
if (MutationObserver = window.WebKitMutationObserver || window.MozMutationObserver || window.OMutationObserver || window.MutationObserver) {
|
||
observer = new MutationObserver(Main.observer);
|
||
observer.observe(board, {
|
||
childList: true,
|
||
subtree: true
|
||
});
|
||
} else {
|
||
$.on(board, 'DOMNodeInserted', Main.listener);
|
||
}
|
||
},
|
||
flatten: function(parent, obj) {
|
||
var key, val;
|
||
if (obj instanceof Array) {
|
||
Conf[parent] = obj[0];
|
||
} else if (typeof obj === 'object') {
|
||
for (key in obj) {
|
||
val = obj[key];
|
||
Main.flatten(key, val);
|
||
}
|
||
} else {
|
||
Conf[parent] = obj;
|
||
}
|
||
},
|
||
addStyle: function() {
|
||
$.off(d, 'DOMNodeInserted', Main.addStyle);
|
||
if (d.head) {
|
||
return $.addStyle(Main.css);
|
||
} else {
|
||
return $.on(d, 'DOMNodeInserted', Main.addStyle);
|
||
}
|
||
},
|
||
message: function(e) {
|
||
var version;
|
||
version = e.data.version;
|
||
if (version && version !== Main.version && confirm('An updated version of 4chan X is available, would you like to install it now?')) {
|
||
return window.location = "https://raw.github.com/mayhemydg/4chan-x/" + version + "/4chan_x.user.js";
|
||
}
|
||
},
|
||
preParse: function(node) {
|
||
var el, img, parentClass, post;
|
||
parentClass = node.parentNode.className;
|
||
el = $('.post', node);
|
||
post = {
|
||
root: node,
|
||
el: el,
|
||
"class": el.className,
|
||
ID: el.id.match(/\d+$/)[0],
|
||
threadID: g.THREAD_ID || $.x('ancestor::div[parent::div[@class="board"]]', node).id.match(/\d+$/)[0],
|
||
isArchived: /\barchivedPost\b/.test(parentClass),
|
||
isInlined: /\binline\b/.test(parentClass),
|
||
isCrosspost: /\bcrosspost\b/.test(parentClass),
|
||
blockquote: el.lastElementChild,
|
||
quotes: el.getElementsByClassName('quotelink'),
|
||
backlinks: el.getElementsByClassName('backlink'),
|
||
fileInfo: false,
|
||
img: false
|
||
};
|
||
if (img = $('img[data-md5]', el)) {
|
||
post.fileInfo = img.parentNode.previousElementSibling;
|
||
post.img = img;
|
||
}
|
||
Main.prettify(post.blockquote);
|
||
return post;
|
||
},
|
||
node: function(nodes, notify) {
|
||
var callback, node, _i, _j, _len, _len1, _ref;
|
||
_ref = Main.callbacks;
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
callback = _ref[_i];
|
||
try {
|
||
for (_j = 0, _len1 = nodes.length; _j < _len1; _j++) {
|
||
node = nodes[_j];
|
||
callback(node);
|
||
}
|
||
} catch (err) {
|
||
if (notify) {
|
||
alert("4chan X (" + Main.version + ") error: " + err.message + "\nReport the bug at mayhemydg.github.com/4chan-x/#bug-report\n\nURL: " + window.location + "\n" + err.stack);
|
||
}
|
||
}
|
||
}
|
||
},
|
||
observer: function(mutations) {
|
||
var addedNode, mutation, nodes, _i, _j, _len, _len1, _ref;
|
||
nodes = [];
|
||
for (_i = 0, _len = mutations.length; _i < _len; _i++) {
|
||
mutation = mutations[_i];
|
||
_ref = mutation.addedNodes;
|
||
for (_j = 0, _len1 = _ref.length; _j < _len1; _j++) {
|
||
addedNode = _ref[_j];
|
||
if (/\bpostContainer\b/.test(addedNode.className)) {
|
||
nodes.push(Main.preParse(addedNode));
|
||
}
|
||
}
|
||
}
|
||
if (nodes.length) {
|
||
return Main.node(nodes);
|
||
}
|
||
},
|
||
listener: function(e) {
|
||
var target;
|
||
target = e.target;
|
||
if (/\bpostContainer\b/.test(target.className)) {
|
||
return Main.node([Main.preParse(target)]);
|
||
}
|
||
},
|
||
prettify: function(bq) {
|
||
var code;
|
||
if (!Main.hasCodeTags) {
|
||
return;
|
||
}
|
||
code = function() {
|
||
var pre, _i, _len, _ref;
|
||
_ref = document.getElementById('_id_').getElementsByClassName('prettyprint');
|
||
for (_i = 0, _len = _ref.length; _i < _len; _i++) {
|
||
pre = _ref[_i];
|
||
pre.innerHTML = prettyPrintOne(pre.innerHTML.replace(/\s/g, ' '));
|
||
}
|
||
};
|
||
return $.globalEval(("(" + code + ")()").replace('_id_', bq.id));
|
||
},
|
||
namespace: '4chan_x.',
|
||
version: '2.33.8',
|
||
callbacks: [],
|
||
css: '\
|
||
/* dialog styling */\
|
||
.dialog.reply {\
|
||
display: block;\
|
||
border: 1px solid rgba(0,0,0,.25);\
|
||
padding: 0;\
|
||
}\
|
||
.move {\
|
||
cursor: move;\
|
||
}\
|
||
label, .favicon {\
|
||
cursor: pointer;\
|
||
}\
|
||
a[href="javascript:;"] {\
|
||
text-decoration: none;\
|
||
}\
|
||
.warning {\
|
||
color: red;\
|
||
}\
|
||
\
|
||
.hide_thread_button:not(.hidden_thread) {\
|
||
float: left;\
|
||
}\
|
||
\
|
||
.thread > .hidden_thread ~ *,\
|
||
[hidden],\
|
||
#content > [name=tab]:not(:checked) + div,\
|
||
#updater:not(:hover) > :not(.move),\
|
||
.autohide:not(:hover) > form,\
|
||
#qp input, #qp .inline, .forwarded {\
|
||
display: none !important;\
|
||
}\
|
||
\
|
||
.menu_button {\
|
||
display: inline-block;\
|
||
}\
|
||
.menu_button > span {\
|
||
border-top: .5em solid;\
|
||
border-right: .3em solid transparent;\
|
||
border-left: .3em solid transparent;\
|
||
display: inline-block;\
|
||
margin: 2px;\
|
||
vertical-align: middle;\
|
||
}\
|
||
#menu {\
|
||
position: absolute;\
|
||
outline: none;\
|
||
}\
|
||
.entry {\
|
||
border-bottom: 1px solid rgba(0, 0, 0, .25);\
|
||
cursor: pointer;\
|
||
display: block;\
|
||
outline: none;\
|
||
padding: 3px 7px;\
|
||
position: relative;\
|
||
text-decoration: none;\
|
||
white-space: nowrap;\
|
||
}\
|
||
.entry:last-child {\
|
||
border: none;\
|
||
}\
|
||
.focused.entry {\
|
||
background: rgba(255, 255, 255, .33);\
|
||
}\
|
||
.entry.hasSubMenu {\
|
||
padding-right: 1.5em;\
|
||
}\
|
||
.hasSubMenu::after {\
|
||
content: "";\
|
||
border-left: .5em solid;\
|
||
border-top: .3em solid transparent;\
|
||
border-bottom: .3em solid transparent;\
|
||
display: inline-block;\
|
||
margin: .3em;\
|
||
position: absolute;\
|
||
right: 3px;\
|
||
}\
|
||
.hasSubMenu:not(.focused) > .subMenu {\
|
||
display: none;\
|
||
}\
|
||
.subMenu {\
|
||
position: absolute;\
|
||
left: 100%;\
|
||
top: 0;\
|
||
margin-top: -1px;\
|
||
}\
|
||
\
|
||
h1 {\
|
||
text-align: center;\
|
||
}\
|
||
#qr > .move {\
|
||
min-width: 300px;\
|
||
overflow: hidden;\
|
||
box-sizing: border-box;\
|
||
-moz-box-sizing: border-box;\
|
||
padding: 0 2px;\
|
||
}\
|
||
#qr > .move > span {\
|
||
float: right;\
|
||
}\
|
||
#autohide, .close, #qr select, #dump, .remove, .captchaimg, #qr div.warning {\
|
||
cursor: pointer;\
|
||
}\
|
||
#qr select,\
|
||
#qr > form {\
|
||
margin: 0;\
|
||
}\
|
||
#dump {\
|
||
background: -webkit-linear-gradient(#EEE, #CCC);\
|
||
background: -moz-linear-gradient(#EEE, #CCC);\
|
||
background: -o-linear-gradient(#EEE, #CCC);\
|
||
background: linear-gradient(#EEE, #CCC);\
|
||
width: 10%;\
|
||
padding: -moz-calc(1px) 0 2px;\
|
||
}\
|
||
#dump:hover, #dump:focus {\
|
||
background: -webkit-linear-gradient(#FFF, #DDD);\
|
||
background: -moz-linear-gradient(#FFF, #DDD);\
|
||
background: -o-linear-gradient(#FFF, #DDD);\
|
||
background: linear-gradient(#FFF, #DDD);\
|
||
}\
|
||
#dump:active, .dump #dump:not(:hover):not(:focus) {\
|
||
background: -webkit-linear-gradient(#CCC, #DDD);\
|
||
background: -moz-linear-gradient(#CCC, #DDD);\
|
||
background: -o-linear-gradient(#CCC, #DDD);\
|
||
background: linear-gradient(#CCC, #DDD);\
|
||
}\
|
||
#qr:not(.dump) #replies, .dump > form > label {\
|
||
display: none;\
|
||
}\
|
||
#replies {\
|
||
display: block;\
|
||
height: 100px;\
|
||
position: relative;\
|
||
-webkit-user-select: none;\
|
||
-moz-user-select: none;\
|
||
-o-user-select: none;\
|
||
user-select: none;\
|
||
}\
|
||
#replies > div {\
|
||
counter-reset: thumbnails;\
|
||
top: 0; right: 0; bottom: 0; left: 0;\
|
||
margin: 0; padding: 0;\
|
||
overflow: hidden;\
|
||
position: absolute;\
|
||
white-space: pre;\
|
||
}\
|
||
#replies > div:hover {\
|
||
bottom: -10px;\
|
||
overflow-x: auto;\
|
||
z-index: 1;\
|
||
}\
|
||
.thumbnail {\
|
||
background-color: rgba(0,0,0,.2) !important;\
|
||
background-position: 50% 20% !important;\
|
||
background-size: cover !important;\
|
||
border: 1px solid #666;\
|
||
box-sizing: border-box;\
|
||
-moz-box-sizing: border-box;\
|
||
cursor: move;\
|
||
display: inline-block;\
|
||
height: 90px; width: 90px;\
|
||
margin: 5px; padding: 2px;\
|
||
opacity: .5;\
|
||
outline: none;\
|
||
overflow: hidden;\
|
||
position: relative;\
|
||
text-shadow: 0 1px 1px #000;\
|
||
-webkit-transition: opacity .25s ease-in-out;\
|
||
-moz-transition: opacity .25s ease-in-out;\
|
||
-o-transition: opacity .25s ease-in-out;\
|
||
transition: opacity .25s ease-in-out;\
|
||
vertical-align: top;\
|
||
}\
|
||
.thumbnail:hover, .thumbnail:focus {\
|
||
opacity: .9;\
|
||
}\
|
||
.thumbnail#selected {\
|
||
opacity: 1;\
|
||
}\
|
||
.thumbnail::before {\
|
||
counter-increment: thumbnails;\
|
||
content: counter(thumbnails);\
|
||
color: #FFF;\
|
||
font-weight: 700;\
|
||
padding: 3px;\
|
||
position: absolute;\
|
||
top: 0;\
|
||
right: 0;\
|
||
text-shadow: 0 0 3px #000, 0 0 8px #000;\
|
||
}\
|
||
.thumbnail.drag {\
|
||
box-shadow: 0 0 10px rgba(0,0,0,.5);\
|
||
}\
|
||
.thumbnail.over {\
|
||
border-color: #FFF;\
|
||
}\
|
||
.thumbnail > span {\
|
||
color: #FFF;\
|
||
}\
|
||
.remove {\
|
||
background: none;\
|
||
color: #E00;\
|
||
font-weight: 700;\
|
||
padding: 3px;\
|
||
}\
|
||
.remove:hover::after {\
|
||
content: " Remove";\
|
||
}\
|
||
.thumbnail > label {\
|
||
background: rgba(0,0,0,.5);\
|
||
color: #FFF;\
|
||
right: 0; bottom: 0; left: 0;\
|
||
position: absolute;\
|
||
text-align: center;\
|
||
}\
|
||
.thumbnail > label > input {\
|
||
margin: 0;\
|
||
}\
|
||
#addReply {\
|
||
color: #333;\
|
||
font-size: 3.5em;\
|
||
line-height: 100px;\
|
||
}\
|
||
#addReply:hover, #addReply:focus {\
|
||
color: #000;\
|
||
}\
|
||
.field {\
|
||
border: 1px solid #CCC;\
|
||
box-sizing: border-box;\
|
||
-moz-box-sizing: border-box;\
|
||
color: #333;\
|
||
font: 13px sans-serif;\
|
||
margin: 0;\
|
||
padding: 2px 4px 3px;\
|
||
-webkit-transition: color .25s, border .25s;\
|
||
-moz-transition: color .25s, border .25s;\
|
||
-o-transition: color .25s, border .25s;\
|
||
transition: color .25s, border .25s;\
|
||
}\
|
||
.field:-moz-placeholder,\
|
||
.field:hover:-moz-placeholder {\
|
||
color: #AAA;\
|
||
}\
|
||
.field:hover, .field:focus {\
|
||
border-color: #999;\
|
||
color: #000;\
|
||
outline: none;\
|
||
}\
|
||
#qr > form > div:first-child > .field:not(#dump) {\
|
||
width: 30%;\
|
||
}\
|
||
#qr textarea.field {\
|
||
display: -webkit-box;\
|
||
min-height: 120px;\
|
||
min-width: 100%;\
|
||
}\
|
||
.textarea {\
|
||
position: relative;\
|
||
}\
|
||
#charCount {\
|
||
color: #000;\
|
||
background: hsla(0, 0%, 100%, .5);\
|
||
position: absolute;\
|
||
top: 100%;\
|
||
right: 0;\
|
||
}\
|
||
#charCount.warning {\
|
||
color: red;\
|
||
}\
|
||
.captchainput > .field {\
|
||
min-width: 100%;\
|
||
}\
|
||
.captchaimg {\
|
||
background: #FFF;\
|
||
outline: 1px solid #CCC;\
|
||
outline-offset: -1px;\
|
||
text-align: center;\
|
||
}\
|
||
.captchaimg > img {\
|
||
display: block;\
|
||
height: 57px;\
|
||
width: 300px;\
|
||
}\
|
||
#qr [type=file] {\
|
||
margin: 1px 0;\
|
||
width: 70%;\
|
||
}\
|
||
#qr [type=submit] {\
|
||
margin: 1px 0;\
|
||
padding: 1px; /* not Gecko */\
|
||
padding: 0 -moz-calc(1px); /* Gecko does not respect box-sizing: border-box */\
|
||
width: 30%;\
|
||
}\
|
||
\
|
||
.fileText:hover .fntrunc,\
|
||
.fileText:not(:hover) .fnfull {\
|
||
display: none;\
|
||
}\
|
||
.fitwidth img[data-md5] + img {\
|
||
max-width: 100%;\
|
||
}\
|
||
.gecko .fitwidth img[data-md5] + img,\
|
||
.presto .fitwidth img[data-md5] + img {\
|
||
width: 100%;\
|
||
}\
|
||
\
|
||
#qr, #qp, #updater, #stats, #ihover, #overlay, #navlinks {\
|
||
position: fixed;\
|
||
}\
|
||
\
|
||
#ihover {\
|
||
max-height: 97%;\
|
||
max-width: 75%;\
|
||
padding-bottom: 18px;\
|
||
}\
|
||
\
|
||
#navlinks {\
|
||
font-size: 16px;\
|
||
top: 25px;\
|
||
right: 5px;\
|
||
}\
|
||
\
|
||
body {\
|
||
box-sizing: border-box;\
|
||
-moz-box-sizing: border-box;\
|
||
}\
|
||
body.unscroll {\
|
||
overflow: hidden;\
|
||
}\
|
||
#overlay {\
|
||
top: 0;\
|
||
right: 0;\
|
||
left: 0;\
|
||
bottom: 0;\
|
||
text-align: center;\
|
||
background: rgba(0,0,0,.5);\
|
||
z-index: 1;\
|
||
}\
|
||
#overlay::after {\
|
||
content: "";\
|
||
display: inline-block;\
|
||
height: 100%;\
|
||
vertical-align: middle;\
|
||
}\
|
||
#options {\
|
||
display: inline-block;\
|
||
padding: 5px;\
|
||
text-align: left;\
|
||
vertical-align: middle;\
|
||
width: 600px;\
|
||
}\
|
||
#credits {\
|
||
float: right;\
|
||
}\
|
||
#options ul {\
|
||
padding: 0;\
|
||
}\
|
||
#options article li {\
|
||
margin: 10px 0 10px 2em;\
|
||
}\
|
||
#options code {\
|
||
background: hsla(0, 0%, 100%, .5);\
|
||
color: #000;\
|
||
padding: 0 1px;\
|
||
}\
|
||
#options label {\
|
||
text-decoration: underline;\
|
||
}\
|
||
#content {\
|
||
height: 450px;\
|
||
overflow: auto;\
|
||
}\
|
||
#content textarea {\
|
||
font-family: monospace;\
|
||
min-height: 350px;\
|
||
resize: vertical;\
|
||
width: 100%;\
|
||
}\
|
||
\
|
||
#updater {\
|
||
text-align: right;\
|
||
}\
|
||
#updater:not(:hover) {\
|
||
border: none;\
|
||
background: transparent;\
|
||
}\
|
||
.new {\
|
||
background: lime;\
|
||
}\
|
||
\
|
||
#watcher {\
|
||
padding-bottom: 5px;\
|
||
position: absolute;\
|
||
overflow: hidden;\
|
||
white-space: nowrap;\
|
||
}\
|
||
#watcher:not(:hover) {\
|
||
max-height: 220px;\
|
||
}\
|
||
#watcher > div {\
|
||
max-width: 200px;\
|
||
overflow: hidden;\
|
||
padding-left: 5px;\
|
||
padding-right: 5px;\
|
||
text-overflow: ellipsis;\
|
||
}\
|
||
#watcher > .move {\
|
||
padding-top: 5px;\
|
||
text-decoration: underline;\
|
||
}\
|
||
\
|
||
#qp {\
|
||
padding: 2px 2px 5px;\
|
||
}\
|
||
#qp .post {\
|
||
border: none;\
|
||
margin: 0;\
|
||
padding: 0;\
|
||
}\
|
||
#qp img {\
|
||
max-height: 300px;\
|
||
max-width: 500px;\
|
||
}\
|
||
.qphl {\
|
||
outline: 2px solid rgba(216, 94, 49, .7);\
|
||
}\
|
||
.inlined {\
|
||
opacity: .5;\
|
||
}\
|
||
.inline {\
|
||
background-color: rgba(255, 255, 255, 0.15);\
|
||
border: 1px solid rgba(128, 128, 128, 0.5);\
|
||
display: table;\
|
||
margin: 2px;\
|
||
padding: 2px;\
|
||
}\
|
||
.inline .post {\
|
||
background: none;\
|
||
border: none;\
|
||
margin: 0;\
|
||
padding: 0;\
|
||
}\
|
||
div.opContainer {\
|
||
display: block !important;\
|
||
}\
|
||
.opContainer.filter_highlight {\
|
||
box-shadow: inset 5px 0 rgba(255,0,0,0.5);\
|
||
}\
|
||
.filter_highlight > .reply {\
|
||
box-shadow: -5px 0 rgba(255,0,0,0.5);\
|
||
}\
|
||
.filtered {\
|
||
text-decoration: underline line-through;\
|
||
}\
|
||
.quotelink.forwardlink,\
|
||
.backlink.forwardlink {\
|
||
text-decoration: none;\
|
||
border-bottom: 1px dashed;\
|
||
}\
|
||
'
|
||
};
|
||
|
||
Main.init();
|
||
|
||
}).call(this);
|