// ==UserScript== // @name Image Extensions // @description Expand images nicely // @namespace dnsev // @version 3.0.4 // @grant GM_getValue // @grant GM_setValue // @grant GM_deleteValue // @grant GM_listValues // @run-at document-start // @icon  // @include http://boards.4chan.org/* // @include https://boards.4chan.org/* // @include http://boards.4channel.org/* // @include https://boards.4channel.org/* // @include http://i.4cdn.org/* // @include https://i.4cdn.org/* // @updateURL https://raw.githubusercontent.com/dnsev/iex/master/builds/iex.meta.js // @downloadURL https://raw.githubusercontent.com/dnsev/iex/master/builds/iex.user.js // ==/UserScript== (function () { "use strict"; // Timing function var timing = (function () { var perf, now; if ( (perf = window.performance) && (now = perf.now || perf.mozNow || perf.msNow || perf.oNow || perf.webkitNow) ) { return function () { return now.call(perf); }; } else { perf = null; now = null; return function () { return new Date().getTime(); }; } })(); // Variables var doc = document, doc_el = doc.documentElement, user_agent = navigator.userAgent.toString(), is_firefox = (user_agent.indexOf("Firefox") >= 0), is_chrome = (user_agent.indexOf(" Chrome/") >= 0), is_opera = (!is_firefox && !is_chrome && user_agent.indexOf("MSIE") < 0), userscript = {"include":["http://boards.4chan.org/*","https://boards.4chan.org/*","http://boards.4channel.org/*","https://boards.4channel.org/*","http://i.4cdn.org/*","https://i.4cdn.org/*"],"name":"Image Extensions","grant":["GM_getValue","GM_setValue","GM_deleteValue","GM_listValues"],"run-at":"document-start","namespace":"dnsev","updateURL":"https://raw.githubusercontent.com/dnsev/iex/master/builds/iex.meta.js","downloadURL":"https://raw.githubusercontent.com/dnsev/iex/master/builds/iex.user.js","version":"3.0.4","icon":"","description":"Expand images nicely"}, api = null, settings = null, sync = null, style = null, hotkey_manager = null, hover = null, image_hover = null, file_link = null, file_view = null; // Failure if (doc_el === null) { console.log("iex failed to start: document.documentElement = " + doc_el + ";"); console.log(doc_el); return; } // Better binding Function.prototype.bind = function (self) { var fn = this; if (arguments.length > 1) { var slice = Array.prototype.slice, push = Array.prototype.push, args = slice.call(arguments, 1); return function () { var full_args = slice.call(args); push.apply(full_args, arguments); return fn.apply(self, full_args); }; } else { return function () { return fn.apply(self, arguments); }; } }; // Helper functions var MutationObserver = (window.MutationObserver || window.WebKitMutationObserver); var wrap_generic_event = function (self, callback) { // Get any extra arguments var extra_args = Array.prototype.slice.call(arguments, 2); // Return the function wrapped return function (event) { // Setup arguments var new_args = [ event , this ], i = 0, im = extra_args.length; for (; i < im; ++i) new_args.push(extra_args[i]); // Run callback return callback.apply(self, new_args); }; }; var wrap_mouseenterleave_event = (function () { // Handle mouseover/mouseout events to make sure the target is correct var on_mouseenterleave_prehandle = function (event, callback, self, extra_args) { // Must check for same parent element var parent = event.relatedTarget; // Error handling try { // Find parents while (parent) { if (parent === this) return; parent = parent.parentNode; } // Setup event arguments var new_args = [ event , this ], i = 0, im = extra_args.length; for (; i < im; ++i) new_args.push(extra_args[i]); // Okay, trigger event return callback.apply(self, new_args); } catch (e) {} }; // Return a wrapping function return function (self, callback) { // Get any extra arguments var args = Array.prototype.slice.call(arguments, 2); // Return the function wrapped return function (event) { return on_mouseenterleave_prehandle.call(this, event, callback, self, args); }; }; })(); var add_event_listener = function (list, node, event, callback, capture) { node.addEventListener(event, callback, capture); list.push([node, event, callback, capture]); }; var remove_event_listeners = function (list) { var list_len = list.length, i = 0, li; for (; i < list_len; ++i) { li = list[i]; li[0].removeEventListener(li[1], li[2], li[3]); } }; var stop_event = function (event) { event.preventDefault(); event.stopPropagation(); return false; }; var get_event_mouse_button = function (event) { // 1=left, 2=middle, 3=right if (event.which) { if (event.which === 1) return 1; else if (event.which === 2) return 2; else if (event.which === 3) return 3; } else { if ((event.button & 1) !== 0) return 1; else if ((event.button & 2) !== 0) return 3; else if ((event.button & 4) !== 0) return 2; } return 0; }; var select_input = function (node, caret_start, caret_end) { var len = (node.value || "").length; // Bound if (caret_start === undefined) { caret_start = 0; caret_end = len; } else if (caret_end === undefined) { if (caret_start < 0) caret_start = 0; else if (caret_start > len) caret_start = len; caret_end = caret_start; } else { if (caret_start < 0) caret_start = 0; else if (caret_start > len) caret_start = len; if (caret_end < caret_start) caret_end = caret_start; else if (caret_end > len) caret_end = len; } // Select if (node.createTextRange) { var range = node.createTextRange(); range.move("character", caret_start); range.select(); } else { if (node.selectionStart) { node.focus(); node.setSelectionRange(caret_start, caret_end); } else { node.focus(); } } }; var $ = function (tag) { return doc.createElement(tag); }; $.node = function (tag, class_name) { var n = doc.createElement(tag); n.className = class_name + style.theme; return n; }; $.node_ns = function (ns, tag, class_name) { var n = doc.createElementNS(ns, tag); n.setAttribute("class", class_name + style.theme); return n; }; $.text = function (text) { return doc.createTextNode(text); }; $.a = function (class_name) { var n = doc.createElement("a"); n.className = class_name + style.theme; return n; }; $.div = function (class_name) { var n = doc.createElement("div"); n.className = class_name + style.theme; return n; }; $.span = function (class_name) { var n = doc.createElement("span"); n.className = class_name + style.theme; return n; }; $.label = function (class_name) { var n = doc.createElement("label"); n.className = class_name + style.theme; return n; }; $.input = function (class_name) { var n = doc.createElement("input"); n.className = class_name + style.theme; return n; }; $.input.check = function (class_name) { var n = doc.createElement("input"); n.setAttribute("type", "checkbox"); n.className = class_name + style.theme; return n; }; $.input.text = function (class_name) { var n = doc.createElement("input"); n.setAttribute("type", "text"); n.className = class_name + style.theme; return n; }; $.input.file = function (class_name) { var n = doc.createElement("input"); n.setAttribute("type", "file"); n.className = class_name + style.theme; return n; }; $.input.button = function (class_name) { var n = doc.createElement("input"); n.setAttribute("type", "button"); n.className = class_name + style.theme; return n; }; var crc32 = (function () { var crc_table = new Uint32Array([ //{ 0x00000000, 0x77073096, 0xEE0E612C, 0x990951BA, 0x076DC419, 0x706AF48F, 0xE963A535, 0x9E6495A3, 0x0EDB8832, 0x79DCB8A4, 0xE0D5E91E, 0x97D2D988, 0x09B64C2B, 0x7EB17CBD, 0xE7B82D07, 0x90BF1D91, 0x1DB71064, 0x6AB020F2, 0xF3B97148, 0x84BE41DE, 0x1ADAD47D, 0x6DDDE4EB, 0xF4D4B551, 0x83D385C7, 0x136C9856, 0x646BA8C0, 0xFD62F97A, 0x8A65C9EC, 0x14015C4F, 0x63066CD9, 0xFA0F3D63, 0x8D080DF5, 0x3B6E20C8, 0x4C69105E, 0xD56041E4, 0xA2677172, 0x3C03E4D1, 0x4B04D447, 0xD20D85FD, 0xA50AB56B, 0x35B5A8FA, 0x42B2986C, 0xDBBBC9D6, 0xACBCF940, 0x32D86CE3, 0x45DF5C75, 0xDCD60DCF, 0xABD13D59, 0x26D930AC, 0x51DE003A, 0xC8D75180, 0xBFD06116, 0x21B4F4B5, 0x56B3C423, 0xCFBA9599, 0xB8BDA50F, 0x2802B89E, 0x5F058808, 0xC60CD9B2, 0xB10BE924, 0x2F6F7C87, 0x58684C11, 0xC1611DAB, 0xB6662D3D, 0x76DC4190, 0x01DB7106, 0x98D220BC, 0xEFD5102A, 0x71B18589, 0x06B6B51F, 0x9FBFE4A5, 0xE8B8D433, 0x7807C9A2, 0x0F00F934, 0x9609A88E, 0xE10E9818, 0x7F6A0DBB, 0x086D3D2D, 0x91646C97, 0xE6635C01, 0x6B6B51F4, 0x1C6C6162, 0x856530D8, 0xF262004E, 0x6C0695ED, 0x1B01A57B, 0x8208F4C1, 0xF50FC457, 0x65B0D9C6, 0x12B7E950, 0x8BBEB8EA, 0xFCB9887C, 0x62DD1DDF, 0x15DA2D49, 0x8CD37CF3, 0xFBD44C65, 0x4DB26158, 0x3AB551CE, 0xA3BC0074, 0xD4BB30E2, 0x4ADFA541, 0x3DD895D7, 0xA4D1C46D, 0xD3D6F4FB, 0x4369E96A, 0x346ED9FC, 0xAD678846, 0xDA60B8D0, 0x44042D73, 0x33031DE5, 0xAA0A4C5F, 0xDD0D7CC9, 0x5005713C, 0x270241AA, 0xBE0B1010, 0xC90C2086, 0x5768B525, 0x206F85B3, 0xB966D409, 0xCE61E49F, 0x5EDEF90E, 0x29D9C998, 0xB0D09822, 0xC7D7A8B4, 0x59B33D17, 0x2EB40D81, 0xB7BD5C3B, 0xC0BA6CAD, 0xEDB88320, 0x9ABFB3B6, 0x03B6E20C, 0x74B1D29A, 0xEAD54739, 0x9DD277AF, 0x04DB2615, 0x73DC1683, 0xE3630B12, 0x94643B84, 0x0D6D6A3E, 0x7A6A5AA8, 0xE40ECF0B, 0x9309FF9D, 0x0A00AE27, 0x7D079EB1, 0xF00F9344, 0x8708A3D2, 0x1E01F268, 0x6906C2FE, 0xF762575D, 0x806567CB, 0x196C3671, 0x6E6B06E7, 0xFED41B76, 0x89D32BE0, 0x10DA7A5A, 0x67DD4ACC, 0xF9B9DF6F, 0x8EBEEFF9, 0x17B7BE43, 0x60B08ED5, 0xD6D6A3E8, 0xA1D1937E, 0x38D8C2C4, 0x4FDFF252, 0xD1BB67F1, 0xA6BC5767, 0x3FB506DD, 0x48B2364B, 0xD80D2BDA, 0xAF0A1B4C, 0x36034AF6, 0x41047A60, 0xDF60EFC3, 0xA867DF55, 0x316E8EEF, 0x4669BE79, 0xCB61B38C, 0xBC66831A, 0x256FD2A0, 0x5268E236, 0xCC0C7795, 0xBB0B4703, 0x220216B9, 0x5505262F, 0xC5BA3BBE, 0xB2BD0B28, 0x2BB45A92, 0x5CB36A04, 0xC2D7FFA7, 0xB5D0CF31, 0x2CD99E8B, 0x5BDEAE1D, 0x9B64C2B0, 0xEC63F226, 0x756AA39C, 0x026D930A, 0x9C0906A9, 0xEB0E363F, 0x72076785, 0x05005713, 0x95BF4A82, 0xE2B87A14, 0x7BB12BAE, 0x0CB61B38, 0x92D28E9B, 0xE5D5BE0D, 0x7CDCEFB7, 0x0BDBDF21, 0x86D3D2D4, 0xF1D4E242, 0x68DDB3F8, 0x1FDA836E, 0x81BE16CD, 0xF6B9265B, 0x6FB077E1, 0x18B74777, 0x88085AE6, 0xFF0F6A70, 0x66063BCA, 0x11010B5C, 0x8F659EFF, 0xF862AE69, 0x616BFFD3, 0x166CCF45, 0xA00AE278, 0xD70DD2EE, 0x4E048354, 0x3903B3C2, 0xA7672661, 0xD06016F7, 0x4969474D, 0x3E6E77DB, 0xAED16A4A, 0xD9D65ADC, 0x40DF0B66, 0x37D83BF0, 0xA9BCAE53, 0xDEBB9EC5, 0x47B2CF7F, 0x30B5FFE9, 0xBDBDF21C, 0xCABAC28A, 0x53B39330, 0x24B4A3A6, 0xBAD03605, 0xCDD70693, 0x54DE5729, 0x23D967BF, 0xB3667A2E, 0xC4614AB8, 0x5D681B02, 0x2A6F2B94, 0xB40BBE37, 0xC30C8EA1, 0x5A05DF1B, 0x2D02EF8D, ]); //} return function (data) { var crc = (0 ^ (-1)), data_len = data.length, ct = crc_table, i; for (i = 0; i < data_len; ++i) { crc = (crc >>> 8) ^ ct[(crc ^ data[i]) & 0xFF]; } return (crc ^ (-1)) >>> 0; }; })(); // Function for performing actions as soon as possible var on_ready = (function () { // Vars var callbacks = [], check_interval = null, check_interval_time = 250; // Data to store callback and relevant data var Data = function (callback, condition, time_limit) { this.callback = callback; this.condition = condition; this.time_limit = time_limit; }; Data.prototype.delay = function () { var limit = null, condition = this.condition, check; if (this.time_limit) { limit = setTimeout(function () { clearInterval(check); }, this.time_limit * 1000); } check = setInterval(function () { if (condition.call(null)) { clearInterval(check); if (limit !== null) clearTimeout(limit); this.callback.call(null); } }, 20); }; // Check if ready and run callbacks var callback_check = function () { if ( (document.readyState === "interactive" || document.readyState === "complete") && callbacks !== null ) { // Run callbacks var cbs = callbacks, cb_count = cbs.length, cb, i; // Clear callbacks = null; for (i = 0; i < cb_count; ++i) { cb = cbs[i]; if (!cb.condition || cb.condition.call(null)) { cb.callback.call(null); } else { cb.delay(); } } // Clear events and checking interval window.removeEventListener("load", callback_check, false); window.removeEventListener("readystatechange", callback_check, false); if (check_interval !== null) { clearInterval(check_interval); check_interval = null; } // Okay return true; } // Not executed return false; }; // Listen window.addEventListener("load", callback_check, false); window.addEventListener("readystatechange", callback_check, false); // Callback adding function return function (callback, condition, time_limit) { if (callbacks === null) { // Ready to execute if (!condition || condition.call(null)) { callback.call(null); } else { // Delay new Data(callback, condition, time_limit).delay(); } } else { // Delay callbacks.push(new Data(callback, condition, time_limit)); // Set a check interval if (check_interval === null && callback_check() !== true) { check_interval = setInterval(callback_check, check_interval_time); } } }; })(); // Module to manage data saving with an asynchronous paradigm var SaveAsync = (function () { // Storage type var using_localstorage = true, chrome_storage = null; // Check for chrome storage try { chrome_storage = chrome.storage.local || null; } catch (e) { chrome_storage = null; } // Check for GM storage try { if (GM_setValue && GM_getValue && GM_deleteValue && GM_listValues) { using_localstorage = false; } } catch (e) { using_localstorage = true; } // HTML5 storage var decode_value = function (value) { if (value) { try { return JSON.parse(value); } catch (e) {} } return value; }, object_byte_size = function (obj) { // Calculate byte length obj = JSON.stringify(obj); try { // Encode in utf-8 return unescape(encodeURIComponent(obj)).length; } catch (e) { return obj.length; } }, generic_get_value = function (key, callback) { var val = this.getItem(key, null); callback.call(null, decode_value(val)); }, generic_set_value = function (key, value, callback) { this.setItem(key, JSON.stringify(value)); if (callback) callback.call(null); }, generic_del_value = function (key, callback) { this.removeItem(key); if (callback) callback.call(null); }, generic_get_keys = function (callback) { var keys = [], key; for (key in this) { keys.push(key); } callback.call(null, keys); }, generic_get_space_used = function (callback) { var size = 0, key; // Create representation for (key in this) { size += object_byte_size(key) + object_byte_size(decode_value(this.getItem(key, null))); } // Return callback.call(null, size); }, w_get_value = generic_get_value.bind(window.localStorage), w_set_value = generic_set_value.bind(window.localStorage), w_del_value = generic_del_value.bind(window.localStorage), w_get_keys = generic_get_keys.bind(window.localStorage), w_get_space_used = generic_get_space_used.bind(window.localStorage), s_get_value = generic_get_value.bind(window.sessionStorage), s_set_value = generic_set_value.bind(window.sessionStorage), s_del_value = generic_del_value.bind(window.sessionStorage), s_get_keys = generic_get_keys.bind(window.sessionStorage), s_get_space_used = generic_get_space_used.bind(window.sessionStorage), get_value, set_value, del_value, get_keys, get_space_used; // Userscript storage if (chrome_storage) { // Chrome storage var get_keys_callback = function (next_callback, obj) { var keys = [], key; for (key in obj) { keys.push(key); } next_callback.call(null, keys); }; var on_get = function (key, callback, value) { callback.call(null, value[key]); }; get_value = function (key, callback) { this.get(key, on_get.bind(null, key, callback)); }.bind(chrome_storage); set_value = function (key, value, callback) { var obj = {}; obj[key] = value; this.set(obj, callback); }.bind(chrome_storage); del_value = function (key, callback) { this.remove(key, callback); }.bind(chrome_storage); get_keys = function (callback) { this.get(null, get_keys_callback.bind(null, callback)); }.bind(chrome_storage); get_space_used = function (callback) { this.getBytesInUse(null, callback); }.bind(chrome_storage); } else if (using_localstorage) { // Local storage get_value = w_get_value; set_value = w_set_value; del_value = w_del_value; get_keys = w_get_keys; get_space_used = w_get_space_used; } else { // GM storage get_value = function (key, callback) { var val = GM_getValue(key, null); callback.call(null, decode_value(val)); }; set_value = function (key, value, callback) { GM_setValue(key, JSON.stringify(value)); if (callback) callback.call(null); }; del_value = function (key, callback) { GM_deleteValue(key); if (callback) callback.call(null); }; get_keys = function (callback) { var keys = GM_listValues(); callback.call(null, keys); }; get_space_used = function (callback) { var keys = GM_listValues(), size = 0, i; // Create representation for (i = 0; i < keys.length; ++i) { size += object_byte_size(keys[i]) + object_byte_size(decode_value(GM_getValue(keys[i], null))); } // Return callback.call(null, size); }; } // Return function list return { set: set_value, // key, value, callback() get: get_value, // key, callback(value) del: del_value, // key, callback() keys: get_keys, // callback([...]) space_used: get_space_used, // callback(bytes) w_set: w_set_value, w_get: w_get_value, w_del: w_del_value, w_keys: w_get_keys, w_space_used: w_get_space_used, s_set: s_set_value, s_get: s_get_value, s_del: s_del_value, s_keys: s_get_keys, s_space_used: s_get_space_used, mode: chrome_storage ? "chrome" : (using_localstorage ? "localstorage" : "gm"), }; })(); // Module to have "delayed" loops var Delay = (function () { var DelayEach = (function () { var DelayEach = function (i, array, callback, max_count, delay, remove) { this.i = i; this.array = array; this.callback = callback; this.max_count = max_count; this.delay = delay; this.timeout = null; this.timeout_fcn = remove ? run_loop_with_remove.bind(this) : run_loop.bind(this); // Run this.timeout_fcn.call(this); }; var run_loop = function () { // Reset this.timeout = null; // Find max i var i = this.i; var array = this.array; var callback = this.callback; var i_max = i + this.max_count; var continue_loop = (array.length > i_max); if (!continue_loop) i_max = array.length; // Loop for (; i < i_max; ++i) { callback.call(this, array[i], i, this); } // Continue if (continue_loop) { this.timeout = setTimeout(this.timeout_fcn, this.delay); } }; var run_loop_with_remove = function () { // Reset this.timeout = null; // Find max i var i = 0; var array = this.array; var callback = this.callback; var i_max = i + this.max_count; var continue_loop = (array.length > i_max); if (!continue_loop) i_max = array.length; // Loop for (; i < i_max; ++i) { callback.call(this, array[i], i, this); } this.array.splice(0, i); this.i += i; // Continue if (continue_loop) { this.timeout = setTimeout(this.timeout_fcn, this.delay); } }; DelayEach.prototype = { constructor: DelayEach, pop: function (value) { for (var i = this.array.length - 1; i >= 0; --i) { if (this.array[i] === value) { this.array.splice(i, 1); return true; } } return false; }, push: function (item) { this.array.push(item); if (this.timeout === null) this.timeout_fcn.call(this); }, concat: function (other) { this.array = this.array.concat(other); if (this.timeout === null) this.timeout_fcn.call(this); }, }; return DelayEach; })(); return { each: function (array, callback, max_count, delay) { // Start if (arguments.length < 3 || max_count <= 0) max_count = 1; if (arguments.length < 4 || delay <= 0) delay = 1; return new DelayEach(0, array, callback, max_count, delay * 1000, false); }, queue: function (array, callback, max_count, delay) { // Start if (arguments.length < 3 || max_count <= 0) max_count = 1; if (arguments.length < 4 || delay <= 0) delay = 1; return new DelayEach(0, array, callback, max_count, delay * 1000, true); }, }; })(); // Class to manage tab synchronization events var Sync = (function () { var Sync = function () { this.key_name = "iex_sync"; this.listeners = {}; window.addEventListener("storage", (this.on_storage_change_bind = on_storage_change.bind(this)), false); }; var on_storage_change = function (event) { // Make sure it's an "addition" trigger if (event.key == this.key_name && event.newValue !== null) { var key = event.newValue, data = null; try { data = JSON.parse(key); key = data[0]; data = data[1]; } catch (e) { key = event.newValue; data = null; } if (key in this.listeners) { // Signal all var list = this.listeners[key]; for (var i = 0; i < list.length; ++i) { // Signal list[i].call(this, key, data); } } } }; Sync.instance = null; Sync.prototype = { constructor: Sync, /** Remove any bound events or leftover data used by an instance */ destroy: function () { // Remove events window.removeEventListener("storage", this.on_storage_change_bind, false); }, /** Add a synchronization event listener @param key They sync key to listen for @param callback The function to be called Callback format is as follows: callback.call(Sync.instance, key, data) */ on: function (key, callback) { // Add to listener list if (key in this.listeners) { this.listeners[key].push(callback); } else { this.listeners[key] = [ callback ]; } }, /** Remove a synchronization event listener @param key They sync key used in the .on function @param callback The callback function used in the .on function @return true if it was removed successfully, false if it did not exist */ off: function (key, callback) { // Check if it exists if (key in this.listeners) { var list = this.listeners[key]; for (var i = 0; i < list.length; ++i) { if (list[i] === callback) { // Remove from list list.splice(i, 1); // Remove key if empty if (list.length === 0) { delete this.listeners[key]; } // Success return true; } } } // Not found return false; }, /** Trigger a synchronization event @param key They sync key @return true if no errors occur, false otherwise */ trigger: function (key, data) { // Set the value try { // Trigger var value = JSON.stringify([ key , data ]); window.localStorage.setItem(this.key_name, value); window.localStorage.removeItem(this.key_name); // Okay return true; } catch (e) { // Failure return false; } }, }; return Sync; })(); // 4chan site and script interface var API = (function () { var API = function () { // Custom events this.events = { "page_type_detected": [], "script_detected": [], "thread_update": [], "settings_4chanx_open": [], "settings_4chanx_close": [], "settings_4chanx_section_change": [], "settings_vanilla_open": [], "settings_vanilla_close": [], "menu_4chanx_open": [], "menu_4chanx_close": [], "image_hover_open": [], "image_hover_close": [], "post_add": [], "post_remove": [], "quick_reply_add": [], "quick_reply_remove": [], "quick_reply_show": [], "quick_reply_hide": [], "file_info_update": [], }; this.getters = { "posts": get_all_posts.bind(this), }; // Nodes this.settings_4chanx_container = null; this.settings_vanilla_container = null; this.menu_4chanx_container = null; // Script this.is_4chanx = false; this.is_appchanx = false; this.page_type = ""; // Document listening this.doc_observer = null; this.body_observer = null; this.delform_observer = null; this.settings_observer = null; this.hover_ui_observer = null; this.header_settings_menu_observer = null; this.linkify_observer = null; this.quick_reply_observer = null; }; var on_asap = function () { // Hook hook_body_observers.call(this); hook_hover_observers.call(this, null); hook_delform_observers.call(this, null); hook_header_observers.call(this, null); // Detect scripts var el = document.documentElement; if (el) { detect_page_type.call(this, el); detect_scripts.call(this, el); var qr; if ((qr = el.querySelector("body>#qr,body>#quickReply"))) { trigger.call(this, "quick_reply_add", qr); trigger.call(this, "quick_reply_show", qr); hook_quick_reply_observers.call(this, qr); } } }; var on_document_observe = function (records) { var i, r; for (i = 0; i < records.length; ++i) { r = records[i]; if (r.attributeName == "class") { // Detect detect_scripts.call(this, r.target); } } }; var on_body_observe = function (records) { var i, j, nodes; for (i = 0; i < records.length; ++i) { if ((nodes = records[i].addedNodes)) { for (j = 0; j < nodes.length; ++j) { // Check on_body_element_add.call(this, nodes[j]); } } if ((nodes = records[i].removedNodes)) { for (j = 0; j < nodes.length; ++j) { // Check on_body_element_remove.call(this, nodes[j]); } } } }; var on_body_element_add = function (element) { var id = element.getAttribute("id"); if (id == "fourchanx-settings") { // 4chan-x / ccd0 trigger_settings_4chanx_open.call(this, element); } else if (id == "appchanx-settings") { // appchan-x / zixaphir trigger_settings_4chanx_open.call(this, element); } else if (id == "overlay") { // 4chan-x / ihavenoface var e2 = element.querySelector("#fourchanx-settings"); if (e2) { trigger_settings_4chanx_open.call(this, e2); } } else if (id == "settingsMenu") { // 4chan vanilla trigger_settings_vanilla_open.call(this, element); } else if (id == "hoverUI") { hook_hover_observers.call(this, element); } else if (id == "delform") { hook_delform_observers.call(this, element); } else if (id == "image-hover") { // Vanilla 4chan image hover trigger.call(this, "image_hover_open", { container: element }); } else if (id == "qp") { // 4chan-x var pc = element.querySelector(".postContainer"); if (pc) { trigger.call(this, "post_add", pc); } // else, post might not be loaded yet } else if (id == "header-bar" || id == "header") { hook_header_observers.call(this, element); } else if (id == "qr" || id == "quickReply") { // 4chan-x quick reply trigger.call(this, "quick_reply_add", element); trigger.call(this, "quick_reply_show", element); hook_quick_reply_observers.call(this, element); } /*else if (id == "quote-preview") { // Vanilla 4chan // Only a .post; no .postContainer trigger.call(this, "post_add", element); }*/ }; var on_body_element_remove = function (element) { var id = element.getAttribute("id"); if (id == "fourchanx-settings") { trigger_settings_4chanx_close.call(this, element); } else if (id == "appchanx-settings") { trigger_settings_4chanx_close.call(this, element); } else if (id == "overlay") { var e2 = element.querySelector("#fourchanx-settings"); if (e2) { trigger_settings_4chanx_close.call(this, e2); } } else if (id == "settingsMenu") { // 4chan vanilla trigger_settings_vanilla_close.call(this, element); } else if (id == "image-hover") { trigger.call(this, "image_hover_close", { container: element }); } else if (id == "qp") { // 4chan-x var pc = element.querySelector(".postContainer"); if (pc) { trigger.call(this, "post_remove", pc); } } else if (id == "qr" || id == "quickReply") { trigger.call(this, "quick_reply_hide", element); trigger.call(this, "quick_reply_remove", element); unhook_quick_reply_observers.call(this); } /*else if (id == "quote-preview") { // Vanilla 4chan // Only a .post; no .postContainer trigger.call(this, "post_remove", element); }*/ }; var on_delform_post_observe = function (records) { var nodes, n, i, j, k, im, jm, km, pc; i = 0; im = records.length; for (; i < im; ++i) { if ((nodes = records[i].addedNodes)) { j = 0; jm = nodes.length; for (; j < jm; ++j) { // Check n = nodes[j]; if (style.has_class(n, "postContainer")) { trigger.call(this, "post_add", n); } else if (style.has_class(n, "thread") || style.has_class(n, "board")) { pc = n.querySelectorAll(".postContainer"); k = 0; km = pc.length; for (; k < km; ++k) { trigger.call(this, "post_add", pc[k]); } } else if (style.has_class(n, "file-info")) { trigger.call(this, "file_info_update", n); } /*else if (n.tagName == "A" && style.has_class(n, "linkify")) { trigger.call(this, "linkify", n); }*/ } } if ((nodes = records[i].removedNodes)) { j = 0; jm = nodes.length; for (; j < jm; ++j) { // Check n = nodes[j]; if (style.has_class(n, "inline")) { pc = n.querySelector(".postContainer"); if (pc) { trigger.call(this, "post_remove", pc); } } else if (style.has_class(n, "thread") || style.has_class(n, "board")) { pc = n.querySelectorAll(".postContainer"); k = 0; km = pc.length; for (; k < km; ++k) { trigger.call(this, "post_remove", pc[k]); } } } } } }; var on_settings_observe = function (records) { for (var i = 0; i < records.length; ++i) { if (records[i].attributeName == "class") { // Section change trigger.call(this, "settings_4chanx_section_change", { container: this.settings_4chanx_container, section: records[i].target }); } } }; var on_quick_reply_observe = function (records) { var i, r; for (i = 0; i < records.length; ++i) { r = records[i]; if (r.attributeName == "hidden") { // Detect if (r.target.getAttribute(r.attributeName) === null) { trigger.call(this, "quick_reply_show", r.target); } else { trigger.call(this, "quick_reply_hide", r.target); } } } }; var on_hover_ui_observe = function (records) { var i, j, nodes, r; for (i = 0; i < records.length; ++i) { r = records[i]; if ((nodes = r.addedNodes)) { for (j = 0; j < nodes.length; ++j) { // Check on_hover_ui_element_add.call(this, nodes[j]); } } if ((nodes = r.removedNodes)) { for (j = 0; j < nodes.length; ++j) { // Check on_hover_ui_element_remove.call(this, nodes[j]); } } } }; var on_hover_ui_element_add = function (element) { var id = element.getAttribute("id"); if (id == "ihover") { if (element.getAttribute("data-full-i-d")) { trigger.call(this, "image_hover_open", { container: element }); } } else if (id == "qp") { var pc = element.querySelector(".postContainer"); if (pc) { trigger.call(this, "post_add", pc); } // else, post might not be loaded yet } else if (id == "menu") { // 4chan-x / ccd0, appchan-x trigger_menu_4chanx_open.call(this, element, null); } }; var on_hover_ui_element_remove = function (element) { var id = element.getAttribute("id"); if (id == "ihover") { trigger.call(this, "image_hover_close", { container: element }); } else if (id == "qp") { var pc = element.querySelector(".postContainer"); if (pc) { trigger.call(this, "post_remove", pc); } } else if (id == "menu") { // 4chan-x / ccd0, appchan-x trigger_menu_4chanx_close.call(this, element); } }; var on_4chanx_menu_observe = function (type, records) { var i, j, nodes, r; for (i = 0; i < records.length; ++i) { r = records[i]; if ((nodes = r.addedNodes)) { for (j = 0; j < nodes.length; ++j) { // Check on_4chanx_menu_element_add.call(this, nodes[j], type); } } if ((nodes = r.removedNodes)) { for (j = 0; j < nodes.length; ++j) { // Check on_4chanx_menu_element_remove.call(this, nodes[j], type); } } } }; var on_4chanx_menu_element_add = function (element, type) { var id = element.getAttribute("id"); if (id == "menu") { // 4chan-x trigger_menu_4chanx_open.call(this, element, type); } }; var on_4chanx_menu_element_remove = function (element, type) { var id = element.getAttribute("id"); if (id == "menu") { // 4chan-x trigger_menu_4chanx_close.call(this, element); } }; var on_menu_4chanx_entry_mouseenter = function (event, node) { if (this.menu_4chanx_container) { var focused = this.menu_4chanx_container.querySelectorAll(".focused"), i; for (i = 0; i < focused.length; ++i) { style.remove_class(focused[i], "focused"); } } style.add_class(node, "focused"); }; var on_menu_4chanx_entry_mouseleave = function (event, node) { // This is removed even for versions which menu items keep it (4chan-x / ccd0, etc.) style.remove_class(node, "focused"); }; var hook_document_observer = function () { var el = document.documentElement; if (!el) return; if (this.doc_observer === null) { // Create new observer this.doc_observer = new MutationObserver(on_document_observe.bind(this)); // Observe this.doc_observer.observe( el, { attributes: true } ); } }; var hook_body_observers = function () { // Observe the body var body = document.querySelector("body"); if (!body) return; if (this.body_observer === null) { // Create new observer this.body_observer = new MutationObserver(on_body_observe.bind(this)); // Observe this.body_observer.observe( body, { childList: true, } ); } }; var hook_hover_observers = function (hover_ui) { if (hover_ui === null) { hover_ui = document.getElementById("hoverUI"); if (!hover_ui) return; } if (this.hover_ui_observer === null) { // Create new observer this.hover_ui_observer = new MutationObserver(on_hover_ui_observe.bind(this)); // Observe this.hover_ui_observer.observe( hover_ui, { childList: true, } ); } }; var hook_delform_observers = function (delform) { if (delform === null) { delform = document.getElementById("delform"); if (!delform) return; } if (this.delform_observer === null) { // Create new observer this.delform_observer = new MutationObserver(on_delform_post_observe.bind(this)); // Observe this.delform_observer.observe( delform, { childList: true, subtree: true, } ); } }; var hook_header_observers = function (header) { if (header === null) { header = document.getElementById("header-bar"); if (!header) { header = document.getElementById("header"); if (!header) return; } } var button; if (this.header_settings_menu_observer === null && (button = header.querySelector(".menu-button"))) { // Create new observer this.header_settings_menu_observer = new MutationObserver(on_4chanx_menu_observe.bind(this, "main")); // Observe this.header_settings_menu_observer.observe( button, { childList: true, } ); } }; var hook_quick_reply_observers = function (qr) { // Observe the quick reply if (!qr) return; if (this.quick_reply_observer === null) { // Create new observer this.quick_reply_observer = new MutationObserver(on_quick_reply_observe.bind(this)); // Observe this.quick_reply_observer.observe( qr, { attributes: true, attributeOldValue: true, } ); } }; var unhook_quick_reply_observers = function () { // Stop observing if (this.quick_reply_observer !== null) { this.quick_reply_observer.disconnect(); this.quick_reply_observer = null; return true; } return false; }; var detect_page_type = function (doc_el) { // Detect page type var delform = doc_el.querySelector("#delform"); if (delform) { var flash_table = delform.querySelector(".flashListing"); if (flash_table) { // Flash board this.page_type = "board_flash"; } else { // Thread or board or catalog if (delform.querySelector(".board.catalog-small")) { // 4chan-x this.page_type = "catalog"; } else if (doc_el.querySelector("#search-box")) { this.page_type = "board"; } else { this.page_type = "thread"; } } } else { // Index if (doc_el.querySelector("#content>#threads")) { this.page_type = "catalog"; } else if (doc_el.querySelector("#doc")) { this.page_type = "error"; } else { // Image or video var n = doc_el.querySelectorAll("body>*"); if (n.length === 1) { n = n[0]; if (n.tagName === "IMG") { this.page_type = "image"; } else if (n === "VIDEO") { this.page_type = "video"; } } } } // Trigger event trigger.call(this, "page_type_detected", { page_type: this.page_type }); }; var detect_scripts = function (doc_el) { // Detect 4chan-x / appchan-x if (!this.is_4chanx) { this.is_4chanx = style.has_class(doc_el, "fourchan-x"); if (this.is_4chanx) { this.is_appchanx = style.has_class(doc_el, "appchan-x"); // Detected trigger.call(this, "script_detected", { name: "4chan-x" }); } } }; var trigger = function (event, data) { // Trigger if (event in this.events) { var e_list = this.events[event]; for (var i = 0; i < e_list.length; ++i) { e_list[i].call(this, data); } } }; var trigger_menu_4chanx_open = function (element, type) { // Find type var type_checks = { "main": { classes: [ "settings-link" , "image-expansion-link" , "gallery-link" ], count: 0, }, "post": { classes: [ "report-link" , "hide-reply-link" , "delete-link" ], count: 0, }, }, c = element.firstChild, t, tc, i; if (type === null) { type = "other"; while (c) { // Class if (style.has_class(c, "entry")) { // Check for class types for (t in type_checks) { tc = type_checks[t]; for (i = 0; i < tc.classes.length; ++i) { if (style.has_class(c, tc.classes[i])) { ++tc.count; tc = null; break; } } // Done if (!tc) break; } } // Next c = c.nextSibling; } // Check type i = 0; for (t in type_checks) { if (type_checks[t].count > i) { i = type_checks[t].count; type = t; } } } // Event this.menu_4chanx_container = element; trigger.call(this, "menu_4chanx_open", { container: element, type: type }); }; var trigger_menu_4chanx_close = function (element) { // Event this.menu_4chanx_container = null; trigger.call(this, "menu_4chanx_close", { container: element, }); }; var trigger_settings_4chanx_open = function (element) { // Open this.settings_4chanx_container = element; // Event trigger.call(this, "settings_4chanx_open", { container: element }); // Settings observer if (this.settings_observer !== null) { this.settings_observer.disconnect(); this.settings_observer = null; } var section = element.querySelector("section"); if (section) { this.settings_observer = new MutationObserver(on_settings_observe.bind(this)); // Observe this.settings_observer.observe( section, { attributes: true, } ); // Section change trigger.call(this, "settings_4chanx_section_change", { container: element, section: section }); } }; var trigger_settings_4chanx_close = function (element) { // Close this.settings_4chanx_container = null; if (this.settings_observer !== null) { this.settings_observer.disconnect(); this.settings_observer = null; } // Event trigger.call(this, "settings_4chanx_close", { container: element }); }; var trigger_settings_vanilla_open = function (element) { // Open this.settings_vanilla_container = element; // Event trigger.call(this, "settings_vanilla_open", { container: element }); }; var trigger_settings_vanilla_close = function (element) { // Close this.settings_vanilla_container = null; // Event trigger.call(this, "settings_vanilla_close", { container: element }); }; var get_all_posts = function () { // Acquire all .postContainer's var elements = null, e; var container = document.getElementById("delform"); if (container) { elements = Array.prototype.slice.call(container.querySelectorAll(".postContainer"), 0); } container = document.getElementById("hoverUI"); if (container) { e = Array.prototype.slice.call(container.querySelectorAll(".postContainer"), 0); if (elements) elements = elements.concat(e); else elements = e; } // Return return (elements === null ? [] : elements); }; var deep_dom_wrap = (function () { // Internal helper class var Offset = function (text_offset, node) { this.text_offset = text_offset; this.node = node; this.node_text_length = node.nodeValue.length; }; // Main function var deep_dom_wrap = function (container, tag, matcher, element_checker, setup_function, quick) { var text = "", offsets = [], d = document, count = 0, match_pos = 0, node, par, next, check, match, pos_start, pos_end, offset_start, offset_end, prefix, suffix, link_base, link_node, relative_node, relative_par, clone, i, n1, n2, len, offset_current, offset_node; // Create a string of the container's contents (similar to but not exactly the same as node.textContent) // Also lists all text nodes into the offsets array par = container; node = container.firstChild; if (node === null) return 0; // Quick exit for empty container while (true) { if (node !== null) { if (node.nodeType === 3) { // TEXT_NODE // Add to list and text offsets.push(new Offset(text.length, node)); text += node.nodeValue; } else if (node.nodeType === 1) { // ELEMENT_NODE // Action callback check = element_checker.call(null, node); // Line break if ((check & deep_dom_wrap.EL_TYPE_LINE_BREAK) !== 0) { text += "\n"; } // Parse if ((check & deep_dom_wrap.EL_TYPE_NO_PARSE) === 0) { par = node; node = node.firstChild; continue; } } // Next node = node.nextSibling; } else { // Done? if (par === container) break; // Move up node = par; par = node.parentNode; node = node.nextSibling; } } // Quick mode: just find all the matches if (quick) { // Match the text match = matcher.call(null, text, match_pos); if (match === null) return count; ++count; match_pos = match[1]; } // Loop to find all links while (true) { // Match the text match = matcher.call(null, text, match_pos); if (match === null) break; ++count; // Find the beginning and ending text nodes pos_start = match[0]; pos_end = match[1]; for (offset_start = 1; offset_start < offsets.length; ++offset_start) { if (offsets[offset_start].text_offset > pos_start) break; } for (offset_end = offset_start; offset_end < offsets.length; ++offset_end) { if (offsets[offset_end].text_offset > pos_end) break; } --offset_start; --offset_end; // Vars to create the link prefix = text.substr(offsets[offset_start].text_offset, pos_start - offsets[offset_start].text_offset); suffix = text.substr(pos_end, offsets[offset_end].text_offset + offsets[offset_end].node_text_length - pos_end); link_base = d.createElement(tag); link_node = link_base; relative_node = null; // Prefix update i = offset_start; offset_current = offsets[i]; offset_node = offset_current.node; if (prefix.length > 0) { // Insert prefix n1 = d.createTextNode(prefix); offset_node.parentNode.insertBefore(n1, offset_node); // Update text offset_node.nodeValue = offset_node.nodeValue.substr(prefix.length); // Set first relative relative_node = n1; relative_par = n1.parentNode; // Update offset for next search len = prefix.length; offset_current.text_offset += len; offset_current.node_text_length -= len; } else { // Set first relative relative_node = offset_node.previousSibling; relative_par = offset_node.parentNode; } // Loop over ELEMENT_NODEs; add TEXT_NODEs to the link, remove empty nodes where necessary // The only reason the par variable is necessary is because some nodes are removed during this process for (; i < offset_end; ++i) { // Next node = offsets[i].node; next = node.nextSibling; par = node.parentNode; // Add text link_node.appendChild(node); // Node loop while (true) { if (next) { if (next.nodeType == 3) { // TEXT_NODE // Done break; } else if (next.nodeType == 1) { // ELEMENT_NODE // Deeper node = next; next = node.firstChild; par = node; // Update link node clone = node.cloneNode(false); link_node.appendChild(clone); link_node = clone; continue; } else { // Some other node type; continue anyway node = next; next = node.nextSibling; // Update link node link_node.appendChild(node); continue; } } // Shallower node = par; next = node.nextSibling; par = node.parentNode; if (node.firstChild === null) par.removeChild(node); // Update link node if (link_node !== link_base) { // Simply move up tree (link_node still has a parent) link_node = link_node.parentNode; } else { // Create a new wrapper node (link_node has no parent; it's the link_base) clone = node.cloneNode(false); for (n1 = link_base.firstChild; n1; n1 = n2) { n2 = n1.nextSibling; clone.appendChild(n1); } link_base.appendChild(clone); link_node = link_base; // Placement relatives relative_node = (next !== null) ? next.previousSibling : null; relative_par = par; } } } // Suffix update offset_current = offsets[i]; offset_node = offset_current.node; if (suffix.length > 0) { // Insert suffix n1 = d.createTextNode(suffix); if ((n2 = offset_node.nextSibling) !== null) { offset_node.parentNode.insertBefore(n1, n2); } else { offset_node.parentNode.appendChild(n1); } // Update text len = offset_node.nodeValue.length - suffix.length; offset_node.nodeValue = offset_node.nodeValue.substr(0, len); // Update offset for next search offset_current.text_length += len; offset_current.node_text_length -= len; offset_current.node = n1; } // Add the last segment par = offset_node.parentNode; link_node.appendChild(offset_node); // Setup function if (setup_function !== null) setup_function.call(null, link_base, match); // Find the proper relative node relative_node = (relative_node !== null) ? relative_node.nextSibling : relative_par.firstChild; // Insert link if (relative_node !== null) { // Insert before it relative_par.insertBefore(link_base, relative_node); } else { // Add to end relative_par.appendChild(link_base); } // Remove empty suffix tags while (par.firstChild === null) { node = par; par = par.parentNode; par.removeChild(node); } // Update match position offsets[offset_end].text_offset = pos_end; match_pos = pos_end; } // Done return count; }; // Element type constants deep_dom_wrap.EL_TYPE_PARSE = 0; deep_dom_wrap.EL_TYPE_NO_PARSE = 1; deep_dom_wrap.EL_TYPE_LINE_BREAK = 2; // Return the function return deep_dom_wrap; })(); var deep_dom_wrap_filter = function (node) { if (node.tagName === "BR" || node.tagName === "A") { return deep_dom_wrap.EL_TYPE_NO_PARSE | deep_dom_wrap.EL_TYPE_LINE_BREAK; } else if (node.tagName === "WBR") { return deep_dom_wrap.EL_TYPE_NO_PARSE; } else if (node.tagName === "DIV") { if (style.has_class(node, "inline") || style.has_class(node, "inlined")) return deep_dom_wrap.EL_TYPE_NO_PARSE | deep_dom_wrap.EL_TYPE_LINE_BREAK; return deep_dom_wrap.EL_TYPE_LINE_BREAK; } return deep_dom_wrap.EL_TYPE_PARSE; }; var deep_dom_wrap_filter_simple = function (node) { if (node.tagName === "BR") { return deep_dom_wrap.EL_TYPE_NO_PARSE | deep_dom_wrap.EL_TYPE_LINE_BREAK; } else if (node.tagName === "WBR") { return deep_dom_wrap.EL_TYPE_NO_PARSE; } else if (node.tagName === "DIV") { if (style.has_class(node, "inline") || style.has_class(node, "inlined")) return deep_dom_wrap.EL_TYPE_NO_PARSE | deep_dom_wrap.EL_TYPE_LINE_BREAK; return deep_dom_wrap.EL_TYPE_LINE_BREAK; } return deep_dom_wrap.EL_TYPE_PARSE; }; API.prototype = { constructor: API, destroy: function () { // Remove event listeners if (this.doc_observer !== null) { this.doc_observer.disconnect(); this.doc_observer = null; } if (this.body_observer !== null) { this.body_observer.disconnect(); this.body_observer = null; } if (this.settings_observer !== null) { this.settings_observer.disconnect(); this.settings_observer = null; } if (this.hover_ui_observer !== null) { this.hover_ui_observer.disconnect(); this.hover_ui_observer = null; } if (this.delform_observer !== null) { this.delform_observer.disconnect(); this.delform_observer = null; } }, setup: function () { // Check for other addons hook_document_observer.call(this); on_ready(on_asap.bind(this)); }, on: function (event, callback) { // Add callback if (event in this.events) { this.events[event].push(callback); } }, off: function (event, callback) { // Add callback if (event in this.events) { var e_list = this.events[event]; for (var i = 0; i < e_list.length; ++i) { if (e_list[i] === callback) { // Remove e_list.splice(i, 1); return true; } } } return false; }, get: function (getter) { if (getter in this.getters) { return this.getters[getter].call(this); } return null; }, settings_4chanx_open: function (tab) { // Signal 4chan-x to open settings document.dispatchEvent(new CustomEvent("OpenSettings", { detail: tab || "Main" })); }, settings_4chanx_close: function () { if (!this.settings_4chanx_container) return; // Click the overlay var overlay = document.getElementById("overlay"); if (overlay) { overlay.click(); } }, settings_4chanx_change_section: function (section) { if (!this.settings_4chanx_container) return; section = section.replace(/\s+/g, "-").toLowerCase(); var target_tab = this.settings_4chanx_container.querySelector(".tab-" + section); if (target_tab) { target_tab.click(); } }, settings_4chanx_is_open: function () { return (this.settings_4chanx_container !== null); }, settings_vanilla_open: function () { var settings_link = document.getElementById("settingsWindowLink"); if (settings_link) { settings_link.click(); } }, settings_vanilla_close: function () { if (!this.settings_vanilla_container) return; var button = this.settings_vanilla_container.querySelector(".panelHeader>span>.pointer"); if (button) { button.click(); } }, settings_vanilla_save: function () { if (!this.settings_vanilla_container) return; var button = this.settings_vanilla_container.querySelector("button[data-cmd='settings-save']"); if (button) { button.click(); } }, settings_vanilla_is_open: function () { return (this.settings_vanilla_container !== null); }, menu_4chanx_close: function () { if (!this.menu_4chanx_container) return; var par = this.menu_4chanx_container.parentNode; if (par) par.click(); }, menu_4chanx_is_open: function () { return (this.menu_4chanx_container !== null); }, menu_4chanx_add_entry: function (menu_container, element, order) { // Styling style.add_class(element, "entry"); element.style.order = (order || 0); // Events var e_enter = wrap_mouseenterleave_event(this, on_menu_4chanx_entry_mouseenter), e_leave = wrap_mouseenterleave_event(this, on_menu_4chanx_entry_mouseleave); element.addEventListener("mouseover", e_enter, false); element.addEventListener("mouseout", e_leave, false); // Add menu_container.appendChild(element); }, notification: function (type, content, lifetime, callback) { // Setup var detail = { type: type, content: content, }; if (arguments.length >= 3) { detail.lifetime = lifetime; } if (arguments.length >= 4) { detail.cb = callback; } // Display document.dispatchEvent(new CustomEvent("CreateNotification", { detail: detail })); }, post_is_image_expanded_or_expanding: function (post_container) { var n_file, n_img; // Expanded if (this.is_4chanx) { if (style.has_class(post_container, "expanded-image")) return true; // Get the .file container n_file = this.post_get_file_info_container(post_container); // Find image n_img = n_file.querySelector("img"); return (n_img && style.has_class(n_img, "expanding")); } else { // Get the .file container n_file = this.post_get_file_info_container(post_container); // Return return (style.has_class(n_file, "image-expanded") || n_file.querySelector("video.expandedWebm") !== null); } }, post_get_file_info: function (post_container) { return this.post_get_file_info_from_file_info_container(this.post_get_file_info_container(post_container)); }, post_get_file_info_from_file_info_container: function (file_info) { // Setup info var info = { url: "", thumb: null, spoiler: null, name: "", resolution: { width: 0, height: 0 }, resolution_thumb: { width: 0, height: 0 }, size: 0, size_label: "", }; // Nodes var n, m, n_p, style_str; if (file_info) { // Get info if ((n_p = file_info.querySelector(".file-info"))) { // 4chan-x if ((n = n_p.querySelector("a"))) { info.url = n.getAttribute("href") || ""; var n2 = n.firstChild; if (n2 !== null && n2.className && style.has_class(n2, "fnswitch") && (n2 = n2.querySelector(".fnfull")) !== null) { info.name = n2.textContent.trim(); } else { info.name = n.textContent.trim(); } } if ((n = n_p.lastChild)) { m = /([0-9\.]+)\s*(\w?b),(?:\s*([0-9]+)x([0-9]+))?/i.exec((n.textContent || "")); if (m) { info.size = parseFloat(m[1]); info.size_label = m[2].toLowerCase(); if (m[3] !== undefined) { info.resolution.width = parseInt(m[3], 10); info.resolution.height = parseInt(m[4], 10); } } } } else if ((n_p = file_info.querySelector(".fileText"))) { // Vanilla if ((n = n_p.querySelector("a"))) { info.url = n.getAttribute("href") || ""; // File name info.name = n_p.getAttribute("title") || n.getAttribute("title") || n.textContent || ""; // Attributes if ((n = n.nextSibling)) { m = /([0-9\.]+)\s*(\w?b),(?:\s*([0-9]+)x([0-9]+))?/i.exec((n.textContent || "")); if (m) { info.size = parseFloat(m[1]); info.size_label = m[2].toLowerCase(); if (m[3] !== undefined) { info.resolution.width = parseInt(m[3], 10); info.resolution.height = parseInt(m[4], 10); } } } } } // Thumbnail and spoiler status if ((n_p = file_info.querySelector(".fileThumb"))) { if ((n = n_p.querySelector("img:not(.full-image):not(.expanded-thumb)"))) { m = n.getAttribute("src"); if (style.has_class(n_p, "imgspoiler")) { info.spoiler = m; // Assume thumbnail if (info.url) { info.thumb = info.url.replace(/\/\/(.*)i\.4cdn\.org/g, "//t.4cdn.org").replace(/\.([^\.]*)$/, "s.jpg"); } } else { info.thumb = m; } // Resolution style_str = n.getAttribute("style"); m = /width\s*\:\s*([0-9\.]+)px/i.exec(style_str); if (m) { info.resolution_thumb.width = parseFloat(m[1]) || 0; } m = /height\s*\:\s*([0-9\.]+)px/i.exec(style_str); if (m) { info.resolution_thumb.height = parseFloat(m[1]) || 0; } } } } // Info return info; }, get_post_container_from_id: function (id) { return document.getElementById("pc" + id); }, post_get_file_info_container: function (post_container) { // Post body var node = post_container.querySelector(".post"); if (node && (node = node.firstChild)) { while (true) { // File container if (style.has_class(node, "file")) return node; // Next if (!(node = node.nextSibling)) return null; } } return null; }, post_get_image_container: function (post_container) { // Post body var node = post_container.querySelector(".post"); if (node && (node = node.firstChild)) { while (true) { // File container if (style.has_class(node, "file")) { // Get thumbnail container return node.querySelector(".fileThumb"); } // Next if (!(node = node.nextSibling)) return null; } } return null; }, post_get_post_container_from_image_container: function (image_container) { // Parent check var n = image_container; while ((n = n.parentNode) && !style.has_class(n, "postContainer")); return n; }, post_get_image_expanded_from_image_container: function (image_container) { if (this.is_4chanx) { return image_container.querySelector(".full-image"); } else { var n = image_container.querySelector("img.expanded-thumb"); if (n !== null) return n; if (image_container.parentNode === null) return null; return image_container.parentNode.querySelector("video.expandedWebm"); } }, post_get_file_nodes: function (post_container) { var post = post_container.querySelector(".post"), nodes, node; if (post !== null) { // File info container if ((node = post.firstChild)) { while (true) { // File container if (style.has_class(node, "file")) { nodes = { container: node, info_container: node.querySelector(".fileText,.file-info"), link_thumbnail: node.querySelector(".fileThumb"), link: null, }; if (nodes.info_container !== null) { nodes.link = node.querySelector("a"); } return nodes; } // Next if (!(node = node.nextSibling)) break; } } } return null; }, post_has_file: function (post_container) { var post = post_container.querySelector(".post"), node; if (post !== null && (node = post.firstChild)) { while (true) { // File container if (style.has_class(node, "file")) return true; // Next if (!(node = node.nextSibling)) break; } } return false; }, post_get_file_node_link_from_file_info: function (file_info) { return file_info.querySelector("a"); }, post_get_file_name_from_file_info: function (file_info) { var n = file_info.parentNode; if (n !== null && (n = n.parentNode) !== null) { } return null; }, post_get_file_container_from_file_info: function (file_info) { var n = file_info.parentNode; if (n !== null && (n = n.parentNode) !== null) { if (style.has_class(n, "file")) return n; } return null; }, post_get_nodes: function (post_container) { var post = post_container.querySelector(".post"), nodes, node, node2; if (post === null) return null; nodes = { subject: null, name: null, tripcode: null, date: null, no: null, number: null, comment: null, }; // Post info if ((node = post.querySelector(".postInfo"))) { nodes.subject = node.querySelector(".subject"); nodes.name = node.querySelector(".name"); nodes.tripcode = node.querySelector(".posterTrip"); nodes.date = node.querySelector(".dateTime"); if ((node2 = node.querySelectorAll(".postNum>a")).length >= 2) { nodes.no = node2[0]; nodes.number = node2[1]; } } // Comment for (node = post.lastChild; node; node = node.previousSibling) { if (style.has_class(node, "postMessage")) { nodes.comment = node; break; } } return nodes; }, post_get_id: function (post_container) { var id = post_container.getAttribute("id"); return id ? parseInt(id.replace(/[^0-9]+/g, ""), 10) : -1; }, post_get_id_from_node: function (node) { // Parent check while (true) { if (node === null) return -1; if (style.has_class(node, "postContainer")) return this.post_get_id(node); node = node.parentNode; } }, post_get_comment_container: function (post_container) { var post = post_container.querySelector(".post"), comment; if (post === null) return null; // Find comment comment = post.lastChild; while (true) { if (comment === null) return 0; if (style.has_class(comment, "postMessage")) break; comment = comment.previousSibling; } return comment; }, post_get_quotelinks: function (post_container) { return this.post_query_selector_all(post_container, ".quotelink"); }, post_comment_scan: function (post_container, regex, tag, setup_function) { var comment = this.post_get_comment_container(post_container), s_fn = null, m_fn; if (comment === null) return 0; // Matching function if (typeof(regex) === "function") { m_fn = regex; } else { if (!regex.global) return 0; m_fn = function (text, pos) { regex.lastIndex = pos; var m = regex.exec(text); if (m === null) return null; return [ m.index , m.index + m[0].length, m ]; }; } // Scanning function if (setup_function) { s_fn = setup_function; //s_fn = function (node, text) { setup_function(node, text); }; } return deep_dom_wrap(comment, tag, m_fn, tag ? deep_dom_wrap_filter : deep_dom_wrap_filter_simple, s_fn, (!tag) ); }, post_get_comment_text: function (post_container) { var post_text = "", comment = this.post_get_comment_container(post_container); if (comment !== null) { deep_dom_wrap(comment, null, function (text) { post_text = text; return null; }, deep_dom_wrap_filter_simple, null, true ); } return post_text; }, post_query_selector_all: function (post_container, selector) { var nodes = post_container.querySelectorAll(selector), other, other_len, i, first; if (nodes.length > 0 && (other_len = (other = post_container.querySelectorAll("div.inline,div.inlined")).length) > 0) { // Filter var filter = function (node) { while ((node = node.parentNode) !== post_container) { for (var i = 0; i < other_len; ++i) { if (other[i] === node) return true; } } return false; }; // Filter out nodes first = true; for (i = 0; i < nodes.length; ++i) { if (filter(nodes[i])) { if (first) { first = false; nodes = Array.prototype.slice.call(nodes, 0); } nodes.splice(i, 1); continue; } } } return nodes; }, post_is_floating_or_embedded: function (post_container) { var p = post_container.parentNode; if (!p) return false; return ( style.has_class(p, "inline") || // 4chan x style.has_class(post_container, "inlined") || // vanilla p.id === "qp" || // 4chan x p === document.body || // vanilla p.id === "hoverUI" // 4chan x (?) ); }, get_quotelink_target: function (node) { return parseInt(node.textContent.replace(/[^0-9]+/g, ""), 10) || 0; }, get_header_rect: function () { var header = document.getElementById(this.is_4chanx ? "header-bar" : "boardNavMobile"); if (header) { return style.get_object_rect(header); } else { return { left: 0, top: 0, right: 0, bottom: 0, }; } }, observe_children: function (parent_node, cb) { // Create new observer var o = new MutationObserver(function (records) { var i, j, nodes; for (i = 0; i < records.length; ++i) { if ((nodes = records[i].addedNodes)) { for (j = 0; j < nodes.length; ++j) { // Check cb.call(null, true, nodes[j]); } } if ((nodes = records[i].removedNodes)) { for (j = 0; j < nodes.length; ++j) { // Check cb.call(null, false, nodes[j]); } } } }); // Observe o.observe( parent_node, { childList: true, } ); // Done return function () { o.disconnect(); }; }, get_post_container_from_post_number: function (post_number) { return document.getElementById("pc" + post_number); }, get_quick_reply_file_4chanx: function (callback) { // Event listener var temp_listener = function (event) { clearTimeout(timer); document.removeEventListener("QRFile", temp_listener, false); if (event.detail && event.detail instanceof File) { callback.call(event.detail); } else { callback.call(null); } }; // Events document.addEventListener("QRFile", temp_listener, false); document.dispatchEvent(new CustomEvent("QRGetFile", { bubbles: true, detail: null })); // Cancel event var timer = setTimeout(function () { timer = null; document.removeEventListener("QRFile", temp_listener, false); }, 1000); }, set_quick_reply_file_4chanx: function (file) { var detail = { file: this.files[0], name: null, }; if (cloneInto) { detail = cloneInto(detail, document.defaultView); } document.dispatchEvent(new CustomEvent("QRSetFile", { bubbles: true, detail: detail })); }, }; return API; })(); // Notification var Notification = (function () { var Notification = function (data) { this.nodes = null; this.overlay_can_close_on_click = ("overlay_close" in data && data.overlay_close); this.on_overlay_click_bind = null; this.on_body_click_bind = null; this.on_close_click_bind = null; this.events = { "close": [] }; var body = document.querySelector("body"); if (body) { this.nodes = { container: null, body: null, close: null }; // Create notification container var container, n_body_outer, n_aligner, n_body_inner, n_body, n_content_outer, n_content, n_close, n_style, nc_title, nc_body; container = $.div("iex_notification"); container.addEventListener("click", this.on_overlay_click_bind = on_overlay_click.bind(this), false); this.nodes.container = container; n_body_outer = $.div("iex_notification_body_outer"); container.appendChild(n_body_outer); n_aligner = $.div("iex_notification_body_aligner"); n_body_outer.appendChild(n_aligner); n_body_inner = $.div("iex_notification_body_inner"); n_body_outer.appendChild(n_body_inner); if ("style" in data && [ "info" , "error" , "success" , "warning" ].indexOf(data.style) >= 0) { n_style = "iex_notification_" + data.style; } else { n_style = "iex_notification_info"; } n_body = $.div("iex_notification_body " + n_style); n_body.addEventListener("click", this.on_body_click_bind = on_body_click.bind(this), false); n_body_inner.appendChild(n_body); this.nodes.body = n_body; n_content_outer = $.div("iex_notification_content_outer"); n_body.appendChild(n_content_outer); n_close = $.a("iex_notification_close" + (data.close === false ? " iex_notification_close_hidden" : "")); n_close.textContent = "\u00D7"; n_close.addEventListener("click", this.on_close_click_bind = on_close_click.bind(this), false); this.nodes.close = n_close; n_content_outer.appendChild(n_close); n_content = $.div("iex_notification_content"); n_content_outer.appendChild(n_content); // Title if ("title" in data) { nc_title = $.div("iex_notification_content_title"); if (typeof(data.title) == typeof("")) { nc_title.textContent = data.title; } else { try { nc_title.appendChild(data.title); } catch (e) {} } n_content.appendChild(nc_title); } // Content if ("content" in data) { nc_body = $.div("iex_notification_content_body"); if (typeof(data.content) == typeof("")) { nc_body.textContent = data.content; } else { try { nc_body.appendChild(data.content); } catch (e) {} } n_content.appendChild(nc_body); } // Insert body.appendChild(container); // Events if ("on" in data) { for (var event in data.on) { this.on(event, data.on[event]); } } } }; var on_overlay_click = function (event) { // Close if (this.overlay_can_close_on_click) { destroy.call(this, "overlay"); } // Stop return stop_event(event); }; var on_body_click = function (event) { // Stop event.stopPropagation(); return false; }; var on_close_click = function (event) { // Remove destroy.call(this, "x"); // Stop return stop_event(event); }; var destroy = function (reason) { if (this.nodes) { // Close trigger.call(this, "close", { reason: reason }); // Remove events if (this.on_close_click_bind !== null) { this.nodes.close.removeEventListener("click", this.on_close_click_bind, false); this.on_close_click_bind = null; } if (this.on_overlay_click_bind !== null) { this.nodes.container.removeEventListener("click", this.on_overlay_click_bind, false); this.on_overlay_click_bind = null; } if (this.on_body_click_bind !== null) { this.nodes.body.removeEventListener("click", this.on_body_click_bind, false); this.on_body_click_bind = null; } // Remove var par = this.nodes.container.parentNode; if (par) { par.removeChild(this.nodes.container); } // Null this.nodes = null; this.events = {}; } }; var trigger = function (event, data) { // Trigger if (event in this.events) { var e_list = this.events[event]; for (var i = 0; i < e_list.length; ++i) { e_list[i].call(this, data); } } }; Notification.prototype = { constructor: Notification, close: function () { // Close destroy.call(this, "manual"); }, on: function (event, callback) { // Add callback if (event in this.events) { this.events[event].push(callback); } }, off: function (event, callback) { // Add callback if (event in this.events) { var e_list = this.events[event]; for (var i = 0; i < e_list.length; ++i) { if (e_list[i] === callback) { // Remove e_list.splice(i, 1); return true; } } } return false; }, }; return Notification; })(); // Settings control var Settings = (function () { var Settings = function () { // Settings values this.values = { "first_run": true, "image_expansion": { "enabled": false, "normal": { "enabled": true, "timeout": 0.0, "to_fit": true }, "spoiler": { "enabled": true, "timeout": 0.0, "to_fit": true }, "hover": { "zoom_invert": false, "zoom_borders_show": true, "zoom_borders_hide_time": 0.5, "zoom_buttons": true, "mouse_hide": true, "mouse_hide_time": 1.0, "header_overlap": false, "fit_large_allowed": true, "display_stats": 0 }, "extensions": { "jpg": { "background": 1, // 0 = never show, 1 = show before load (transparent), 2 = show before load (opaque) "mouse_wheel": 0 // 0 = zoom, 1 = volume control }, "png": { "background": 1, "mouse_wheel": 0 }, "gif": { "background": 1, "mouse_wheel": 0 }, "webm": { "background": 2, "mouse_wheel": 0 }, }, "video": { "autoplay": 1, // 0 = not at all, 1 = asap, 2 = when it can play "through", 3 = fully loaded "loop": true, "mute_initially": false, "volume": 0.5, "mini_controls": 1, // 0 = never, 1 = when mouse is over the video, 2 = when mouse is NOT over the video, 3 = always "expand_state_save": true }, "style": { "animations_background": true, "controls_rounded_border": true } }, "file_linkification": { "enabled": true, }, "annotations": { "enabled": true, "enabled_standalone": true, "editor": true, "editor_always_enable": false, "modify_urls": true, "transparent_until_hover": false, "toggle_hotkey": "a", "defaults": { "font": 1, "bold": false, "italic": false, }, }, }; this.save_key = "iex_settings"; // Value changing events this.change_events = {}; // Events this.ready_callbacks = []; // Notification instances this.notification_first_run = null; this.notification_install_fail = null; this.notification_image_hover = null; // Modifying other settings this.modifying_settings_4chanx_timer = null; this.modifying_settings_4chanx_callback = null; this.modifying_settings_vanilla_timer = null; this.modifying_settings_vanilla_callback = null; this.can_disable_image_hover = true; // If another image hover is detected // Data this.settings_container = null; this.settings_container_outer = null; this.settings_difficulty_container = null; this.settings_removal_data = null; this.settings_update_other_after_close = false; }; var on_settings_4chanx_section_change = function (data) { // Get section id var id = /section-([a-zA-Z0-9_\-]+)/.exec(data.section.className); id = id ? id[1].toLowerCase() : ""; // Hover settings var modify_image_hover = (id == (api.is_appchanx ? "script" : "main")); if (modify_image_hover) { // Forcibly modify 4chan-x settings if (this.modifying_settings_4chanx_timer !== null) { // Clear timer clearTimeout(this.modifying_settings_4chanx_timer); this.modifying_settings_4chanx_timer = null; // Get the hover setting var image_hover_setting = data.container.querySelector('input[name="Image Hover"]'); if (image_hover_setting) { setTimeout(on_settings_4chanx_section_change_modify.bind(this, image_hover_setting), 100); } else { // Callback trigger_modifying_settings_4chanx_cb.call(this, false, false, "not found"); } } // Modify display modify_settings_4chanx_display.call(this, data.container); } }; var on_settings_4chanx_section_change_modify = function (setting) { // If it's checked, click it to uncheck it var is_checked = setting.checked; if (is_checked) { setting.click(); } // Callback trigger_modifying_settings_4chanx_cb.call(this, true, is_checked, "okay"); sync.trigger("image_expansion_enable"); }; var on_settings_vanilla_open = function (data) { // Forcibly modify 4chan-x settings if (this.modifying_settings_vanilla_timer !== null) { // Clear timer clearTimeout(this.modifying_settings_vanilla_timer); this.modifying_settings_vanilla_timer = null; // Get the hover setting var image_hover_setting = data.container.querySelector("input[data-option='imageHover']"); if (image_hover_setting) { setTimeout(on_settings_vanilla_open_modify.bind(this, image_hover_setting), 10); } else { // Callback trigger_modifying_settings_vanilla_cb.call(this, false, false, "not found"); } } // Modify display modify_settings_vanilla_display.call(this, data.container); }; var on_settings_vanilla_open_modify = function (setting) { // If it's checked, click it to uncheck it var is_checked = setting.checked; if (is_checked) { setting.click(); } // Callback trigger_modifying_settings_vanilla_cb.call(this, true, is_checked, "okay"); sync.trigger("image_expansion_enable"); }; var on_modifying_settings_4chanx_timout = function () { // Callback trigger_modifying_settings_4chanx_cb.call(this, false, false, "timeout"); // Didn't happen this.modifying_settings_4chanx_timer = null; }; var on_modifying_settings_vanilla_timout = function () { // Callback trigger_modifying_settings_vanilla_cb.call(this, false, false, "timeout"); // Didn't happen this.modifying_settings_vanilla_timer = null; }; var on_menu_4chanx_open = function (data) { if (data.type != "main") return; var e = $.a("iex_4chanx_menu_link"); e.textContent = "iex settings"; e.addEventListener("click", on_menu_4chanx_entry_click.bind(this), false); api.menu_4chanx_add_entry(data.container, e, 135); }; var on_menu_4chanx_entry_click = function (event) { // Close api.menu_4chanx_close(); // Open settings this.settings_open(); }; var on_edit_iex_settings_click = function (event) { // Open settings if (api.is_4chanx) { api.settings_4chanx_close(); } else { api.settings_vanilla_close(); } this.settings_open(); // Stop event return stop_event(event); }; var on_main_page_iex_link_click = function (event) { // Open settings this.settings_open(); // Stop event return stop_event(event); }; var on_first_run_link_click = function (install, event) { // Selected this.change_value(["first_run"], false); this.save_values(); // Install or not? if (install) { // Perform install var cb = on_first_run_install_callback.bind(this); if (api.is_4chanx) { this.modify_4chanx_settings(cb); } else { this.modify_vanilla_settings(cb); } } // Close if (this.notification_first_run !== null) { this.notification_first_run.close(); this.notification_first_run = null; } // Stop event return stop_event(event); }; var on_first_run_install_callback = function (okay, settings_were_changed, status) { // Update settings if (okay) { this.change_value(["first_run"], false); this.change_value(["image_expansion", "enabled"], true); this.save_values(on_first_run_install_callback_complete.bind(this, okay, settings_were_changed, status)); } else { // Instant on_first_run_install_callback_complete.call(this, okay, settings_were_changed, status); } }; var on_first_run_install_callback_complete = function (okay, settings_were_changed, status) { // Event if (okay) { sync.trigger("install_complete"); } // Close settings if (api.is_4chanx) { // Close api.settings_4chanx_close(); } else { // Close or save if (settings_were_changed) { api.settings_vanilla_save(); } else { api.settings_vanilla_close(); } } if (okay) { // Reload window.location.reload(false); } else { // Error this.display_install_fail_notification(status); } }; var on_settings_update_other_complete = function (okay, settings_were_changed, status) { // Close settings if (api.is_4chanx) { // Close api.settings_4chanx_close(); } else { // Close or save if (settings_were_changed) { api.settings_vanilla_save(); } else { api.settings_vanilla_close(); } } if (!okay) { // Error this.display_install_fail_notification(status); } }; var on_initial_load = function () { on_ready(on_first_run_check.bind(this)); }; var on_first_run_check = function () { // First run check if (api.page_type == "board" || api.page_type == "thread" || api.page_type == "catalog") { if (this.values.first_run) { // Show message this.display_first_run_notification(); } } // Trigger ready trigger_ready.call(this); }; var on_insert_links = function () { var nav_nodes = [], nav, par, i, c, n, separate; if ((nav = document.getElementById("navtopright"))) { nav_nodes.push({ node: nav, is_parent: true, add_separators: true, before: true, }); } if ((nav = document.getElementById("navbotright"))) { nav_nodes.push({ node: nav, is_parent: true, add_separators: true, before: true, }); } if ((nav = document.getElementById("settingsWindowLinkMobile"))) { nav_nodes.push({ node: nav, is_parent: false, add_separators: false, before: true, }); } // Insert for (i = 0; i < nav_nodes.length; ++i) { par = nav_nodes[i].node; separate = nav_nodes[i].add_separators; if (nav_nodes[i].is_parent) { c = par.firstChild; } else { c = par; par = par.parentNode; if (!par) continue; } if (separate) { if (c && c.nodeType == 3) { // TEXT_NODE c.nodeValue = "] ["; } else { n = $.text("]"); if (c) par.insertBefore(n, c); else par.appendChild(n); c = n; } } n = $.node("a"); n.textContent = "iex"; n.setAttribute("target", "_blank"); n.setAttribute("rel", "noreferrer nofollow"); n.setAttribute("href", "https://dnsev.github.io/iex/"); n.addEventListener("click", on_main_page_iex_link_click.bind(this), false); par.insertBefore(n, c); c = n; if (separate) { n = $.text("["); par.insertBefore(n, c); } } }; var on_insert_links_condition = function () { return document.getElementById("navtopright") || document.getElementById("navbotright") || document.getElementById("settingsWindowLinkMobile"); }; var on_install_complete_sync = function () { // Close notifications if (this.notification_first_run !== null) { this.notification_first_run.close(); this.notification_first_run = null; } }; var on_settings_save_sync = function () { // Reload settings this.load_values(true, null); }; var on_image_expansion_enable_sync = function () { // Image hover has been enabled in another tab; make it so this tab can't re-disable it this.can_disable_image_hover = false; }; var on_notification_link_click = function (settings_open, close_object, event) { // Open if (settings_open) { this.settings_open(); } // Close if (close_object !== null) { close_object.close(); } // Stop event return stop_event(event); }; var on_notification_first_run_close = function () { this.notification_first_run = null; }; var on_notification_install_fail_close = function () { this.notification_install_fail = null; }; var on_notification_image_hover_close = function () { this.notification_image_hover = null; }; var on_iex_settings_overlay_click = function (event) { // Close this.settings_close(); // Stop event return stop_event(event); }; var on_iex_settings_body_click = function (event) { // Stop event event.stopPropagation(); return false; }; var on_iex_settings_close_click = function (event) { // Close this.settings_close(); // Stop event return stop_event(event); }; var on_iex_setting_checkbox_change = function (node, descriptor, event) { // Set var value_new = node.checked, value_old; if ("values" in descriptor) { value_new = descriptor.values[value_new ? 1 : 0]; } if ("modify" in descriptor) { value_new = descriptor.modify.call(this, value_new, descriptor); } // Value tree if ("tree" in descriptor) { // Change value_old = this.change_value(descriptor.tree, value_new); this.save_values(); } else if ("change" in descriptor) { // Modify value_new = descriptor.change.call(this, value_new, value_old); } // Update node node.checked = value_new; // After if ("after_callback" in descriptor) { descriptor.after_callback.call(this, value_new, value_old); } }; var on_iex_setting_textbox_change = function (node, descriptor, event) { // Set var value_new = node.value, value_old; if ("modify" in descriptor) { value_new = descriptor.modify.call(this, value_new, descriptor); } // Value tree if ("tree" in descriptor) { // Change value_old = this.change_value(descriptor.tree, value_new); this.save_values(); } else if ("change" in descriptor) { // Modify value_new = descriptor.change.call(this, value_new, value_old); } // Update node node.value = "" + value_new; // After if ("after_callback" in descriptor) { descriptor.after_callback.call(this, value_new, value_old); } }; var on_iex_setting_text_click = function (node, descriptor, event) { // Get new id var id_current = parseInt(node.getAttribute("data-iex-setting-current-id") || "", 10) || 0; var id_new = (id_current + 1) % descriptor.values.length; var update_id = false; // Set var value_new = descriptor.values[id_new], value_old; if ("modify" in descriptor) { value_new = descriptor.modify.call(this, value_new, descriptor); update_id = true; } // Value tree if ("tree" in descriptor) { // Change value_old = this.change_value(descriptor.tree, value_new); this.save_values(); } else if ("change" in descriptor) { // Modify value_new = descriptor.change.call(this, value_new, value_old); update_id = true; } // Update if (update_id) { var i = id_new; while (true) { i = (i + 1) % descriptor.values.length; if (i == id_new) break; if (descriptor.values[i] === value_new) { id_new = i; break; } } } node.setAttribute("data-iex-setting-current-id", id_new); node.textContent = ("value_labels" in descriptor ? descriptor.value_labels[id_new] : ("" + descriptor.values[id_new])); // After if ("after_callback" in descriptor) { descriptor.after_callback.call(this, value_new, value_old); } }; var on_iex_setting_display_settings = function () { // Close settings this.settings_close(); // Display this.display_settings_info_notification(); }; var on_iex_setting_delete_settings = function () { // Delete this.delete_values(on_iex_setting_delete_settings_complete.bind(this)); }; var on_iex_setting_delete_settings_complete = function () { // Reload this.settings_close(); window.location.reload(false); }; var on_iex_difficulty_link_click = function (event, node) { // Get target var target = node.getAttribute("data-iex-settings-difficulty-choice-level") || ""; // Update difficulty change_settings_difficulty.call(this, target); // Stop event return stop_event(event); }; var on_iex_setting_image_expansion_toggle = function (value_new, value_old) { // Update other settings after close this.settings_update_other_after_close = value_new; }; var on_save_values_callback = function (next_callback) { // Sync sync.trigger("settings_save"); if (next_callback) { next_callback.call(this); } }; var on_load_values_callback = function (events, next_callback, value) { if (value) { update_values.call(this, [], this.values, value, false, events); } if (next_callback) { next_callback.call(this); } }; var on_display_settings_info_notification_callback = function (saved_values) { // Get try { saved_values = JSON.stringify(saved_values, null, " "); } catch (e) {} // Alert message var message = $.node("div"), m_part; // Line 1 m_part = $.div("iex_spaced_div"); m_part.textContent = "These are iex's saved settings, which are useful for debugging purposes:"; message.appendChild(m_part); // Textarea m_part = $.node("textarea", "iex_notification_textarea"); m_part.setAttribute("spellcheck", "false"); m_part.value = saved_values; message.appendChild(m_part); // Display new Notification({ close: true, overlay_close: true, style: "success", title: "Saved iex settings", content: message, }); }; var modify_settings_4chanx_display = function (container) { // Get the hover setting var image_hover_setting = container.querySelector('input[name="Image Hover"]'); if (!image_hover_setting) return; // Find the parent to clone var image_hover_setting_container; try { image_hover_setting_container = image_hover_setting.parentNode.parentNode; } catch (e) { image_hover_setting_container = null; } if (image_hover_setting_container) { // Clone container var container_clone = image_hover_setting_container.cloneNode(true), i, description_old; // Remove input names and disable var inputs = container_clone.querySelectorAll("input"); for (i = 0; i < inputs.length; ++i) { inputs[i].disabled = true; inputs[i].removeAttribute("name"); } // Dim labels and descriptions var labels = container_clone.querySelectorAll("label"); for (i = 0; i < labels.length; ++i) { labels[i].style.opacity = "0.5"; } var descriptions = container_clone.querySelectorAll(".description"); for (i = 0; i < descriptions.length; ++i) { descriptions[i].style.opacity = "0.5"; } // Add new if (api.is_appchanx) { container_clone.appendChild($.text(" (disable iex)")); // Modify old description_old = image_hover_setting_container.querySelector("label"); if (description_old) { description_old.appendChild($.text(" (or use iex)")); } } else { var description_new = $.span("description"); var link_new = $("a"); link_new.setAttribute("href", "//dnsev.github.io/iex/"); link_new.setAttribute("target", "_blank"); link_new.textContent = "disable iex version to use"; link_new.addEventListener("click", on_edit_iex_settings_click.bind(this), false); description_new.appendChild($.text(" (")); description_new.appendChild(link_new); description_new.appendChild($.text(")")); container_clone.appendChild(description_new); // Modify old description_old = image_hover_setting_container.querySelector(".description"); if (description_old) { link_new = $("a"); link_new.setAttribute("href", "//dnsev.github.io/iex/"); link_new.setAttribute("target", "_blank"); link_new.textContent = "or use iex version"; link_new.addEventListener("click", on_edit_iex_settings_click.bind(this), false); description_old.appendChild($.text(" (")); description_old.appendChild(link_new); description_old.appendChild($.text(")")); } } // Insert new container image_hover_setting_container.parentNode.insertBefore(container_clone, image_hover_setting_container); // Hide old if (this.values.image_expansion.enabled) { image_hover_setting_container.style.display = "none"; } else { container_clone.style.display = "none"; } // Update checkbox value setTimeout(modify_settings_4chanx_display_delay.bind(this, inputs, image_hover_setting), 10); } }; var modify_settings_4chanx_display_delay = function (inputs, original) { for (var i = 0; i < inputs.length; ++i) { inputs[i].checked = original.checked; } }; var modify_settings_vanilla_display = function (container) { // Get the hover setting var image_hover_setting = container.querySelector("input[data-option='imageHover']"); if (!image_hover_setting) return; // Acquire nodes to clone var c, containers_original = [], clone, inputs, par, i, j, n, iex_link, iex_descr; try { c = image_hover_setting.parentNode.parentNode; if (c) { if ((par = c.parentNode)) { containers_original.push(c); } else { c = null; } } } catch (e) { c = null; } if (c) { c = c.nextSibling; if (style.has_class(c, "settings-tip")) containers_original.push(c); } for (i = 0; i < containers_original.length; ++i) { c = containers_original[i]; // Clone container clone = c.cloneNode(true); // Disable inputs inputs = clone.querySelectorAll("input"); for (j = 0; j < inputs.length; ++j) { inputs[j].disabled = true; inputs[j].removeAttribute("data-option"); } if (i === 0) { // Dim if ((n = clone.querySelector("label"))) { n.style.opacity = "0.5"; } // Text for new iex_link = $("a"); iex_link.setAttribute("href", "//dnsev.github.io/iex/"); iex_link.setAttribute("target", "_blank"); iex_link.textContent = "disable iex version to use"; iex_link.addEventListener("click", on_edit_iex_settings_click.bind(this), false); iex_descr = $("span"); iex_descr.appendChild($.text(" (")); iex_descr.appendChild(iex_link); iex_descr.appendChild($.text(")")); clone.appendChild(iex_descr); // Text for old iex_link = $("a"); iex_link.setAttribute("href", "//dnsev.github.io/iex/"); iex_link.setAttribute("target", "_blank"); iex_link.textContent = "or use iex version"; iex_link.addEventListener("click", on_edit_iex_settings_click.bind(this), false); iex_descr = $("span"); iex_descr.appendChild($.text(" (")); iex_descr.appendChild(iex_link); iex_descr.appendChild($.text(")")); c.appendChild(iex_descr); } else { // Dim clone.style.opacity = "0.5"; } // Hide if (this.values.image_expansion.enabled) { c.style.display = "none"; } else { clone.style.display = "none"; } // Append par.insertBefore(clone, containers_original[0]); // Update checkbox value setTimeout(modify_settings_vanilla_display_delay.bind(this, inputs, image_hover_setting), 10); } }; var modify_settings_vanilla_display_delay = function (inputs, original) { for (var i = 0; i < inputs.length; ++i) { inputs[i].checked = original.checked; } }; var trigger_ready = function () { // Trigger ready event if (this.ready_callbacks !== null) { for (var i = 0; i < this.ready_callbacks.length; ++i) { this.ready_callbacks[i].call(this); } this.ready_callbacks = null; } }; var trigger_change = function (tree, old_value, new_value, source) { var key = tree.join("-"); if (key in this.change_events) { // Get the proper tree tree = this.change_events[key].tree; // Callback list var list = this.change_events[key].callbacks; for (var i = 0; i < list.length; ++i) { list[i].call(this, { tree: tree, old_value: old_value, new_value: new_value, source: source, }); } } }; var trigger_modifying_settings_4chanx_cb = function (okay, settings_were_changed, status) { // Show var body = document.querySelector("body"); if (body) style.remove_class(body, "iex_hide_other_settings"); // Callback if (this.modifying_settings_4chanx_callback !== null) { this.modifying_settings_4chanx_callback.call(this, okay, settings_were_changed, status); this.modifying_settings_4chanx_callback = null; } }; var trigger_modifying_settings_vanilla_cb = function (okay, settings_were_changed, status) { // Show var body = document.querySelector("body"); if (body) style.remove_class(body, "iex_hide_other_settings"); // Callback if (this.modifying_settings_vanilla_callback !== null) { this.modifying_settings_vanilla_callback.call(this, okay, settings_were_changed, status); this.modifying_settings_vanilla_callback = null; } }; var update_values = function (tree, values, new_values, remove, events) { // Updates var old_value, key; for (key in new_values) { if (key in values) { // Update if (new_values[key] instanceof Object) { // Deeper if (!(values[key] instanceof Object)) { values[key] = {}; } tree.push(key); update_values.call(this, tree, values[key], new_values[key], remove, events); tree.pop(); } else { // Replace old_value = values[key]; values[key] = new_values[key]; if (events && old_value !== values[key]) { tree.push(key); trigger_change.call(this, tree, old_value, values[key], "update"); tree.pop(); } } } else { // New if (new_values[key] instanceof Object) { // Deeper values[key] = {}; tree.push(key); update_values.call(this, tree, values[key], new_values[key], remove, events); tree.pop(); } else { // Replace values[key] = new_values[key]; if (events) { tree.push(key); trigger_change.call(this, tree, undefined, values[key], "update"); tree.pop(); } } } } // Removals if (remove) { for (key in values) { if (!(key in new_values)) { // Remove old_value = values[key]; delete values[key]; if (events) { tree.push(key); trigger_change.call(this, tree, old_value, undefined, "update"); tree.pop(); } } } } }; var string_to_float = function (value) { return parseFloat(value.trim()) || 0; }; var string_to_hotkey = function (value, descriptor) { value = value.trim().toLowerCase(); if (hotkey_manager.key_to_keycode(value) === 0) { // Return previous return this.get_value(descriptor.tree); } return value; }; var settings_open = function () { if (this.settings_container_outer !== null) return; var body = document.querySelector("body"), bg_color, settings_content, settings_buttons, cb, overlay, container, inner, n1, n2, n3, n4; if (!body) return; // Get background color bg_color = style.parse_css_color(style.get_true_style_of_class("post reply", "backgroundColor", "div")); settings_buttons = generate_settings_difficulty_selector.call(this); settings_content = generate_settings_container.call(this, generate_settings_descriptors.call(this)); this.settings_removal_data = settings_content[1].concat(settings_buttons[1]); // Create overlay = $.div("iex_settings_popup_overlay"); overlay.addEventListener("click", (cb = on_iex_settings_overlay_click.bind(this)), false); this.settings_removal_data.push({ node: overlay, event: "click", callback: cb, capture: false }); this.settings_container_outer = overlay; container = $.div("iex_settings_popup"); overlay.appendChild(container); inner = $.div("iex_settings_popup_aligner"); container.appendChild(inner); inner = $.div("iex_settings_popup_inner"); container.appendChild(inner); n1 = $.div("iex_settings_popup_table"); n1.style.backgroundColor = style.color_to_css(bg_color); inner.appendChild(n1); n1.addEventListener("click", (cb = on_iex_settings_body_click.bind(this)), false); this.settings_removal_data.push({ node: n1, event: "click", callback: cb, capture: false }); // Top n2 = $.div("iex_settings_popup_top"); n1.appendChild(n2); n3 = $.div("iex_settings_popup_top_content"); n2.appendChild(n3); n4 = $.div("iex_settings_popup_top_label"); n3.appendChild(n4); n4.textContent = "Image Extensions - Settings"; settings_buttons[0].className = "iex_settings_popup_top_right" + style.theme; n3.appendChild(settings_buttons[0]); this.settings_difficulty_container = settings_buttons[0]; // Middle n2 = $.div("iex_settings_popup_middle"); n1.appendChild(n2); n3 = $.div("iex_settings_popup_middle_content_pad"); n2.appendChild(n3); n2 = n3; n3 = $.div("iex_settings_popup_middle_content"); n2.appendChild(n3); n2 = n3; n3 = $.div("iex_settings_popup_middle_content_inner"); n2.appendChild(n3); n3.appendChild(settings_content[0]); this.settings_container = settings_content[0]; // Bottom n2 = $.div("iex_settings_popup_bottom"); n1.appendChild(n2); // Update difficulty change_settings_difficulty.call(this, "normal"); // Append body.appendChild(overlay); }; var generate_settings_difficulty_selector = function () { var d_container, d_choice, d_span, cb, settings_removal_data = []; d_container = $("div"); // Normal d_choice = $.a("iex_settings_difficulty_choice"); d_choice.textContent = "normal"; d_choice.setAttribute("data-iex-settings-difficulty-choice-level", "normal"); d_container.appendChild(d_choice); cb = wrap_generic_event(this, on_iex_difficulty_link_click); d_choice.addEventListener("click", cb, false); settings_removal_data.push({ node: d_choice, event: "click", callback: cb, capture: false }); d_span = $.span("iex_settings_difficulty_separator"); d_span.textContent = " | "; d_container.appendChild(d_span); // Advanced d_choice = $.a("iex_settings_difficulty_choice"); d_choice.textContent = "advanced"; d_choice.setAttribute("data-iex-settings-difficulty-choice-level", "advanced"); d_container.appendChild(d_choice); d_choice.addEventListener("click", cb, false); settings_removal_data.push({ node: d_choice, event: "click", callback: cb, capture: false }); d_span = $.span("iex_settings_difficulty_separator"); d_span.textContent = " | "; d_container.appendChild(d_span); // Homepage d_choice = $.a("iex_settings_homepage_link"); d_choice.setAttribute("target", "_blank"); d_choice.setAttribute("href", "//dnsev.github.io/iex/"); d_choice.textContent = "homepage"; d_container.appendChild(d_choice); d_span = $.span("iex_settings_difficulty_separator"); d_span.textContent = " | "; d_container.appendChild(d_span); // Close d_choice = $.a("iex_settings_popup_close_link"); d_choice.setAttribute("target", "_blank"); d_choice.setAttribute("href", "//dnsev.github.io/iex/"); d_choice.textContent = "\u00D7"; d_container.appendChild(d_choice); d_choice.addEventListener("click", (cb = on_iex_settings_close_click.bind(this)), false); settings_removal_data.push({ node: d_choice, event: "click", callback: cb, capture: false }); // Return return [ d_container , settings_removal_data ]; }; var generate_settings_descriptors = function () { var descriptors = []; // Image expansion descriptors.push({ level: "normal", section: "Image Expansion", tree: [ "image_expansion" , "enabled" ], label: "Enabled", description: "Enable custom expansion of images on mouse hover", type: "checkbox", values: [ false , true ], value_labels: [ "disabled" , "enabled" ], descriptors: [], after_callback: on_iex_setting_image_expansion_toggle.bind(this) }); descriptors.push({ level: "normal", section: "Image Expansion", tree: [ "image_expansion" , "normal" , "enabled" ], label: "Enabled", sublabel: "for non-spoiler images", description: "Enable hover for non-spoiler images", type: "checkbox", values: [ false , true ], value_labels: [ "disabled" , "enabled" ], }); descriptors.push({ level: "normal", section: "Image Expansion", tree: [ "image_expansion" , "spoiler" , "enabled" ], label: "Enabled", sublabel: "for spoiler images", description: "Enable hover for spoiler images", type: "checkbox", values: [ false , true ], value_labels: [ "disabled" , "enabled" ], }); descriptors.push({ level: "normal", section: "Image Expansion", tree: [ "image_expansion" , "normal" , "to_fit" ], label: "Fit image", sublabel: "for non-spoiler images", description: "Automatically fit the image to the best screen size on hover", type: "checkbox", values: [ false , true ], value_labels: [ "don't fit" , "fit" ], }); descriptors.push({ level: "normal", section: "Image Expansion", tree: [ "image_expansion" , "spoiler" , "to_fit" ], label: "Fit image", sublabel: "for spoiler images", description: "Automatically fit the image to the best screen size on hover", type: "checkbox", values: [ false , true ], value_labels: [ "don't fit" , "fit" ], }); descriptors.push({ level: "normal", section: "Image Expansion", tree: [ "image_expansion" , "normal" , "timeout" ], modify: string_to_float, label: "Open timeout", sublabel: "for non-spoiler images", description: "Time to wait before displaying the image (in seconds)", type: "textbox", }); descriptors.push({ level: "normal", section: "Image Expansion", tree: [ "image_expansion" , "spoiler" , "timeout" ], modify: string_to_float, label: "Open timeout", sublabel: "for spoiler images", description: "Time to wait before displaying the image (in seconds)", type: "textbox", }); // Hover settings descriptors.push({ level: "normal", section: "Hover", tree: [ "image_expansion" , "hover" , "header_overlap" ], label: "Overlap header", description: "If the header is visible, the preview will not overlap it", type: "checkbox", values: [ false , true ], value_labels: [ "don't overlap" , "overlap" ], }); descriptors.push({ level: "normal", section: "Hover", tree: [ "image_expansion" , "hover" , "fit_large_allowed" ], label: "Fit large", description: "When enabled, image zooming can be snapped to both vertical and horizontal scales", type: "checkbox", values: [ false , true ], value_labels: [ "disabled" , "enabled" ], }); descriptors.push({ level: "normal", section: "Hover", tree: [ "image_expansion" , "hover" , "display_stats" ], label: "Display stats", description: "Show the file name and relevant stats on top of the preview", type: "text", values: [ 0 , 1 , 2 ], value_labels: [ "show" , "hide" , "hide all" ] }); descriptors.push({ level: "normal", section: "Hover", tree: [ "image_expansion" , "hover" , "zoom_invert" ], label: "Zoom invert mouse", description: "Zoom location based on mouse will be inverted", type: "checkbox", values: [ false , true ], value_labels: [ "not inverted" , "inverted" ], }); descriptors.push({ level: "normal", section: "Hover", tree: [ "image_expansion" , "hover" , "zoom_borders_show" ], label: "Zoom borders", description: "Display borders inside the preview displaying the mouse movement region", type: "checkbox", values: [ false , true ], value_labels: [ "disabled" , "enabled" ], }); descriptors.push({ level: "normal", section: "Hover", tree: [ "image_expansion" , "hover" , "zoom_borders_hide_time" ], modify: string_to_float, label: "Zoom borders hide timeout", description: "Time to wait after mouse has stopped to hide the zoom borders (in seconds)", type: "textbox", }); descriptors.push({ level: "normal", section: "Hover", tree: [ "image_expansion" , "hover" , "zoom_buttons" ], label: "Zoom buttons", description: "Show zooming buttons when the zoom% is hovered", type: "checkbox", values: [ false , true ], value_labels: [ "disabled" , "enabled" ], }); descriptors.push({ level: "normal", section: "Hover", tree: [ "image_expansion" , "hover" , "mouse_hide" ], label: "Cursor hide", description: "Hide the mouse cursor when hovering the image after inactivity", type: "checkbox", values: [ false , true ], value_labels: [ "don't hide" , "hide" ], }); descriptors.push({ level: "normal", section: "Hover", tree: [ "image_expansion" , "hover" , "mouse_hide_time" ], modify: string_to_float, label: "Cursor hide timeout", description: "Time to wait after mouse has stopped to hide cursor (in seconds)", type: "textbox", }); descriptors.push({ level: "normal", section: "Hover", tree: [ "image_expansion" , "hover" , "mouse_hide_time" ], modify: string_to_float, label: "Cursor hide timeout", description: "Time to wait after mouse has stopped to hide cursor (in seconds)", type: "textbox", }); // File types descriptors.push({ level: "normal", section: "Hover", tree: [ "image_expansion" , "extensions" , "jpg" , "background" ], label: "Thumbnail background (.jpg)", description: "How to display the thumbnail image in the background while loading", type: "text", values: [ 0 , 1 , 2 ], value_labels: [ "never show" , "semi-transparent" , "opaque" ] }); descriptors.push({ level: "normal", section: "Hover", tree: [ "image_expansion" , "extensions" , "png" , "background" ], label: "Thumbnail background (.png)", description: "How to display the thumbnail image in the background while loading", type: "text", values: [ 0 , 1 , 2 ], value_labels: [ "never show" , "semi-transparent" , "opaque" ] }); descriptors.push({ level: "normal", section: "Hover", tree: [ "image_expansion" , "extensions" , "gif" , "background" ], label: "Thumbnail background (.gif)", description: "How to display the thumbnail image in the background while loading", type: "text", values: [ 0 , 1 , 2 ], value_labels: [ "never show" , "semi-transparent" , "opaque" ] }); descriptors.push({ level: "normal", section: "Hover", tree: [ "image_expansion" , "extensions" , "webm" , "background" ], label: "Thumbnail background (.webm)", description: "How to display the thumbnail image in the background while loading", type: "text", values: [ 0 , 1 , 2 ], value_labels: [ "never show" , "transparent" , "opaque" ] }); // Video settings descriptors.push({ level: "normal", section: "Video", tree: [ "image_expansion" , "video" , "mini_controls" ], label: "Mini controls", description: "When to display the mini control bar", type: "text", values: [ 0 , 1 , 2 , 3 ], value_labels: [ "never" , "mouse over preview" , "mouse NOT over preview" , "always" ] }); descriptors.push({ level: "normal", section: "Video", tree: [ "image_expansion" , "video" , "autoplay" ], label: "Autoplay", description: "When the video should automatically play", type: "text", values: [ 0 , 1 , 2 , 3 ], value_labels: [ "never" , "as soon as possible" , "when it can play \"through\"" , "when fully loaded" ] }); descriptors.push({ level: "normal", section: "Video", tree: [ "image_expansion" , "video" , "loop" ], label: "Loop", description: "Enable automatic video looping", type: "checkbox", values: [ false , true ], value_labels: [ "don't loop" , "loop" ], }); descriptors.push({ level: "normal", section: "Video", tree: [ "image_expansion" , "video" , "mute_initially" ], label: "Mute", description: "Mute the video when the preview is opened", type: "checkbox", values: [ false , true ], value_labels: [ "don't mute" , "mute" ], }); descriptors.push({ level: "normal", section: "Video", tree: [ "image_expansion" , "video" , "volume" ], modify: string_to_float, label: "Default volume", description: "The default volume of a video when it is opened", type: "textbox", }); descriptors.push({ level: "normal", section: "Video", tree: [ "image_expansion" , "video" , "expand_state_save" ], label: "Expansion state save", description: "The video state should be saved and resumed when expanding", type: "checkbox", values: [ false , true ], value_labels: [ "don't save" , "save" ], }); descriptors.push({ level: "normal", section: "Video", tree: [ "image_expansion" , "extensions" , "webm" , "mouse_wheel" ], label: "Mouse wheel action (.webm)", description: "What scrolling the mouse should do for .webm files", type: "text", values: [ 0 , 1 ], value_labels: [ "zoom" , "volume control" ] }); // Link settings descriptors.push({ level: "normal", section: "Linkification", tree: [ "file_linkification" , "enabled" ], label: "Named file URLs", description: "Appends the filename to the url as a #fragment (asthetic purposes)", type: "checkbox", values: [ false , true ], value_labels: [ "disabled" , "enabled" ], }); // Style settings descriptors.push({ level: "normal", section: "Style", tree: [ "image_expansion" , "style" , "animations_background" ], label: "Background image animations", description: "Use animated CSS transitions for the background image", type: "checkbox", values: [ false , true ], value_labels: [ "unanimated" , "animated" ], }); descriptors.push({ level: "normal", section: "Style", tree: [ "image_expansion" , "style" , "controls_rounded_border" ], label: "Rounded borders on controls", description: "Use rounded corners on video control bars (doesn't work on Chrome)", type: "checkbox", values: [ false , true ], value_labels: [ "square" , "rounded" ], }); // Annotations descriptors.push({ level: "normal", section: "Annotations", tree: [ "annotations" , "enabled" ], label: "Enabled", description: "Enable image annotations", type: "checkbox", values: [ false , true ], value_labels: [ "disabled" , "enabled" ], }); descriptors.push({ level: "normal", section: "Annotations", tree: [ "annotations" , "enabled_standalone" ], label: "Enabled on standalone images", description: "Enable image annotations on images opened in a new tab", type: "checkbox", values: [ false , true ], value_labels: [ "disabled" , "enabled" ], }); descriptors.push({ level: "normal", section: "Annotations", tree: [ "annotations" , "editor" ], label: "Annotation editor", description: "Show the annotation editor option on the quick reply form", type: "checkbox", values: [ false , true ], value_labels: [ "disabled" , "enabled" ], }); descriptors.push({ level: "normal", section: "Annotations", tree: [ "annotations" , "editor_always_enable" ], label: "Annotation editor always enabled", description: "Always enable the annotation editor when opening quick reply", type: "checkbox", values: [ false , true ], value_labels: [ "disabled" , "enabled" ], }); descriptors.push({ level: "normal", section: "Annotations", tree: [ "annotations" , "modify_urls" ], label: "Modify image URLs", description: "Append a #fragment to image urls to allow standalone image annotations", type: "checkbox", values: [ false , true ], value_labels: [ "disabled" , "enabled" ], }); descriptors.push({ level: "normal", section: "Annotations", tree: [ "annotations" , "transparent_until_hover" ], label: "Annotation transparency", description: "Make annotations more transparent when the mouse is not over them", type: "checkbox", values: [ false , true ], value_labels: [ "regular" , "transparent" ], }); descriptors.push({ level: "normal", section: "Annotations", tree: [ "annotations" , "toggle_hotkey" ], modify: string_to_hotkey, label: "Toggle Hotkey", description: "The key to use for toggling annotations on/off", type: "textbox", }); var vals = [], val_labels = [], i; for (i = 0; i < Annotation.fonts.length; ++i) { vals.push(i); val_labels.push(Annotation.fonts[i].name); } descriptors.push({ level: "normal", section: "Annotation Editor Defaults", tree: [ "annotations" , "defaults" , "font" ], label: "Default font", description: "Default font family for new annotations", type: "text", values: vals, value_labels: val_labels }); descriptors.push({ level: "normal", section: "Annotation Editor Defaults", tree: [ "annotations" , "defaults" , "bold" ], label: "Default bold", description: "Default bold setting for new annotations", type: "text", values: [ false , true ], value_labels: [ "not bold" , "bold" ], }); descriptors.push({ level: "normal", section: "Annotation Editor Defaults", tree: [ "annotations" , "defaults" , "italic" ], label: "Default italic", description: "Default italic setting for new annotations", type: "text", values: [ false , true ], value_labels: [ "not italic" , "italic" ], }); // Meta descriptors.push({ level: "advanced", section: "Debugging", change: on_iex_setting_display_settings.bind(this), label: "Display local settings", description: "Display all saved local data in a pop-up textbox", type: "text", values: [ true ], value_labels: [ "display" ] }); descriptors.push({ level: "advanced", section: "Debugging", change: on_iex_setting_delete_settings.bind(this), label: "Delete local settings", description: "Delete all saved local data and refresh the page", type: "text", values: [ true ], value_labels: [ "delete" ] }); // Return return descriptors; }; var generate_settings_container = function (descriptors, data) { // Create container var container = $.div("iex_settings_region"); // Groups data = data || []; var groups = {}; var group_count = 0; // Setup var d, i, j, g_key, g_container, g_label, g_set, g_span, g_div, g_table, g_half_left, g_half_right, g_input, g_input_label, cb, val, s_val, odd = true; for (i = 0; i < descriptors.length; ++i) { d = descriptors[i]; // Get group g_key = d.section || ""; if (g_key in groups) { g_container = groups[g_key]; } else { // New g_container = $.div("iex_settings_group" + (group_count > 0 ? " iex_settings_group_padding_top" : "")); container.appendChild(g_container); g_label = $.div("iex_settings_group_label"); g_label.textContent = g_key; g_container.appendChild(g_label); groups[g_key] = g_container; group_count += 1; odd = true; } // Add setting g_set = $.div("iex_settings_setting" + (odd ? " iex_settings_setting_odd" : "") + (i > 0 ? " iex_settings_setting_top_padding" : "")); g_set.setAttribute("data-iex-setting-level", d.level || "normal"); g_container.appendChild(g_set); g_table = $.div("iex_settings_setting_table"); g_set.appendChild(g_table); // Halves g_half_right = $.div("iex_settings_setting_right"); g_table.appendChild(g_half_right); g_half_left = $.div("iex_settings_setting_left"); g_table.appendChild(g_half_left); // Left half g_label = $.div("iex_settings_setting_label"); g_half_left.appendChild(g_label); g_span = $.span("iex_settings_setting_label_title"); g_span.textContent = d.label; g_label.appendChild(g_span); if ("sublabel" in d) { g_span = $.span("iex_settings_setting_label_subtitle"); g_span.textContent = d.sublabel; g_label.appendChild(g_span); } g_label = $.div("iex_settings_setting_description"); g_label.textContent = d.description; g_half_left.appendChild(g_label); // Right half: input g_div = $.label("iex_settings_setting_input_container"); g_half_right.appendChild(g_div); if (d.type == "checkbox") { g_div.className += " iex_settings_setting_input_container_reverse"; g_input = $.input.check("iex_settings_setting_input_checkbox"); g_div.appendChild(g_input); g_input_label = $.span("iex_settings_setting_input_label"); if ("value_labels" in d) { g_input_label.setAttribute("data-iex-checkbox-label-off", d.value_labels[0]); g_input_label.setAttribute("data-iex-checkbox-label-on", d.value_labels[1]); } else if ("values" in d) { g_input_label.setAttribute("data-iex-checkbox-label-off", "" + d.values[0]); g_input_label.setAttribute("data-iex-checkbox-label-on", "" + d.values[1]); } else { g_input_label.setAttribute("data-iex-checkbox-label-off", "off"); g_input_label.setAttribute("data-iex-checkbox-label-on", "on"); } g_div.appendChild(g_input_label); // Events cb = on_iex_setting_checkbox_change.bind(this, g_input, d); g_input.addEventListener("change", cb, false); data.push({ node: g_input, event: "change", callback: cb, capture: false }); // Check status if ("tree" in d) { val = this.get_value(d.tree); if ("values" in d) { g_input.checked = (val == d.values[1]); } else { g_input.checked = val; } } } else if (d.type == "textbox") { g_input = $.input.text("iex_settings_setting_input_textbox"); g_div.appendChild(g_input); // Events cb = on_iex_setting_textbox_change.bind(this, g_input, d); g_input.addEventListener("change", cb, false); data.push({ node: g_input, event: "change", callback: cb, capture: false }); // Text if ("tree" in d) { g_input.value = this.get_value(d.tree).toString(); } } else if (d.type == "text") { // Get current val = 0; if ("tree" in d) { s_val = this.get_value(d.tree); for (j = 0; j < d.values.length; ++j) { if (d.values[j] == s_val) { val = j; break; } } } else if ("get" in d) { val = d.get.call(this, d); } // Text g_input = $.a("iex_settings_setting_input_text"); g_input.textContent = ("value_labels" in d ? d.value_labels[val] : ("" + d.values[val])); g_input.setAttribute("data-iex-setting-current-id", val); g_div.appendChild(g_input); // Events cb = on_iex_setting_text_click.bind(this, g_input, d); g_input.addEventListener("click", cb, false); data.push({ node: g_input, event: "click", callback: cb, capture: false }); } // Sub-options if ("descriptors" in d) { g_div = $.div("iex_settings_region_container"); g_set.appendChild(g_div); g_div.appendChild(generate_settings_container.call(this, d.descriptors, data)[0]); } // Next odd = !odd; } // Return return [ container , data ]; }; var destroy_settings_data = function (data) { // Remove events for (var i = 0; i < data.length; ++i) { data[i].node.removeEventListener(data[i].event, data[i].callback, data[i].capture); } }; var change_settings_difficulty = function (target) { if (this.settings_container === null) return; // Setup var groups = this.settings_container.querySelectorAll(".iex_settings_group"), levels = [ "normal" , "advanced" ], level_target = levels.indexOf(target), settings, s, s_pre, i, j, odd, count, level, choices; // Modify settings for (i = 0; i < groups.length; ++i) { settings = groups[i].querySelectorAll(".iex_settings_setting"); odd = true; s_pre = null; count = 0; for (j = 0; j < settings.length; ++j) { s = settings[j]; level = levels.indexOf(s.getAttribute("data-iex-setting-level")); // Display if (level <= level_target) { // Un-hide style.remove_class(s, "iex_settings_setting_hidden"); ++count; // Oddity if (odd) { style.add_class(s, "iex_settings_setting_odd"); } else { style.remove_class(s, "iex_settings_setting_odd"); } odd = !odd; // Previous if (s_pre !== null) { style.add_class(s, "iex_settings_setting_top_padding"); } else { style.remove_class(s, "iex_settings_setting_top_padding"); } s_pre = s; } else { // Hide style.add_class(s, "iex_settings_setting_hidden"); } } if (count === 0) { style.add_class(groups[i], "iex_settings_group_hidden"); } else { style.remove_class(groups[i], "iex_settings_group_hidden"); } } // Modify difficulty choices = this.settings_difficulty_container.querySelectorAll(".iex_settings_difficulty_choice"); for (i = 0; i < choices.length; ++i) { level = choices[i].getAttribute("data-iex-settings-difficulty-choice-level"); if (level == target) { style.add_class(choices[i], "iex_settings_difficulty_choice_selected"); } else { style.remove_class(choices[i], "iex_settings_difficulty_choice_selected"); } } }; Settings.prototype = { constructor: Settings, setup: function () { // Events sync.on("install_complete", this.on_install_complete_sync_bind = on_install_complete_sync.bind(this)); sync.on("settings_save", this.on_settings_save_sync_bind = on_settings_save_sync.bind(this)); sync.on("image_expansion_enable", this.on_image_expansion_enable_sync_bind = on_image_expansion_enable_sync.bind(this)); // Modify other settings api.on("settings_4chanx_section_change", this.on_settings_4chanx_section_change_bind = on_settings_4chanx_section_change.bind(this)); api.on("settings_vanilla_open", this.on_settings_vanilla_open_bind = on_settings_vanilla_open.bind(this)); api.on("menu_4chanx_open", this.on_menu_4chanx_open_bind = on_menu_4chanx_open.bind(this)); // First load this.load_values(false, on_initial_load.bind(this)); on_ready(on_insert_links.bind(this), on_insert_links_condition, 0.5); }, get_value: function (tree) { // Traverse tree var val = this.values, i = 0, j = tree.length - 1; for (; i < j; ++i) { val = val[tree[i]]; } // Get return val[tree[i]]; }, change_value: function (tree, new_value) { // Traverse tree var val = this.values, i = 0, j = tree.length - 1; for (; i < j; ++i) { val = val[tree[i]]; } // Update var old_value = val[tree[i]]; val[tree[i]] = new_value; // Change event trigger_change.call(this, tree, old_value, new_value, "same"); // Return old return old_value; }, load_values: function (events, complete_callback) { // Async save SaveAsync.get(this.save_key, on_load_values_callback.bind(this, events, complete_callback)); }, save_values: function (complete_callback) { // Async save SaveAsync.set(this.save_key, this.values, on_save_values_callback.bind(this, complete_callback)); }, delete_values: function (complete_callback) { // Async delete SaveAsync.del(this.save_key, complete_callback); }, on_ready: function (callback) { if (this.ready_callbacks !== null) { // Add to callback list this.ready_callbacks.push(callback); } else { // Already ready callback.call(this); } }, on_change: function (tree, callback) { var key = tree.join("-"); if (key in this.change_events) { // Update old this.change_events[key].callbacks.push(callback); } else { // Create new var tree_clone = []; for (var i = 0; i < tree.length; ++i) { tree_clone.push(tree[i]); } this.change_events[key] = { tree: tree_clone, callbacks: [ callback ] }; } }, off_change: function (tree, callback) { var key = tree.join("-"); if (key in this.change_events) { // Update old var list = this.change_events[key].callbacks; for (var i = 0; i < list.length; ++i) { if (list[i] === callback) { // Remove list[i].splice(i, 1); if (list.length === 0) { delete this.change_events[key]; } return true; } } } return false; }, modify_4chanx_settings: function (callback) { if (api.is_4chanx) { var body = document.querySelector("body"); if (body) style.add_class(body, "iex_hide_other_settings"); // Set the timer this.modifying_settings_4chanx_timer = setTimeout(on_modifying_settings_4chanx_timout.bind(this), 1000); this.modifying_settings_4chanx_callback = callback || null; // Open settings to Main tab var target_tab = api.is_appchanx ? "Script" : "Main"; if (api.settings_4chanx_is_open()) { api.settings_4chanx_change_section(target_tab); } else { api.settings_4chanx_open(target_tab); } } }, modify_vanilla_settings: function (callback) { if (!api.is_4chanx) { var body = document.querySelector("body"); if (body) style.add_class(body, "iex_hide_other_settings"); // Set the timer this.modifying_settings_vanilla_timer = setTimeout(on_modifying_settings_vanilla_timout.bind(this), 1000); this.modifying_settings_vanilla_callback = callback || null; // Open settings api.settings_vanilla_open(); } }, disable_image_expansion: function () { if (!this.can_disable_image_hover) return; // Disable settings.change_value(["image_expansion", "enabled"], false); settings.save_values(); // Alert message this.display_image_hover_notification(); }, settings_open: function () { settings_open.call(this); }, settings_close: function () { if (this.settings_container === null) return; // Remove destroy_settings_data.call(this, this.settings_removal_data); var par = this.settings_container_outer.parentNode; if (par) par.removeChild(this.settings_container_outer); this.settings_removal_data = null; this.settings_container = null; this.settings_container_outer = null; this.settings_difficulty_container = null; // Modify other settings if (this.settings_update_other_after_close) { this.settings_update_other_after_close = false; if (this.values.image_expansion.enabled) { // Modify var cb = on_settings_update_other_complete.bind(this); if (api.is_4chanx) { this.modify_4chanx_settings(cb); } else { this.modify_vanilla_settings(cb); } } } }, display_first_run_notification: function () { if (this.notification_first_run !== null) return; // Alert message var message = $("div"), m_title, m_part, m_link; // Line 1 m_title = $("div"); m_title.appendChild($.text("You've just installed ")); m_link = $("a"); m_link.setAttribute("target", "_blank"); m_link.setAttribute("rel", "nofollow noreferrer"); m_link.setAttribute("href", "https://dnsev.github.io/iex/"); m_link.textContent = "iex - Image Extensions"; m_title.appendChild(m_link); m_title.appendChild($.text("!")); // Line 2 m_part = $.div("iex_spaced_div"); m_part.textContent = "If you use 4chan-x or appchan-x, you can access iex settings through the header menu. Otherwise, look for the [ iex ] link at the top of the page."; message.appendChild(m_part); // Line 3 m_part = $.div("iex_spaced_div"); m_part.appendChild($.text("Any questions or issues can be addressed on ")); m_link = $("a"); m_link.setAttribute("target", "_blank"); m_link.setAttribute("rel", "nofollow noreferrer"); m_link.setAttribute("href", "https://github.com/dnsev/iex/issues"); m_link.textContent = "the github page"; m_part.appendChild(m_link); m_part.appendChild($.text(".")); message.appendChild(m_part); // Line 4 m_part = $.div("iex_spaced_div"); m_part.textContent = "Would you like to enable iex now? (page will be refreshed)"; message.appendChild(m_part); // Line 5 m_part = $("div"); m_link = $.a("iex_highlighted_link"); m_link.setAttribute("target", "_blank"); m_link.setAttribute("href", "//dnsev.github.io/iex/"); m_link.textContent = "enable now"; m_link.addEventListener("click", on_first_run_link_click.bind(this, true), false); m_part.appendChild(m_link); m_part.appendChild($.text(" or ")); m_link = $.a("iex_highlighted_link"); m_link.setAttribute("target", "_blank"); m_link.setAttribute("href", "//dnsev.github.io/iex/"); m_link.textContent = "leave disabled"; m_link.addEventListener("click", on_first_run_link_click.bind(this, false), false); m_part.appendChild(m_link); message.appendChild(m_part); // Display this.notification_first_run = new Notification({ close: false, overlay_close: false, style: "info", title: m_title, content: message, on: { close: on_notification_first_run_close.bind(this) } }); }, display_install_fail_notification: function (status) { if (this.notification_install_fail !== null) return; // Alert message var message = $("div"), m_part, m_link; // Line 1 m_part = $.div("iex_spaced_div"); m_part.textContent = "Failed to resolve settings conflicts for reason: " + status + "."; message.appendChild(m_part); // Line 3 m_part = $("div"); m_part.textContent = "This is potentially due to other userscript extensions you have installed."; message.appendChild(m_part); // Line 4 m_part = $.div("iex_spaced_div"); m_part.textContent = "You can try manually turning off any \"Image Hover\" settings to get iex to work."; message.appendChild(m_part); // Line 5 m_part = $.div("iex_spaced_div"); m_part.appendChild($.text("If you would like to get this fixed, ")); m_link = $.a("iex_underlined_link"); m_link.setAttribute("target", "_blank"); m_link.setAttribute("href", "http://github.com/dnsev/iex/issues"); m_link.textContent = "file an issue request"; m_part.appendChild(m_link); m_part.appendChild($.text(".")); message.appendChild(m_part); // Line 5 m_part = $("div"); m_part.textContent = "This script can be re-enabled through its settings page."; message.appendChild(m_part); // Display this.notification_install_fail = new Notification({ close: true, overlay_close: false, style: "error", title: "iex error: Failed to modify settings", content: message, on: { close: on_notification_install_fail_close.bind(this) } }); }, display_settings_info_notification: function () { // Async get SaveAsync.get(this.save_key, on_display_settings_info_notification_callback.bind(this)); }, display_image_hover_notification: function () { // Alert message var message = $("div"), m_part, m_link; // Line 1 m_part = $("div"); // Text m_part.appendChild($.text("Image Extensions will be disabled and can be re-enabled through ")); // Link m_link = $.a("iex_underlined_link"); m_link.setAttribute("target", "_blank"); m_link.setAttribute("href", "//dnsev.github.io/iex/"); m_link.textContent = "settings"; m_part.appendChild(m_link); // Text m_part.appendChild($.text(".")); // Add to message message.appendChild(m_part); // Display this.notification_image_hover = new Notification({ close: true, overlay_close: true, style: "warning", title: "Default image hovering still enabled", content: message, on: { close: on_notification_image_hover_close.bind(this) } }); m_link.addEventListener("click", on_notification_link_click.bind(this, true, this.notification_image_hover), false); }, }; return Settings; })(); // Class to manage page styling var Style = (function () { var Style = function () { // Append style this.stylesheet = null; this.stylesheet_annotations = null; // Observers this.head_observer = null; this.style_observers = []; this.stylesheet_observers = []; this.on_style_observe_bind = on_style_observe.bind(this); this.on_stylesheet_observe_bind = on_stylesheet_observe.bind(this); // Theme this.theme = " iex_no_theme"; this.theme_check_count = 0; this.theme_check_count_max = 100; this.theme_check_timer = null; this.theme_check_timeout_bind = on_theme_check_timeout.bind(this); on_ready(on_asap.bind(this), on_asap_condition.bind(this)); }; var loaded_fonts = null; var on_insert_stylesheet = function (stylesheet) { var head = document.querySelector("head"); if (head) head.appendChild(stylesheet); }; var on_insert_stylesheet_condition = function () { return document.querySelector("head"); }; var on_asap = function () { // Theme check this.theme_check_timeout_bind.call(this); // Style changing var head = document.querySelector("head"); if (!head) return; // Create new observer this.head_observer = new MutationObserver(on_head_observe.bind(this)); this.head_observer.observe( head, { childList: true } ); // Add event listeners to stylesheets var stylesheets = head.querySelectorAll("link[rel='stylesheet']"), styles = head.querySelectorAll("style"), i; for (i = 0; i < styles.length; ++i) { // Create new observer add_style_observer.call(this, styles[i]); } for (i = 0; i < stylesheets.length; ++i) { // Create new observer add_stylesheet_observer.call(this, stylesheets[i]); } }; var on_asap_condition = function () { return document.querySelector("head") && document.querySelector("body"); }; var on_theme_check_timeout = function () { // Stop timer this.theme_check_timer = null; // New theme if (!theme_update.call(this)) { // Timeout if (++this.theme_check_count < this.theme_check_count_max) { this.theme_check_timer = setTimeout(this.theme_check_timeout_bind, 50); } } }; var on_head_observe = function (records) { var nodes, i, j, r; for (i = 0; i < records.length; ++i) { r = records[i]; if ((nodes = r.addedNodes)) { for (j = 0; j < nodes.length; ++j) { // Check on_head_observe_add.call(this, nodes[j]); } } if ((nodes = r.removedNodes)) { for (j = 0; j < nodes.length; ++j) { // Check on_head_observe_remove.call(this, nodes[j]); } } } }; var on_head_observe_add = function (element) { if (element.tagName === "STYLE") { add_style_observer.call(this, element); } else if (element.tagName === "LINK") { if (element.getAttribute("rel") == "stylesheet") { add_stylesheet_observer.call(this, element); } } }; var on_head_observe_remove = function (element) { if (element.tagName === "STYLE") { remove_style_observer.call(this, element); } else if (element.tagName === "LINK") { if (element.getAttribute("rel") == "stylesheet") { remove_stylesheet_observer.call(this, element); } } }; var on_stylesheet_observe = function (records) { var i, r; for (i = 0; i < records.length; ++i) { r = records[i]; if (r.attributeName === "href") { // Theme update theme_update.call(this); } } }; var on_style_observe = function (records) { var i; for (i = 0; i < records.length; ++i) { // Theme update theme_update.call(this); } }; var add_style_observer = function (element) { // Create new observer var observer = new MutationObserver(this.on_style_observe_bind); observer.observe( element, { childList: true, characterData: true } ); this.style_observers.push({ target: element, observer: observer, }); }; var add_stylesheet_observer = function (element) { // Create new observer var observer = new MutationObserver(this.on_stylesheet_observe_bind); observer.observe( element, { attributes: true } ); this.stylesheet_observers.push({ target: element, observer: observer, }); }; var remove_style_observer = function (element) { // Delete observer for (var i = 0; i < this.style_observers.length; ++i) { if (this.style_observers[i].target === element) { this.style_observers[i].observer.disconnect(); this.style_observers.splice(i, 1); return; } } }; var remove_stylesheet_observer = function (element) { // Delete observer for (var i = 0; i < this.stylesheet_observers.length; ++i) { if (this.stylesheet_observers[i].target === element) { this.stylesheet_observers[i].observer.disconnect(); this.stylesheet_observers.splice(i, 1); return; } } }; var theme_update = function () { // Find theme var theme_new = this.theme_detect(); if (theme_new !== null) { // Clear timer if (this.theme_check_timer !== null) { clearTimeout(this.theme_check_timer); this.theme_check_timer = null; } // Change theme this.theme_change(theme_new); // Okay return true; } // Not okay return false; }; var get_true_style = function (element, style_name) { var s; try { s = document.defaultView.getComputedStyle(element); } catch (e) {} if (!s) { s = element.style || {}; } return style_name ? s[style_name] : s; }; Style.prototype = { constructor: Style, insert_stylesheet: function () { // Create stylesheet var stylesheet_text = "body.iex_hide_other_settings #settingsMenu,body.iex_hide_other_settings #overlay,body.iex_hide_other_settings #appchanx-settings,body.iex_hide_other_settings #fourchanx-settings{visibility:hidden !important;}.iex_no_padding_br{padding-bottom:0 !important;padding-right:0 !important;margin-bottom:0 !important;margin-right:0 !important;border-bottom-width:0 !important;border-right-width:0 !important;}.iex_no_padding_tl{padding-top:0 !important;padding-left:0 !important;margin-top:0 !important;margin-left:0 !important;border-top-width:0 !important;border-left-width:0 !important;}.iex_settings_popup_overlay{position:fixed;z-index:200;left:0;top:0;bottom:0;right:0;font-size:14px;background:rgba(0,0,0,0.25);}.iex_settings_popup_overlay.iex_dark{background:rgba(255,255,255,0.25);}.iex_settings_popup{margin:0;padding:0;width:100%;height:100%;text-align:center;white-space:nowrap;z-index:200;line-height:0;}.iex_settings_popup_aligner{height:100%;width:0;display:inline-block;vertical-align:middle;}.iex_settings_popup_inner{display:inline-block;vertical-align:middle;text-align:left;line-height:normal;white-space:normal;position:relative;width:40em;height:80%;min-height:14em;}.iex_settings_popup_table{display:table;position:relative;table-layout:fixed;width:100%;height:100%;box-shadow:0 0 1em 0.25em #000000;}.iex_settings_popup_table.iex_dark{box-shadow:0 0 1em 0.25em #ffffff;}.iex_settings_popup_top{display:table-row;width:100%;}.iex_settings_popup_top_content{display:table;width:100%;background:rgba(0,0,0,0.0625);}.iex_settings_popup_top_content.iex_dark{background:rgba(255,255,255,0.0625);}.iex_settings_popup_top_label{display:table-cell;vertical-align:middle;padding:0.25em;font-size:1.125em;font-weight:bold;text-shadow:0 0.0625em 0.25em #ffffff;}.iex_settings_popup_top_label.iex_dark{text-shadow:0 0.0625em 0.25em #000000;}.iex_settings_popup_top_right{display:table-cell;vertical-align:middle;padding:0.25em;text-align:right;}.iex_settings_popup_middle{display:table-row;width:100%;height:100%;}.iex_settings_popup_middle_content_pad{display:block;width:100%;height:100%;position:relative;}.iex_settings_popup_middle_content{display:block;position:absolute;left:0;top:0;bottom:0;right:0;padding:0;overflow:auto;}.iex_settings_popup_middle_content_inner{display:block;padding:0.5em;}.iex_settings_popup_close_link{font-weight:bold;text-decoration:none !important;padding:0em 0.2em;}.iex_settings_popup_bottom{display:table-row;width:100%;}.iex_settings_container{position:relative;}.iex_settings_region_container{margin:0.25em -0.25em -0.25em 2em;}.iex_settings_region{}.iex_settings_group{}.iex_settings_group.iex_settings_group_hidden{display:none;}.iex_settings_group_label{font-size:1.125em;padding-bottom:0.125em;margin-bottom:0.25em;font-weight:bold;text-align:left;color:#000000;border-bottom:0.125em solid rgba(0,0,0,0.5);text-shadow:0em 0em 0.25em #ffffff;}.iex_settings_group_label.iex_dark{color:#ffffff;border-bottom:0.125em solid rgba(255,255,255,0.5);text-shadow:0em 0em 0.25em #000000;}.iex_settings_group.iex_settings_group_padding_top{margin-top:2em;}.iex_settings_setting{padding:0.25em;background:transparent;}.iex_settings_setting.iex_settings_setting_top_padding{margin-top:0.25em;}.iex_settings_setting.iex_settings_setting_hidden{display:none;}.iex_settings_setting.iex_settings_setting_odd{background:rgba(0,0,0,0.0625);}.iex_settings_setting.iex_settings_setting_odd.iex_dark{background:rgba(255,255,255,0.0625);}.iex_settings_setting_table{display:block;width:100%;}.iex_settings_setting_left{display:block;overflow:hidden;text-align:left;vertical-align:top;}.iex_settings_setting_right{display:inline-block;float:right;text-align:right;vertical-align:middle;padding:0.5em 0em;}.iex_settings_setting_label{display:block;}.iex_settings_setting_label_title{color:#000000;font-weight:bold;font-size:1.125em;vertical-align:baseline;}.iex_settings_setting_label_title.iex_dark{color:#ffffff;}.iex_settings_setting_label_subtitle{color:#000000;font-weight:bold;font-size:1em;vertical-align:baseline;opacity:0.625;margin-left:0.5em;}.iex_settings_setting_label_subtitle.iex_dark{color:#ffffff;}.iex_settings_setting_label_subtitle:before{content:\"(\";}.iex_settings_setting_label_subtitle:after{content:\")\";}.iex_settings_setting_description{margin-left:0.5em;}.iex_settings_setting_input_container{display:inline-block;white-space:nowrap;cursor:pointer;}.iex_settings_setting_input_container.iex_settings_setting_input_container_reverse{direction:rtl;}.iex_settings_setting_input_checkbox{vertical-align:middle;}.iex_settings_setting_input_checkbox+.iex_settings_setting_input_label,.iex_settings_setting_input_checkbox+*+.iex_settings_setting_input_label{vertical-align:middle;padding-right:0.5em;}.iex_settings_setting_input_checkbox+.iex_settings_setting_input_label:after,.iex_settings_setting_input_checkbox+*+.iex_settings_setting_input_label:after{content:attr(data-iex-checkbox-label-off);}.iex_settings_setting_input_checkbox:checked+.iex_settings_setting_input_label:after,.iex_settings_setting_input_checkbox:checked+*+.iex_settings_setting_input_label:after{content:attr(data-iex-checkbox-label-on);}input.iex_settings_setting_input_textbox{width:5em;padding:0.125em;border:1px solid rgba(16,16,16,0.5) !important;}input.iex_settings_setting_input_textbox:focus{border-color:rgba(16,16,16,1) !important;}input.iex_settings_setting_input_textbox.iex_dark{border-color:rgba(240,240,240,0.5) !important;}input.iex_settings_setting_input_textbox.iex_dark:focus{border-color:rgba(240,240,240,1) !important;}.iex_settings_difficulty_container{position:absolute;right:0;top:0;margin-top:0.5em;}.iex_settings_popup a.iex_settings_difficulty_choice{cursor:pointer;text-decoration:none !important;}.iex_settings_popup a.iex_settings_difficulty_choice.iex_settings_difficulty_choice_selected{text-decoration:underline !important;}.iex_settings_popup a.iex_settings_homepage_link{text-decoration:none !important;}.iex_settings_difficulty_separator{}.iex_notification{position:fixed;left:0;top:0;bottom:0;right:0;z-index:100;background:rgba(255,255,255,0.5);font-size:14px;}.iex_notification.iex_dark{background:rgba(0,0,0,0.5);}.iex_notification_body_outer{margin:0;padding:0;width:100%;height:100%;text-align:center;white-space:nowrap;z-index:200;line-height:0;}.iex_notification_body_aligner{height:100%;width:0;display:inline-block;vertical-align:middle;}.iex_notification_body_inner{display:inline-block;vertical-align:middle;line-height:normal;}.iex_notification_body{position:relative;text-align:center;white-space:normal;width:36em;padding:0.375em 1em;border-radius:0.25em;box-shadow:0em 0em 0.25em 0.125em rgba(0,0,0,0.5);}.iex_notification_body.iex_dark{box-shadow:0em 0em 0.25em 0.125em rgba(255,255,255,0.5);}.iex_notification_body.iex_notification_info{background:rgba(0,120,220,0.9);}.iex_notification_body.iex_notification_success{background:rgba(0,168,20,0.9);}.iex_notification_body.iex_notification_error{background:rgba(220,0,0,0.9);}.iex_notification_body.iex_notification_warning{background:rgba(220,120,0,0.9);}a.iex_notification_close{color:#ffffff !important;position:absolute;top:0;right:0;z-index:10;cursor:pointer;font-weight:bold;}a.iex_notification_close:hover{color:#000000 !important;text-shadow:0em 0.0625em 0.125em #ffffff;}a.iex_notification_close.iex_notification_close_hidden{display:none;}.iex_notification_content_outer{position:relative;}.iex_notification_content{position:relative;}.iex_notification_content_title{font-weight:bold;margin-bottom:0.75em;color:#ffffff;text-shadow:0em 0.0625em 0.125em #000000;}.iex_notification_content_body{color:#ffffff;text-shadow:0em 0.0625em 0.125em rgba(0,0,0,0.5);}.iex_notification_content a{color:#ffffff !important;text-decoration:none;}.iex_notification_content a:hover{color:#c010e0 !important;}.iex_notification_content.iex_dark a:hover{color:#d040e8 !important;}.iex_notification_content .iex_spaced_div{margin-bottom:0.5em;}.iex_notification_content a.iex_highlighted_link{font-weight:bold;text-decoration:underline !important;}.iex_notification_content a.iex_underlined_link{text-decoration:underline !important;}textarea.iex_notification_textarea{width:100%;height:5em;min-height:2em;font-family:Courier New;font-size:1em;text-align:left;color:#ffffff !important;background-color:transparent !important;border:0.08em solid rgba(255,255,255,0.5) !important;padding:0.25em;margin:0;resize:vertical;box-sizing:border-box;-moz-box-sizing:border-box;}textarea.iex_notification_textarea:hover,textarea.iex_notification_textarea:focus{background-color:rgba(255,255,255,0.0625) !important;border:0.08em solid rgba(255,255,255,0.75) !important;}.iex_floating_container{display:block;margin:0;padding:0;border:0em hidden;}.iex_floating_image_connector{display:block;position:absolute;z-index:100;}.iex_floating_image_connector:not(.iex_floating_image_connector_visible){display:none;}.iex_cpreview_container{display:block;position:absolute;z-index:100;}.iex_cpreview_container.iex_cpreview_container_fixed{position:fixed;}.iex_cpreview_container:not(.iex_cpreview_container_visible){display:none;}.iex_cpreview_container.iex_cpreview_container_auto{z-index:0;left:0;top:0;bottom:0;right:0;}.iex_cpreview_container.iex_cpreview_container_auto>.iex_cpreview_padding,.iex_cpreview_container.iex_cpreview_container_auto>.iex_cpreview_padding>.iex_cpreview_overflow{width:100%;height:100%;}.iex_cpreview_padding{display:block;position:relative;margin:0;padding:0;left:0;top:0;}.iex_cpreview_overflow{display:block;position:relative;overflow:hidden;}.iex_cpreview_offset{display:block;position:relative;width:100%;height:100%;text-align:center;}.iex_cpreview_offset.iex_cpreview_offset_sized{}.iex_cpreview_overlay{}.iex_cpreview_content{}.iex_cpreview_padding.iex_mpreview_padding{box-shadow:0em 0em 0.6em 0.3em rgba(0,0,0,0.75);background-color:rgba(255,255,255,0.5);}.iex_cpreview_padding.iex_mpreview_padding.iex_dark{box-shadow:0em 0em 0.6em 0.3em rgba(255,255,255,0.75);background-color:rgba(0,0,0,0.5);}.iex_cpreview_padding.iex_mpreview_mouse_hidden{cursor:none;}.iex_mpreview_background{display:block;position:absolute;left:0;top:0;bottom:0;right:0;background-color:transparent;background-repeat:no-repeat;background-position:left top;background-size:100% 100%;opacity:0.375;}.iex_mpreview_background.iex_mpreview_background_disabled{opacity:0;visibility:hidden;}.iex_mpreview_background.iex_mpreview_background_visible_full{opacity:1;}.iex_mpreview_background:not(.iex_mpreview_background_visible){display:none;}.iex_mpreview_background.iex_transitions.iex_mpreview_background_visible{transition:opacity 0.5s, visibility 0s linear 0s;}.iex_mpreview_background.iex_transitions.iex_mpreview_background_visible.iex_mpreview_background_disabled{transition:opacity 0.5s, visibility 0s linear 0.5s;}.iex_mpreview_zoom_borders{display:block;position:absolute;border:0.08em solid #ffffff;opacity:0.25;left:0;top:0;bottom:0;right:0;pointer-events:none;}.iex_mpreview_zoom_borders_inner{display:block;position:absolute;border:0.08em solid #000000;left:0;top:0;bottom:0;right:0;}.iex_mpreview_zoom_borders:not(.iex_mpreview_zoom_borders_visible),.iex_mpreview_zoom_borders:not(.iex_mpreview_zoom_borders_vertical):not(.iex_mpreview_zoom_borders_horizontal){opacity:0;visibility:hidden;}.iex_mpreview_zoom_borders:not(.iex_mpreview_zoom_borders_vertical){top:0 !important;bottom:0 !important;border-top:0 hidden;border-bottom:0 hidden;}.iex_mpreview_zoom_borders:not(.iex_mpreview_zoom_borders_horizontal){left:0 !important;right:0 !important;border-left:0 hidden;border-right:0 hidden;}.iex_mpreview_zoom_borders:not(.iex_mpreview_zoom_borders_vertical)>.iex_mpreview_zoom_borders_inner{border-top:0 hidden;border-bottom:0 hidden;}.iex_mpreview_zoom_borders:not(.iex_mpreview_zoom_borders_horizontal)>.iex_mpreview_zoom_borders_inner{border-left:0 hidden;border-right:0 hidden;}.iex_mpreview_stats_container{display:block;position:absolute;padding:0.125em 0em 0em 0.5em;left:-0.5em;right:0;bottom:100%;white-space:nowrap;overflow:hidden;color:#ffffff;text-shadow:0em 0em 0.4em #000000,0em 0em 0.3em #000000;}.iex_mpreview_stats_container.iex_dark{color:#000000;text-shadow:0em 0em 0.4em #ffffff,0em 0em 0.3em #ffffff;}.iex_mpreview_stat{font-weight:bold;}.iex_mpreview_stat.iex_mpreview_stat_red{color:#f03030;}.iex_mpreview_stat.iex_mpreview_stat_red.iex_dark{color:#a00000;}.iex_mpreview_stat_sep{opacity:0.825;}.iex_mpreview_stat:not(.iex_mpreview_stat_visible),.iex_mpreview_stat_sep:not(.iex_mpreview_stat_sep_visible){display:none;}.iex_mpreview_stat.iex_mpreview_stat_zoom_controls{margin-right:0.25em;position:absolute;}.iex_mpreview_stat.iex_mpreview_stat_zoom_controls.iex_mpreview_stat_zoom_controls_fixed{position:fixed;}.iex_mpreview_stat_zoom_controls_offset{display:none;}.iex_mpreview_stat.iex_mpreview_stat_zoom_controls.iex_mpreview_stat_visible+.iex_mpreview_stat_zoom_controls_offset{display:inline-block;}.iex_mpreview_stat_zoom_control{padding:0em 0.25em;cursor:pointer;}.iex_mpreview_stat_zoom_control_increase:hover{color:#80d0ff;}.iex_mpreview_stat_zoom_control_increase.iex_dark:hover{color:#a0e0ff;}.iex_mpreview_stat_zoom_control_decrease:hover{color:#f08060;}.iex_mpreview_stat_zoom_control_decrease.iex_dark:hover{color:#ffb0a0;}.iex_mpreview_image{display:inline-block;border:none;margin:0;padding:0;outline:none;}.iex_mpreview_image:not(.iex_mpreview_image_visible){display:none;}.iex_mpreview_image.iex_mpreview_image_unsized{max-width:100%;max-height:100%;}.iex_cpreview_offset.iex_cpreview_offset_sized>.iex_mpreview_image{position:absolute;left:0;top:0;bottom:0;right:0;width:100%;height:100%;}.iex_cpreview_offset:not(.iex_cpreview_offset_sized)>.iex_mpreview_image{max-width:100%;max-height:100%;width:auto;height:auto;}.iex_mpreview_video{display:inline-block;border:0em hidden;margin:0;padding:0;outline:0em hidden;}.iex_mpreview_video:not(.iex_mpreview_video_visible){display:none;}.iex_mpreview_video.iex_mpreview_video_unsized{max-width:100%;max-height:100%;}.iex_mpreview_video.iex_mpreview_video_not_ready{visibility:hidden;}.iex_cpreview_offset.iex_cpreview_offset_sized>.iex_mpreview_video{position:absolute;left:0;top:0;bottom:0;right:0;width:100%;height:100%;}.iex_cpreview_offset:not(.iex_cpreview_offset_sized)>.iex_mpreview_video{max-width:100%;max-height:100%;width:auto;height:auto;}.iex_mpreview_vcontrols_container{display:block;position:absolute;left:0;top:0;bottom:0;right:0;background:transparent;font-size:1.5em;}.iex_mpreview_vcontrols_container_inner{display:block;position:absolute;left:0;bottom:0;right:0;padding-top:2em;height:2em;background:transparent;}.iex_mpreview_vcontrols_table{display:table;width:100%;height:100%;}.iex_mpreview_vcontrols_table:not(.iex_mpreview_vcontrols_table_visible):not(.iex_mpreview_vcontrols_table_visible_temp):not(.iex_mpreview_vcontrols_table_visible_important):not(.iex_mpreview_vcontrols_table_mini),.iex_mpreview_vcontrols_table.iex_mpreview_vcontrols_table_mini_disabled{display:none;}.iex_mpreview_vcontrols_seek_container{display:table-cell;vertical-align:bottom;min-width:1em;}.iex_mpreview_vcontrols_seek_container_inner{display:block;width:100%;height:2em;position:relative;}.iex_mpreview_vcontrols_seek_bar{display:block;position:absolute;left:0;right:0;top:0.5em;bottom:0.5em;border-radius:0.5em;overflow:hidden;cursor:pointer;border:1px solid rgba(0,0,0,0.5);}.iex_mpreview_vcontrols_seek_bar.iex_mpreview_vcontrols_no_border_radius{border-radius:0;}.iex_mpreview_vcontrols_seek_bar_bg{position:absolute;left:0;top:0;bottom:0;right:0;background-color:#c0c0c0;opacity:0.825;}.iex_mpreview_vcontrols_seek_bar_loaded{position:absolute;left:0;top:0;bottom:0;width:0;background:#ffffff;}.iex_mpreview_vcontrols_seek_bar_played{position:absolute;left:0;top:0;bottom:0;width:0;background:#66cc33;}.iex_mpreview_vcontrols_seek_time_table{position:relative;display:table;width:100%;height:100%;font-size:0.625em;}.iex_mpreview_vcontrols_seek_time_current{display:table-cell;text-align:left;vertical-align:middle;padding:0em 0.25em 0em 0.375em;}.iex_mpreview_vcontrols_seek_time_duration{display:table-cell;text-align:right;vertical-align:middle;padding:0em 0.375em 0em 0.25em;}.iex_mpreview_vcontrols_seek_time_current,.iex_mpreview_vcontrols_seek_time_duration{color:#101010;text-shadow:0em 0.125em 0.125em #ffffff,0em 0.0625em 0.0625em #ffffff;}.iex_mpreview_vcontrols_volume_container_position{display:block;position:absolute;bottom:100%;right:0;margin-bottom:-0.5em;}.iex_mpreview_vcontrols_volume_container{width:2em;height:9em;position:relative;}.iex_mpreview_vcontrols_volume_container:not(.iex_mpreview_vcontrols_volume_container_visible):not(.iex_mpreview_vcontrols_volume_container_visible_important):not(.iex_mpreview_vcontrols_volume_container_visible_temp){display:none;}.iex_mpreview_vcontrols_volume_bar{position:absolute;left:0.5em;right:0.5em;top:0.5em;bottom:0.5em;border-radius:0.5em;overflow:hidden;cursor:pointer;border:1px solid rgba(0,0,0,0.5);}.iex_mpreview_vcontrols_volume_bar.iex_mpreview_vcontrols_no_border_radius{border-radius:0;}.iex_mpreview_vcontrols_volume_bar_bg{position:absolute;left:0;top:0;bottom:0;right:0;background-color:#ffffff;opacity:0.825;}.iex_mpreview_vcontrols_volume_bar_level{position:absolute;left:0;right:0;bottom:0;height:0;background-color:#ffbb50;}.iex_mpreview_vcontrols_button_container{display:table-cell;vertical-align:bottom;width:2em;}.iex_mpreview_vcontrols_button_container_inner{display:block;width:100%;height:2em;position:relative;}.iex_mpreview_vcontrols_button_container_inner2{display:block;position:absolute;left:0;top:0;bottom:0;right:0;}.iex_mpreview_vcontrols_button_mouse_controller{position:absolute;left:0;top:0;bottom:0;right:0;margin:0.4em;cursor:pointer;}.iex_mpreview_vcontrols_table.iex_mpreview_vcontrols_table_mini:not(.iex_mpreview_vcontrols_table_visible):not(.iex_mpreview_vcontrols_table_visible_temp):not(.iex_mpreview_vcontrols_table_visible_important)>.iex_mpreview_vcontrols_button_container{display:none;}.iex_mpreview_vcontrols_table.iex_mpreview_vcontrols_table_mini:not(.iex_mpreview_vcontrols_table_visible):not(.iex_mpreview_vcontrols_table_visible_temp):not(.iex_mpreview_vcontrols_table_visible_important)>.iex_mpreview_vcontrols_seek_container{min-width:0;}.iex_mpreview_vcontrols_table.iex_mpreview_vcontrols_table_mini:not(.iex_mpreview_vcontrols_table_visible):not(.iex_mpreview_vcontrols_table_visible_temp):not(.iex_mpreview_vcontrols_table_visible_important)>*>.iex_mpreview_vcontrols_seek_container_inner{height:0.125em;}.iex_mpreview_vcontrols_table.iex_mpreview_vcontrols_table_mini:not(.iex_mpreview_vcontrols_table_visible):not(.iex_mpreview_vcontrols_table_visible_temp):not(.iex_mpreview_vcontrols_table_visible_important)>*>*>.iex_mpreview_vcontrols_seek_bar{top:0;bottom:0;border-radius:0;border:0 hidden;}.iex_mpreview_vcontrols_table.iex_mpreview_vcontrols_table_mini:not(.iex_mpreview_vcontrols_table_visible):not(.iex_mpreview_vcontrols_table_visible_temp):not(.iex_mpreview_vcontrols_table_visible_important)>*>*>*>*>.iex_mpreview_vcontrols_seek_time_table{display:none;}.iex_svg_play_button{display:block;}.iex_svg_play_button.iex_svg_play_button_playing>.iex_svg_button_scale_group>.iex_svg_play_button_play_icon,.iex_svg_play_button.iex_svg_play_button_playing>.iex_svg_button_scale_group>.iex_svg_play_button_loop_icon,.iex_svg_play_button:not(.iex_svg_play_button_playing)>.iex_svg_button_scale_group>.iex_svg_play_button_pause_icon,.iex_svg_play_button:not(.iex_svg_play_button_playing).iex_svg_play_button_looping>.iex_svg_button_scale_group>.iex_svg_play_button_play_icon,.iex_svg_play_button:not(.iex_svg_play_button_playing):not(.iex_svg_play_button_looping)>.iex_svg_button_scale_group>.iex_svg_play_button_loop_icon{visibility:hidden;}.iex_svg_volume_button{display:block;}.iex_svg_volume_button.iex_svg_volume_button_muted>.iex_svg_button_scale_group>.iex_svg_volume_button_wave_big,.iex_svg_volume_button.iex_svg_volume_button_muted>.iex_svg_button_scale_group>.iex_svg_volume_button_wave_small,.iex_svg_volume_button:not(.iex_svg_volume_button_muted)>.iex_svg_button_scale_group>.iex_svg_volume_button_wave_mute_icon,.iex_svg_volume_button.iex_svg_volume_button_high>.iex_svg_button_scale_group>.iex_svg_volume_button_wave_small,.iex_svg_volume_button:not(.iex_svg_volume_button_medium)>.iex_svg_button_scale_group>.iex_svg_volume_button_wave_small,.iex_svg_volume_button:not(.iex_svg_volume_button_high)>.iex_svg_button_scale_group>.iex_svg_volume_button_wave_big{visibility:hidden;}.iex_svg_volume_button_wave_mute_icon{opacity:0.75;}.iex_svg_volume_button_wave_mute_icon_polygon{fill:#000000;}.iex_svg_button_fill{fill:rgba(255,255,255,0.825);stroke:rgba(0,0,0,0.5);stroke-width:1px;vector-effect:non-scaling-stroke;}.iex_mpreview_vcontrols_button_mouse_controller:hover+.iex_svg_play_button>.iex_svg_button_scale_group>.iex_svg_button_fill{fill:#44aaff;}.iex_mpreview_vcontrols_button_mouse_controller:hover+.iex_svg_volume_button>.iex_svg_button_scale_group>.iex_svg_button_fill{fill:#ff88aa;}.iex_mpreview_vcontrols_button_mouse_controller:hover+.iex_svg_volume_button>.iex_svg_button_scale_group>.iex_svg_volume_button_wave_mute_icon>.iex_svg_volume_button_wave_mute_icon_polygon{fill:#a00000;}"; // Append style this.stylesheet = $("style"); this.stylesheet.textContent = stylesheet_text; on_ready(on_insert_stylesheet.bind(this, this.stylesheet), on_insert_stylesheet_condition); }, insert_stylesheet_annotations: function () { // Create stylesheet var stylesheet_text = ".iex_quick_reply_extra{margin:0.5em 0.125em 0.25em;}.iex_quick_reply_extra.iex_quick_reply_extra_4chanx{}.iex_quick_reply_label_table{display:table;width:100%;}.iex_quick_reply_label_row{display:table-row;height:100%;}.iex_quick_reply_label_cell{display:table-cell;width:100%;height:100%;vertical-align:middle;}.iex_quick_reply_label_cell.iex_quick_reply_label_cell_small{width:0;}.iex_quick_reply_label,#qr label.iex_quick_reply_label{display:inline-block;vertical-align:middle;text-transform:none;font-size:1em !important;}.iex_quick_reply_label_check{vertical-align:middle;margin:0.125em;padding:0;}.iex_quick_reply_label_check+.riceCheck{vertical-align:middle;}.iex_quick_reply_label_text{vertical-align:middle;}.iex_quick_reply_help_link_container{white-space:nowrap;display:inline-block;vertical-align:middle;}.iex_quick_reply_extra:not(.iex_quick_reply_extra_enabled) .iex_quick_reply_help_link_container{display:none;}.iex_quick_reply_help_link{cursor:pointer;display:inline-block;padding:0 0.125em;}.iex_quick_reply_content{font-size:0.8em;}.iex_quick_reply_extra:not(.iex_quick_reply_extra_enabled)>.iex_quick_reply_content{display:none;}.iex_quick_reply_info{margin:0.25em 0;}.iex_quick_reply_source_selection{}.iex_quick_reply_source_selection_text{vertical-align:middle;}.iex_quick_reply_source_selection_file,.iex_quick_reply_source_selection_url{padding:0.125em 0.25em;vertical-align:middle;font-size:inherit !important;}.iex_quick_reply_source_selection_file.iex_quick_reply_source_selection_file_selected,.iex_quick_reply_source_selection_url.iex_quick_reply_source_selection_url_selected{font-weight:bold;}.iex_quick_reply_source_selection_post{padding:0.125em 0.25em;vertical-align:middle;font-size:inherit !important;width:8em;}.iex_quick_reply_source_selection_post.iex_quick_reply_source_selection_post_selected{font-weight:bold;}.iex_quick_reply_source_selection_post_option{font-weight:bold;font-size:inherit !important}.iex_quick_reply_source_selection_post_option_default{opacity:0.5;font-weight:normal;font-size:inherit !important}.iex_quick_reply_error{margin-top:0.25em;font-weight:bold;font-style:italic;}.iex_quick_reply_error:not(.iex_quick_reply_error_visible){display:none;}.iex_quick_reply_file_input{display:none;visibility:hidden;}#quickReply,#quickReply~.preview,#quickReply~.dd-menu{z-index:20;}.iex_annotation_editor_outer_paddings{padding:1em;}.iex_annotation_editor{display:block;position:fixed;z-index:19;}.iex_annotation_editor:not(.iex_annotation_editor_visible){display:none;}.iex_annotation_editor{box-shadow:0em 0em 0.6em 0.3em rgba(0,0,0,0.75);background-color:rgba(255,255,255,0.75);}.iex_annotation_editor.iex_dark{box-shadow:0em 0em 0.6em 0.3em rgba(255,255,255,0.75);background-color:rgba(0,0,0,0.75);}.iex_ae_t1c1{display:table;table-layout:fixed;width:100%;height:100%;}.iex_ae_t1c2{display:table-row;width:100%;height:100%;}.iex_ae_t1c3{display:table-cell;width:auto;height:100%;vertical-align:top;}.iex_ae_t1c3:not(:first-of-type){width:16em;height:100%;}.iex_ae_t1c4{display:block;width:100%;height:100%;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t1c1{display:block;width:100%;height:100%;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t1c2{display:table;table-layout:fixed;width:100%;height:100%;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t1c3{display:table-row;width:100%;height:100%;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t1c3:not(:first-of-type){width:100%;height:16em;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t1c4{display:table-cell;width:100%;height:inherit;vertical-align:top;}.iex_ae_t2c0{display:block;width:100%;height:100%;box-sizing:border-box;-moz-box-sizing:border-box;xpadding:0 0.25em;border-style:solid;border-color:#111111;border-width:0 0 0 0.125em;background:#eeeeee;}.iex_ae_t2c0.iex_dark{background:#111111;border-color:#eeeeee;}.iex_ae_t2c1{display:block;width:100%;height:100%;}.iex_ae_t2c2{display:table;table-layout:fixed;width:100%;height:100%;}.iex_ae_t2c3{display:table-row;width:100%;height:100%;}.iex_ae_t2c3:first-of-type{width:100%;height:auto;}.iex_ae_t2c4{display:table-cell;width:100%;height:inherit;vertical-align:top;}.iex_ae_t2c3:not(:first-of-type)>.iex_ae_t2c4{vertical-align:bottom;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t2c0{xpadding:0.25em 0;border-width:0.125em 0 0 0;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t2c1{display:table;table-layout:auto;width:100%;height:100%;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t2c2{display:table-row;width:100%;height:100%;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t2c3{display:table-cell;width:100%;height:100%;vertical-align:bottom;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t2c3:first-of-type{width:auto;height:100%;vertical-align:middle;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t2c4{display:inline-block;width:100%;height:100%;vertical-align:middle;}.iex_ae_t3c0{text-align:center;padding:0 0.25em;}.iex_ae_t3c1{display:inline-block;vertical-align:middle;padding:0.25em;text-align:left;border-box;-moz-box-sizing:border-box;}.iex_ae_t3c2{display:block;width:auto;height:100%;border-box;-moz-box-sizing:border-box;}.iex_ae_t3c3{display:table;table-layout:fixed;width:auto;height:100%;}.iex_ae_t3c4{display:table-row;width:100%;height:auto;}.iex_ae_t3c5{display:table-cell;width:100%;height:inherit;vertical-align:top;}.iex_ae_t3c4:first-of-type>.iex_ae_t3c5{text-align:center;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t3c0{padding:0.25em 0;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t3c1{width:100%;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t3c2{display:table;table-layout:auto;width:100%;height:100%;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t3c3{display:table-row;width:100%;height:100%;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t3c4{display:table-cell;width:100%;height:100%;vertical-align:top;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t3c4:first-of-type{width:0;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t3c5{display:block;width:100%;height:100%;box-sizing:border-box;-moz-box-sizing:border-box;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t3c4+.iex_ae_t3c4>.iex_ae_t3c5{padding-left:0.25em;}.iex_ae_t4c0{display:inline-block;vertical-align:middle;}.iex_ae_t4c1{display:table;table-layout:fixed;height:100%;}.iex_ae_t4c2{display:table-row;width:100%;height:100%;}.iex_ae_t4c3{display:table-cell;width:auto;height:100%;vertical-align:middle;}.iex_ae_t4c4{display:block;width:100%;white-space:nowrap;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t4c1{height:100%;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t4c2{display:table;table-layout:auto;width:100%;height:100%;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t4c3{display:table-row;width:100%;height:auto;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t4c4{display:table-cell;width:100%;height:inherit;vertical-align:top;text-align:center;}.iex_ae_color_selector{padding:0.125em;cursor:pointer;position:relative;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_color_selector{display:inline-block;vertical-align:middle;}.iex_ae_color_selector_inner{width:1em;height:1em;position:relative;}.iex_ae_color_selector_border{position:absolute;left:0;top:0;right:0;bottom:0;border:1px solid #000000;opacity:0.5;}.iex_ae_color_selector_border:after{content:\"\";display:block;box-sizing:border-box;-moz-box-sizing:border-box;width:100%;height:100%;border:1px solid #ffffff;}.iex_ae_color_selector_border.iex_dark{border-color:#ffffff;}.iex_ae_color_selector_border.iex_dark:after{border-color:#000000;}.iex_ae_color_selector.iex_ae_color_selector_disabled>.iex_ae_color_selector_inner{background-color:#b0b0b0 !important;}.iex_ae_color_selector.iex_ae_color_selector_disabled>.iex_ae_color_selector_inner.iex_dark{background-color:#404040 !important;}.iex_ae_color_selector_border2{position:absolute;left:0;top:0;right:0;bottom:0;border:2px solid #ffffff;}.iex_ae_color_selector_border2_alt{border-color:#808080;}.iex_ae_color_selector:not(.iex_ae_color_selector_selected)>.iex_ae_color_selector_inner>.iex_ae_color_selector_border2{display:none;}.iex_ae_t5c1{display:table;font-size:0.8em;margin-top:0.5em;}.iex_ae_t5c2{display:table-row;}.iex_ae_t5c3{display:table-cell;vertical-align:middle;}.iex_ae_t5c3:nth-of-type(n+2){padding-left:0.5em;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t5c1{display:block;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t5c2{display:block;text-align:center;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t5c3{display:block;padding-left:0;text-align:left;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t5c3+.iex_ae_t5c3{margin-top:1em;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t5c3.iex_ae_t5c3_small_space+.iex_ae_t5c3{margin-top:0.25em;}.iex_ae_new_annotation_button{display:inline-block;width:3em;height:3em;border:0.15em dashed #000000;text-align:center;white-space:nowrap;line-height:0;font-weight:bold;cursor:pointer;}.iex_ae_new_annotation_button.iex_dark{border-color:#ffffff;}.iex_ae_new_annotation_button:hover{background-color:#ffffff;color:#000000;}.iex_ae_new_annotation_button.iex_dark:hover{background-color:#000000;color:#ffffff;}.iex_ae_new_annotation_button{transition:background-color 0.25s ease-in-out 0s,color 0.25s ease-in-out 0s;}.iex_ae_new_annotation_button:before{content:\"\";display:inline-block;vertical-align:middle;width:0;height:100%;}.iex_ae_new_annotation_button_inner{display:inline-block;vertical-align:middle;white-space:normal;line-height:normal;}select.iex_ae_selection{display:block;font-size:inherit !important;}.iex_ae_selection:disabled{opacity:0.5;}.iex_ae_selection_option{}.iex_ae_option_label{display:block;white-space:nowrap;}.iex_ae_option_checkbox{vertical-align:middle;margin:0;padding:0;}.iex_ae_option_checkbox+.riceCheck{vertical-align:middle;}.iex_ae_option_label_text{vertical-align:middle;}.iex_ae_option_checkbox:disabled,.iex_ae_option_checkbox:disabled+.riceCheck,.iex_ae_option_checkbox:disabled~.iex_ae_option_label_text{opacity:0.5;}.iex_ae_option_checkbox.iex_ae_option_checkbox_bold:checked+.iex_ae_option_label_text{font-weight:bold;}.iex_ae_option_checkbox.iex_ae_option_checkbox_italic:checked+.iex_ae_option_label_text{font-style:italic;}.iex_ae_option_label+.iex_ae_option_label,.iex_ae_selection+.iex_ae_selection{margin-top:0.25em;}.iex_ae_t6c0{display:block;width:100%;height:100%;position:relative;}.iex_ae_t6c1{position:absolute;left:0;top:0;bottom:0;right:0;overflow-x:hidden;overflow-y:auto;white-space:0;line-height:0;}.iex_ae_t6c1:after{content:\"\";display:inline-block;vertical-align:bottom;width:0;height:100%;}.iex_ae_t6c2{display:inline-block;vertical-align:bottom;white-space:normal;box-sizing:border-box;-moz-box-sizing:border-box;line-height:normal;width:100%;padding:0 0.25em 0.25em;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t6c2{padding-top:0.25em;}.iex_ae_add_annotation_container_empty{text-align:center;}.iex_ae_add_annotation_container_empty:not(.iex_ae_add_annotation_container_empty_visible){display:none;}.iex_ae_add_annotation_container_empty_link{cursor:pointer;}.iex_ae_restore_annotations_container_empty{text-align:center;}.iex_ae_restore_annotations_container_empty:not(.iex_ae_restore_annotations_container_empty_visible){display:none;}.iex_ae_restore_annotations_container_empty_link{cursor:pointer;}.iex_ae_annotation_editor{}.iex_ae_annotation_editor+.iex_ae_annotation_editor{margin-top:0.5em;}.iex_ae_annotation_editor.iex_ae_annotation_editor_selected{}.iex_ae_annotation_editor_top{display:table;width:100%;font-size:0.8em;}.iex_ae_annotation_editor_top_row{display:table-row;height:100%;}.iex_ae_annotation_editor_top_cell{display:table-cell;height:100%;width:100%;white-space:nowrap;text-align:right;vertical-align:bottom;}.iex_ae_annotation_editor_top_cell:first-child{width:0;text-align:left;}.iex_ae_annotation_editor_number{display:inline-block;line-height:1em;opacity:0.8;}.iex_ae_annotation_button{position:relative;display:inline-block;cursor:pointer;width:1.5em;height:1.5em;margin-left:0.25em;background-color:#e5e5e5;color:#111111;border:1px solid #aaaaaa;border-bottom:none;text-align:center;line-height:0;vertical-align:bottom;}.iex_ae_annotation_button.iex_dark{color:#eeeeee;}.iex_ae_annotation_button.iex_dark{background-color:#1a1a1a;border-color:#555555;}.iex_ae_annotation_button:after{content:\"\";display:inline-block;vertical-align:middle;width:0;height:100%;}.iex_ae_annotation_button_inner{display:inline-block;vertical-align:middle;line-height:1em;height:1em;position:relative;}.iex_ae_annotation_button.iex_ae_annotation_button_small_text>.iex_ae_annotation_button_inner{font-size:0.8em;line-height:1em;}.iex_ae_annotation_button>.iex_ae_annotation_button_inner_color{display:block;position:absolute;left:0;top:0;right:0;bottom:0;padding:0.25em;height:auto;}.iex_ae_annotation_button>.iex_ae_annotation_button_color_text{font-size:0.8em;text-shadow:1px 1px 0 #ffffff;}.iex_ae_annotation_button>.iex_ae_annotation_button_color_text.iex_dark{text-shadow:1px 1px 0 #000000;}.iex_ae_annotation_button_color{display:block;width:100%;height:100%;box-sizing:border-box;-moz-box-sizing:border-box;border:1px solid rgba(0,0,0,0.5);}.iex_ae_annotation_button_color.iex_dark{border-color:rgba(255,255,255,0.5);}.iex_ae_annotation_button_color:after{content:\"\";display:block;width:100%;height:100%;box-sizing:border-box;-moz-box-sizing:border-box;border:1px solid rgba(255,255,255,0.5);}.iex_ae_annotation_button_color.iex_dark:after{border-color:rgba(0,0,0,0.5);}.iex_ae_annotation_button.iex_ae_annotation_button_align{position:relative;}.iex_ae_annotation_button.iex_ae_annotation_button_align>.iex_ae_annotation_button_inner{position:absolute;width:33.33%;height:33.33%;background-color:#000000;opacity:0.5;}.iex_ae_annotation_button.iex_ae_annotation_button_align>.iex_ae_annotation_button_inner.iex_dark{background-color:#ffffff;}.iex_ae_annotation_button.iex_ae_annotation_button_align_left>.iex_ae_annotation_button_inner{left:0;}.iex_ae_annotation_button.iex_ae_annotation_button_align_right>.iex_ae_annotation_button_inner{right:0;}.iex_ae_annotation_button.iex_ae_annotation_button_align_center>.iex_ae_annotation_button_inner{left:50%;margin-left:-16.66%}.iex_ae_annotation_button.iex_ae_annotation_button_align_justify>.iex_ae_annotation_button_inner{left:0;width:100%;}.iex_ae_annotation_button.iex_ae_annotation_button_valign_top>.iex_ae_annotation_button_inner{top:0;}.iex_ae_annotation_button.iex_ae_annotation_button_valign_bottom>.iex_ae_annotation_button_inner{bottom:0;}.iex_ae_annotation_button.iex_ae_annotation_button_valign_middle>.iex_ae_annotation_button_inner{top:50%;margin-top:-16.66%}.iex_ae_annotation_editor_bottom{position:relative;}.iex_ae_annotation_editor_bottom:after{content:\"\";display:block;position:absolute;left:0;top:0;bottom:0;right:0;border:0.125em dashed #c0c0c0;pointer-events:none;visibility:hidden;}input.iex_ae_annotation_editor_input_text{position:relative;font-size:inherit !important;font-family:inherit !important;padding:0.25em !important;border:1px solid #808080 !important;background-color:#dddddd !important;box-sizing:border-box;-moz-box-sizing:border-box;width:100%;background-color:#d0d0d0;}input.iex_ae_annotation_editor_input_text:hover{background-color:#c0c0c0 !important;}input.iex_ae_annotation_editor_input_text.iex_dark{background-color:#202020 !important;}input.iex_ae_annotation_editor_input_text.iex_dark:hover{background-color:#303030 !important;}input[type=text].iex_ae_annotation_editor_input_text:focus{border-color:#404040 !important;}.iex_ae_annotation_editor_input_text.iex_dark{background-color:#222222 !important;}input[type=text].iex_ae_annotation_editor_input_text.iex_dark:focus{border-color:#b0b0b0 !important;}.iex_ae_annotation_editor.iex_ae_annotation_editor_selected>.iex_ae_annotation_editor_bottom:after{visibility:visible;}.iex_ae_annotation_editor.iex_ae_annotation_editor_selected>.iex_ae_annotation_editor_bottom>input.iex_ae_annotation_editor_input_text[type=text],.iex_ae_annotation_editor.iex_ae_annotation_editor_selected>.iex_ae_annotation_editor_bottom>input.iex_ae_annotation_editor_input_text[type=text]:focus{border-color:rgba(0,0,0,0) !important;}.iex_ae_t7c0{font-size:0.8em;text-align:left;margin-bottom:0.375em;}.iex_ae_t7c1{display:inline-block;vertical-align:baseline;}.iex_ae_t7c1+.iex_ae_t7c1{margin-left:0.75em;}.iex_ae_top_color_link{cursor:pointer;border-bottom:0.125em solid rgba(0,0,0,0);}.iex_ae_top_color_link.iex_ae_top_color_link_selected{border-bottom-color:#000000;color:#000000;}.iex_ae_top_color_link.iex_ae_top_color_link_selected.iex_dark{border-bottom-color:#ffffff;color:#ffffff;}.iex_ae_top_label{font-weight: bold;}.iex_ae_top_hyphen{display:none;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_top_hyphen{display:inline;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t7c0{margin-bottom:0;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_t7c1{margin-left:0;margin-bottom:0.375em;display:block;text-align:center;}.iex_annotation_editor.iex_annotation_editor_vertical .iex_ae_top_label{display:block;}.iex_ae_preview_container{position:relative;width:100%;height:100%;}.iex_ae_preview_image{border:none;padding:0;margin:0;display:block;}.iex_ae_preview_image:not(.iex_ae_preview_image_visible){visibility:hidden;}.iex_ae_preview_image.iex_ae_preview_image_visible{width:100%;height:100%;}.iex_ae_preview_message{position:absolute;left:0;top:0;bottom:0;right:0;white-space:0;line-height:0;text-align:center;}.iex_ae_preview_message:before{content:\"\";display:inline-block;vertical-align:middle;width:0;height:100%;}.iex_ae_preview_message_inner{display:inline-block;vertical-align:middle;white-space:normal;line-height:normal;font-weight:bold;color:#000000;text-shadow:0 0 0.125em #ffffff;}.iex_ae_preview_message_inner.iex_dark{color:#ffffff;text-shadow:0 0 0.125em #000000;}.iex_annotation_overlay{position:absolute;left:0;top:0;bottom:0;right:0;text-align:left;}.iex_annotation_container{position:relative;width:100%;height:100%;transform-origin:0 0;-o-transform-origin:0 0;-moz-transform-origin:0 0;-webkit-transform-origin:0 0;}.iex_ae_preview_scroll_h{position:absolute;left:0;right:0;bottom:0;height:0.5em;pointer-events:none;}.iex_ae_preview_scroll_h_inner{height:100%;}.iex_ae_preview_scroll_v{position:absolute;top:0;bottom:0;right:0;width:0.5em;pointer-events:none;}.iex_ae_preview_scroll_v_inner{width:100%;}.iex_ae_preview_scroll_h:not(.iex_ae_preview_scroll_visible),.iex_ae_preview_scroll_v:not(.iex_ae_preview_scroll_visible),.iex_ae_preview_scroll_h:not(.iex_ae_preview_scroll_displayable),.iex_ae_preview_scroll_v:not(.iex_ae_preview_scroll_displayable){display:none;}.iex_ae_preview_scroll_h_inner,.iex_ae_preview_scroll_v_inner{position:absolute;border-radius:0.25em;background-color:#ffffff;box-sizing:border-box;-moz-box-sizing:border-box;border:1px solid #000000;opacity:0.25;}.iex_ae_preview_scroll_h_inner.iex_dark,.iex_ae_preview_scroll_v_inner.iex_dark{background-color:#000000;border-color:#ffffff;}.iex_ae_preview_scroll_h.iex_ae_preview_scroll_displayable.iex_ae_preview_scroll_visible+.iex_ae_preview_scroll_v{bottom:0.5em;}.iex_annotation{position:absolute;z-index:0;pointer-events:auto;}.iex_annotation.iex_annotation_scaling_px{}.iex_annotation:hover{z-index:1}.iex_annotation.iex_annotation_selected{z-index:2;}.iex_annotation_borders{position:absolute;left:0;top:0;bottom:0;right:0;}.iex_annotation_border{position:absolute;}.iex_annotation_border_top_left{cursor:nwse-resize;left:-16px;top:-16px;width:32px;height:32px;}.iex_annotation_border_top_right{cursor:nesw-resize;right:-16px;top:-16px;width:32px;height:32px;}.iex_annotation_border_bottom_left{cursor:nesw-resize;left:-16px;bottom:-16px;width:32px;height:32px;}.iex_annotation_border_bottom_right{cursor:nwse-resize;right:-16px;bottom:-16px;width:32px;height:32px;}.iex_annotation_border_top{cursor:ns-resize;left:16px;right:16px;top:-16px;height:32px;}.iex_annotation_border_bottom{cursor:ns-resize;left:16px;right:16px;bottom:-16px;height:32px;}.iex_annotation_border_left{cursor:ew-resize;left:-16px;width:32px;top:16px;bottom:16px;}.iex_annotation_border_right{cursor:ew-resize;right:-16px;width:32px;top:16px;bottom:16px;}.iex_annotation_content{position:absolute;left:0;top:0;bottom:0;right:0;line-height:0;white-space:nowrap;text-align:center;}.iex_annotation.iex_annotation_editing>.iex_annotation_content{cursor:move;}.iex_annotation_content:before{content:\"\";display:inline-block;vertical-align:middle;width:0;height:100%;}.iex_annotation_content2{line-height:normal;white-space:normal;display:inline-block;width:100%;vertical-align:middle;position:relative;pointer-events:none;}.iex_annotation_content3{display:inline-block;line-height:1.2em;cursor:default;pointer-events:auto;}.iex_annotation.iex_annotation_editing .iex_annotation_content3{cursor:move;pointer-events:auto;}.iex_annotation_content3.iex_annotation_content_bold{font-weight:bold;}.iex_annotation_content3.iex_annotation_content_italic{font-style:italic;}.iex_annotation_format_bold{font-weight:bold;}.iex_annotation_content3.iex_annotation_content_bold>.iex_annotation_format_bold{font-weight:normal;}.iex_annotation_format_italic{font-style:italic;}.iex_annotation_content3.iex_annotation_content_italic>.iex_annotation_format_italic{font-style:normal;}.iex_annotation_background{position:absolute;left:0;top:0;bottom:0;right:0;background:#ffffff;opacity:0.25;}.iex_annotation_outline{position:absolute;left:0;top:0;bottom:0;right:0;border:2px solid #000000;opacity:0.25;}.iex_annotation.iex_annotation_selected>.iex_annotation_background{opacity:0.5;}.iex_annotation.iex_annotation_selected>.iex_annotation_outline{border-style:dashed;opacity:0.5;}.iex_annotation:not(.iex_annotation_editing)>.iex_annotation_background{opacity:0.5;transition:opacity 0.25s ease-in-out 0s;}.iex_annotation:not(.iex_annotation_editing):hover>.iex_annotation_background{opacity:0.8;}.iex_annotation:not(.iex_annotation_editing).iex_annotation_transparent>.iex_annotation_background{opacity:0.25;}.iex_annotation:not(.iex_annotation_editing).iex_annotation_transparent:hover>.iex_annotation_background{opacity:0.75;}.iex_annotation:not(.iex_annotation_editing)>.iex_annotation_outline{opacity:0.5;transition:opacity 0.25s ease-in-out 0s;}.iex_annotation:not(.iex_annotation_editing):hover>.iex_annotation_outline{opacity:0.5;}.iex_annotation:not(.iex_annotation_editing).iex_annotation_transparent>.iex_annotation_outline{opacity:0.25;}.iex_annotation:not(.iex_annotation_editing).iex_annotation_transparent:hover>.iex_annotation_outline{opacity:0.8;}.iex_annotation:not(.iex_annotation_editing)>.iex_annotation_content{opacity:1;transition:opacity 0.25s ease-in-out 0s;}.iex_annotation:not(.iex_annotation_editing):hover>.iex_annotation_content{opacity:1;}.iex_annotation:not(.iex_annotation_editing).iex_annotation_transparent>.iex_annotation_content{opacity:0.5;}.iex_annotation:not(.iex_annotation_editing).iex_annotation_transparent:hover>.iex_annotation_content{opacity:1;}.iex_annotation.iex_annotation_align_left>.iex_annotation_content{text-align:left;}.iex_annotation.iex_annotation_align_right>.iex_annotation_content{text-align:right;}.iex_annotation.iex_annotation_align_center>.iex_annotation_content{text-align:center;}.iex_annotation.iex_annotation_align_justify>.iex_annotation_content{text-align:justify;}.iex_annotation.iex_annotation_valign_top>.iex_annotation_content>.iex_annotation_content2{vertical-align:top;}.iex_annotation.iex_annotation_valign_middle>.iex_annotation_content>.iex_annotation_content2{vertical-align:middle;}.iex_annotation.iex_annotation_valign_bottom>.iex_annotation_content>.iex_annotation_content2{vertical-align:bottom;}.iex_annotation.iex_annotation_valign_x>.iex_annotation_content:before{content:none;}.iex_annotation.iex_annotation_align_x>.iex_annotation_content>.iex_annotation_content2,.iex_annotation.iex_annotation_valign_x>.iex_annotation_content>.iex_annotation_content2{position:absolute;display:inline-block;left:0;top:0;width:100%;}.iex_annotation.iex_annotation_valign_x.iex_annotation_valign_top>.iex_annotation_content>.iex_annotation_content2{top:0;}.iex_annotation.iex_annotation_valign_x.iex_annotation_valign_bottom>.iex_annotation_content>.iex_annotation_content2{bottom:0;top:auto;}.iex_annotation.iex_annotation_valign_x.iex_annotation_valign_middle>.iex_annotation_content>.iex_annotation_content2{top:50%;}.iex_annotation.iex_annotation_valign_x.iex_annotation_valign_middle>.iex_annotation_content>.iex_annotation_content2>.iex_annotation_content3{transform:translate(0,-50%);}.iex_annotation.iex_annotation_align_x.iex_annotation_align_left>.iex_annotation_content>.iex_annotation_content2{left:0;}.iex_annotation.iex_annotation_align_x.iex_annotation_align_right>.iex_annotation_content>.iex_annotation_content2{right:0;left:auto;}.iex_annotation.iex_annotation_align_x.iex_annotation_align_center>.iex_annotation_content>.iex_annotation_content2,.iex_annotation.iex_annotation_align_x.iex_annotation_align_justify>.iex_annotation_content>.iex_annotation_content2{left:50%;text-align:left;}.iex_annotation.iex_annotation_align_x.iex_annotation_align_center>.iex_annotation_content>.iex_annotation_content2>.iex_annotation_content3{transform:translate(-50%,0);text-align:center;}.iex_annotation.iex_annotation_align_x.iex_annotation_align_justify>.iex_annotation_content>.iex_annotation_content2>.iex_annotation_content3{transform:translate(-50%,0);text-align:justify;min-width:100%;}.iex_annotation.iex_annotation_align_x.iex_annotation_align_center.iex_annotation_valign_x.iex_annotation_valign_middle>.iex_annotation_content>.iex_annotation_content2>.iex_annotation_content3,.iex_annotation.iex_annotation_align_x.iex_annotation_align_justify.iex_annotation_valign_x.iex_annotation_valign_middle>.iex_annotation_content>.iex_annotation_content2>.iex_annotation_content3{transform:translate(-50%,-50%)}.iex_annotation_standalone_overlay{display:block;position:absolute;z-index:1;pointer-events:none;font-size:1px;}"; // Append style this.stylesheet_annotations = $("style"); this.stylesheet_annotations.textContent = stylesheet_text; on_ready(on_insert_stylesheet.bind(this, this.stylesheet_annotations), on_insert_stylesheet_condition); }, theme_detect: function () { // Theme detection var doc_el = document.documentElement, body = document.querySelector("body"), bg_colors, bg_color, e, i, j, a, a_inv; if (doc_el && body) { // Get background colors e = $("div"); e.className = "post reply"; bg_colors = [ this.parse_css_color(this.get_true_style(doc_el, "backgroundColor")), this.parse_css_color(this.get_true_style(body, "backgroundColor")), this.parse_css_color(this.get_true_style(e, "backgroundColor")), ]; // Average bg_color = bg_colors[0]; for (i = 1; i < bg_colors.length; ++i) { // Alphas a = bg_colors[i][3]; a_inv = (1.0 - a) * bg_color[3]; // Modify for (j = 0; j < 3; ++j) { bg_color[j] = (bg_color[j] * a_inv + bg_colors[i][j] * a); } bg_color[3] = Math.max(bg_color[3], a); } if (bg_color[3] > 0) { // Found return (bg_color[0] + bg_color[1] + bg_color[2] < (256 + 128)) ? "iex_dark" : "iex_light"; } } // Not found return null; }, theme_change: function (new_theme) { var theme_old = this.theme.trim(), theme_old_nodes, i, n; // No change if (theme_old == new_theme) return; // Update this.theme = " " + new_theme; // Replace themes theme_old_nodes = document.querySelectorAll("." + theme_old); for (i = 0; i < theme_old_nodes.length; ++i) { n = theme_old_nodes[i]; this.remove_class(n, theme_old); n.className += this.theme; } }, has_class: function (element, classname) { return (new RegExp("(\\s|^)" + classname + "(\\s|$)")).test(element.className); }, add_class: function (element, classname) { if (element.classList) { element.classList.add(classname); } else { if (!(new RegExp("(\\s|^)" + classname + "(\\s|$)")).test(element.className)) { element.className = (element.className + " " + classname).trim(); } } }, remove_class: function (element, classname) { if (element.classList) { // classList element.classList.remove(classname); } else { // Regex var reg = new RegExp("(\\s|^)" + classname + "(\\s|$)", "g"), newCls = element.className.replace(reg, " ").trim(); if (newCls != element.className) element.className = newCls; } }, remove_classes: function (element, classnames) { // Split var i; classnames = classnames.split(" "); if (element.classList) { // classList for (i = 0; i < classnames.length; ++i) { element.classList.remove(classnames[i]); } } else { // Regex var newCls = element.className, reg; for (i = 0; i < classnames.length; ++i) { reg = new RegExp("(\\s|^)" + classnames[i] + "(\\s|$)", "g"); newCls = newCls.replace(reg, " ").trim(); } if (newCls != element.className) element.className = newCls; } }, has_class_svg: function (element, classname) { return (new RegExp("(\\s|^)" + classname + "(\\s|$)")).test(element.getAttribute("class")); }, add_class_svg: function (element, classname) { if (!(new RegExp("(\\s|^)" + classname + "(\\s|$)")).test(element.getAttribute("class"))) { element.setAttribute("class", (element.getAttribute("class") + " " + classname).trim()); } }, remove_class_svg: function (element, classname) { // Regex var reg = new RegExp("(\\s|^)" + classname + "(\\s|$)", "g"), oldCls = element.getAttribute("class"), newCls = oldCls.replace(reg, " ").trim(); if (newCls != oldCls) element.setAttribute("class", newCls); }, remove_classes_svg: function (element, classnames) { var oldCls = element.getAttribute("class"), newCls = oldCls, i, reg; // Split classnames = classnames.split(" "); for (i = 0; i < classnames.length; ++i) { reg = new RegExp("(\\s|^)" + classnames[i] + "(\\s|$)", "g"); newCls = newCls.replace(reg, " ").trim(); } if (newCls != oldCls) element.setAttribute("class", newCls); }, change_classes_svg: function (element, classnames_add, classnames_remove) { var oldCls = element.getAttribute("class"), newCls = oldCls, i, reg; // Add classes if (classnames_add) { classnames_add = classnames_add.split(" "); for (i = 0; i < classnames_add.length; ++i) { reg = new RegExp("(\\s|^)" + classnames_add[i] + "(\\s|$)"); if (!reg.test(newCls)) { newCls += " " + classnames_add[i]; } } } // Remove classes if (classnames_remove) { classnames_remove = classnames_remove.split(" "); for (i = 0; i < classnames_remove.length; ++i) { reg = new RegExp("(\\s|^)" + classnames_remove[i] + "(\\s|$)", "g"); newCls = newCls.replace(reg, " "); } } // Update class if ((newCls = newCls.trim()) != oldCls) { element.setAttribute("class", newCls); } }, get_true_style_of_class: function (class_name, style_name, tag_name, parent_node) { var e = $(tag_name || "div"), s, v; // Set class e.className = class_name; // Add if (parent_node || (parent_node = document.querySelector("body"))) { parent_node.appendChild(e); v = get_true_style(e, style_name); parent_node.removeChild(e); } else { v = get_true_style(e, style_name); } // Return style return v; }, get_true_style: get_true_style, parse_css_color: function (color) { if (/^transparent$/.test(color)) return [ 0 , 0 , 0 , 0 ]; var m; if ((m = /^#([0-9a-fA-F]{2})([0-9a-fA-F]{2})([0-9a-fA-F]{2})$/.exec(color))) { return [ parseInt(m[1], 16), parseInt(m[2], 16), parseInt(m[3], 16), 1 ]; } else if ((m = /^#([0-9a-fA-F])([0-9a-fA-F])([0-9a-fA-F])$/.exec(color))) { return [ Math.floor((parseInt(m[1], 16) * 255.0) / 0xF + 0.5), Math.floor((parseInt(m[2], 16) * 255.0) / 0xF + 0.5), Math.floor((parseInt(m[3], 16) * 255.0) / 0xF + 0.5), 1 ]; } else if ((m = /^rgb\(\s*([0-9]+)\s*,\s*([0-9]+)\s*,\s*([0-9]+)\s*\)$/.exec(color))) { return [ parseInt(m[1], 10), parseInt(m[2], 10), parseInt(m[3], 10), 1 ]; } else if ((m = /^rgba\(\s*([0-9]+)\s*,\s*([0-9]+)\s*,\s*([0-9]+)\s*,\s*([0-9\.]+)\s*\)$/.exec(color))) { return [ parseInt(m[1], 10), parseInt(m[2], 10), parseInt(m[3], 10), parseFloat(m[4]) ]; } else { // Default return [ 0 , 0 , 0 , 0 ]; } }, color_to_css: function (color) { return "rgba(" + color[0] + "," + color[1] + "," + color[2] + "," + color[3] + ")"; }, get_window_rect: function () { var doc_el = document.documentElement, left = (window.pageXOffset || doc_el.scrollLeft || 0) - (doc_el.clientLeft || 0), top = (window.pageYOffset || doc_el.scrollTop || 0) - (doc_el.clientTop || 0); return { left: left, top: top, right: left + (window.innerWidth || doc_el.clientWidth || 0), bottom: top + (window.innerHeight || doc_el.clientHeight || 0), }; }, get_document_rect: function () { var doc_el = document.documentElement, left = (window.pageXOffset || doc_el.scrollLeft || 0) - (doc_el.clientLeft || 0), top = (window.pageYOffset || doc_el.scrollTop || 0) - (doc_el.clientTop || 0); return { left: left, top: top, right: left + (doc_el.clientWidth || window.innerWidth || 0), bottom: top + (doc_el.clientHeight || window.innerHeight || 0), }; }, get_document_size: function () { var doc_el = document.documentElement; return { width: (doc_el.clientWidth || window.innerWidth || 0), height: (doc_el.clientHeight || window.innerHeight || 0), }; }, get_document_offset: function () { var doc_el = document.documentElement; return { left: (window.pageXOffset || doc_el.scrollLeft || 0) - (doc_el.clientLeft || 0), top: (window.pageYOffset || doc_el.scrollTop || 0) - (doc_el.clientTop || 0), }; }, get_object_rect: function (obj) { var bounds = obj.getBoundingClientRect(), doc_el = document.documentElement, left = (window.pageXOffset || doc_el.scrollLeft || 0) - (doc_el.clientLeft || 0), top = (window.pageYOffset || doc_el.scrollTop || 0) - (doc_el.clientTop || 0); return { left: left + bounds.left, top: top + bounds.top, right: left + bounds.right, bottom: top + bounds.bottom, width: bounds.width, height: bounds.height, }; }, get_object_rect_relative: function (obj) { return obj.getBoundingClientRect(); }, get_object_size: function (obj) { var bounds = obj.getBoundingClientRect(); return { width: bounds.width, height: bounds.height, }; }, get_object_rect_inner: function (obj) { // Document scroll offset var doc_el = document.documentElement, left = (window.pageXOffset || doc_el.scrollLeft || 0) - (doc_el.clientLeft || 0), top = (window.pageYOffset || doc_el.scrollTop || 0) - (doc_el.clientTop || 0), bounds0, bounds1, bounds2, b1_w, b1_h; // Measure bounds with and without paddings bounds0 = obj.getBoundingClientRect(); this.add_class(obj, "iex_no_padding_br"); bounds1 = obj.getBoundingClientRect(); this.add_class(obj, "iex_no_padding_tl"); bounds2 = obj.getBoundingClientRect(); this.remove_classes(obj, "iex_no_padding_br iex_no_padding_tl"); b1_w = bounds1.right - bounds1.left; b1_h = bounds1.bottom - bounds1.top; return { left: left + bounds1.left + (b1_w - (bounds2.right - bounds2.left)), top: top + bounds1.top + (b1_h - (bounds2.bottom - bounds2.top)), right: left + bounds0.right + (b1_w - (bounds0.right - bounds0.left)), bottom: top + bounds0.bottom + (b1_h - (bounds0.bottom - bounds0.top)), }; }, get_prefixed_style: function (name) { // Returns var e = $("div"), prefixes, prefixes2, name2, n, i; name2 = name.replace(/[A-Z]/g, function (match) { return "-" + match.toLowerCase(); }); name = name.replace(/-[a-z]/g, function (match) { return match[1].toUpperCase(); }); // Un-prefixed if (typeof(e.style[name]) === "string") { return [ name, name2 ]; } // Check prefixes prefixes = [ "webkit" , "Moz" , "O" , "ms" ]; prefixes2 = [ "-webkit-" , "-moz-" , "-o-" , "-ms-" ]; name = name[0].toUpperCase() + name.substr(1); for (i = 0; i < prefixes.length; ++i) { n = prefixes[i] + name; if (typeof(e.style[n]) === "string") { return [ n, prefixes2[i] + name2 ]; } } name = name[0].toLowerCase() + name.substr(1); return [ name, name2 ]; }, add_font: function (font_name, external_name, styles) { if (loaded_fonts !== null && font_name in loaded_fonts) return true; var styles_names = [ "r", "i", "b", "bi" ], styles_selected = [ false, false, false, false ], styles_count = 0, base_url = "https://dnsev.github.io/web/fonts/", src = "", url, i, j, f_weight, f_style, n, p; // Parent node if (!(p = doc.head || doc.body || doc_el)) return false; // External name base_url += (external_name ? external_name : font_name) + "/"; // Styles to load if (styles) { for (i = 0; i < styles.length; ++i) { j = styles_names.indexOf(arguments[i]); if (j >= 0) { styles_selected[j] = true; ++styles_count; } } } // Must load at least one if (styles_count === 0) styles_selected[0] = true; // Prepare source string for (i = 0; i < styles_names.length; ++i) { if (!styles_selected[i]) continue; url = base_url + styles_names[i] + "/webfont"; f_weight = ((i < 2) ? "normal" : "bold"); f_style = ((i % 2) === 0 ? "normal" : "italic"); src += "@font-face{" + "font-family:'" + font_name + "';" + "src:url('" + url + ".eot');" + "src:url('" + url + ".eot?#iefix') format('embedded-opentype')," + "url('" + url + ".woff2') format('woff2')," + "url('" + url + ".woff') format('woff')," + "url('" + url + ".ttf') format('truetype')," + "url('" + url + ".svg#" + external_name + "') format('svg');" + "font-weight:" + f_weight + ";" + "font-style:" + f_style + ";" + "}"; } // Create node n = document.createElement("style"); n.textContent = src; // State if (loaded_fonts === null) loaded_fonts = {}; loaded_fonts[font_name] = n; // Add to doc p.appendChild(n); }, }; return Style; })(); // Class to manage hotkey interactions var HotkeyManager = (function () { // Hotkey registration class var Hotkey = function (key, keyCode, modifiers, callback) { this.key = key; this.keyCode = keyCode; this.modifiers = modifiers; this.callback = callback; }; // Setup key conversions var key_conversions = { "backspace": 8, "tab": 9, "enter": 13, "shift": 16, "ctrl": 17, "alt": 18, "pause": 19, "capslock": 20, "esc": 27, "space": 32, "page up": 33, "page down": 34, "end": 35, "home": 36, "left": 37, "up": 38, "right": 39, "down": 40, "insert": 45, "delete": 46, ",": 188, ".": 190, "/": 191, "`": 192, "[": 219, "\\": 220, "]": 221, "'": 222, ";": (is_firefox || is_opera ? 59 : 186), "=": (is_firefox ? 107 : (is_opera ? 61 : 187)), "-": (is_firefox || is_opera ? 109 : 189), "numpad0": 96, "numpad1": 97, "numpad2": 98, "numpad3": 99, "numpad4": 100, "numpad5": 101, "numpad6": 102, "numpad7": 103, "numpad8": 104, "numpad9": 105, "numpad*": 106, "numpad+": 107, "numpad-": 109, "numpad.": 110, "numpad/": 111, }; var i, iEnd, iDiff; for (i = "a".charCodeAt(0), iEnd = "z".charCodeAt(0), iDiff = (i - "A".charCodeAt(0)); i <= iEnd; ++i) { key_conversions[String.fromCharCode(i)] = i - iDiff; } for (i = "0".charCodeAt(0), iEnd = "9".charCodeAt(0); i <= iEnd; ++i) { key_conversions[String.fromCharCode(i)] = i; } for (i = 112; i <= 123; ++i) { key_conversions["f" + (i - 112)] = i; } // Constructor var HotkeyManager = function () { // Event document.addEventListener("keydown", this.on_window_keydown_bound = this.on_window_keydown.bind(this), false); // Registrations this.registrations = {}; }; // Static values HotkeyManager.instance = null; HotkeyManager.MODIFIER_NONE = 0; HotkeyManager.MODIFIER_SHIFT = 1; HotkeyManager.MODIFIER_CTRL = 2; HotkeyManager.MODIFIER_ALT = 4; HotkeyManager.MODIFIER_TYPING = 8; // Public functions HotkeyManager.prototype = { constructor: HotkeyManager, /** Destroy any events and/or data created by the instance that will not be necessary after this object is no longer used. */ destroy: function () { // Remove event listener document.removeEventListener("keydown", this.on_window_keydown_bound, false); // Remove registrations this.registrations = {}; }, /** Convert a string-based key to an integer key code. @param key The string representing a key This should be lowercase @return An integer value representing the key, or 0 if not found */ key_to_keycode: function (key) { // Convert the key if (key in key_conversions) { return key_conversions[key]; } else { return 0; } }, /** Register a hotkey event. @param key The key to use, as a lowercase string. If the string contains a non-alphabetic character, use whichever symbol is on the key with the shift key NOT held. Check key_conversions for any other issues. @param [modifiers] Bitflags of which key modifiers are required. Available values are: HotkeyManager.MODIFIER_NONE HotkeyManager.MODIFIER_SHIFT HotkeyManager.MODIFIER_CTRL HotkeyManager.MODIFIER_ALT HotkeyManager.MODIFIER_TYPING Optional parameter @param callback A callback function, which will be called using: callback.call(HotkeyManager.instance, keydown_event, hotkey_instance) If the callback function returns false, no other hotkeys registered for this key will recieve the event. The default action of the event can be canceled with: event.preventDefault() event.stopPropagation() @return An object describing the registered hotkey. This value can be passed to the unregister function to remove the hotkey. */ register: function (key, modifiers, callback) { // Argument shifting if (arguments.length < 3) { modifiers = 0; callback = arguments[1]; } // Get key code var keyCode = this.key_to_keycode(key); // Create registration var reg = new Hotkey(key, keyCode, modifiers, callback); // Add to or create a list if (keyCode in this.registrations) { this.registrations[keyCode].push(reg); } else { this.registrations[keyCode] = [ reg ]; } // Return return reg; }, /** Unregister a hotkey event. @param hotkey The hotkey registration value returned from calling the register function. @return true if the hotkey was removed, false if it was not found (maybe already removed) */ unregister: function (hotkey) { if (hotkey.keyCode in this.registrations) { var list = this.registrations[hotkey.keyCode]; for (var i = 0; i < list.length; ++i) { if (list[i] === hotkey) { // Successfully unregistered list.splice(i, 1); if (list.length === 0) { // Remove if empty delete this.registrations[hotkey.keyCode]; } return true; } } } // Not found return false; }, /** The event handler for the document.onkeydown event. This will propagate the event to any hotkeys if necessary. @param event The event event passed in */ on_window_keydown: function (event) { // Key and modifiers var key = event.which || event.keyCode || (event = window.event).keyCode; // Check if (key in this.registrations) { // Get modifiers var modifiers = (event.shiftKey ? 1 : 0) | (event.ctrlKey ? 2 : 0) | (event.altKey ? 4 : 0); // Check if typing var t = document.activeElement; if (t) { t = t.tagName; if (t == "INPUT" || t == "TEXTAREA") { modifiers |= 8; } } // Get registration list var list = this.registrations[key]; for (var i = 0; i < list.length; ++i) { // Check modifiers if (modifiers === list[i].modifiers) { // Callback if (list[i].callback.call(this, event, list[i]) === false) { break; } } } } } }; // Return the class return HotkeyManager; })(); // Class to re-linkify file links var FileLink = (function () { var FileLink = function () { // Enabled this.enabled = false; this.disabled = false; this.post_queue = null; this.modifiers = []; // Binds this.on_api_post_add_bind = on_api_post_add.bind(this); this.on_api_file_info_update_bind = on_api_file_info_update.bind(this); }; var modify_href = function (node, filename) { var href = node.getAttribute("href"), i; if (href !== null) { href = href.replace(/#.*/, "") + "#!" + this.escape(filename); for (i = 0; i < this.modifiers.length; ++i) { href = this.modifiers[i].call(this, node, href); } node.setAttribute("href", href); } }; var on_api_post_add = function (post_container) { // Queue add hooks this.post_queue.push(post_container); }; var on_api_file_info_update = function (node) { var c = api.post_get_file_node_link_from_file_info(node), fn, n; if ( c !== null && (c.getAttribute("href") || "").indexOf("#") < 0 && (n = api.post_get_file_container_from_file_info(node)) !== null && (fn = api.post_get_file_info_from_file_info_container(n).name) ) { modify_href.call(this, c, fn); } }; var on_post_queue_callback = function (post_container) { // Find file var file_nodes = api.post_get_file_nodes(post_container), file_info; if ( file_nodes && (file_info = api.post_get_file_info(post_container)).name ) { // Modify hrefs if (file_nodes.link !== null) { modify_href.call(this, file_nodes.link, file_info.name); } if (file_nodes.link_thumbnail !== null) { modify_href.call(this, file_nodes.link_thumbnail, file_info.name); } } }; FileLink.prototype = { constructor: FileLink, destroy: function () { // Remove api events api.off("post_add", this.on_api_post_add_bind); api.off("file_info_update", this.on_api_file_info_update_bind); }, start: function () { // Enable if settings allow it if (settings.values.file_linkification.enabled) { this.enable(); } }, enable: function () { // Enable if (this.enabled) return; this.enabled = true; // Get posts this.post_queue = Delay.queue(api.get("posts"), on_post_queue_callback.bind(this), 50, 0.25); // Bind post acquiring api.on("post_add", this.on_api_post_add_bind); api.on("file_info_update", this.on_api_file_info_update_bind); }, disable: function () { // Not enabled or already disabled if (!this.enabled || this.disabled) return; // Disable this.disabled = true; // Unbind post acquiring api.off("post_add", this.on_api_post_add_bind); api.off("file_info_update", this.on_api_file_info_update_bind); }, escape: function (s) { return encodeURIComponent(s).replace(/\%20/g, "+"); }, unescape: function (s) { return decodeURIComponent(s.replace(/\+/g, "%20")); }, register_modifier: function (modifier) { this.modifiers.push(modifier); }, }; return FileLink; })(); // Class to manage files in a separate window var FileView = (function () { var FileView = function () { this.events = { "image": [], "video": [], }; }; var trigger_event = function (event, data) { var list = this.events[event], i; for (i = 0; i < list.length; ++i) { list[i].call(null, data); } }; FileView.prototype = { constructor: FileView, start: function () { // Auto-loop .webm's var n = document.body.querySelector("video"); if (n) { n.loop = true; trigger_event.call(this, "video", n); } else if ((n = document.body.querySelector("img")) !== null) { trigger_event.call(this, "image", n); } }, on: function (event, callback) { if (event in this.events) { this.events[event].push(callback); return true; } return false; }, off: function (event, callback) { if (event in this.events) { var list = this.events[event], i; for (i = 0; i < list.length; ++i) { if (callback === list[i]) { list.splice(i, 1); return true; } } } return false; }, }; return FileView; })(); // Hover container control var Hover = (function () { var Hover = function () { this.container = null; }; var on_asap = function () { var body = document.querySelector("body"); // Parent if (body) { // Create floating container this.container = $.div("iex_floating_container"); // Append body.appendChild(this.container); } }; var on_asap_condition = function () { return document.querySelector("body"); }; Hover.prototype = { constructor: Hover, destroy: function () { // Remove container var par = this.container.parentNode; if (par) par.removeChild(this.container); this.container = null; }, start: function () { // Setup on_ready(on_asap.bind(this), on_asap_condition); }, }; return Hover; })(); // Image hover control var ImageHover = (function () { var ImageHover = function (hover) { // Preview object this.mpreview = null; // Enabled this.enabled = false; this.disabled = false; this.current_image_container = null; this.current_hover_container = null; // Post queue this.post_queue = null; // Extension settings this.extensions_valid = { ".jpg": { ext: "jpg", mime: "image/jpeg", type: "image", }, ".jpeg": { ext: "jpg", mime: "image/jpeg", type: "image", }, ".png": { ext: "png", mime: "image/png", type: "image", }, ".gif": { ext: "gif", mime: "image/gif", type: "image", }, ".webm": { ext: "webm", mime: "video/webm", type: "video", }, }; // Parent node this.hover = hover; this.connector = null; // Open and closing timers this.preview_open_timer = null; this.preview_close_timer = null; this.zoom_borders_hide_timer = null; this.mouse_hide_timer = null; // Events this.events = { "change": [], }; // Event bindings this.on_image_mouseenter_bind = wrap_mouseenterleave_event(this, on_image_mouseenter); this.on_image_mouseleave_bind = wrap_mouseenterleave_event(this, on_image_mouseleave); this.on_image_connector_mouseenter_bind = wrap_mouseenterleave_event(this, on_image_connector_mouseenter); this.on_image_connector_mouseleave_bind = wrap_mouseenterleave_event(this, on_image_connector_mouseleave); this.on_image_click_bind = wrap_generic_event(this, on_image_click); // More bindings this.on_preview_close_timeout_bind = on_preview_close_timeout.bind(this); this.on_preview_zoom_borders_hide_timeout_bind = on_preview_zoom_borders_hide_timeout.bind(this); this.on_preview_mouse_hide_timeout_bind = on_preview_mouse_hide_timeout.bind(this); this.on_preview_image_load_bind = on_preview_image_load.bind(this); this.on_preview_image_error_bind = on_preview_image_error.bind(this); this.on_preview_video_ready_bind = on_preview_video_ready.bind(this); this.on_preview_video_can_play_bind = on_preview_video_can_play.bind(this); this.on_preview_video_can_play_through_bind = on_preview_video_can_play_through.bind(this); this.on_preview_video_load_bind = on_preview_video_load.bind(this); this.on_preview_video_error_bind = on_preview_video_error.bind(this); //this.on_preview_video_volume_change_bind = on_preview_video_volume_change.bind(this); this.on_preview_size_change_bind = on_preview_size_change.bind(this); this.on_preview_mouse_wheel_bind = on_preview_mouse_wheel.bind(this); this.on_preview_mouse_enter_bind = on_preview_mouse_enter.bind(this); this.on_preview_mouse_leave_bind = on_preview_mouse_leave.bind(this); this.on_preview_mouse_down_bind = on_preview_mouse_down.bind(this); this.on_preview_mouse_move_bind = on_preview_mouse_move.bind(this); this.on_preview_mouse_enter_main_bind = on_preview_mouse_enter_main.bind(this); this.on_preview_stats_zoom_control_click_bind = on_preview_stats_zoom_control_click.bind(this); this.on_image_connector_mousedown_bind = on_image_connector_mousedown.bind(this); this.on_window_resize_bind = on_window_resize.bind(this); this.on_window_scroll_bind = on_window_scroll.bind(this); this.on_api_post_add_bind = on_api_post_add.bind(this); this.on_api_post_remove_bind = on_api_post_remove.bind(this); this.on_api_image_hover_open_bind = on_api_image_hover_open.bind(this); // Callbacks this.on_settings_value_change_bind = on_settings_value_change.bind(this); settings.on_change(["image_expansion", "hover", "zoom_invert"], this.on_settings_value_change_bind); settings.on_change(["image_expansion", "hover", "zoom_borders_show"], this.on_settings_value_change_bind); settings.on_change(["image_expansion", "hover", "zoom_borders_hide_time"], this.on_settings_value_change_bind); settings.on_change(["image_expansion", "hover", "zoom_buttons"], this.on_settings_value_change_bind); settings.on_change(["image_expansion", "hover", "mouse_hide"], this.on_settings_value_change_bind); settings.on_change(["image_expansion", "hover", "mouse_hide_time"], this.on_settings_value_change_bind); settings.on_change(["image_expansion", "hover", "header_overlap"], this.on_settings_value_change_bind); settings.on_change(["image_expansion", "hover", "fit_large_allowed"], this.on_settings_value_change_bind); settings.on_change(["image_expansion", "hover", "display_stats"], this.on_settings_value_change_bind); settings.on_change(["image_expansion", "video", "autoplay"], this.on_settings_value_change_bind); settings.on_change(["image_expansion", "video", "loop"], this.on_settings_value_change_bind); settings.on_change(["image_expansion", "video", "mute_initially"], this.on_settings_value_change_bind); settings.on_change(["image_expansion", "video", "volume"], this.on_settings_value_change_bind); settings.on_change(["image_expansion", "video", "mini_controls"], this.on_settings_value_change_bind); settings.on_change(["image_expansion", "video", "expand_state_save"], this.on_settings_value_change_bind); settings.on_change(["image_expansion", "style", "controls_rounded_border"], this.on_settings_value_change_bind); settings.on_change(["image_expansion", "style", "animations_background"], this.on_settings_value_change_bind); // Settings this.hotkeys = null; this.mouse_x = 0; this.mouse_y = 0; this.paddings_active = false; this.display_window_max = { x: 0, y: 0, width: 0, height: 0, }; this.settings = { fit_large_allowed: true, display_stats: 0, // 0: all, 1: some, 2: none zoom_limits: { increase: 2.0, decrease: 2.0, min: 1.0, max: 32.0, }, zoom_margins: { horizontal: 0.1, vertical: 0.1 }, zoom_invert: false, zoom_borders_show: true, zoom_borders_hide_time: 500, zoom_buttons: true, mouse_hide: true, mouse_hide_time: 1000, padding_extra: { left: 10, top: 10, right: 10, bottom: 10 }, header_overlap: false, region_paddings: { left: 20, top: 20, right: 20, bottom: 20 }, video: { autoplay: 1, // 0 = not at all, 1 = asap, 2 = when it can play "through", 3 = fully loaded loop: true, mute_initially: false, volume: 0.5, mini_controls: 1, // 0 = never, 1 = when mouse is over the video, 2 = when mouse is NOT over the video, 3 = always expand_state_save: true }, style: { controls_rounded_border: true, animations_background: true, }, }; }; var log2 = function (x) { return Math.log(x) / Math.LN2; }; var add_post_container_callbacks = function (post_container) { // Find file var image_container = api.post_get_image_container(post_container); if (image_container !== null) { // Add mouseover/mouseout/click listeners image_container.addEventListener("mouseover", this.on_image_mouseenter_bind, false); image_container.addEventListener("mouseout", this.on_image_mouseleave_bind, false); image_container.addEventListener("click", this.on_image_click_bind, false); } }; var remove_post_container_callbacks = function (post_container) { // Find file var image_container = api.post_get_image_container(post_container); if (image_container !== null) { // Remove listeners image_container.removeEventListener("mouseover", this.on_image_mouseenter_bind, false); image_container.removeEventListener("mouseout", this.on_image_mouseleave_bind, false); image_container.removeEventListener("click", this.on_image_click_bind, false); } }; var get_image_url_extension = function (url) { var ext = /(\.[^\.\?#]+)(?:[\?#]|$)/.exec(url); return (ext === null) ? "" : ext[1].toLowerCase(); }; var preview_open_test = function (image_container, post_container, obey_timer) { // Don't open if same if (this.current_image_container === image_container) { this.preview_close_cancel(); return; } // Get info if (post_container === null) post_container = api.post_get_post_container_from_image_container(image_container); var post_info = api.post_get_file_info(post_container); if (!post_info.url) return; // Don't open if expanded if (api.post_is_image_expanded_or_expanding(post_container)) return; // Spoiler check var val_check = (post_info.spoiler === null) ? settings.values.image_expansion.normal : settings.values.image_expansion.spoiler; if (!val_check.enabled) return; // Extension check var ext = get_image_url_extension.call(this, post_info.url); if (!(ext in this.extensions_valid)) return; // Open preview var time = val_check.timeout; if (time === 0 || !obey_timer) { this.preview_open(image_container, post_info, val_check.to_fit); } else { if (this.preview_open_timer !== null) { clearTimeout(this.preview_open_timer); } this.preview_open_timer = setTimeout(on_preview_open_timeout.bind(this, image_container, post_info, val_check.to_fit), time * 1000); } }; var preview_create = function () { var zoom_margins = this.settings.zoom_margins; // Create new this.mpreview = new MediaPreview(); // Setup size this.mpreview.set_view_borders(zoom_margins.horizontal, zoom_margins.vertical); // Hook events this.mpreview.on("image_load", this.on_preview_image_load_bind); this.mpreview.on("image_error", this.on_preview_image_error_bind); this.mpreview.on("video_ready", this.on_preview_video_ready_bind); this.mpreview.on("video_can_play", this.on_preview_video_can_play_bind); this.mpreview.on("video_can_play_through", this.on_preview_video_can_play_through_bind); this.mpreview.on("video_load", this.on_preview_video_load_bind); this.mpreview.on("video_error", this.on_preview_video_error_bind); //this.mpreview.on("video_volume_change", this.on_preview_video_volume_change_bind); this.mpreview.on("size_change", this.on_preview_size_change_bind); this.mpreview.on("mouse_wheel", this.on_preview_mouse_wheel_bind); this.mpreview.on("mouse_enter", this.on_preview_mouse_enter_bind); this.mpreview.on("mouse_leave", this.on_preview_mouse_leave_bind); this.mpreview.on("mouse_down", this.on_preview_mouse_down_bind); this.mpreview.on("mouse_move", this.on_preview_mouse_move_bind); this.mpreview.on("mouse_enter_main", this.on_preview_mouse_enter_main_bind); this.mpreview.on("stats_zoom_control_click", this.on_preview_stats_zoom_control_click_bind); window.addEventListener("resize", this.on_window_resize_bind, false); window.addEventListener("scroll", this.on_window_scroll_bind, false); // Add this.mpreview.add_to(this.hover.container); }; var preview_detach = function (destroy) { if (this.mpreview === null) return; // Unhook events this.mpreview.off("image_load", this.off_preview_image_load_bind); this.mpreview.off("image_error", this.off_preview_image_error_bind); this.mpreview.off("video_ready", this.off_preview_video_ready_bind); this.mpreview.off("video_can_play", this.off_preview_video_can_play_bind); this.mpreview.off("video_can_play_through", this.off_preview_video_can_play_through_bind); this.mpreview.off("video_load", this.off_preview_video_load_bind); this.mpreview.off("video_error", this.off_preview_video_error_bind); this.mpreview.off("video_volume_change", this.off_preview_video_volume_change_bind); this.mpreview.off("size_change", this.off_preview_size_change_bind); this.mpreview.off("mouse_wheel", this.off_preview_mouse_wheel_bind); this.mpreview.off("mouse_enter", this.off_preview_mouse_enter_bind); this.mpreview.off("mouse_leave", this.off_preview_mouse_leave_bind); this.mpreview.off("mouse_down", this.off_preview_mouse_down_bind); this.mpreview.off("mouse_move", this.off_preview_mouse_move_bind); this.mpreview.off("mouse_enter_main", this.off_preview_mouse_enter_main_bind); this.mpreview.off("stats_zoom_control_click", this.off_preview_stats_zoom_control_click_bind); window.removeEventListener("resize", this.on_window_resize_bind, false); window.removeEventListener("scroll", this.on_window_scroll_bind, false); // Nullify if (destroy) this.mpreview.destroy(); this.mpreview = null; }; var change_zoom_level = function (delta) { // Update zoom var set = this.settings, size_update = false, zoom_limits = set.zoom_limits, img_size = this.mpreview.size, disp_size_max = this.display_window_max, zoom = this.mpreview.zoom, fit = this.mpreview.fit, fit_large = this.mpreview.fit_large, fit_large_allowed = set.fit_large_allowed, z, w_scale, h_scale, z_scale, scale; if (delta > 0) { // Larger if (fit) { if (fit_large || !fit_large_allowed) { // Zoom in z = zoom * zoom_limits.increase; if (z > zoom_limits.max) z = zoom_limits.max; if (zoom != z) { // Update zoom = z; size_update = true; } } else { // Zoom in w_scale = disp_size_max.width / img_size.width; h_scale = disp_size_max.height / img_size.height; z_scale = (w_scale < h_scale) ? w_scale : h_scale; z = zoom * zoom_limits.increase; w_scale = disp_size_max.width / (img_size.width * z * z_scale); h_scale = disp_size_max.height / (img_size.height * z * z_scale); if (w_scale <= 1.0 && h_scale <= 1.0) { // Fit large fit_large = true; zoom = 1.0; } else { // Not fit large zoom = z; } // Update size_update = true; } } else { // Zoom in z = zoom * zoom_limits.increase; w_scale = disp_size_max.width / (img_size.width * z); h_scale = disp_size_max.height / (img_size.height * z); if (w_scale <= 1.0 || h_scale <= 1.0) { // Fit fit = true; fit_large = (Math.abs(w_scale - h_scale) < 1e-5) && fit_large_allowed; zoom = 1.0; } else { // Not fit zoom = z; } // Update size_update = true; } } else if (delta < 0) { // Smaller if (fit) { if (zoom == 1.0) { if (fit_large) { // Switch to normal fit w_scale = disp_size_max.width / img_size.width; h_scale = disp_size_max.height / img_size.height; z_scale = (w_scale < h_scale) ? w_scale : h_scale; w_scale = disp_size_max.width / (img_size.width * z_scale); h_scale = disp_size_max.height / (img_size.height * z_scale); scale = (w_scale > h_scale) ? w_scale : h_scale; scale = Math.pow(2, Math.floor(log2(scale))); fit_large = false; zoom = scale; size_update = true; } else { // Stop fitting if possible w_scale = disp_size_max.width / img_size.width; h_scale = disp_size_max.height / img_size.height; if (w_scale > 1.0 && h_scale > 1.0) { // Calculate new zoom scale = (w_scale < h_scale) ? w_scale : h_scale; scale = Math.pow(2, Math.floor(log2(scale))); // Update fit = false; fit_large = false; zoom = scale; size_update = true; } } } else { // Zoom out zoom /= zoom_limits.decrease; if (zoom < 1.0) zoom = 1.0; // Update size_update = true; } } else { // Zoom out z = zoom / zoom_limits.decrease; if (z < zoom_limits.min) z = zoom_limits.min; if (zoom != z) { // Update zoom = z; size_update = true; } } } // Update if (size_update) { // Update this.mpreview.set_zoom(zoom, fit, fit_large); this.preview_update(true, false, false); update_zoom_offset.call(this); } }; var change_zoom_borders_visibility = function (visible) { this.mpreview.set_view_borders_visible(visible); if (visible) { if (this.zoom_borders_hide_timer !== null) clearTimeout(this.zoom_borders_hide_timer); this.zoom_borders_hide_timer = setTimeout(this.on_preview_zoom_borders_hide_timeout_bind, this.settings.zoom_borders_hide_time); } }; var change_mouse_visibility = function (visible) { this.mpreview.set_mouse_visible(visible); if (visible && this.settings.mouse_hide) { if (this.mouse_hide_timer !== null) clearTimeout(this.mouse_hide_timer); this.mouse_hide_timer = setTimeout(this.on_preview_mouse_hide_timeout_bind, this.settings.mouse_hide_time); } }; var update_zoom_offset = function () { // Change zoom offset var rect = this.mpreview.get_inner_rect(), r_w = (rect.right - rect.left), r_h = (rect.bottom - rect.top), zoom_margins = this.settings.zoom_margins, x = (this.mouse_x - (rect.left + zoom_margins.horizontal * r_w)) / (r_w * (1.0 - 2.0 * zoom_margins.horizontal)), y = (this.mouse_y - (rect.top + zoom_margins.vertical * r_h)) / (r_h * (1.0 - 2.0 * zoom_margins.vertical)); if (x < 0) x = 0; else if (x > 1) x = 1; if (y < 0) y = 0; else if (y > 1) y = 1; if (this.settings.zoom_invert) { x = 1.0 - x; y = 1.0 - y; } this.mpreview.set_offset(x, y); }; var on_asap = function () { var body = document.querySelector("body"); // Parent if (body) { // Create connector this.connector = $.div("iex_floating_image_connector"); this.connector.addEventListener("mouseover", this.on_image_connector_mouseenter_bind, false); this.connector.addEventListener("mouseout", this.on_image_connector_mouseleave_bind, false); this.connector.addEventListener("mousedown", this.on_image_connector_mousedown_bind, false); this.hover.container.appendChild(this.connector); } }; var on_asap_condition = function () { return document.querySelector("body"); }; var on_post_queue_callback = function (post_container) { // Add hooks add_post_container_callbacks.call(this, post_container); }; var on_preview_image_load = function (event) { // Hide fallback this.mpreview.set_background_style(0); }; var on_preview_image_error = function (event) { // Display fallback this.mpreview.set_background_style(2); // Show error this.mpreview.set_stat_status(true, event.reason, "error"); }; var on_preview_video_ready = function (event) { // Hide fallback this.mpreview.set_background_style(0); }; var on_preview_video_can_play = function (event) { // Auto-play if (this.settings.video.autoplay == 1 && !this.mpreview.video_interacted()) { this.mpreview.set_video_paused(false); } }; var on_preview_video_can_play_through = function (event) { // Auto-play if (this.settings.video.autoplay == 2 && !this.mpreview.video_interacted()) { this.mpreview.set_video_paused(false); } }; var on_preview_video_load = function (event) { // Auto-play if (this.settings.video.autoplay == 3 && !this.mpreview.video_interacted()) { this.mpreview.set_video_paused(false); } }; var on_preview_video_error = function (event) { // Display fallback this.mpreview.set_background_style(2); // Show error this.mpreview.set_stat_status(true, event.reason, "error"); }; // var on_preview_video_volume_change = function (event) { // Save volume changes // if (event.reason == "seek") { // settings.change_value(["image_expansion", "video", "volume"], event.volume); // settings.save_values(); // } // }; var on_preview_size_change = function (event) { // Update size this.mpreview.set_size(event.width, event.height, true); this.mpreview.set_stat_resolution(undefined, event.width + "x" + event.height); // Reposition this.preview_update(false, false, false); }; var on_preview_mouse_wheel = function (event) { if (event.mode === 0) { // Zoom change_zoom_level.call(this, event.delta); if (this.settings.zoom_borders_show) change_zoom_borders_visibility.call(this, true); } else { // if (event.mode === 1) { // Change volume if (this.mpreview.get_type() === MediaPreview.TYPE_VIDEO) { // New volume var v = this.mpreview.get_video_volume() + (event.delta * 5.0 / 100.0); // Bound if (v < 0.0) v = 0.0; else if (v > 1.0) v = 1.0; // Update this.mpreview.set_video_volume(v); if (this.mpreview.get_video_muted()) { this.mpreview.set_video_muted(false); } this.mpreview.set_show_volume_controls_temp(true, 0.5); } } change_mouse_visibility.call(this, true); }; var on_preview_mouse_enter = function (event) { // Cancel close this.preview_close_cancel(); change_mouse_visibility.call(this, true); // Mini controls var mc = this.settings.video.mini_controls; if (mc < 3) this.mpreview.set_vcontrols_mini_visible(mc == 1); }; var on_preview_mouse_leave = function (event) { // Delay close this.preview_close(false); // Mini controls var mc = this.settings.video.mini_controls; if (mc < 3) this.mpreview.set_vcontrols_mini_visible(mc == 2); }; var on_preview_mouse_down = function (event) { // Close this.preview_close(true); }; var on_preview_mouse_move = function (event) { // Change zoom offset this.mouse_x = event.x; this.mouse_y = event.y; update_zoom_offset.call(this); if (this.settings.zoom_borders_show) change_zoom_borders_visibility.call(this, true); change_mouse_visibility.call(this, true); }; var on_preview_mouse_enter_main = function (event) { // Hide paddings if (this.paddings_active) { this.paddings_active = false; this.mpreview.clear_paddings(); } }; var on_preview_stats_zoom_control_click = function (event) { // Zoom in/out change_zoom_level.call(this, event.delta); }; var on_preview_open_timeout = function (image_container, post_info, auto_fit) { // Nullify timer this.preview_open_timer = null; // Close this.preview_open(image_container, post_info, auto_fit); }; var on_preview_close_timeout = function () { // Nullify timer this.preview_close_timer = null; // Close this.preview_close(true); }; var on_preview_zoom_borders_hide_timeout = function () { this.zoom_borders_hide_timer = null; change_zoom_borders_visibility.call(this, false); }; var on_preview_mouse_hide_timeout = function () { this.mouse_hide_timer = null; change_mouse_visibility.call(this, false); }; var on_image_mouseenter = function (event, image_container) { // Don't run if disabled if (this.disabled) return; // Attempt to open this.current_hover_container = image_container; preview_open_test.call(this, image_container, null, true); }; var on_image_mouseleave = function (event, image_container) { // Don't run if disabled if (this.disabled) return; // Close preview this.current_hover_container = null; this.preview_close(false); }; var on_image_click = function (event, image_container) { // Don't run if disabled if (this.disabled) return; // Close preview setTimeout(on_image_click_delay.bind(this, image_container), 10); }; var on_image_click_delay = function (image_container) { var post_container = api.post_get_post_container_from_image_container(image_container); if (post_container === null) return; if (api.post_is_image_expanded_or_expanding(post_container)) { // Transfer state var expanded_node = null; if (this.settings.video.expand_state_save) { expanded_node = api.post_get_image_expanded_from_image_container(image_container); if (expanded_node !== null) { this.mpreview.transfer_video_state(expanded_node); } } // Close this.preview_close(true); // Observe expanded node if (expanded_node !== null && this.settings.video.expand_state_save) { observe_expanded.call(this, expanded_node, image_container); } } else { // Attempt to open if still hovered if (this.current_hover_container === image_container) { preview_open_test.call(this, image_container, post_container, false); } } }; var on_image_connector_mouseenter = function (event, image_container) { // Cancel close this.preview_close_cancel(); }; var on_image_connector_mouseleave = function (event, image_container) { // Close preview this.preview_close(false); }; var on_image_connector_mousedown = function (event) { // Close this.preview_close(true); }; var on_window_resize = function (event) { // Reposition this.preview_update(false, false, false); }; var on_window_scroll = function (event) { // Reposition this.preview_update(false, false, false); }; var on_api_post_add = function (container) { // Queue add hooks this.post_queue.push(container); }; var on_api_post_remove = function (container) { // Remove hooks remove_post_container_callbacks.call(this, container); }; var on_api_image_hover_open = function () { // Disable if (this.disabled) return; this.disable(); // Remove event api.off("image_hover_open", this.on_api_image_hover_open_bind); // Update settings settings.disable_image_expansion(); }; var on_settings_value_change = function (data) { // Update this.update_settings_from_global(); }; var on_hotkey = function (event, hotkey) { // Don't run if disabled if (this.disabled) return; if (hotkey.key == "esc") { // Close this.preview_close(true); return stop_event(event); } }; var observe_expanded = function (node, image_container) { // Ignore if invalid node if (node.parentNode === null || node.tagName !== "VIDEO" || style.has_class(node, "iex_observing")) return; style.add_class(node, "iex_observing"); var State = function (time, n) { this.time = time; this.volume = n.volume; this.muted = n.muted; this.paused = n.paused; }; var self = this, time_min_interval = 0.25 * 1000, state_pre = new State(timing() - time_min_interval, node), state = state_pre, current_time = 0.0, disconnect, on_volumechange, on_timeupdate, on_pause, on_play; // Listen for video state changing events on_volumechange = function (event) { var t = timing(); if (t - state.time >= time_min_interval) { state_pre = state; state = new State(t, this); } else { state.volume = this.volume; state.muted = this.muted; } }; on_pause = function (event) { var t = timing(); if (t - state.time >= time_min_interval) { state_pre = state; state = new State(t, this); } else { state.paused = this.paused; } }; on_play = function (event) { var t = timing(); if (t - state.time >= time_min_interval) { state_pre = state; state = new State(t, this); } else { state.paused = this.paused; } }; on_timeupdate = function (event) { current_time = this.currentTime; }; node.addEventListener("play", on_volumechange, false); node.addEventListener("pause", on_volumechange, false); node.addEventListener("timeupdate", on_timeupdate, false); node.addEventListener("volumechange", on_volumechange, false); // Observe for removal disconnect = api.observe_children(node.parentNode, function (added, n) { if (!added && n === node) { // Node removed disconnect(); n.removeEventListener("play", on_volumechange, false); n.removeEventListener("pause", on_volumechange, false); n.removeEventListener("timeupdate", on_timeupdate, false); n.removeEventListener("volumechange", on_volumechange, false); style.remove_class(n, "iex_observing"); // Update if (self.mpreview !== null && self.mpreview.is_visible()) { var t = timing(), s = state; if (t - s.time < time_min_interval) s = state_pre; // Update state self.mpreview.load_video_state({ time: current_time, volume: s.volume, muted: s.muted, paused: s.paused, }); } } }); }; var trigger_event = function (event, data) { var list = this.events[event], i; for (i = 0; i < list.length; ++i) { list[i].call(null, data); } }; ImageHover.prototype = { constructor: ImageHover, destroy: function () { var par; // Detach preview preview_detach.call(this, true); // Create connector par = this.connector.parentNode; if (par) { this.connector.removeEventListener("mouseover", this.on_image_connector_mouseenter_bind, false); this.connector.removeEventListener("mouseout", this.on_image_connector_mouseleave_bind, false); this.connector.removeEventListener("mousedown", this.on_image_connector_mousedown_bind, false); par.removeChild(this.connector); } // Remove hotkeys if (this.hotkeys !== null) { for (var i = 0; i < this.hotkeys.length; ++i) { hotkey_manager.unregister(this.hotkeys[i]); } this.hotkeys = null; } // Remove api events api.off("post_add", this.on_api_post_add_bind); api.off("post_remove", this.on_api_post_remove_bind); api.off("image_hover_open", this.on_api_image_hover_open_bind); // Remove settings events settings.off_change(["image_expansion", "hover", "zoom_invert"], this.off_settings_value_change_bind); settings.off_change(["image_expansion", "hover", "zoom_borders_show"], this.off_settings_value_change_bind); settings.off_change(["image_expansion", "hover", "zoom_borders_hide_time"], this.off_settings_value_change_bind); settings.off_change(["image_expansion", "hover", "zoom_buttons"], this.off_settings_value_change_bind); settings.off_change(["image_expansion", "hover", "mouse_hide"], this.off_settings_value_change_bind); settings.off_change(["image_expansion", "hover", "mouse_hide_time"], this.off_settings_value_change_bind); settings.off_change(["image_expansion", "hover", "header_overlap"], this.off_settings_value_change_bind); settings.off_change(["image_expansion", "hover", "fit_large_allowed"], this.off_settings_value_change_bind); settings.off_change(["image_expansion", "hover", "display_stats"], this.off_settings_value_change_bind); settings.off_change(["image_expansion", "video", "autoplay"], this.off_settings_value_change_bind); settings.off_change(["image_expansion", "video", "loop"], this.off_settings_value_change_bind); settings.off_change(["image_expansion", "video", "mute_initially"], this.off_settings_value_change_bind); settings.off_change(["image_expansion", "video", "volume"], this.off_settings_value_change_bind); settings.off_change(["image_expansion", "video", "mini_controls"], this.off_settings_value_change_bind); settings.off_change(["image_expansion", "style", "controls_rounded_border"], this.off_settings_value_change_bind); settings.off_change(["image_expansion", "style", "animations_background"], this.off_settings_value_change_bind); }, start: function () { // Enable if settings allow it if (settings.values.image_expansion.enabled) { this.update_settings_from_global(); this.enable(); } }, enable: function () { // Enable if (this.enabled) return; this.enabled = true; // Setup nodes on_ready(on_asap.bind(this), on_asap_condition); // Get posts this.post_queue = Delay.queue(api.get("posts"), on_post_queue_callback.bind(this), 50, 0.25); // Bind post acquiring api.on("post_add", this.on_api_post_add_bind); api.on("post_remove", this.on_api_post_remove_bind); // Bind default image hover test api.on("image_hover_open", this.on_api_image_hover_open_bind); // Hotkeys var on_hotkey_bind = on_hotkey.bind(this); this.hotkeys = [ hotkey_manager.register("esc", 0, on_hotkey_bind), ]; }, disable: function () { // Not enabled or already disabled if (!this.enabled || this.disabled) return; // Disable this.disabled = true; // Stop the current preview this.preview_close(true); }, preview_open: function (image_container, post_info, auto_fit) { var display_stats = this.settings.display_stats, v_set = this.settings.video, mc = this.settings.video.mini_controls, ext, ext_info, ext_bg_show; // Clear timer if (this.preview_open_timer !== null) { clearTimeout(this.preview_open_timer); this.preview_open_timer = null; } if (this.preview_close_timer !== null) { clearTimeout(this.preview_close_timer); this.preview_close_timer = null; } // Set this.current_image_container = image_container; // Create new if (this.mpreview === null) { preview_create.call(this); this.update_mpreview(); } // Expansion type ext = get_image_url_extension.call(this, post_info.url); ext_info = this.extensions_valid[ext] || this.extensions_valid[""]; ext_bg_show = settings.values.image_expansion.extensions[ext_info.ext].background; if (ext_info.type == "video") { // Set video this.mpreview.set_vcontrols_mini_available(true); this.mpreview.set_video(post_info.url); // Playback settings this.mpreview.set_video_muted(v_set.mute_initially); this.mpreview.set_video_volume(v_set.volume); this.mpreview.set_video_paused(true, v_set.loop); this.mpreview.clear_video_interactions(); } else { // Set image this.mpreview.set_vcontrols_mini_available(false); this.mpreview.set_image(post_info.url); } // Background this.mpreview.set_background(post_info.thumb, ext_bg_show); this.mpreview.set_mouse_wheel_mode(settings.values.image_expansion.extensions[ext_info.ext].mouse_wheel); // Size if (post_info.resolution.width > 0 && post_info.resolution.height > 0) { this.mpreview.set_size(post_info.resolution.width, post_info.resolution.height, true); this.mpreview.set_stat_resolution((display_stats === 0), post_info.resolution.width + "x" + post_info.resolution.height); } else { this.mpreview.set_size(post_info.resolution_thumb.width, post_info.resolution_thumb.height, false); this.mpreview.set_stat_resolution((display_stats === 0), "unknown"); } // Stats if (post_info.name) { this.mpreview.set_stat_file_name((display_stats === 0), post_info.name); } this.mpreview.set_stat_file_size((display_stats === 0), post_info.size + post_info.size_label); // Mini controls this.mpreview.set_vcontrols_mini_visible(mc == 3 || mc == 2); // Set visible this.mpreview.set_visible(true); style.add_class(this.connector, "iex_floating_image_connector_visible"); // Update position this.mpreview.set_fixed(true); this.preview_update(false, true, auto_fit); // Trigger trigger_event.call(this, "change", { image_hover: this, image_container: image_container, type: ext_info.type, }); }, preview_close: function (immediate) { // Cancel the open timer if (this.preview_open_timer !== null) { clearTimeout(this.preview_open_timer); this.preview_open_timer = null; } if (this.mpreview === null || !this.mpreview.is_visible()) return; if (immediate) { // Immediate this.mpreview.set_visible(false); this.current_image_container = null; // Connector style.remove_class(this.connector, "iex_floating_image_connector_visible"); // Trigger trigger_event.call(this, "change", { image_hover: this, image_container: null, type: null, }); } else { // Delay if (this.preview_close_timer !== null) { clearTimeout(this.preview_close_timer); } this.preview_close_timer = setTimeout(this.on_preview_close_timeout_bind, 20); } }, preview_close_cancel: function () { // Cancel the timer if (this.preview_close_timer !== null) { clearTimeout(this.preview_close_timer); this.preview_close_timer = null; } }, preview_update: function (keep_mouse_inside, fit_reset, fit_start) { // Update the position and sizing of the preview after a zoom, resize, or open event if (this.mpreview === null || !this.mpreview.is_visible()) return; // Vars var set = this.settings, header_overlap = set.header_overlap, paddings = set.region_paddings, max_rect = style.get_document_rect(), img = this.current_image_container.querySelector("img") || this.current_image_container, img_rect = style.get_object_rect_inner(img), left = max_rect.left, top = max_rect.top, fit = this.mpreview.fit, fit_large = this.mpreview.fit_large, fit_large_allowed = set.fit_large_allowed, fit_axis = 0, w = this.mpreview.size.width, h = this.mpreview.size.height, zoom = this.mpreview.zoom, x, y, w_max, h_max, w_scale, h_scale; // Include header if (!header_overlap) { var header_rect = api.get_header_rect(); // Subtract if (header_rect.top <= max_rect.top && header_rect.bottom > max_rect.top) { // It's on the top max_rect.top = header_rect.bottom; } else if (header_rect.bottom >= max_rect.bottom && header_rect.top < max_rect.bottom) { // It's on the bottom max_rect.bottom = header_rect.top; } } // Update max_rect max_rect.left = img_rect.right; max_rect.left += paddings.left; max_rect.top += paddings.top; max_rect.right -= paddings.right; max_rect.bottom -= paddings.bottom; // Maximum size w_max = (max_rect.right - max_rect.left); h_max = (max_rect.bottom - max_rect.top); if (w_max < 1) w_max = 1; if (h_max < 1) h_max = 1; this.display_window_max.width = w_max; this.display_window_max.height = h_max; // Get the minimum scale and apply it w_scale = w_max / w; h_scale = h_max / h; if (fit_reset) { zoom = 1; fit = fit_start; fit_large = false; if (w_scale < 1.0 || h_scale < 1.0) { fit = true; fit_large = (Math.abs(w_scale - h_scale) < 1e-5) && fit_large_allowed; } } if (fit) { var best_scale; if ((w_scale > h_scale) == fit_large) { best_scale = w_scale; fit_axis = 0; } else { best_scale = h_scale; fit_axis = 1; } w *= best_scale; h *= best_scale; } w *= zoom; h *= zoom; // Limit this.mpreview.set_view_borders_visible_sides((w - w_max) > 1e-5, (h - h_max) > 1e-5); if (w > w_max) w = w_max; if (h > h_max) h = h_max; // Position x = max_rect.left; y = (img_rect.top + img_rect.bottom - h) / 2.0; if (y + h >= max_rect.bottom) y = max_rect.bottom - h; if (y < max_rect.top) y = max_rect.top; // Mouse inside if (keep_mouse_inside) { // Setup var off_right = 0, off_top = 0, off_bottom = 0, padding_extra = set.padding_extra, x_r = x + w, y_b = y + h; off_right = (this.mouse_x - x_r); off_top = (y - this.mouse_y); off_bottom = (this.mouse_y - y_b); // Right offset if (off_right <= 0) { off_right = 0; } else { off_right += padding_extra.right; if (x_r + off_right > max_rect.right) { off_right = Math.max(0, max_rect.right - x_r); } } // Top offset if (off_top <= 0) { off_top = 0; } else { off_top += padding_extra.top; if (y - off_top < max_rect.top) { off_top = Math.max(0, y - max_rect.top); } } // Bottom offset if (off_bottom <= 0) { off_bottom = 0; } else { off_bottom += padding_extra.bottom; if (y_b + off_bottom > max_rect.bottom) { off_bottom = Math.max(0, max_rect.bottom - y_b); } } // Set if (off_top > 0 || off_right > 0 || off_bottom > 0) { this.paddings_active = true; this.mpreview.set_paddings(off_top, off_right, off_bottom, 0); } else { this.paddings_active = false; this.mpreview.clear_paddings(); } } else { // No paddings this.paddings_active = false; this.mpreview.clear_paddings(); } // Set this.mpreview.set_window(x - left, y - top, w, h); this.mpreview.set_zoom(zoom, fit, fit_large, fit_axis); // Connector this.connector.style.left = (img_rect.right).toFixed(2) + "px"; this.connector.style.top = (img_rect.top).toFixed(2) + "px"; this.connector.style.width = (paddings.left).toFixed(2) + "px"; this.connector.style.height = (img_rect.bottom - img_rect.top).toFixed(2) + "px"; // Stats this.mpreview.set_stat_zoom((set.display_stats <= 1), (zoom * 100) + "%", fit ? " fit-" + (this.mpreview.fit_axis === 0 ? "x" : "y") : ""); }, update_settings_from_global: function () { // Copy settings var hs = settings.values.image_expansion.hover, vs = settings.values.image_expansion.video, ss = settings.values.image_expansion.style, set = this.settings, v_set = set.video, s_set = set.style; set.zoom_invert = hs.zoom_invert; set.zoom_borders_show = hs.zoom_borders_show; set.zoom_borders_hide_time = hs.zoom_borders_hide_time * 1000; set.zoom_buttons = hs.zoom_buttons; set.mouse_hide = hs.mouse_hide; set.mouse_hide_time = hs.mouse_hide_time * 1000; set.header_overlap = hs.header_overlap; set.fit_large_allowed = hs.fit_large_allowed; set.display_stats = hs.display_stats; v_set.autoplay = vs.autoplay; v_set.loop = vs.loop; v_set.mute_initially = vs.mute_initially; v_set.volume = vs.volume; v_set.mini_controls = vs.mini_controls; v_set.expand_state_save = vs.expand_state_save; s_set.controls_rounded_border = ss.controls_rounded_border; s_set.animations_background = ss.animations_background; if (this.mpreview !== null) { this.update_mpreview(); } }, update_mpreview: function () { // Rounded borders this.mpreview.set_vcontrols_borders_rounded(!is_chrome && this.settings.style.controls_rounded_border); this.mpreview.set_background_animations(this.settings.style.animations_background); // Zoom buttons this.mpreview.set_zoom_control_buttons_enabled(this.settings.zoom_buttons); }, on: function (event, callback) { if (event in this.events) { this.events[event].push(callback); return true; } return false; }, off: function (event, callback) { if (event in this.events) { var list = this.events[event], i; for (i = 0; i < list.length; ++i) { if (callback === list[i]) { list.splice(i, 1); return true; } } } return false; }, }; return ImageHover; })(); // Media preview base var MediaPreview = (function () { var MediaPreviewGenericNodes = function (parent) { // Create the preview container var zoom_border_inner, stat_zoom_inc, stat_zoom_dec, stat_sep; // Vars this.stats_zoom_controls_enabled = true; this.zoom_controls_hide_timer = null; this.mouse_in_stats = false; this.on_hide_zoom_controls_timeout_bind = hide_zoom_controls.bind(parent, true, false); // Background this.background = $.div("iex_mpreview_background"); // iex_mpreview_background_disabled iex_mpreview_background_visible iex_mpreview_background_fallback // Zoom borders this.zoom_borders = $.div("iex_mpreview_zoom_borders"); // iex_mpreview_zoom_borders_visible iex_mpreview_zoom_borders_vertical iex_mpreview_zoom_borders_horizontal zoom_border_inner = $.div("iex_mpreview_zoom_borders_inner"); this.zoom_borders.appendChild(zoom_border_inner); // Stats container this.stats_container = $.div("iex_mpreview_stats_container"); // Zoom controls this.zoom_controls = $.span("iex_mpreview_stat iex_mpreview_stat_zoom_controls"); // iex_mpreview_stat_zoom_controls_visible iex_mpreview_stat_zoom_controls_fixed this.stats_container.appendChild(this.zoom_controls); stat_zoom_inc = $.span("iex_mpreview_stat_zoom_control iex_mpreview_stat_zoom_control_increase"); stat_zoom_inc.textContent = "+"; this.zoom_controls.appendChild(stat_zoom_inc); stat_zoom_dec = $.span("iex_mpreview_stat_zoom_control iex_mpreview_stat_zoom_control_decrease"); stat_zoom_dec.textContent = "\u2212"; this.zoom_controls.appendChild(stat_zoom_dec); this.zoom_controls_offset = $.span("iex_mpreview_stat_zoom_controls_offset"); this.stats_container.appendChild(this.zoom_controls_offset); // Stats items this.stat_zoom_container = $.span("iex_mpreview_stat"); this.stats_container.appendChild(this.stat_zoom_container); this.stat_zoom = $.span(""); this.stat_zoom_container.appendChild(this.stat_zoom); this.stat_zoom_fit = $.span(""); this.stat_zoom_container.appendChild(this.stat_zoom_fit); stat_sep = $.span("iex_mpreview_stat_sep"); stat_sep.textContent = ", "; this.stats_container.appendChild(stat_sep); this.stat_status = $.span("iex_mpreview_stat"); this.stats_container.appendChild(this.stat_status); stat_sep = $.span("iex_mpreview_stat_sep"); stat_sep.textContent = ", "; this.stats_container.appendChild(stat_sep); this.stat_resolution = $.span("iex_mpreview_stat"); this.stats_container.appendChild(this.stat_resolution); stat_sep = $.span("iex_mpreview_stat_sep"); stat_sep.textContent = ", "; this.stats_container.appendChild(stat_sep); this.stat_filesize = $.span("iex_mpreview_stat"); this.stats_container.appendChild(this.stat_filesize); stat_sep = $.span("iex_mpreview_stat_sep"); stat_sep.textContent = ", "; this.stats_container.appendChild(stat_sep); this.stat_filename = $.span("iex_mpreview_stat"); this.stats_container.appendChild(this.stat_filename); // Events parent.add_event_listener(this.stats_container, "mouseover", wrap_mouseenterleave_event(parent, on_stats_mouseenter), false); parent.add_event_listener(this.stats_container, "mouseout", wrap_mouseenterleave_event(parent, on_stats_mouseleave), false); parent.add_event_listener(this.stat_zoom, "mouseover", wrap_mouseenterleave_event(parent, on_stats_zoom_mouseenter), false); parent.add_event_listener(this.zoom_controls, "mousedown", on_stats_zoom_controls_mousedown.bind(parent), false); parent.add_event_listener(this.zoom_controls, "mouseover", wrap_mouseenterleave_event(parent, on_stats_zoom_controls_mouseenter), false); parent.add_event_listener(this.zoom_controls, "mouseout", wrap_mouseenterleave_event(parent, on_stats_zoom_controls_mouseleave), false); parent.add_event_listener(stat_zoom_inc, "click", on_stats_zoom_control_click.bind(parent, 1), false); parent.add_event_listener(stat_zoom_dec, "click", on_stats_zoom_control_click.bind(parent, -1), false); // Add parent.cpreview.add_content(this.background); parent.cpreview.add_overlay(this.stats_container, true); parent.cpreview.add_overlay(this.zoom_borders); }; var MediaPreviewImageNodes = function (parent) { // Image this.image = $.node("img", "iex_mpreview_image"); this.image.setAttribute("alt", ""); this.image.setAttribute("title", ""); // Events parent.add_event_listener(this.image, "load", on_image_load.bind(parent), false); parent.add_event_listener(this.image, "error", on_image_error.bind(parent), false); // Add parent.cpreview.add_content(this.image); }; var MediaPreviewVideoNodes = function (parent) { var svgns = "http://www.w3.org/2000/svg", c_inner, svg_e1, svg_e2, c_div1, c_div2, c_div3, c_div4, c_buttons_left, c_buttons_right, c_play_button, c_volume_button; // Vars this.interacted = false; this.seeking = false; this.seeking_paused = false; this.volume_modifying = false; this.mouse_capturing_rect = null; this.mouse_capturing_element = null; this.on_capture_mousemove = on_vcontrols_capture_mousemove.bind(parent); this.on_capture_mouseup = on_vcontrols_capture_mouseup.bind(parent); this.on_capture_window_resize = on_window_resize.bind(parent); this.on_capture_window_scroll = on_window_scroll.bind(parent); // Video this.video = $.node("video", "iex_mpreview_video"); this.video.setAttribute("preload", "auto"); // Controller overlay this.overlay = $.div("iex_mpreview_vcontrols_container"); c_inner = $.div("iex_mpreview_vcontrols_container_inner"); this.overlay.appendChild(c_inner); this.overlay_table = $.div("iex_mpreview_vcontrols_table"); // iex_mpreview_vcontrols_table_visible iex_mpreview_vcontrols_table_visible_important iex_mpreview_vcontrols_table_mini c_inner.appendChild(this.overlay_table); // Pause/play button c_buttons_left = $.div("iex_mpreview_vcontrols_button_container"); this.overlay_table.appendChild(c_buttons_left); c_div1 = $.div("iex_mpreview_vcontrols_button_container_inner"); c_buttons_left.appendChild(c_div1); c_div2 = $.div("iex_mpreview_vcontrols_button_container_inner2"); c_div1.appendChild(c_div2); // Mouse event controller c_play_button = $.div("iex_mpreview_vcontrols_button_mouse_controller"); c_div2.appendChild(c_play_button); // SVG image this.svg_play = $.node_ns(svgns, "svg", "iex_svg_play_button"); // iex_svg_play_button_playing iex_svg_play_button_looping this.svg_play.setAttribute("svgns", svgns); this.svg_play.setAttribute("width", "2em"); this.svg_play.setAttribute("height", "2em"); this.svg_play.setAttribute("viewBox", "0 0 1 1"); c_div2.appendChild(this.svg_play); svg_e1 = $.node_ns(svgns, "g", "iex_svg_button_scale_group"); svg_e1.setAttribute("transform", "translate(0.25,0.25) scale(0.5)"); this.svg_play.appendChild(svg_e1); svg_e2 = $.node_ns(svgns, "polygon", "iex_svg_play_button_play_icon iex_svg_button_fill"); svg_e2.setAttribute("points", "0,0 0,1 1,0.5"); svg_e1.appendChild(svg_e2); svg_e2 = $.node_ns(svgns, "polygon", "iex_svg_play_button_loop_icon iex_svg_button_fill"); svg_e2.setAttribute("points", "0.1,0.05 1,0.5 0.1,0.95"); svg_e1.appendChild(svg_e2); svg_e2 = $.node_ns(svgns, "polygon", "iex_svg_play_button_loop_icon iex_svg_button_fill"); svg_e2.setAttribute("points", "0,0.7 0.4,0.5 0,0.3"); svg_e1.appendChild(svg_e2); svg_e2 = $.node_ns(svgns, "polygon", "iex_svg_play_button_pause_icon iex_svg_button_fill"); svg_e2.setAttribute("points", "0.125,0 0.375,0 0.375,1 0.125,1"); svg_e1.appendChild(svg_e2); svg_e2 = $.node_ns(svgns, "polygon", "iex_svg_play_button_pause_icon iex_svg_button_fill"); svg_e2.setAttribute("points", "0.625,0 0.875,0 0.875,1 0.625,1"); svg_e1.appendChild(svg_e2); // Seek bar c_div1 = $.div("iex_mpreview_vcontrols_seek_container"); this.overlay_table.appendChild(c_div1); c_div2 = $.div("iex_mpreview_vcontrols_seek_container_inner"); c_div1.appendChild(c_div2); this.seek_bar = $.div("iex_mpreview_vcontrols_seek_bar"); // iex_mpreview_vcontrols_no_border_radius c_div2.appendChild(this.seek_bar); c_div3 = $.div("iex_mpreview_vcontrols_seek_bar_bg"); this.seek_bar.appendChild(c_div3); this.load_progress = $.div("iex_mpreview_vcontrols_seek_bar_loaded"); c_div3.appendChild(this.load_progress); this.play_progress = $.div("iex_mpreview_vcontrols_seek_bar_played"); c_div3.appendChild(this.play_progress); c_div4 = $.div("iex_mpreview_vcontrols_seek_time_table"); c_div3.appendChild(c_div4); this.time_current = $.div("iex_mpreview_vcontrols_seek_time_current"); c_div4.appendChild(this.time_current); this.time_duration = $.div("iex_mpreview_vcontrols_seek_time_duration"); c_div4.appendChild(this.time_duration); // Volume bar/mute button c_buttons_right = $.div("iex_mpreview_vcontrols_button_container"); this.overlay_table.appendChild(c_buttons_right); c_div1 = $.div("iex_mpreview_vcontrols_button_container_inner"); c_buttons_right.appendChild(c_div1); c_div2 = $.div("iex_mpreview_vcontrols_button_container_inner2"); c_div1.appendChild(c_div2); // Mouse event controller c_volume_button = $.div("iex_mpreview_vcontrols_button_mouse_controller"); c_div2.appendChild(c_volume_button); // SVG image this.svg_volume = $.node_ns(svgns, "svg", "iex_svg_volume_button"); // iex_svg_volume_button_muted iex_svg_volume_button_high iex_svg_volume_button_medium iex_svg_volume_button_low this.svg_volume.setAttribute("svgns", svgns); this.svg_volume.setAttribute("width", "2em"); this.svg_volume.setAttribute("height", "2em"); this.svg_volume.setAttribute("viewBox", "0 0 1 1"); c_div2.appendChild(this.svg_volume); svg_e1 = $.node_ns(svgns, "g", "iex_svg_button_scale_group"); svg_e1.setAttribute("transform", "translate(0.25,0.25) scale(0.5)"); this.svg_volume.appendChild(svg_e1); svg_e2 = $.node_ns(svgns, "polygon", "iex_svg_volume_button_speaker iex_svg_button_fill"); svg_e2.setAttribute("points", "0,0.3 0.2,0.3 0.5,0 0.6,0 0.6,1 0.5,1 0.2,0.7 0,0.7"); svg_e1.appendChild(svg_e2); svg_e2 = $.node_ns(svgns, "path", "iex_svg_volume_button_wave_big iex_svg_button_fill"); svg_e2.setAttribute("d", "M 0.75,0.1 Q 1.3,0.5 0.75,0.9 L 0.75,0.75 Q 1.05,0.5 0.75,0.25 Z"); svg_e1.appendChild(svg_e2); svg_e2 = $.node_ns(svgns, "path", "iex_svg_volume_button_wave_small iex_svg_button_fill"); svg_e2.setAttribute("d", "M 0.75,0.75 Q 1.05,0.5 0.75,0.25 Z"); svg_e1.appendChild(svg_e2); svg_e2 = $.node_ns(svgns, "g", "iex_svg_volume_button_wave_mute_icon"); svg_e1.appendChild(svg_e2); svg_e1 = svg_e2; svg_e2 = $.node_ns(svgns, "polygon", "iex_svg_volume_button_wave_mute_icon_polygon"); svg_e2.setAttribute("points", "0.7,0.3 1.0,0.6 0.9,0.7 0.6,0.4"); svg_e1.appendChild(svg_e2); svg_e2 = $.node_ns(svgns, "polygon", "iex_svg_volume_button_wave_mute_icon_polygon"); svg_e2.setAttribute("points", "0.7,0.7 1.0,0.4 0.9,0.3 0.6,0.6"); svg_e1.appendChild(svg_e2); // Volume bar c_div2 = $.div("iex_mpreview_vcontrols_volume_container_position"); c_div1.appendChild(c_div2); this.volume_container = $.div("iex_mpreview_vcontrols_volume_container"); // iex_mpreview_vcontrols_volume_container_visible iex_mpreview_vcontrols_volume_container_visible_important c_div2.appendChild(this.volume_container); this.volume_bar = $.div("iex_mpreview_vcontrols_volume_bar"); // iex_mpreview_vcontrols_no_border_radius this.volume_container.appendChild(this.volume_bar); c_div3 = $.div("iex_mpreview_vcontrols_volume_bar_bg"); this.volume_bar.appendChild(c_div3); this.volume_progress = $.div("iex_mpreview_vcontrols_volume_bar_level"); c_div3.appendChild(this.volume_progress); // Events parent.add_event_listener(this.video, "loadedmetadata", on_video_loadedmetadata.bind(parent), false); parent.add_event_listener(this.video, "canplay", on_video_canplay.bind(parent), false); parent.add_event_listener(this.video, "canplaythrough", on_video_canplaythrough.bind(parent), false); parent.add_event_listener(this.video, "error", on_video_error.bind(parent), false); parent.add_event_listener(this.video, "progress", on_video_progress.bind(parent), false); parent.add_event_listener(this.video, "timeupdate", on_video_timeupdate.bind(parent), false); parent.add_event_listener(this.video, "ended", on_video_ended.bind(parent), false); var prevent = on_vcontrols_prevent_default_mousedown.bind(parent); parent.add_event_listener(c_inner, "mouseover", wrap_mouseenterleave_event(parent, on_vcontrols_mouseenter), false); parent.add_event_listener(c_inner, "mouseout", wrap_mouseenterleave_event(parent, on_vcontrols_mouseleave), false); parent.add_event_listener(this.overlay_table, "mousedown", on_vcontrols_container_mousedown.bind(parent), false); parent.add_event_listener(c_play_button, "click", on_vcontrols_play_button_click.bind(parent), false); parent.add_event_listener(c_volume_button, "click", on_vcontrols_volume_button_click.bind(parent), false); parent.add_event_listener(c_volume_button, "mouseover", wrap_mouseenterleave_event(parent, on_vcontrols_volume_button_mouseenter), false); parent.add_event_listener(c_buttons_right, "mouseout", wrap_mouseenterleave_event(parent, on_vcontrols_volume_button_container_mouseleave), false); parent.add_event_listener(this.seek_bar, "mousedown", on_vcontrols_seek_bar_mousedown.bind(parent), false); parent.add_event_listener(this.volume_bar, "mousedown", on_vcontrols_volume_bar_mousedown.bind(parent), false); parent.add_event_listener(this.time_current, "mousedown", prevent, false); parent.add_event_listener(this.time_duration, "mousedown", prevent, false); // Add parent.cpreview.add_content(this.video); parent.cpreview.add_overlay(this.overlay); }; var MediaPreview = function () { // Setup preview this.cpreview = new ContentPreview(); this.event_listeners = []; // Events var cp_nodes = this.cpreview.nodes, event_bind; this.add_event_listener(cp_nodes.container, "mouseover", wrap_mouseenterleave_event(this, on_mouseenter), false); this.add_event_listener(cp_nodes.container, "mouseout", wrap_mouseenterleave_event(this, on_mouseleave), false); this.add_event_listener(cp_nodes.container, "mousemove", on_mousemove.bind(this), false); this.add_event_listener(cp_nodes.container, "mousewheel", (event_bind = on_mousewheel.bind(this)), false); this.add_event_listener(cp_nodes.container, "DOMMouseScroll", event_bind, false); this.add_event_listener(cp_nodes.container, "mousedown", on_mousedown.bind(this), false); this.add_event_listener(cp_nodes.container, "contextmenu", on_contextmenu.bind(this), false); this.add_event_listener(cp_nodes.overflow, "mouseover", wrap_mouseenterleave_event(this, on_overflow_mouseenter), false); // Setup classes style.add_class(this.cpreview.nodes.padding, "iex_mpreview_padding"); // Setup nodes this.nodes = new MediaPreviewGenericNodes(this); this.nodes_image = null; this.nodes_video = null; // Settings this.type = MediaPreview.TYPE_NONE; this.zoom = 1; this.fit = false; this.fit_large = false; this.fit_large_allowed = true; this.fit_axis = 0; this.mouse_wheel_mode = 0; this.volume_controls_temp_timer = null; this.style = { vcontrols_rounded: true, }; this.size = { width: 0, height: 0, acquired: false }; // Events this.events = { "image_load": [], "image_error": [], "video_ready": [], "video_can_play": [], "video_can_play_through": [], "video_load": [], "video_error": [], "video_volume_change": [], "size_change": [], "mouse_wheel": [], "mouse_enter": [], "mouse_leave": [], "mouse_down": [], "mouse_move": [], "mouse_enter_main": [], "stats_zoom_control_click": [], "resize": [], }; }; MediaPreview.TYPE_NONE = 0; MediaPreview.TYPE_IMAGE = 1; MediaPreview.TYPE_VIDEO = 2; var trigger = function (event, data) { // Trigger if (event in this.events) { var e_list = this.events[event]; for (var i = 0; i < e_list.length; ++i) { e_list[i].call(this, data); } } }; var format_video_time = function (time) { time = Math.floor(time + 0.5); var s = (time % 60).toString(); if (s.length < 2) s = "0" + s; time = Math.floor(time / 60); return time + ":" + s; }; var update_size = function (width_new, height_new) { // Change this.size.width = width_new; this.size.height = height_new; this.size.acquired = true; // Event trigger.call(this, "size_change", { width: width_new, height: height_new, }); }; var update_video_duration_status = function () { var video = this.nodes_video.video, duration = video.duration; // Update seek bar if (isNaN(duration)) { // Update time numbers this.nodes_video.time_duration.textContent = ""; } else { // Update time numbers this.nodes_video.time_duration.textContent = format_video_time.call(this, duration); } }; var update_video_seek_status = function (time) { var video = this.nodes_video.video, duration = video.duration; // Update seek bar if (time < 0 || isNaN(duration)) { // Bar this.nodes_video.play_progress.style.width = "0"; // Update time numbers this.nodes_video.time_current.textContent = ""; } else { // Bar this.nodes_video.play_progress.style.width = ((time / duration) * 100).toFixed(2) + "%"; // Update time numbers this.nodes_video.time_current.textContent = format_video_time.call(this, time); } }; var update_video_play_status = function (paused) { var loop = this.nodes_video.video.loop, svg_play = this.nodes_video.svg_play; // Update button if (paused) { style.remove_class(svg_play, "iex_svg_play_button_playing"); } else { style.add_class(svg_play, "iex_svg_play_button_playing"); } if (loop) { style.add_class(svg_play, "iex_svg_play_button_looping"); } else { style.remove_class(svg_play, "iex_svg_play_button_looping"); } }; var set_video_volume = function (volume, reason, interacted) { var svg_volume = this.nodes_video.svg_volume; // Set volume on video if (volume >= 0) { if (volume > 1) volume = 1; this.nodes_video.video.volume = volume; } else { volume = this.nodes_video.video.volume; } // Update bar this.nodes_video.volume_progress.style.height = (volume * 100).toFixed(2) + "%"; // Update icons if (volume > 0.625) { style.change_classes_svg(svg_volume, "iex_svg_volume_button_high", "iex_svg_volume_button_low iex_svg_volume_button_medium"); } else if (volume > 0.125) { style.change_classes_svg(svg_volume, "iex_svg_volume_button_medium", "iex_svg_volume_button_low iex_svg_volume_button_high"); } else { style.change_classes_svg(svg_volume, "iex_svg_volume_button_low", "iex_svg_volume_button_medium iex_svg_volume_button_high"); } // Interacted if (interacted) this.nodes_video.interacted = true; // Event trigger.call(this, "video_volume_change", { reason: reason, volume: volume }); }; var setup_vcontrols_mouse_capture = function () { // Setup mouse capturing events if ( this.nodes_video.mouse_capturing_element === null && (this.nodes_video.mouse_capturing_element = document.documentElement) !== null ) { this.nodes_video.mouse_capturing_element.addEventListener("mousemove", this.nodes_video.on_capture_mousemove, false); this.nodes_video.mouse_capturing_element.addEventListener("mouseup", this.nodes_video.on_capture_mouseup, false); window.addEventListener("resize", this.nodes_video.on_capture_window_resize, false); window.addEventListener("scroll", this.nodes_video.on_capture_window_scroll, false); update_vcontrols_mouse_capture.call(this); } }; var teardown_vcontrols_mouse_capture = function () { // Destroy mouse capturing events if (this.nodes_video.mouse_capturing_element !== null) { this.nodes_video.mouse_capturing_element.removeEventListener("mousemove", this.nodes_video.on_capture_mousemove, false); this.nodes_video.mouse_capturing_element.removeEventListener("mouseup", this.nodes_video.on_capture_mouseup, false); window.removeEventListener("resize", this.nodes_video.on_capture_window_resize, false); window.removeEventListener("scroll", this.nodes_video.on_capture_window_scroll, false); this.nodes_video.mouse_capturing_element = null; } }; var update_vcontrols_mouse_capture = function () { // Update node rectangle if (this.nodes_video.seeking) { this.nodes_video.mouse_capturing_rect = style.get_object_rect(this.nodes_video.seek_bar); } else if (this.nodes_video.volume_modifying) { this.nodes_video.mouse_capturing_rect = style.get_object_rect(this.nodes_video.volume_bar); } }; var show_zoom_controls = function (force_open) { // Stop timer if (this.nodes.zoom_controls_hide_timer !== null) { clearTimeout(this.nodes.zoom_controls_hide_timer); this.nodes.zoom_controls_hide_timer = null; } // Display if (force_open) { var zoom_controls = this.nodes.zoom_controls, z_rect; style.add_class(zoom_controls, "iex_mpreview_stat_visible"); z_rect = style.get_object_rect(zoom_controls); this.nodes.zoom_controls_offset.style.width = (z_rect.right - z_rect.left).toFixed(2) + "px"; } }; var hide_zoom_controls = function (instant, timer_check) { if (instant) { // Hide style.remove_class(this.nodes.zoom_controls, "iex_mpreview_stat_visible"); this.nodes.zoom_controls_offset.style.width = ""; // Stop timer if (timer_check && this.nodes.zoom_controls_hide_timer !== null) { clearTimeout(this.nodes.zoom_controls_hide_timer); } this.nodes.zoom_controls_hide_timer = null; } else { // Set timer this.nodes.zoom_controls_hide_timer = setTimeout(this.nodes.on_hide_zoom_controls_timeout_bind, 10); } }; var set_zoom_controls_fixed = function (fixed) { // Set fixed var zoom_controls = this.nodes.zoom_controls; if (fixed) { var w_rect = style.get_document_rect(), z_rect = style.get_object_rect(zoom_controls); // Fix style.add_class(zoom_controls, "iex_mpreview_stat_zoom_controls_fixed"); zoom_controls.style.left = (z_rect.left - w_rect.left).toFixed(2) + "px"; zoom_controls.style.top = (z_rect.top - w_rect.top).toFixed(2) + "px"; } else { // Un-fix style.remove_class(zoom_controls, "iex_mpreview_stat_zoom_controls_fixed"); zoom_controls.style.left = ""; zoom_controls.style.top = ""; } }; var update_stat_sep_visibility = function (node, visible) { var n; if (visible) { // Previous for (n = node; (n = n.previousSibling); ) { if (style.has_class(n, "iex_mpreview_stat_sep_visible")) break; if (style.has_class(n, "iex_mpreview_stat_visible")) { if ((n = node.previousSibling) !== null && style.has_class(n, "iex_mpreview_stat_sep")) { style.add_class(n, "iex_mpreview_stat_sep_visible"); } break; } } // Next for (n = node; (n = n.nextSibling); ) { if (style.has_class(n, "iex_mpreview_stat_sep_visible")) break; if (style.has_class(n, "iex_mpreview_stat_visible")) { if ((n = node.nextSibling) !== null && style.has_class(n, "iex_mpreview_stat_sep")) { style.add_class(n, "iex_mpreview_stat_sep_visible"); } break; } } } else { // Hide next n = node.nextSibling; if (n !== null && style.has_class(n, "iex_mpreview_stat_sep")) { if (style.has_class(n, "iex_mpreview_stat_sep_visible")) { style.remove_class(n, "iex_mpreview_stat_sep_visible"); return; } } // Hide previous n = node.previousSibling; if (n !== null && style.has_class(n, "iex_mpreview_stat_sep")) { style.remove_class(n, "iex_mpreview_stat_sep_visible"); } } }; var on_window_resize = function (event) { // Update bar position if (this.nodes_video.mouse_capturing_element !== null) { update_vcontrols_mouse_capture.call(this); } }; var on_window_scroll = function (event) { // Update bar position if (this.nodes_video.mouse_capturing_element !== null) { update_vcontrols_mouse_capture.call(this); } }; var on_video_loadedmetadata = function (event) { if (this.type !== MediaPreview.TYPE_VIDEO) return; var video = this.nodes_video.video, w = video.videoWidth, h = video.videoHeight; // Update seek bar update_video_duration_status.call(this); update_video_seek_status.call(this, 0); // Un-hide video style.remove_class(video, "iex_mpreview_video_not_ready"); // Update size if (!this.size.acquired || (this.size.width != w || this.size.height != h)) { update_size.call(this, w, h); style.remove_class(video, "iex_mpreview_video_unsized"); } // Event trigger.call(this, "video_ready", {}); }; var on_video_canplay = function (event) { if (this.type !== MediaPreview.TYPE_VIDEO) return; // Event on_video_progress.call(this, event); trigger.call(this, "video_can_play", {}); }; var on_video_canplaythrough = function (event) { if (this.type !== MediaPreview.TYPE_VIDEO) return; // Event on_video_progress.call(this, event); trigger.call(this, "video_can_play_through", {}); }; var on_video_progress = function (event) { if (this.type !== MediaPreview.TYPE_VIDEO) return; var video = this.nodes_video.video, percent = 0.0; if (video.buffered.length > 0) { percent = (video.buffered.end(0) / video.duration); } // Update progress bar this.nodes_video.load_progress.style.width = (percent * 100).toFixed(2) + "%"; // Complete if (percent >= 1.0) { trigger.call(this, "video_load", {}); } }; var on_video_error = function (event) { if (this.type !== MediaPreview.TYPE_VIDEO) return; // Hide video style.remove_class(this.nodes_video.video, "iex_mpreview_video_visible"); this.nodes_video.video.removeAttribute("src"); // Error trigger.call(this, "video_error", { reason: "error" }); }; var on_video_timeupdate = function (event) { if (this.type !== MediaPreview.TYPE_VIDEO) return; // Update time progress update_video_seek_status.call(this, this.nodes_video.video.currentTime); }; var on_video_ended = function (event) { if (this.type !== MediaPreview.TYPE_VIDEO) return; // Update buttons update_video_play_status.call(this, this.nodes_video.video.paused); }; var on_vcontrols_mouseenter = function (event, node) { // Do not show for non-video if (this.type !== MediaPreview.TYPE_VIDEO) return; // Show controls style.remove_class(this.nodes_video.volume_container, "iex_mpreview_vcontrols_volume_container_visible"); style.add_class(this.nodes_video.overlay_table, "iex_mpreview_vcontrols_table_visible"); }; var on_vcontrols_mouseleave = function (event, node) { // Hide controls style.remove_class(this.nodes_video.overlay_table, "iex_mpreview_vcontrols_table_visible"); }; var on_vcontrols_container_mousedown = function (event) { // Ignore if not left mb if (get_event_mouse_button(event) !== 1) return; // Stop event return stop_event(event); }; var on_vcontrols_play_button_click = function (event) { // Play/pause var video = this.nodes_video.video; this.set_video_paused(!video.paused, (event.shiftKey && video.paused) ? (!video.loop) : undefined); }; var on_vcontrols_volume_button_click = function (event) { // Mute/unmute this.set_video_muted(!this.nodes_video.video.muted); }; var on_vcontrols_volume_button_mouseenter = function (event, node) { // Show volume bar style.add_class(this.nodes_video.volume_container, "iex_mpreview_vcontrols_volume_container_visible"); }; var on_vcontrols_volume_button_container_mouseleave = function (event, node) { // Hide volume bar style.remove_class(this.nodes_video.volume_container, "iex_mpreview_vcontrols_volume_container_visible"); }; var on_vcontrols_seek_bar_mousedown = function (event) { // Get the mouse button if (get_event_mouse_button(event) !== 1) return; // Begin seeking var video = this.nodes_video.video; // Set paused state this.nodes_video.seeking = true; this.nodes_video.seeking_paused = video.paused; if (!video.paused) video.pause(); // Force visibility style.add_class(this.nodes_video.overlay_table, "iex_mpreview_vcontrols_table_visible_important"); // Setup window mousemove and mouseup events setup_vcontrols_mouse_capture.call(this); // Initial update on_vcontrols_capture_mousemove.call(this, event); // Stop return stop_event(event); }; var on_vcontrols_volume_bar_mousedown = function (event) { // Get the mouse button if (get_event_mouse_button(event) !== 1) return; // Set paused state this.nodes_video.volume_modifying = true; // Force visibility style.add_class(this.nodes_video.overlay_table, "iex_mpreview_vcontrols_table_visible_important"); style.add_class(this.nodes_video.volume_container, "iex_mpreview_vcontrols_volume_container_visible_important"); // Setup window mousemove and mouseup events setup_vcontrols_mouse_capture.call(this); // Initial update on_vcontrols_capture_mousemove.call(this, event); // Stop return stop_event(event); }; var on_vcontrols_prevent_default_mousedown = function (event) { event.preventDefault(); return false; }; var on_vcontrols_capture_mousemove = function (event) { // Begin seeking var video = this.nodes_video.video, bar_rect = this.nodes_video.mouse_capturing_rect, percent, x, y; // Get mouse x/y if (event.pageX === undefined && (event = event || window.event).clientX !== undefined) { var html = document.documentElement, body = document.body; if (html && body) { x = ((event.clientX || 0) + (html.scrollLeft || (body && body.scrollLeft) || 0)) - (html.clientLeft || 0); y = ((event.clientY || 0) + (html.scrollTop || (body && body.scrollTop) || 0)) - (html.clientTop || 0); } } else { x = event.pageX || 0; y = event.pageY || 0; } // Set paused state if (this.nodes_video.seeking) { var duration = video.duration; if (!isNaN(duration)) { var loaded = (video.buffered.length > 0) ? video.buffered.end(0) / duration : 0.0; // Get percent percent = (x - bar_rect.left) / (bar_rect.right - bar_rect.left); // Bound if (percent < 0) percent = 0; else if (percent > loaded) percent = loaded; // Apply percent *= duration; video.currentTime = percent; update_video_seek_status.call(this, percent); } } else if (this.nodes_video.volume_modifying) { // Get percent percent = 1.0 - (y - bar_rect.top) / (bar_rect.bottom - bar_rect.top); // Bound if (percent < 0) percent = 0; else if (percent > 1.0) percent = 1.0; // Set volume set_video_volume.call(this, percent, "seeking", true); } }; var on_vcontrols_capture_mouseup = function (event) { if (this.nodes_video.seeking) { // Unpause if (!this.nodes_video.seeking_paused) this.nodes_video.video.play(); // Un-force visibility style.remove_class(this.nodes_video.overlay_table, "iex_mpreview_vcontrols_table_visible_important"); // Done this.nodes_video.seeking = false; } else if (this.nodes_video.volume_modifying) { // Save volume set_video_volume.call(this, -1, "seek", true); // Un-force visibility style.remove_class(this.nodes_video.overlay_table, "iex_mpreview_vcontrols_table_visible_important"); style.remove_class(this.nodes_video.volume_container, "iex_mpreview_vcontrols_volume_container_visible_important"); // Done this.nodes_video.volume_modifying = false; } // Stop capture teardown_vcontrols_mouse_capture.call(this); }; var on_image_load = function (event) { if (this.type != MediaPreview.TYPE_IMAGE) return; // True size var w = this.nodes_image.image.naturalWidth, h = this.nodes_image.image.naturalHeight; // Update size if (!this.size.acquired || (this.size.width != w || this.size.height != h)) { update_size.call(this, w, h); style.remove_class(this.nodes_image.image, "iex_mpreview_image_unsized"); } // Event trigger.call(this, "image_load", {}); }; var on_image_error = function (event) { if (this.type != MediaPreview.TYPE_IMAGE) return; // Hide image style.remove_class(this.nodes_image.image, "iex_mpreview_image_visible"); this.nodes_image.image.removeAttribute("src"); // Error trigger.call(this, "image_error", { reason: "error" }); }; var on_mousewheel = function (event) { // Get direction var delta = (event.wheelDelta || -event.detail || 0); if (delta < 0) delta = -1; else if (delta > 0) delta = 1; // Trigger event trigger.call(this, "mouse_wheel", { delta: delta, mode: this.mouse_wheel_mode }); // Prevent return stop_event(event); }; var on_mouseenter = function (event, node) { // Trigger event trigger.call(this, "mouse_enter", {}); }; var on_mouseleave = function (event, node) { // Trigger event trigger.call(this, "mouse_leave", {}); }; var on_mousedown = function (event) { // Get the mouse button var button = get_event_mouse_button(event); // Activate close trigger.call(this, "mouse_down", { button: button }); return stop_event(event); }; var on_contextmenu = function (event) { // Stop return stop_event(event); }; var on_mousemove = function (event) { // Get mouse x/y var x, y; if (event.pageX === undefined && (event = event || window.event).clientX !== undefined) { var html = document.documentElement, body = document.body; if (html && body) { x = ((event.clientX || 0) + (html.scrollLeft || (body && body.scrollLeft) || 0)) - (html.clientLeft || 0); y = ((event.clientY || 0) + (html.scrollTop || (body && body.scrollTop) || 0)) - (html.clientTop || 0); } } else { x = event.pageX || 0; y = event.pageY || 0; } // Event trigger.call(this, "mouse_move", { x: x, y: y, }); }; var on_overflow_mouseenter = function (event, node) { // Trigger event trigger.call(this, "mouse_enter_main", {}); }; var on_stats_mouseenter = function (event, node) { // Stop hiding if it's already open this.nodes.mouse_in_stats = true; show_zoom_controls.call(this, false); }; var on_stats_mouseleave = function (event, node) { // Hide controls this.nodes.mouse_in_stats = false; hide_zoom_controls.call(this, false, true); }; var on_stats_zoom_mouseenter = function (event, node) { // Show controls if (this.nodes.stats_zoom_controls_enabled) { show_zoom_controls.call(this, true); } }; var on_stats_zoom_controls_mouseenter = function (event, node) { // Display controls and make fixed show_zoom_controls.call(this, true); set_zoom_controls_fixed.call(this, true); }; var on_stats_zoom_controls_mouseleave = function (event, node) { // Make un-fixed and possibly hide set_zoom_controls_fixed.call(this, false); if (!this.nodes.mouse_in_stats) { hide_zoom_controls.call(this, true, true); } }; var on_stats_zoom_controls_mousedown = function (event) { // Stop return stop_event(event); }; var on_stats_zoom_control_click = function (delta, event) { // Event trigger.call(this, "stats_zoom_control_click", { delta: delta }); // Stop return stop_event(event); }; MediaPreview.prototype = { constructor: MediaPreview, destroy: function () { // Clear this.clear(); // Remove events for (var i = 0, el; i < this.event_listeners.length; ++i) { el = this.event_listeners[i]; this.remove_event_listener(el[0], el[1], el[2], el[3]); } this.event_listeners = []; // Remove this.remove(); }, clear: function () { var type = this.type; this.type = MediaPreview.TYPE_NONE; // Clear background this.nodes.background.style.backgroundImage = ""; style.remove_classes(this.nodes.background, "iex_mpreview_background_visible iex_mpreview_background_visible_full iex_mpreview_background_disabled iex_mpreview_background_fallback"); if (type == MediaPreview.TYPE_IMAGE) { // Remove previous image style.remove_class(this.nodes_image.image, "iex_mpreview_image_visible"); this.nodes_image.image.removeAttribute("src"); } else if (type == MediaPreview.TYPE_VIDEO) { // Clear video style.remove_classes(this.nodes_video.video, "iex_mpreview_video_visible iex_mpreview_video_not_ready"); this.set_video_paused(true); this.nodes_video.video.removeAttribute("src"); // Clear seek status this.nodes_video.load_progress.style.width = "0"; this.nodes_video.play_progress.style.width = "0"; // Update time numbers this.nodes_video.time_current.textContent = ""; this.nodes_video.time_duration.textContent = ""; update_video_play_status.call(this, true); } // Clear stats this.clear_stats(); }, on: function (event, callback) { // Add callback if (event in this.events) { this.events[event].push(callback); } }, off: function (event, callback) { // Add callback if (event in this.events) { var e_list = this.events[event]; for (var i = 0; i < e_list.length; ++i) { if (e_list[i] === callback) { // Remove e_list.splice(i, 1); return true; } } } return false; }, get_type: function () { return this.type; }, get_video_volume: function () { return (this.type !== MediaPreview.TYPE_VIDEO) ? 0.0 : this.nodes_video.video.volume; }, get_video_muted: function () { return (this.type !== MediaPreview.TYPE_VIDEO) ? true : this.nodes_video.video.muted; }, set_image: function (image) { // Unset previous this.clear(); // Set type this.type = MediaPreview.TYPE_IMAGE; if (this.nodes_image === null) this.nodes_image = new MediaPreviewImageNodes(this); // Set style.add_class(this.nodes_image.image, "iex_mpreview_image_visible"); this.nodes_image.image.setAttribute("src", image.replace(/#.*$/, "")); }, set_video: function (video) { // Unset previous this.clear(); // Set type this.type = MediaPreview.TYPE_VIDEO; if (this.nodes_video === null) { this.nodes_video = new MediaPreviewVideoNodes(this); this.set_vcontrols_borders_rounded(this.style.vcontrols_rounded); } // Set style.add_class(this.nodes_video.video, "iex_mpreview_video_visible"); style.add_class(this.nodes_video.video, "iex_mpreview_video_not_ready"); this.nodes_video.video.setAttribute("src", video.replace(/#.*$/, "")); }, set_background: function (image, visibility_level) { // visibility_level: 0=hidden, 1=transparent, 2=full if (image) { // Set background style.add_class(this.nodes.background, "iex_mpreview_background_visible"); this.set_background_style(visibility_level); this.nodes.background.style.backgroundImage = 'url("' + image + '")'; } else { // Clear background style.remove_classes(this.nodes.background, "iex_mpreview_background_visible iex_mpreview_background_disabled iex_mpreview_background_visible_full"); this.nodes.background.style.backgroundImage = ""; } }, set_background_style: function (visibility_level) { // visibility_level: 0=hidden, 1=transparent, 2=full if (visibility_level === 0) { style.add_class(this.nodes.background, "iex_mpreview_background_disabled"); style.remove_class(this.nodes.background, "iex_mpreview_background_visible_full"); } else if (visibility_level == 1) { style.remove_classes(this.nodes.background, "iex_mpreview_background_disabled iex_mpreview_background_visible_full"); } else { style.add_class(this.nodes.background, "iex_mpreview_background_visible_full"); style.remove_class(this.nodes.background, "iex_mpreview_background_disabled"); } }, set_background_animations: function (animate) { if (animate) style.add_class(this.nodes.background, "iex_transitions"); else style.remove_class(this.nodes.background, "iex_transitions"); }, set_mouse_wheel_mode: function (mode) { this.mouse_wheel_mode = mode; }, set_view_borders: function (horizontal, vertical) { var zoom_borders = this.nodes.zoom_borders, w = (horizontal * 100).toFixed(2) + "%", h = (vertical * 100).toFixed(2) + "%"; // Set view borders sizing zoom_borders.style.left = w; zoom_borders.style.top = h; zoom_borders.style.right = w; zoom_borders.style.bottom = h; }, set_view_borders_visible: function (visible) { var zoom_borders = this.nodes.zoom_borders; // Change visibility if (visible === true) { style.add_class(zoom_borders, "iex_mpreview_zoom_borders_visible"); } else { style.remove_class(zoom_borders, "iex_mpreview_zoom_borders_visible"); } }, set_view_borders_visible_sides: function (horizontal, vertical) { var zoom_borders = this.nodes.zoom_borders; // Change visibility if (horizontal) style.add_class(zoom_borders, "iex_mpreview_zoom_borders_horizontal"); else style.remove_class(zoom_borders, "iex_mpreview_zoom_borders_horizontal"); if (vertical) style.add_class(zoom_borders, "iex_mpreview_zoom_borders_vertical"); else style.remove_class(zoom_borders, "iex_mpreview_zoom_borders_vertical"); }, set_mouse_visible: function (visible) { if (visible) style.remove_class(this.cpreview.nodes.padding, "iex_mpreview_mouse_hidden"); else style.add_class(this.cpreview.nodes.padding, "iex_mpreview_mouse_hidden"); }, add_to: function (parent) { this.cpreview.add_to(parent); }, remove: function () { this.cpreview.remove(); }, set_visible: function (visible) { this.cpreview.set_visible(visible); if (!visible) this.clear(); }, set_fixed: function (fixed) { this.cpreview.set_fixed(fixed); }, set_window: function (x, y, width, height, fixed) { this.cpreview.set_window(x, y, width, height, fixed); trigger.call(this, "resize", {}); }, set_zoom: function (zoom, fit, large, axis) { // Update fit if (fit !== undefined) { this.fit = fit; this.fit_large = large; } // Detect scale if (axis !== undefined) this.fit_axis = axis; var scale = 1.0; if (this.fit) { scale = (this.fit_axis === 0) ? (this.cpreview.display.width / this.size.width) : (this.cpreview.display.height / this.size.height); } // Set zoom if (zoom !== undefined) { this.zoom = zoom; } scale *= this.zoom; this.cpreview.set_size(this.size.width * scale, this.size.height * scale); trigger.call(this, "resize", {}); }, set_size: function (width, height, acquired) { this.size.width = width; this.size.height = height; if (acquired === true || acquired === false) this.size.acquired = acquired; }, set_offset: function (x, y) { this.cpreview.set_offset(x, y); }, set_paddings: function (top, right, bottom, left) { this.cpreview.set_paddings(top, right, bottom, left); }, clear_size: function () { this.cpreview.clear_size(); }, clear_paddings: function () { this.cpreview.clear_paddings(); }, video_interacted: function () { return (this.nodes_video !== null && this.nodes_video.interacted); }, clear_video_interactions: function () { if (this.nodes_video !== null) this.nodes_video.interacted = false; }, set_video_muted: function (muted) { if (this.nodes_video === null) return; // Apply this.nodes_video.video.muted = muted; // Change icon if (muted) style.add_class_svg(this.nodes_video.svg_volume, "iex_svg_volume_button_muted"); else style.remove_class_svg(this.nodes_video.svg_volume, "iex_svg_volume_button_muted"); // Interacted this.nodes_video.interacted = true; }, set_video_volume: function (volume) { if (this.nodes_video === null) return; set_video_volume.call(this, volume, "set", true); }, set_video_paused: function (paused, loop) { if (this.nodes_video === null) return; var video = this.nodes_video.video; // Loop if (loop === true || loop === false) { video.loop = loop; } // Pause or play if (paused) video.pause(); else video.play(); // Interacted this.nodes_video.interacted = true; // Update buttons update_video_play_status.call(this, paused); }, set_show_volume_controls_temp: function (show, duration) { if (this.type !== MediaPreview.TYPE_VIDEO) return; // Toggle var cls1 = "iex_mpreview_vcontrols_table_visible_temp", cls2 = "iex_mpreview_vcontrols_volume_container_visible_temp"; if (show) { style.add_class(this.nodes_video.overlay_table, cls1); style.add_class(this.nodes_video.volume_container, cls2); // Timer if (duration !== undefined) { if (this.volume_controls_temp_timer !== null) { clearTimeout(this.volume_controls_temp_timer); } this.volume_controls_temp_timer = setTimeout(this.set_show_volume_controls_temp.bind(this, false, 0.0), duration * 1000); } } else { // Hide style.remove_class(this.nodes_video.overlay_table, cls1); style.remove_class(this.nodes_video.volume_container, cls2); // Clear timeout if (this.volume_controls_temp_timer !== null) { clearTimeout(this.volume_controls_temp_timer); this.volume_controls_temp_timer = null; } } }, is_visible: function () { return this.cpreview.visible; }, get_inner_rect: function () { return style.get_object_rect(this.cpreview.nodes.overflow); }, get_window: function () { return this.cpreview.display; }, add_event_listener: function (node, event, callback, capture) { node.addEventListener(event, callback, capture); this.event_listeners.push([node, event, callback, capture]); }, remove_event_listener: function (node, event, callback, capture, remove) { node.removeEventListener(event, callback, capture); if (remove) { for (var i = 0, j = this.event_listeners.length; i < j; ++i) { if ( this.event_listeners[i][0] === event && this.event_listeners[i][1] === callback && this.event_listeners[i][2] === capture ) { this.event_listeners.splice(i, 1); return; } } } }, set_vcontrols_mini_available: function (available) { if (this.nodes_video === null) return; if (available) style.remove_class(this.nodes_video.overlay_table, "iex_mpreview_vcontrols_table_mini_disabled"); else style.add_class(this.nodes_video.overlay_table, "iex_mpreview_vcontrols_table_mini_disabled"); }, set_vcontrols_mini_visible: function (visible) { if (this.nodes_video === null) return; if (visible) style.add_class(this.nodes_video.overlay_table, "iex_mpreview_vcontrols_table_mini"); else style.remove_class(this.nodes_video.overlay_table, "iex_mpreview_vcontrols_table_mini"); }, set_vcontrols_borders_rounded: function (rounded) { this.style.vcontrols_rounded = rounded; if (this.nodes_video === null) return; if (rounded) { style.remove_class(this.nodes_video.volume_bar, "iex_mpreview_vcontrols_no_border_radius"); style.remove_class(this.nodes_video.seek_bar, "iex_mpreview_vcontrols_no_border_radius"); } else { style.add_class(this.nodes_video.volume_bar, "iex_mpreview_vcontrols_no_border_radius"); style.add_class(this.nodes_video.seek_bar, "iex_mpreview_vcontrols_no_border_radius"); } }, clear_stats: function () { this.set_stat_zoom(false); this.set_stat_status(false); this.set_stat_resolution(false); this.set_stat_file_size(false); this.set_stat_file_name(false); }, set_stat_zoom: function (visible, text, text_fit) { if (visible === false) { style.remove_class(this.nodes.stat_zoom_container, "iex_mpreview_stat_visible"); update_stat_sep_visibility.call(this, this.nodes.stat_zoom_container, false); this.nodes.stat_zoom.textContent = ""; this.nodes.stat_zoom_fit.textContent = ""; } else { if (visible) { style.add_class(this.nodes.stat_zoom_container, "iex_mpreview_stat_visible"); update_stat_sep_visibility.call(this, this.nodes.stat_zoom_container, true); } this.nodes.stat_zoom.textContent = text; this.nodes.stat_zoom_fit.textContent = text_fit; } }, set_stat_status: function (visible, text, type) { if (visible === false) { style.remove_classes(this.nodes.stat_status, "iex_mpreview_stat_visible iex_mpreview_stat_red"); update_stat_sep_visibility.call(this, this.nodes.stat_status, false); this.nodes.stat_status.textContent = ""; } else { if (visible) { style.add_class(this.nodes.stat_status, "iex_mpreview_stat_visible"); update_stat_sep_visibility.call(this, this.nodes.stat_status, true); } if (type == "error") { style.add_class(this.nodes.stat_status, "iex_mpreview_stat_red"); } else { style.remove_class(this.nodes.stat_status, "iex_mpreview_stat_red"); } this.nodes.stat_status.textContent = text; } }, set_stat_resolution: function (visible, text) { if (visible === false) { style.remove_class(this.nodes.stat_resolution, "iex_mpreview_stat_visible"); update_stat_sep_visibility.call(this, this.nodes.stat_resolution, false); this.nodes.stat_resolution.textContent = ""; } else { if (visible) { style.add_class(this.nodes.stat_resolution, "iex_mpreview_stat_visible"); update_stat_sep_visibility.call(this, this.nodes.stat_resolution, true); } this.nodes.stat_resolution.textContent = text; } }, set_stat_file_size: function (visible, text) { if (visible === false) { style.remove_class(this.nodes.stat_filesize, "iex_mpreview_stat_visible"); update_stat_sep_visibility.call(this, this.nodes.stat_filesize, false); this.nodes.stat_filesize.textContent = ""; } else { if (visible) { style.add_class(this.nodes.stat_filesize, "iex_mpreview_stat_visible"); update_stat_sep_visibility.call(this, this.nodes.stat_filesize, true); } this.nodes.stat_filesize.textContent = text; } }, set_stat_file_name: function (visible, text) { if (visible === false) { style.remove_class(this.nodes.stat_filename, "iex_mpreview_stat_visible"); update_stat_sep_visibility.call(this, this.nodes.stat_filename, false); this.nodes.stat_filename.textContent = ""; } else { if (visible) { style.add_class(this.nodes.stat_filename, "iex_mpreview_stat_visible"); update_stat_sep_visibility.call(this, this.nodes.stat_filename, true); } this.nodes.stat_filename.textContent = text; } }, set_zoom_control_buttons_enabled: function (enabled) { if (this.nodes === null) return; this.nodes.stats_zoom_controls_enabled = enabled; }, transfer_video_state: function (node) { if (node.tagName === "VIDEO" && this.type === MediaPreview.TYPE_VIDEO) { // Transfer video state var v = this.nodes_video.video, muted = v.muted, volume = v.volume, time = v.currentTime, paused = v.paused, okay = false, fn; fn = function () { this.muted = muted; this.volume = volume; this.currentTime = time; if (paused) this.pause(); else this.play(); }; if (!node.paused || !isNaN(node.duration)) { // Immediate try { fn.call(node); okay = true; } catch (e) {} } if (!okay) { // Delay until playing var play_fn = function (event) { try { fn.call(this); } catch (e) {} this.removeEventListener("play", play_fn, false); }; node.addEventListener("play", play_fn, false); } } }, load_video_state: function (state) { if (this.type === MediaPreview.TYPE_VIDEO) { // Update var self = this, v = this.nodes_video.video, target = v.getAttribute("src") || "", event_fn, fn; fn = function () { // Set state this.set_video_muted(state.muted); this.set_video_volume(state.volume); this.set_video_paused(state.paused); this.nodes_video.video.currentTime = state.time; }; if (isNaN(v.currentTime) || v.buffered.length <= 0) { // Delay self = this; event_fn = function () { if (target === this.getAttribute("src") || "") { fn.call(self); } this.removeEventListener("loadedmetadata", event_fn, false); }; // Listen v.addEventListener("loadedmetadata", event_fn, false); } else { fn.call(this); } } }, get_size: function () { return this.size; }, }; return MediaPreview; })(); // Content preview base var ContentPreview = (function () { var ContentPreviewNodes = function () { // Main container this.container = $.div("iex_cpreview_container iex_cpreview_container_visible"); // Position/padding container this.padding = $.div("iex_cpreview_padding"); this.container.appendChild(this.padding); // Overflow container this.overflow = $.div("iex_cpreview_overflow"); this.padding.appendChild(this.overflow); // Offset container this.offset = $.div("iex_cpreview_offset"); this.overflow.appendChild(this.offset); }; var ContentPreview = function () { this.nodes = new ContentPreviewNodes(); this.visible = true; this.offset_x = 0; this.offset_y = 0; this.size = { width: 0, height: 0, }; this.display = { x: 0, y: 0, width: 0, height: 0, }; this.fixed = false; this.auto = false; }; ContentPreview.prototype = { constructor: ContentPreview, set_visible: function (visible) { // Add or remove visibility this.visible = visible; if (visible) style.add_class(this.nodes.container, "iex_cpreview_container_visible"); else style.remove_class(this.nodes.container, "iex_cpreview_container_visible"); }, set_fixed: function (fixed) { // No change if (this.fixed == fixed) return; var container = this.nodes.container, doc_offset = style.get_document_offset(); // Fixed position this.fixed = fixed; if (this.fixed) { this.display.x -= doc_offset.left; this.display.y -= doc_offset.top; style.add_class(container, "iex_cpreview_container_fixed"); } else { this.display.x += doc_offset.left; this.display.y += doc_offset.top; style.remove_class(container, "iex_cpreview_container_fixed"); } // Position container.style.left = this.display.x.toFixed(2) + "px"; container.style.top = this.display.y.toFixed(2) + "px"; }, set_window: function (x, y, width, height, fixed) { var container = this.nodes.container, overflow = this.nodes.overflow; // Fixed position if (fixed !== undefined) { this.fixed = fixed; if (fixed) style.add_class(this.nodes.container, "iex_cpreview_container_fixed"); else style.remove_class(this.nodes.container, "iex_cpreview_container_fixed"); } // Position container.style.left = x.toFixed(2) + "px"; container.style.top = y.toFixed(2) + "px"; // Size overflow.style.width = width.toFixed(2) + "px"; overflow.style.height = height.toFixed(2) + "px"; // Set this.display.x = x; this.display.y = y; this.display.width = width; this.display.height = height; }, set_size: function (width, height, offset_x, offset_y) { var offset = this.nodes.offset, w_diff = width - this.display.width, h_diff = height - this.display.height; // Bound if (w_diff < 0) w_diff = 0; if (h_diff < 0) h_diff = 0; // Set size this.size.width = width; this.size.height = height; // Set offset if (offset_x !== undefined && offset_y !== undefined) { this.offset_x = offset_x; this.offset_y = offset_y; } // Set visible size offset.style.width = width.toFixed(2) + "px"; offset.style.height = height.toFixed(2) + "px"; style.add_class(offset, "iex_cpreview_offset_sized"); // Set visible offset offset.style.left = (w_diff * -this.offset_x).toFixed(2) + "px"; offset.style.top = (h_diff * -this.offset_y).toFixed(2) + "px"; }, set_offset: function (x, y) { var offset = this.nodes.offset, w_diff = this.size.width - this.display.width, h_diff = this.size.height - this.display.height; // Bound if (w_diff < 0) w_diff = 0; if (h_diff < 0) h_diff = 0; // Set offset this.offset_x = x; this.offset_y = y; // Set visible offset offset.style.left = (w_diff * -x).toFixed(2) + "px"; offset.style.top = (h_diff * -y).toFixed(2) + "px"; }, set_paddings: function (top, right, bottom, left) { // Set paddings this.nodes.padding.style.top = (-top).toFixed(2) + "px"; this.nodes.padding.style.left = (-left).toFixed(2) + "px"; this.nodes.padding.style.padding = top.toFixed(2) + "px " + right.toFixed(2) + "px " + bottom.toFixed(2) + "px " + left.toFixed(2) + "px"; }, set_auto_size: function (auto) { // No change if (this.auto == auto) return; var container = this.nodes.container; // Fixed position this.auto = auto; if (this.auto) { style.add_class(container, "iex_cpreview_container_auto"); } else { style.remove_class(container, "iex_cpreview_container_auto"); } // Position this.display.x = 0; this.display.y = 0; }, update_auto_size: function () { // Get size var size = style.get_object_size(this.nodes.container); // Set this.size.width = size.width; this.size.height = size.height; this.display.width = size.width; this.display.height = size.height; }, clear_size: function () { this.nodes.offset.style.width = ""; this.nodes.offset.style.height = ""; this.size.width = 0; this.size.height = 0; }, clear_paddings: function () { // Clear paddings this.nodes.padding.style.top = ""; this.nodes.padding.style.left = ""; this.nodes.padding.style.padding = ""; }, add_to: function (parent) { parent.appendChild(this.nodes.container); }, remove: function () { var par = this.nodes.container.parentNode; if (par) par.removeChild(this.nodes.container); }, add_content: function (node) { // Add to offset style.add_class(node, "iex_cpreview_content"); this.nodes.offset.appendChild(node); }, add_overlay: function (node, outside) { // Add to overflow or padding style.add_class(node, "iex_cpreview_overlay"); (outside ? this.nodes.padding : this.nodes.overflow).appendChild(node); }, }; return ContentPreview; })(); // Class for bit streaming to a string var BitStream = (function () { var BitStream = function () { this.value_pre = 0; this.value = 0; this.bits = 0; this.string = ""; this.pos = 0; this.pos_bit = 0; this.value_pre_decode = 0; }; BitStream.bits_per_char = 6; BitStream.bits_per_char_mask = (1 << BitStream.bits_per_char) - 1; BitStream.alphabet = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+-"; BitStream.re_alphabet = /^[a-zA-Z0-9\+\-]*/; BitStream.alphabet_map = (function (alphabet) { var length = alphabet.length, map = {}, i; for (i = 0; i < length; ++i) map[alphabet[i]] = i; return map; })(BitStream.alphabet); BitStream.prototype = { constructor: BitStream, value_to_char: function (v) { return BitStream.alphabet[v]; }, pos_to_value: function (pos) { return (pos >= this.string.length) ? 0 : BitStream.alphabet_map[this.string[pos]]; }, encode_value: function (v) { return (v ^ ((this.value_pre + this.string.length) % BitStream.alphabet.length)); }, decode_value: function (v) { return (v ^ ((this.value_pre_decode + this.pos) % BitStream.alphabet.length)); }, add: function (value, bits) { var b; while (bits >= (b = BitStream.bits_per_char - this.bits)) { this.value |= (value << this.bits) & BitStream.bits_per_char_mask; this.value = this.encode_value(this.value); this.string += this.value_to_char(this.value); value >>>= b; bits -= b; this.bits = 0; this.value_pre = this.value; this.value = 0; } this.value |= (value & ((1 << bits) - 1)) << this.bits; this.bits += bits; if (this.bits<0)console.log("value=",value,bits); }, get: function (bits) { if (bits === undefined) bits = 1; var value = 0, shift = 0, c, cr, b; while (bits >= (b = BitStream.bits_per_char - this.pos_bit)) { cr = this.pos_to_value(this.pos); c = this.decode_value(cr); //c = (cr ^ ((this.value_pre_decode + this.pos) % BitStream.alphabet.length)); value |= ((c >>> this.pos_bit) & ((1 << b) - 1)) << shift; bits -= b; shift += b; this.pos_bit = 0; ++this.pos; this.value_pre_decode = cr; } if (bits > 0) { //c = (this.pos_to_value(this.pos) ^ ((this.value_pre_decode + this.pos) % BitStream.alphabet.length)); c = this.decode_value(this.pos_to_value(this.pos)); value |= ((c >>> this.pos_bit) & ((1 << bits) - 1)) << shift; this.pos_bit += bits; } return value; }, encode: function () { var s = this.string; if (this.bits > 0) s += this.value_to_char(this.encode_value(this.value)); return s; }, decode: function (string) { this.string = string; this.pos = 0; this.pos_bit = 0; this.value_pre_decode = 0; }, to_encoded_string: function () { // Converts the content decoded thus far into a new encoded string var s = this.string.substr(0, this.pos), c; if (this.pos_bit === 0) return s; c = this.decode_value(this.pos_to_value(this.pos)); c &= (1 << this.pos_bit) - 1; // mask off the end c = (c ^ ((this.value_pre_decode + this.pos) % BitStream.alphabet.length)); s += this.value_to_char(c); return s; }, remaining: function () { if (this.pos >= this.string.length) return 0; return (this.string.length - this.pos + 1) * BitStream.bits_per_char - this.pos_bit; }, eof: function () { return (this.pos >= this.string.length); }, }; return BitStream; })(); // Image annotation var Annotation = (function () { var Annotation = function (parent, settings) { this.parent = parent; this.selected = false; this.index = 0; this.font_index = 0; this.font_size_index = 8; this.font_size = 0; this.font_bold = false; this.font_italic = false; this.text_before = null; this.text = ""; this.text_display = ""; this.text_formatters = []; this.text_align = Annotation.AlignDefault; this.text_line_spacing = Annotation.line_spacing.to_index(0); this.text_char_spacing = Annotation.char_spacing.to_index(0); this.text_decoration = Annotation.TextDecorationNone; this.colors = [ 0, 1, 1 ]; // text, alt, background this.color_index_selected = 0; this.position = [ 32, 32 ]; this.size = [ 100, 100 ]; this.nodes = { edit: null, annotation: null, }; this.events = []; this.edit_data = (settings && settings.edit) ? new Annotation.EditData() : null; if (settings !== null) { this.index = settings.index || 0; if ("font_size_index" in settings) { this.font_size_index = settings.font_size_index; } if ("x" in settings) this.position[0] = settings.x; if ("y" in settings) this.position[1] = settings.y; if ("width" in settings) this.size[0] = settings.width; if ("height" in settings) this.size[1] = settings.height; if ("text_before" in settings) this.text_before = settings.text_before; if ("font" in settings) this.font_index = Math.min(Math.max(0, settings.font), Annotation.fonts.length); if ("bold" in settings) this.font_bold = settings.bold; if ("italic" in settings) this.font_italic = settings.italic; } this.font_size = Annotation.font_size_index_to_size(this.font_size_index); this.create_nodes((settings && settings.edit) || false); }; Annotation.EditData = function () { this.move_events = []; this.move_mode = 0; this.move_origin = [ 0, 0 ]; this.move_position = [ 0, 0 ]; this.move_size = [ 0, 0 ]; this.double_click_ignore = false; }; Annotation.prototype = { constructor: Annotation, set_annotation_rect: function (x, y, width, height) { this.position[0] = x; this.position[1] = y; this.size[0] = width; this.size[1] = height; this.update_annotation_rect(); }, update_annotation_rect: function () { var n1s = this.nodes.annotation.container.style, u = Annotation.scaling_mode_units; n1s.left = this.position[0] + u; n1s.top = this.position[1] + u; n1s.width = this.size[0] + u; n1s.height = this.size[1] + u; }, auto_size: function () { var a_rect = style.get_object_rect_relative(this.nodes.annotation.container), scale = this.parent.get_image_scale(), image_size = this.parent.get_image_size(), rect, n, x, y, w, h; // Size n = this.nodes.annotation.text; n.style.display = "inline-block"; n.style.width = "auto"; rect = style.get_object_rect_relative(n); n.style.display = ""; n.style.width = ""; // Update x = Math.round(this.position[0] + (rect.left - a_rect.left) / scale); y = Math.round(this.position[1] + (rect.top - a_rect.top) / scale); w = Math.ceil(rect.width / scale); h = Math.ceil(rect.height / scale); // Limits if (x + w > image_size.width) { x = image_size.width - w; } if (y + h > image_size.height) { y = image_size.height - h; } if (x < 0) { x = 0; if (w > image_size.width) { w = image_size.width; } } if (y < 0) { y = 0; if (h > image_size.height) { h = image_size.height; } } // Set this.set_annotation_rect(x, y, w, h); this.trigger("rect_change"); }, set_index: function (index) { this.index = index; this.nodes.edit.index_text.textContent = "#" + (index + 1); }, set_text_before: function (text) { this.text_before = text; }, set_text: function (text, dont_set_input) { // Parse var tf = new TextFormatter(); tf.parse(text); // Update values this.text = tf.raw_text; this.text_display = tf.text; this.text_formatters = tf.formatters; // Update input if (!dont_set_input && this.nodes.edit !== null) { this.nodes.edit.text.value = this.text; } // Apply this.apply_text(); }, set_font: function (index) { this.font_index = index; var f = Annotation.fonts[index], fn = f.css_name + "," + Annotation.fonts[0].css_name; if (f.external_name !== null) { // Load font style.add_font(f.css_name, f.external_name, f.external_settings); } this.nodes.annotation.content.style.fontFamily = fn; if (this.nodes.edit !== null) { this.nodes.edit.button_font.style.fontFamily = fn; } }, set_font_size: function (index) { this.font_size_index = index; this.font_size = Annotation.font_size_index_to_size(this.font_size_index); this.nodes.annotation.content.style.fontSize = this.font_size + Annotation.scaling_mode_units; if (this.nodes.edit !== null) { this.nodes.edit.button_font_size_text.textContent = this.font_size; } }, set_font_bold: function (bold) { this.font_bold = bold; if (bold) { style.add_class(this.nodes.annotation.text, "iex_annotation_content_bold"); if (this.nodes.edit !== null) { this.nodes.edit.button_font.style.fontWeight = "bold"; } } else { style.remove_class(this.nodes.annotation.text, "iex_annotation_content_bold"); if (this.nodes.edit !== null) { this.nodes.edit.button_font.style.fontWeight = ""; } } }, set_font_italic: function (italic) { this.font_italic = italic; if (italic) { style.add_class(this.nodes.annotation.text, "iex_annotation_content_italic"); if (this.nodes.edit !== null) { this.nodes.edit.button_font.style.fontStyle = "italic"; } } else { style.remove_class(this.nodes.annotation.text, "iex_annotation_content_italic"); if (this.nodes.edit !== null) { this.nodes.edit.button_font.style.fontStyle = ""; } } }, set_text_line_spacing: function (spacing) { this.text_line_spacing = spacing; this.nodes.annotation.content.style.lineHeight = (1 + Annotation.line_spacing.from_index(spacing)) + "em"; }, set_text_char_spacing: function (spacing) { this.text_char_spacing = spacing; this.nodes.annotation.content.style.letterSpacing = Annotation.char_spacing.from_index(spacing) + "em"; }, set_text_decoration: function (decoration) { this.text_decoration = decoration; this.nodes.annotation.text.style.textShadow = this.decoration_to_shadow_style(decoration, "#" + Annotation.colors[this.colors[1]]); if (this.text_formatters.length > 0) { this.apply_text(); // in case any formatting contains a decoration color modification } }, set_align: function (align) { var classes_old = this.align_to_classes(this.text_align), temp = this.align_to_classes_and_strings(align), classes_new = temp[0], align_strs = temp[1], cl = this.nodes.annotation.container.classList, i; this.text_align = align; for (i = 0; i < classes_old.length; ++i) { cl.remove(classes_old[i]); } for (i = 0; i < classes_new.length; ++i) { cl.add(classes_new[i]); } if (this.nodes.edit !== null) { this.nodes.edit.button_align.className = "iex_ae_annotation_button iex_ae_annotation_button_align iex_ae_annotation_button_align_" + align_strs[0] + " iex_ae_annotation_button_valign_" + align_strs[1] + style.theme; } }, set_color: function (index, color) { this.colors[index] = color; var c = "#" + Annotation.colors[color]; if (this.nodes.edit !== null && index === this.color_index_selected) { this.nodes.edit.button_color_bg.style.backgroundColor = c; } if (index === 0) { this.nodes.annotation.text.style.color = c; this.nodes.annotation.border1.style.borderColor = c; } else if (index === 1) { if (this.text_decoration !== Annotation.TextDecorationNone) { this.set_text_decoration(this.text_decoration); } else if (this.text_formatters.length > 0) { this.apply_text(); // in case any formatting contains a decoration color modification } } else { // if (index === 2) { this.nodes.annotation.background.style.backgroundColor = c; } }, set_color_index_selected: function (index) { this.color_index_selected = index; if (this.nodes.edit !== null) { this.nodes.edit.button_color_text.textContent = (index === 1 ? "alt" : (index === 2 ? "bg" : "txt" )); this.nodes.edit.button_color_bg.style.backgroundColor = "#" + Annotation.colors[this.colors[this.color_index_selected]]; } }, set_selected: function (selected, dont_select_text) { if (selected === this.selected) return; this.selected = selected; if (selected) { style.add_class(this.nodes.annotation.container, "iex_annotation_selected"); if (this.nodes.edit !== null) { style.add_class(this.nodes.edit.container, "iex_ae_annotation_editor_selected"); if (!dont_select_text) { this.select_text(null, false); } } } else { style.remove_class(this.nodes.annotation.container, "iex_annotation_selected"); if (this.nodes.edit !== null) { style.remove_class(this.nodes.edit.container, "iex_ae_annotation_editor_selected"); } } }, get_index: function () { return this.index; }, get_text: function () { return this.text; }, get_text_display: function () { return this.text_display; }, get_text_formatters: function () { return this.text_formatters; }, get_text_before: function () { return this.text_before; }, get_font: function () { return this.font_index; }, get_font_size: function () { return this.font_size_index; }, get_font_bold: function () { return this.font_bold; }, get_font_italic: function () { return this.font_italic; }, get_align: function () { return this.text_align; }, get_text_decoration: function () { return this.text_decoration; }, get_text_line_spacing: function () { return this.text_line_spacing; }, get_text_char_spacing: function () { return this.text_char_spacing; }, get_color: function (index) { return this.colors[index]; }, get_color_count: function () { return this.colors.length; }, get_color_index_selected: function () { return this.color_index_selected; }, get_position: function () { return this.position; }, get_size: function () { return this.size; }, is_selected: function () { return this.selected; }, get_formatter_text: function (text, formatters) { return TextFormatter.get_raw(text, formatters); }, escape_text: function (text) { return text.trim().replace(/\\/g, "\\\\"); }, load_from_state: function (state, text) { this.set_annotation_rect(state.rect.x, state.rect.y, state.rect.width, state.rect.height); this.set_font(state.font); this.set_font_bold(state.bold); this.set_font_italic(state.italic); this.set_font_size(state.font_size); this.set_align(state.align); this.set_color(0, state.colors[0]); this.set_color(1, state.colors[1]); this.set_color(2, state.colors[2]); this.set_text_decoration(state.text_decoration); this.set_text_line_spacing(state.line_spacing); this.set_text_char_spacing(state.char_spacing); // Update text if (state.text !== null) text = state.text; //this.set_text(text); this.text_display = text; this.text_formatters = state.formatters; if (this.nodes.edit !== null) { this.text = this.get_formatter_text(this.text_display, this.text_formatters); this.nodes.edit.text.value = this.text; } this.apply_text(); }, align_to_classes: function (align) { var classes = [], s, a; a = (align & Annotation.AlignHorizontal); if (a === Annotation.AlignJustify) { s = "justify"; } else if (a === Annotation.AlignLeft) { s = "left"; } else if (a === Annotation.AlignRight) { s = "right"; } else { // if (a === Annotation.AlignCenter) { s = "center"; } classes.push("iex_annotation_align_" + s); a = (align & Annotation.AlignVertical); if (a === Annotation.AlignTop) { s = "top"; } else if (a === Annotation.AlignBottom) { s = "bottom"; } else { // if (a === Annotation.AlignMiddle) { s = "middle"; } classes.push("iex_annotation_valign_" + s); if ((align & Annotation.AlignHOverflow) !== 0) { classes.push("iex_annotation_align_x"); } if ((align & Annotation.AlignVOverflow) !== 0) { classes.push("iex_annotation_valign_x"); } return classes; }, align_to_classes_and_strings: function (align) { var classes = [], strings = [], s, a; a = (align & Annotation.AlignHorizontal); if (a === Annotation.AlignJustify) { s = "justify"; } else if (a === Annotation.AlignLeft) { s = "left"; } else if (a === Annotation.AlignRight) { s = "right"; } else { // if (a === Annotation.AlignCenter) { s = "center"; } strings.push(s); classes.push("iex_annotation_align_" + s); a = (align & Annotation.AlignVertical); if (a === Annotation.AlignTop) { s = "top"; } else if (a === Annotation.AlignBottom) { s = "bottom"; } else { // if (a === Annotation.AlignMiddle) { s = "middle"; } strings.push(s); classes.push("iex_annotation_valign_" + s); if ((align & Annotation.AlignHOverflow) !== 0) { classes.push("iex_annotation_align_x"); } if ((align & Annotation.AlignVOverflow) !== 0) { classes.push("iex_annotation_valign_x"); } return [ classes , strings ]; }, decoration_to_shadow_style: function (decoration, color) { var s = ""; if (decoration === Annotation.TextDecorationShadow) { s = "0.0625em 0.0625em 0 " + color; } else if (decoration === Annotation.TextDecorationShadowBlur) { s = "0 0 0.0625em " + color; s += "," + s; } else if (decoration === Annotation.TextDecorationShadowBlurStrong) { s = "0 0 0.125em " + color; s += "," + s; s += "," + s; } return s; }, apply_text: function () { var container = this.nodes.annotation.text, pos = 0, pos_last = -1, bold = false, italic = false, decoration = -1, colors = [ -1, -1, -1 ], self = this, i, j, f, s, type, i_max; var add_node = function (text) { if (text.length === 0) return; var n = $("span"), c = "", d; n.textContent = text; if (bold) { c = "iex_annotation_format_bold"; if (italic) c += " iex_annotation_format_italic"; n.className = c; } else if (italic) { n.className = "iex_annotation_format_italic"; } if (colors[0] >= 0) { n.style.color = "#" + Annotation.colors[colors[0]]; } if (colors[2] >= 0) { n.style.backgroundColor = "#" + Annotation.colors[colors[2]]; } if (colors[1] >= 0) { if ((d = decoration) >= 0 || (d = self.text_decoration) > 0) { c = self.decoration_to_shadow_style(d, "#" + Annotation.colors[colors[1]]); if (c.length === 0) c = "none"; n.style.textShadow = c; } } else if (decoration >= 0) { c = self.decoration_to_shadow_style(decoration, "#" + Annotation.colors[self.colors[1]]); if (c.length === 0) c = "none"; n.style.textShadow = c; } container.appendChild(n); }; // Clear container.textContent = ""; // Add content i_max = this.text_formatters.length; for (i = 0; i < i_max; ++i) { f = this.text_formatters[i]; type = f[0]; s = this.text_display.substr(pos, f[1] - pos); pos = f[1]; if (type === Annotation.TextFormattingBold) { add_node(s); bold = !bold; } else if (type === Annotation.TextFormattingItalic) { add_node( s); italic = !italic; } else if (type === Annotation.TextFormattingNewline) { add_node(s); container.appendChild($("br")); ++pos; pos_last = pos; } else if (type === Annotation.TextFormattingHyphen) { s += "-"; add_node(s); container.appendChild($("br")); } else if (type === Annotation.TextFormattingColor) { add_node(s); j = f[2]; if (j >= colors.length) { for (j = 0; j < colors.length; ++j) colors[j] = -1; } else { colors[j] = f[3]; } } else if (type === Annotation.TextFormattingDecoration) { add_node(s); decoration = f[2]; } else if (type === Annotation.TextFormattingDecorationReset) { add_node(s); decoration = -1; } else { // if (type === Annotation.TextFormattingReset) { add_node(s); for (j = 0; j < colors.length; ++j) colors[j] = -1; bold = false; italic = false; decoration = -1; } } // Last s = this.text_display.substr(pos); add_node(s); }, destroy: function () { this.remove_nodes(); remove_event_listeners(this.events); this.events = []; }, remove_nodes: function () { var n; // Deselect if (this.is_selected()) this.set_selected(false); // Stop events if (this.edit_data !== null && this.edit_data.move_events.length > 0) { this.on_annotation_move_document_mouseup(null); } // Remove nodes if ((n = this.nodes.annotation.container).parentNode !== null) { n.parentNode.removeChild(n); } if (this.nodes.edit !== null) { if ((n = this.nodes.edit.container).parentNode !== null) { n.parentNode.removeChild(n); } } }, insert_before: function (annotation) { var n = this.nodes.annotation.container, n_o = annotation.nodes.annotation.container; n_o.parentNode.insertBefore(n, n_o); if (this.nodes.edit !== null && annotation.nodes.edit !== null) { n = this.nodes.edit.container; n_o = annotation.nodes.edit.container; n_o.parentNode.insertBefore(n, n_o); } }, add_to: function (annotation_container, edit_container) { var n = this.nodes.annotation.container; annotation_container.appendChild(n); if (this.nodes.edit !== null && edit_container !== undefined) { n = this.nodes.edit.container; edit_container.appendChild(n); } }, create_nodes: function (edit) { // Edit if (edit) { this.nodes.edit = {}; this.create_nodes_edit(); } // Annotation this.nodes.annotation = {}; this.create_nodes_annotation(edit); // Setup this.update_annotation_rect(); this.set_align(this.text_align); this.set_color(0, this.colors[0]); this.set_color(1, this.colors[1]); this.set_color(2, this.colors[2]); this.set_font(this.font_index); this.set_font_size(this.font_size_index); this.set_text_line_spacing(this.text_line_spacing); this.set_text_char_spacing(this.text_char_spacing); }, create_nodes_annotation: function (edit) { var n1, n2, n3, n4, borders, i; n1 = $.div("iex_annotation"); this.nodes.annotation.container = n1; if (edit) { style.add_class(n1, "iex_annotation_editing"); add_event_listener(this.events, n1, "mousedown", this.on_annotation_mousedown.bind(this), false); add_event_listener(this.events, n1, "mouseover", wrap_mouseenterleave_event(this, this.on_annotation_mouseover), false); add_event_listener(this.events, n1, "mouseout", wrap_mouseenterleave_event(this, this.on_annotation_mouseout), false); } else { add_event_listener(this.events, n1, "mousedown", this.on_annotation_mousedown_no_edit.bind(this), false); } if (Annotation.scaling_mode_px) { style.add_class(n1, "iex_annotation_scaling_px"); } if (settings.values.annotations.transparent_until_hover) { style.add_class(n1, "iex_annotation_transparent"); } n2 = $.div("iex_annotation_borders"); n1.appendChild(n2); if (edit) { borders = [ "top", "bottom", "left", "right", "top_left", "top_right", "bottom_left", "bottom_right" ]; for (i = 0; i < borders.length; ++i) { n3 = $.div("iex_annotation_border iex_annotation_border_" + borders[i]); n2.appendChild(n3); add_event_listener(this.events, n3, "mousedown", this.on_annotation_resize_mousedown.bind(this, i), false); } } n2 = $.div("iex_annotation_background"); n1.appendChild(n2); this.nodes.annotation.background = n2; n2.style.borderColor = "#" + Annotation.colors[this.colors[2]]; n2 = $.div("iex_annotation_outline"); n1.appendChild(n2); this.nodes.annotation.border1 = n2; n2.style.borderColor = "#" + Annotation.colors[this.colors[0]]; n2 = $.div("iex_annotation_content"); n1.appendChild(n2); if (edit) { add_event_listener(this.events, n2, "mousedown", this.on_annotation_move_mousedown.bind(this), false); add_event_listener(this.events, n2, "contextmenu", this.on_annotation_contextmenu.bind(this), false); add_event_listener(this.events, n2, "dblclick", this.on_annotation_dblclick.bind(this), false); } this.nodes.annotation.content = n2; n3 = $.div("iex_annotation_content2"); n2.appendChild(n3); n4 = $.div("iex_annotation_content3"); n3.appendChild(n4); this.nodes.annotation.text = n4; }, on_annotation_mousedown_no_edit: function (event) { //event.preventDefault(); }, create_nodes_edit: function () { var n1, n2, n3, n4, n5; n1 = $.div("iex_ae_annotation_editor"); this.nodes.edit.container = n1; add_event_listener(this.events, n1, "mousedown", this.on_edit_region_mousedown.bind(this), false); // Info bar n2 = $.div("iex_ae_annotation_editor_top"); n1.appendChild(n2); n3 = $.div("iex_ae_annotation_editor_top_row"); n2.appendChild(n3); n4 = $.div("iex_ae_annotation_editor_top_cell"); n3.appendChild(n4); n5 = $.div("iex_ae_annotation_editor_number"); n4.appendChild(n5); n5.textContent = "#" + (this.index + 1); this.nodes.edit.index_text = n5; n4 = $.div("iex_ae_annotation_editor_top_cell"); n3.appendChild(n4); n5 = this.create_button_alignment(); n4.appendChild(n5); add_event_listener(this.events, n5, "click", this.on_edit_align_click.bind(this), false); add_event_listener(this.events, n5, "mousedown", this.on_edit_button_mousedown.bind(this), false); this.nodes.edit.button_align = n5; n5 = this.create_button_color(); n4.appendChild(n5); add_event_listener(this.events, n5, "click", this.on_edit_color_click.bind(this), false); add_event_listener(this.events, n5, "mousedown", this.on_edit_button_mousedown.bind(this), false); this.nodes.edit.button_color = n5; n5 = this.create_button("A"); n4.appendChild(n5); add_event_listener(this.events, n5, "click", this.on_edit_font_click.bind(this), false); add_event_listener(this.events, n5, "mousedown", this.on_edit_button_mousedown.bind(this), false); n5.setAttribute("title", "Font"); this.nodes.edit.button_font = n5; n5 = this.create_button("", "iex_ae_annotation_button_small_text"); n4.appendChild(n5); add_event_listener(this.events, n5, "click", this.on_edit_font_size_click.bind(this), false); add_event_listener(this.events, n5, "mousedown", this.on_edit_button_mousedown.bind(this), false); n5.setAttribute("title", "Font size"); this.nodes.edit.button_font_size = n5; this.nodes.edit.button_font_size_text = n5.firstChild; n5 = this.create_button("\u2191"); n4.appendChild(n5); add_event_listener(this.events, n5, "click", this.on_edit_move_up_click.bind(this), false); add_event_listener(this.events, n5, "mousedown", this.on_edit_button_mousedown.bind(this), false); n5.setAttribute("title", "Move up"); this.nodes.edit.button_move_up = n5; n5 = this.create_button("\u2193"); n4.appendChild(n5); add_event_listener(this.events, n5, "click", this.on_edit_move_down_click.bind(this), false); add_event_listener(this.events, n5, "mousedown", this.on_edit_button_mousedown.bind(this), false); n5.setAttribute("title", "Move down"); this.nodes.edit.button_move_down = n5; n5 = this.create_button("\u00D7"); n4.appendChild(n5); add_event_listener(this.events, n5, "click", this.on_edit_remove_click.bind(this), false); add_event_listener(this.events, n5, "mousedown", this.on_edit_button_mousedown.bind(this), false); n5.setAttribute("title", "Remove"); this.nodes.edit.button_remove = n5; // Input n2 = $.div("iex_ae_annotation_editor_bottom"); n1.appendChild(n2); n3 = $.input.text("iex_ae_annotation_editor_input_text"); n2.appendChild(n3); n3.setAttribute("placeholder", "annotation"); n3.style.width = "100%"; n3.style.boxSizing = "border-box"; add_event_listener(this.events, n3, "input", this.on_edit_text_input.bind(this), false); add_event_listener(this.events, n3, "change", this.on_edit_text_change.bind(this), false); add_event_listener(this.events, n3, "focus", this.on_edit_text_focus.bind(this), false); this.nodes.edit.text = n3; }, create_button: function (text, extra_class) { var n1, n2, cls; cls = "iex_ae_annotation_button"; if (extra_class) { cls += " "; cls += extra_class; } n1 = $.span(cls); n2 = $.span("iex_ae_annotation_button_inner"); n1.appendChild(n2); n2.textContent = text; return n1; }, create_button_color: function () { var n1, n2, n3; n1 = $.span("iex_ae_annotation_button iex_ae_annotation_button_small_text"); n2 = $.span("iex_ae_annotation_button_inner iex_ae_annotation_button_inner_color"); n1.appendChild(n2); n3 = $.span("iex_ae_annotation_button_color"); n2.appendChild(n3); n2 = $.span("iex_ae_annotation_button_inner iex_ae_annotation_button_color_text"); n2.textContent = "txt"; n1.appendChild(n2); this.nodes.edit.button_color_bg = n3; this.nodes.edit.button_color_text = n2; n1.setAttribute("title", "Color"); return n1; }, create_button_alignment: function () { var n1, n2; n1 = $.span("iex_ae_annotation_button iex_ae_annotation_button_align"); n2 = $.span("iex_ae_annotation_button_inner"); n1.appendChild(n2); n1.setAttribute("title", "Alignment"); return n1; }, initialize_move: function (mode, event) { this.edit_data.move_mode = mode; this.edit_data.move_origin[0] = event.clientX; this.edit_data.move_origin[1] = event.clientY; this.edit_data.move_position[0] = this.position[0]; this.edit_data.move_position[1] = this.position[1]; this.edit_data.move_size[0] = this.size[0]; this.edit_data.move_size[1] = this.size[1]; remove_event_listeners(this.edit_data.move_events); this.edit_data.move_events = []; add_event_listener(this.edit_data.move_events, document, "mousemove", this.on_annotation_move_document_mousemove.bind(this), true); add_event_listener(this.edit_data.move_events, document, "mouseup", this.on_annotation_move_document_mouseup.bind(this), true); }, scroll_to_editor: function () { var scroll_region = this.nodes.edit.container.parentNode, scroll_container = scroll_region.parentNode, check = this.nodes.edit.text, bound = this.nodes.edit.container, r1 = style.get_object_rect_relative(check), r2 = style.get_object_rect_relative(scroll_container); if (r1.top < r2.top) { r1 = style.get_object_rect_relative(bound); r2 = style.get_object_rect_relative(scroll_region); scroll_container.scrollTop = r1.top - r2.top; } else if (r1.bottom > r2.bottom) { r1 = style.get_object_rect_relative(bound); scroll_container.scrollTop = r1.bottom - style.get_object_rect_relative(scroll_region).top - r2.height; } }, select_text: function (target, event) { this.scroll_to_editor(); var ret = true, active_change = (document.activeElement !== this.nodes.edit.text); if (target !== this.nodes.edit.text) { if (active_change) { select_input(this.nodes.edit.text, this.nodes.edit.text.value.length || 0); } ret = false; } if (event && !this.is_selected()) { this.trigger("select"); } return ret; }, on_edit_button_mousedown: function (event) { return stop_event(event); }, on_edit_align_click: function (event) { this.trigger("align"); }, on_edit_color_click: function (event) { this.trigger("color"); }, on_edit_font_click: function (event) { this.trigger("font"); }, on_edit_font_size_click: function (event) { this.trigger("font_size"); }, on_edit_move_up_click: function (event) { this.trigger("move_up"); }, on_edit_move_down_click: function (event) { this.trigger("move_down"); }, on_edit_remove_click: function (event) { this.trigger("remove"); }, on_edit_region_mousedown: function (event) { if (this.select_text(event.target, true)) return true; return true;//stop_event(event); }, on_edit_text_input: function (event) { this.set_text(this.nodes.edit.text.value, true); this.trigger("text_input"); }, on_edit_text_change: function (event) { this.set_text(this.nodes.edit.text.value, false); this.trigger("text_change"); }, on_edit_text_focus: function (event) { if (!this.is_selected()) { this.trigger("select"); } }, on_annotation_mousedown: function (event) { this.edit_data.double_click_ignore = false; if (get_event_mouse_button(event) === 2) return; event.stopPropagation(); }, on_annotation_mouseover: function (event) { }, on_annotation_mouseout: function (event) { }, on_annotation_contextmenu: function (event) { event.stopPropagation(); }, on_annotation_resize_mousedown: function (index, event) { if (get_event_mouse_button(event) !== 1) return; var resize_dir; if (index < 4) { resize_dir = (1 << index); } else { resize_dir = (index < 6) ? Annotation.ResizeNorth : Annotation.ResizeSouth; resize_dir |= ((index % 2) === 0) ? Annotation.ResizeWest : Annotation.ResizeEast; } // Move this.initialize_move(resize_dir, event); return stop_event(event); }, on_annotation_move_mousedown: function (event) { this.edit_data.double_click_ignore = false; var button = get_event_mouse_button(event); if (button === 2) return; if (button !== 1) return stop_event(event); // Move this.initialize_move(0, event); if (this.select_text(null, true)) return true; return stop_event(event); }, on_annotation_dblclick: function (event) { if (!this.edit_data.double_click_ignore) { this.auto_size(); } this.edit_data.double_click_ignore = false; return stop_event(event); }, on_annotation_move_document_mousemove: function (event) { this.edit_data.double_click_ignore = true; var scale = this.parent.get_image_scale(), image_size = this.parent.get_image_size(), xo = (event.clientX - this.edit_data.move_origin[0]) / scale, yo = (event.clientY - this.edit_data.move_origin[1]) / scale, x = this.edit_data.move_position[0], y = this.edit_data.move_position[1], w = this.edit_data.move_size[0], h = this.edit_data.move_size[1]; if (this.edit_data.move_mode === 0) { // Move x = Math.round(x + xo); y = Math.round(y + yo); // Limit if (x + w > image_size.width) { x = image_size.width - w; } if (y + h > image_size.height) { y = image_size.height - h; } if (x < 0) x = 0; if (y < 0) y = 0; } else { if ((this.edit_data.move_mode & Annotation.ResizeWest) !== 0) { // Change x and width x = Math.round(x + xo); w = Math.round(w + (this.edit_data.move_position[0] - x)); if (x < 0) { w += x; x = 0; } if (w < 0) w = 0; } else if ((this.edit_data.move_mode & Annotation.ResizeEast) !== 0) { // Change width w = Math.round(w + xo); if (x + w > image_size.width) { w = image_size.width - x; } if (w < 0) w = 0; } if ((this.edit_data.move_mode & Annotation.ResizeNorth) !== 0) { // Change y and height y = Math.round(y + yo); h = Math.round(h + (this.edit_data.move_position[1] - y)); if (y < 0) { h += y; y = 0; } if (h < 0) h = 0; } else if ((this.edit_data.move_mode & Annotation.ResizeSouth) !== 0) { // Change height h = Math.round(h + yo); if (y + h > image_size.height) { h = image_size.height - y; } if (h < 0) h = 0; } } // Update this.set_annotation_rect(x, y, w, h); this.trigger("rect_change"); }, on_annotation_move_document_mouseup: function (event) { remove_event_listeners(this.edit_data.move_events); this.edit_data.move_events = []; this.trigger("rect_changed"); }, trigger: function (event_name, data) { this.parent.on_annotation_event(this, event_name, data); }, }; Annotation.ResizeNorth = 0x1; Annotation.ResizeSouth = 0x2; Annotation.ResizeWest = 0x4; Annotation.ResizeEast = 0x8; Annotation.AlignCenter = 0x0; Annotation.AlignLeft = 0x1; Annotation.AlignRight = 0x2; Annotation.AlignJustify = 0x3; Annotation.AlignMiddle = 0x0; Annotation.AlignTop = 0x4; Annotation.AlignBottom = 0x8; Annotation.AlignHOverflow = 0x10; Annotation.AlignVOverflow = 0x20; Annotation.AlignHorizontal = 0x3; Annotation.AlignVertical = 0xC; Annotation.AlignDefault = Annotation.AlignCenter | Annotation.AlignMiddle | Annotation.AlignHOverflow | Annotation.AlignVOverflow; Annotation.TextDecorationNone = 0x0; Annotation.TextDecorationShadow = 0x1; Annotation.TextDecorationShadowBlur = 0x2; Annotation.TextDecorationShadowBlurStrong = 0x3; Annotation.TextDecorationCount = 0x4; Annotation.TextFormattingBold = 0x0; Annotation.TextFormattingItalic = 0x1; Annotation.TextFormattingNewline = 0x2; Annotation.TextFormattingHyphen = 0x3; Annotation.TextFormattingColor = 0x4; Annotation.TextFormattingDecoration = 0x5; Annotation.TextFormattingDecorationReset = 0x6; Annotation.TextFormattingReset = 0x7; Annotation.TextCharSpacingRange = { min: -0.5, range: 1.5, steps: 15, }; Annotation.TextLineSpacingRange = { min: 0.0, range: 1.5, steps: 15, }; Annotation.scaling_mode_px = true; Annotation.scaling_mode_units = (Annotation.scaling_mode_px ? "em" : "px"); Annotation.colors = [ //{ // dark, normal, light "000000", "ffffff", // black and white "333333", "808080", "cccccc", // gray "5a0707", "e11212", "f3a0a0", // red "5a3807", "e18c12", "f3d1a0", // orange "544e07", "d4c412", "ede7a0", // yellow "2a560c", "6ad91f", "c3efa5", // green "0a5648", "19d7b6", "a3efe1", // teal "0a4356", "19a8d7", "a3dcef", // lt blue "141157", "342cdb", "adaaf0", // blue "340a56", "8419d7", "cda3ef", // purple "560a43", "d719a8", "efa3dc", // fuchsia ]; //} Annotation.color_shades = 3; Annotation.fonts = [{ name: "Arial", css_name: "arial", external_name: null, external_settings: null, }, { name: "Wild Words", css_name: "iex_wildwords", external_name: "ww", external_settings: [ "r", "b", "bi" ], }]; Annotation.font_size_settings = { base: 8, base_step: 1, step_duration: 8, step_count: 5, step_multiplier: 2, }; Annotation.font_size_index_to_size = function (index) { var fss = Annotation.font_size_settings; if (index < 0) { index = 0; } else if (index >= fss.step_duration * fss.step_count) { index = fss.step_duration * fss.step_count - 1; } var step = Math.floor(index / fss.step_duration), step_index = index % fss.step_duration, size = fss.base, step_size = fss.base_step, i; for (i = 0; i < step; ++i) { size += step_size * fss.step_duration; step_size *= fss.step_multiplier; } size += step_index * step_size; return size; }; Annotation.font_size_to_size_info = function (size) { var fss = Annotation.font_size_settings, base = fss.base, step = fss.base_step, base_next, i, off; // Min if (size < base) return [ 0 , base ]; // [ index, size ] for (i = 0; i < fss.step_count; ++i) { base_next = base + step * fss.step_duration; if (size < base_next) { off = (fss.step_duration - Math.round((base_next - size) / step)); return [ i * fss.step_duration + off, base + off * step ]; } step *= fss.step_multiplier; base = base_next; } // Max return [ i * fss.step_duration - 1, base - step / fss.step_multiplier ]; }; Annotation.char_spacing = { from_index: function (index) { var index_limit = Annotation.TextCharSpacingRange.steps; if (index < 0) { index = 0; } else if (index > index_limit) { index = index_limit; } return Annotation.TextCharSpacingRange.min + Annotation.TextCharSpacingRange.range * (index / index_limit); }, to_index: function (spacing) { if (spacing <= Annotation.TextCharSpacingRange.min) return 0; if (spacing >= Annotation.TextCharSpacingRange.min + Annotation.TextCharSpacingRange.range) return Annotation.TextCharSpacingRange.steps; return Math.round((spacing - Annotation.TextCharSpacingRange.min) / Annotation.TextCharSpacingRange.range * Annotation.TextCharSpacingRange.steps); }, }; Annotation.line_spacing = { from_index: function (index) { var index_limit = Annotation.TextLineSpacingRange.steps; if (index < 0) { index = 0; } else if (index > index_limit) { index = index_limit; } return Annotation.TextLineSpacingRange.min + Annotation.TextLineSpacingRange.range * (index / index_limit); }, to_index: function (spacing) { if (spacing <= Annotation.TextLineSpacingRange.min) return 0; if (spacing >= Annotation.TextLineSpacingRange.min + Annotation.TextLineSpacingRange.range) return Annotation.TextLineSpacingRange.steps; return Math.round((spacing - Annotation.TextLineSpacingRange.min) / Annotation.TextLineSpacingRange.range * Annotation.TextLineSpacingRange.steps); }, }; var TextFormatter = function () { this.code = ""; this.before = ""; this.after = ""; this.no_spaces = false; this.force_update = false; this.raw_text = ""; this.text = ""; this.formatters = []; this.match = null; this.match_length = 0; }; TextFormatter.bind = function (fn, args) { return function () { return fn.apply(this, args); }; }; TextFormatter.code_simple = function (formatting_code, extra) { if (this.before.length > 0 || (this.force_update = (this.after.length > 0))) { this.before = " "; this.after = ""; this.text += " "; } var f = [ formatting_code, this.text.length ]; if (extra !== undefined) Array.prototype.push.apply(f, extra); this.formatters.push(f); return true; }; TextFormatter.codes = { "\\": function () { this.no_spaces = false; this.text += this.before; this.text += this.code; this.text += this.after; return true; }, "h": function () { this.before = ""; this.after = ""; this.formatters.push([ Annotation.TextFormattingHyphen, this.text.length ]); return true; }, "n": function () { this.before = ""; this.after = ""; if (this.text.length > 0 && /^\s/.test(this.text[this.text.length - 1])) { // Sequential newlines this.text = this.text.substr(0, this.text.length - 1); } this.formatters.push([ Annotation.TextFormattingNewline, this.text.length ]); this.text += " "; return true; }, "c": function () { var c = 0, i, m, p; // Invalid check if ( this.after.length > 0 || (m = TextFormatter.re_color.exec(this.raw_text.substr(this.match.index + this.match_length))) === null ) { // Invalid return false; } // Move next this.match_length += m[0].length; p = this.match.index + this.match_length; // Whitespace if (this.before.length > 0) this.text += " "; // Color index if (m[1] !== undefined) { c = Math.min(parseInt(m[1], 10), 2); if (c > 0) { this.code += c; this.code += "."; } } // Color i = Math.min(parseInt(m[2], 10), Annotation.colors.length - 1); this.code += i; if (p < this.raw_text.length && TextFormatter.re_number.test(this.raw_text[p])) this.code += ":"; // Formatting this.formatters.push([ Annotation.TextFormattingColor, this.text.length, c, i ]); this.force_update = true; return true; }, "d": function () { var i, m; // Invalid check if ( this.after.length > 0 || (m = TextFormatter.re_number.exec(this.raw_text.substr(this.match.index + this.match_length))) === null ) { // Invalid return false; } // Move next this.match_length += m[0].length; // Whitespace if (this.before.length > 0) this.text += " "; // Decoration i = Math.min(parseInt(m[0], 10), Annotation.TextDecorationCount - 1); this.code += i; // Formatting this.formatters.push([ Annotation.TextFormattingDecoration, this.text.length, i ]); this.force_update = true; return true; }, "b": TextFormatter.bind(TextFormatter.code_simple, [ Annotation.TextFormattingBold ]), "i": TextFormatter.bind(TextFormatter.code_simple, [ Annotation.TextFormattingItalic ]), "C": TextFormatter.bind(TextFormatter.code_simple, [ Annotation.TextFormattingColor, [ 3 ] ]), "D": TextFormatter.bind(TextFormatter.code_simple, [ Annotation.TextFormattingDecorationReset ]), "R": TextFormatter.bind(TextFormatter.code_simple, [ Annotation.TextFormattingReset ]), }; TextFormatter.re_formatter = /(?:(\s*)\\([\w\W])(\s*))/g; TextFormatter.re_color = /^(?:([0-9])\.)?([0-9]+):?/; TextFormatter.re_number = /^[0-9]/; TextFormatter.re_space = /^\s/; TextFormatter.MaxTextFormatters = 127; // (2^(2^3 - 1) - 1) where 3 is TagWriter.bits_per.formatter_count_bits TextFormatter.get_raw = function (text, formatters) { var raw_text = "", pos = 0, i, f, s, type; // Add content for (i = 0; i < formatters.length; ++i) { f = formatters[i]; type = f[0]; s = text.substr(pos, f[1] - pos); pos = f[1]; raw_text += s.replace(/\\/g, "\\\\"); if (type === Annotation.TextFormattingBold) { raw_text += "\\b"; } else if (type === Annotation.TextFormattingItalic) { raw_text += "\\i"; } else if (type === Annotation.TextFormattingNewline) { if (raw_text.length > 0 && TextFormatter.re_space.test(raw_text[raw_text.length - 1])) { raw_text = raw_text.substr(0, raw_text.length - 1); // multiple newlines } raw_text += "\\n"; if (text.length > pos && TextFormatter.re_space.test(text[pos])) { ++pos; // whitespace } } else if (type === Annotation.TextFormattingHyphen) { raw_text += "\\h"; } else if (type === Annotation.TextFormattingColor) { if (f[2] >= 3) { // colors.length) { raw_text += "\\C"; } else { raw_text += "\\c"; if (f[2] > 0) { raw_text += f[2]; raw_text += "."; } raw_text += f[3]; if (text.length > pos && TextFormatter.re_number.test(text[pos])) { raw_text += ":"; // optional } } } else if (type === Annotation.TextFormattingDecoration) { raw_text += "\\d1"; } else if (type === Annotation.TextFormattingDecorationReset) { raw_text += "\\D"; } else { // if (type === Annotation.TextFormattingReset) { raw_text += "\\R"; } } // Last raw_text += text.substr(pos).replace(/\\/g, "\\\\"); return raw_text; }; TextFormatter.normalize_text = function (text) { return text.trim().replace(/\t\s*| \s+/g, " "); }; TextFormatter.prototype = { constructor: TextFormatter, parse: function (text) { var i = 0, fn, match_text; this.no_spaces = false; this.raw_text = TextFormatter.normalize_text(text); TextFormatter.re_formatter.lastIndex = 0; while ((this.match = TextFormatter.re_formatter.exec(this.raw_text)) !== null) { // Setup this.code = this.match[2]; // char this.before = this.match[1]; // before this.after = this.match[3]; // after this.text += this.raw_text.substr(i, this.match.index - i); this.match_length = this.match[0].length; // Process fn = TextFormatter.codes[this.code]; if (fn === undefined || !fn.call(this)) { // Invalid this.no_spaces = false; this.text += this.match[0]; } else { // Don't overflow if (this.formatters.length > TextFormatter.MaxTextFormatters) { this.formatters.splice(TextFormatter.MaxTextFormatters, this.formatters.length - TextFormatter.MaxTextFormatters); } } // Remove extra spaces between things like " \b \i \b \i " (turns into " \b\i\b\i") if (this.no_spaces && this.match.index === i) { this.before = ""; this.after = ""; } this.no_spaces = (this.match[1].length + this.match[3].length > 0); // Text update match_text = this.before + "\\" + this.code + this.after; if (this.force_update || match_text.length !== this.match_length) { this.raw_text = this.raw_text.substr(0, this.match.index) + match_text + this.raw_text.substr(this.match.index + this.match_length); this.force_update = false; } // Position update i = this.match.index + match_text.length; TextFormatter.re_formatter.lastIndex = i; } // Remainder if (i === 0) { this.text = this.raw_text; } else if (i < this.raw_text.length) { this.text += this.raw_text.substr(i); } }, }; Annotation.TextFormatter = TextFormatter; var State = function () { this.rect = { x: 0, y: 0, width: 0, height: 0 }; this.font = 0; this.bold = false; this.italic = false; this.font_size = 0; this.align = 0; this.colors = [ 0, 0, 0 ]; this.text_decoration = 0; this.line_spacing = 0; this.char_spacing = 0; this.formatters = []; this.text = null; }; Annotation.State = State; return Annotation; })(); // Image annotation control var Annotator = (function () { // Annotator class var Annotator = function () { // Get posts this.post_queue = null; this.post_id_current = -1; this.annotation_nodes = { container: null, overlay: null, }; this.annotation_data = {}; this.annotations = []; this.annotations_active = false; }; // Public functions Annotator.prototype = { constructor: Annotator, start: function (image_hover, file_link) { // Get posts this.post_queue = Delay.queue(api.get("posts"), on_post_queue_callback.bind(this), 5, 0.125); // Bind post acquiring api.on("post_add", on_api_post_add.bind(this)); api.on("post_remove", on_api_post_remove.bind(this)); // Hover image_hover.on("change", on_image_hover_change.bind(this)); this.on_preview_size_change_bind = null; this.on_preview_resize_bind = null; this.toggle_hotkey = null; // Links if (settings.values.annotations.modify_urls) { file_link.register_modifier(on_image_link_change.bind(this)); } }, }; // Annotation data for a single image var AnnotationData = function (data) { this.datas = [ data ]; this.current = 0; }; // Private functions var re_annotation_texts = /^(?:|([\w\W]*?)(?:\r\n?|\n))[ \t]*>((?:[^>\r\n][^\r\n]*)?)(\r\n?|\n|$)/; var re_annotation = /([ \t]*)(\[iex:a\/([^\]]*)\])\s*$/g; var checked_attr = "data-iex-annotation-checked"; var on_api_post_add = function (post_container) { if (post_container.getAttribute(checked_attr) === "true") { post_container.setAttribute(checked_attr, "false"); } if (api.post_is_floating_or_embedded(post_container)) { // Immediate on_post_queue_callback.call(this, post_container); } else { this.post_queue.push(post_container); } }; var on_api_post_remove = function (post_container) { if (post_container.getAttribute(checked_attr) === "false") { this.post_queue.pop(post_container); } }; var on_post_queue_callback = function (post_container) { var not_processed = !post_container.getAttribute(checked_attr); post_container.setAttribute(checked_attr, "true"); // Process container if (not_processed) { // Not processed yet var tag = null, tag_node = null; api.post_comment_scan(post_container, function (text, pos) { re_annotation.lastIndex = pos; var m = re_annotation.exec(text), i; if (m === null) return null; i = m.index + m[1].length; return [ i , i + m[2].length , m ]; }, "a", function (node, match) { node.className = "iex_annotation_tag"; tag = match[2][3]; tag_node = node; } ); if (tag !== null) { setup_tag.call(this, post_container, tag, tag_node); } } /*else { // Already processed var tags = api.post_query_selector_all(post_container, "a.iex_annotation_tag"), i; }*/ }; var on_image_hover_change = function (data) { var n = data.image_container, p_id = (n === null || (n = api.post_get_post_container_from_image_container(n)) === null) ? -1 : api.post_get_id(n); if (p_id < 0) { if (this.post_id_current >= 0) { // Update set_current_post.call(this, data.image_hover, data.type, p_id); } } else if (p_id !== this.post_id_current) { // Update set_current_post.call(this, data.image_hover, data.type, p_id); } }; var on_preview_resize = function (image_hover, annotation_container) { var mpreview = image_hover.mpreview, cpreview = mpreview.cpreview, w = mpreview.get_size().width, scale = (w > 0 ? (cpreview.size.width / w) : 1.0); annotation_container.style.fontSize = scale.toFixed(16) + "px"; }; var on_preview_size_change = function (image_hover, annotation_data) { var true_size = image_hover.mpreview.get_size(), current = annotation_data.datas[annotation_data.current], tag_size = current.image; if (true_size.width !== tag_size.width || true_size.height !== tag_size.height) { // Scale var xs = true_size.width / tag_size.width, ys = true_size.height / tag_size.height, i, r; for (i = 0; i < this.annotations.length; ++i) { r = current.annotations[i].rect; this.annotations[i].set_annotation_rect( Math.round(r.x * xs), Math.round(r.y * ys), Math.round(r.height * xs), Math.round(r.width * ys) ); } } }; var on_image_link_change = function (node, href) { var a = node.getAttribute("data-iex-annotations"), post_id, data; // Fast mode if (a) return href + "#" + a; // Get id post_id = api.post_get_id_from_node(node); data = this.annotation_data[post_id]; if (data) { href += "#"; href += get_url_extension.call(this, data); } return href; }; var on_toggle_hotkey = function () { var post_id = this.post_id_current, annotation_data = this.annotation_data[post_id]; // Change annotation annotation_data.current = ((annotation_data.current + 2) % (annotation_data.datas.length + 1)) - 1; // Clear and create clear_annotations.call(this, image_hover); create_annotations.call(this, image_hover, post_id, annotation_data); }; var setup_tag = function (post_container, tag, tag_node) { var target_post_id = api.post_get_id(post_container), tr, post_info, file_nodes, post_id, data, texts, i, url_ex, url; // Validate tag tr = new TagReader(); if (!tr.read(tag)) { return set_tag_error.call(this, tag_node, tr.get_error()); } // Find post reference post_info = setup_tag_find_reference_post.call(this, post_container, tr); if (post_info[0] === null) { return set_tag_error.call(this, tag_node, post_info[2]); } // Find image if ((file_nodes = api.post_get_file_nodes(post_info[0])) === null) { // No image return set_tag_error.call(this, tag_node, "No image attached to post " + post_info[1]); } if (file_nodes.link === null || file_nodes.link_thumbnail === null) { // Invalid return set_tag_error.call(this, tag_node, "Image link and/or thumbnail missing for post " + post_info[1]); } // Find texts texts = setup_tag_find_annotation_texts.call(this, post_container); if (texts.length === 0) { // No text return set_tag_error.call(this, tag_node, "No annotation texts found"); } // Set annotation texts if (tr.annotations.length === 0) return; // okay, but no need to do anything if (texts.length > tr.annotations.length) { texts.splice(tr.annotations.length, texts.length - tr.annotations.length); } else if (texts.length < tr.annotations.length) { tr.annotations.splice(texts.length, tr.annotations.length - texts.length); } for (i = 0; i < texts.length; ++i) { tr.annotations[i].text = texts[i]; } // Set data tr.tag = tag; post_id = post_info[1]; if (post_id in this.annotation_data) { data = this.annotation_data[post_id]; data.datas.push(tr); } else { data = new AnnotationData(tr); this.annotation_data[post_id] = data; } // Update links url_ex = get_url_extension.call(this, data); url = (file_nodes.link.getAttribute("href") || "").replace(/#.*$/, ""); tag_node.setAttribute("target", "_blank"); tag_node.setAttribute("href", url + "#!" + url_ex); if (settings.values.annotations.modify_urls) { update_image_link.call(this, file_nodes.link, url_ex); update_image_link.call(this, file_nodes.link_thumbnail, url_ex); } // Apply annotations if open if (this.post_id_current === target_post_id && !this.annotations_active) { // Apply annotations create_annotations.call(this, image_hover, post_id, data); } return true; }; var setup_tag_find_annotation_texts = function (post_container) { var post_text = api.post_get_comment_text(post_container), lines = [], m; // Find lines while ((m = re_annotation_texts.exec(post_text)) !== null) { post_text = post_text.substr(m.index + m[0].length); lines.push(Annotation.TextFormatter.normalize_text(m[2])); } // Done return lines; }; var setup_tag_find_reference_post = function (post_container, tag_reader) { if (tag_reader.image.mode === "post") { var post_id = -1, comment, links, i, j, n, modval; if ( (comment = api.post_get_comment_container(post_container)) !== null && (links = api.post_get_quotelinks(comment)).length > 0 ) { // Find the post number if (tag_reader.image.post_number_suffix < 0) { for (i = 0; i < links.length; ++i) { post_id = api.get_quotelink_target(links[i]); n = api.get_post_container_from_id(post_id); if (n !== null && api.post_has_file(n)) { break; } } if (i >= links.length) { // Error return [ null, -1, "No post reference with an image found in the comment" ]; } } else { modval = Math.pow(10, tag_reader.image.post_number_suffix_digits); for (i = 0; i < links.length; ++i) { j = api.get_quotelink_target(links[i]); if ((j % modval) === tag_reader.image.post_number_suffix) { post_id = j; break; } } if (post_id < 0) { // Error i = tag_reader.image.post_number_suffix.toString(); while (i.length < tag_reader.image.post_number_suffix_digits) i = "0" + i; return [ null, -1, "No post reference ending in " + i + " found in the comment" ]; } n = api.get_post_container_from_id(post_id); } // Find the post container from the post number if (n === null) { // Error return [ null, -1, "Post number " + post_id + " could not be found" ]; } return [ n, post_id ]; } else { return [ null, -1, "No post references found in the comment" ]; } } // Else, assume it's the posted image return [ post_container, api.post_get_id(post_container) ]; }; var set_tag_error = function (tag_node, error) { style.add_class(tag_node, "iex_annotation_tag_error"); tag_node.setAttribute("title", "Invalid annotation tag: " + error); return false; }; var update_image_link = function (node, url_ex) { var url = node.getAttribute("href") || ""; url += (url.indexOf("#") < 0) ? "#!" : "#"; url += url_ex; node.setAttribute("data-iex-annotations", url_ex); node.setAttribute("href", url); }; var get_url_extension = function (a_data) { var url_ex = "", i, j, d; for (i = 0; i < a_data.datas.length; ++i) { if (i > 0) url_ex += "#"; d = a_data.datas[i]; url_ex += "iex:a/" + d.tag; d = d.annotations; url_ex += "?" + encodeURIComponent(d[0].text); for (j = 1; j < d.length; ++j) { url_ex += "&"; url_ex += encodeURIComponent(d[j].text); } } return url_ex; }; var set_current_post = function (image_hover, type, post_id) { if (this.annotations_active) { clear_annotations.call(this, image_hover); } this.post_id_current = post_id; var d; if (type === "image" && (d = this.annotation_data[post_id])) { // Check if this post_id has any annotations create_annotations.call(this, image_hover, post_id, d); } }; var clear_annotations = function (image_hover) { this.annotations_active = false; if (this.annotation_nodes.overlay !== null) { // Remove size event image_hover.mpreview.off("resize", this.on_preview_resize_bind); image_hover.mpreview.on("size_change", this.on_preview_size_change_bind); this.on_preview_resize_bind = null; this.on_preview_size_change_bind = null; // Remove overlay var n = this.annotation_nodes.overlay.parentNode; if (n !== null) n.removeChild(this.annotation_nodes.overlay); this.annotation_nodes.overlay = null; this.annotation_nodes.container = null; // Remove annotations for (var i = 0; i < this.annotations.length; ++i) { this.annotations[i].destroy(); } this.annotations = []; } // Remove hotkeys hotkey_manager.unregister(this.toggle_hotkey); this.toggle_hotkey = null; }; var create_annotations = function (image_hover, post_id, annotation_data) { var mpreview = image_hover.mpreview, cpreview = mpreview.cpreview; // Overlay this.annotations_active = true; if (annotation_data.current >= 0) { this.annotation_nodes.overlay = $.div("iex_annotation_overlay"); this.annotation_nodes.container = $.div("iex_annotation_container"); this.annotation_nodes.overlay.appendChild(this.annotation_nodes.container); this.annotation_nodes.overlay.style.pointerEvents = "none"; // Create annotations create_annotations_from_list.call(this, annotation_data.datas[annotation_data.current].annotations, this.annotations, this.annotation_nodes.container ); // Add cpreview.add_content(this.annotation_nodes.overlay); // Bind events this.on_preview_resize_bind = on_preview_resize.bind(this, image_hover, this.annotation_nodes.container); this.on_preview_size_change_bind = on_preview_size_change.bind(this, image_hover, annotation_data); mpreview.on("resize", this.on_preview_resize_bind); mpreview.on("size_change", this.on_preview_size_change_bind); // Trigger size update this.on_preview_resize_bind(); this.on_preview_size_change_bind(); } // Hotkey this.toggle_hotkey = hotkey_manager.register(settings.values.annotations.toggle_hotkey, HotkeyManager.MODIFIER_NONE, on_toggle_hotkey.bind(this)); }; var create_annotations_from_list = function (list, new_list, target) { var i, a; for (i = 0; i < list.length; ++i) { a = new Annotation(null, null); a.load_from_state(list[i], null); a.add_to(target, null); new_list.push(a); } }; // Standalone annotator, for a single image var Standalone = function (image_node, annotation_data) { // Vars this.overlay = $.div("iex_annotation_standalone_overlay"); this.overlay.style.display = "none"; this.annotation_data = annotation_data; this.image_node = image_node; this.annotations = []; this.image_size = { width: 0, height: 0, acquired: false, }; this.image_node_rect = null; this.secondary_timer = null; this.on_secondary_timer_bind = this.on_secondary_timer.bind(this); // Image events this.image_node.addEventListener("load", this.on_image_load.bind(this), false); this.image_node.addEventListener("error", this.on_image_error.bind(this), false); this.image_size_poll_interval = setInterval(this.on_image_poll.bind(this), 200); // Changes window.addEventListener("resize", this.on_window_resize.bind(this), false); var ob = new MutationObserver(this.on_image_attribute_change.bind(this)); ob.observe(this.image_node, { attributes: true }); // Create annotations this.create_annotations(); // Insert overlay var rel = this.image_node.nextSibling; if (rel !== null) { this.image_node.parentNode.insertBefore(this.overlay, rel); } else { this.image_node.parentNode.appendChild(this.overlay); } // Start this.on_image_poll(); this.update_position(); // Hotkey this.toggle_hotkey = hotkey_manager.register(settings.values.annotations.toggle_hotkey, HotkeyManager.MODIFIER_NONE, this.toggle.bind(this)); }; Standalone.prototype = { constructor: Standalone, clear_annotations: function () { for (var i = 0; i < this.annotations.length; ++i) { this.annotations[i].destroy(); } this.annotations = []; }, create_annotations: function () { if (this.annotation_data.current < 0) return; // Create annotations create_annotations_from_list.call(this, this.annotation_data.datas[this.annotation_data.current].annotations, this.annotations, this.overlay ); }, toggle: function () { // Change annotation this.annotation_data.current = ((this.annotation_data.current + 2) % (this.annotation_data.datas.length + 1)) - 1; // Clear and create this.clear_annotations(); this.create_annotations(); }, on_image_load: function () { if (!this.image_size.acquired) { this.acquire_size(); } }, on_image_error: function () { this.overlay.style.display = "none"; }, on_image_poll: function () { if (this.image_node.naturalWidth && this.image_node.naturalHeight) { this.acquire_size(); } }, on_window_resize: function () { this.update_position(); if (this.secondary_timer !== null) clearTimeout(this.secondary_timer); this.secondary_timer = setTimeout(this.on_secondary_timer_bind, 10); }, on_image_attribute_change: function () { this.update_position(); if (this.secondary_timer !== null) clearTimeout(this.secondary_timer); this.secondary_timer = setTimeout(this.on_secondary_timer_bind, 10); }, on_secondary_timer: function () { this.secondary_timer = null; this.update_position(); }, acquire_size: function () { this.image_size.width = this.image_node.naturalWidth; this.image_size.height = this.image_node.naturalHeight; this.image_size.acquired = true; this.overlay.style.display = ""; clearInterval(this.image_size_poll_interval); this.image_size_poll_interval = null; if (this.image_node_size !== null) { this.update_position(); if (this.secondary_timer !== null) clearTimeout(this.secondary_timer); this.secondary_timer = setTimeout(this.on_secondary_timer_bind, 10); } }, update_position: function () { this.image_node_rect = style.get_object_rect(this.image_node); var s = this.overlay.style; s.left = (this.image_node_rect.left) + "px"; s.top = (this.image_node_rect.top) + "px"; s.width = this.image_node_rect.width + "px"; s.height = this.image_node_rect.height + "px"; if (this.image_size.acquired) this.update_size(); }, update_size: function () { var scale = this.image_node_rect.width / this.image_size.width; this.overlay.style.fontSize = scale.toFixed(16) + "px"; }, }; // Annotation tag writer var TagWriter = function () { this.bit_stream = null; this.defaults = null; this.bits_per = { width: 0, height: 0, post_number: 0, formatter_count: 0, formatter_position: 0, }; }; TagWriter.prototype = { constructor: TagWriter, write: function (image, annotations, suffix_text) { // Setup this.bit_stream = new BitStream(); this.bits_per.width = TagWriter.log2bits(image.size.width); this.bits_per.height = TagWriter.log2bits(image.size.height); // Header this.write_header(image, annotations, suffix_text); // Annotations if (annotations.length > 0) { // Default values this.write_annotations_header(annotations); // Annotations for (var i = 0; i < annotations.length; ++i) { this.write_annotation_info(annotations[i]); } // Clear this.defaults = null; } // CRC var crc = crc32(this.bit_stream.encode()); this.bit_stream.add(crc, 32); // Clean var s = this.bit_stream.encode(); this.bit_stream = null; return "[iex:a/" + s + "]"; }, write_header: function (image, annotations, suffix_text) { var bits_per = TagWriter.bits_per; this.bit_stream.add(TagWriter.version, bits_per.version); // version this.bit_stream.add(annotations.length, bits_per.annotation_count); // number of annotations this.bit_stream.add((image.mode === "file" ? 0 : (image.mode === "post" ? 1 : 2)), bits_per.image_mode); // image mode [0=file, 1=post, 2=url] if (image.mode === "post") { // Find how unique the post number is var s = image.extra.toString(), refs = TagWriter.get_all_post_references(annotations, suffix_text), unique_digits = TagWriter.get_longest_same_end(s, refs); if (refs.length <= 1) { this.bit_stream.add(0, bits_per.post_number_bits); } else { ++unique_digits; this.bits_per.post_number = Math.ceil(Math.log(10) / TagWriter.ln2 * unique_digits); this.bit_stream.add(this.bits_per.post_number, bits_per.post_number_bits); this.bit_stream.add(parseInt(s.substr(s.length - unique_digits), 10), this.bits_per.post_number); } } this.bit_stream.add(this.bits_per.width, bits_per.width_bits); // bits for width this.bit_stream.add(this.bits_per.height, bits_per.height_bits); // bits for height this.bit_stream.add(image.size.width, this.bits_per.width); // width this.bit_stream.add(image.size.height, this.bits_per.height); // height }, write_annotations_header: function (annotations) { // Default values var bits_per = TagWriter.bits_per, i, k, f, v; this.defaults = {}; for (k in TagWriter.defaults_formatters) { f = TagWriter.defaults_formatters[k]; this.defaults[k] = TagWriter.find_most_common(annotations, f[0], f[1]); } // Write defaults this.bit_stream.add(this.defaults.font[0], bits_per.font); // font this.bit_stream.add(this.defaults.font[1] ? 1 : 0, bits_per.font_bold); // bold this.bit_stream.add(this.defaults.font[2] ? 1 : 0, bits_per.font_italic); // italic this.bit_stream.add(this.defaults.font_size, bits_per.font_size); // font size this.bit_stream.add(this.defaults.align, bits_per.align); // align this.bit_stream.add(this.defaults.color_deco[0], bits_per.color); // color this.bit_stream.add(this.defaults.color_deco[1], bits_per.color); // color alt this.bit_stream.add(this.defaults.color_deco[2], bits_per.color); // color bg this.bit_stream.add(this.defaults.color_deco[3], bits_per.text_decoration); // text decoration this.bit_stream.add(this.defaults.spacing[0], bits_per.line_spacing); // line spacing this.bit_stream.add(this.defaults.spacing[1], bits_per.char_spacing); // char spacing // Bit ranges this.bits_per.formatter_count = 0; this.bits_per.formatter_position = 0; for (i = 0; i < annotations.length; ++i) { f = annotations[i].get_text_formatters(); if (f.length > this.bits_per.formatter_count) this.bits_per.formatter_count = f.length; for (k = 0; k < f.length; ++k) { v = f[k][1]; if (v > this.bits_per.formatter_position) this.bits_per.formatter_position = v; } } if (this.bits_per.formatter_count > 0) this.bits_per.formatter_count = TagWriter.log2bits(this.bits_per.formatter_count); if (this.bits_per.formatter_position > 0) this.bits_per.formatter_position = TagWriter.log2bits(this.bits_per.formatter_position); this.bit_stream.add(this.bits_per.formatter_count, bits_per.formatter_count_bits); // bits per formatter count number this.bit_stream.add(this.bits_per.formatter_position, bits_per.formatter_position_bits); // bits per formatter position number }, write_annotation_info: function (annotation) { var pos = annotation.get_position(), size = annotation.get_size(), formatters = annotation.get_text_formatters(), bits_per = TagWriter.bits_per, i, f; // rect this.bit_stream.add(pos[0], this.bits_per.width); // x this.bit_stream.add(pos[1], this.bits_per.height); // y this.bit_stream.add(size[0], this.bits_per.width); // width this.bit_stream.add(size[1], this.bits_per.height); // height // font/bold/italic if (!this.is_default("font", annotation)) { this.bit_stream.add(annotation.get_font(), bits_per.font); // font this.bit_stream.add(annotation.get_font_bold() ? 1 : 0, bits_per.font_bold); // bold this.bit_stream.add(annotation.get_font_italic() ? 1 : 0, bits_per.font_italic); // italic } // font size if (!this.is_default("font_size", annotation)) { this.bit_stream.add(annotation.get_font_size(), bits_per.font_size); // font size } // text align if (!this.is_default("align", annotation)) { this.bit_stream.add(annotation.get_align(), bits_per.align); // align } // color / decoration if (!this.is_default("color_deco", annotation)) { this.bit_stream.add(annotation.get_color(0), bits_per.color); // color this.bit_stream.add(annotation.get_color(1), bits_per.color); // color alt this.bit_stream.add(annotation.get_color(2), bits_per.color); // color bg this.bit_stream.add(annotation.get_text_decoration(), bits_per.text_decoration); // text decoration } // spacing if (!this.is_default("spacing", annotation)) { this.bit_stream.add(annotation.get_text_char_spacing(), bits_per.line_spacing); // line spacing this.bit_stream.add(annotation.get_text_line_spacing(), bits_per.char_spacing); // char spacing } // Formatters this.bit_stream.add(formatters.length, this.bits_per.formatter_count); for (i = 0; i < formatters.length; ++i) { f = formatters[i]; this.bit_stream.add(f[0], bits_per.formatter_type); // formatter this.bit_stream.add(f[1], this.bits_per.formatter_position); // position if (f[0] === Annotation.TextFormattingColor) { this.bit_stream.add(f[2], bits_per.color_index); // color index this.bit_stream.add(f[3], bits_per.color); // color } else if (f[0] === Annotation.TextFormattingDecoration) { this.bit_stream.add(f[2], bits_per.text_decoration); // decoration } } }, is_default: function (key, annotation) { var formatter = TagWriter.defaults_formatters[key], value = formatter[1](this.defaults[key], formatter[0](annotation)); this.bit_stream.add(value ? 1 : 0, 1); return value; }, }; TagWriter.get_all_post_references = function (annotations, suffix_text) { var posts = [], post_map = {}, re_pattern = />>([1-9][0-9]*)/g, i, t; for (i = 0; i < annotations.length; ++i) { if ((t = annotations[i].get_text_before()) !== null) { TagWriter.get_post_references(t, re_pattern, posts, post_map); } TagWriter.get_post_references(annotations[i].get_text(), re_pattern, posts, post_map); } if (suffix_text !== null) { TagWriter.get_post_references(suffix_text, re_pattern, posts, post_map); } return posts; }; TagWriter.get_post_references = function (text, re_pattern, posts, post_map) { var m; while ((m = re_pattern.exec(text)) !== null) { m = m[1]; if (!(m in post_map)) { post_map[m] = true; posts.push(m); } } }; TagWriter.get_longest_same_end = function (value, all_values) { var length = 0, i, j, k, v, j_end; for (i = 0; i < all_values.length; ++i) { v = all_values[i]; if (value === v) continue; j_end = v.length - Math.min(v.length, value.length); j = v.length - 1; k = value.length - 1; while (j >= j_end && v[j] === value[k]) { --j; --k; } j = (v.length - 1) - j; if (j > length) length = j; } return length; }; TagWriter.find_most_common = function (array, formatter, compare_same) { var best = [ formatter(array[0]), 1 ], values = [ best ], i, j, f, v; for (i = 1; i < array.length; ++i) { f = formatter(array[i]); for (j = 0; j < values.length; ++j) { v = values[j]; if (compare_same(f, v[0])) { if (++v[1] > best[1]) best = v; break; } } if (j >= values.length) { values.push([ f , 1 ]); } } return best[0]; }; TagWriter.compare_same_arrays = function (a1, a2) { for (var i = 0; i < a1.length; ++i) { if (a1[i] !== a2[i]) return false; } return true; }; TagWriter.compare_same_values = function (a1, a2) { return (a1 === a2); }; TagWriter.log2bits = function (v) { if (v < 1) return 0; return Math.floor(Math.log(v) / TagWriter.ln2) + 1; }; TagWriter.ln2 = Math.log(2); TagWriter.version = 1; TagWriter.version_min = 1; TagWriter.bits_per = { version: 4, annotation_count: 6, image_mode: 2, post_number_bits: 5, width_bits: 4, height_bits: 4, formatter_count_bits: 3, formatter_position_bits: 5, font: 2, font_size: 5, font_bold: 1, font_italic: 1, text_decoration: 2, align: 6, color: 5, formatter_type: 3, color_index: 2, line_spacing: 4, char_spacing: 4, }; TagWriter.defaults_formatters = { font: [ function (annotation) { return [ annotation.get_font(), annotation.get_font_bold(), annotation.get_font_italic() ]; }, TagWriter.compare_same_arrays ], color_deco: [ function (annotation) { return [ annotation.get_color(0), annotation.get_color(1), annotation.get_color(2), annotation.get_text_decoration() ]; }, TagWriter.compare_same_arrays ], font_size: [ function (annotation) { return annotation.get_font_size(); }, TagWriter.compare_same_values ], align: [ function (annotation) { return annotation.get_align(); }, TagWriter.compare_same_values ], spacing: [ function (annotation) { return [ annotation.get_text_line_spacing(), annotation.get_text_char_spacing() ]; }, TagWriter.compare_same_arrays ], }; // Annotation tag reader var TagReader = function () { this.annotation_count = 0; this.image = { mode: null, post_number_suffix: -1, post_number_suffix_digits: 0, width: 0, height: 0, }; this.bits_per = { width: 0, height: 0, formatter_count: 0, formatter_position: 0, }; this.annotations = null; this.tag = null; this.bit_stream = null; this.defaults = null; this.error_message = null; }; TagReader.prototype = { constructor: TagReader, read: function (tag, header_validate) { this.bit_stream = new BitStream(); this.bit_stream.decode(BitStream.re_alphabet.exec(tag)[0]); // Header if (!this.read_header()) return false; if (header_validate && !header_validate(this)) return this.error("Invalid header"); // Annotations this.annotations = []; if (this.annotation_count > 0) { if (this.bit_stream.eof()) return this.error("Invalid stream"); // Defaults if (!this.read_annotations_header()) return false; // End of stream if (this.bit_stream.eof()) return this.error("Invalid stream"); // Annotations for (var i = 0; true; ) { if (!this.read_annotation()) return false; if (++i >= this.annotation_count) break; if (this.bit_stream.eof()) return this.error("Invalid stream"); } } // Get crc if (this.bit_stream.remaining() < 32) return this.error("Missing CRC"); var crc_true = crc32(this.bit_stream.to_encoded_string()), crc; crc = (this.bit_stream.get(32) >>> 0); if (crc_true !== crc) return this.error("CRC invalid"); // Okay return true; }, read_header: function () { var bits_per = TagWriter.bits_per, v, v2; // Version v = this.bit_stream.get(bits_per.version); if (v < TagWriter.version_min || v > TagWriter.version) return this.error("Invalid version"); // Number of annotations this.annotation_count = this.bit_stream.get(bits_per.annotation_count); // Image mode v = this.bit_stream.get(bits_per.image_mode); if (v === 0) { this.image.mode = "file"; } else if (v === 1) { // Post this.image.mode = "post"; v = this.bit_stream.get(bits_per.post_number_bits); if (v > 0) { // Post number suffix this.image.post_number_suffix = this.bit_stream.get(v); this.image.post_number_suffix_digits = Math.floor(Math.log(v) / Math.log(10)); } } else if (v === 2) { this.image.mode = "url"; } else if (v > 2) { return this.error("Invalid image mode"); } // Resolution v = this.bit_stream.get(bits_per.width_bits); v2 = this.bit_stream.get(bits_per.height_bits); this.bits_per.width = v; this.bits_per.height = v2; this.image.width = this.bit_stream.get(v); this.image.height = this.bit_stream.get(v2); return true; }, read_annotations_header: function () { var bits_per = TagWriter.bits_per; this.defaults = new Annotation.State(); this.defaults.font = this.bit_stream.get(bits_per.font); // font if (this.defaults.font >= Annotation.fonts.length) this.defaults.font = 0; this.defaults.bold = (this.bit_stream.get(bits_per.font_bold) === 1); // bold this.defaults.italic = (this.bit_stream.get(bits_per.font_italic) === 1); // italic this.defaults.font_size = this.bit_stream.get(bits_per.font_size); // font size this.defaults.align = this.bit_stream.get(bits_per.align); // align this.defaults.colors[0] = this.bit_stream.get(bits_per.color); // color[0] this.defaults.colors[1] = this.bit_stream.get(bits_per.color); // color[1] this.defaults.colors[2] = this.bit_stream.get(bits_per.color); // color[2] this.defaults.text_decoration = this.bit_stream.get(bits_per.text_decoration); // text_decoration this.defaults.line_spacing = this.bit_stream.get(bits_per.line_spacing); // line spacing this.defaults.char_spacing = this.bit_stream.get(bits_per.char_spacing); // char spacing this.bits_per.formatter_count = this.bit_stream.get(bits_per.formatter_count_bits); // bits per formatter count number this.bits_per.formatter_position = this.bit_stream.get(bits_per.formatter_position_bits); // bits per formatter position number return true; }, read_annotation: function () { var bits_per = TagWriter.bits_per, annotation = new Annotation.State(), formatter_count, i, f; // rect annotation.rect.x = this.bit_stream.get(this.bits_per.width); annotation.rect.y = this.bit_stream.get(this.bits_per.height); annotation.rect.width = this.bit_stream.get(this.bits_per.width); annotation.rect.height = this.bit_stream.get(this.bits_per.height); if (annotation.rect.x > this.image.width) { annotation.rect.x = this.image.width; annotation.rect.width = 0; } else if (annotation.rect.x > this.image.width) { annotation.rect.width = this.image.width - annotation.rect.x; } if (annotation.rect.y > this.image.height) { annotation.rect.y = this.image.height; annotation.rect.height = 0; } else if (annotation.rect.y > this.image.height) { annotation.rect.height = this.image.height - annotation.rect.y; } // font/bold/italic if (this.is_default()) { // Default annotation.font = this.defaults.font; annotation.bold = this.defaults.bold; annotation.italic = this.defaults.italic; } else { annotation.font = this.bit_stream.get(bits_per.font); annotation.bold = (this.bit_stream.get(bits_per.font_bold) === 1); annotation.italic = (this.bit_stream.get(bits_per.font_italic) === 1); if (annotation.font >= Annotation.fonts.length) annotation.font = 0; } // font size if (this.is_default()) { // Default annotation.font_size = this.defaults.font_size; } else { annotation.font_size = this.bit_stream.get(bits_per.font_size); } // text align if (this.is_default()) { // Default annotation.align = this.defaults.align; } else { annotation.align = this.bit_stream.get(bits_per.align); } // color / decoration if (this.is_default()) { // Default annotation.colors[0] = this.defaults.colors[0]; annotation.colors[1] = this.defaults.colors[1]; annotation.colors[2] = this.defaults.colors[2]; annotation.text_decoration = this.defaults.text_decoration; } else { annotation.colors[0] = this.bit_stream.get(bits_per.color); annotation.colors[1] = this.bit_stream.get(bits_per.color); annotation.colors[2] = this.bit_stream.get(bits_per.color); annotation.text_decoration = this.bit_stream.get(bits_per.text_decoration); } // spacing if (this.is_default()) { // Default annotation.line_spacing = this.defaults.line_spacing; annotation.char_spacing = this.defaults.char_spacing; } else { annotation.line_spacing = this.bit_stream.get(bits_per.line_spacing); annotation.char_spacing = this.bit_stream.get(bits_per.char_spacing); } // Formatters formatter_count = this.bit_stream.get(this.bits_per.formatter_count); for (i = 0; i < formatter_count; ++i) { f = []; f.push(this.bit_stream.get(bits_per.formatter_type)); f.push(this.bit_stream.get(this.bits_per.formatter_position)); if (f[0] === Annotation.TextFormattingColor) { f.push(this.bit_stream.get(bits_per.color_index)); f.push(this.bit_stream.get(bits_per.color)); } else if (f[0] === Annotation.TextFormattingDecoration) { f.push(this.bit_stream.get(bits_per.text_decoration)); } annotation.formatters.push(f); } // Done this.annotations.push(annotation); return true; }, is_default: function () { return (this.bit_stream.get(1) === 1); }, error: function (message) { this.defaults = null; this.annotations = null; this.error_message = message; return false; }, clear: function () { this.defaults = null; this.annotations = null; this.bit_stream = null; this.error_message = null; }, get_error: function () { return this.error_message; }, }; // Static functions Annotator.create_image_overlay = function (image_node, annotation_data) { new Standalone(image_node, annotation_data); }; Annotator.check_url = function () { var hash = window.location.href, p = hash.indexOf("#"), ret = null, re, m, i, tag, lines, tr; if (p < 0) return null; hash = hash.substr(p); re = /#!?iex:a\/([^?#]*)(?:\?([^#]*))?/g; while ((m = re.exec(hash)) !== null) { tag = m[1]; if (m[2] === undefined) { lines = null; } else { lines = m[2].split("&"); for (i = 0; i < lines.length; ++i) { lines[i] = decodeURIComponent(lines[i]); } } tr = new TagReader(); if (lines.length !== null && tr.read(tag) && tr.annotations.length > 0) { if (ret === null) { ret = new AnnotationData(tr); } else { ret.datas.push(tr); } if (tr.annotations.length > lines.length) { tr.annotations.splice(lines.length, tr.annotations.length - lines.length); } for (i = 0; i < tr.annotations.length; ++i) { tr.annotations[i].text = lines[i]; } } } return ret; }; Annotator.process_tag = function (tag, header_validate) { var tr = new TagReader(); if (tr.read(tag, header_validate)) { return tr; } return null; }; Annotator.generate_tag = function (image, annotations, suffix_text) { return (new TagWriter()).write(image, annotations, suffix_text); }; return Annotator; })(); // Quick reply controller var QuickReplyController = (function () { var QuickReplyController = function () { api.on("quick_reply_add", on_quick_reply_add.bind(this)); api.on("quick_reply_remove", on_quick_reply_remove.bind(this)); api.on("quick_reply_show", on_quick_reply_show.bind(this)); api.on("quick_reply_hide", on_quick_reply_hide.bind(this)); document.addEventListener("QRFile", on_file_4chanx_poll.bind(this), false); document.addEventListener("QRPostSuccessful", on_file_post.bind(this), false); this.quick_reply_container = null; this.quick_reply_events = []; this.quick_reply_observers = []; this.quick_reply_file_input = null; this.quick_reply_comment = null; this.quick_reply_comment_text = ""; this.quick_reply_comment_length = 0; this.quick_reply_comment_references = []; this.quick_reply_poll_interval = null; this.quick_reply_poll_input_timer = null; this.nodes = { extra_container: null, content_container: null, enabled_checkbox: null, input_file_button: null, input_url_button: null, input_post_number: null, error_message: null, fallback_file_input: null, }; this.node_events = []; this.enabled = false; this.file = null; this.file_url = null; this.can_directly_use_files = false; this.file_poll_error_timeout = null; this.editor = null; this.force_change_to_file = false; this.file_n_submit = null; this.mode = "none"; this.custom_url = null; this.post_reference_number = null; this.post_reference_number_image = null; }; var on_quick_reply_add = function (node) { this.quick_reply_container = node; setup_qr_nodes.call(this, node); setup_qr_events.call(this, node); if (this.editor !== null) { this.editor.set_qr_container(node); } }; var on_quick_reply_remove = function (node) { remove_qr_events.call(this); remove_qr_nodes.call(this); if (this.editor !== null) { this.editor.set_qr_container(null); } }; var on_quick_reply_show = function (node) { this.set_enabled(settings.values.annotations.editor_always_enable); poll_file_4chanx.call(this); if (this.quick_reply_comment !== null) { if (this.quick_reply_poll_interval !== null) { clearInterval(this.quick_reply_poll_interval); } this.quick_reply_poll_interval = setInterval(on_comment_value_poll.bind(this), 2000); on_comment_value_poll.call(this); } }; var on_quick_reply_hide = function (node) { this.set_enabled(false); if (this.quick_reply_poll_interval !== null) { clearInterval(this.quick_reply_poll_interval); this.quick_reply_poll_interval = null; } if (this.quick_reply_poll_input_timer !== null) { clearTimeout(this.quick_reply_poll_input_timer); this.quick_reply_poll_input_timer = null; } this.quick_reply_comment_length = 0; this.quick_reply_comment_text = ""; update_available_post_references.call(this, []); }; var on_file_post = function (event) { this.set_enabled(false); this.clear(); }; var on_file_4chanx_poll = function (event) { var file = null; if (event.detail && event.detail instanceof File) { file = event.detail; } change_file.call(this, file, false); if ( file === null && this.quick_reply_file_n_submit !== null && style.has_class(this.quick_reply_file_n_submit, "has-file") ) { setTimeout(on_file_poll_check_failure.bind(this), 10); } else { // Clear poll error timeout if (this.file_poll_error_timeout !== null) { clearTimeout(this.file_poll_error_timeout); this.file_poll_error_timeout = null; } } }; var on_file_poll_check_failure = function () { if (style.has_class(this.quick_reply_file_n_submit, "has-file")) { // Clear poll error timeout if (this.file_poll_error_timeout === null) { on_file_poll_failure.call(this, null); } } else { // Clear poll error timeout if (this.file_poll_error_timeout !== null) { clearTimeout(this.file_poll_error_timeout); this.file_poll_error_timeout = null; } } }; var on_file_poll_failure = function () { this.file_poll_error_timeout = null; show_error_message.call(this); }; var on_enabled_check_change = function (event) { this.set_enabled(this.nodes.enabled_checkbox.checked); return stop_event(event); }; var on_input_file_button_click = function (event) { // If there's an error, redirect this to the secondary file input if (this.mode !== "file" && this.mode !== "none" && this.file !== null) { // Switch mode to file set_mode.call(this, "file"); } else if (this.quick_reply_file_input !== null && !is_showing_error_message.call(this)) { // Force change file this.force_change_to_file = true; var self = this, first = true; var click_fn = function (event) { if (event.target === self.quick_reply_file_input && first) { first = false; return; } self.force_change_to_file = false; document.removeEventListener("click", click_fn, true); }; document.addEventListener("click", click_fn, true); // this is to account for clicking "cancel" // Activate this.quick_reply_file_input.click(); } else { // Click custom file this.nodes.fallback_file_input.click(); } }; var on_input_url_button_click = function (event) { var v = null; try { v = prompt("Enter a URL:", this.custom_url || ""); if (v === null) return; } catch (e) { } set_custom_url.call(this, v ? v : null); }; var on_input_post_number_change = function (event) { // Must refer to a valid post with a valid image var v = this.nodes.input_post_number.options[this.nodes.input_post_number.selectedIndex]; v = v ? parseInt(v.value, 10) || null : null; set_post_reference.call(this, v); }; var on_file_n_submit_attr_change = function (records) { var i; for (i = 0; i < records.length; ++i) { if (records[i].attributeName == "class") { // "has-file" class probably changed if (style.has_class(records[i].target, "has-file")) { poll_file_4chanx_with_errors.call(this); } else { // Set file to null change_file.call(this, null, false); } break; } } }; var on_qr_file_change = function (node, event) { if (this.can_directly_use_files || (node.files && node.files.length > 0)) { // Use file directly this.can_directly_use_files = true; change_file.call(this, (node.files.length > 0 ? node.files[0] : null), true); } else { poll_file_4chanx_with_errors.call(this); } }; var on_qr_file_click = function (node, event) { if (event.shiftKey && event.which === 1) { setTimeout(update_qr_input_file_events.bind(this, this.quick_reply_container, node), 10); } }; var on_input_file_fallback_change = function (event) { var file = null; if (this.nodes.fallback_file_input.files.length > 0) { file = this.nodes.fallback_file_input.files[0]; } this.nodes.fallback_file_input.value = null; change_file.call(this, file, true); }; var on_comment_change = function (event) { if (this.quick_reply_poll_input_timer !== null) { clearTimeout(this.quick_reply_poll_input_timer); this.quick_reply_poll_input_timer = null; } comment_value_update.call(this); update_editor_annotations.call(this); }; var on_comment_input = function (event) { if (this.quick_reply_poll_input_timer !== null) { clearTimeout(this.quick_reply_poll_input_timer); } var self = this; this.quick_reply_poll_input_timer = setTimeout(function () { self.quick_reply_poll_input_timer = null; comment_value_update.call(self); }, 200); }; var on_comment_value_poll = function () { if ( this.quick_reply_poll_input_timer === null && this.quick_reply_comment.value.length !== this.quick_reply_comment_length ) { comment_value_update.call(this); update_editor_annotations.call(this); } }; var update_qr_input_file_events = function (qr_container, old_node) { if (old_node !== null) { for (var i = 0; i < this.quick_reply_events.length; ++i) { if (this.quick_reply_events[i][0] === old_node) { this.quick_reply_events.splice(i, 1); continue; } } // Nullify change_file.call(this, null, true); } // Find new var n; if ((n = qr_container.querySelector("input[type=file]")) !== null) { // Watch for changes this.quick_reply_file_input = n; add_event_listener(this.quick_reply_events, n, "change", on_qr_file_change.bind(this, n), false); if (n.getAttribute("id") === "qrFile") { add_event_listener(this.quick_reply_events, n, "click", on_qr_file_click.bind(this, n), false); } } }; var poll_file_4chanx = function () { document.dispatchEvent(new CustomEvent("QRGetFile", { bubbles: true, detail: null })); }; var poll_file_4chanx_with_errors = function () { if (this.file_poll_error_timeout !== null) { clearTimeout(this.file_poll_error_timeout); } this.file_poll_error_timeout = setTimeout(on_file_poll_failure.bind(this), 1000); poll_file_4chanx.call(this); }; var comment_value_update = function () { if (this.quick_reply_container !== null) { this.quick_reply_comment_length = this.quick_reply_comment.value.length; this.quick_reply_comment_text = this.quick_reply_comment.value; var re_number_matcher = />>([1-9][0-9]*)/g, references = [], already_used = {}, number, number_key, image_info, m, pc; while ((m = re_number_matcher.exec(this.quick_reply_comment_text)) !== null) { number = parseInt(m[1], 10); number_key = number.toString(); if (number_key in already_used) continue; if ((pc = api.get_post_container_from_post_number(number)) !== null) { already_used[number_key] = true; image_info = api.post_get_file_nodes(pc); references.push([ number, (image_info !== null && image_info.link !== null) ]); } } update_available_post_references.call(this, references); } else { this.quick_reply_comment_length = 0; this.quick_reply_comment_text = ""; update_available_post_references.call(this, []); } }; var change_file = function (file, forced) { if (this.file === null) { hide_error_message.call(this); if (file !== null) { set_file.call(this, file); } // else, both null } else { if ( forced || file === null || this.file.size != file.size || this.file.type != file.type || this.file.name != file.name || ( file.lastModifiedDate instanceof Date && this.file.lastModifiedDate instanceof Date && this.file.lastModifiedDate.getTime() != file.lastModifiedDate.getTime() ) ) { hide_error_message.call(this); set_file.call(this, file); } } }; var setup_qr_nodes = function (qr_container) { this.quick_reply_container = qr_container; var chan4x = (qr_container.getAttribute("id") == "qr"); // Setup DOM nodes var n1, n2, n3, n4, n5, n6; n1 = $.div("iex_quick_reply_extra" + (chan4x ? " iex_quick_reply_extra_4chanx" : "")); this.nodes.extra_container = n1; n2 = $.div("iex_quick_reply_label_table"); n1.appendChild(n2); n3 = $.div("iex_quick_reply_label_row"); n2.appendChild(n3); n4 = $.div("iex_quick_reply_label_cell"); n3.appendChild(n4); n5 = $.label("iex_quick_reply_label"); n4.appendChild(n5); n6 = $.input.check("iex_quick_reply_label_check"); n5.appendChild(n6); this.nodes.enabled_checkbox = n6; n6 = $.span("iex_quick_reply_label_text"); n6.textContent = "iex image annotations"; n5.appendChild(n6); n4 = $.div("iex_quick_reply_label_cell"); n3.appendChild(n4); n5 = $.span("iex_quick_reply_help_link_container"); n5.appendChild($.text("[")); n6 = $.a("iex_quick_reply_help_link"); n6.setAttribute("href", "https://dnsev.github.io/iex/annotations/help.html"); n6.setAttribute("rel", "nofollow noreferrer"); n6.setAttribute("target", "_blank"); n6.textContent = "Help"; n5.appendChild(n6); n5.appendChild($.text("]")); n4.appendChild(n5); n2 = $.div("iex_quick_reply_content"); n1.appendChild(n2); this.nodes.content_container = n2; n3 = $.div("iex_quick_reply_info"); n3.textContent = "Select an image to annotate:"; n2.appendChild(n3); n3 = $.div("iex_quick_reply_source_selection"); n4 = $.span("iex_quick_reply_source_selection_text"); n4.textContent = "From "; n3.appendChild(n4); n4 = $.input.button("iex_quick_reply_source_selection_file"); n4.value = "File"; this.nodes.input_file_button = n4; n3.appendChild(n4); n4 = $.span("iex_quick_reply_source_selection_text"); n4.textContent = " or "; n3.appendChild(n4); n4 = $.input.button("iex_quick_reply_source_selection_url"); n4.value = "URL"; this.nodes.input_url_button = n4; n3.appendChild(n4); n4 = $.span("iex_quick_reply_source_selection_text"); n4.textContent = " or reference post "; n3.appendChild(n4); n4 = $.node("select", "iex_quick_reply_source_selection_post"); n4.value = ""; this.nodes.input_post_number = n4; n5 = $.node("option", "iex_quick_reply_source_selection_post_option_default"); n5.value = ""; n5.textContent = ">>number"; n4.appendChild(n5); n3.appendChild(n4); n2.appendChild(n3); n3 = $.div("iex_quick_reply_error"); n3.appendChild($.text("Warning: image not detected properly.")); n3.appendChild($("br")); n3.appendChild($.text("Click \"Image\" to select for editing")); n3.setAttribute( "title", "This typically occurs due to an incompatability between scripts.\n" + "For example, appchan-x does not support sharing the post file.\n" + "Chrome may have similar issues." ); this.nodes.error_message = n3; n2.appendChild(n3); n3 = $.input.file("iex_quick_reply_file_input"); this.nodes.fallback_file_input = n3; n2.appendChild(n3); // Events add_event_listener(this.node_events, this.nodes.enabled_checkbox, "change", on_enabled_check_change.bind(this), false); add_event_listener(this.node_events, this.nodes.input_file_button, "click", on_input_file_button_click.bind(this), false); add_event_listener(this.node_events, this.nodes.input_url_button, "click", on_input_url_button_click.bind(this), false); add_event_listener(this.node_events, this.nodes.input_post_number, "change", on_input_post_number_change.bind(this), false); add_event_listener(this.node_events, this.nodes.fallback_file_input, "change", on_input_file_fallback_change.bind(this), false); // Add to QR qr_container.appendChild(n1); }; var setup_qr_events = function (qr_container) { var n, ob; if ((n = qr_container.querySelector("#file-n-submit")) !== null) { // Watch for class changes this.quick_reply_file_n_submit = n; ob = new MutationObserver(on_file_n_submit_attr_change.bind(this)); ob.observe(n, { attributes: true }); this.quick_reply_observers.push(ob); } if ((n = qr_container.querySelector("textarea[name=com],textarea[data-name=com]")) !== null) { // Watch for changes this.quick_reply_comment = n; this.quick_reply_comment_length = 0; add_event_listener(this.quick_reply_events, n, "change", on_comment_change.bind(this), false); add_event_listener(this.quick_reply_events, n, "input", on_comment_input.bind(this), false); } update_qr_input_file_events.call(this, qr_container, null); }; var remove_qr_nodes = function () { this.quick_reply_container = null; if (this.nodes.extra_container.parentNode !== null) { this.nodes.extra_container.parentNode.removeChild(this.nodes.extra_container); } for (var k in this.nodes) { this.nodes[k] = null; } }; var remove_qr_events = function () { for (var i = 0; i < this.quick_reply_observers.length; ++i) { this.quick_reply_observers[i].disconnect(); } this.quick_reply_observers = []; remove_event_listeners(this.quick_reply_events); remove_event_listeners(this.node_events); this.quick_reply_events = []; this.node_events = []; this.quick_reply_file_input = null; this.quick_reply_file_n_submit = null; this.quick_reply_comment = null; this.quick_reply_comment_text = ""; this.quick_reply_comment_length = 0; this.quick_reply_comment_references = []; }; var is_showing_error_message = function () { return (this.quick_reply_container !== null && style.has_class(this.nodes.error_message, "iex_quick_reply_error_visible")); }; var show_error_message = function () { if (this.quick_reply_container !== null) { style.add_class(this.nodes.error_message, "iex_quick_reply_error_visible"); } }; var hide_error_message = function () { if (this.quick_reply_container !== null) { style.remove_class(this.nodes.error_message, "iex_quick_reply_error_visible"); } }; var update_editor = function () { if (this.mode === "file") { this.editor.set_file(this.file, this.file_url, this.mode); return; } else if (this.mode === "post") { if (this.post_reference_number_image !== null) { this.editor.set_file(null, this.post_reference_number_image, this.mode, this.post_reference_number); return; } } else if (this.mode === "url") { this.editor.set_file(null, this.custom_url, this.mode); return; } // "none" or invalid this.editor.hide(); this.editor.set_file(null, null, "none"); }; var update_editor_annotations = function () { if (this.editor !== null) { this.editor.load_annotations_from_text(this.quick_reply_comment_text); } }; var update_available_post_references = function (references) { // Only update this if the references are different var i, j; if (this.quick_reply_comment_references.length === references.length) { if (references.length === 0) return; // same var r1, r2; i = 0; while (true) { r1 = this.quick_reply_comment_references[i]; r2 = references[i]; if (r1[0] !== r2[0] || r1[1] !== r2[1]) break; if (++i >= references.length) return; // same } } // Update this.quick_reply_comment_references = references; var pn_select = this.nodes.input_post_number, selected_index = 0, first_selected_index = 0, first_selected = null, n, n2; // Remove j = 0; for (n = pn_select.firstChild; n !== null; n = n2) { n2 = n.nextSibling; if (n.value !== "") { pn_select.removeChild(n); } else { ++j; } } // Add for (i = 0; i < references.length; ++i) { n = references[i][0]; if (n === this.post_reference_number) { selected_index = i + j; } n2 = $.node("option", "iex_quick_reply_source_selection_post_option"); n2.value = n.toString(); n2.textContent = ">>" + n; if (!references[i][1]) { n2.disabled = true; } else if (first_selected === null) { first_selected = n2; first_selected_index = i; } pn_select.appendChild(n2); } // Select pn_select.selectedIndex = selected_index; // Select based on mode if (this.enabled && this.mode === "none" && first_selected !== null) { pn_select.selectedIndex = first_selected_index + j; set_post_reference.call(this, references[first_selected_index][0]); } else if (selected_index === 0) { // Deselect auto_detect_mode.call(this); } }; var set_file = function (new_file) { if (this.file_url !== null) { window.URL.revokeObjectURL(this.file_url); this.file_url = null; } this.file = new_file; if (this.file === null) { if (this.mode === "file") set_mode.call(this, "none"); } else { // Create url this.file_url = window.URL.createObjectURL(this.file); if ((this.mode === "none" && this.enabled) || this.force_change_to_file) { this.force_change_to_file = false; set_mode.call(this, "file"); } } }; var set_custom_url = function (url) { hide_error_message.call(this); if (url === null) { this.custom_url = null; auto_detect_mode.call(this); } else { this.custom_url = url; if (this.enabled) { if (!set_mode.call(this, "url")) { update_editor.call(this); } } } }; var set_post_reference = function (number) { hide_error_message.call(this); if (number === null) { this.post_reference_number = null; auto_detect_mode.call(this); } else { this.post_reference_number = number; // Get the image var pc, image_info, href; if ( (pc = api.get_post_container_from_post_number(this.post_reference_number)) !== null && (image_info = api.post_get_file_nodes(pc)) !== null && image_info.link !== null && (href = image_info.link.getAttribute("href")) ) { this.post_reference_number_image = href; } else { this.post_reference_number_image = null; } if (this.enabled) { if (!set_mode.call(this, "post")) { update_editor.call(this); } } } }; var set_mode = function (mode) { if (this.mode === mode) return false; this.mode = mode; if (mode === "file") { style.add_class(this.nodes.input_file_button, "iex_quick_reply_source_selection_file_selected"); } else { style.remove_class(this.nodes.input_file_button, "iex_quick_reply_source_selection_file_selected"); } if (mode === "post") { style.add_class(this.nodes.input_post_number, "iex_quick_reply_source_selection_post_selected"); } else { style.remove_class(this.nodes.input_post_number, "iex_quick_reply_source_selection_post_selected"); this.post_reference_number = null; this.post_reference_number_image = null; this.nodes.input_post_number.selectedIndex = 0; } if (mode === "url") { style.add_class(this.nodes.input_url_button, "iex_quick_reply_source_selection_url_selected"); } else { style.remove_class(this.nodes.input_url_button, "iex_quick_reply_source_selection_url_selected"); this.custom_url = null; } // Editor if (mode === "none") { // Hide if (this.editor !== null) { update_editor.call(this); this.editor.hide(); } } else { // Show if (this.enabled) { if (this.editor === null) { this.editor = new AnnotationEditor(this.quick_reply_container, this); } update_editor.call(this); if (!this.editor.is_visible()) { this.editor.show(); update_editor_annotations.call(this); } } } return true; }; var auto_detect_mode = function () { if (this.file !== null) { if (this.mode !== "file") { set_mode.call(this, "file"); return; } } if (this.mode !== "post") { var pn_select = this.nodes.input_post_number, opts = pn_select.options, i, n; for (i = 0; i < opts.length; ++i) { if (!opts[i].disabled && opts[i].value && (n = parseInt(opts[i].value, 10))) { pn_select.selectedIndex = i; set_post_reference.call(this, n); return; } } } set_mode.call(this, "none"); }; QuickReplyController.prototype = { constructor: QuickReplyController, is_enabled: function () { return this.enabled; }, set_enabled: function (enabled) { if (this.quick_reply_container === null || this.enabled === enabled) { if (this.nodes.enabled_checkbox !== null) { this.nodes.enabled_checkbox.checked = this.enabled; } return; } this.enabled = enabled; this.nodes.enabled_checkbox.checked = this.enabled; if (this.enabled) { style.add_class(this.nodes.extra_container, "iex_quick_reply_extra_enabled"); poll_file_4chanx.call(this); auto_detect_mode.call(this); } else { style.remove_class(this.nodes.extra_container, "iex_quick_reply_extra_enabled"); if (this.editor !== null) { this.editor.hide(); } set_mode.call(this, "none"); } }, get_text: function () { return this.quick_reply_comment_text; }, set_text: function (text) { this.quick_reply_comment_length = text.length; this.quick_reply_comment_text = text; this.quick_reply_comment.value = text; }, clear: function () { // Clear statuses set_mode.call(this, "none"); }, }; return QuickReplyController; })(); // Image annotation editor var AnnotationEditor = (function () { var AnnotationEditor = function (qr_container, qr_controller) { this.qr_controller = qr_controller; this.style_transform = style.get_prefixed_style("transform")[0]; this.visible = false; this.qr_container = null; this.qr_resize_timer = null; this.qr_resized_during_timer = false; this.qr_event_list = []; this.move_event_list = []; this.minimum_ratio_for_wide = 4.0 / 3.0; this.paddings = { left: 0, top: 0, right: 0, bottom: 0, }; this.image = { url: null, mode: null, extra: null, size: { width: 0, height: 0, }, }; this.cpreview = null; this.cpreview_zoom = 1; this.nodes = { container: null, image: null, image_error: null, edit_container: null, edit_container_empty: null, edit_color_buttons: [], edit_color_selected: -1, edit_font_selector: null, edit_font_size_selector: null, edit_font_bold_check: null, edit_font_italic_check: null, annotation_container: null, annotation_scroll_h: null, annotation_scroll_v: null, annotation_scroll_h_size: null, annotation_scroll_v_size: null, }; this.image_events = []; this.image_load_poll = null; this.preview_scale = 1.0; this.preview_move_events = []; this.preview_move_speed = [0, 0]; this.preview_move_origin = [0, 0]; this.preview_move_offset = [0, 0]; this.preview_scroll_range = [0, 0]; this.annotations = []; this.annotations_previous = this.annotations; this.annotation_text_suffix = null; this.annotation_text_suffix_previous = this.annotation_text_suffix; this.annotation_selected_index = -1; this.update_annotation_sizes_on_image_load = null; acquire_paddings.call(this); window.addEventListener("resize", on_window_resize.bind(this), false); this.set_qr_container(qr_container); }; var re_load_annotations = /^(?:|([\w\W]*?)(?:\r\n?|\n))[ \t]*>((?:[^>\r\n][^\r\n]*)?)(\r\n?|\n|$)/, re_load_tag = /^(?:|([\w\W]*?)(?:\r\n?|\n))[ \t]*(\[iex:a\/([^\]]*)\])\s*$/; var acquire_paddings = function () { var n = $("div"), r1, r2; n.className = "iex_annotation_editor_outer_paddings"; n.style.paddingRight = "0"; n.style.paddingBottom = "0"; n.style.display = "inline-block"; n.style.width = "0"; n.style.height = "0"; document.body.appendChild(n); r1 = style.get_object_size(n); n.style.paddingTop = "0"; n.style.paddingLeft = "0"; n.style.paddingRight = ""; n.style.paddingBottom = ""; r2 = style.get_object_size(n); document.body.removeChild(n); this.paddings.left = r1.width; this.paddings.right = r2.width; this.paddings.top = r1.height; this.paddings.bottom = r2.height; }; var create_nodes = function () { var n1, n2, n3, n4, n5; // Create n1 = $.div("iex_annotation_editor"); this.nodes.container = n1; n2 = $.div("iex_ae_t1c1"); n1.appendChild(n2); n3 = $.div("iex_ae_t1c2"); n2.appendChild(n3); n4 = $.div("iex_ae_t1c3"); n3.appendChild(n4); n5 = $.div("iex_ae_t1c4"); n4.appendChild(n5); create_nodes_preview.call(this, n5); n4 = $.div("iex_ae_t1c3"); n3.appendChild(n4); n5 = $.div("iex_ae_t1c4"); n4.appendChild(n5); create_nodes_editing.call(this, n5); // Set size document.body.appendChild(n1); update_size.call(this); }; var create_nodes_preview = function (container) { // Create preview var n1, n2, n3, event_bind; n1 = $.div("iex_ae_preview_container"); container.appendChild(n1); // Message n2 = $.div("iex_ae_preview_message"); n1.appendChild(n2); n3 = $.div("iex_ae_preview_message_inner"); n2.appendChild(n3); this.nodes.image_error = n3; // Preview this.cpreview = new ContentPreview(); this.cpreview.set_auto_size(true); this.cpreview.add_to(n1); // Image n2 = $.node("img", "iex_ae_preview_image"); n2.setAttribute("alt", ""); n2.setAttribute("title", ""); this.nodes.image = n2; this.cpreview.add_content(n2); // Overlay n2 = $.div("iex_annotation_overlay"); this.cpreview.add_content(n2); n3 = $.div("iex_annotation_container"); n2.appendChild(n3); this.nodes.annotation_container = n3; // Events n2.addEventListener("mouseover", wrap_mouseenterleave_event(this, on_container_mouseover), false); n2.addEventListener("mouseout", wrap_mouseenterleave_event(this, on_container_mouseout), false); n2.addEventListener("mousewheel", (event_bind = on_container_mousewheel.bind(this)), false); n2.addEventListener("DOMMouseScroll", event_bind, false); n2.addEventListener("mousedown", on_container_mousedown.bind(this), false); n2.addEventListener("contextmenu", on_container_contextmenu.bind(this), false); // Scroll bars n2 = $.div("iex_ae_preview_scroll_h"); n1.appendChild(n2); this.nodes.annotation_scroll_h = n2; n3 = $.div("iex_ae_preview_scroll_h_inner"); n2.appendChild(n3); this.nodes.annotation_scroll_h_size = n3; n2 = $.div("iex_ae_preview_scroll_v"); n1.appendChild(n2); this.nodes.annotation_scroll_v = n2; n3 = $.div("iex_ae_preview_scroll_v_inner"); n2.appendChild(n3); this.nodes.annotation_scroll_v_size = n3; }; var create_nodes_editing = function (container) { var n1, n2, n3, n4, n5; // Create n1 = $.div("iex_ae_t2c0"); container.appendChild(n1); n2 = $.div("iex_ae_t2c1"); n1.appendChild(n2); n3 = $.div("iex_ae_t2c2"); n2.appendChild(n3); n4 = $.div("iex_ae_t2c3"); n3.appendChild(n4); n5 = $.div("iex_ae_t2c4"); n4.appendChild(n5); create_nodes_editing_tools.call(this, n5); n4 = $.div("iex_ae_t2c3"); n3.appendChild(n4); n5 = $.div("iex_ae_t2c4"); n4.appendChild(n5); create_nodes_editing_text.call(this, n5); }; var create_nodes_editing_tools = function (container) { var n1, n2, n3, n4, n5, n6; // Create n1 = $.div("iex_ae_t3c0"); container.appendChild(n1); n2 = $.div("iex_ae_t3c1"); n1.appendChild(n2); n3 = $.div("iex_ae_t3c2"); n2.appendChild(n3); n4 = $.div("iex_ae_t3c3"); n3.appendChild(n4); // Top (colors) n5 = $.div("iex_ae_t3c4"); n4.appendChild(n5); n6 = $.div("iex_ae_t3c5"); n5.appendChild(n6); create_nodes_editing_tools_top.call(this, n6); // Top (colors) n5 = $.div("iex_ae_t3c4"); n4.appendChild(n5); n6 = $.div("iex_ae_t3c5"); n5.appendChild(n6); create_nodes_editing_tools_colors.call(this, n6); // Middle (fonts) n5 = $.div("iex_ae_t3c4"); n4.appendChild(n5); n6 = $.div("iex_ae_t3c5"); n5.appendChild(n6); create_nodes_editing_tools_fonts.call(this, n6); // Bottom (align) create_nodes_editing_tools_align.call(this, n6); }; var create_nodes_editing_tools_top = function (container) { var n1, n2, n3; n1 = $.div("iex_ae_t7c0"); this.nodes.edit_color_index_labels = []; n2 = $.span("iex_ae_t7c1 iex_ae_top_label"); n2.textContent = "Color:"; n1.appendChild(n2); n2 = $.span("iex_ae_t7c1 iex_ae_top_color_link"); n2.textContent = "Text"; n1.appendChild(n2); n2.addEventListener("click", on_color_index_select.bind(this, 0), false); this.nodes.edit_color_index_labels.push(n2); n2 = $.span("iex_ae_t7c1 iex_ae_top_color_link"); n2.appendChild($.text("Deco")); n3 = $.span("iex_ae_top_hyphen"); n3.textContent = "-"; n3.appendChild($("br")); n2.appendChild(n3); n2.appendChild($.text("ration")); n1.appendChild(n2); n2.addEventListener("click", on_color_index_select.bind(this, 1), false); this.nodes.edit_color_index_labels.push(n2); n2 = $.span("iex_ae_t7c1 iex_ae_top_color_link"); n2.appendChild($.text("Back")); n3 = $.span("iex_ae_top_hyphen"); n3.textContent = "-"; n3.appendChild($("br")); n2.appendChild(n3); n2.appendChild($.text("ground")); n1.appendChild(n2); n2.addEventListener("click", on_color_index_select.bind(this, 2), false); this.nodes.edit_color_index_labels.push(n2); container.appendChild(n1); }; var create_nodes_editing_tools_colors = function (container) { var shades = Annotation.color_shades, i, j, n1, n2, n3, n4, n5; // Create n1 = $.div("iex_ae_t4c0"); container.appendChild(n1); n2 = $.div("iex_ae_t4c1"); n1.appendChild(n2); n3 = $.div("iex_ae_t4c2"); n2.appendChild(n3); // Black and white n4 = $.div("iex_ae_t4c3"); n3.appendChild(n4); n5 = $.div("iex_ae_t4c4"); n4.appendChild(n5); n5.appendChild(create_nodes_color_selector.call(this, 0)); n5.appendChild(create_nodes_color_selector.call(this, 1)); for (i = 2; i < Annotation.colors.length; i += shades) { n4 = $.div("iex_ae_t4c3"); n3.appendChild(n4); n5 = $.div("iex_ae_t4c4"); n4.appendChild(n5); for (j = 0; j < shades; ++j) { n5.appendChild(create_nodes_color_selector.call(this, i + j)); } } }; var create_nodes_color_selector = function (color_index) { var n1, n2, n3, cls; n1 = $.div("iex_ae_color_selector iex_ae_color_selector_disabled"); n1.setAttribute("title", "Color index: " + color_index); n2 = $.div("iex_ae_color_selector_inner"); n2.style.backgroundColor = "#" + Annotation.colors[color_index]; n1.appendChild(n2); n3 = $.div("iex_ae_color_selector_border"); n2.appendChild(n3); cls = "iex_ae_color_selector_border2"; if (color_index === 1) cls += " iex_ae_color_selector_border2_alt"; n3 = $.div(cls); n2.appendChild(n3); n1.addEventListener("click", on_color_select.bind(this, color_index), false); this.nodes.edit_color_buttons.push(n1); return n1; }; var create_nodes_editing_tools_fonts = function (container) { var n1, n2, n3, n4, n5; n1 = $.div("iex_ae_t5c1"); container.appendChild(n1); n2 = $.div("iex_ae_t5c2"); n1.appendChild(n2); // Column 1 n3 = $.div("iex_ae_t5c3"); n2.appendChild(n3); n4 = $.div("iex_ae_new_annotation_button"); n3.appendChild(n4); n5 = $.div("iex_ae_new_annotation_button_inner"); n4.appendChild(n5); n5.textContent = "New Note"; n4.addEventListener("click", on_new_note_click.bind(this), false); // Column 2 n3 = $.div("iex_ae_t5c3"); n2.appendChild(n3); n4 = $.node("select", "iex_ae_selection"); n4.addEventListener("change", on_font_select.bind(this, n4), false); n4.setAttribute("title", "Font to use for the annotation"); n3.appendChild(n4); n4.disabled = true; this.nodes.edit_font_selector = n4; create_nodes_font_selection.call(this, n4); n4 = $.node("select", "iex_ae_selection"); n4.addEventListener("change", on_font_scale_select.bind(this, n4), false); n4.setAttribute("title", "Font size to use for the annotation"); n3.appendChild(n4); n4.disabled = true; this.nodes.edit_font_size_selector = n4; create_nodes_font_size_selection.call(this, n4); // Column 3 n3 = $.div("iex_ae_t5c3"); n2.appendChild(n3); n4 = $.label("iex_ae_option_label"); n3.appendChild(n4); n5 = $.input.check("iex_ae_option_checkbox iex_ae_option_checkbox_bold"); n5.addEventListener("change", on_font_bold_change.bind(this, n5), false); this.nodes.edit_font_bold_check = n5; n5.disabled = true; n4.appendChild(n5); n5 = $.span("iex_ae_option_label_text"); n5.textContent = " Bold"; n4.appendChild(n5); n4 = $.label("iex_ae_option_label"); n3.appendChild(n4); n5 = $.input.check("iex_ae_option_checkbox iex_ae_option_checkbox_italic"); n5.addEventListener("change", on_font_italic_change.bind(this, n5), false); this.nodes.edit_font_italic_check = n5; n5.disabled = true; n4.appendChild(n5); n5 = $.span("iex_ae_option_label_text"); n5.textContent = " Italic"; n4.appendChild(n5); }; var create_nodes_editing_tools_align = function (container) { var n1, n2, n3, n4; var add_opt = function (par, text, value) { var n1 = $.node("option", "iex_ae_selection_option"); n1.textContent = text; n1.value = value.toString(); par.appendChild(n1); return n1; }; n1 = $.div("iex_ae_t5c1"); container.appendChild(n1); n2 = $.div("iex_ae_t5c2"); n1.appendChild(n2); // Column 1 n3 = $.div("iex_ae_t5c3 iex_ae_t5c3_small_space"); n2.appendChild(n3); n4 = $.node("select", "iex_ae_selection"); n4.addEventListener("change", on_halign_select.bind(this, n4), false); n4.setAttribute("title", "Horizontal alignment for the annotation text"); n3.appendChild(n4); n4.disabled = true; this.nodes.edit_halign_selector = n4; add_opt(n4, "Center", Annotation.AlignCenter); add_opt(n4, "Left", Annotation.AlignLeft); add_opt(n4, "Right", Annotation.AlignRight); add_opt(n4, "Justify", Annotation.AlignJustify); // Column 2 n3 = $.div("iex_ae_t5c3 iex_ae_t5c3_small_space"); n2.appendChild(n3); n4 = $.node("select", "iex_ae_selection"); n4.addEventListener("change", on_valign_select.bind(this, n4), false); n4.setAttribute("title", "Vertical alignment for the annotation text"); n3.appendChild(n4); n4.disabled = true; this.nodes.edit_valign_selector = n4; add_opt(n4, "Middle", Annotation.AlignMiddle); add_opt(n4, "Top", Annotation.AlignTop); add_opt(n4, "Bottom", Annotation.AlignBottom); // Column 3 n3 = $.div("iex_ae_t5c3 iex_ae_t5c3_small_space"); n2.appendChild(n3); n4 = $.node("select", "iex_ae_selection"); n4.addEventListener("change", on_text_decoration_select.bind(this, n4), false); n4.setAttribute("title", "Text decoration annotation text"); n3.appendChild(n4); n4.disabled = true; this.nodes.edit_decoration_selector = n4; add_opt(n4, "Normal", Annotation.TextDecorationNone).setAttribute("title", "Decoration index: 0"); add_opt(n4, "Shadow", Annotation.TextDecorationShadow).setAttribute("title", "Decoration index: 1"); add_opt(n4, "Blur", Annotation.TextDecorationShadowBlur).setAttribute("title", "Decoration index: 2"); add_opt(n4, "Strong Blur", Annotation.TextDecorationShadowBlurStrong).setAttribute("title", "Decoration index: 3"); }; var create_nodes_font_selection = function (container) { var i, n1; for (i = 0; i < Annotation.fonts.length; ++i) { n1 = $.node("option", "iex_ae_selection_option"); n1.textContent = Annotation.fonts[i].name; n1.value = i.toString(); container.appendChild(n1); } }; var create_nodes_font_size_selection = function (container) { var size = Annotation.font_size_settings.base, size_incr = Annotation.font_size_settings.base_step, i, j, n1; for (i = 0; i < Annotation.font_size_settings.step_count; ++i) { for (j = 0; j < Annotation.font_size_settings.step_duration; ++j) { n1 = $.node("option", "iex_ae_selection_option"); n1.textContent = "Size: " + size + "px"; n1.value = (i * Annotation.font_size_settings.step_duration + j).toString(); container.appendChild(n1); size += size_incr; } size_incr *= Annotation.font_size_settings.step_multiplier; } }; var create_nodes_editing_text = function (container) { var n1, n2, n3, n4, n5; n1 = $.div("iex_ae_t6c0"); container.appendChild(n1); n2 = $.div("iex_ae_t6c1"); n1.appendChild(n2); n3 = $.div("iex_ae_t6c2"); n2.appendChild(n3); this.nodes.edit_container = n3; n4 = $.div("iex_ae_add_annotation_container_empty"); n3.appendChild(n4); this.nodes.edit_container_empty = n4; n5 = $.a("iex_ae_add_annotation_container_empty_link"); n4.appendChild(n5); n5.textContent = "Add annotation"; n5.addEventListener("click", on_new_note_empty_click.bind(this), false); n4 = $.div("iex_ae_restore_annotations_container_empty"); n3.appendChild(n4); this.nodes.edit_container_empty_restore = n4; n5 = $.a("iex_ae_restore_annotations_container_empty_link"); n4.appendChild(n5); n5.textContent = "Restore recent annotations"; n5.addEventListener("click", on_restore_annotations_click.bind(this), false); set_empty_links_visible.call(this, true); }; var remove_nodes = function () { }; var setup_qr_events = function () { var movable_sections = this.qr_container.querySelectorAll(".move,.drag,textarea"), i; // Events for (i = 0; i < movable_sections.length; ++i) { add_event_listener(this.qr_event_list, movable_sections[i], "mousedown", on_movable_mousedown.bind(this), false); } }; var remove_qr_events = function () { remove_event_listeners(this.qr_event_list); this.qr_event_list = []; }; var update_size = function () { var max_rect = style.get_document_rect(), doc_left = max_rect.left, doc_top = max_rect.top, header_rect = api.get_header_rect(), qr_rect = style.get_object_rect(this.qr_container), container = this.nodes.container, x_mid, x, y, w, h; // Subtract header if (header_rect.top <= max_rect.top && header_rect.bottom > max_rect.top) { // It's on the top max_rect.top = header_rect.bottom; } else if (header_rect.bottom >= max_rect.bottom && header_rect.top < max_rect.bottom) { // It's on the bottom max_rect.bottom = header_rect.top; } // Subtract size of quick reply if it results in a large enough container x_mid = (doc_left + max_rect.right) / 2.0; if (qr_rect.left >= x_mid + this.paddings.left) { // On the right max_rect.right = qr_rect.left; } else if (qr_rect.right <= x_mid - this.paddings.right) { // On the left max_rect.left = qr_rect.right; } // Finalize position x = (max_rect.left + this.paddings.left); y = (max_rect.top + this.paddings.top); w = (max_rect.right - this.paddings.right) - x; h = (max_rect.bottom - this.paddings.bottom) - y; x -= doc_left; y -= doc_top; if (w < 16) w = 16; if (h < 16) h = 16; // Set size container.style.left = x + "px"; container.style.top = y + "px"; container.style.width = w + "px"; container.style.height = h + "px"; // Ratio if (w / h >= this.minimum_ratio_for_wide) { style.remove_class(container, "iex_annotation_editor_vertical"); } else { style.add_class(container, "iex_annotation_editor_vertical"); } // Preview size update_preview_size.call(this); }; var update_preview_size = function (offset_x, offset_y) { var w, h, size, r; // Resize this.cpreview.update_auto_size(); size = this.cpreview.size; // Get image size w = this.image.size.width; h = this.image.size.height; // Ratios r = w / h; if ((size.width / size.height) > r) { // fit width w = size.width; h = w / r; } else { // fit height h = size.height; w = h * r; } // Zoom w *= this.cpreview_zoom; h *= this.cpreview_zoom; this.cpreview.set_size(w, h, offset_x, offset_y); // Annotation scaling this.preview_scale = 1.0; if (this.image.size.width > 0) this.preview_scale = (this.cpreview.size.width / this.image.size.width); if (Annotation.scaling_mode_px) { this.nodes.annotation_container.style.fontSize = this.preview_scale.toFixed(4) + "px"; } else { this.nodes.annotation_container.style[this.style_transform] = "scale(" + this.preview_scale.toFixed(4) + ")"; } // Scrollbars update_preview_scrollbars.call(this); show_preview_scrollbars.call(this); }; var update_preview_scrollbars = function () { var size = this.cpreview.size, display_size = this.cpreview.display, w, h; if (size.width > display_size.width + 0.001) { style.add_class(this.nodes.annotation_scroll_h, "iex_ae_preview_scroll_displayable"); w = (size.width - display_size.width) / size.width; this.nodes.annotation_scroll_h_size.style.width = ((1 - w) * 100).toFixed(4) + "%"; } else { style.remove_class(this.nodes.annotation_scroll_h, "iex_ae_preview_scroll_displayable"); w = 0; } if (size.height > display_size.height + 0.001) { style.add_class(this.nodes.annotation_scroll_v, "iex_ae_preview_scroll_displayable"); h = (size.height - display_size.height) / size.height; this.nodes.annotation_scroll_v_size.style.height = ((1 - h) * 100).toFixed(4) + "%"; } else { style.remove_class(this.nodes.annotation_scroll_v, "iex_ae_preview_scroll_displayable"); h = 0; } this.preview_scroll_range[0] = w * 100; this.preview_scroll_range[1] = h * 100; update_preview_scrollbar_positions.call(this); }; var update_preview_scrollbar_positions = function () { this.nodes.annotation_scroll_h_size.style.left = (this.cpreview.offset_x * this.preview_scroll_range[0]).toFixed(4) + "%"; this.nodes.annotation_scroll_v_size.style.top = (this.cpreview.offset_y * this.preview_scroll_range[1]).toFixed(4) + "%"; }; var show_preview_scrollbars = function () { style.add_class(this.nodes.annotation_scroll_h, "iex_ae_preview_scroll_visible"); style.add_class(this.nodes.annotation_scroll_v, "iex_ae_preview_scroll_visible"); }; var hide_preview_scrollbars = function () { style.remove_class(this.nodes.annotation_scroll_h, "iex_ae_preview_scroll_visible"); style.remove_class(this.nodes.annotation_scroll_v, "iex_ae_preview_scroll_visible"); }; var set_image = function (url, mode, extra) { // Events clear_image_events.call(this); this.image.url = url; this.image.mode = mode; this.image.extra = extra; this.image.size.width = 0; this.image.size.height = 0; var image = this.nodes.image; style.remove_class(image, "iex_ae_preview_image_visible"); style.remove_class(image, "iex_ae_preview_image_loaded"); style.remove_class(image, "iex_ae_preview_image_error"); style.remove_class(this.nodes.annotation_scroll_h, "iex_ae_preview_scroll_visible"); style.remove_class(this.nodes.annotation_scroll_v, "iex_ae_preview_scroll_visible"); image.removeAttribute("src"); if (url === null) { // Image not available this.nodes.image_error.textContent = "Image not available"; return; } this.nodes.image_error.textContent = ""; add_event_listener(this.image_events, image, "load", on_image_load.bind(this, image), false); add_event_listener(this.image_events, image, "error", on_image_error.bind(this, image), false); this.image_load_poll = setInterval(on_image_status_poll.bind(this, image), 200); update_preview_size.call(this); image.setAttribute("src", url); update_annotation_full_text.call(this, false); }; var clear_image_events = function () { if (this.image_events.length > 0) { on_image_abort.call(this, this.nodes.image); remove_event_listeners(this.image_events); this.image_events = []; if (this.image_load_poll !== null) { clearInterval(this.image_load_poll); this.image_load_poll = null; } } }; var change_zoom_level = function (new_zoom, update_offset, mx, my) { var left, top; if (update_offset) { var rect = style.get_object_rect_relative(this.cpreview.nodes.container), image_rect = style.get_object_rect_relative(this.nodes.image), scale = (this.cpreview_zoom / new_zoom), w = image_rect.width / scale, h = image_rect.height / scale, left_max = (w - rect.width) / w, top_max = (h - rect.height) / h; left = (((mx - rect.left) * (1 - scale)) + (rect.left - image_rect.left)) / image_rect.width; top = (((my - rect.top) * (1 - scale)) + (rect.top - image_rect.top)) / image_rect.height; if (left_max > 0) { if (left < 0) left = 0; else if (left > left_max) left = left_max; left /= left_max; } else { left = 0; } if (top_max > 0) { if (top < 0) top = 0; else if (top > top_max) top = top_max; top /= top_max; } else { top = 0; } } this.cpreview_zoom = new_zoom; update_preview_size.call(this, left, top); }; var set_empty_links_visible = function (visible) { var s1 = "iex_ae_add_annotation_container_empty_visible", s2 = "iex_ae_restore_annotations_container_empty_visible"; if (visible) { style.add_class(this.nodes.edit_container_empty, s1); } else { style.remove_class(this.nodes.edit_container_empty, s1); } if (visible && this.annotations_previous !== this.annotations && this.annotations_previous.length > 0) { style.add_class(this.nodes.edit_container_empty_restore, s2); } else { style.remove_class(this.nodes.edit_container_empty_restore, s2); } }; var clear_previous_annotations = function () { // Clear previous if (this.annotations_previous !== this.annotations) { for (var i = 0; i < this.annotations_previous.length; ++i) { this.annotations_previous[i].destroy(); } this.annotations_previous = this.annotations; this.annotation_text_suffix_previous = null; } }; var set_no_annotations_selected = function () { var ns = this.nodes.edit_color_buttons, i; i = this.nodes.edit_color_selected; if (i >= 0) { style.remove_class(ns[i], "iex_ae_color_selector_selected"); for (i = 0; i < ns.length; ++i) { style.add_class(ns[i], "iex_ae_color_selector_disabled"); } this.nodes.edit_color_selected = -1; } ns = this.nodes.edit_color_index_labels; for (i = 0; i < ns.length; ++i) { style.remove_class(ns[i], "iex_ae_top_color_link_selected"); } this.nodes.edit_font_selector.disabled = true; this.nodes.edit_font_selector.selectedIndex = 0; this.nodes.edit_font_size_selector.disabled = true; this.nodes.edit_font_size_selector.selectedIndex = 0; this.nodes.edit_halign_selector.disabled = true; this.nodes.edit_halign_selector.selectedIndex = 0; this.nodes.edit_valign_selector.disabled = true; this.nodes.edit_valign_selector.selectedIndex = 0; this.nodes.edit_decoration_selector.disabled = true; this.nodes.edit_decoration_selector.selectedIndex = 0; this.nodes.edit_font_bold_check.disabled = true; this.nodes.edit_font_bold_check.checked = false; this.nodes.edit_font_italic_check.disabled = true; this.nodes.edit_font_italic_check.checked = false; this.annotation_selected_index = -1; }; var set_annotation_selected = function (annotation_index, select_text) { if (annotation_index === this.annotation_selected_index) return; // Update selection if (this.annotation_selected_index >= 0) { this.annotations[this.annotation_selected_index].set_selected(false); } this.annotation_selected_index = annotation_index; update_annotation_color_selected.call(this); update_annotation_color_index_selected.call(this); update_annotation_align_selected.call(this); update_annotation_font_selected.call(this); update_annotation_font_size_selected.call(this); update_annotation_text_decoration_selected.call(this); update_annotation_font_bold_selected.call(this); update_annotation_font_italic_selected.call(this); this.annotations[this.annotation_selected_index].set_selected(true, !select_text); }; var add_new_annotation = function (text_before, simple_mode) { if (this.annotations_previous !== this.annotations) { clear_previous_annotations.call(this); } // Create new var index = this.annotations.length, sz = get_annotation_default_size.call(this, simple_mode), a, m, i; if (text_before === undefined) { if (this.annotation_text_suffix === null) { text_before = null; this.annotation_text_suffix = ""; // will add an additional newline before the tag } else { m = /\s*$/.exec(this.annotation_text_suffix)[0]; i = this.annotation_text_suffix.length - m.length; text_before = this.annotation_text_suffix.substr(0, i); this.annotation_text_suffix = this.annotation_text_suffix.substr(i).replace(/^(?:\r\n?|\n)/, ""); if (text_before.length === 0) text_before = null; } } a = new Annotation(this, { edit: true, index: index, font_size_index: sz.font_size_index, font: settings.values.annotations.defaults.font, bold: settings.values.annotations.defaults.bold, italic: settings.values.annotations.defaults.italic, x: sz.x, y: sz.y, width: sz.width, height: sz.height, text_before: text_before }); this.annotations.push(a); a.add_to(this.nodes.annotation_container, this.nodes.edit_container); if (!simple_mode) { set_empty_links_visible.call(this, false); set_annotation_selected.call(this, index, true); update_annotation_full_text.call(this, false); } return a; }; var remove_annotation = function (index, update) { var a = this.annotations[index], tb = a.get_text_before(), tb2, i; // Remove if (this.annotations.length === 1) { this.annotations = []; a.remove_nodes(); this.annotation_text_suffix_previous = this.annotation_text_suffix; } else { this.annotations.splice(index, 1); a.destroy(); } // Update indices for (i = index; i < this.annotations.length; ++i) { this.annotations[i].set_index(i); } // Update text before if (tb !== null) { if (index < this.annotations.length) { tb2 = this.annotations[index].get_text_before(); tb2 = (tb2 === null) ? tb : (tb + "\n" + tb2); this.annotations[index].set_text_before(tb2); } else { tb2 = this.annotation_text_suffix; this.annotation_text_suffix = (tb2 === null) ? tb : (tb + "\n" + tb2); } } // Update selection if (this.annotation_selected_index === index) { i = this.annotation_selected_index; if (i > 0) --i; if (i < this.annotations.length) { this.annotation_selected_index = -1; set_annotation_selected.call(this, i, true); } else { set_no_annotations_selected.call(this); } } else if (this.annotation_selected_index > index) { --this.annotation_selected_index; } // Update text if (update) update_annotation_full_text.call(this, false); // Show links if (this.annotations.length === 0) { // Show set_empty_links_visible.call(this, true); } }; var get_annotation_default_size = function (simple_mode) { // New size var x = 0, y = 0, w = 100, h = 100, font_size, s; if (this.image.size.width > 0 && this.image.size.height > 0 && !simple_mode) { // Font size s = Math.min(this.image.size.width, this.image.size.height) / 32; font_size = Annotation.font_size_to_size_info(s); w = Math.round(w * font_size[1] / 16); h = Math.round(h * font_size[1] / 16); // Size if (w > this.image.size.width) { w = this.image.size.width; } else { s = this.image.size.width / this.cpreview.size.width; x = (this.cpreview.size.width - this.cpreview.display.width) * s * this.cpreview.offset_x + (this.cpreview.display.width * s - w) / 2; x = Math.round(x); } if (h > this.image.size.height) { h = this.image.size.height; } else { s = this.image.size.height / this.cpreview.size.height; y = (this.cpreview.size.height - this.cpreview.display.height) * s * this.cpreview.offset_y + (this.cpreview.display.height * s - h) / 2; y = Math.round(y); } } else { font_size = Annotation.font_size_to_size_info(16); } return { font_size_index: font_size[0], x: x, y: y, width: w, height: h, }; }; var rebound_annotations = function () { if (this.image.size.width <= 0 || this.image.size.height <= 0) return; // Update annotation bounds var i, a, sz, pos, x, y, w, h, update; for (i = 0; i < this.annotations.length; ++i) { update = false; a = this.annotations[i]; sz = a.get_size(); pos = a.get_position(); x = pos[0]; y = pos[1]; w = sz[0]; h = sz[1]; if (w > this.image.size.width) { x = 0; w = this.image.size.width; update = true; } else if (x + w > this.image.size.width) { x = this.image.size.width - w; update = true; } if (h > this.image.size.height) { y = 0; h = this.image.size.height; update = true; } else if (y + h > this.image.size.height) { y = this.image.size.height - h; update = true; } if (update) { a.set_annotation_rect(x, y, w, h); } } }; var apply_default_size_to_annotations = function (annotations) { var sz = get_annotation_default_size.call(this, false), i, a; for (i = 0; i < annotations.length; ++i) { a = annotations[i]; a.set_font_size(sz.font_size_index); a.set_annotation_rect(sz.x, sz.y, sz.width, sz.height); } }; var update_annotation_color_selected = function () { var a = this.annotations[this.annotation_selected_index], ns = this.nodes.edit_color_buttons, i; if (this.nodes.edit_color_selected >= 0) { style.remove_class(ns[this.nodes.edit_color_selected], "iex_ae_color_selector_selected"); } else { for (i = 0; i < ns.length; ++i) { style.remove_class(ns[i], "iex_ae_color_selector_disabled"); } } this.nodes.edit_color_selected = a.get_color(a.get_color_index_selected()); style.add_class(ns[this.nodes.edit_color_selected], "iex_ae_color_selector_selected"); }; var update_annotation_color_index_selected = function () { var sel = this.annotations[this.annotation_selected_index].get_color_index_selected(), cls = "iex_ae_top_color_link_selected", ns, i; ns = this.nodes.edit_color_index_labels; for (i = 0; i < ns.length; ++i) { if (i === sel) { style.add_class(ns[i], cls); } else { style.remove_class(ns[i], cls); } } }; var update_annotation_align_selected = function () { var a = this.annotations[this.annotation_selected_index]; this.nodes.edit_halign_selector.disabled = false; this.nodes.edit_halign_selector.selectedIndex = (a.get_align() & Annotation.AlignHorizontal); this.nodes.edit_valign_selector.disabled = false; this.nodes.edit_valign_selector.selectedIndex = (a.get_align() & Annotation.AlignVertical) >> 2; }; var update_annotation_font_selected = function () { var a = this.annotations[this.annotation_selected_index]; this.nodes.edit_font_selector.disabled = false; this.nodes.edit_font_selector.selectedIndex = a.get_font(); }; var update_annotation_font_size_selected = function () { var a = this.annotations[this.annotation_selected_index]; this.nodes.edit_font_size_selector.disabled = false; this.nodes.edit_font_size_selector.selectedIndex = a.get_font_size(); }; var update_annotation_text_decoration_selected = function () { var a = this.annotations[this.annotation_selected_index]; this.nodes.edit_decoration_selector.disabled = false; this.nodes.edit_decoration_selector.selectedIndex = a.get_text_decoration(); }; var update_annotation_font_bold_selected = function () { var a = this.annotations[this.annotation_selected_index]; this.nodes.edit_font_bold_check.disabled = false; this.nodes.edit_font_bold_check.checked = a.get_font_bold(); }; var update_annotation_font_italic_selected = function () { var a = this.annotations[this.annotation_selected_index]; this.nodes.edit_font_italic_check.disabled = false; this.nodes.edit_font_italic_check.checked = a.get_font_italic(); }; var clear_annotation_lines = function () { var i; // Clear, move into previous if (this.annotations.length > 0) { // Remove nodes of every annotation set_no_annotations_selected.call(this); for (i = 0; i < this.annotations.length; ++i) { this.annotations[i].remove_nodes(); } this.annotations = []; this.annotation_text_suffix_previous = this.annotation_text_suffix; } // Show set_empty_links_visible.call(this, true); }; var load_annotation_lines = function (lines, lines_before, remaining, tag) { if (!this.visible) return; var a_len_pre = this.annotations.length, sz = null, i, j, a, t, can_size; this.annotation_text_suffix = remaining; if (lines.length === 0) { clear_annotation_lines.call(this); } else { // Completely remove previous clear_previous_annotations.call(this); // Hide set_empty_links_visible.call(this, false); // Create new annotations for (i = 0; i < lines.length; ++i) { if (i < a_len_pre) { a = this.annotations[i]; a.set_text_before(lines_before[i]); } else { a = add_new_annotation.call(this, lines_before[i], true); } } // Select if (this.annotation_selected_index < 0 || this.annotation_selected_index >= i) { set_annotation_selected.call(this, i - 1, true); } // Remove extras while (i < this.annotations.length) { remove_annotation.call(this, i, false); } // Tag and text i = 0; if (tag !== null && (t = Annotator.process_tag(tag)) !== null) { // Load from state j = Math.min(this.annotations.length, t.annotations.length); for (; i < j; ++i) { this.annotations[i].load_from_state(t.annotations[i], lines[i]); } } // Text and default size can_size = (this.image.size.width > 0 && this.image.size.height > 0); if (!can_size) this.update_annotation_sizes_on_image_load = []; for (; i < lines.length; ++i) { a = this.annotations[i]; a.set_text(a.escape_text(lines[i])); if (i >= a_len_pre) { if (can_size) { // Default position if (sz === null) sz = get_annotation_default_size.call(this, false); a.set_font_size(sz.font_size_index); a.set_annotation_rect(sz.x, sz.y, sz.width, sz.height); } else { this.update_annotation_sizes_on_image_load.push(a); } } } // Re-bound rebound_annotations.call(this); } // Update update_annotation_full_text.call(this, false); }; var update_annotation_full_text = function (omit_tag) { if (!omit_tag && (this.image.size.width <= 0 || this.image.size.height <= 0 || !this.visible)) return; var text = "", i, a, t; for (i = 0; i < this.annotations.length; ++i) { if (i > 0) text += "\n"; a = this.annotations[i]; if ((t = a.get_text_before()) !== null) { text += t; text += "\n"; } text += ">"; if (a.text_display.length > 0 && a.text_display[0] === ">") text += " "; text += a.text_display; } if (this.annotation_text_suffix !== null) { if (i > 0) text += "\n"; text += this.annotation_text_suffix; ++i; } // Tag if (this.annotations.length > 0 && !omit_tag) { if (i > 0) text += "\n"; text += Annotator.generate_tag(this.image, this.annotations, this.annotation_text_suffix); } // Set text this.qr_controller.set_text(text); }; var on_movable_mousedown = function (event) { if (this.move_event_list.length > 0) { remove_event_listeners(this.move_event_list); this.move_event_list = []; } add_event_listener(this.move_event_list, document, "mousemove", on_movable_document_mousemove.bind(this), true); add_event_listener(this.move_event_list, document, "mouseup", on_movable_document_mouseup.bind(this), true); }; var on_movable_document_mousemove = function (event) { // Potentially moving if (this.qr_resize_timer === null) { this.qr_resize_timer = setTimeout(on_resize_delay_timeout.bind(this), 100); update_size.call(this); } else { this.qr_resized_during_timer = true; } }; var on_movable_document_mouseup = function (event) { // Clear events if (this.qr_resize_timer !== null) { clearTimeout(this.qr_resize_timer); this.qr_resize_timer = null; this.qr_resized_during_timer = false; } remove_event_listeners(this.move_event_list); this.move_event_list = []; update_size.call(this); }; var on_resize_delay_timeout = function () { this.qr_resize_timer = null; if (this.qr_resized_during_timer) { this.qr_resized_during_timer = false; update_size.call(this); } }; var on_window_resize = function (event) { if (this.qr_container !== null) { if (this.qr_resize_timer === null) { this.qr_resize_timer = setTimeout(on_resize_delay_timeout.bind(this), 100); update_size.call(this); } else { this.qr_resized_during_timer = true; } } }; var on_image_load = function (node, event) { if (this.image_load_poll !== null) { // Update image size this.image.size.width = node.naturalWidth; this.image.size.height = node.naturalHeight; style.add_class(node, "iex_ae_preview_image_visible"); update_preview_size.call(this); rebound_annotations.call(this); if (this.update_annotation_sizes_on_image_load !== null) { apply_default_size_to_annotations.call(this, this.update_annotation_sizes_on_image_load); this.update_annotation_sizes_on_image_load = null; } update_annotation_full_text.call(this, false); } clear_image_events.call(this); style.add_class(node, "iex_ae_preview_image_loaded"); }; var on_image_error = function (node, event) { clear_image_events.call(this); style.remove_class(node, "iex_ae_preview_image_visible"); style.remove_class(node, "iex_ae_preview_image_loaded"); style.add_class(node, "iex_ae_preview_image_error"); this.nodes.image_error.textContent = "Image failed to load"; }; var on_image_abort = function (node) { }; var on_image_status_poll = function (node) { if (node.naturalWidth > 0 && node.naturalHeight > 0) { clearInterval(this.image_load_poll); this.image_load_poll = null; // Update image size this.image.size.width = node.naturalWidth; this.image.size.height = node.naturalHeight; style.add_class(node, "iex_ae_preview_image_visible"); update_preview_size.call(this); rebound_annotations.call(this); if (this.update_annotation_sizes_on_image_load !== null) { apply_default_size_to_annotations.call(this, this.update_annotation_sizes_on_image_load); this.update_annotation_sizes_on_image_load = null; } update_annotation_full_text.call(this, false); } }; var on_container_mouseover = function (event) { show_preview_scrollbars.call(this); }; var on_container_mouseout = function (event) { hide_preview_scrollbars.call(this); }; var on_container_mousewheel = function (event) { if (this.preview_move_events.length === 0) { // Get direction var delta = (event.wheelDelta || -event.detail || 0), new_zoom; if (delta > 0) { if (this.cpreview_zoom < 4) { new_zoom = this.cpreview_zoom + 0.5; } else { new_zoom = this.cpreview_zoom * 2; } if (new_zoom > 32) new_zoom = 32; } else { if (this.cpreview_zoom <= 4) { new_zoom = this.cpreview_zoom - 0.5; } else { new_zoom = this.cpreview_zoom / 2; } if (new_zoom < 1) new_zoom = 1; } if (new_zoom !== this.cpreview_zoom) { change_zoom_level.call(this, new_zoom, true, event.clientX, event.clientY); } } // Stop return stop_event(event); }; var on_container_mousedown = function (event) { var button = get_event_mouse_button(event), display_size = this.cpreview.display, inner_size = this.cpreview.size, w, h; // Stop if (button !== 1 && button !== 2) { return stop_event(event); } // Else, start dragging w = (inner_size.width - display_size.width); h = (inner_size.height - display_size.height); if (w > 0) w = 1.0 / w; else w = 0; if (h > 0) h = 1.0 / h; else h = 0; this.preview_move_speed[0] = w; this.preview_move_speed[1] = h; this.preview_move_origin[0] = event.clientX; this.preview_move_origin[1] = event.clientY; this.preview_move_offset[0] = this.cpreview.offset_x; this.preview_move_offset[1] = this.cpreview.offset_y; remove_event_listeners(this.preview_move_events); this.preview_move_events = []; add_event_listener(this.preview_move_events, document, "mousemove", on_conatiner_move_document_mousemove.bind(this), true); add_event_listener(this.preview_move_events, document, "mouseup", on_conatiner_move_document_mouseup.bind(this), true); // Stop return stop_event(event); }; var on_container_contextmenu = function (event) { return stop_event(event); }; var on_conatiner_move_document_mousemove = function (event) { var xo, yo; xo = this.preview_move_offset[0] + (this.preview_move_origin[0] - event.clientX) * this.preview_move_speed[0]; yo = this.preview_move_offset[1] + (this.preview_move_origin[1] - event.clientY) * this.preview_move_speed[1]; if (xo < 0) xo = 0; else if (xo > 1) xo = 1; if (yo < 0) yo = 0; else if (yo > 1) yo = 1; this.cpreview.set_offset(xo, yo); update_preview_scrollbar_positions.call(this); }; var on_conatiner_move_document_mouseup = function (event) { on_conatiner_move_document_mousemove.call(this, event); remove_event_listeners(this.preview_move_events); this.preview_move_events = []; }; var on_color_index_select = function (index, event) { if (this.annotation_selected_index >= 0) { var a = this.annotations[this.annotation_selected_index]; a.set_color_index_selected(index); update_annotation_color_index_selected.call(this); update_annotation_color_selected.call(this); } }; var on_color_select = function (index, event) { if (this.annotation_selected_index >= 0) { var a = this.annotations[this.annotation_selected_index]; a.set_color(a.get_color_index_selected(), index); update_annotation_color_selected.call(this); update_annotation_full_text.call(this, false); } }; var on_new_note_empty_click = function (event) { on_new_note_click.call(this, event); }; var on_new_note_click = function (event) { add_new_annotation.call(this); }; var on_font_select = function (node, event) { if (this.annotation_selected_index >= 0) { var a = this.annotations[this.annotation_selected_index]; a.set_font(parseInt(node.value, 10)); update_annotation_full_text.call(this, false); } }; var on_font_scale_select = function (node, event) { if (this.annotation_selected_index >= 0) { var a = this.annotations[this.annotation_selected_index]; a.set_font_size(parseInt(node.value, 10)); update_annotation_full_text.call(this, false); } }; var on_font_bold_change = function (node, event) { if (this.annotation_selected_index >= 0) { var a = this.annotations[this.annotation_selected_index]; a.set_font_bold(node.checked); update_annotation_full_text.call(this, false); } }; var on_font_italic_change = function (node, event) { if (this.annotation_selected_index >= 0) { var a = this.annotations[this.annotation_selected_index]; a.set_font_italic(node.checked); update_annotation_full_text.call(this, false); } }; var on_halign_select = function (node, event) { if (this.annotation_selected_index >= 0) { var a = this.annotations[this.annotation_selected_index], align = a.get_align(); align = (align & ~Annotation.AlignHorizontal) | parseInt(node.value, 10); a.set_align(align); update_annotation_full_text.call(this, false); } }; var on_valign_select = function (node, event) { if (this.annotation_selected_index >= 0) { var a = this.annotations[this.annotation_selected_index], align = a.get_align(); align = (align & ~Annotation.AlignVertical) | parseInt(node.value, 10); a.set_align(align); update_annotation_full_text.call(this, false); } }; var on_text_decoration_select = function (node, event) { if (this.annotation_selected_index >= 0) { var a = this.annotations[this.annotation_selected_index]; a.set_text_decoration(parseInt(node.value, 10)); update_annotation_full_text.call(this, false); } }; var on_restore_annotations_click = function (event) { if (this.annotations_previous !== this.annotations && this.annotations_previous.length > 0) { // Hide links set_empty_links_visible.call(this, false); // Clear var i; for (i = 0; i < this.annotations.length; ++i) { this.annotations[i].destroy(); } // Restore this.annotations = this.annotations_previous; for (i = 0; i < this.annotations.length; ++i) { this.annotations[i].add_to(this.nodes.annotation_container, this.nodes.edit_container); } this.annotation_text_suffix = this.annotation_text_suffix_previous; this.annotation_text_suffix_previous = null; // Select set_annotation_selected.call(this, 0, true); // Update update_annotation_full_text.call(this, false); } }; var on_annotation_event = { select: function (annotation) { if (!annotation.is_selected()) { set_annotation_selected.call(this, annotation.index, false); } }, align: function (annotation) { var a = annotation.get_align(), ha = a & Annotation.AlignHorizontal; a &= ~Annotation.AlignHorizontal; if (ha === Annotation.AlignCenter) ha = Annotation.AlignRight; else if (ha === Annotation.AlignRight) ha = Annotation.AlignLeft; else if (ha === Annotation.AlignLeft) ha = Annotation.AlignJustify; else ha = Annotation.AlignCenter; annotation.set_align(a | ha); if (annotation.is_selected()) { update_annotation_align_selected.call(this); } update_annotation_full_text.call(this, false); }, color: function (annotation) { annotation.set_color_index_selected((annotation.get_color_index_selected() + 1) % annotation.get_color_count()); if (annotation.is_selected()) { update_annotation_color_index_selected.call(this); update_annotation_color_selected.call(this); } update_annotation_full_text.call(this, false); }, font: function (annotation) { if (annotation.get_font_bold()) { if (annotation.get_font_italic()) { annotation.set_font_bold(false); } else { annotation.set_font_italic(true); } } else { if (annotation.get_font_italic()) { annotation.set_font_italic(false); } else { annotation.set_font_bold(true); } } if (annotation.is_selected()) { update_annotation_font_bold_selected.call(this); update_annotation_font_italic_selected.call(this); } update_annotation_full_text.call(this, false); }, font_size: function (annotation) { var size = annotation.get_font_size(); size = (size + 1) % (Annotation.font_size_settings.step_duration * Annotation.font_size_settings.step_count); annotation.set_font_size(size); if (annotation.is_selected()) { update_annotation_font_size_selected.call(this); } update_annotation_full_text.call(this, false); }, move_up: function (annotation) { if (annotation.index > 0) { var annotation_index_old = annotation.index, annotation_index_new = annotation_index_old - 1, other = this.annotations[annotation_index_new], b = annotation.get_text_before(); annotation.insert_before(other); annotation.set_index(annotation_index_new); other.set_index(annotation_index_old); this.annotations.splice(annotation_index_old, 1); this.annotations.splice(annotation_index_new, 0, annotation); annotation.set_text_before(other.get_text_before()); other.set_text_before(b); if (this.annotation_selected_index === annotation_index_old) { this.annotation_selected_index = annotation_index_new; } else if (this.annotation_selected_index === annotation_index_new) { this.annotation_selected_index = annotation_index_old; } update_annotation_full_text.call(this, false); } }, move_down: function (annotation) { var index_max = this.annotations.length - 1, other, annotation_index_new, annotation_index_old, b; if (annotation.index < index_max) { annotation_index_old = annotation.index; annotation_index_new = annotation_index_old + 1; other = this.annotations[annotation_index_new]; b = annotation.get_text_before(); if (annotation_index_old === index_max - 1) { annotation.add_to(this.nodes.annotation_container, this.nodes.edit_container); } else { annotation.insert_before(this.annotations[annotation_index_new + 1]); } annotation.set_index(annotation_index_new); other.set_index(annotation_index_old); this.annotations.splice(annotation_index_old, 1); this.annotations.splice(annotation_index_new, 0, annotation); annotation.set_text_before(other.get_text_before()); other.set_text_before(b); if (this.annotation_selected_index === annotation_index_old) { this.annotation_selected_index = annotation_index_new; } else if (this.annotation_selected_index === annotation_index_new) { this.annotation_selected_index = annotation_index_old; } update_annotation_full_text.call(this, false); } }, remove: function (annotation) { remove_annotation.call(this, annotation.index, true); }, // text_input: function (annotation) {}, text_change: function (/*annotation*/) { update_annotation_full_text.call(this, false); }, rect_changed: function (/*annotation*/) { update_annotation_full_text.call(this, false); }, }; AnnotationEditor.prototype = { constructor: AnnotationEditor, set_qr_container: function (qr_container) { if (this.qr_container !== qr_container) { if (this.qr_container !== null) { remove_nodes.call(this); remove_qr_events.call(this); } this.qr_container = qr_container; if (this.qr_container !== null) { create_nodes.call(this); setup_qr_events.call(this); } } }, set_file: function (file, url, mode, extra) { // Set image set_image.call(this, url, mode, extra); }, show: function () { if (this.qr_container === null || this.visible) return; this.visible = true; style.add_class(this.nodes.container, "iex_annotation_editor_visible"); update_size.call(this); //this.load_annotations_from_text(this.qr_controller.get_text()); }, hide: function () { if (this.qr_container === null || !this.visible) return; this.visible = false; style.remove_class(this.nodes.container, "iex_annotation_editor_visible"); update_annotation_full_text.call(this, true); clear_annotation_lines.call(this); }, is_visible: function () { return this.visible; }, load_annotations_from_text: function (text) { var lines = [], lines_before = [], tag = null, m; // Find lines while ((m = re_load_annotations.exec(text)) !== null) { text = text.substr(m.index + m[0].length); lines.push(m[2].trim()); lines_before.push(m[1] === undefined ? null : m[1]); if (m[3].length === 0) { text = null; break; } } // Find tag if (text !== null && (m = re_load_tag.exec(text)) !== null) { // Load tag text = (m[1] === undefined ? null : m[1]); tag = m[3]; } // Load load_annotation_lines.call(this, lines, lines_before, text, tag); }, get_image_scale: function () { return this.preview_scale; }, get_image_size: function () { return this.image.size; }, on_annotation_event: function (annotation, event_name) { var fn = on_annotation_event[event_name]; if (fn !== undefined) { fn.call(this, annotation); } }, }; return AnnotationEditor; })(); // Setup objects style = new Style(); api = new API(); sync = new Sync(); settings = new Settings(); hotkey_manager = new HotkeyManager(); // Execute once page type is detected api.on("page_type_detected", function (event) { if (event.page_type == "board" || event.page_type == "thread" || event.page_type == "catalog") { // Settings settings.setup(); // Insert stylesheet style.insert_stylesheet(); // Create new hover hover = new Hover(); hover.start(); // Create image hover manager image_hover = new ImageHover(hover); settings.on_ready(image_hover.start.bind(image_hover)); // Create image link modifier file_link = new FileLink(); settings.on_ready(file_link.start.bind(file_link)); settings.on_ready(function () { // Check if enabled if (settings.values.annotations.enabled) { // Insert stylesheet style.insert_stylesheet_annotations(); // Annotator var annotator = new Annotator(); settings.on_ready(annotator.start.bind(annotator, image_hover, file_link)); if (settings.values.annotations.editor) { // Quick reply controller new QuickReplyController(); } } }); } else if (event.page_type == "image" || event.page_type == "video") { // Settings settings.setup(); // File view file_view = new FileView(); file_view.on("image", function (image) { // Check if enabled if (settings.values.annotations.enabled_standalone) { // Check url var d = Annotator.check_url(); if (d !== null) { // Insert stylesheet style.insert_stylesheet_annotations(); // Create overlay Annotator.create_image_overlay(image, d); } } }); settings.on_ready(file_view.start.bind(file_view)); } }); api.setup(); })();