// ==UserScript== // @id GAF-QQRE // @name NeoGAF: Quick Quote, Reply, and Edit // @namespace hateradio))) // @author hateradio // @version 9.0 // @description This script adds the ability to quickly quote, reply, and edit posts on NeoGAF. Just click and go. // @homepage https://greasyfork.org/scripts/1022-neogaf-quick-quote-reply-and-edit // @icon  // @screenshot https://dl.dropboxusercontent.com/u/14626536/userscripts/updt/qqre/qqre_09.png // @include http://*neogaf.com/forum/showthread.php?* // @include http://*neogaf.com/forum/newreply.php* // @include http://*neogaf.com/forum/editpost.php* // @include http://*neogaf.com/forum/newthread.php* // @include http://*neogaf.com/forum/subscription.php* // @include http://*neogaf.com/forum/showpost.php* // @include http://*neogaf.com/forum/private.php* // @match http://*.neogaf.com/forum/showthread.php?* // @match http://*.neogaf.com/forum/newreply.php* // @match http://*.neogaf.com/forum/editpost.php* // @match http://*.neogaf.com/forum/newthread.php* // @match http://*.neogaf.com/forum/subscription.php* // @match http://*.neogaf.com/forum/showpost.php* // @match http://*.neogaf.com/forum/private.php* // @updated 18 NOV 2014 // @since 30 OCT 2010 // @grant GM_xmlhttpRequest // @grant unsafeWindow // (c) 2010+, hateradio // @downloadURL none // ==/UserScript== /* Changes ## 9.0 * **New**: Keyboard toggle for images in quotes is now ALT+SHIFT+T instead of ALT+T for greater compatibility. * **New**: Improved shortcuts for Mac (eg, CMD+B for Bold) * **New**: Temporary storage saves text as it is written. * **New**: Subscription setting overhauled. * - Will now only provide an override option, as all posts made will default to the settings provided by the [User Control Panel](http://www.neogaf.com/forum/profile.php?do=editoptions#sel_autosubscribe) * **New**: Middle clicking on Emoji will link to [CodePoints.net](https://codepoints.net) for information about that character * **New**: AutoPager & Live Update extension support * **New**: Informs anonymous users to sign in before submitting posts * **New**: Resize toggle added to Emoji box * **New**: Tabbing will send the focus from the textarea to the Submit button * **New**: Routing mechanism to load the different code on the different pages * **New**: New Form class to create Quick Reply and Edit forms * **New**: Character escaping mechanism to convert JavaScript characters to HTML entities for AJAX transmissions; removes VB method * **Fix**: Removed vertical alignment of inputs * **Fix**: Posts that have been saved now clear after the post has been processed * Renames "Delete" button with "Clear" * Restructure of the code base // todo implement preview ## 8.0 * **New**: Emoji picker! * **New**: (Windows) Emoji support to Chrome-based browsers! * - _Note 1_: Segoe UI Symbol, Segoe UI Emoji and Symbola added to the font list * - _Note 2_: This may change default fonts a bit * - _Note 3_: Windows 7 users who have issues displaying Emoji should install [Symbola](http://users.teilar.gr/~g1951d/) * **New**: USC-2 decoder and entity encoder to work with forum AJAX to transmit Emoji data * **Fix**: Removing recursive quotes no longer causes an error 7.4.1 7.4 7.3 7.1 - 7.2 7.0 6.3 6.1.2 - 6.2 6.1

6.0

4,156 */ (function () { 'use strict'; var $, greaseWindow, strg, update, Extra, Editor, Form, app, bond, emoji; if (!String.prototype.trim) { String.prototype.trim = function () { return this.replace(/^\s+|\s+$/g, ''); }; } bond = function (o, m) { return function () { return m.apply(o, arguments); }; }; // UCS-2 Decoder https://github.com/bestiejs/punycode.js function ucs2decode(string) { var output = [], counter = 0, length = string.length, value, extra; while (counter < length) { value = string.charCodeAt(counter++); if ((value & 0xF800) === 0xD800 && counter < length) { // high surrogate, and there is a next character extra = string.charCodeAt(counter++); if ((extra & 0xFC00) === 0xDC00) { // low surrogate output.push(((value & 0x3FF) << 10) + (extra & 0x3FF) + 0x10000); } else { output.push(value, extra); } } else { output.push(value); } } return output; } // Window | No longer works in Chrome 27+ greaseWindow = (function (w) { try { w = unsafeWindow === window ? w : unsafeWindow; } catch (e) {} return w || (function () { try { w = document.createElement('p'); w.setAttribute('onclick', 'return window;'); return w.onclick(); } catch (e) { return window; } }()); }()); // console.log(greaseWindow === this); function embedScript(func) { var script = document.createElement('script'); script.type = 'application/javascript'; script.textContent = '(' + func + ')();'; document.body.appendChild(script); document.body.removeChild(script); } // S T O R A G E HANDLE strg = { on: (function () { try { var s = window.localStorage; return (s.getItem && s.setItem && s.removeItem); } catch (e) { return false; } }()), read: function (key) { return this.on ? JSON.parse(window.localStorage.getItem(key)) : false; }, grab: function (key) { return this.read(key) || {}; }, save: function (key, dat) { return this.on ? !window.localStorage.setItem(key, JSON.stringify(dat)) : false; }, wipe: function (key) { return this.on ? !window.localStorage.removeItem(key) : false; }, zero: function (obj) { var k; for (k in obj) { if (obj.hasOwnProperty(k)) { return false; } } return true; } }; // U P D A T E HANDLE update = { name: 'NeoGAF: Quick Quote, Reply, and Edit', version: 9000, key: 'ujs_QQRE_UPDT', callback: 'qqreupdater', page: 'https://greasyfork.org/scripts/1022-neogaf-quick-quote-reply-and-edit', uric: 'https://dl.dropboxusercontent.com/u/14626536/userscripts/updt/qqre/qqrej.js', checkchrome: true, interval: 5, day: (new Date()).getTime(), time: function () {return (new Date(this.day + (1000 * 60 * 60 * 24 * this.interval))).getTime(); }, top: document.head || document.getElementsByTagName('head')[0], css: function (t) { if (!this.style) { this.style = document.createElement('style'); this.style.type = 'text/css'; this.top.appendChild(this.style); } this.style.appendChild(document.createTextNode(t + '\n')); }, js: function (t) { var j = document.createElement('script'); j.type = 'text/javascript'; j[/^https?\:\/\//i.test(t) ? 'src' : 'textContent'] = t; this.top.appendChild(j); }, notification: function (j) { if (j) { if (this.version < j.version) { window.localStorage.setItem(this.key, JSON.stringify({date: this.time(), version: j.version, page: j.page })); } else { return true; } } var a = document.createElement('a'), b = JSON.parse(window.localStorage.getItem(this.key)); a.href = b.page || '#'; a.target = '_blank'; a.id = 'userscriptupdater'; a.title = 'Update Now!'; a.textContent = 'An update for ' + this.name + ' is available.'; document.body.appendChild(a); return true; }, check: function (force) { if (this.extension) { return; } if (!strg.on) { return; } // || typeof (GM_updatingEnabled) === 'boolean' var stored = strg.read(this.key), j, page; this.csstxt(); if (force || !stored || stored.date < this.day) { page = stored && stored.page ? stored.page : this.page; strg.save(this.key, {date: this.time(), version: this.version, page: page}); j = this.notification.toString() .replace('function', 'function ' + this.callback) .replace('this.version', this.version) .replace(/(?:this\.key)/g, "'" + this.key + "'") .replace('this.time()', this.time()) .replace('this.name', "'" + this.name + "'"); this.js(j); this.js(this.uric); } else if (this.version < stored.version) { this.notification(); } }, csstxt: function () { if (!this.pop) { this.pop = true; this.css('#userscriptupdater,#userscriptupdater:visited{box-shadow:0 0 6px #787878;border:1px solid #777;border-radius:4px;cursor:pointer;color:#555;font-family:Arial, Verdana, sans-serif;font-size:11px;font-weight:700;text-align:justify;min-height:45px;position:fixed;z-index:999999;right:10px;top:10px;width:170px;background:#ebebeb url() no-repeat 13px 15px;padding:12px 20px 10px 65px}#userscriptupdater:hover,#userscriptupdater:visited:hover{color:#55698c!important;background-position:13px -85px;border-color:#8f8d96}'); } }, extension: window.chrome && window.chrome.extensions }; update.check(); // DOM/HTML Helpers $ = { a: function (e) { var i = 1, j = arguments.length, f = document.createDocumentFragment(); for (i; i < j; i++) { f.appendChild(arguments[i]); } e.appendChild(f); return e; }, e: function (t, o, e, p) { var a, b, c = document.createElement(t); if (typeof (o) === 'object') { for (a in o) { if (o.hasOwnProperty(a)) { b = a.charAt(0); switch (b) { case '_': c.style[a.substring(1)] = o[a]; break; case '$': c.setAttribute(a.substring(1), o[a]); break; default: c[a] = o[a]; break; } } } } if (e) { if (p) { c.appendChild(e); } else { e.appendChild(c); } } return c; }, t: document.createElement('textarea'), h: function (t) { this.t.innerHTML = t; return this.t.value.trim(); }, htmlEncode: function (text) { var chars = ucs2decode(text), c, i, j = chars.length, str = []; for (i = 0; i < j; i++) { c = chars[i]; str.push(c >= 127 ? ('&#' + c + ';') : String.fromCharCode(c)); } return str.join(''); }, uriHtmlEncode: function (text) { return encodeURIComponent(this.htmlEncode(text)); } }; // Extra Reply Buttons Extra = (function () { function Extra(id, custom) { id = id || '001'; this.editor = 'vB_Editor_' + id; this.custum = !!custom; this.txt = document.getElementById(this.editor + '_textarea'); if (this.txt) { this.src = this.custum ? document.getElementById(this.editor + '_cmd_underline') : document.getElementById(this.editor + '_cmd_underline').parentNode; this.BB = {}; this.add(); this.setup(); } } Extra.buttons = ['Spoiler', 'Highlight', 'Strike']; Extra.images = [ '/forum/images/neogaf2/misc/spoiler.gif', '', '' ]; Extra.ctrl = { b: 'b', i: 'i', u: 'u', s: 'Strike', h: 'Highlight', p: 'Spoiler', l: 'url', m: 'img', e: 'save', d: 'clear' }; Extra.mac = /(?:mac)/i.test(navigator.platform); Extra.moz = window.InstallTrigger !== undefined; Extra.keyEvent = Extra.moz ? 'keypress' : 'keydown'; Extra.click = function (el) { var evt = document.createEvent('MouseEvents'); evt.initMouseEvent('click', true, true, window, 0, 0, 0, 0, 0, false, false, false, false, 0, null); // window.setTimeout(function () { el.dispatchEvent(evt); // }, 1); }; Extra.prototype = { constructor: Extra, insert: function (evt) { // based on: http://parentnode.org/javascript/working-with-the-cursor-position/ // gecko parts only var t = evt.target.getAttribute('alt').toUpperCase(), start, end, pos, temp; if (this.txt.selectionStart >= 0) { start = this.txt.selectionStart || 0; end = this.txt.selectionEnd || 0; temp = this.txt.value.substr(0, start) + '[' + t + ']' + this.txt.value.substr(start, end - start); pos = temp.length; this.txt.value = temp + '[/' + t + ']' + this.txt.value.substr(end, this.txt.value.length); this.txt.focus(); start = start + 2 + t.length; this.txt.setSelectionRange(start, pos); } }, add: function () { var i = Extra.buttons.length, div, td; while (i--) { div = $.e('div', {className: 'imagebutton', title: 'new'}); this.BB[Extra.buttons[i]] = $.e('img', {height: 20, width: 21, src: Extra.images[i], alt: Extra.buttons[i], title: Extra.buttons[i]}, div); this.BB[Extra.buttons[i]].addEventListener('click', this.insert.bind(this), false); if (this.custum) { this.src.parentNode.insertBefore(div, this.src.nextSibling); } else { td = $.e('td', {className: 'imgbtnemu'}, div, true); this.src.parentNode.insertBefore(td, this.src.nextSibling); } } }, keys: function (e) { if (!e.shiftKey && (e.ctrlKey || e.metaKey)) { // console.log('keys--which', e.which, e); var k = Extra.ctrl[String.fromCharCode(e.which).toLowerCase()]; // console.log('k', k); if (k && (this.BB[k] || (e.metaKey && this.BB.meta[k]))) { // console.log('keys found'); e.preventDefault(); Extra.click(this.BB[k] || this.BB.meta[k]); } } }, setup: function () { this.BB.meta = {}; this.BB.b = document.getElementById(this.editor + '_cmd_bold'); this.BB.i = document.getElementById(this.editor + '_cmd_italic'); this.BB.u = document.getElementById(this.editor + '_cmd_underline'); this.BB.img = document.getElementById(this.editor + '_cmd_insertimage'); this.BB.url = document.getElementById(this.editor + '_cmd_createlink'); this.BB.meta.b = this.BB.b; this.BB.meta.i = this.BB.i; this.BB.meta.u = this.BB.u; if (Extra.moz) { this.BB.b = this.BB.i = this.BB.u = null; } this.txt.addEventListener(Extra.keyEvent, this.keys.bind(this), false); } }; return Extra; }()); update.css('td.imgbtnemu div:hover { border:1px solid #316ac5; margin: -1px; background: #c1d2ee; } td.imgbtnemu div:active { background: #98b5e2; } #vB_Editor_001_cmd_wrap0_spoiler { visibility:hidden !important } .quickreplyform_hotspot{text-align:center;margin-top:6px} .text_emo_container { text-align: center; margin-top: 8px; } .text_emo_container > div {vertical-align: top} .text_emo_container .text_emo_container_text {display:inline-block;margin: 0 0 0 4%;width:70%} .text_emo_container .text_emo_container_emo {width:18%;margin: 0 0 0 1%;display:inline-block;overflow:hidden} .text_emo_container_text textarea {width:99%} .gm_emoji {height:157px;overflow: auto;resize:both} .gm_emoji a {font-size:18px;line-height:1.2;cursor:pointer}'); // Emoji Stuff <3 OSX 10.9+ Command+Control+Space emoji = { div: null, dbGenerator: function (ranges) { var list = [], i; ranges.forEach(function (range) { for (i = range[0]; i <= range[1]; i++) { list.push(i.toString(16)); } }); return list; }, db: { faces: ['1f604', '1f603', '1f600', '1f60a', /*'263a',*/ '1f609', '1f60d', '1f618', '1f61a', '1f617', '1f619', '1f61c', '1f61d', '1f61b', '1f633', '1f601', '1f614', '1f60c', '1f612', '1f61e', '1f623', '1f622', '1f602', '1f62d', '1f62a', '1f625', '1f630', '1f605', '1f613', '1f629', '1f62b', '1f628', '1f631', '1f620', '1f621', '1f624', '1f616', '1f606', '1f60b', '1f637', '1f60e', '1f634', '1f635', '1f632', '1f61f', '1f626', '1f627', '1f608', '1f47f', '1f62e', '1f62c', '1f610', '1f615', '1f62f', '1f636', '1f607', '1f60f', '1f611', '1f472', '1f473', '1f46e', '1f477', '1f482', '1f476', '1f466', '1f467', '1f468', '1f469', '1f474', '1f475', '1f471', '1f47c', '1f478'], cats: ['1f63a', '1f638', '1f63b', '1f63d', '1f63c', '1f640', '1f63f', '1f639', '1f63e'], otherFaces: ['1f479', '1f47a', '1f648', '1f649', '1f64a', '1f480', '1f47d', '1f4a9'], misc: ['1f525', '2728', '1f31f', '1f4ab', '1f4a5', '1f4a2', '1f4a6', '1f4a7', '1f4a4', '1f4a8', '1f442', '1f440', '1f443', '1f445', '1f444', '1f44d', '1f44e', '1f44c', '1f44a', '270a', '270c', '1f44b', '270b', '1f450', '1f446', '1f447', '1f449', '1f448', '1f64c', '1f64f', '261d', '1f44f', '1f4aa', '1f6b6', '1f3c3', '1f483', '1f46b', '1f46a', '1f46c', '1f46d', '1f48f', '1f491', '1f46f', '1f646', '1f645', '1f481', '1f64b', '1f486', '1f487', '1f485', '1f470', '1f64e', '1f64d', '1f647'], fashion: ['1f3a9', '1f451', '1f452', '1f45f', '1f45e', '1f461', '1f460', '1f462', '1f455', '1f454', '1f45a', '1f457', '1f3bd', '1f456', '1f458', '1f459', '1f4bc', '1f45c', '1f45d', '1f45b', '1f453', '1f380', '1f302', '1f484'], hearts: ['1f49b', '1f499', '1f49c', '1f49a', '2764', '1f494', '1f497', '1f493', '1f495', '1f496', '1f49e', '1f498', '1f48c', '1f48b', '1f48d', '1f48e', '1f464', '1f465', '1f4ac', '1f463', '1f4ad'], animals: ['1f436', '1f43a', '1f431', '1f42d', '1f439', '1f430', '1f438', '1f42f', '1f428', '1f43b', '1f437', '1f43d', '1f42e', '1f417', '1f435', '1f412', '1f434', '1f411', '1f418', '1f43c', '1f427', '1f426', '1f424', '1f425', '1f423', '1f414', '1f40d', '1f422', '1f41b', '1f41d', '1f41c', '1f41e', '1f40c', '1f419', '1f41a', '1f420', '1f41f', '1f42c', '1f433', '1f40b', '1f404', '1f40f', '1f400', '1f403', '1f405', '1f407', '1f409', '1f40e', '1f410', '1f413', '1f415', '1f416', '1f401', '1f402', '1f432', '1f421', '1f40a', '1f42b', '1f42a', '1f406', '1f408', '1f429', '1f43e'], plants: ['1f490', '1f338', '1f337', '1f340', '1f339', '1f33b', '1f33a', '1f341', '1f343', '1f342', '1f33f', '1f33e', '1f344', '1f335', '1f334', '1f332', '1f333', '1f330', '1f331', '1f33c'], weather: ['1f310', '1f31e', '1f31d', '1f31a', '1f311', '1f312', '1f313', '1f314', '1f315', '1f316', '1f317', '1f318', '1f31c', '1f31b', '1f319', '1f30d', '1f30e', '1f30f', '1f30b', '1f30c', '1f320', '2b50', '2600', '26c5', '2601', '26a1', '2614', '2744', '26c4', '1f300', '1f301', '1f308', '1f30a'], items: ['1f38d', '1f49d', '1f38e', '1f392', '1f393', '1f38f', '1f386', '1f387', '1f390', '1f391', '1f383', '1f47b', '1f385', '1f384', '1f381', '1f38b', '1f389', '1f38a', '1f388', '1f38c', '1f52e', '1f3a5', '1f4f7', '1f4f9', '1f4fc', '1f4bf', '1f4c0', '1f4bd', '1f4be', '1f4bb', '1f4f1', '260e', '1f4de', '1f4df', '1f4e0', '1f4e1', '1f4fa', '1f4fb', '1f50a', '1f509', '1f508', '1f507', '1f514', '1f515', '1f4e2', '1f4e3', '23f3', '231b', '23f0', '231a', '1f513', '1f512', '1f50f', '1f510', '1f511', '1f50e', '1f4a1', '1f526', '1f506', '1f505', '1f50c', '1f50b', '1f50d', '1f6c1', '1f6c0', '1f6bf', '1f6bd', '1f527', '1f529', '1f528', '1f6aa', '1f6ac', '1f4a3', '1f52b', '1f52a', '1f48a', '1f489', '1f4b0', '1f4b4', '1f4b5', '1f4b7', '1f4b6', '1f4b3', '1f4b8', '1f4f2'], stationary: ['1f4e7', '1f4e5', '1f4e4', '2709', '1f4e9', '1f4e8', '1f4ef', '1f4eb', '1f4ea', '1f4ec', '1f4ed', '1f4ee', '1f4e6', '1f4dd', '1f4c4', '1f4c3', '1f4d1', '1f4ca', '1f4c8', '1f4c9', '1f4dc', '1f4cb', '1f4c5', '1f4c6', '1f4c7', '1f4c1', '1f4c2', '2702', '1f4cc', '1f4ce', '2712', '270f', '1f4cf', '1f4d0', '1f4d5', '1f4d7', '1f4d8', '1f4d9', '1f4d3', '1f4d4', '1f4d2', '1f4da', '1f4d6', '1f516', '1f4db', '1f52c', '1f52d', '1f4f0'], musicArt: ['1f3a8', '1f3ac', '1f3a4', '1f3a7', '1f3bc', '1f3b5', '1f3b6', '1f3b9', '1f3bb', '1f3ba', '1f3b7', '1f3b8'], sportsGames: ['1f47e', '1f3ae', '1f0cf', '1f3b4', '1f004', '1f3b2', '1f3af', '1f3c8', '1f3c0', '26bd', '26be', '1f3be', '1f3b1', '1f3c9', '1f3b3', '26f3', '1f6b5', '1f6b4', '1f3c1', '1f3c7', '1f3c6', '1f3bf', '1f3c2', '1f3ca', '1f3c4', '1f3a3'], foods: ['2615', '1f375', '1f376', '1f37c', '1f37a', '1f37b', '1f378', '1f379', '1f377', '1f374', '1f355', '1f354', '1f35f', '1f357', '1f356', '1f35d', '1f35b', '1f364', '1f371', '1f363', '1f365', '1f359', '1f358', '1f35a', '1f35c', '1f372', '1f362', '1f361', '1f373', '1f35e', '1f369', '1f36e', '1f366', '1f368', '1f367', '1f382', '1f370', '1f36a', '1f36b', '1f36c', '1f36d', '1f36f'], fruits: ['1f34e', '1f34f', '1f34a', '1f34b', '1f352', '1f347', '1f349', '1f353', '1f351', '1f348', '1f34c', '1f350', '1f34d', '1f360', '1f346', '1f345', '1f33d'], buildings: ['1f3e0', '1f3e1', '1f3eb', '1f3e2', '1f3e3', '1f3e5', '1f3e6', '1f3ea', '1f3e9', '1f3e8', '1f492', '26ea', '1f3ec', '1f3e4', '1f307', '1f306', '1f3ef', '1f3f0', '26fa', '1f3ed', '1f5fc', '1f5fe', '1f5fb', '1f304', '1f305', '1f303', '1f5fd', '1f309', '1f3a0', '1f3a1', '26f2', '1f3a2', '1f6a2'], transport: ['26f5', '1f6a4', '1f6a3', '2693', '1f680', '2708', '1f4ba', '1f681', '1f682', '1f68a', '1f689', '1f69e', '1f686', '1f684', '1f685', '1f688', '1f687', '1f69d', '1f68b', '1f683', '1f68e', '1f68c', '1f68d', '1f699', '1f698', '1f697', '1f695', '1f696', '1f69b', '1f69a', '1f6a8', '1f693', '1f694', '1f692', '1f691', '1f690', '1f6b2', '1f6a1', '1f69f', '1f6a0', '1f69c', '1f488', '1f68f', '1f3ab', '1f6a6', '1f6a5', '26a0', '1f6a7', '1f530', '26fd', '1f3ee', '1f3b0', '2668', '1f5ff', '1f3aa', '1f3ad', '1f4cd', '1f6a9'], arrows: ['1f51f', '1f522', /*'20e3',*/ '1f523', '2b06', '2b07', '2b05', '27a1', '1f520', '1f521', '1f524', '2197', '2196', '2198', '2199', '2194', '2195', '1f504', '25c0', '25b6', '1f53c', '1f53d', '21a9', '21aa', '2139', '23ea', '23e9', '23eb', '23ec', '2935', '2934'], textLabels: ['1f197', '1f500', '1f501', '1f502', '1f195', '1f199', '1f192', '1f193', '1f196', '1f4f6', '1f3a6', '1f201', '1f22f', '1f233', '1f235', '1f234', '1f232', '1f250', '1f239', '1f23a', '1f236', '1f21a', '1f6bb', '1f6b9', '1f6ba', '1f6bc', '1f6be', '1f6b0', '1f6ae', '1f17f', '267f', '1f6ad', '1f237', '1f238', '1f202', '24c2', '1f6c2', '1f6c4', '1f6c5', '1f6c3', '1f251', '3299', '3297', '1f191', '1f198', '1f194', '1f6ab', '1f51e', '1f4f5', '1f6af', '1f6b1', '1f6b3', '1f6b7', '1f6b8', '26d4', '2733', '2747', '274e', '2705', '2734', '1f49f', '1f19a', '1f4f3', '1f4f4', '1f170', '1f171', '1f18e', '1f17e', '1f4a0', '27bf', '267b'], zodiac: ['2648', '2649', '264a', '264b', '264c', '264d', '264e', '264f', '2650', '2651', '2652', '2653', '26ce'], symbols: ['1f52f', '1f3e7', '1f4b9', '1f4b2', '1f4b1', 'a9', 'ae', '2122', '274c', '203c', '2049', '2757', '2753', '2755', '2754', '2b55', '1f51d', '1f51a', '1f519', '1f51b', '1f51c', '1f503', '1f55b', '1f567', '1f550', '1f55c', '1f551', '1f55d', '1f552', '1f55e', '1f553', '1f55f', '1f554', '1f560', '1f555', '1f556', '1f557', '1f558', '1f559', '1f55a', '1f561', '1f562', '1f563', '1f564', '1f565', '1f566', '2716', '2795', '2796', '2797', '2660', '2665', '2663', '2666', '1f4ae', '1f4af', '2714', '2611', '1f518', '1f517', '27b0', '3030', '303d', '1f531', '25fc', '25fb', '25fe', '25fd', '25aa', '25ab', '1f53a', '1f532', '1f533', '26ab', '26aa', '1f534', '1f535', '1f53b', '2b1c', '2b1b', '1f536', '1f537', '1f538', '1f539'], dingbatsAndMore: (function () { var base = ['2588', '2318'], more = [[0x2600, 0x2647], [0x2654, 0x266F], [0xb0, 0xb0]], i; more.forEach(function (range) { for (i = range[0]; i <= range[1]; i++) { base.push(i.toString(16)); } }); return base; }()) }, create: function (point) { var a = document.createElement('a'); a.innerHTML = '&#x' + point + ';'; a.setAttribute('data-char', a.innerHTML); a.title = 'U+' + point; // a.innerHTML; a.href = 'https://codepoints.net/U+' + point; return a; }, injectTo: function (div) { var e, i, l, f = document.createDocumentFragment(); for (e in this.db) { if (this.db.hasOwnProperty(e)) { l = this.db[e]; for (i = 0; i < l.length; i++) { f.appendChild(this.create(l[i])); f.appendChild(document.createTextNode(' ')); } } } div.appendChild(f); }, events: { insert: function (e, textarea) { var target = e.target.hasAttribute('data-char') ? e.target : e.target.parentElement, start, end, pre, post, emo; if (textarea.selectionStart < 0) { return; } if (target.hasAttribute('data-char')) { if (textarea.selectionEnd > textarea.value.length) { textarea.selectionEnd = textarea.value.length; } start = textarea.selectionStart || 0; end = textarea.selectionEnd || 0; pre = textarea.value.substring(0, start); emo = (start > 0 ? ' ' : '') + target.getAttribute('data-char'); post = emo + textarea.value.substring(end); textarea.value = pre + post; textarea.focus(); textarea.setSelectionRange(start + emo.length, start + emo.length); } }, wire: function (div, textarea) { div.addEventListener('click', function (e) { if (e.which === 1) { e.preventDefault(); emoji.events.insert(e, textarea); } }, false); } }, make: function (parent, textarea) { var div = $.e('div', { className: 'gm_emoji' }); parent.appendChild(div); this.injectTo(div); this.events.wire(div, textarea); }, css: (function () { /* Windows Chromium Browser Emoji Fix * Emoji in Chrome-based browsers! * * Modifies font-family to include fonts that display Emoji * * Windows 7 Users: Install Symbola.ttf if you do not see all Emoji * Find it here: http://users.teilar.gr/~g1951d/ */ if (window.chrome) { // && /(?:win)/i.test(navigator.platform) update.css('.post, .postbit-post, .normal, textarea, input, .gm_emoji, a { font-family: "Helvetica Neue", sans-serif, "Apple Color Emoji", "Android Emoji", "Segoe UI Emoji", "Segoe UI Symbol", Symbola; }'); } }()) }; // Quick Editor Editor = (function () { function Editor(el) { this.id = el.name.split('::')[2]; this.show = 0; this.num = document.querySelector('a[href^="showpost.php?p=' + this.id + '&postcount="]').textContent; //.href.match(/(?:postcount=)(\d+)/)[1]; this.dom(el); Editor.set[this.id] = this; } Editor.set = {}; Editor.toggle = { displays : [ ['block', 'none', 'none'], ['none', 'block', 'inline'] ], spin : ['none', 'inherit'] }; Editor.url = { gets : 'ajax.php?do=quickedit&p=', posts : 'editpost.php?do=updatepost&postid=undefined' }; Editor.fromLink = function (link) { link.textContent += '+'; link.title = 'Quick Editor++'; link.onclick = null; link.dataset.qqre = true; return new Editor(link); }; Editor.fromOldLink = function (link) { return new Editor(link); }; Editor.events = { quickEdit: function (e) { var qe; e.preventDefault(); if (e.target && e.target.hasAttribute('data-qqre')) { qe = Editor.set[e.target.dataset.qqre] || Editor.fromOldLink(e.target); if (qe.x) { qe.toggle(); } else { qe.gets(); } } } }; Editor.prototype = { constuctor: Editor, dom: function (el) { this.el = el; this.el.dataset.qqre = this.id; // this.el.addEventListener('click', Editor.events.quickEdit.bind(this), false); this.spin = document.getElementById('progress_' + this.id); this.post = document.getElementById('post_message_' + this.id); this.div = $.e('div', {className : 'inpost'}, this.post.parentNode); }, gets: function () { this.x = new XMLHttpRequest(); this.x.onprogress = bond(this, this.spinner)(true); this.x.open('POST', Editor.url.gets + this.id, true); this.x.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded'); //charset=UTF-8 this.x.setRequestHeader('X-Requested-With', 'XMLHttpRequest'); this.x.onload = this.load.bind(this); this.x.onerror = this.error.bind(this); this.x.send('securitytoken=' + Form.token() + '&do=quickedit&p=' + this.id + '&editorid=vB_Editor_QE_1'); }, buildPost: function () { return [ 'securitytoken=', Form.token(), '&do=updatepost&ajax=1&postid=', this.id, '&wysiwyg=0&message=', $.uriHtmlEncode(this.div.querySelector('textarea').value), '&reason=', $.uriHtmlEncode(this.div.querySelector('input[name="reason"]').value), '&postcount=', this.num ].join(''); }, posts : function (e) { e.preventDefault(); var s = this.buildPost(); //.call(this); this.p = new XMLHttpRequest(); this.p.open('POST', Editor.url.posts + this.id, true); this.p.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded; charset=UTF-8'); this.p.setRequestHeader('X-Requested-With', 'XMLHttpRequest'); this.p.setRequestHeader('Content-Length', s.length); this.p.onload = this.update.bind(this); this.p.onerror = this.error.bind(this); this.p.send(s); }, toggle : function () { var toggle = Editor.toggle.displays[this.show]; this.el.style.display = toggle[2]; this.div.style.display = toggle[0]; this.post.style.display = toggle[1]; app.state.activeBox = this.show ? app.state.form.elements.message : this.text; this.show = Number(!this.show); }, spinner : function (show) { this.spin.style.display = Editor.toggle.spin[Number(!!show)]; }, load: function () { var t = this.xmlhtml(this.x, 'editor', 'textarea'), f; if (t && t[0]) { f = new Form(this.id, this.div, t[0].value, false, true, t[1]); this.extra = new Extra(this.id, true); this.text = this.div.querySelector('textarea'); this.div.querySelector('.cancel_button').addEventListener('click', this.toggle.bind(this), false); document.getElementById('vB_Editor_' + this.id + '_save').addEventListener('click', this.posts.bind(this), false); update.js(';(function () { $vb_hook(' + this.id + '); }());'); } this.spinner(); this.toggle(); }, update: function () { var post = this.xmlhtml(this.p, 'postbit', '.post'); try { this.post.innerHTML = post[0].innerHTML; this.toggle(); } catch (e) {} //console.log(e); }, error : function () { alert('QQRE: Request failed.'); this.spin.style.display = 'none'; this.x = null; }, xmlhtml : function (attr, b, selector) { // b may be unnecessary try { // console.log(attr, this[attr].responseXML); var div = $.e('div', {innerHTML: attr.responseText}), r = ''; // div.innerHTML = attr.responseText; div = div.querySelector(b); try { r = div.querySelector('editor[reason]').getAttribute('reason'); } catch (er) {} return [div.querySelector(selector), r]; } catch (e) {} } }; return Editor; }()); // Form Maker Form = (function () { function Form(id, el, message, title, edit, reason) { if (!id || !el) { return; } this.id = id; this.el = el; this.message = $.h(message || ''); this.title = $.h(title || ''); this.edit = !!edit; this.reason = $.h(reason || ''); this.elements = {}; this.make(); } Form.PRIMARY = 'quickreplyform_primary'; Form.POST_KEY = 'uQQR.post'; Form.STORAGE_KEY = 'uQQR'; Form.userStorage = strg.read(Form.STORAGE_KEY); Form.threadId = null; Form.postId = null; Form.shared = { elements: { main: document.getElementById('main'), closed: !!document.querySelector('a.large-button.disabled'), logged: document.querySelector('#usercptools'), postbits: function () { return document.querySelectorAll('.postbit:not(.ignored):not([data-qqre])'); }, editButtons: function () { return document.querySelectorAll('a[href*="editpost"]:not([data-qqre])'); } }, scriptVersion: (function () { try { return '?v=' + document.querySelector('script[src*="v="]').src.split('v=')[1]; } catch (e) { return ''; } }()), reg: { '
' : '[indent]', '
[^]+?
([^]+?)<\\/pre>[^]+?<\\/blockquote>' : '[code]$1[/code]',
				'<\\/blockquote>' : '[/indent]',
				'
  • ' : '[*]', '<\\/li>' : '', '' : '[b]', '<\\/b>' : '[/b]', '' : '[i]', '<\\/i>' : '[/i]', '' : '[strike]', '<\\/strike>' : '[/strike]', '' : '[u]', '<\\/u>' : '[/u]', '
      ' : '[list]', '<\\/ul>' : '[/list]\n', '
        ' : '[list=1]', '<\\/ol>' : '[/list]\n', '' : '', '' : '[img]$1[/img]', '([^]+?)<\\/a>' : '[email="$1"]$2[/email]', '' : '[url="$1"]', '' : '[/url]', '([^]+?)<\\/span>' : '$1', '([^]+?)<\\/span>' : '[spoiler]$1[/spoiler]', '([^]+?)<\\/span>' : '[highlight]$1[/highlight]', '' : '', '' : '', '
        ' : '', '<.*?>' : '', '\\[I\'m an idiot\\.\\]' : 'lol' }, buttons: ['removeformat', 'space', 'undo', 'redo', 'space', 'bold', 'italic', 'underline', 'space', 'insertorderedlist', 'insertunorderedlist', 'outdent', 'indent', 'space', 'createlink', 'unlink', 'email', 'insertimage', 'space', 'quote', 'code', 'space', 'resize'], quote: { $replaceImgWithUrlP1: function (text) { return text.replace(/\[img\]/i, 'Link[/url] : [url="'); }, replaceImgWithUrl: function (text) { return text.replace(/\[url\=\".+?\"\]\[img\]/ig, Form.shared.quote.$replaceImgWithUrlP1) .replace(/\[\/img\]\[\/url/ig, '"]Image[/url') .replace(/(?:\[(\/?)img\])/ig, '[$1url]'); }, add: function (post) { if (Form.userStorage.u) { post = Form.shared.quote.replaceImgWithUrl(post); } app.state.activeBox.value += app.state.activeBox.value ? ('\n\n' + post) : post; if (app.state.activeBox.id === app.state.form.elements.message.id) { Form.events.saveStorage(); } }, unquote: function (post) { var div = $.e('div', {innerHTML: post}), e = div.querySelectorAll('.quote:not(.code)'), i = e.length; while (i--) { try { div.removeChild(e[i]); } catch (er) {} } return div.innerHTML; } }, loadIds: function () { try { Form.threadId = +document.querySelector('.left a[href*="showthread.php?t="]').href.match(/t\=(\d+)/)[1]; Form.postId = +document.querySelector('a[href^="newreply.php?do=newreply&noquote=1&p="]').href.match(/p\=(\d+)/)[1]; } catch (e) {} return (Form.threadId && Form.postId); }, tokenFromPage: function () { try { return document.getElementsByTagName('head')[0].textContent.match(/SECURITYTOKEN = "(.+?)"/)[1]; } catch (e) { /*alert('Quick Quote, Reply and Edit encountered an error.'); */ return ''; } }, loadScript: function () { var s = $.e('script', {type: 'text/javascript', src: '/forum/clientscript/vbulletin_textedit.js' + Form.shared.scriptVersion}); s.addEventListener('load', function () { embedScript(function () { window.$vb_hook = function (id) { window.vB_Editor['vB_Editor_' + id] = new window.vB_Text_Editor('vB_Editor_' + id, 0, 3, 1); }; var vbphrase = window.vbphrase || []; vbphrase.enter_list_item = 'Enter a list item.'; vbphrase.message_too_short = 'Your message is too short.'; vbphrase.enter_link_url = 'URL:'; vbphrase.enter_image_url = 'Image URL:'; vbphrase.enter_email_link = 'E-mail:'; window.vbphrase = vbphrase; window.$vb_hook('001'); }); }, false); document.body.appendChild(s); } }; Form.events = { clickDelegate: function (e) { if (e.target && e.target.title === 'Quick Quote') { Form.events.quickQuote(e); } if (e.target && e.target.title === 'Quick Editor++') { Editor.events.quickEdit(e); } }, quickQuote: function (e) { // if (e.target && e.target.title === 'Quick Quote') { e = e.target; var user = e.getAttribute('data-user'), pid = e.getAttribute('data-post-id'), id = e.getAttribute('data-id'), i, re, post = Form.shared.quote.unquote(document.getElementById(pid).innerHTML); for (i in Form.shared.reg) { if (Form.shared.reg.hasOwnProperty(i)) { re = new RegExp(i, 'ig'); post = post.replace(re, Form.shared.reg[i]); } } post = '[quote=' + user + ';' + id + ']' + $.h(post) + '[/quote]\n'; e.className = 'multiquotelink quickquotes quickquoted'; Form.shared.quote.add(post); // } }, submit: function (e) { if (Form.shared.elements.logged) { strg.save(Form.POST_KEY, {threadId: Form.threadId}); } else { e.preventDefault(); document.location.hash = '#'; alert('Please login!'); document.location.hash = '#top'; Form.events.saveStorage(); } }, toggleImageQuotes: function (e) { var k = String.fromCharCode(e.which).toLowerCase(); if (e.altKey && e.shiftKey && (k === 't' || k === '\u02C7')) { e.preventDefault(); Extra.click(app.state.form.elements.convert); alert('Notice: Converting IMG to URL ' + (app.state.form.elements.convert.checked ? 'enabled.' : 'disabled.')); } }, saveStorage: function () { app.model.saveStorage(); }, clear: function () { if (this.value === 'Clear') { app.state.form.elements.title.value = app.state.form.elements.message.value = ''; app.state.form.elements.message.focus(); } app.model.clearStorage(); }, textStore: function (e) { sessionStorage.setItem('qqre-temp', JSON.stringify({ id: Form.threadId, title: app.state.form.elements.title.value, text: e.target.value, z: +(new Date()) })); } }; Form.token = function () { var s = greaseWindow.SECURITYTOKEN; return typeof s === 'string' ? s : ''; }; Form.canAdd = function () { return Form.shared.elements.main && Form.shared.loadIds(); }; Form.removePreviousPost = function () { if (Form.hasJustPosted()) { // console.log('I think you have just posted, so I will clear previous post . . .'); app.model.clearStorage(); } }; Form.hasJustPosted = function () { var id, tid = +strg.grab(Form.POST_KEY).threadId, hash = document.location.hash.substr(1), last = (function (posts) { return posts.length > 0 ? posts[posts.length - 1] : null; }(document.querySelectorAll('.mypost'))); console.log('just posted?'); if (tid === Form.threadId && last) { id = last.id; console.log(id, hash, tid); if (/(?:posted=1)/.test(document.location.href) && id === hash) { console.log('just posted : 1'); return true; } if (/(?:post(\d+))/.test(document.location.hash) && id === hash) { console.log('just posted : 2'); return true; } } return false; }; Form.prototype = { constructor: Form, lastTabIndex: function () { var tabs = document.querySelectorAll('[tabindex]'), i, tabIndex = 0; for (i = 0; i < tabs.length; i++) { if (tabs[i].tabIndex > tabIndex) { tabIndex = tabs[i].tabIndex; } } return tabIndex; }, make: function () { var f = $.e('form', {className: 'quickreplyformp', action: this.id === '001' ? 'newreply.php?post=1' : 'editpost.php', name: 'vbform', method: 'post', onsubmit: 'return vB_Editor["vB_Editor_' + this.id + '"].prepare_submit(0, 1)'}, this.el), d = $.e('div', {className: this.id === '001' ? 'alt1 newreplybox' : 'alt2 newreplybox'}, f), p = $.e('p', null, d), editor = $.e('div', {id: 'vB_Editor_' + this.id, className: 'vBulletin_editor'}, d), container = $.e('div', {innerHTML: '
        '}, d), r = $.e('div', {className: 'quickreplyform_hotspot'}, d), index = this.lastTabIndex() + 1; this.elements.buttons = $.e('div', {className: 'quickreplyform_buttons'}, r); this.elements.message = $.e('textarea', {dir: 'ltr', tabIndex: index, cols: 60, rows: 10, _width: '98%', _height: '150px', id: 'vB_Editor_' + this.id + '_textarea', name: 'message', value: this.message}); this.elements.control = $.e('div', {id: 'vB_Editor_' + this.id + '_controls', className: 'control'}, editor); container.querySelector('.text_emo_container_text').appendChild(this.elements.message); emoji.make(container.querySelector('.text_emo_container_emo'), this.elements.message); if (this.edit) { this.elements.reason = $.e('input', {className: 'biginput', type: 'text', title: 'Optional', maxlength: 125, size: 50, name: 'reason', value: this.reason}, p); $.e('small', {title: 'Reason for editing.', _cursor: 'help', textContent: ' Reason'}, p); } else { this.elements.title = $.e('input', {size: 50, name: 'title', className: 'biginput', type: 'text', value: this.title}, p); $.e('small', {title: 'Optionally, set a title for your post.', _cursor: 'help', textContent: ' Title'}, p); } $.e('input', {type: 'hidden', name: 'securitytoken', id: '', value: Form.token()}, r); $.e('input', {type: 'hidden', name: 'wysiwyg', id: '', value: 0}, r); $.e('input', {type: 'hidden', name: 's', value: ''}, r); $.e('input', {type: 'hidden', name: 'do', value: this.id === '001' ? 'postreply' : 'updatepost'}, r); $.e('input', {type: 'hidden', name: 't', value: Form.threadId}, r); $.e('input', {type: 'hidden', name: 'p', value: this.id === '001' ? Form.postId : this.id}, r); $.e('input', {type: 'hidden', name: 'posthash', value: ''}, r); $.e('input', {type: 'hidden', name: 'poststarttime', value: ''}, r); $.e('input', {type: 'hidden', name: 'parseurl', value: 1}, r); this.elements.submit = $.e('input', {type: 'submit', name: 'sbutton', value: 'Submit', id: 'vB_Editor_' + this.id + '_save', className: 'large-button submit', tabIndex: index, accesskey: 's', title: 'Submit your reply.'}, this.elements.buttons); if (this.edit) { this.elements.cancel = $.e('input', {type: 'button', name: 'cancel', value: 'Cancel', className: 'large-button submit cancel_button', tabIndex: index, accesskey: 'c', title: 'Cancel your reply.'}, this.elements.buttons); } this.elements.preview = $.e('input', {type: 'submit', name: 'preview', value: 'Preview', className: 'large-button submit', tabIndex: index, accesskey: 'p', title: 'Preview your reply.'}, this.elements.buttons); $.e('div', {innerHTML: '

        View ' + (Extra.mac ? '⌘' : 'CTRL') + ' Shortcuts

        '}, r); this.addButtons(); this.elements.body = f; this.elements.hotspot = r; }, makePrimary: function () { this.elements.body.id = Form.PRIMARY; this.elements.save = $.e('input', {type: 'button', name: 'sbutton1', value: 'Save', id: 'quicksavebutton', className: 'large-button submit', tabIndex: this.elements.submit.tabIndex, title: 'Save your reply.', $action: 'save'}); this.elements.buttons.insertBefore(this.elements.save, this.elements.submit.nextSibling); this.elements.clear = $.e('input', {type: 'button', name: 'sbutton2', value: 'Clear', id: 'quickclearbutton', className: 'large-button submit', tabIndex: this.elements.preview.tabIndex, title: 'Clear text.'}, this.elements.buttons); this.elements.temp = this.elements.message; this.elements.subscribe = $.e('div', {innerHTML: '