// ==UserScript==
// @name FastTech Forum Enhancements
// @namespace ftil
// @description Improvements for the FastTech forums and store
// @include https://*.fasttech.com/*
// @version 2.5.1
// @grant GM_getValue
// @grant GM_setValue
// @run-at document-start
// @icon 
// ==/UserScript==
'use strict';
// Delay function calls until their preconditions have been met.
var block = (function() {
var counters = { };
return {
get: function(name) { return counters[name].val !== 0; },
block: function(name) {
var c = counters[name];
if (c === undefined)
counters[name] = { val: 1, watches: [] };
else if (c.val !== 0)
c.val++;
},
unblock: function(name) {
var c = counters[name];
if (c.val !== 0)
c.val--;
if (c.val === 0) {
var w = c.watches;
while (w.length !== 0)
w.shift()();
}
},
watch: function(name, func) {
var c = counters[name];
if (c.val !== 0)
c.watches.push(func);
else
func();
},
};
})();
/* Userscript settings:
* This is just terrible. GM_[sg]etValue is used to initialize settings since
* it's shared across subdomains, while localStorage is used to notify other
* tabs of new settings.
*/
var settings = (function() {
var settings = { };
var watches = { };
var prefix = 'ignore_list:';
var prere = new RegExp('^' + prefix + '(.+)');
var have_ls = (function() {
try {
localStorage.getItem('test');
return true;
} catch (e) {
return false;
}
})();
// Unblock after execution of the script
block.block('ready');
function inner_set(name, value) {
// Ugh.
if (JSON.stringify(settings[name].value) === JSON.stringify(value))
return false;
settings[name].value = value;
if (name in watches) {
var w = watches[name];
for (var i = 0; i < w.length; i++)
w[i](name, value);
}
return true;
}
if (have_ls) {
window.addEventListener('storage', function(e) {
var set = e.key.match(prere);
if (set !== null && set[1] in settings)
inner_set(set[1], JSON.parse(e.newValue));
});
}
function get_sync(name) {
var val = GM_getValue(name, null);
if (val !== null)
return JSON.parse(val);
else
return settings[name].def;
}
return {
get: function(name) { return settings[name].value; },
get_sync: get_sync,
set: function(name, value) {
if (inner_set(name, value)) {
var s = JSON.stringify(value);
if (have_ls)
localStorage.setItem(prefix + name, s);
GM_setValue(name, s);
}
},
all: function() { return settings; },
register: function(name, desc, def, dep, opt) {
settings[name] = {
desc: desc,
def: def,
dep: dep,
opt: opt,
};
inner_set(name, get_sync(name));
},
watch: function(name, func) {
if (!(name in watches))
watches[name] = [];
watches[name].push(func);
if (settings[name] !== undefined)
func(name, settings[name].value);
},
update: function() { block.unblock('ready'); },
};
})();
// Dynamic stylesheets
var styles = (function() {
var urls = {};
var txts = {};
var elms = {};
var active_elm;
function try_gmgrt(name) {
try {
return GM_getResourceText('theme_' + name);
} catch (e) {}
}
function apply() {
var enable = settings.get('use_theme');
var name = settings.get('theme_name');
if (enable === true && name !== undefined) {
if (active_elm !== undefined) {
if (active_elm === elms[name])
return;
active_elm.disabled = true;
}
if (name in elms) {
elms[name].disabled = false;
} else {
var e, u;
var c = try_gmgrt(name);
if (c === undefined || c === null) {
u = urls[name];
c = txts[name];
}
if (c !== undefined) {
e = document.createElement('style');
e.type = 'text/css';
e.innerHTML = c;
} else {
e = document.createElement('link');
e.rel = 'stylesheet';
e.type = 'text/css';
e.href = u;
}
document.head.appendChild(e);
elms[name] = e;
}
active_elm = elms[name];
} else if (active_elm !== undefined) {
active_elm.disabled = true;
active_elm = undefined;
}
}
return {
register: function(name, url, txt) {
if (url !== undefined)
urls[name] = url;
if (txt !== undefined)
txts[name] = txt;
},
update: function() {
settings.watch('use_theme', apply);
settings.watch('theme_name', apply);
},
};
})();
// Sanitize external values that may be fed into an innerHTML
var strip_tags = (function() {
var div = document.createElement('div');
return function(t) {
div.innerHTML = t;
return div.textContent;
};
})();
// Case-insensitive indexOf (needed in case of case errors in user input)
function ci_indexof(a, n) {
var l = n.toLowerCase();
for (var i = 0; i < a.length; i++) {
if (a[i].toLowerCase() === l)
return i;
}
return -1;
}
// Case-insensitive attribute lookup
function ci_lookup(a, n) {
var l = n.toLowerCase();
for (var k in a) {
if (k.toLowerCase() === l)
return k;
}
}
// Assorted generic handlers for user input.
var handlers = (function() {
function Handler(funcs, elm, data, extra) {
this.elm = elm;
this.data = data;
this.extra = extra;
if (funcs.setting !== undefined)
settings.watch(data, funcs.setting.bind(this));
for (var k in funcs) {
if (k.slice(0, 2) === 'on')
elm.addEventListener(k.slice(2), funcs[k].bind(this));
}
}
function handler(fs) {
return function(e, d, x) { return new Handler(fs, e, d, x); };
}
return {
generic: handler,
ignore_toggle: handler({
onclick: function() { ignore_list.set(this.data); }
}),
checkbox: handler({
onchange: function() { settings.set(this.data, this.elm.checked); },
setting: function(n, v) { this.elm.checked = v; }
}),
top_click: handler({
onclick: function() { scrollTo(0, 0); }
}),
hide_toggle: handler({
onclick: function() { unhide_posts.set(this.data); }
}),
};
})();
// Manage the state of the ignore list
function create_ignore_list(list_var) {
// Use two stages so that filters can handle updates before being called
var watches = { 'early': [], 'late': [] };
var il = [];
function fire_watches(user, state, idx) {
['early', 'late'].forEach(function(stage) {
var w = watches[stage];
for (var i = 0; i < w.length; i++)
w[i](user, state, idx);
});
}
function inner_set(user, state, idx) {
// In order to support userscripts, we need to rebuild the list on every set
if (settings.get_sync) {
il = [];
var l = settings.get_sync(list_var);
for (var i = 0; i < l.length; i++)
il.push(l[i]);
}
if (state)
il.push(user);
else
il.splice(idx, 1);
settings.set(list_var, il);
fire_watches(user, state, idx);
}
settings.register(list_var, null, [], null);
// Watch for ignorelist changes, and fire incremental watch events
settings.watch(list_var, function(name, val) {
var i, j;
if (il.length === val.length)
return;
if (il.length === 0) {
for (i = 0; i < val.length; i++)
il.push(val[i]);
fire_watches(il, true);
return;
}
i = 0;
j = 0;
while (true) {
if (il[i] === undefined && val[j] === undefined)
break;
if (il[i] !== val[j]) {
var user, state, idx;
if (il[i] === undefined) {
// New entry @ end of list: user added
user = val[j];
il.push(user);
state = true;
idx = i;
} else {
// Missing entry inside list: user removed
user = il[i];
il.splice(i, 1);
state = false;
idx = i;
}
fire_watches(user, state, idx);
continue;
}
i++;
j++;
}
});
return {
// Get ignored state of user (true == ignored)
get: function(user) { return (ci_indexof(il, user) !== -1); },
// Set ignored state of user (undefined == toggle)
set: function(user, state) {
var idx = ci_indexof(il, user);
if (state === undefined)
state = (idx === -1);
if ((state && (idx === -1)) || (!state && (idx !== -1)))
inner_set(user, state, idx);
},
/* Watch the ignorelist for changes (stage == ('early'|'late')).
* func may be called with a single user and a new state, or an array of
* users (with implicit ignored state for all)
*/
watch: function(stage, func) {
watches[stage].push(func);
if (il !== [])
func(il, true, -1);
},
};
}
/* Core post filtering logic
* post vars are expected to be in the format
* { user: string, content_elm: element, post_elms: [element] }
*/
var post_filter = (function() {
var posts;
var watches = [];
var filters;
settings.register('hide_ignored', 'Hide posts by ignored users', true, null);
block.block('filter');
function update() {
if (posts === undefined)
return;
for (var i = 0; i < posts.length; i++) {
var post = posts[i];
var hide = false;
var j;
for (j = 0; j < filters.length; j++) {
var tmp = filters[j](post);
if (tmp === true) {
hide = true;
} else if (tmp === false) {
hide = false;
break;
}
}
var display = hide ? 'none' : '';
for (j = 0; j < post.post_elms.length; j++)
post.post_elms[j].style.display = display;
for (j = 0; j < watches.length; j++)
watches[j](post, hide);
}
block.unblock('filter');
}
function builtin() {
filters = [];
filters.unshift((function() {
settings.watch('hide_ignored', post_filter.update);
ignore_list.watch('late', post_filter.update);
return function(post) {
if (settings.get('hide_ignored') && ignore_list.get(post.user))
return true;
};
})());
unhide_posts = (function() {
var posts = [];
filters.unshift(function(post) {
if (posts.indexOf(post) !== -1)
return false;
});
return {
// Set the unhide state of a post (true == unhide, undefined == toggle)
set: function(post, state) {
var idx = posts.indexOf(post);
if (state === undefined)
state = (idx === -1);
if (state && idx === -1)
posts.push(post);
else if (!state && idx !== -1)
posts.splice(idx, 1);
else
return;
post_filter.update();
}
};
})();
}
return {
// Register a list of posts to be filtered
register: function(list) {
posts = list;
update();
},
/* Add a new filter. Filters return true (should hide), false (must not
* hide), or undefined (no judgement). Does not trigger update().
*/
filter: function(func) {
if (filters === undefined)
builtin();
filters.push(func);
},
// Run all filters again
update: update,
/* Call func with changes to the hidden status of each post (true ==
* hidden). Does not trigger update().
*/
watch: function(func) { watches.push(func); },
};
})();
var unhide_posts;
// Generic buttons to toggle ignored state of a user
var ignore_buttons = (function() {
var posts;
var inner_ign = '';
var inner_unign = '';
function update_btns(users, state) {
var i;
if (users instanceof Array) {
for (i = 0; i < posts.length; i++) {
posts[i].ignbtn.innerHTML =
ci_indexof(users, posts[i].user) !== -1 ? inner_unign : inner_ign;
}
} else {
var ul = users.toLowerCase();
for (i = 0; i < posts.length; i++) {
if (!ul.localeCompare(posts[i].user.toLowerCase()))
posts[i].ignbtn.innerHTML = state ? inner_unign : inner_ign;
}
}
}
return {
// Set the (HTML) format of the (un)ignore buttons
set: function(ign, unign) {
inner_ign = ign;
inner_unign = unign;
},
// Register posts to have their ignore buttons handled
register: function(p) {
posts = p;
for (var i = 0; i < p.length; i++)
handlers.ignore_toggle(p[i].ignbtn, p[i].user);
ignore_list.watch('late', update_btns);
},
};
})();
// Handle RES-style tagging of users
var user_tags = (function() {
var posts;
function update_badges(n, val) {
for (var i = 0; i < posts.length; i++) {
var post = posts[i];
var k = ci_lookup(val, post.user);
post.badge.innerHTML = k !== undefined ? val[k] : post.saved_badge;
}
}
var tag_prompt = handlers.generic({ onclick: function(e) {
e.preventDefault();
e.stopPropagation();
var user = this.data;
var tmp =
(settings.get_sync ? settings.get_sync : settings.get)('user_tags');
var def = '';
if (tmp[user] !== undefined)
def = tmp[user];
var tag = prompt('Tag for ' + user + ' or blank for default:', def);
if (tag === null)
return;
tag = strip_tags(tag);
// Ugh.
var tags = { };
for (var u in tmp)
tags[u] = tmp[u];
if (tags[user] !== undefined && tag === '')
delete tags[user];
else if (tag !== '')
tags[user] = tag;
settings.set('user_tags', tags);
}});
return {
// Register posts to have their badges handled
register: function(p) {
posts = p;
for (var i = 0; i < p.length; i++) {
var post = p[i];
post.saved_badge = post.badge.innerHTML;
tag_prompt(post.badge, post.user);
}
settings.watch('user_tags', update_badges);
},
update: function() {
settings.register('user_tags', null, { }, null);
},
};
})();
// Track the last viewed page and post count for threads
var last_viewed = (function() {
var db;
var store = 'thread_data';
function GetCmd(args) { this.args = args; }
GetCmd.prototype.mode = 'readonly';
GetCmd.prototype.success = function(e) {
var t = e.target.result;
if (t !== undefined)
this.args.f(this.args.e, t.lvp, t.lpn, t.lp);
};
function SetCmd(args) { this.args = args; }
SetCmd.prototype.mode = 'readwrite';
SetCmd.prototype.success = function(e) {
var t = e.target.result;
if (t === undefined)
t = {id: this.args.id};
t.lvt = this.args.lvt;
t.lvp = this.args.lvp;
if (t.lpn === undefined || t.lpn < this.args.lpn) {
t.lpn = this.args.lpn;
t.lp = this.args.lp;
}
this.os.put(t);
};
GetCmd.prototype.exec = SetCmd.prototype.exec = function() {
this.os = db.transaction(store, this.mode).objectStore(store);
this.os.get(this.args.id).onsuccess = this.success.bind(this);
};
// Accumulate commands and fire them once idb is open
var cmds = [];
function run_cmds() {
if (db === undefined || db === null)
return open_db();
var cmd;
while ((cmd = cmds.shift()) !== undefined)
cmd.exec();
}
// Open idb and maybe remove old threads
function open_db() {
if (db === undefined)
db = null;
else
return;
var req = indexedDB.open('ignore_list', 2);
req.onupgradeneeded = function(e) {
var tmp = e.target.result;
if (!tmp.objectStoreNames.contains(store)) {
var os = tmp.createObjectStore(store, { keyPath: 'id' });
os.createIndex('lvt', 'lvt');
}
};
req.onsuccess = function(e) {
db = e.target.result;
run_cmds();
/* Periodically scrub the database (FIXME this is awful & may run in
* multiple tabs simultaneously)
*/
settings.register('lv_scrub_time', null, 0, null);
settings.watch('lv_scrub_time', function(n, v) {
var now = Date.now();
if (v + 24 * 60 * 60 * 1000 > now)
return;
var tr = db.transaction(store, 'readwrite');
var os = tr.objectStore(store);
os.index('lvt')
.openCursor(IDBKeyRange.upperBound(now - 60 * 24 * 60 * 60 * 1000))
.onsuccess = function(e) {
var c = e.target.result;
if (c) {
os.delete(c.primaryKey);
c.continue();
} else {
settings.set('lv_scrub_time', now);
}
};
});
};
}
return {
/* Record a visit to thread_id, on page last_viewed_page, with up to
* last_post_nr (of id last_post_id) visible. Should be called on every
* (thread) pageview.
*/
set: function(thread_id, last_viewed_page, last_post_nr, last_post_id) {
cmds.push(new SetCmd({id: thread_id, lvp: last_viewed_page,
lpn: last_post_nr, lp: last_post_id}));
run_cmds();
},
// Call func with the view history of thread_id (and pass extra as well)
get: function(thread_id, func, extra) {
cmds.push(new GetCmd({id: thread_id, f: func, e: extra}));
run_cmds();
},
update: open_db,
};
})();
// Generic settings menu
var settings_menu = (function() {
function mangle_opts(opts) {
var ret = '';
for (var k in opts)
ret = ret + '<option value="' + k + '">' + opts[k].name + '</option>';
return ret;
}
var watch_disable = handlers.generic({ setting: function(n, v) {
this.elm.disabled = v ^ this.extra;
}});
var select = handlers.generic({
onchange: function() { settings.set(this.data, this.elm.value); },
setting: function(n, v) { this.elm.value = v; }
});
return {
/* Attach a settings menu to elm. Menus is a list of submenus, in the form
* [ { title: 'str' | undefined, vars: [ "name" ] } ]
*/
register: function(elm, menus) {
var s = settings.all();
var d = document.createElement('div');
var html = '';
var vs = [];
for (var i = 0; i < menus.length; i++) {
var menu = menus[i];
if (i !== 0)
html += '<hr></hr>';
if (menu.title !== undefined)
html += '<b>' + menu.title + '</b>';
for (var j = 0; j < menu.vars.length; j++) {
var v = menu.vars[j];
vs.push(v);
var sv = s[v];
if (sv.opt !== undefined) {
html += '<div><select style="margin-left: 21px">' +
mangle_opts(sv.opt) + '</select></div>';
} else {
html += '<div><input type="checkbox" id="ilcb_' + v + '"></input>' +
'<label for="ilcb_' + v + '">' + sv.desc + '</label></div>';
}
}
}
d.innerHTML = html;
var e = d.firstElementChild;
var depre = /(!)?(.*)/;
function h(e, v) {
var sv = s[v];
if (sv.opt !== undefined)
select(e, v);
else
handlers.checkbox(e, v);
var d = sv.dep;
if (d) {
var ds = sv.dep.match(depre);
var i = !ds[1];
d = ds[2];
watch_disable(e, d, i);
}
}
while (e !== null) {
if (e.tagName === 'DIV')
h(e.firstElementChild, vs.shift());
e = e.nextElementSibling;
}
elm.appendChild(d);
}
};
})();
// Display a nice menu for managing the ignorelist
var manage_menu = (function() {
var menu_list;
var menu_elms = [];
var template = document.createElement('tr');
template.innerHTML =
'<td></td><td align="right"><a class="itmremove"></a></td>';
function create_elm(user) {
var tr = template.cloneNode(true);
tr.firstElementChild.innerHTML = user;
var a = tr.lastElementChild.firstElementChild;
a.href = 'javascript:void(0)';
handlers.ignore_toggle(a, user);
if (this.appendChild)
this.appendChild(tr);
return { user: user.toLowerCase(), elm: tr };
}
function lc_comp(a, b) {
return a.toLowerCase().localeCompare(b.toLowerCase());
}
function update(user, state) {
if (menu_list === undefined)
return;
if (user instanceof Array) {
var frag = document.createDocumentFragment();
menu_elms = user.sort(lc_comp).map(create_elm, frag);
menu_list.appendChild(frag);
} else {
var lc = user.toLowerCase();
var idx;
if (state) {
for (idx = 0; idx < menu_elms.length; idx++) {
if (menu_elms[idx].user.localeCompare(lc) >= 0)
break;
}
var elm = create_elm.call(0, user);
menu_elms.splice(idx, 0, elm);
if (idx + 1 < menu_elms.length) {
menu_list.insertBefore(menu_elms[idx].elm, menu_elms[idx + 1].elm);
} else {
menu_list.appendChild(menu_elms[idx].elm);
}
} else {
for (idx = 0; idx < menu_elms.length; idx++) {
if (menu_elms[idx].user.localeCompare(lc) >= 0)
break;
}
if (idx < menu_elms.length) {
menu_elms[idx].elm.remove();
menu_elms.splice(idx, 1);
}
}
}
}
var addbox_enter = handlers.generic({
onkeydown: function(k) {
if (k.which !== 13)
return true;
k.stopPropagation();
k.preventDefault();
ignore_list.set(strip_tags(this.elm.value), true);
this.elm.value = '';
return false;
}
});
var addbox_click = handlers.generic({
onclick: function() {
ignore_list.set(strip_tags(this.data.value), true);
this.data.value = '';
}
});
return {
// Attach management elements to elm
register: function(elm) {
var tbl = document.createElement('table');
tbl.className = 'itmanage';
tbl.innerHTML = '<tbody><tr><td colspan="2">' +
'<input type="text" placeholder="Add user..." width="auto">' +
'</input> <div class="il_addbtn">Add</div></td></tr></tbody>';
menu_list = tbl.firstElementChild;
var td = menu_list.firstElementChild.firstElementChild;
addbox_enter(td.firstElementChild);
addbox_click(td.lastElementChild, td.firstElementChild);
ignore_list.watch('late', update);
elm.appendChild(tbl);
},
};
})();
// Where are we?
// In the interest of sanity, ftl.thread is true for all thread-like pages
// (i.e. any page with one or more posts); ftl.threadlist is true for all
// threadlist-like pages (i.e. any page with a list of threads). The
// formatting for all such pages is similar enough to treat them equivalently.
var ftl = {
// Mobile pages
mobile: '^https?://m\\.',
// All forum pages
forums: '^https?://(m|www)\\.fasttech\\.com/forums',
// Thread-like pages (i.e. those with posts)
thread: '^https?://(m|www)\\.fasttech\\.com/forums/[^/]+/t/[0-9]+/.',
// Permalink pages
perma: '^https?://(m|www)\\.fasttech\\.com/forums' +
'/[^/]+/t/[0-9]+/[^/]+\\?[0-9]+$',
// Threadlist-like pages (i.e. those with thread links)
threadlist: '^https?://(m|www)\\.fasttech\\.com/forums' +
'(/[^/]+(/page/[0-9]+)?(/search\\?.*)?)?$',
// Lists of threads for a single author
author: '^https?://(m|www)\\.fasttech\\.com/forums/author/',
// Lists of threads for a single product
product: '^https?://(m|www)\\.fasttech\\.com/p(roducts?)?/',
// New thread page for a product
pnthread: '^https?://(m|www)\\.fasttech\\.com/forums/[0-9]+/post',
// Forum settings page
settings: '^https?://www\\.fasttech\\.com/forums/settings$',
// New arrivals pages
new_arrivals: '^https?://(m|www)\\.fasttech\\.com/category/1/new-products',
// Review pages
reviews: '^https?://(m|www)\\.fasttech\\.com/reviews',
// Product search
prsearch: '^https?://(m|www)\\.fasttech\\.com/search',
// Product category
prcategory: '^https?://(m|www)\\.fasttech\\.com/c(ategory)?(/|$)',
};
for (var k in ftl)
ftl[k] = new RegExp(ftl[k]).test(location.href);
// ftl.editor matches all pages with a post editor
ftl.editor = ftl.thread ||
(ftl.threadlist && !ftl.settings && !ftl.author) ||
ftl.pnthread;
// ftl.threadlist matches all pages listing threads
ftl.threadlist = ftl.threadlist || ftl.author || ftl.product || ftl.settings;
// ftl.forums matches all pages with forum elements (threads or posts)
ftl.forums = ftl.forums || ftl.product;
// ftl.reviews matches all pages with reviews
ftl.reviews = ftl.reviews || ftl.product;
if (ftl.thread || ftl.settings) {
// Menu configuration
var full_menus = [
{ title: 'Browsing', vars: ['use_theme', 'theme_name', 'track_viewed',
'track_posts', 'always_unread', 'hide_threads'] },
{ title: 'Posting', vars: ['quote_quotes', 'quote_imgs'] },
{ title: 'Thread Menu', vars: ['inthread_menu', 'inthread_manage'] },
{ title: 'Post Hiding', vars: ['placeholders', 'hide_ignored',
'hide_quotes', 'hide_deleted'] },
];
var thread_menu;
if (true) {
thread_menu = [
{ title: 'General', vars: ['use_theme', 'quote_quotes', 'quote_imgs'] },
{ title: 'Post Hiding',
vars: ['hide_ignored', 'hide_quotes', 'hide_deleted'] },
];
} else {
thread_menu = full_menus;
}
settings.register('inthread_menu', 'Show menu in thread toolbar', true, null);
settings.register('inthread_manage', 'Manage ignored users from the menu',
false, 'inthread_menu');
settings.register('placeholders', 'Show placeholders for hidden posts', true,
null);
settings.register('hide_quotes', 'Hide posts quoting ignored users', false,
'hide_ignored');
settings.register('hide_deleted', 'Hide deleted posts', true, null);
settings.register('quote_quotes', 'Include quotes in quotes', true, null);
settings.register('quote_imgs', 'Include images in quotes', true, null);
}
if (ftl.threadlist) {
settings.register('track_viewed', 'Link to last viewed page', false, null);
settings.register('track_posts', 'Link to unread posts', true, null);
settings.register('always_unread', 'Always show unread posts button', false,
'track_posts');
settings.register('hide_threads', 'Hide threads by ignored users', false,
null);
}
if (ftl.editor) {
settings.register('agree_tou', null, false, null);
}
block.block('ready');
if (ftl.forums) {
var ignore_list = create_ignore_list('ignorelist');
last_viewed.update();
if (ftl.thread || ftl.reviews)
user_tags.update();
block.block('late');
block.watch('ready', function() {
setTimeout(function() { block.unblock('late'); }, 0);
});
} else if (ftl.new_arrivals) {
var ignore_categories = create_ignore_list('ignored_categories');
} else if (ftl.reviews) {
user_tags.update();
}
settings.update();
if (document.readyState === 'loading') {
document.addEventListener('readystatechange', function() {
if (document.readyState === 'interactive') block.unblock('ready');
});
} else {
block.unblock('ready');
}
/* Fix up display of certain post elements:
* - Fix align tags (which should have been 100% divs to begin with)
* - Avoid breaking SKU links across lines
*/
var style = document.createElement('style');
style.innerHTML =
'.PostContent span[style*="text-align:"] {display:block;width:100%}\n' +
'a.SKUAutoLink {display: inline-block}\n' +
'.PopoutPanel>ul {margin: -5px -20px 0px -50px!important}\n' +
'.PopoutPanel>p {margin: 0px!important}\n' +
'.Badges {cursor:pointer}\n' +
'.itmanage tbody tr td {padding:0px}\n' +
'.itmanage .itmremove {padding-left: 16px; height: 16px; background: ' +
'url(https://www.fasttech.com/images/minus-button.png) no-repeat}';
document.head.appendChild(style);
// RE to pull a thread id from a URL
if (ftl.forums)
var thread_id_re = new RegExp('/forums/[^/]+/t/([0-9]+)/[^?]*$');
if (ftl.thread) {
// Add a filter for posts quoting ignored users
post_filter.filter((function() {
var quote_res = [];
function new_qre(user) {
quote_res.push(
new RegExp('\\b' + user + ' wrote(</a>)?:?[\\r\\n]*<br>', 'i'));
}
function update(user, state, idx) {
if (user instanceof Array) {
quote_res = [];
for (var i = 0; i < user.length; i++)
new_qre(user[i]);
} else {
if (state)
new_qre(user);
else
quote_res.splice(idx, 1);
}
}
ignore_list.watch('early', update);
settings.watch('hide_quotes', post_filter.update);
return function(post) {
if (settings.get('hide_ignored') && settings.get('hide_quotes')) {
for (var i = 0; i < quote_res.length; i++) {
if (quote_res[i].test(post.content_elm.innerHTML))
return true;
}
}
};
})());
// Add a filter for posts that the admins have deleted
post_filter.filter((function() {
var deleted_re = /post deleted/i;
var edited_re = /^[ \r\n]*$/;
settings.watch('hide_deleted', post_filter.update);
return function(post) {
if (settings.get('hide_deleted')) {
var wb = post.content_elm.getElementsByClassName('WarningBox')[0];
if ((wb !== undefined && deleted_re.test(wb.innerHTML)) ||
edited_re.test(post.content_elm.innerHTML))
return true;
}
};
})());
// Get all post bodies, desktop and mobile
var posts;
block.watch('ready', function() {
posts = [];
var elms = document.getElementsByClassName('PostContent');
for (var i = 0; i < elms.length; i++) {
var pc = elms[i];
var user = pc.getAttribute('data-username');
if (user !== null)
posts.push({ user: user, content_elm: pc });
}
});
// Create a function to create placeholders as needed
var gen_watch_filter = function(ph_func) {
return function(post, state) {
var placeholders = settings.get('placeholders');
if (state && placeholders) {
if (!post.placeholder)
post.placeholder = ph_func(post);
post.placeholder.style.display = '';
} else {
if (post.placeholder !== undefined)
post.placeholder.style.display = 'none';
}
}
};
// Get this thread's ID
var thread_id = location.href.match(thread_id_re);
if (thread_id !== null && thread_id[1] !== undefined)
thread_id = parseInt(thread_id[1]);
else
thread_id = undefined;
}
// Get the current and last pages from a pager element, desktop and mobile
function parse_pager(elm) {
var cur, last, sel;
if (elm === undefined || elm === null) {
cur = last = 1;
} else if (!ftl.mobile) {
sel = elm.getElementsByClassName('PageLink_Selected')[0];
if (sel !== undefined)
cur = parseInt(sel.innerHTML);
last = parseInt(elm.lastElementChild.previousElementSibling.innerHTML);
} else {
sel = elm.getElementsByClassName('active')[0];
if (sel !== undefined)
cur = parseInt(sel.firstElementChild.innerHTML);
last = parseInt(elm.lastElementChild.firstElementChild.innerHTML);
}
return {
cur: cur,
last: last
};
}
if (!ftl.mobile) {
/* Jump-to-page boxes are broken site-wide. Fix them.
* The existing JumpToPage box could be fixed by changing "function (e)" to
* "function (event)" in the event listener, but instead, I'm doing all this.
*/
var fix_jtp = function() {
var jtps = document.getElementsByClassName('JumpToBox');
if (jtps.length === 0)
return;
var lastpg = parse_pager(jtps[0].parentElement).last;
function handle_kp(k) {
if (k.which === 13) {
k.preventDefault();
k.stopPropagation();
var pg = parseInt(this.value);
if (pg > 0 && pg <= lastpg)
location.href = base[1] + pg + base[2];
}
return false;
}
for (var i = 0; i < jtps.length; i++) {
var jtpbox = jtps[i];
var as = jtpbox.parentElement.getElementsByTagName('a');
if (as[0] !== undefined) {
var base = as[0].href.match(new RegExp('(.*/)[0-9]+($|[?#/].*)'));
if (base !== null)
jtpbox.addEventListener('keypress', handle_kp);
}
}
};
block.watch(ftl.forums ? 'late' : 'ready', fix_jtp);
}
if (ftl.threadlist) {
// Translate a post number into a page number, handling FT's off-by-one quirk
var postnr_to_pagenr = function(post, maxpage) {
return Math.floor(Math.min(maxpage, post / 10 + 1));
};
// Create a link to a specific page of a thread
var gen_page_link = function(base, page, hash) {
var ret = base;
if (page !== 1)
ret += '/' + page;
if (hash)
ret += '#' + hash;
return ret;
};
}
if (!ftl.mobile) {
// Re-implement FT's PopoutMenu, only nicer
var PM2 = (function() {
var active, timeout;
function show() {
var p = active.par;
var c = active.ch;
p.className += ' focused';
c.style.left = c.style.top = '0px';
c.style.display = 'inline';
var ppb = p.parentElement.getBoundingClientRect();
var pb = p.getBoundingClientRect();
var cb = c.getBoundingClientRect();
c.style.left = Math.min(pb.left - cb.left,
document.body.clientWidth - cb.width) + 'px';
c.style.top = (ppb.bottom - cb.top) + 'px';
}
function hide() {
if (active === undefined)
return;
active.par.className = active.cname;
active.ch.style.display = 'none';
active = undefined;
}
function min(pm) {
clearTimeout(timeout);
if (pm !== active)
hide();
active = pm;
show();
}
function mout() { timeout = setTimeout(hide, 300); }
function add_listeners(elem, pm) {
elem.addEventListener('mouseover', min.bind(null, pm));
elem.addEventListener('mouseout', mout);
}
/* Hack: we can't remove anonymous event listeners, so duplicate elements.
* We hide the existing element to serve as a decoy for the original
* PopoutMenu in case it hasn't run yet, then steal all its children.
*/
function dupe(id) {
var elm = document.getElementById(id);
var nelm = elm.cloneNode(false);
elm.style.display = 'none';
while (elm.childNodes[0])
nelm.appendChild(elm.childNodes[0]);
if (elm.nextElementSibling)
elm.parentElement.insertBefore(nelm, elm.nextElementSibling);
else
elm.parentElement.appendChild(nelm);
return nelm;
}
var def = {dupe: true};
return function(pm) {
for (var k in def) {
if (!(k in pm))
pm[k] = def[k];
}
try {
if (pm.dupe) {
pm.ch = dupe(pm.ch || (pm.par.replace('Button', '') + 'Popout'));
pm.par = dupe(pm.par);
} else {
pm.ch = document.getElementById(
pm.ch || (pm.par.replace('Button', '') + 'Popout'));
pm.par = document.getElementById(pm.par);
}
pm.cname = pm.par.className;
pm.ch.style.padding = '20px';
add_listeners(pm.par, pm);
add_listeners(pm.ch, pm);
} catch (e) {}
};
})();
block.watch('ready', function() {
['Cart', 'Support', 'Community', 'Account']
.forEach(function(n) { PM2({par: n + 'Button'}); });
if (ftl.thread) {
['RateThread', 'ForumTools']
.forEach(function(n) { PM2({par: n}); });
}
});
}
styles.register('nightmode', undefined,
'#SearchKeywords,#images ol,#searchBarCategory,button.close{background:0 ' +
'0!important}#images ol{border:0!important}.btn.active,li.active>a{backgr' +
'ound:#0d7bad!important}.FormButton.green{background:#20944b!important}#P' +
'roductDetails,#RightPanel,#categoryButton,#container .list-group li a,#f' +
'ooter,#searchBarOptionsCell,#searchBoxCell,#searchButtonCell,.AFEntry,.A' +
'ttachments,.BGShadow,.BGShadowLight,.OrdersShipmentHeading,.PageNavigati' +
'onalPanel,.ProductBackdrop,.ProductFilters,.ProductsGrid,.TicketResponse' +
',.TicketResponseFrame,.alert-info,.jumbotron,.list-group li,.mbbc-popups' +
' li,.mbbc-popups ul,.ui-dialog,body,td.Pros,ul.nav a{background:#222!imp' +
'ortant}.FormButton.blue{background:#3182b9!important}#AccountButton.focu' +
'sed,#LeftPanel,#UpdateWarningMsg,#container li a,#container li.active a.' +
'dropdown-toggle,#container ol,#navbar>tbody>tr:first-child,.CardShadow,.' +
'ControlArrows,.FilterEntry:hover,.FormButton.white,.ForumHeader,.ForumLi' +
'nk:hover,.GridItem,.PopoutPanel,.ProductGridItem,.ProductPanelBackdrop,.' +
'ReviewContent,.Steps>div,.btn,.dropdown-menu,.dropit-submenu>li,.mbbc-po' +
'pups li:hover,.navbarsearch,.panel-body,.panel-heading,.panel-title,.tab' +
'le,.table td,.tabs-min,button,div.well,input:not([valbbcode]),select,tex' +
'tarea,ul.dropdown-menu>li>a{background:#333!important}.FilterSelected,.F' +
'orumLink.Selected,.PageLink:hover,.StaticPageLink_Selected,.divider,.dro' +
'p-hover>a,.dropit-submenu,.subCategories li:hover,hr{background:#444!imp' +
'ortant}.FormButton.purple{background:#6a3b7f!important}.FormButton.red{b' +
'ackground:#9e1531!important}.FormButton.orange{background:#c4571d!import' +
'ant}.topLevelCategories li.focused{background-image:url(data:image/png;b' +
'ase64,iVBORw0KGgoAAAANSUhEUgAAAA8AAAAPCAYAAAA71pVKAAAAZklEQVR4AWPwySwkG1' +
'NPs2TjRjuyNUs0brwl0bRxj3jzei2yNEs2bfwPwkBD+sTb1wuQrhlhwCvJ5o25ZGhGYJA4KD' +
'zI0YwwpGF9LMmaQYEIsplkP0s0b4wnO7TpE88gTXRN22RjAIeVNYXGl+mUAAAAAElFTkSuQm' +
'CC)!important;background-position:right 2px!important;background-repeat:' +
'no-repeat!important}.mbbc-buttons li{background-image:url(' +
'YmS+kI/MzMzMzMzMzMzMzMzAxLuoqKiamppWV1cuLi7MzMzMzMzAx8qNjo5FRUaoRkZ8fH1N' +
'Tk/MzMzMzMyUPz55OjfMzMzMzMzMzMzMzMzMzMxWeJzJbgbGpxKCoqY8YIC5uqhzqsmwlV99' +
'pIB1lTxlghlvfMlxYbKnRy1sk7skfGsNbDRy3rXhAAAAHnRSTlMA/////42hdf//////s8L/' +
'///////W6///NiMRW0Sx+uQWAAAC5klEQVR4AezSRZLAMAxEUY3SQTkM8v0vOhxG7fNdXr9A' +
'09vbqo9VZIuZTVaARcGshVEUhfGNlaSZWDDAwWFsxgggylFcWSWzVGLD0lOMgKv3YqLagjVA' +
'61rnHH4OGguWsnxjnQn7ZY6xEP0FxmXdjZYFcz93i/XDzS+rK5NFzXqNTz8jM39bTkwWabNK' +
'N1gBnGz+2yJr2gD4YXCI5cChxVzVzo6NzDE2IDq0SMbNm7HfgwBYYBHQR31xsnkiqToz5hvF' +
'yCkaTw/ijEVau0VeFQr8Xqg+wkiqcfNGzOsi/wyjrvtip761hllBKAzvArGSTtcS8P5v8/x+' +
'OceRk3iaPd07+QetXXsGKaX0bzQLl4kolRRhWBBnVIRRmghTB+III4yRI4yTIUzriKKV+kKk' +
'lFJKiZ7ABZyZF6CFi+JSzlyX45lCjm3Y+1eGH6jNsMaLmJNiE1yr745/JXbBM+tigAVdWaWx' +
'Sz+4MvyADxJFyJWtPxpRnW/G+MNnhu/ixsyNRsFTk/whdqVdQSU8Jw27tS5+Zl13seE561IL' +
'Oy4mUpzF8JIuu/7faLCl+BvJI0dKKaUvMbe7xXHOhm3tPax2u13BcZXmw56m1LAZMY5jcmyF' +
'Fr7F1jOGLxikD/sti55Z+AIivVuLiPVeFHa7h28jbXy3DWcJbQ971twNuduGw6aMog+bUkop' +
'fYPI7U43Xcdj2LTVWbqExHQ4UKmExDoDKDQjYnUYgDYQERsMwAZHxBY5ACaPiFWagHeyiJhT' +
'sVpF1HHYEKDTcPRR8V8jzw4OG3IPKaX/vb/aqQcchmIAAMPzmjzGC2rz/rebbfvvAb7637eX' +
'rdfYCq13DwxjnBFCMkrpLtYBoJrVYQ/HOCFUTKuegGEphNL0XtjRMwO1ENQYfS/MWps55zLv' +
'/S5WjTdQG2PUV2JCmSedmZo0oEqBu2AhhCzGmKWUdjFu9CzDn4EtegIGxj/IeIzrPP7MqlVj' +
'7LEhtuoWbAQUelFEFIeMmgAAAABJRU5ErkJggg==)!important;border-radius:5px!im' +
'portant}.HourGlass{background:inherit!important}.Stars{background:url(da' +
'ta:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAA4AAAAOCAMAAAAolt3jAAAAXVBM' +
'VEVMaXH/rVz/s2f/jCv/uGz/vXD+iSf+jSz+kjD+xH3+iiX+jCr+s2P/uG/+wXf+iif+m0T+' +
'n0f+pE7+p1L+v3X+wHj+wn3/kjP/mD3/oEj/p1P/tGX/vnT/iCX/xX3uTskzAAAAF3RSTlMA' +
'/grK/i3p1yHXJX7WA3aZbPXD7o2sP+NiEhsAAABkSURBVHgBZcvFAcMwEEXBL2aFzXb/ZUYY' +
'fLdZQI0GfHaPnxo2Lt6iZls9bRDRbHwl17NGgjmOrGWeRmWBx1aVeEJ69GtSXrLyLkjdSY3S' +
'rWhUqF3mKWmXjQmM7Z1BOQ1Yx/DXE5S8Bqiv9nzRAAAAAElFTkSuQmCC)!important}.Gra' +
'yStars{background:url(' +
'AAAOCAYAAAAfSC3RAAAA4UlEQVR4AWLwySzEinfu3MkGwrjkcWo8efJk76lTp8pI0njmzBn7' +
'Y8eO/T948OB/IG2AXyPCJl5AZ3RthjAARAF4F0omZBWquKfBrcPdOmyEMELk8XFxL+L5T0+n' +
'04fgbDa7p5UcBk2Utt1u74fDwSE4nU6dwWCwkGW5JYpiw4coBRnwM444wtHtdh3TNB1FURye' +
'558I8NNeJpNAGgpDjuPaMH5fKBEgDwmCsKd+Iz0SSEEo0UKJqVMdjUZfAmFEvWWuIwzCiGXZ' +
'fEiA+sEB9IdWKkTTiBrfF+7xDkHisNbhAqOEltIAVAofAAAAAElFTkSuQmCC)!important}' +
'.FilterEntry.FilterCancellable{background:url(' +
'Rw0KGgoAAAANSUhEUgAAAAwAAAALCAYAAABLcGxfAAAASklEQVR4AWPwySxkuCUk9J8QBqkD' +
'YbBiYjGKBigAs3GJMTAwNDBgUYDOxtSAoQmhmCwNlDmJTE8TxggNiIgjrBimAYxBHEIYpA4A' +
'F8kQ7Ga9snUAAAAASUVORK5CYII=) 10px 3px no-repeat,#444!important;border-c' +
'olor:#0d7bad!important;box-shadow:0 2px 2px #111!important}input[src="/i' +
'mages/errorreport.png"]{background:url(' +
'AAAANSUhEUgAAABAAAAAQBAMAAADt3eJSAAAAMFBMVEX////////////////////////////' +
'///////////////////////////////////9Or7hAAAAAEHRSTlMBCBIcJS45RlVkcYOVqbz' +
'NHir32QAAAGNJREFUeAEUTAkBBEEIUhMIDeAazPbvdjgvn0xVAXkCGocCwKcNaH4/9Sn08ym' +
'Qae00DUBRAtbGIHJUDc1Y1DRuKRY25Y3/gOa8BwKQFe/ugcAHpnfv7p49e/bcPab/Z8HgPgC' +
'LAx2nigq2jAAAAABJRU5ErkJggg==)!important}.CellMode .GridItemPrice{backgr' +
'ound:url(' +
'1ElAAAAi0lEQVR4Ae3aSRXDIABAwZiIjgrAQkV0X46VUT1RFAOV0M3Gf3MYBR8IEKbXfvMhZ' +
'RG1Z4jasm7vj6kVlUswqlnaisqlFZX1Z25F5fmP2YrKHIxqlraiMovasvwjtqIyglHN0lZUR' +
'iAqolp+sVHqcKTB5QOuCXGhj19vfpLjOQsenuGJKB5zM0TtWb7rxpkaVlv9JQAAAABJRU5Er' +
'kJggg==) no-repeat!important;width:134px!important}.ForumHeader input[ty' +
'pe=image]{border:0!important;vertical-align:text-top!important}#AddedToC' +
'artPopupPanel,#container ol,.CardShadow,.ForumThread,.GridItem,.ProductG' +
'ridItem,.ProductPanelBackdrop,.TicketResponseFrame,.dropit-submenu,.mbbc' +
'-buttons li,.panel-body,.panel-heading,td.Pros{border:1px solid #444!imp' +
'ortant}#searchBarOptionsCell,#searchBoxCell,#searchButtonCell,.ForumThre' +
'ad tr td,.navbarsearch{border-bottom:1px solid #444!important}#Ignorelis' +
'tPopout tr td,.PostFlags ul li{border-bottom:none!important}.FilterEntry' +
':hover,.FilterSelected,.StaticPageLink_Selected{border-color:#0d7bad!imp' +
'ortant}#categoryButton,#container li a,#container ul,.ControlArrows,.For' +
'umQuote,.ForumsNavigation,.PageLink,.PageNavigationalPanel,.PostFlags,.S' +
'KUAutoLink,.Signature,.TightInformationBox,.list-group li,.mbbc-popups u' +
'l,body .FormButton[class],div,hr,td{border-color:#444!important}.btn,but' +
'ton,input:not([valbbcode]),select,textarea{border-color:#444!important;b' +
'order-style:solid!important}#searchBarOptionsCell{border-left:1px solid ' +
'#444!important}.ForumQuote{border-left:5px solid #0d7bad!important}.drop' +
'it-submenu>li,blockquote{border:none!important}#searchButtonCell{border-' +
'right:1px solid #444!important}.ForumLink.Selected,.ForumLink:hover{bord' +
'er-right:3px solid #0d7bad!important}#searchBarOptionsCell,#searchBoxCel' +
'l,#searchButtonCell{border-top:1px solid #444!important}#LeftPanel,.Popo' +
'utPanel{box-shadow:0 15px 30px #111!important}.ProductGridItem{box-shado' +
'w:0 3px 3px #111!important}.BGShadow,.BGShadowLight,.CardShadow,.FilterS' +
'elected,.ForumLink.Selected,.GridItem,.StaticPageLink_Selected,.subCateg' +
'ories li:hover{box-shadow:none!important}.SysMsgBanner{color:#000!import' +
'ant}#AccountButton a{color:#0d7bad!important}#StaticPageContent [style],' +
'#categoryButton a,#content_ProductName,#footer,.AFTitle,.ControlArrows a' +
',.FieldLabel,.FlagValue,.Gray,.GrayText,.GridItemDesc>div,.GridItemModel' +
's,.Name>a,.NewList,.PageLink,.Pager,.btn,.dropit a,.navmenubutton a,.pan' +
'el-default>.panel-heading,.panel-title,.subCategories a,.topLevelCategor' +
'ies a,a.SortOrder,body,button,input:not([valbbcode]),select,textarea,ul.' +
'dropdown-menu>li>a{color:#bbb!important}.mbbc-popups li{color:#ccc!impor' +
'tant}.btn.active,body .FormButton[class],ul.dropdown-menu>li>a:hover{col' +
'or:#fff!important}.GridItemPackSize{display:inline!important;float:none!' +
'important;margin:0 0 0 -22px!important;vertical-align:sub!important}#DMC' +
'A,.logo,.navseals,a[href$="/faq/money-back-guarantee"]>img{display:none!' +
'important}#categoryButton{margin:0 20px!important}#footer,.PageContentPa' +
'nel>div:last-child[style]{margin:0!important}#CartPopupClose{margin:-10p' +
'x -20px!important}.mbbc-row1{margin-bottom:1px!important}.ForumLink:hove' +
'r{margin-right:-3px!important}#Tab{margin-top:10px!important}#AddedToCar' +
'tPopupPanel{padding:0 10px!important;position:fixed!important;right:40px' +
'!important;top:40px!important}.menubuttons{padding:0 20px!important}#Ans' +
'wer{padding:0!important}#searchbar{padding-right:20px!important}#AddedTo' +
'CartPopupPanel,#categoryButton{width:auto!important}');
styles.register('darkft', undefined,
'.FieldFlag,.Pager .PageLink{font-size:9pt!important}.Pager .PageLink:hov' +
'er{background:#111!important;color:#fefefe!important}.FieldFlag{backgrou' +
'nd:#1980b0!important;font-weight:400!important;padding:3px 5px!important' +
'}.ForumLink.Selected{background:#222!important;border-right:3px solid #6' +
'66!important}.ForumHeader,.ForumThread .ThreadCommandBar,.logo,body,div#' +
'contents{background:#333!important}.logo{visibility:hidden!important}.Fo' +
'rumThread tr td{background:#363636!important}#AccountButton.focused,#Acc' +
'ountPopout,#AddedToCartPopupPanel,#CartPopout,#CartPopupPanel,#Community' +
'Popout,#LeftPanel,#SupportPopout,#categoryButton,#categoryPopout,#navbar' +
',#searchBarOptionsButton,.CellMode .ProductGridItem,.FilterEntry:hover,.' +
'FormButton.white,.FormButton.white:hover,.LargeFilterEntry:hover,.LineMo' +
'de .ProductGridItem,.ModalDialog,.OrdersShipmentHeading,.PageNavigationP' +
'anel,.ProductBackdrop,.ProductFilters,.ProductFilters .FilterCancellable' +
',.ProductFilters .FilterSelected,.ProductPanelBackdrop,.mbbc .mbbc-edito' +
'r,div#footer,hr,tr{background:#444!important}.HourGlass{background:#444f' +
'e5!important}.Cons .FieldFlag,.Pros .FieldFlag{background:#4c9ed9!import' +
'ant}#navbar .focused,.BGShadow,.BGShadowLight,.PopoutPanel,.Steps .focus' +
'ed{background:#666!important}.PopoutPanel{border-radius:0!important;box-' +
'shadow:0 5px 5px #888!important}#searchBarCategory{background:#777!impor' +
'tant}.Pager .ControlArrows{background:#888!important;font-weight:400!imp' +
'ortant}.Pager .PageLink_Selected{background:#999!important}#ForumPopout,' +
'#ForumToolsPopout,#RateThreadPopout{background:#fff!important}#CartButto' +
'n,#CommunityButton,#SupportButton,#searchBarOptionsCell,#searchButtonCel' +
'l,.GridViewIcon,.GroupViewIcon,.ProductGridMenu,.searchbar_focus #search' +
'BarOptionsCell,.searchbar_focus #searchButtonCell{background-image:url(d' +
'ata:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAUAAAAJYBAMAAADh0yLyAAAAJ1B' +
'MVEVMaXFqbG11dXX9/f3l5eXd3eBadIdWbnlWc4RYieRmZ2dpaWlsbW2OPQlLAAAADXRSTlM' +
'A////////////r0YgYtO1qQAAAq9JREFUeAHt2sd160YYx9FxKgBcekc24ADsHTizVwBakEr' +
'g6+CpgBdLcHYJLsA9KTmnZ0PQof+U7j0q4Hc+fMM0KvcAALpI9xsocL2K+/t94GoT6KACNyZ' +
'4oIF20AQF2kGBdtAEBdpBgXbQBAXaQYF20AQF2kGBdtAEI5UMAAAA4V9oBC43938PV/sO3My' +
'1FihQoECB+yBQoECBAgUKFChQoECBAgUKFChQoEA3TcsBAAAA9CXbUEuq+pOhpNoO/fVfTZ5' +
'ga3Wqn6Y/4hLr3ToM4YFj9gRLHdIDb5VwyTuYPsFh4SExwXeGeh35YQEAAACAf/h1K/8aogq' +
'8q7rt+74lB/bXhvjAfkwPrAIXHZKaHNj6oUYHXsf1/ZAcONRtn/5O0o/ZgW3rvTjx4xYAAAA' +
'ArsIEtjrWlj3BbfojHrbxOziEBw7pE0x/xKfN6+ACday1tpY8wTF7gq2168ZxfHw76CoMAAA' +
'AAM4ild+c7xL9PrBLdEiBT1ebQA8hUKBAgQIFChQoUKBAgQIFChQoUKBAgQIFChSYftN0cRE' +
'ZmAEAAOBstrJfu7melP3q5hL4J5u54gN3AgUKFChQoECBAgUKFChQoECBAgUKFChQoMCMwN1' +
'cjzgQAABgmt6bjqZJoEA8YoF4xHsgMA0AADD9ROBs2EHsoEDs4OEDAADaNH1akm1bbSXZONa' +
'jo/CvnUfvlUUcklpryz4ktT2PPiRTe/08+pCMS3/68E7yXvg7yadH4e8kU0l/Jyl3BAAAfPm' +
'T70uqzU/eTw9c/5Ae+P0jDRQoUKDArutWJrgwcP3QJ+iQdE5x+ATtoAl6oRYoUOAeCBRYUnU' +
'/+aqk+vEnBQAAF9qRXGgLFChQoMBlXt9YCbSD/5+zGw7Jwgme28EFuht20Au1QIECBQp0oe1' +
'COx0AAFf2ljch8gqLTAAAAABJRU5ErkJggg==)!important}#navbar tr.navbarsearch' +
',.navsealsbg,body.stripe{background-image:url(' +
'Rw0KGgoAAAANSUhEUgAAAGQAAABGAQMAAAAASKMqAAAAA1BMVE2.5EQ1TRdOAAAAEElEQVR4' +
'AWMYVmAUjIJRAAAD1AABVz/btAAAAABJRU5ErkJggg==)!important}.searchbar_focus' +
' #searchBoxCell{background-image:url(' +
'AANSUhEUgAAAD0AAAAtBAMAAADvmn5BAAAAD1BMVEWYmJmzs7PBwcHGxsagoaK4fiRWAAAAI' +
'ElEQVR4AWNiwA+YBPECgcGuf6DkR+VH5UflR+UpLd8AJUsIkz10gdcAAAAASUVORK5CYII=)' +
'!important}#searchBoxCell{background-image:url(' +
'ORw0KGgoAAAANSUhEUgAAAD0AAAAtAQMAAAAnevExAAAAA1BMVEV3d3dY20ihAAAADUlEQVR' +
'4AWMYfGAUAAABlQABpWSZFgAAAABJRU5ErkJggg==)!important}.SysMsgBanner{borde' +
'r:1px dotted #444!important}#DMCA div.dm-1,#DMCA div.dm-2,#LeftPanel,#ca' +
'tegoryButton,#searchBarOptionsButton,.BorderedPanel,.FormButton,.ForumTh' +
'read,.ForumThread .Signature,.InformationBox,.Pager .ControlArrows,.Page' +
'r .PageLink,.Pager .PageLink_Selected,.ProductThumbnails,.Reviews .Cons,' +
'.SKUAutoLink,.SingleLineTextBox,.TightInformationBox,.VerticalTextArea,.' +
'WarningBox{border:1px solid #444!important}.ForumQuote{border:1px solid ' +
'#555!important;border-left:5px solid #666!important}.mbbc .mbbc-editor{b' +
'order:none!important}td{border-bottom:0!important}#ProductDetails .Attri' +
'butesTableRowGroupTitle{border-bottom:1px dashed #444!important}.Discoun' +
'tsTable,.ProductDescriptions,.fixedCategoryLink,.scHeading{border-bottom' +
':1px dotted #444!important}#navbar,.FilterHeading,.FooterBanner,.PostFla' +
'gs ul li,.tabs-min .ui-widget-header{border-bottom:1px solid #444!import' +
'ant}.SingleLineTextBox:focus,.VerticalTextArea:focus{border-color:#444!i' +
'mportant}.ForumThread .Signature,.Reviews .ReviewContent{border-left:5px' +
' solid #444!important}.ForumsNavigation{border-right:0 solid #e0e0e0!imp' +
'ortant}#ProductSpecifications,.FooterLeft,.navmenubutton_separator{borde' +
'r-right:1px dotted #444!important}.PostFlags,td.PageNavigationalPanel{bo' +
'rder-right:1px solid #444!important}.FilterEntry:hover,.LargeFilterEntry' +
':hover,.ProductFilters .FilterCancellable,.ProductFilters .FilterSelecte' +
'd,.subCategories li:hover,div.StaticPageLink_Selected{border-right:3px s' +
'olid #444!important}#navbar .hdivider,.FooterBanner,.OrdersShipmentHeadi' +
'ng,.PopoutPanel{border-top:1px solid #444!important}.BGShadow{box-shadow' +
':0 1px 1px #666!important}.BGShadowLight{box-shadow:0 3px 3px #666!impor' +
'tant}#CartPopout .ViewCart a:hover,#DMCA,#DMCA div.dm-1 a,#DMCA div.dm-2' +
' a,#ProductDetailsPanel .Price,#navbar .focused a,#searchButtonCell,.Cel' +
'lMode .GridItemPrice,.FieldFlag a,.FormButton,.FormButton.blue:hover,.Fo' +
'rmButton.green:hover,.FormButton.orange:hover,.FormButton.purple:hover,.' +
'FormButton.red:hover,.LineMode .GridItemPrice,.MemberRankIndex,.Pager .P' +
'ageLink_Selected,.Pager .PageLink_Selected:hover,.Pager .PageLink_Select' +
'ed:visited,.Steps .focused,.Steps .focused .StepText,.Steps .focused .St' +
'epTitle,.tabs-min .ui-state-default a,.unitSelector{color:#444!important' +
'}.tabs-min .ui-state-active a{color:#666!important}#navbar .menulinks sp' +
'an a:active,a,a:visited{color:#d5d5d5!important}#categoryButton,#searchB' +
'arCategory,.CartTable .Name a,.CellMode .GridItemModels,.FormButton.whit' +
'e,.FormButton.white:hover,.ForumThread .ThreadCommand,.Information,.Line' +
'Mode .GridItemModels,.Pager .PageLink,.navmenubutton a,.subCategories a,' +
'.topLevelCategories a,.topLevelCategories li.focused a{color:#e1e1e1!imp' +
'ortant}.Black{color:#e2e2e2!important}.PageTitle{color:#e5e5e5!important' +
'}#categoryButton a,.PostFlags .FlagValue,.SingleLineTextBox:focus,.Verti' +
'calTextArea:focus{color:#f1f1f1!important}body{color:#f3f3f3!important}#' +
'content_ProductName,.mbbc .mbbc-align-popup li,.mbbc .mbbc-code-popup li' +
',.mbbc .mbbc-editor,.mbbc .mbbc-font-popup li,.mbbc .mbbc-list-popup li,' +
'.mbbc .mbbc-simple-popup li,.mbbc .mbbc-size-popup li,.subCategories a:h' +
'over,.topLevelCategories a:hover,a:hover{color:#fff!important}.MemberRan' +
'kUsername{font-weight:700!important}.ForumGroup{padding:3px 10px!importa' +
'nt}.mbbc .mbbc-buttons ul{padding:5px 0 3px!important}.mbbc .mbbc-emotic' +
'on-popup .mbbc-group li{padding:8px!important}');
settings.register('use_theme', 'Use custom theme', false, null);
settings.register('theme_name', null, 'nightmode', 'use_theme', {
nightmode: { name: 'Night mode' },
darkft: { name: 'Pafapaf\'s DarkFT' },
});
styles.update();
// Add a "Top of page" button at the bottom of the page, aligned with the pager
if ((ftl.thread && !ftl.perma) || (ftl.threadlist && !ftl.settings)) {
var add_top_btn;
if (!ftl.mobile) {
add_top_btn = function() {
var elm = document.getElementsByClassName('ForumThread TightTable')[0];
if (elm)
elm = elm.nextElementSibling;
if (!elm)
return;
elm.className = 'Pager';
var tbtn = document.createElement('span');
tbtn.className = 'ControlArrows';
tbtn.setAttribute('style', 'float: left; min-width: 150px; ' +
'max-width: 250px; text-align: center');
tbtn.innerHTML = '<a>Top of page</a>';
tbtn.firstElementChild.href = 'javascript:void(0)';
handlers.top_click(tbtn);
var pager = elm.firstElementChild;
elm.appendChild(tbtn);
if (pager) {
var pspan = document.createElement('span');
pspan.appendChild(pager);
elm.appendChild(pspan);
} else {
elm.parentElement
.insertBefore(document.createElement('br'), elm.nextElementSibling);
}
};
} else {
add_top_btn = function() {
var elm = document.getElementsByClassName('ListRow')[0];
if (elm)
elm = elm.parentElement.nextElementSibling;
if (!elm)
return;
var tbtn = document.createElement('button');
tbtn.className = 'btn btn-default';
tbtn.innerHTML = 'Top';
handlers.top_click(tbtn);
if (elm.firstElementChild) {
tbtn.style.marginTop = '-5px';
elm = elm.firstElementChild;
var td = document.createElement('td');
td.style.textAlign = 'right';
td.appendChild(tbtn);
elm.style.width = '100%';
elm.firstElementChild.firstElementChild.appendChild(td);
} else {
var d = elm.appendChild(document.createElement('div'));
d.setAttribute('style', 'width: 100%; text-align: right; ' +
'margin-bottom: 8px');
d.appendChild(tbtn);
}
};
}
block.watch('ready', add_top_btn);
}
if (ftl.editor) {
var bbeeditor; // element to scroll to
var bbetext; // textarea
var bbeexpand; // expand button (mobile)
var bbepanel; // parent of editor
if (!ftl.mobile) {
var fix_editor = function() {
if (bbetext === undefined)
bbetext = document.getElementsByClassName('mbbc-editor')[0];
if (bbetext !== undefined) {
// Save & restore the editor text
if (history.state && history.state.saved_post)
bbetext.value = history.state.saved_post;
window.addEventListener('pagehide', function() {
var state = history.state;
if (state === null)
state = {};
state.saved_post = bbetext.value;
history.replaceState(state, '');
});
// Remove the broken size=[67] buttons
var size_popup = document
.getElementsByClassName('mbbc-size-popup')[0];
if (size_popup !== undefined) {
size_popup.lastElementChild.remove();
size_popup.lastElementChild.remove();
}
// Fix the editor width
bbetext.style.boxSizing = 'border-box';
bbetext.style.width = '100%';
// Add a YouTube button
var url_btn = document.getElementsByClassName('mbbc-url')[0];
if (url_btn !== undefined) {
var yt_btn = document.createElement('li');
yt_btn.innerHTML = ' ';
yt_btn.className = 'mbbc-youtube';
yt_btn.addEventListener('click', function() {
var sp = bbetext.selectionStart;
var ep = bbetext.selectionEnd;
var sel;
if (ep < sp) {
sp = ep;
ep = bbetext.selectionStart;
}
if (ep !== sp) {
sel = bbetext.value.slice(sp, ep);
} else {
sel = prompt('Video URL:', '');
if (sel === null)
return;
var vid = sel
.match(new RegExp('[/?=#]([-\\w]{11})([^-\\w]|$)'));
if (vid !== null)
sel = 'https://www.youtube.com/watch?v=' + vid[1];
}
if (sel === undefined)
return;
bbetext.value = bbetext.value.slice(0, sp) +
'[youtube]' + sel + '[/youtube]' +
bbetext.value.slice(ep);
bbetext.selectionStart = bbetext.selectionEnd = sp + 9 +
(sel === '' ? 0 : 10 + sel.length);
bbetext.focus();
});
url_btn.parentElement.insertBefore(yt_btn, url_btn);
}
}
};
block.watch('late', function() {
// Persist the terms of use checkbox
var toubox = document.getElementById('AgreeTerms');
if (toubox !== null)
handlers.checkbox(toubox, 'agree_tou');
bbeeditor = document.getElementById('bbEditor');
if (bbeeditor !== null) {
bbetext = bbeeditor.getElementsByClassName('mbbc-editor')[0];
if (bbetext !== undefined)
fix_editor();
else
new MutationObserver(function(es, mo) {
mo.disconnect();
fix_editor();
}).observe(bbeeditor, { childList: true });
}
});
} else {
block.watch('ready', function() {
bbetext = document.getElementById('RawContent');
if (bbetext !== null) {
bbepanel = bbetext.parentElement.parentElement.parentElement;
bbeeditor = bbepanel.parentElement;
bbeexpand = bbeeditor.previousElementSibling.firstElementChild
.firstElementChild;
}
});
}
}
if (ftl.thread && !ftl.perma) {
// Re-implement quotes in a way that sort of works
var fix_quote_btns = (function() {
function wrap_tag(tag, val, str) {
return '[' + tag + (val ? '=' + val : '') + ']' + str + '[/' + tag + ']';
}
function inner_quote(elm, range, depth) {
var tmp = '';
for (var e = elm.firstChild; e !== null; e = e.nextSibling) {
var sp, ep;
if (range !== undefined) {
if (range.startContainer.compareDocumentPosition(e) === 2)
continue;
if (range.endContainer.compareDocumentPosition(e) === 4)
break;
if (range.startContainer === e)
sp = range.startOffset;
if (range.endContainer === e)
ep = range.endOffset;
}
switch (e.tagName) {
case 'BR':
tmp += '\n';
break;
case 'STRONG':
tmp += wrap_tag('b', '', inner_quote(e, range, depth));
break;
case 'EM':
tmp += wrap_tag('i', '', inner_quote(e, range, depth));
break;
case 'HR':
tmp += '[hr]';
break;
case 'P':
if (e.innerHTML !== '')
tmp += '\n' + inner_quote(e, range, depth);
break;
case 'SPAN':
var y = e.style;
var t = null;
var v = '';
if (y.textDecoration === 'underline') {
t = 'u';
} else if (y.textDecoration === 'line-through') {
t = 'y';
} else if (y.fontSize !== '') {
t = 'size';
v = parseInt(y.fontSize);
} else if (y.color !== '') {
t = 'color';
v = y.color;
} else if (y.textAlign !== '') {
t = 'align';
v = y.textAlign;
} else {
console.log('Unknown span!');
}
tmp += wrap_tag(t, v, inner_quote(e, range, depth));
break;
case 'DIV':
if (e.className === 'ForumQuote') {
if (tmp[tmp.length - 1] !== '\n')
tmp += '\n';
if (depth === 0 && settings.get('quote_quotes'))
tmp += wrap_tag('quote', '',
inner_quote(e.firstElementChild, range, depth + 1).trim());
} else {
console.log('Unknown div!');
}
break;
case 'IFRAME':
var m = e.src.match(/youtube[^\/]*.com.*\/embed\/([^?]+)/);
if (m !== null && m[1] !== undefined) {
/* A YouTube tag inside a quote breaks the parsing of the quote.
tmp += wrap_tag('youtube', '',
'https://www.youtube.com/watch?v=' + m[1])
*/
tmp += wrap_tag('url',
'https://www.youtube.com/watch?v=' + m[1], 'YouTube video');
} else {
console.warn('Unknown iframe!');
}
break;
case 'IMG':
if (settings.get('quote_imgs')) {
tmp += wrap_tag('img', '', e.src);
} else if (e.parentElement.tagName !== 'A') {
tmp += wrap_tag('url', e.src, 'Image');
} else {
tmp += 'Image';
}
break;
case 'A':
if (e.className === 'SKUAutoLink') {
tmp += e.textContent;
} else {
var do_quote = true;
if ((depth > 0 || !settings.get('quote_quotes') &&
e.innerHTML.match(/\bwrote$/) !== null)) {
// Is this a permalink for a quote we'll remove?
var s = e;
var l;
do {
if (s.nextSibling === null) {
l = (l === undefined) ? s : l;
s = s.parentElement.nextSibling;
} else {
s = s.nextSibling;
}
/* Between the <a> and the <div>, we only want to see a
* colon and an unpredictable number of <br> and empty <p>
* elements. The <div> may be outside the current element.
*/
if (s.tagName === 'DIV' && s.className === 'ForumQuote') {
do_quote = false;
e = (l === undefined) ? s : l;
break;
} else if (s.tagName === undefined) {
if (s.textContent.match(/[^:\r\n]/) !== null)
break;
} else if (s.innerHTML !== '' ||
(s.tagName !== 'P' && s.tagName !== 'BR')) {
break;
}
} while (true);
}
if (do_quote)
tmp += wrap_tag('url', e.href, inner_quote(e, range, depth));
}
break;
case undefined:
// Filter out newlines and empty tags
tmp += e.wholeText.slice(sp ? sp : 0, ep)
.replace(/[\r\n]+/mg, '')
.replace(/\[([^\]=]*)(=[^\]]*)?\]\[\/\1\]/mg, '');
break;
default:
console.warn('Unhandled tag ' + e.tagName);
}
}
return tmp;
}
function quote(user, elm) {
if (bbetext === null) {
console.error('bbEditor text box not found!');
return;
}
var post_id;
var pid = elm.id.match(/^POST([0-9]+)$/);
if (pid !== null)
post_id = pid[1];
var sel = getSelection();
if (sel.rangeCount > 0) {
sel = sel.getRangeAt(0);
if (sel.collapsed || !sel.intersectsNode(elm))
sel = undefined;
} else {
sel = undefined;
}
if (bbetext.value.length > 0 &&
bbetext.value[bbetext.value.length - 1] !== '\n')
bbetext.value += '\n';
if (thread_id !== undefined && post_id !== undefined)
bbetext.value += wrap_tag('url', '/forums/-/t/' + thread_id + '?' +
post_id, user + ' wrote') + ':\n';
else
bbetext.value += user + ' wrote:\n';
bbetext.value += wrap_tag('quote', '', inner_quote(elm, sel, 0)
.trim().replace(/[\r\n]{3,}/mg, '\n\n')) + '\n';
if (bbeeditor)
bbeeditor.scrollIntoView();
if (bbeexpand && bbepanel && bbepanel.className &&
!/\bin\b/.test(bbepanel.className))
bbeexpand.click();
}
var quote_click = handlers.generic({ onclick: function() {
quote(this.data.getAttribute('data-username'), this.data);
}});
return function(elms, func) {
for (var i = 0; i < elms.length; i++) {
var elm = elms[i].content_elm;
quote_click(func(elm), elm);
}
};
})();
if (!ftl.mobile) {
var replace_quotebtn = function(elm) {
// Ugh.
var oldbtn = elm.parentElement.parentElement.parentElement
.previousElementSibling.firstElementChild.lastElementChild
.firstElementChild;
var newbtn = document.createElement('a');
newbtn.innerHTML = 'quote/reply';
newbtn.href = 'javascript:void(0)';
oldbtn.parentElement.insertBefore(newbtn, oldbtn);
oldbtn.remove();
return newbtn;
};
block.watch('late', function() {
fix_quote_btns(posts, replace_quotebtn);
});
} else { // mobile
var replace_quotebtn_m = function(elm) {
// Ugh.
var oldbtn = elm.parentElement.firstElementChild.lastElementChild
.lastElementChild.firstElementChild;
var newbtn = document.createElement('a');
newbtn.innerHTML = oldbtn.innerHTML;
newbtn.href = 'javascript:void(0)';
newbtn.tabindex = '-1';
newbtn.role = 'menuitem';
oldbtn.parentElement.insertBefore(newbtn, oldbtn);
oldbtn.remove();
return newbtn;
};
block.watch('late', function() {
fix_quote_btns(posts, replace_quotebtn_m);
// Add a link to forum settings to allow ignorelist management
// Ugh. There's no semi-robust way to select this.
var li = document.querySelector('ul.nav:nth-child(1)>li:nth-child(3)' +
'>ul:nth-child(2)>li:nth-child(8)');
if (li !== null) {
var new_li = document.createElement('li');
new_li.innerHTML = '<a title="Forum Settings" ' +
'href="https://www.fasttech.com/forums/settings">' +
'Forum Settings</a>';
li.parentElement.insertBefore(new_li, li);
}
});
}
}
if (ftl.thread && ftl.mobile) {
// Fix up YT iframe size: CSS doesn't work well here.
var fr_resize = (function() {
var frs;
var raf;
function do_resize() {
for (var i = 0; i < frs.length; i++) {
var fr = frs[i];
fr.elm.style.height = Math.ceil(fr.elm.clientWidth * fr.ar) + 'px';
}
raf = undefined;
}
return function(fr) {
if (frs === undefined) {
frs = [];
window.addEventListener('resize', function() {
if (raf === undefined)
raf = requestAnimationFrame(do_resize);
});
}
fr.style.maxWidth = fr.width + 'px';
fr.style.width = '100%';
frs.push({elm: fr, ar: fr.height / fr.width});
};
})();
block.watch('ready', function() {
var frs = document.getElementsByTagName('iframe');
for (var i = 0; i < frs.length; i++)
fr_resize(frs[i]);
});
}
if (ftl.thread && !ftl.perma) {
var scroll_to_hash = function() {
function scroll(elm) {
if (elm) {
var sd = elm.style.display;
elm.style.display = '';
elm.scrollIntoView();
elm.style.display = sd;
}
}
if (location.hash === '#unread' || location.hash === '#last')
last_viewed.get(thread_id, function(hash, lvp, lpn, lp) {
block.watch('filter', function() {
var i;
var pid = 'POST' + lp;
for (i = posts.length - 1; i >= 0; i--) {
if (posts[i].content_elm.id === pid)
break;
}
if (i < posts.length - 1)
scroll(posts[i + 1].post_elms[0]);
else if (hash === '#last')
scroll(posts[posts.length - 1].post_elms[0]);
});
}, location.hash);
else if (location.hash !== '')
block.watch('filter', function() {
scroll(document.getElementById(location.hash));
});
// Drop the hash so we won't run again
history.replaceState(history.state, '', location.pathname);
};
}
if (ftl.threadlist) {
if (!ftl.mobile) {
var add_last_viewed =
function(elm, last_viewed_page, last_post_nr, last_post_id) {
var row = elm.parentElement;
var pelm = row.parentElement.getElementsByClassName('Pager')[0];
var p = parse_pager(pelm);
var skip_margin = false;
if (pelm === undefined) {
skip_margin = true;
if (!ftl.settings) {
pelm = document.createElement('div');
pelm.className = 'Pager';
row.appendChild(pelm);
} else {
pelm = row.parentElement.appendChild(document.createElement('div'));
pelm.className = 'Pager';
}
}
if (settings.get('track_viewed')) {
var lv = document.createElement('a');
lv.href = gen_page_link(elm.href, last_viewed_page);
lv.innerHTML = 'Last viewed';
lv.className = 'FormButton blue';
if (!skip_margin)
lv.style.marginLeft = '5px';
pelm.appendChild(lv);
skip_margin = false;
}
var tr = row.parentElement.parentElement;
var ts = tr.getElementsByClassName('ThreadStats')[0];
var pc;
if (ts !== undefined)
pc = (pc = ts.innerHTML.match(/Replies: ([0-9]+)/)) ?
parseInt(pc[1]) : undefined;
var unread = pc > 0 && pc >= last_post_nr;
if ((unread || settings.get('always_unread')) &&
settings.get('track_posts')) {
var np = document.createElement('a');
if (unread) {
np.innerHTML = 'Unread posts';
np.className = 'FormButton red';
np.href = gen_page_link(elm.href,
postnr_to_pagenr(last_post_nr, p.last), 'unread');
} else {
np.innerHTML = 'No unread posts';
np.className = 'FormButton white';
np.href = gen_page_link(elm.href,
postnr_to_pagenr(last_post_nr, p.last), 'last');
}
if (!skip_margin)
np.style.marginLeft = '5px';
pelm.appendChild(np);
skip_margin = false;
}
};
var mangle_list = function() {
var header = document.getElementsByClassName('ForumHeader')[0];
var filter = !ftl.settings && !ftl.author && settings.get('hide_threads');
var sku_re = new RegExp('fasttechcdn.com/[0-9]+/([0-9]+)/.*');
var author_re = new RegExp('started\\W+by\\W+(\\w+)');
if (header !== undefined) {
var ch = header.parentElement.children;
for (var i = 1; i < ch.length; i += 2) {
var tr = ch[i];
var tl = tr.getElementsByClassName('ThreadLink')[0];
var tid;
if (tl !== undefined &&
(settings.get('track_posts') || settings.get('track_viewed'))) {
tid = parseInt(tl.href.match(thread_id_re)[1]);
if (!isNaN(tid))
last_viewed.get(tid, add_last_viewed, tl);
}
var img = tr.firstElementChild.firstElementChild;
if (img && img.tagName === 'IMG') {
var sku = img.src.match(sku_re);
if (sku !== null) {
var p = img.parentElement;
var a = document.createElement('a');
a.href = '/products/' + sku[1];
a.appendChild(img);
p.appendChild(a);
}
}
if (filter) {
var u = tl.parentElement.textContent
.match(author_re);
if (u && u[1] && ignore_list.get(u[1])) {
tr.style.display = 'none';
if (tr.previousElementSibling !== header)
tr.previousElementSibling.style.display = 'none';
}
}
}
}
};
block.watch('ready', mangle_list);
} else { // mobile
var add_last_viewed =
function(elm, last_viewed_page, last_post_nr, last_post_id) {
var rh = elm.parentElement;
var tr = rh.parentElement;
var p = parse_pager(tr.getElementsByClassName('pagination')[0]);
var d = document.createElement('div');
d.style.marginLeft = '-5px';
if (settings.get('track_viewed')) {
var lv = document.createElement('span');
lv.innerHTML = '<a href="' +
gen_page_link(elm.href, last_viewed_page) + '">Last viewed</a>';
lv.className = 'btn btn-default';
lv.style.margin = '5px';
d.appendChild(lv);
}
var pc = (pc = tr.textContent.match(/R(eplies)?: ([0-9]+)/)) ?
parseInt(pc[2]) : undefined;
var unread = pc > 0 && pc >= last_post_nr;
if ((unread || settings.get('always_unread')) &&
settings.get('track_posts')) {
var np = document.createElement('span');
np.className = 'btn btn-default';
np.style.margin = '5px';
var npa = document.createElement('a');
if (unread) {
npa.innerHTML = 'Unread posts';
npa.href = gen_page_link(elm.href,
postnr_to_pagenr(last_post_nr, p.last), 'unread');
npa.style.color = '#C64148';
} else {
npa.innerHTML = 'No unread posts';
npa.style.color = '#888';
npa.href = gen_page_link(elm.href,
postnr_to_pagenr(last_post_nr, p.last), 'last');
}
np.appendChild(npa);
d.appendChild(np);
}
rh.appendChild(d);
};
var mangle_list = function() {
var filter = settings.get('hide_threads');
var elms = document.getElementsByClassName('ListRow');
var author_re = new RegExp('started\\W+by\\W+(\\w+)');
for (var i = 0; i < elms.length; i++) {
var elm = elms[i];
var rh = elm.getElementsByClassName('RowHeading')[0];
if (rh === undefined)
continue;
var l = rh.firstElementChild;
if (l !== undefined &&
(settings.get('track_posts') || settings.get('track_viewed'))) {
var tid = parseInt(l.href.match(thread_id_re)[1]);
if (!isNaN(tid))
last_viewed.get(tid, add_last_viewed, l);
}
if (filter) {
var u = rh.parentElement.textContent.match(author_re);
if (u && u[1] && ignore_list.get(u[1]))
elm.style.display = 'none';
}
}
};
block.watch('ready', function() {
if (ftl.product) {
// Wait for thread list to load before mangling
var div = document.getElementById('forum');
if (div !== null)
new MutationObserver(function(es, mo) {
mo.disconnect();
mangle_list();
}).observe(div, { childList: true });
} else {
mangle_list();
}
});
}
}
if (!ftl.mobile && ftl.thread) {
var have_itmenu = false;
var have_itmanage = false;
var update_itmanage = function(n, itm) {
if (!have_itmenu)
return;
var div = document.getElementById('IgnorelistPopout');
if (div === null)
return;
if (!have_itmanage && itm) {
var frag = document.createDocumentFragment();
var hr = document.createElement('hr');
hr.className = 'itmanage';
frag.appendChild(hr);
var l = document.createElement('div');
l.innerHTML = '<b>Ignored Users</b>';
l.className = 'itmanage';
l.style.marginTop = '10px';
manage_menu.register(l);
l.getElementsByClassName('il_addbtn')[0].className = 'FormButton blue';
frag.appendChild(l);
div.appendChild(frag);
have_itmanage = true;
} else if (have_itmanage) {
var elms = div.getElementsByClassName('itmanage');
for (var i = 0; i < elms.length; i++)
elms[i].style.display = itm ? '' : 'none';
}
};
var update_itmenu = function(n, itm) {
if (!have_itmenu && itm) {
var b = document.getElementsByClassName('ThreadCommandBar')[0];
var sp = document.createElement('span');
sp.id = 'Ignorelist';
sp.className = 'ThreadCommand';
sp.innerHTML = 'Settings';
b.appendChild(sp);
block.watch('late', function() {
var div = document.createElement('div');
div.id = 'IgnorelistPopout';
div.className = 'PopoutPanel';
div.align = 'left';
div.setAttribute('style',
'position: absolute; background: white; padding: 20px');
settings_menu.register(div, thread_menu);
b.appendChild(div);
PM2({par: 'Ignorelist', dupe: false});
have_itmenu = true;
block.watch('late', function() {
update_itmanage(undefined, settings.get('inthread_manage'));
});
});
} else if (have_itmenu) {
var elm = document.getElementById('Ignorelist');
if (elm !== null)
elm.style.display = itm ? '' : 'none';
}
};
post_filter.watch(gen_watch_filter(function(post) {
var tbl = post.post_elms[0].parentElement;
var ph = post.placeholder = document.createElement('tr');
ph.innerHTML = '<td colspan="2" class="ForumHeader" ' +
'style="padding: 10px; text-align: center"><a>Post by ' +
strip_tags(post.user) + ' hidden. Click to show.</a></td>';
ph.firstElementChild.firstElementChild.href = 'javascript:void(0)';
handlers.hide_toggle(ph.firstElementChild.firstElementChild, post);
tbl.insertBefore(ph, post.post_elms[0]);
return ph;
}));
settings.watch('placeholders', post_filter.update);
block.watch('ready', function() {
// Wire posts & ignorebuttons up
ignore_buttons.set('+ ignore', '+ unignore');
for (var i = 0; i < posts.length; i++) {
var post = posts[i];
var tl = post.content_elm.parentElement.parentElement.parentElement;
post.post_elms = [tl.previousElementSibling, tl];
var head = tl.previousElementSibling.firstElementChild.lastElementChild;
var new_a = document.createElement('a');
new_a.href = 'javascript:void(0)';
head.appendChild(new_a);
post.ignbtn = new_a;
post.badge = tl.getElementsByClassName('Badges')[0];
}
post_filter.register(posts);
ignore_buttons.register(posts);
user_tags.register(posts);
settings.watch('inthread_manage', update_itmanage);
settings.watch('inthread_menu', update_itmenu);
});
block.watch('late', function() {
var p = parse_pager(document.getElementsByClassName('Pager')[1]);
var lpn = (p.cur - 1) * 10 + posts.length;
var lp = parseInt(posts[posts.length - 1].content_elm.id.slice(4));
if (!ftl.perma)
last_viewed.set(thread_id, p.cur, lpn, lp);
});
if (!ftl.perma && thread_id !== undefined) {
if (location.hash !== '')
scroll_to_hash();
}
}
if (ftl.mobile && ftl.thread) {
post_filter.watch(gen_watch_filter(function(post) {
var tbl = post.post_elms[0].parentElement;
var ph = post.placeholder = document.createElement('div');
ph.className = 'col-xs-12 ListRow';
ph.style.paddingBottom = '5px';
ph.style.textAlign = 'center';
ph.innerHTML = '<a>Post by ' + strip_tags(post.user) +
' hidden. Click to show.</a>';
ph.firstElementChild.href = 'javascript:void(0)';
handlers.hide_toggle(ph.firstElementChild, post);
tbl.insertBefore(ph, post.post_elms[0]);
return ph;
}));
settings.watch('placeholders', post_filter.update);
ignore_buttons.set(' Ignore', ' Unignore');
var have_itmenu = false;
var have_itmanage = false;
var update_itmanage = function(n, itm) {
if (!have_itmenu)
return;
var menu = document.getElementById('hideSettings');
if (menu === null)
return;
if (!have_itmanage && itm) {
var frag = document.createDocumentFragment();
var inner = menu.getElementsByClassName('panel-body')[0];
var hr = document.createElement('hr');
hr.className = 'itmanage';
frag.appendChild(hr);
var l = document.createElement('div');
l.className = 'itmanage';
l.innerHTML = '<b>Ignored Users</b>';
l.style.marginTop = '10px';
manage_menu.register(l);
l.getElementsByClassName('il_addbtn')[0].remove();
frag.appendChild(l);
inner.appendChild(frag);
have_itmanage = true;
} else if (have_itmanage) {
var elms = menu.getElementsByClassName('itmanage');
for (var i = 0; i < elms.length; i++)
elms[i].style.display = itm ? '' : 'none';
}
};
var update_itmenu = function(n, itm) {
if (!have_itmenu && itm) {
var div = document.createElement('div');
div.id = 'itmenu';
div.className = 'panel panel-default';
div.innerHTML = '<div class="panel-heading"><h5 class="panel-title">' +
'<a href="#hideSettings" data-toggle="collapse">Settings ' +
'<span class="caret"></span></a></h5>' +
'<div id="hideSettings" class="panel-collapse collapse">' +
'<div class="panel-body"></div></div>';
block.watch('late', function() {
var inner = div.getElementsByClassName('panel-body')[0];
settings_menu.register(inner, thread_menu);
document.getElementsByClassName('body-content')[0].appendChild(div);
have_itmenu = true;
if (settings.get('inthread_manage') !== undefined)
update_itmanage(undefined, settings.get('inthread_manage'));
});
} else if (have_itmenu) {
document.getElementById('itmenu').style.display = itm ? '' : 'none';
}
};
block.watch('ready', function() {
for (var i = 0; i < posts.length; i++) {
var post = posts[i];
var pe = post.content_elm.parentElement;
post.post_elms = [pe];
var menu = pe.firstElementChild.lastElementChild;
var li = document.createElement('li');
li.setAttribute('role', 'presentation');
li.innerHTML = '<a tabindex="-1" role="menuitem">' +
'<span class="glyphicon glyphicon-plus red"></span><span></span></a>';
li.firstElementChild.href = 'javascript:void(0)';
var rp = menu.lastElementChild;
if (!rp.firstElementChild.lastChild.data.startsWith(' Report'))
rp = rp.previousElementSibling;
menu.insertBefore(li, rp);
post.ignbtn = li.firstElementChild.lastElementChild;
var badge = menu.firstElementChild.firstElementChild;
var m = badge.innerHTML.match(/(.*)\((.*)\)/);
badge.innerHTML = m[1] + '(<span>' + m[2] + '</span>)';
post.badge = badge.lastElementChild;
}
post_filter.register(posts);
ignore_buttons.register(posts);
user_tags.register(posts);
settings.watch('inthread_manage', update_itmanage);
settings.watch('inthread_menu', update_itmenu);
});
block.watch('late', function() {
var p = parse_pager(document.getElementsByClassName('pagination')[0]);
var lpn = (p.cur - 1) * 10 + posts.length;
var lp = parseInt(posts[posts.length - 1].content_elm.id.slice(4));
if (!ftl.perma)
last_viewed.set(thread_id, p.cur, lpn, lp);
});
if (!ftl.perma && thread_id !== undefined) {
if (location.hash !== '')
scroll_to_hash();
}
}
if (ftl.new_arrivals) {
var product_filter = (function() {
var items;
function update() {
if (items === undefined)
return;
for (var i = 0; i < items.length; i++) {
items[i].content_elm.style.display =
ignore_categories.get(items[i].category) ? 'none' : '';
}
}
return {
register: function(list) {
items = list;
ignore_categories.watch('late', update);
},
};
})();
var catnr_re = new RegExp('categor(y|ies)/([0-9]+)/');
var catstr_re = new RegExp('\\bf=([a-zA-Z0-9]+)');
var catstr_cleanup_re = new RegExp('^.*=([0-9]+)$');
var ignore_category = function(elm, cat) {
elm.addEventListener('click', function() { ignore_categories.set(cat); });
ignore_categories.watch('late', function(c, s) {
if (c === cat)
elm.checked = !s;
});
elm.checked = !ignore_categories.get(cat);
};
var add_ign_cb = function(elm) {
try {
var cat = atob(elm.lastElementChild.href.match(catstr_re)[1])
.match(catstr_cleanup_re)[1];
var cb = document.createElement('input');
cb.type = 'checkbox';
cb.style.float = 'right';
cb.style.margin = '2px -2px 0px -11px';
ignore_category(cb, cat);
elm.insertBefore(cb, elm.firstElementChild);
} catch (e) {}
};
block.watch('ready', function() {
var container = ftl.mobile ?
document.getElementsByClassName('ProductsGrid')[0] :
document.getElementById('Products_Grid');
if (container === null || container === undefined)
return;
var list = [];
for (var elm = container.firstElementChild; elm !== null;
elm = elm.nextElementSibling) {
try {
var cat = elm.lastElementChild.lastElementChild.firstElementChild.href
.match(catnr_re)[2];
list.push({ content_elm: elm, category: cat });
} catch (e) {}
}
product_filter.register(list);
var cats, i;
if (ftl.mobile) {
var heads = document.getElementsByClassName('panel-heading');
for (i = 0; i < heads.length; i++) {
if (heads[i].textContent.trim() === 'Category') {
cats = heads[i];
break;
}
}
if (cats === undefined)
return;
cats = cats.parentElement.lastElementChild.firstElementChild;
while (cats !== null) {
add_ign_cb(cats.lastElementChild);
cats = cats.nextElementSibling;
}
} else {
var groups = document.getElementsByClassName('FilterGroup');
for (i = 0; i < groups.length; i++) {
if (groups[i].textContent.trim() === 'Category') {
cats = groups[i];
break;
}
}
if (cats === undefined)
return;
cats = cats.nextElementSibling;
while (cats !== null && cats.classList.contains('FilterEntry')) {
add_ign_cb(cats);
cats = cats.nextElementSibling;
}
}
});
}
// Mobile reviews don't show tags
if (ftl.reviews && !ftl.mobile) {
var reg_tags = function(badges) {
var posts = [];
for (var i in badges) {
var badge = badges[i];
if (badge.innerText === '' ||
!badge.previousElementSibling ||
badge.previousElementSibling.className !== 'Nickname')
continue;
posts.push({
user: badge.previousElementSibling.innerText.trim(),
badge: badge,
});
}
if (posts.length !== 0)
user_tags.register(posts);
};
block.watch('ready', function() {
var badges = document.getElementsByClassName('Badges');
if (badges.length !== 0) {
reg_tags(badges);
} else {
var elm;
if (!ftl.mobile)
elm = document.getElementById('Reviews-tab');
else
elm = document.getElementById('reviews');
if (elm !== null) {
new MutationObserver(function(es, mo) {
mo.disconnect();
reg_tags(document.getElementsByClassName('Badges'));
}).observe(elm, { childlist: true });
}
}
});
}
if (!ftl.mobile) {
if (ftl.prsearch) {
var search = location.href.match(/\/search(\/([0-9]+)?-?(.*))?\?([^&]+)/);
if (search !== null) {
if (!search[2])
search[2] = '0';
var sortcookie = document.cookie.match(/\bsort=([a-zA-Z0-9]+)/);
sortcookie = sortcookie ? sortcookie[1] : 'r';
location.replace(location.origin +
'/category/' + search[2] + '/search/-/p/0?f=-&keywords=' + search[4] +
'&sort=' + sortcookie);
}
}
if (ftl.prcategory && location.search) {
block.watch('ready', function() {
var sortelm = document.getElementById('content_SortOrder');
if (sortelm) {
sortelm.onchange = function() {
document.cookie = 'sort=' + this.value + ';path=/';
location.href = location.href.replace(/\bsort=[^&]*/, '') +
'&sort=' + this.value;
};
}
var searchcat = document.getElementById('searchBarCategory');
var urlcat = location.search.match(/\bf=([a-zA-Z0-9]+)/);
if (searchcat !== null && urlcat !== null) {
searchcat.value = atob(urlcat[1]).replace(/^.*=/, '');
if (!searchcat.value)
searchcat.value = '';
}
});
}
} else {
if (ftl.prsearch) {
var search = location.href.match(/\/search\/([0-9]+).*\?(.*)/);
if (search !== null) {
var sortcookie = document.cookie.match(/\bsort=([a-zA-Z0-9]+)/);
sortcookie = sortcookie ? sortcookie[1] : '0';
location.replace(location.origin +
'/c/' + search[1] + '/-/-/p/0?f=-&sort=' + sortcookie +
'&' + search[2]);
}
}
if (ftl.prcategory) {
var fix_onclick = function(elm, sortnr) {
elm.onclick = function() {
document.cookie = 'sort=' + sortnr + ';path=/';
location.href = location.href.replace(/\bsort=[^&]*/, '') +
'&sort=' + sortnr;
};
};
block.watch('ready', function () {
try {
var li = document.querySelector('.clearfix button .glyphicon-sort')
.parentElement.parentElement.lastElementChild.firstElementChild;
for ( ; li !== null; li = li.nextElementSibling) {
console.log(li);
var inner = li.firstElementChild;
var sortnr = String.match(inner.onclick, /\bsort=([0-9]+)/);
if (sortnr === null)
continue;
fix_onclick(inner, sortnr[1]);
}
} catch (e) {}
});
}
}
if (ftl.settings) {
block.watch('ready', function() {
var panel = document.getElementsByClassName('PageContentPanel')[0];
var div = document.createElement('div');
div.style.marginTop = '5px';
div.innerHTML = '<table style="width: auto"><tbody><tr><td class="MediumL' +
'abel Bold EndOfInlineSection" style="padding: 5px">Ignored Users</td' +
'><td width=15></td><td class="MediumLabel Bold EndOfInlineSection" s' +
'tyle="padding: 5px">FastTech Forum Enhancements Settings</td></tr><t' +
'r><td style="padding: 0px"><div class="BGShadow" style="padding: 10p' +
'x"></div></td><td width=15></td><td style="padding:10px"><div class=' +
'"BGShadow" style="padding: 0px"></div></td></tr></tbody></table>';
var divs = div.getElementsByTagName('div');
settings_menu.register(divs[1], full_menus);
manage_menu.register(divs[0]);
divs[0].getElementsByClassName('il_addbtn')[0].className =
'FormButton blue';
panel.appendChild(div);
});
}