NeoGAF: Quick Quote, Reply, and Edit

This script adds the ability to quickly quote, reply, and edit posts on NeoGAF. Just click and go.

  1. // ==UserScript==
  2. // @id GAF-QQRE
  3. // @name NeoGAF: Quick Quote, Reply, and Edit
  4. // @namespace hateradio)))
  5. // @author hateradio
  6. // @version 10.1
  7. // @description This script adds the ability to quickly quote, reply, and edit posts on NeoGAF. Just click and go.
  8. // @homepage https://greasyfork.org/scripts/1022-neogaf-quick-quote-reply-and-edit
  9. // @icon 
  10. // @screenshot https://raw.githubusercontent.com/hateradio/qqre/master/screenshot.png
  11.  
  12. // @include http*://*neogaf.com/forum/showthread.php?*
  13. // @include http*://*neogaf.com/forum/newreply.php*
  14. // @include http*://*neogaf.com/forum/editpost.php*
  15. // @include http*://*neogaf.com/forum/newthread.php*
  16. // @include http*://*neogaf.com/forum/subscription.php*
  17. // @include http*://*neogaf.com/forum/showpost.php*
  18. // @include http*://*neogaf.com/forum/private.php*
  19.  
  20. // @match *://*.neogaf.com/forum/showthread.php?*
  21. // @match *://*.neogaf.com/forum/newreply.php*
  22. // @match *://*.neogaf.com/forum/editpost.php*
  23. // @match *://*.neogaf.com/forum/newthread.php*
  24. // @match *://*.neogaf.com/forum/subscription.php*
  25. // @match *://*.neogaf.com/forum/showpost.php*
  26. // @match *://*.neogaf.com/forum/private.php*
  27.  
  28. // @grant GM_log
  29.  
  30. // @updated 04 OCT 2017
  31. // @since 30 OCT 2010
  32. // (c) 2010+, hateradio
  33. // ==/UserScript==
  34.  
  35. /*
  36.  
  37. ## Updates
  38.  
  39. ### 10.1
  40. * **New**: Adds new emoji!
  41. * **Fix**: Inserting emoji will no longer add a space between emoji
  42. * Adds ESLint support
  43.  
  44. ### 10.0.2
  45. * **New**: Adds Grunt for task management -- generates readme.md and update.json
  46. * **Fix**: Safari - Shortcuts now work
  47. * - Note: `CMD+M` and `CMD+H` cannot be overriden. Please use `CTRL+H` and `CTRL+M`
  48. * **Fix**: Safari - Textareas in PMs now display
  49. * Removed extraneous elements and added `placeholder` text to input fields
  50.  
  51. ### 10.0.1
  52. * **Fix**: Search input now matches the style of the site
  53. * **Fix**: Quick Editor now toggling correctly
  54.  
  55. ### 10
  56. * **New**: Emoji toolbar to toggle which emoji to display
  57. * **New**: Adds the `[NOPARSE]` tag -- won't parse any BBCode within it
  58. * **Fix**: Posts without user names won't stop the script execution
  59.  
  60. ### 9.0.2
  61. * **FIX**: HTTPS for NeoGAF
  62.  
  63. ### 9.0.1
  64. * **FIX**: Saved text would appear in the title field instead of the textarea.
  65.  
  66. ### 9.0
  67. * **New**: Keyboard toggle for images in quotes is now ALT+SHIFT+T instead of ALT+T for greater compatibility.
  68. * **New**: Improved shortcuts for Mac (eg, CMD+B for Bold)
  69. * **New**: Temporary storage saves text as it is written.
  70. * **New**: Subscription setting overhauled.
  71. * - 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)
  72. * **New**: Middle clicking on Emoji will link to [CodePoints.net](https://codepoints.net) for information about that character
  73. * **New**: AutoPager & Live Update extension support
  74. * **New**: Informs anonymous users to sign in before submitting posts
  75. * **New**: Resize toggle added to Emoji box
  76. * **New**: Tabbing will send the focus from the textarea to the Submit button
  77. * **New**: Routing mechanism to load the different code on the different pages
  78. * **New**: New Form class to create Quick Reply and Edit forms
  79. * **New**: Character escaping mechanism to convert JavaScript characters to HTML entities for AJAX transmissions; removes VB method
  80. * **Fix**: Removed vertical alignment of inputs
  81. * **Fix**: Posts that have been saved now clear after the post has been processed
  82. * Renames "Delete" button with "Clear"
  83. * Restructure of the code base
  84.  
  85. ### 8.0
  86.  
  87. * **New**: Emoji picker!
  88. * **New**: (Windows) Emoji support to Chrome-based browsers!
  89. * - _Note 1_: Segoe UI Symbol, Segoe UI Emoji and Symbola added to the font list
  90. * - _Note 2_: This may change default fonts a bit
  91. * - _Note 3_: Windows 7 users who have issues displaying Emoji should install [Symbola](http://users.teilar.gr/~g1951d/)
  92. * **New**: USC-2 decoder and entity encoder to work with forum AJAX to transmit Emoji data
  93. * **Fix**: Removing recursive quotes no longer causes an error
  94.  
  95. ### 7.4.1
  96.  
  97. * Chrome fix for greaseWindow
  98.  
  99. ### 7.4
  100.  
  101. * Fixes quotes when user has ignored others
  102. * Fixes for getting user key
  103.  
  104. ### 7.3
  105.  
  106. * Fixes update notification
  107. * Closed threads do not show the reply box
  108. * Changes to form to match new dark theme
  109. * Improves quick quote text replacements
  110.  
  111. ### 7.1 - 7.2
  112.  
  113. * Changes for Chrome extension
  114. * Changes to form to match new dark theme
  115. * Improves quick quote text replacements
  116.  
  117. ### 7.0
  118.  
  119. * Fix for new site
  120. * Fix for quick reply AJAX response (no more `responseXML`)
  121.  
  122. ### 6.3
  123.  
  124. * Chrome 27+, no more `window` access
  125. * Added message listener
  126. * Chanced to dropboxusercontent.com, since it's now used by dropbox
  127.  
  128. ### 6.1.2 - 6.2
  129.  
  130. * Chrome - button issues, again
  131.  
  132. ### 6.1
  133.  
  134. * Chrome - button issues
  135.  
  136. ### 6.0
  137.  
  138. * Removes quick reply bar, now uses the native button to create the quick editor
  139. * The editor starts when pressing the Edit+ link
  140. * Incorporates some of the site's native AJAX functionality
  141. * [IMG] to [URL] replacement now creates "Link : Image" when an image is inside a link
  142. * Fixes integration with Smilies script, data image links will be removed
  143. * Updates the Extra reply button code
  144. * Enhances compatibility with Opera
  145. * Adds keyboad shortcut list
  146. * Adds keyboard shortcut (`ALT+T`) for toggling "Auto-convert IMG to URL tags" checkbox
  147.  
  148. (DLs: 4,156)
  149.  
  150. // #Updates
  151. // todo implement preview
  152.  
  153.  
  154. */
  155.  
  156. (function () {
  157. 'use strict';
  158.  
  159. var $, greaseWindow, strg, update, Extra, Editor, Form, app, bond, emoji, dom;
  160.  
  161. if (!String.prototype.trim) { String.prototype.trim = function () { return this.replace(/^\s+|\s+$/g, ''); }; }
  162. if (!Array.from) { Array.from = function (list) { return Array.prototype.slice.call(list); }; }
  163.  
  164. function throttle(f, delay) {
  165. var timer = null;
  166. return function () {
  167. var context = this, args = arguments;
  168. clearTimeout(timer);
  169. timer = window.setTimeout(function () {
  170. f.apply(context, args);
  171. }, delay || 1000);
  172. };
  173. }
  174.  
  175. bond = function (o, m) { return function () { return m.apply(o, arguments); }; };
  176.  
  177. dom = {
  178. top: document.head || document.body,
  179. css: function (t) {
  180. if (!this.style) {
  181. this.style = document.createElement('style');
  182. this.style.type = 'text/css';
  183. this.top.appendChild(this.style);
  184. }
  185. this.style.appendChild(document.createTextNode(t + '\n'));
  186. },
  187. js: function (t) {
  188. var j = document.createElement('script');
  189. j.type = 'text/javascript';
  190. j[/^https?:\/\//i.test(t) ? 'src' : 'textContent'] = t;
  191. this.top.appendChild(j);
  192. }
  193. };
  194.  
  195. // UCS-2 Decoder https://github.com/bestiejs/punycode.js
  196. function ucs2decode(string) {
  197. var output = [],
  198. counter = 0,
  199. length = string.length,
  200. value,
  201. extra;
  202. while (counter < length) {
  203. value = string.charCodeAt(counter++);
  204. if ((value & 0xF800) === 0xD800 && counter < length) {
  205. // high surrogate, and there is a next character
  206. extra = string.charCodeAt(counter++);
  207. if ((extra & 0xFC00) === 0xDC00) { // low surrogate
  208. output.push(((value & 0x3FF) << 10) + (extra & 0x3FF) + 0x10000);
  209. } else {
  210. output.push(value, extra);
  211. }
  212. } else {
  213. output.push(value);
  214. }
  215. }
  216. return output;
  217. }
  218.  
  219. // Window | No longer works in Chrome 27+
  220. greaseWindow = {};
  221.  
  222. function embedScript(func) {
  223. var script = document.createElement('script');
  224. script.type = 'application/javascript';
  225. script.textContent = '(' + func + ')();';
  226. document.body.appendChild(script);
  227. document.body.removeChild(script);
  228. }
  229.  
  230. // S T O R A G E HANDLE
  231. strg = {
  232. on: (function () { try { var s = window.localStorage; return (s.getItem && s.setItem && s.removeItem); } catch (e) { return false; } }()),
  233. read: function (key) { return this.on ? JSON.parse(window.localStorage.getItem(key)) : false; },
  234. grab: function (key) { return this.read(key) || {}; },
  235. save: function (key, dat) { return this.on ? !window.localStorage.setItem(key, JSON.stringify(dat)) : false; },
  236. wipe: function (key) { return this.on ? !window.localStorage.removeItem(key) : false; },
  237. zero: function (obj) { var k; for (k in obj) { if (obj.hasOwnProperty(k)) { return false; } } return true; }
  238. };
  239.  
  240. // U P D A T E HANDLE
  241. update = {
  242. name: 'NeoGAF: Quick Quote, Reply, and Edit',
  243. version: 10100,
  244. key: 'ujs_QQRE_UPDT',
  245. callback: 'qqreupdater',
  246. page: 'https://greasyfork.org/scripts/1022-neogaf-quick-quote-reply-and-edit',
  247. urij: 'https://hateradio.github.io/qqre/update.json',
  248. interval: 5,
  249. day: (new Date()).getTime(),
  250. time: function () { return new Date(this.day + (1000 * 60 * 60 * 24 * this.interval)).getTime(); },
  251. notification: function (j) {
  252. if (this.version < j.version) {
  253. strg.save(this.key, { date: this.time(), version: j.version, page: j.page });
  254. this.link();
  255. }
  256. },
  257. link: function () {
  258. this.csstxt();
  259.  
  260. var a = document.createElement('a'), b = strg.read(this.key);
  261. a.href = b.page || '#';
  262. a.id = 'userscriptupdater2';
  263. a.title = 'Update now.';
  264. a.target = '_blank';
  265. a.textContent = 'An update for ' + this.name + ' is available.';
  266. a.addEventListener('click', function () { this.style.display = 'none'; }, false);
  267. document.body.appendChild(a);
  268. },
  269. xhr: function () {
  270. var x = new XMLHttpRequest();
  271. x.addEventListener('load', function () { update.notification(JSON.parse(this.responseText)); }, false);
  272. x.open('get', update.urij, true);
  273. x.send();
  274. },
  275. check: function (opt) {
  276. if (!strg.on) { return; }
  277. var stored = strg.read(this.key), page;
  278.  
  279. if (opt || !stored || stored.date < this.day) {
  280. page = (stored && stored.page) || '#';
  281. strg.save(this.key, {date: this.time(), version: this.version, page: page});
  282. this.xhr();
  283. } else if (this.version < stored.version) {
  284. this.link();
  285. }
  286. },
  287. csstxt: function () {
  288. if (!this.pop) { this.pop = true; dom.css('#userscriptupdater2,#userscriptupdater2:visited{box-shadow:1px 1px 6px #7776;border-bottom:3px solid #d65e55;cursor:pointer;color:#555;font-family:sans-serif;font-size:12px;font-weight:700;text-align:justify;position:fixed;z-index:999999;right:10px;top:10px;background:#ebebeb url() no-repeat 10px center;background-size:40px;padding:0 20px 0 60px;height:55px;line-height:55px}#userscriptupdater2:hover,#userscriptupdater2:visited:hover{color:#b33a3a !important;border-color:#ce4b30}'); }
  289. }
  290. };
  291. update.check();
  292.  
  293. // DOM/HTML Helpers
  294. $ = {
  295. 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; },
  296. 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; },
  297. t: document.createElement('textarea'),
  298. h: function (t) { this.t.innerHTML = t; return this.t.value.trim(); },
  299. htmlEncode: function (text) {
  300. var chars = ucs2decode(text), c, i, j = chars.length, str = [];
  301.  
  302. for (i = 0; i < j; i++) {
  303. c = chars[i];
  304. str.push(c >= 127 ? ('&#' + c + ';') : String.fromCharCode(c));
  305. }
  306.  
  307. return str.join('');
  308. },
  309. uriHtmlEncode: function (text) { return encodeURIComponent(this.htmlEncode(text)); }
  310. };
  311.  
  312. // Extra Reply Buttons
  313. Extra = (function () {
  314. function Extra(id, custom) {
  315. id = id || '001';
  316. this.editor = 'vB_Editor_' + id;
  317. this.custum = !!custom;
  318. this.txt = document.getElementById(this.editor + '_textarea');
  319. if (this.txt) {
  320. this.src = this.custum ? document.getElementById(this.editor + '_cmd_underline') : document.getElementById(this.editor + '_cmd_underline').parentNode;
  321. this.BB = {};
  322. this.add();
  323. this.setup();
  324. }
  325. }
  326.  
  327. Extra.buttons = ['Spoiler', 'Highlight', 'Strike', 'NoParse'];
  328. Extra.images = [
  329. '/forum/images/neogaf2/misc/spoiler.gif',
  330. '',
  331. '',
  332. ''
  333. ];
  334. Extra.ctrl = { b: 'b', i: 'i', u: 'u', s: 'Strike', h: 'Highlight', p: 'Spoiler', l: 'url', m: 'img', e: 'save', d: 'clear' };
  335. Extra.mac = /(?:mac)/i.test(navigator.platform);
  336. Extra.moz = window.InstallTrigger !== undefined;
  337. Extra.keyEvent = Extra.moz ? 'keypress' : 'keydown';
  338. Extra.click = function (el) {
  339. try {
  340. el.click();
  341. } catch (e) {
  342. var evt = document.createEvent('MouseEvents');
  343. evt.initMouseEvent('click', true, true, window, 0, 0, 0, 0, 0, false, false, false, false, 0, null);
  344. // window.setTimeout(function () {
  345. el.dispatchEvent(evt);
  346. // }, 1);
  347. }
  348. };
  349.  
  350. Extra.prototype = {
  351. constructor: Extra,
  352. insert: function (evt) { // based on: http://parentnode.org/javascript/working-with-the-cursor-position/ // gecko parts only
  353. var t = evt.target.getAttribute('alt').toUpperCase(), start, end, pos, temp;
  354. if (this.txt.selectionStart >= 0) {
  355. start = this.txt.selectionStart || 0;
  356. end = this.txt.selectionEnd || 0;
  357. temp = this.txt.value.substr(0, start) + '[' + t + ']' + this.txt.value.substr(start, end - start);
  358. pos = temp.length;
  359. this.txt.value = temp + '[/' + t + ']' + this.txt.value.substr(end, this.txt.value.length);
  360. this.txt.focus();
  361. start = start + 2 + t.length;
  362. this.txt.setSelectionRange(start, pos);
  363. }
  364. },
  365. add: function () {
  366. var i = Extra.buttons.length, div, td;
  367. while (i--) {
  368. div = $.e('div', {className: 'imagebutton', title: 'new'});
  369. this.BB[Extra.buttons[i]] = $.e('img', {height: 20, width: 21, src: Extra.images[i], alt: Extra.buttons[i], title: Extra.buttons[i]}, div);
  370. this.BB[Extra.buttons[i]].addEventListener('click', this.insert.bind(this), false);
  371. if (this.custum) {
  372. this.src.parentNode.insertBefore(div, this.src.nextSibling);
  373. } else {
  374. td = $.e('td', {className: 'imgbtnemu'}, div, true);
  375. this.src.parentNode.insertBefore(td, this.src.nextSibling);
  376. }
  377. }
  378. },
  379. keys: function (e) {
  380. if (!e.shiftKey && (e.ctrlKey || e.metaKey)) {
  381. // console.log('keys--which', e.which, e);
  382. var k = Extra.ctrl[String.fromCharCode(e.which).toLowerCase()];
  383. // console.log('k', k);
  384. if (k && (this.BB[k] || (e.metaKey && this.BB.meta[k]))) {
  385. // console.log('keys found');
  386. e.preventDefault();
  387. Extra.click(this.BB[k] || this.BB.meta[k]);
  388. }
  389. }
  390. },
  391. setup: function () {
  392. this.BB.meta = {};
  393. this.BB.b = document.getElementById(this.editor + '_cmd_bold');
  394. this.BB.i = document.getElementById(this.editor + '_cmd_italic');
  395. this.BB.u = document.getElementById(this.editor + '_cmd_underline');
  396. this.BB.img = document.getElementById(this.editor + '_cmd_insertimage');
  397. this.BB.url = document.getElementById(this.editor + '_cmd_createlink');
  398.  
  399. this.BB.meta.b = this.BB.b;
  400. this.BB.meta.i = this.BB.i;
  401. this.BB.meta.u = this.BB.u;
  402.  
  403. if (Extra.moz) {
  404. this.BB.b = this.BB.i = this.BB.u = null;
  405. }
  406.  
  407. this.txt.addEventListener(Extra.keyEvent, this.keys.bind(this), false);
  408. }
  409. };
  410. return Extra;
  411. }());
  412. dom.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:64%} .text_emo_container .text_emo_container_emo {width:27%;min-width:180px;margin: 0 0 0 1%;display:inline-block;overflow:hidden;font-size:13px} .text_emo_container_text textarea {width:99%} .gm_emoji {line-height: 1.3;overflow: hidden;resize:both} .gm_emoji a {font-size:13px;line-height:1.3;cursor:pointer} .gm_emoji_list {display:inline} .gm_emoji_box {height: 100px; overflow: auto; } .gm_emoji_toolbar { margin-bottom: 4px } .gm_emoji_search{margin:5px 0;box-sizing:border-box;width:95%}');
  413.  
  414. // Emoji Stuff <3 OSX 10.9+ Command+Control+Space
  415. // http://konieczny.be/article/utf.php
  416. // http://apps.timwhitlock.info/emoji/tables/unicode
  417. emoji = {
  418. div: null,
  419. ranges: /(?:[\u2700-\u27bf]|(?:\ud83c[\udde6-\uddff]){2}|[\ud800-\udbff][\udc00-\udfff]|[\u0023-\u0039]\ufe0f?\u20e3|\u3299|\u3297|\u303d|\u3030|\u24c2|\ud83c[\udd70-\udd71]|\ud83c[\udd7e-\udd7f]|\ud83c\udd8e|\ud83c[\udd91-\udd9a]|\ud83c[\udde6-\uddff]|[\ud83c[\ude01-\ude02]|\ud83c\ude1a|\ud83c\ude2f|[\ud83c[\ude32-\ude3a]|[\ud83c[\ude50-\ude51]|\u203c|\u2049|[\u25aa-\u25ab]|\u25b6|\u25c0|[\u25fb-\u25fe]|\u00a9|\u00ae|\u2122|\u2139|\ud83c\udc04|[\u2600-\u26FF]|\u2b05|\u2b06|\u2b07|\u2b1b|\u2b1c|\u2b50|\u2b55|\u231a|\u231b|\u2328|\u23cf|[\u23e9-\u23f3]|[\u23f8-\u23fa]|\ud83c\udccf|\u2934|\u2935|[\u2190-\u21ff])/g,
  420. dbGenerator: function (ranges) {
  421. var list = [], i;
  422.  
  423. ranges.forEach(function (range) {
  424. for (i = range[0]; i <= range[1]; i++) {
  425. list.push(i.toString(16));
  426. }
  427. });
  428.  
  429. return list;
  430. },
  431. presentation: {
  432. nameReg: /([A-Z])/g,
  433. readableName: function (name) {
  434. var title = name.replace(emoji.presentation.nameReg, ' & $1');
  435. title = title[0].toUpperCase() + title.slice(1);
  436.  
  437. return title;
  438. },
  439. toolbarIcon: function (points, name) {
  440. var b = document.createElement('div'), point;
  441.  
  442. if (typeof points === 'string' || points instanceof String) {
  443. point = points;
  444. } else {
  445. point = points[0];
  446. }
  447.  
  448. b.innerHTML = '&#x' + point + ';';
  449. b.setAttribute('data-toolbar', name);
  450. b.title = emoji.presentation.readableName(name);
  451. b.className = 'gm_emoji_toolbar_button';
  452. return b;
  453. },
  454. icon: function (points) {
  455. var a = document.createElement('a'), title, point;
  456.  
  457. if (typeof points === 'string' || points instanceof String) {
  458. point = points;
  459. title = 'U+' + points;
  460. } else {
  461. point = points[0];
  462. title = points[1];
  463. }
  464.  
  465. a.innerHTML = '&#x' + point + ';';
  466. a.setAttribute('data-char', a.innerHTML);
  467. a.title = title;
  468. a.href = 'https://codepoints.net/U+' + point;
  469. return a;
  470. },
  471. panel: function (div) {
  472. var e, i, l, f = document.createDocumentFragment(), innerDiv, innerFragment, toolbar, search, emos, hide = false;
  473.  
  474. toolbar = document.createElement('div');
  475. toolbar.className = 'gm_emoji_toolbar';
  476.  
  477. search = document.createElement('input');
  478. search.setAttribute('type', 'text');
  479. search.className = 'biginput gm_emoji_search';
  480. search.placeholder = 'Search';
  481.  
  482. emos = document.createElement('div');
  483. emos.className = 'gm_emoji_box';
  484.  
  485. for (e in emoji.db) {
  486. if (emoji.db.hasOwnProperty(e)) {
  487. innerFragment = document.createDocumentFragment();
  488.  
  489. innerDiv = document.createElement('div');
  490.  
  491. if (hide) {
  492. innerDiv.className = 'gm_emoji_list gm_emoji_hidden gm_emoji_' + e;
  493. } else {
  494. hide = true;
  495. innerDiv.className = 'gm_emoji_list gm_emoji_' + e;
  496. }
  497.  
  498. l = emoji.db[e];
  499. for (i = 0; i < l.length; i++) {
  500. if (i === 0) {
  501. toolbar.appendChild(emoji.presentation.toolbarIcon(l[i], e));
  502. toolbar.appendChild(document.createTextNode(' '));
  503. }
  504. innerFragment.appendChild(emoji.presentation.icon(l[i]));
  505. innerFragment.appendChild(document.createTextNode(' '));
  506. }
  507. }
  508.  
  509. innerDiv.appendChild(innerFragment);
  510. f.appendChild(innerDiv);
  511. }
  512. div.appendChild(toolbar);
  513. div.appendChild(search);
  514. emos.appendChild(f);
  515. div.appendChild(emos);
  516. toolbar.querySelector('.gm_emoji_toolbar_button').classList.add('gm_emoji_selected');
  517.  
  518. search.addEventListener(Extra.keyEvent, function (e) {
  519. if (e.keyCode === 13) {
  520. e.preventDefault();
  521. e.stopPropagation();
  522. }
  523. }, true);
  524. search.addEventListener(Extra.keyEvent, throttle(emoji.events.search, 500), false);
  525. }
  526. },
  527. db: {
  528. smileysPeople: [['1F600', 'Grinning Face'], ['1F601', 'Grinning Face With Smiling Eyes'], ['1F602', 'Face With Tears of Joy'], ['1F923', 'Rolling on the Floor Laughing'], ['1F603', 'Smiling Face With Open Mouth'], ['1F604', 'Smiling Face With Open Mouth & Smiling Eyes'], ['1F605', 'Smiling Face With Open Mouth & Cold Sweat'], ['1F606', 'Smiling Face With Open Mouth & Closed Eyes'], ['1F609', 'Winking Face'], ['1F60A', 'Smiling Face With Smiling Eyes'], ['1F60B', 'Face Savouring Delicious Food'], ['1F60E', 'Smiling Face With Sunglasses'], ['1F60D', 'Smiling Face With Heart-Eyes'], ['1F618', 'Face Blowing a Kiss'], ['1F617', 'Kissing Face'], ['1F619', 'Kissing Face With Smiling Eyes'], ['1F61A', 'Kissing Face With Closed Eyes'], ['263A', 'Smiling Face'], ['1F642', 'Slightly Smiling Face'], ['1F917', 'Hugging Face'], ['1F914', 'Thinking Face'], ['1F610', 'Neutral Face'], ['1F611', 'Expressionless Face'], ['1F636', 'Face Without Mouth'], ['1F644', 'Face With Rolling Eyes'], ['1F60F', 'Smirking Face'], ['1F623', 'Persevering Face'], ['1F625', 'Disappointed but Relieved Face'], ['1F62E', 'Face With Open Mouth'], ['1F910', 'Zipper-Mouth Face'], ['1F62F', 'Hushed Face'], ['1F62A', 'Sleepy Face'], ['1F62B', 'Tired Face'], ['1F634', 'Sleeping Face'], ['1F60C', 'Relieved Face'], ['1F61B', 'Face With Stuck-Out Tongue'], ['1F61C', 'Face With Stuck-Out Tongue & Winking Eye'], ['1F61D', 'Face With Stuck-Out Tongue & Closed Eyes'], ['1F924', 'Drooling Face'], ['1F612', 'Unamused Face'], ['1F613', 'Face With Cold Sweat'], ['1F614', 'Pensive Face'], ['1F615', 'Confused Face'], ['1F643', 'Upside-Down Face'], ['1F911', 'Money-Mouth Face'], ['1F632', 'Astonished Face'], ['2639', 'Frowning Face'], ['1F641', 'Slightly Frowning Face'], ['1F616', 'Confounded Face'], ['1F61E', 'Disappointed Face'], ['1F61F', 'Worried Face'], ['1F624', 'Face With Steam From Nose'], ['1F622', 'Crying Face'], ['1F62D', 'Loudly Crying Face'], ['1F626', 'Frowning Face With Open Mouth'], ['1F627', 'Anguished Face'], ['1F628', 'Fearful Face'], ['1F629', 'Weary Face'], ['1F62C', 'Grimacing Face'], ['1F630', 'Face With Open Mouth & Cold Sweat'], ['1F631', 'Face Screaming in Fear'], ['1F633', 'Flushed Face'], ['1F635', 'Dizzy Face'], ['1F621', 'Pouting Face'], ['1F620', 'Angry Face'], ['1F637', 'Face With Medical Mask'], ['1F912', 'Face With Thermometer'], ['1F915', 'Face With Head-Bandage'], ['1F922', 'Nauseated Face'], ['1F927', 'Sneezing Face'], ['1F607', 'Smiling Face With Halo'], ['1F920', 'Cowboy Hat Face'], ['1F921', 'Clown Face'], ['1F925', 'Lying Face'], ['1F913', 'Nerd Face'], ['1F608', 'Smiling Face With Horns'], ['1F47F', 'Angry Face With Horns'], ['1F479', 'Ogre'], ['1F47A', 'Goblin'], ['1F480', 'Skull'], ['1F47B', 'Ghost'], ['1F47D', 'Alien'], ['1F916', 'Robot Face'], ['1F4A9', 'Pile of Poo'], ['1F63A', 'Smiling Cat Face With Open Mouth'], ['1F638', 'Grinning Cat Face With Smiling Eyes'], ['1F639', 'Cat Face With Tears of Joy'], ['1F63B', 'Smiling Cat Face With Heart-Eyes'], ['1F63C', 'Cat Face With Wry Smile'], ['1F63D', 'Kissing Cat Face With Closed Eyes'], ['1F640', 'Weary Cat Face'], ['1F63F', 'Crying Cat Face'], ['1F63E', 'Pouting Cat Face'], ['1F476', 'Baby'], ['1F466', 'Boy'], ['1F467', 'Girl'], ['1F468', 'Man'], ['1F469', 'Woman'], ['1F474', 'Old Man'], ['1F475', 'Old Woman'], ['1F468;&#x200D;&#x2695;&#xFE0F', 'Man Health Worker'], ['1F469;&#x200D;&#x2695;&#xFE0F', 'Woman Health Worker'], ['1F468;&#x200D;&#x1F393', 'Man Student'], ['1F469;&#x200D;&#x1F393', 'Woman Student'], ['1F468;&#x200D;&#x2696;&#xFE0F', 'Man Judge'], ['1F469;&#x200D;&#x2696;&#xFE0F', 'Woman Judge'], ['1F468;&#x200D;&#x1F33E', 'Man Farmer'], ['1F469;&#x200D;&#x1F33E', 'Woman Farmer'], ['1F468;&#x200D;&#x1F373', 'Man Cook'], ['1F469;&#x200D;&#x1F373', 'Woman Cook'], ['1F468;&#x200D;&#x1F527', 'Man Mechanic'], ['1F469;&#x200D;&#x1F527', 'Woman Mechanic'], ['1F468;&#x200D;&#x1F3ED', 'Man Factory Worker'], ['1F469;&#x200D;&#x1F3ED', 'Woman Factory Worker'], ['1F468;&#x200D;&#x1F4BC', 'Man Office Worker'], ['1F469;&#x200D;&#x1F4BC', 'Woman Office Worker'], ['1F468;&#x200D;&#x1F52C', 'Man Scientist'], ['1F469;&#x200D;&#x1F52C', 'Woman Scientist'], ['1F468;&#x200D;&#x1F4BB', 'Man Technologist'], ['1F469;&#x200D;&#x1F4BB', 'Woman Technologist'], ['1F468;&#x200D;&#x1F3A4', 'Man Singer'], ['1F469;&#x200D;&#x1F3A4', 'Woman Singer'], ['1F468;&#x200D;&#x1F3A8', 'Man Artist'], ['1F469;&#x200D;&#x1F3A8', 'Woman Artist'], ['1F468;&#x200D;&#x2708;&#xFE0F', 'Man Pilot'], ['1F469;&#x200D;&#x2708;&#xFE0F', 'Woman Pilot'], ['1F468;&#x200D;&#x1F680', 'Man Astronaut'], ['1F469;&#x200D;&#x1F680', 'Woman Astronaut'], ['1F468;&#x200D;&#x1F692', 'Man Firefighter'], ['1F469;&#x200D;&#x1F692', 'Woman Firefighter'], ['1F46E', 'Police Officer'], ['1F46E;&#x200D;&#x2642;&#xFE0F', 'Man Police Officer'], ['1F46E;&#x200D;&#x2640;&#xFE0F', 'Woman Police Officer'], ['1F575', 'Detective'], ['1F575;&#xFE0F;&#x200D;&#x2642;&#xFE0F', 'Man Detective'], ['1F575;&#xFE0F;&#x200D;&#x2640;&#xFE0F', 'Woman Detective'], ['1F482', 'Guard'], ['1F482;&#x200D;&#x2642;&#xFE0F', 'Man Guard'], ['1F482;&#x200D;&#x2640;&#xFE0F', 'Woman Guard'], ['1F477', 'Construction Worker'], ['1F477;&#x200D;&#x2642;&#xFE0F', 'Man Construction Worker'], ['1F477;&#x200D;&#x2640;&#xFE0F', 'Woman Construction Worker'], ['1F934', 'Prince'], ['1F478', 'Princess'], ['1F473', 'Person Wearing Turban'], ['1F473;&#x200D;&#x2642;&#xFE0F', 'Man Wearing Turban'], ['1F473;&#x200D;&#x2640;&#xFE0F', 'Woman Wearing Turban'], ['1F472', 'Man With Chinese Cap'], ['1F471', 'Blond-Haired Person'], ['1F471;&#x200D;&#x2642;&#xFE0F', 'Blond-Haired Man'], ['1F471;&#x200D;&#x2640;&#xFE0F', 'Blond-Haired Woman'], ['1F935', 'Man in Tuxedo'], ['1F470', 'Bride With Veil'], ['1F930', 'Pregnant Woman'], ['1F47C', 'Baby Angel'], ['1F385', 'Santa Claus'], ['1F936', 'Mrs. Claus'], ['1F64D', 'Person Frowning'], ['1F64D;&#x200D;&#x2642;&#xFE0F', 'Man Frowning'], ['1F64D;&#x200D;&#x2640;&#xFE0F', 'Woman Frowning'], ['1F64E', 'Person Pouting'], ['1F64E;&#x200D;&#x2642;&#xFE0F', 'Man Pouting'], ['1F64E;&#x200D;&#x2640;&#xFE0F', 'Woman Pouting'], ['1F645', 'Person Gesturing No'], ['1F645;&#x200D;&#x2642;&#xFE0F', 'Man Gesturing No'], ['1F645;&#x200D;&#x2640;&#xFE0F', 'Woman Gesturing No'], ['1F646', 'Person Gesturing OK'], ['1F646;&#x200D;&#x2642;&#xFE0F', 'Man Gesturing OK'], ['1F646;&#x200D;&#x2640;&#xFE0F', 'Woman Gesturing OK'], ['1F481', 'Person Tipping Hand'], ['1F481;&#x200D;&#x2642;&#xFE0F', 'Man Tipping Hand'], ['1F481;&#x200D;&#x2640;&#xFE0F', 'Woman Tipping Hand'], ['1F64B', 'Person Raising Hand'], ['1F64B;&#x200D;&#x2642;&#xFE0F', 'Man Raising Hand'], ['1F64B;&#x200D;&#x2640;&#xFE0F', 'Woman Raising Hand'], ['1F647', 'Person Bowing'], ['1F647;&#x200D;&#x2642;&#xFE0F', 'Man Bowing'], ['1F647;&#x200D;&#x2640;&#xFE0F', 'Woman Bowing'], ['1F926', 'Person Facepalming'], ['1F926;&#x200D;&#x2642;&#xFE0F', 'Man Facepalming'], ['1F926;&#x200D;&#x2640;&#xFE0F', 'Woman Facepalming'], ['1F937', 'Person Shrugging'], ['1F937;&#x200D;&#x2642;&#xFE0F', 'Man Shrugging'], ['1F937;&#x200D;&#x2640;&#xFE0F', 'Woman Shrugging'], ['1F486', 'Person Getting Massage'], ['1F486;&#x200D;&#x2642;&#xFE0F', 'Man Getting Massage'], ['1F486;&#x200D;&#x2640;&#xFE0F', 'Woman Getting Massage'], ['1F487', 'Person Getting Haircut'], ['1F487;&#x200D;&#x2642;&#xFE0F', 'Man Getting Haircut'], ['1F487;&#x200D;&#x2640;&#xFE0F', 'Woman Getting Haircut'], ['1F6B6', 'Person Walking'], ['1F6B6;&#x200D;&#x2642;&#xFE0F', 'Man Walking'], ['1F6B6;&#x200D;&#x2640;&#xFE0F', 'Woman Walking'], ['1F3C3', 'Person Running'], ['1F3C3;&#x200D;&#x2642;&#xFE0F', 'Man Running'], ['1F3C3;&#x200D;&#x2640;&#xFE0F', 'Woman Running'], ['1F483', 'Woman Dancing'], ['1F57A', 'Man Dancing'], ['1F46F', 'People With Bunny Ears Partying'], ['1F46F;&#x200D;&#x2642;&#xFE0F', 'Men With Bunny Ears Partying'], ['1F46F;&#x200D;&#x2640;&#xFE0F', 'Women With Bunny Ears Partying'], ['1F574', 'Man in Business Suit Levitating'], ['1F5E3', 'Speaking Head'], ['1F464', 'Bust in Silhouette'], ['1F465', 'Busts in Silhouette'], ['1F46B', 'Man and Woman Holding Hands'], ['1F46C', 'Two Men Holding Hands'], ['1F46D', 'Two Women Holding Hands'], ['1F48F', 'Kiss'], ['1F468;&#x200D;&#x2764;&#xFE0F;&#x200D;&#x1F48B;&#x200D;&#x1F468', 'Kiss: Man, Man'], ['1F469;&#x200D;&#x2764;&#xFE0F;&#x200D;&#x1F48B;&#x200D;&#x1F469', 'Kiss: Woman, Woman'], ['1F491', 'Couple With Heart'], ['1F468;&#x200D;&#x2764;&#xFE0F;&#x200D;&#x1F468', 'Couple With Heart: Man, Man'], ['1F469;&#x200D;&#x2764;&#xFE0F;&#x200D;&#x1F469', 'Couple With Heart: Woman, Woman'], ['1F46A', 'Family'], ['1F468;&#x200D;&#x1F469;&#x200D;&#x1F466', 'Family: Man, Woman, Boy'], ['1F468;&#x200D;&#x1F469;&#x200D;&#x1F467', 'Family: Man, Woman, Girl'], ['1F468;&#x200D;&#x1F469;&#x200D;&#x1F467;&#x200D;&#x1F466', 'Family: Man, Woman, Girl, Boy'], ['1F468;&#x200D;&#x1F469;&#x200D;&#x1F466;&#x200D;&#x1F466', 'Family: Man, Woman, Boy, Boy'], ['1F468;&#x200D;&#x1F469;&#x200D;&#x1F467;&#x200D;&#x1F467', 'Family: Man, Woman, Girl, Girl'], ['1F468;&#x200D;&#x1F468;&#x200D;&#x1F466', 'Family: Man, Man, Boy'], ['1F468;&#x200D;&#x1F468;&#x200D;&#x1F467', 'Family: Man, Man, Girl'], ['1F468;&#x200D;&#x1F468;&#x200D;&#x1F467;&#x200D;&#x1F466', 'Family: Man, Man, Girl, Boy'], ['1F468;&#x200D;&#x1F468;&#x200D;&#x1F466;&#x200D;&#x1F466', 'Family: Man, Man, Boy, Boy'], ['1F468;&#x200D;&#x1F468;&#x200D;&#x1F467;&#x200D;&#x1F467', 'Family: Man, Man, Girl, Girl'], ['1F469;&#x200D;&#x1F469;&#x200D;&#x1F466', 'Family: Woman, Woman, Boy'], ['1F469;&#x200D;&#x1F469;&#x200D;&#x1F467', 'Family: Woman, Woman, Girl'], ['1F469;&#x200D;&#x1F469;&#x200D;&#x1F467;&#x200D;&#x1F466', 'Family: Woman, Woman, Girl, Boy'], ['1F469;&#x200D;&#x1F469;&#x200D;&#x1F466;&#x200D;&#x1F466', 'Family: Woman, Woman, Boy, Boy'], ['1F469;&#x200D;&#x1F469;&#x200D;&#x1F467;&#x200D;&#x1F467', 'Family: Woman, Woman, Girl, Girl'], ['1F468;&#x200D;&#x1F466', 'Family: Man, Boy'], ['1F468;&#x200D;&#x1F466;&#x200D;&#x1F466', 'Family: Man, Boy, Boy'], ['1F468;&#x200D;&#x1F467', 'Family: Man, Girl'], ['1F468;&#x200D;&#x1F467;&#x200D;&#x1F466', 'Family: Man, Girl, Boy'], ['1F468;&#x200D;&#x1F467;&#x200D;&#x1F467', 'Family: Man, Girl, Girl'], ['1F469;&#x200D;&#x1F466', 'Family: Woman, Boy'], ['1F469;&#x200D;&#x1F466;&#x200D;&#x1F466', 'Family: Woman, Boy, Boy'], ['1F469;&#x200D;&#x1F467', 'Family: Woman, Girl'], ['1F469;&#x200D;&#x1F467;&#x200D;&#x1F466', 'Family: Woman, Girl, Boy'], ['1F469;&#x200D;&#x1F467;&#x200D;&#x1F467', 'Family: Woman, Girl, Girl'], ['1F933', 'Selfie'], ['1F4AA', 'Flexed Biceps'], ['1F448', 'Backhand Index Pointing Left'], ['1F449', 'Backhand Index Pointing Right'], ['261D', 'Index Pointing Up'], ['1F446', 'Backhand Index Pointing Up'], ['1F595', 'Middle Finger'], ['1F447', 'Backhand Index Pointing Down'], ['270C', 'Victory Hand'], ['1F91E', 'Crossed Fingers'], ['1F596', 'Vulcan Salute'], ['1F918', 'Sign of the Horns'], ['1F590', 'Raised Hand With Fingers Splayed'], ['270B', 'Raised Hand'], ['1F44C', 'OK Hand'], ['1F44D', 'Thumbs Up'], ['1F44E', 'Thumbs Down'], ['270A', 'Raised Fist'], ['1F44A', 'Oncoming Fist'], ['1F91B', 'Left-Facing Fist'], ['1F91C', 'Right-Facing Fist'], ['1F91A', 'Raised Back of Hand'], ['1F44B', 'Waving Hand'], ['270D', 'Writing Hand'], ['1F44F', 'Clapping Hands'], ['1F450', 'Open Hands'], ['1F64C', 'Raising Hands'], ['1F64F', 'Folded Hands'], ['1F91D', 'Handshake'], ['1F485', 'Nail Polish'], ['1F442', 'Ear'], ['1F443', 'Nose'], ['1F463', 'Footprints'], ['1F440', 'Eyes'], ['1F441', 'Eye'], ['1F445', 'Tongue'], ['1F444', 'Mouth'], ['1F48B', 'Kiss Mark'], ['1F453', 'Glasses'], ['1F576', 'Sunglasses'], ['1F454', 'Necktie'], ['1F455', 'T-Shirt'], ['1F456', 'Jeans'], ['1F457', 'Dress'], ['1F458', 'Kimono'], ['1F459', 'Bikini'], ['1F45A', 'Woman’s Clothes'], ['1F45B', 'Purse'], ['1F45C', 'Handbag'], ['1F45D', 'Clutch Bag'], ['1F392', 'School Backpack'], ['1F45E', 'Man’s Shoe'], ['1F45F', 'Running Shoe'], ['1F460', 'High-Heeled Shoe'], ['1F461', 'Woman’s Sandal'], ['1F462', 'Woman’s Boot'], ['1F451', 'Crown'], ['1F452', 'Woman’s Hat'], ['1F3A9', 'Top Hat'], ['1F393', 'Graduation Cap'], ['26D1', 'Rescue Worker’s Helmet'], ['1F484', 'Lipstick'], ['1F48D', 'Ring'], ['1F302', 'Closed Umbrella'], ['2602', 'Umbrella'], ['1F4BC', 'Briefcase']],
  529. animalsNature: [['1F648', 'See-No-Evil Monkey'], ['1F649', 'Hear-No-Evil Monkey'], ['1F64A', 'Speak-No-Evil Monkey'], ['1F4A5', 'Collision'], ['1F4A6', 'Sweat Droplets'], ['1F4A8', 'Dashing Away'], ['1F4AB', 'Dizzy'], ['1F435', 'Monkey Face'], ['1F412', 'Monkey'], ['1F98D', 'Gorilla'], ['1F436', 'Dog Face'], ['1F415', 'Dog'], ['1F429', 'Poodle'], ['1F43A', 'Wolf Face'], ['1F98A', 'Fox Face'], ['1F431', 'Cat Face'], ['1F408', 'Cat'], ['1F981', 'Lion Face'], ['1F42F', 'Tiger Face'], ['1F405', 'Tiger'], ['1F406', 'Leopard'], ['1F434', 'Horse Face'], ['1F40E', 'Horse'], ['1F984', 'Unicorn Face'], ['1F42E', 'Cow Face'], ['1F402', 'Ox'], ['1F403', 'Water Buffalo'], ['1F404', 'Cow'], ['1F437', 'Pig Face'], ['1F416', 'Pig'], ['1F417', 'Boar'], ['1F43D', 'Pig Nose'], ['1F40F', 'Ram'], ['1F411', 'Ewe'], ['1F410', 'Goat'], ['1F42A', 'Camel'], ['1F42B', 'Two-Hump Camel'], ['1F418', 'Elephant'], ['1F98F', 'Rhinoceros'], ['1F42D', 'Mouse Face'], ['1F401', 'Mouse'], ['1F400', 'Rat'], ['1F439', 'Hamster Face'], ['1F430', 'Rabbit Face'], ['1F407', 'Rabbit'], ['1F43F', 'Chipmunk'], ['1F987', 'Bat'], ['1F43B', 'Bear Face'], ['1F428', 'Koala'], ['1F43C', 'Panda Face'], ['1F43E', 'Paw Prints'], ['1F983', 'Turkey'], ['1F414', 'Chicken'], ['1F413', 'Rooster'], ['1F423', 'Hatching Chick'], ['1F424', 'Baby Chick'], ['1F425', 'Front-Facing Baby Chick'], ['1F426', 'Bird'], ['1F427', 'Penguin'], ['1F54A', 'Dove'], ['1F985', 'Eagle'], ['1F986', 'Duck'], ['1F989', 'Owl'], ['1F438', 'Frog Face'], ['1F40A', 'Crocodile'], ['1F422', 'Turtle'], ['1F98E', 'Lizard'], ['1F40D', 'Snake'], ['1F432', 'Dragon Face'], ['1F409', 'Dragon'], ['1F433', 'Spouting Whale'], ['1F40B', 'Whale'], ['1F42C', 'Dolphin'], ['1F41F', 'Fish'], ['1F420', 'Tropical Fish'], ['1F421', 'Blowfish'], ['1F988', 'Shark'], ['1F419', 'Octopus'], ['1F41A', 'Spiral Shell'], ['1F980', 'Crab'], ['1F990', 'Shrimp'], ['1F991', 'Squid'], ['1F40C', 'Snail'], ['1F98B', 'Butterfly'], ['1F41B', 'Bug'], ['1F41C', 'Ant'], ['1F41D', 'Honeybee'], ['1F41E', 'Lady Beetle'], ['1F577', 'Spider'], ['1F578', 'Spider Web'], ['1F982', 'Scorpion'], ['1F490', 'Bouquet'], ['1F338', 'Cherry Blossom'], ['1F4AE', 'White Flower'], ['1F3F5', 'Rosette'], ['1F339', 'Rose'], ['1F940', 'Wilted Flower'], ['1F33A', 'Hibiscus'], ['1F33B', 'Sunflower'], ['1F33C', 'Blossom'], ['1F337', 'Tulip'], ['1F331', 'Seedling'], ['1F332', 'Evergreen Tree'], ['1F333', 'Deciduous Tree'], ['1F334', 'Palm Tree'], ['1F335', 'Cactus'], ['1F33E', 'Sheaf of Rice'], ['1F33F', 'Herb'], ['2618', 'Shamrock'], ['1F340', 'Four Leaf Clover'], ['1F341', 'Maple Leaf'], ['1F342', 'Fallen Leaf'], ['1F343', 'Leaf Fluttering in Wind'], ['1F344', 'Mushroom'], ['1F330', 'Chestnut'], ['1F30D', 'Globe Showing Europe-Africa'], ['1F30E', 'Globe Showing Americas'], ['1F30F', 'Globe Showing Asia-Australia'], ['1F310', 'Globe With Meridians'], ['1F311', 'New Moon'], ['1F312', 'Waxing Crescent Moon'], ['1F313', 'First Quarter Moon'], ['1F314', 'Waxing Gibbous Moon'], ['1F315', 'Full Moon'], ['1F316', 'Waning Gibbous Moon'], ['1F317', 'Last Quarter Moon'], ['1F318', 'Waning Crescent Moon'], ['1F319', 'Crescent Moon'], ['1F31A', 'New Moon Face'], ['1F31B', 'First Quarter Moon With Face'], ['1F31C', 'Last Quarter Moon With Face'], ['2600', 'Sun'], ['1F31D', 'Full Moon With Face'], ['1F31E', 'Sun With Face'], ['2B50', 'White Medium Star'], ['1F31F', 'Glowing Star'], ['1F320', 'Shooting Star'], ['2601', 'Cloud'], ['26C5', 'Sun Behind Cloud'], ['26C8', 'Cloud With Lightning and Rain'], ['1F324', 'Sun Behind Small Cloud'], ['1F325', 'Sun Behind Large Cloud'], ['1F326', 'Sun Behind Rain Cloud'], ['1F327', 'Cloud With Rain'], ['1F328', 'Cloud With Snow'], ['1F329', 'Cloud With Lightning'], ['1F32A', 'Tornado'], ['1F32B', 'Fog'], ['1F32C', 'Wind Face'], ['1F308', 'Rainbow'], ['2602', 'Umbrella'], ['2614', 'Umbrella With Rain Drops'], ['26A1', 'High Voltage'], ['2744', 'Snowflake'], ['2603', 'Snowman'], ['26C4', 'Snowman Without Snow'], ['2604', 'Comet'], ['1F525', 'Fire'], ['1F4A7', 'Droplet'], ['1F30A', 'Water Wave'], ['1F384', 'Christmas Tree'], ['2728', 'Sparkles'], ['1F38B', 'Tanabata Tree'], ['1F38D', 'Pine Decoration']],
  530. foodDrink: [['1F347', 'Grapes'], ['1F348', 'Melon'], ['1F349', 'Watermelon'], ['1F34A', 'Tangerine'], ['1F34B', 'Lemon'], ['1F34C', 'Banana'], ['1F34D', 'Pineapple'], ['1F34E', 'Red Apple'], ['1F34F', 'Green Apple'], ['1F350', 'Pear'], ['1F351', 'Peach'], ['1F352', 'Cherries'], ['1F353', 'Strawberry'], ['1F95D', 'Kiwi Fruit'], ['1F345', 'Tomato'], ['1F951', 'Avocado'], ['1F346', 'Eggplant'], ['1F954', 'Potato'], ['1F955', 'Carrot'], ['1F33D', 'Ear of Corn'], ['1F336', 'Hot Pepper'], ['1F952', 'Cucumber'], ['1F344', 'Mushroom'], ['1F95C', 'Peanuts'], ['1F330', 'Chestnut'], ['1F35E', 'Bread'], ['1F950', 'Croissant'], ['1F956', 'Baguette Bread'], ['1F95E', 'Pancakes'], ['1F9C0', 'Cheese Wedge'], ['1F356', 'Meat on Bone'], ['1F357', 'Poultry Leg'], ['1F953', 'Bacon'], ['1F354', 'Hamburger'], ['1F35F', 'French Fries'], ['1F355', 'Pizza'], ['1F32D', 'Hot Dog'], ['1F32E', 'Taco'], ['1F32F', 'Burrito'], ['1F373', 'Cooking'], ['1F372', 'Pot of Food'], ['1F957', 'Green Salad'], ['1F37F', 'Popcorn'], ['1F371', 'Bento Box'], ['1F358', 'Rice Cracker'], ['1F359', 'Rice Ball'], ['1F35A', 'Cooked Rice'], ['1F35B', 'Curry Rice'], ['1F35C', 'Steaming Bowl'], ['1F35D', 'Spaghetti'], ['1F360', 'Roasted Sweet Potato'], ['1F362', 'Oden'], ['1F363', 'Sushi'], ['1F364', 'Fried Shrimp'], ['1F365', 'Fish Cake With Swirl'], ['1F361', 'Dango'], ['1F366', 'Soft Ice Cream'], ['1F367', 'Shaved Ice'], ['1F368', 'Ice Cream'], ['1F369', 'Doughnut'], ['1F36A', 'Cookie'], ['1F382', 'Birthday Cake'], ['1F370', 'Shortcake'], ['1F36B', 'Chocolate Bar'], ['1F36C', 'Candy'], ['1F36D', 'Lollipop'], ['1F36E', 'Custard'], ['1F36F', 'Honey Pot'], ['1F37C', 'Baby Bottle'], ['1F95B', 'Glass of Milk'], ['2615', 'Hot Beverage'], ['1F375', 'Teacup Without Handle'], ['1F376', 'Sake'], ['1F37E', 'Bottle With Popping Cork'], ['1F377', 'Wine Glass'], ['1F378', 'Cocktail Glass'], ['1F379', 'Tropical Drink'], ['1F37A', 'Beer Mug'], ['1F37B', 'Clinking Beer Mugs'], ['1F942', 'Clinking Glasses'], ['1F943', 'Tumbler Glass'], ['1F37D', 'Fork and Knife With Plate'], ['1F374', 'Fork and Knife'], ['1F944', 'Spoon']],
  531. activity: [['1F47E', 'Alien Monster'], ['1F574', 'Man in Business Suit Levitating'], ['1F3C7', 'Horse Racing'], ['26F7', 'Skier'], ['1F3C2', 'Snowboarder'], ['1F3CC', 'Person Golfing'], ['1F3CC;&#xFE0F;&#x200D;&#x2642;&#xFE0F', 'Man Golfing'], ['1F3CC;&#xFE0F;&#x200D;&#x2640;&#xFE0F', 'Woman Golfing'], ['1F3C4', 'Person Surfing'], ['1F3C4;&#x200D;&#x2642;&#xFE0F', 'Man Surfing'], ['1F3C4;&#x200D;&#x2640;&#xFE0F', 'Woman Surfing'], ['1F6A3', 'Person Rowing Boat'], ['1F6A3;&#x200D;&#x2642;&#xFE0F', 'Man Rowing Boat'], ['1F6A3;&#x200D;&#x2640;&#xFE0F', 'Woman Rowing Boat'], ['1F3CA', 'Person Swimming'], ['1F3CA;&#x200D;&#x2642;&#xFE0F', 'Man Swimming'], ['1F3CA;&#x200D;&#x2640;&#xFE0F', 'Woman Swimming'], ['26F9', 'Person Bouncing Ball'], ['26F9;&#xFE0F;&#x200D;&#x2642;&#xFE0F', 'Man Bouncing Ball'], ['26F9;&#xFE0F;&#x200D;&#x2640;&#xFE0F', 'Woman Bouncing Ball'], ['1F3CB', 'Person Lifting Weights'], ['1F3CB;&#xFE0F;&#x200D;&#x2642;&#xFE0F', 'Man Lifting Weights'], ['1F3CB;&#xFE0F;&#x200D;&#x2640;&#xFE0F', 'Woman Lifting Weights'], ['1F6B4', 'Person Biking'], ['1F6B4;&#x200D;&#x2642;&#xFE0F', 'Man Biking'], ['1F6B4;&#x200D;&#x2640;&#xFE0F', 'Woman Biking'], ['1F6B5', 'Person Mountain Biking'], ['1F6B5;&#x200D;&#x2642;&#xFE0F', 'Man Mountain Biking'], ['1F6B5;&#x200D;&#x2640;&#xFE0F', 'Woman Mountain Biking'], ['1F938', 'Person Cartwheeling'], ['1F938;&#x200D;&#x2642;&#xFE0F', 'Man Cartwheeling'], ['1F938;&#x200D;&#x2640;&#xFE0F', 'Woman Cartwheeling'], ['1F93C', 'People Wrestling'], ['1F93C;&#x200D;&#x2642;&#xFE0F', 'Men Wrestling'], ['1F93C;&#x200D;&#x2640;&#xFE0F', 'Women Wrestling'], ['1F93D', 'Person Playing Water Polo'], ['1F93D;&#x200D;&#x2642;&#xFE0F', 'Man Playing Water Polo'], ['1F93D;&#x200D;&#x2640;&#xFE0F', 'Woman Playing Water Polo'], ['1F93E', 'Person Playing Handball'], ['1F93E;&#x200D;&#x2642;&#xFE0F', 'Man Playing Handball'], ['1F93E;&#x200D;&#x2640;&#xFE0F', 'Woman Playing Handball'], ['1F939', 'Person Juggling'], ['1F939;&#x200D;&#x2642;&#xFE0F', 'Man Juggling'], ['1F939;&#x200D;&#x2640;&#xFE0F', 'Woman Juggling'], ['1F3AA', 'Circus Tent'], ['1F3AD', 'Performing Arts'], ['1F3A8', 'Artist Palette'], ['1F3B0', 'Slot Machine'], ['1F397', 'Reminder Ribbon'], ['1F39F', 'Admission Tickets'], ['1F3AB', 'Ticket'], ['1F396', 'Military Medal'], ['1F3C6', 'Trophy'], ['1F3C5', 'Sports Medal'], ['1F947', '1st Place Medal'], ['1F948', '2nd Place Medal'], ['1F949', '3rd Place Medal'], ['26BD', 'Soccer Ball'], ['26BE', 'Baseball'], ['1F3C0', 'Basketball'], ['1F3D0', 'Volleyball'], ['1F3C8', 'American Football'], ['1F3C9', 'Rugby Football'], ['1F3BE', 'Tennis'], ['1F3B1', 'Pool 8 Ball'], ['1F3B3', 'Bowling'], ['1F3CF', 'Cricket'], ['1F3D1', 'Field Hockey'], ['1F3D2', 'Ice Hockey'], ['1F3D3', 'Ping Pong'], ['1F3F8', 'Badminton'], ['1F94A', 'Boxing Glove'], ['1F94B', 'Martial Arts Uniform'], ['1F3AF', 'Direct Hit'], ['26F3', 'Flag in Hole'], ['26F8', 'Ice Skate'], ['1F3A3', 'Fishing Pole'], ['1F3BD', 'Running Shirt'], ['1F3BF', 'Skis'], ['1F3AE', 'Video Game'], ['1F3B2', 'Game Die'], ['1F3BC', 'Musical Score'], ['1F3A4', 'Microphone'], ['1F3A7', 'Headphone'], ['1F3B7', 'Saxophone'], ['1F3B8', 'Guitar'], ['1F3B9', 'Musical Keyboard'], ['1F3BA', 'Trumpet'], ['1F3BB', 'Violin'], ['1F941', 'Drum'], ['1F3AC', 'Clapper Board'], ['1F3F9', 'Bow and Arrow']],
  532. travelPlaces: [['1F6A3', 'Person Rowing Boat'], ['1F3CE', 'Racing Car'], ['1F3CD', 'Motorcycle'], ['1F5FE', 'Map of Japan'], ['1F3D4', 'Snow-Capped Mountain'], ['26F0', 'Mountain'], ['1F30B', 'Volcano'], ['1F5FB', 'Mount Fuji'], ['1F3D5', 'Camping'], ['1F3D6', 'Beach With Umbrella'], ['1F3DC', 'Desert'], ['1F3DD', 'Desert Island'], ['1F3DE', 'National Park'], ['1F3DF', 'Stadium'], ['1F3DB', 'Classical Building'], ['1F3D7', 'Building Construction'], ['1F3D8', 'House'], ['1F3D9', 'Cityscape'], ['1F3DA', 'Derelict House'], ['1F3E0', 'House'], ['1F3E1', 'House With Garden'], ['1F3E2', 'Office Building'], ['1F3E3', 'Japanese Post Office'], ['1F3E4', 'Post Office'], ['1F3E5', 'Hospital'], ['1F3E6', 'Bank'], ['1F3E8', 'Hotel'], ['1F3E9', 'Love Hotel'], ['1F3EA', 'Convenience Store'], ['1F3EB', 'School'], ['1F3EC', 'Department Store'], ['1F3ED', 'Factory'], ['1F3EF', 'Japanese Castle'], ['1F3F0', 'Castle'], ['1F492', 'Wedding'], ['1F5FC', 'Tokyo Tower'], ['1F5FD', 'Statue of Liberty'], ['26EA', 'Church'], ['1F54C', 'Mosque'], ['1F54D', 'Synagogue'], ['26E9', 'Shinto Shrine'], ['1F54B', 'Kaaba'], ['26F2', 'Fountain'], ['26FA', 'Tent'], ['1F301', 'Foggy'], ['1F303', 'Night With Stars'], ['1F304', 'Sunrise Over Mountains'], ['1F305', 'Sunrise'], ['1F306', 'Cityscape at Dusk'], ['1F307', 'Sunset'], ['1F309', 'Bridge at Night'], ['1F30C', 'Milky Way'], ['1F3A0', 'Carousel Horse'], ['1F3A1', 'Ferris Wheel'], ['1F3A2', 'Roller Coaster'], ['1F682', 'Locomotive'], ['1F683', 'Railway Car'], ['1F684', 'High-Speed Train'], ['1F685', 'High-Speed Train With Bullet Nose'], ['1F686', 'Train'], ['1F687', 'Metro'], ['1F688', 'Light Rail'], ['1F689', 'Station'], ['1F68A', 'Tram'], ['1F69D', 'Monorail'], ['1F69E', 'Mountain Railway'], ['1F68B', 'Tram Car'], ['1F68C', 'Bus'], ['1F68D', 'Oncoming Bus'], ['1F68E', 'Trolleybus'], ['1F690', 'Minibus'], ['1F691', 'Ambulance'], ['1F692', 'Fire Engine'], ['1F693', 'Police Car'], ['1F694', 'Oncoming Police Car'], ['1F695', 'Taxi'], ['1F696', 'Oncoming Taxi'], ['1F697', 'Automobile'], ['1F698', 'Oncoming Automobile'], ['1F69A', 'Delivery Truck'], ['1F69B', 'Articulated Lorry'], ['1F69C', 'Tractor'], ['1F6B2', 'Bicycle'], ['1F6F4', 'Kick Scooter'], ['1F6F5', 'Motor Scooter'], ['1F68F', 'Bus Stop'], ['1F6E4', 'Railway Track'], ['26FD', 'Fuel Pump'], ['1F6A8', 'Police Car Light'], ['1F6A5', 'Horizontal Traffic Light'], ['1F6A6', 'Vertical Traffic Light'], ['1F6A7', 'Construction'], ['2693', 'Anchor'], ['26F5', 'Sailboat'], ['1F6A4', 'Speedboat'], ['1F6F3', 'Passenger Ship'], ['26F4', 'Ferry'], ['1F6E5', 'Motor Boat'], ['1F6A2', 'Ship'], ['2708', 'Airplane'], ['1F6E9', 'Small Airplane'], ['1F6EB', 'Airplane Departure'], ['1F6EC', 'Airplane Arrival'], ['1F4BA', 'Seat'], ['1F681', 'Helicopter'], ['1F69F', 'Suspension Railway'], ['1F6A0', 'Mountain Cableway'], ['1F6A1', 'Aerial Tramway'], ['1F6F0', 'Satellite'], ['1F680', 'Rocket'], ['1F320', 'Shooting Star'], ['26F1', 'Umbrella on Ground'], ['1F386', 'Fireworks'], ['1F387', 'Sparkler'], ['1F391', 'Moon Viewing Ceremony'], ['1F4B4', 'Yen Banknote'], ['1F4B5', 'Dollar Banknote'], ['1F4B6', 'Euro Banknote'], ['1F4B7', 'Pound Banknote'], ['1F5FF', 'Moai'], ['1F6C2', 'Passport Control'], ['1F6C3', 'Customs'], ['1F6C4', 'Baggage Claim'], ['1F6C5', 'Left Luggage']],
  533. objects: [['2620', 'Skull and Crossbones'], ['1F6C0', 'Person Taking Bath'], ['1F6CC', 'Person in Bed'], ['1F48C', 'Love Letter'], ['1F4A3', 'Bomb'], ['1F573', 'Hole'], ['1F6CD', 'Shopping Bags'], ['1F4FF', 'Prayer Beads'], ['1F48E', 'Gem Stone'], ['1F52A', 'Kitchen Knife'], ['1F3FA', 'Amphora'], ['1F5FA', 'World Map'], ['1F488', 'Barber Pole'], ['1F5BC', 'Framed Picture'], ['1F6CE', 'Bellhop Bell'], ['1F6AA', 'Door'], ['1F6CF', 'Bed'], ['1F6CB', 'Couch and Lamp'], ['1F6BD', 'Toilet'], ['1F6BF', 'Shower'], ['1F6C1', 'Bathtub'], ['231B', 'Hourglass'], ['23F3', 'Hourglass With Flowing Sand'], ['231A', 'Watch'], ['23F0', 'Alarm Clock'], ['23F1', 'Stopwatch'], ['23F2', 'Timer Clock'], ['1F570', 'Mantelpiece Clock'], ['1F321', 'Thermometer'], ['26F1', 'Umbrella on Ground'], ['1F388', 'Balloon'], ['1F389', 'Party Popper'], ['1F38A', 'Confetti Ball'], ['1F38E', 'Japanese Dolls'], ['1F38F', 'Carp Streamer'], ['1F390', 'Wind Chime'], ['1F380', 'Ribbon'], ['1F381', 'Wrapped Gift'], ['1F579', 'Joystick'], ['1F4EF', 'Postal Horn'], ['1F399', 'Studio Microphone'], ['1F39A', 'Level Slider'], ['1F39B', 'Control Knobs'], ['1F4FB', 'Radio'], ['1F4F1', 'Mobile Phone'], ['1F4F2', 'Mobile Phone With Arrow'], ['260E', 'Telephone'], ['1F4DE', 'Telephone Receiver'], ['1F4DF', 'Pager'], ['1F4E0', 'Fax Machine'], ['1F50B', 'Battery'], ['1F50C', 'Electric Plug'], ['1F4BB', 'Laptop Computer'], ['1F5A5', 'Desktop Computer'], ['1F5A8', 'Printer'], ['2328', 'Keyboard'], ['1F5B1', 'Computer Mouse'], ['1F5B2', 'Trackball'], ['1F4BD', 'Computer Disk'], ['1F4BE', 'Floppy Disk'], ['1F4BF', 'Optical Disk'], ['1F4C0', 'DVD'], ['1F3A5', 'Movie Camera'], ['1F39E', 'Film Frames'], ['1F4FD', 'Film Projector'], ['1F4FA', 'Television'], ['1F4F7', 'Camera'], ['1F4F8', 'Camera With Flash'], ['1F4F9', 'Video Camera'], ['1F4FC', 'Videocassette'], ['1F50D', 'Left-Pointing Magnifying Glass'], ['1F50E', 'Right-Pointing Magnifying Glass'], ['1F52C', 'Microscope'], ['1F52D', 'Telescope'], ['1F4E1', 'Satellite Antenna'], ['1F56F', 'Candle'], ['1F4A1', 'Light Bulb'], ['1F526', 'Flashlight'], ['1F3EE', 'Red Paper Lantern'], ['1F4D4', 'Notebook With Decorative Cover'], ['1F4D5', 'Closed Book'], ['1F4D6', 'Open Book'], ['1F4D7', 'Green Book'], ['1F4D8', 'Blue Book'], ['1F4D9', 'Orange Book'], ['1F4DA', 'Books'], ['1F4D3', 'Notebook'], ['1F4C3', 'Page With Curl'], ['1F4DC', 'Scroll'], ['1F4C4', 'Page Facing Up'], ['1F4F0', 'Newspaper'], ['1F5DE', 'Rolled-Up Newspaper'], ['1F4D1', 'Bookmark Tabs'], ['1F516', 'Bookmark'], ['1F3F7', 'Label'], ['1F4B0', 'Money Bag'], ['1F4B4', 'Yen Banknote'], ['1F4B5', 'Dollar Banknote'], ['1F4B6', 'Euro Banknote'], ['1F4B7', 'Pound Banknote'], ['1F4B8', 'Money With Wings'], ['1F4B3', 'Credit Card'], ['2709', 'Envelope'], ['1F4E7', 'E-Mail'], ['1F4E8', 'Incoming Envelope'], ['1F4E9', 'Envelope With Arrow'], ['1F4E4', 'Outbox Tray'], ['1F4E5', 'Inbox Tray'], ['1F4E6', 'Package'], ['1F4EB', 'Closed Mailbox With Raised Flag'], ['1F4EA', 'Closed Mailbox With Lowered Flag'], ['1F4EC', 'Open Mailbox With Raised Flag'], ['1F4ED', 'Open Mailbox With Lowered Flag'], ['1F4EE', 'Postbox'], ['1F5F3', 'Ballot Box With Ballot'], ['270F', 'Pencil'], ['2712', 'Black Nib'], ['1F58B', 'Fountain Pen'], ['1F58A', 'Pen'], ['1F58C', 'Paintbrush'], ['1F58D', 'Crayon'], ['1F4DD', 'Memo'], ['1F4C1', 'File Folder'], ['1F4C2', 'Open File Folder'], ['1F5C2', 'Card Index Dividers'], ['1F4C5', 'Calendar'], ['1F4C6', 'Tear-Off Calendar'], ['1F5D2', 'Spiral Notepad'], ['1F5D3', 'Spiral Calendar'], ['1F4C7', 'Card Index'], ['1F4C8', 'Chart Increasing'], ['1F4C9', 'Chart Decreasing'], ['1F4CA', 'Bar Chart'], ['1F4CB', 'Clipboard'], ['1F4CC', 'Pushpin'], ['1F4CD', 'Round Pushpin'], ['1F4CE', 'Paperclip'], ['1F587', 'Linked Paperclips'], ['1F4CF', 'Straight Ruler'], ['1F4D0', 'Triangular Ruler'], ['2702', 'Scissors'], ['1F5C3', 'Card File Box'], ['1F5C4', 'File Cabinet'], ['1F5D1', 'Wastebasket'], ['1F512', 'Locked'], ['1F513', 'Unlocked'], ['1F50F', 'Locked With Pen'], ['1F510', 'Locked With Key'], ['1F511', 'Key'], ['1F5DD', 'Old Key'], ['1F528', 'Hammer'], ['26CF', 'Pick'], ['2692', 'Hammer and Pick'], ['1F6E0', 'Hammer and Wrench'], ['1F5E1', 'Dagger'], ['2694', 'Crossed Swords'], ['1F52B', 'Pistol'], ['1F6E1', 'Shield'], ['1F527', 'Wrench'], ['1F529', 'Nut and Bolt'], ['2699', 'Gear'], ['1F5DC', 'Clamp'], ['2697', 'Alembic'], ['2696', 'Balance Scale'], ['1F517', 'Link'], ['26D3', 'Chains'], ['1F489', 'Syringe'], ['1F48A', 'Pill'], ['1F6AC', 'Cigarette'], ['26B0', 'Coffin'], ['26B1', 'Funeral Urn'], ['1F5FF', 'Moai'], ['1F6E2', 'Oil Drum'], ['1F52E', 'Crystal Ball'], ['1F6B0', 'Potable Water'], ['1F3C1', 'Chequered Flag'], ['1F6A9', 'Triangular Flag'], ['1F38C', 'Crossed Flags'], ['1F3F4', 'Black Flag'], ['1F3F3', 'White Flag'], ['1F3F3;&#xFE0F;&#x200D;&#x1F308', 'Rainbow Flag']],
  534. symbols: [['1F441;&#xFE0F;&#x200D;&#x1F5E8;&#xFE0F', 'Eye in Speech Bubble'], ['1F498', 'Heart With Arrow'], ['2764', 'Red Heart'], ['1F493', 'Beating Heart'], ['1F494', 'Broken Heart'], ['1F495', 'Two Hearts'], ['1F496', 'Sparkling Heart'], ['1F497', 'Growing Heart'], ['1F499', 'Blue Heart'], ['1F49A', 'Green Heart'], ['1F49B', 'Yellow Heart'], ['1F49C', 'Purple Heart'], ['1F5A4', 'Black Heart'], ['1F49D', 'Heart With Ribbon'], ['1F49E', 'Revolving Hearts'], ['1F49F', 'Heart Decoration'], ['2763', 'Heavy Heart Exclamation'], ['1F4A4', 'Zzz'], ['1F4A2', 'Anger Symbol'], ['1F4AC', 'Speech Balloon'], ['1F5EF', 'Right Anger Bubble'], ['1F4AD', 'Thought Balloon'], ['1F4AE', 'White Flower'], ['2668', 'Hot Springs'], ['1F488', 'Barber Pole'], ['1F6D1', 'Stop Sign'], ['1F55B', 'Twelve O’clock'], ['1F567', 'Twelve-Thirty'], ['1F550', 'One O’clock'], ['1F55C', 'One-Thirty'], ['1F551', 'Two O’clock'], ['1F55D', 'Two-Thirty'], ['1F552', 'Three O’clock'], ['1F55E', 'Three-Thirty'], ['1F553', 'Four O’clock'], ['1F55F', 'Four-Thirty'], ['1F554', 'Five O’clock'], ['1F560', 'Five-Thirty'], ['1F555', 'Six O’clock'], ['1F561', 'Six-Thirty'], ['1F556', 'Seven O’clock'], ['1F562', 'Seven-Thirty'], ['1F557', 'Eight O’clock'], ['1F563', 'Eight-Thirty'], ['1F558', 'Nine O’clock'], ['1F564', 'Nine-Thirty'], ['1F559', 'Ten O’clock'], ['1F565', 'Ten-Thirty'], ['1F55A', 'Eleven O’clock'], ['1F566', 'Eleven-Thirty'], ['1F300', 'Cyclone'], ['2660', 'Spade Suit'], ['2665', 'Heart Suit'], ['2666', 'Diamond Suit'], ['2663', 'Club Suit'], ['1F0CF', 'Joker'], ['1F004', 'Mahjong Red Dragon'], ['1F3B4', 'Flower Playing Cards'], ['1F507', 'Muted Speaker'], ['1F508', 'Speaker Low Volume'], ['1F509', 'Speaker Medium Volume'], ['1F50A', 'Speaker High Volume'], ['1F4E2', 'Loudspeaker'], ['1F4E3', 'Megaphone'], ['1F4EF', 'Postal Horn'], ['1F514', 'Bell'], ['1F515', 'Bell With Slash'], ['1F3B5', 'Musical Note'], ['1F3B6', 'Musical Notes'], ['1F3E7', 'Atm Sign'], ['1F6AE', 'Litter in Bin Sign'], ['1F6B0', 'Potable Water'], ['267F', 'Wheelchair Symbol'], ['1F6B9', 'Men’s Room'], ['1F6BA', 'Women’s Room'], ['1F6BB', 'Restroom'], ['1F6BC', 'Baby Symbol'], ['1F6BE', 'Water Closet'], ['26A0', 'Warning'], ['1F6B8', 'Children Crossing'], ['26D4', 'No Entry'], ['1F6AB', 'Prohibited'], ['1F6B3', 'No Bicycles'], ['1F6AD', 'No Smoking'], ['1F6AF', 'No Littering'], ['1F6B1', 'Non-Potable Water'], ['1F6B7', 'No Pedestrians'], ['1F51E', 'No One Under Eighteen'], ['2622', 'Radioactive'], ['2623', 'Biohazard'], ['2B06', 'Up Arrow'], ['2197', 'Up-Right Arrow'], ['27A1', 'Right Arrow'], ['2198', 'Down-Right Arrow'], ['2B07', 'Down Arrow'], ['2199', 'Down-Left Arrow'], ['2B05', 'Left Arrow'], ['2196', 'Up-Left Arrow'], ['2195', 'Up-Down Arrow'], ['2194', 'Left-Right Arrow'], ['21A9', 'Right Arrow Curving Left'], ['21AA', 'Left Arrow Curving Right'], ['2934', 'Right Arrow Curving Up'], ['2935', 'Right Arrow Curving Down'], ['1F503', 'Clockwise Vertical Arrows'], ['1F504', 'Anticlockwise Arrows Button'], ['1F519', 'Back Arrow'], ['1F51A', 'End Arrow'], ['1F51B', 'On! Arrow'], ['1F51C', 'Soon Arrow'], ['1F51D', 'Top Arrow'], ['1F6D0', 'Place of Worship'], ['269B', 'Atom Symbol'], ['1F549', 'Om'], ['2721', 'Star of David'], ['2638', 'Wheel of Dharma'], ['262F', 'Yin Yang'], ['271D', 'Latin Cross'], ['2626', 'Orthodox Cross'], ['262A', 'Star and Crescent'], ['262E', 'Peace Symbol'], ['1F54E', 'Menorah'], ['1F52F', 'Dotted Six-Pointed Star'], ['2648', 'Aries'], ['2649', 'Taurus'], ['264A', 'Gemini'], ['264B', 'Cancer'], ['264C', 'Leo'], ['264D', 'Virgo'], ['264E', 'Libra'], ['264F', 'Scorpius'], ['2650', 'Sagittarius'], ['2651', 'Capricorn'], ['2652', 'Aquarius'], ['2653', 'Pisces'], ['26CE', 'Ophiuchus'], ['1F500', 'Shuffle Tracks Button'], ['1F501', 'Repeat Button'], ['1F502', 'Repeat Single Button'], ['25B6', 'Play Button'], ['23E9', 'Fast-Forward Button'], ['25C0', 'Reverse Button'], ['23EA', 'Fast Reverse Button'], ['1F53C', 'Up Button'], ['23EB', 'Fast Up Button'], ['1F53D', 'Down Button'], ['23EC', 'Fast Down Button'], ['23F9', 'Stop Button'], ['1F3A6', 'Cinema'], ['1F505', 'Dim Button'], ['1F506', 'Bright Button'], ['1F4F6', 'Antenna Bars'], ['1F4F3', 'Vibration Mode'], ['1F4F4', 'Mobile Phone Off'], ['267B', 'Recycling Symbol'], ['1F531', 'Trident Emblem'], ['1F4DB', 'Name Badge'], ['1F530', 'Japanese Symbol for Beginner'], ['2B55', 'Heavy Large Circle'], ['2705', 'White Heavy Check Mark'], ['2611', 'Ballot Box With Check'], ['2714', 'Heavy Check Mark'], ['2716', 'Heavy Multiplication X'], ['274C', 'Cross Mark'], ['274E', 'Cross Mark Button'], ['2795', 'Heavy Plus Sign'], ['2796', 'Heavy Minus Sign'], ['2797', 'Heavy Division Sign'], ['27B0', 'Curly Loop'], ['27BF', 'Double Curly Loop'], ['303D', 'Part Alternation Mark'], ['2733', 'Eight-Spoked Asterisk'], ['2734', 'Eight-Pointed Star'], ['2747', 'Sparkle'], ['203C', 'Double Exclamation Mark'], ['2049', 'Exclamation Question Mark'], ['2753', 'Question Mark'], ['2754', 'White Question Mark'], ['2755', 'White Exclamation Mark'], ['2757', 'Exclamation Mark'], ['A9', 'Copyright'], ['AE', 'Registered'], ['2122', 'Trade Mark'], ['23;&#xFE0F;&#x20E3', 'Keycap Number Sign'], ['30;&#xFE0F;&#x20E3', 'Keycap Digit Zero'], ['31;&#xFE0F;&#x20E3', 'Keycap Digit One'], ['32;&#xFE0F;&#x20E3', 'Keycap Digit Two'], ['33;&#xFE0F;&#x20E3', 'Keycap Digit Three'], ['34;&#xFE0F;&#x20E3', 'Keycap Digit Four'], ['35;&#xFE0F;&#x20E3', 'Keycap Digit Five'], ['36;&#xFE0F;&#x20E3', 'Keycap Digit Six'], ['37;&#xFE0F;&#x20E3', 'Keycap Digit Seven'], ['38;&#xFE0F;&#x20E3', 'Keycap Digit Eight'], ['39;&#xFE0F;&#x20E3', 'Keycap Digit Nine'], ['1F51F', 'Keycap 10'], ['1F4AF', 'Hundred Points'], ['1F520', 'Input Latin Uppercase'], ['1F521', 'Input Latin Lowercase'], ['1F522', 'Input Numbers'], ['1F523', 'Input Symbols'], ['1F524', 'Input Latin Letters'], ['1F170', 'A Button (blood Type)'], ['1F18E', 'Ab Button (blood Type)'], ['1F171', 'B Button (blood Type)'], ['1F191', 'CL Button'], ['1F192', 'Cool Button'], ['1F193', 'Free Button'], ['2139', 'Information'], ['1F194', 'ID Button'], ['24C2', 'Circled M'], ['1F195', 'New Button'], ['1F196', 'NG Button'], ['1F17E', 'O Button (blood Type)'], ['1F197', 'OK Button'], ['1F17F', 'P Button'], ['1F198', 'SOS Button'], ['1F199', 'Up! Button'], ['1F19A', 'Vs Button'], ['1F201', 'Japanese “here” Button'], ['1F202', 'Japanese “service Charge” Button'], ['1F237', 'Japanese “monthly Amount” Button'], ['1F236', 'Japanese “not Free of Charge” Button'], ['1F22F', 'Japanese “reserved” Button'], ['1F250', 'Japanese “bargain” Button'], ['1F239', 'Japanese “discount” Button'], ['1F21A', 'Japanese “free of Charge” Button'], ['1F232', 'Japanese “prohibited” Button'], ['1F251', 'Japanese “acceptable” Button'], ['1F238', 'Japanese “application” Button'], ['1F234', 'Japanese “passing Grade” Button'], ['1F233', 'Japanese “vacancy” Button'], ['3297', 'Japanese “congratulations” Button'], ['3299', 'Japanese “secret” Button'], ['1F23A', 'Japanese “open for Business” Button'], ['1F235', 'Japanese “no Vacancy” Button'], ['25AA', 'Black Small Square'], ['25AB', 'White Small Square'], ['25FB', 'White Medium Square'], ['25FC', 'Black Medium Square'], ['25FD', 'White Medium-Small Square'], ['25FE', 'Black Medium-Small Square'], ['2B1B', 'Black Large Square'], ['2B1C', 'White Large Square'], ['1F536', 'Large Orange Diamond'], ['1F537', 'Large Blue Diamond'], ['1F538', 'Small Orange Diamond'], ['1F539', 'Small Blue Diamond'], ['1F53A', 'Red Triangle Pointed Up'], ['1F53B', 'Red Triangle Pointed Down'], ['1F4A0', 'Diamond With a Dot'], ['1F532', 'Black Square Button'], ['1F533', 'White Square Button'], ['26AA', 'White Circle'], ['26AB', 'Black Circle'], ['1F534', 'Red Circle'], ['1F535', 'Blue Circle']],
  535. flags: [['1F3C1', 'Chequered Flag'], ['1F6A9', 'Triangular Flag'], ['1F38C', 'Crossed Flags'], ['1F3F4', 'Black Flag'], ['1F3F3', 'White Flag'], ['1F3F3;&#xFE0F;&#x200D;&#x1F308', 'Rainbow Flag'], ['1F1E6;&#x1F1E8', 'Ascension Island'], ['1F1E6;&#x1F1E9', 'Andorra'], ['1F1E6;&#x1F1EA', 'United Arab Emirates'], ['1F1E6;&#x1F1EB', 'Afghanistan'], ['1F1E6;&#x1F1EC', 'Antigua & Barbuda'], ['1F1E6;&#x1F1EE', 'Anguilla'], ['1F1E6;&#x1F1F1', 'Albania'], ['1F1E6;&#x1F1F2', 'Armenia'], ['1F1E6;&#x1F1F4', 'Angola'], ['1F1E6;&#x1F1F6', 'Antarctica'], ['1F1E6;&#x1F1F7', 'Argentina'], ['1F1E6;&#x1F1F8', 'American Samoa'], ['1F1E6;&#x1F1F9', 'Austria'], ['1F1E6;&#x1F1FA', 'Australia'], ['1F1E6;&#x1F1FC', 'Aruba'], ['1F1E6;&#x1F1FD', 'Åland Islands'], ['1F1E6;&#x1F1FF', 'Azerbaijan'], ['1F1E7;&#x1F1E6', 'Bosnia & Herzegovina'], ['1F1E7;&#x1F1E7', 'Barbados'], ['1F1E7;&#x1F1E9', 'Bangladesh'], ['1F1E7;&#x1F1EA', 'Belgium'], ['1F1E7;&#x1F1EB', 'Burkina Faso'], ['1F1E7;&#x1F1EC', 'Bulgaria'], ['1F1E7;&#x1F1ED', 'Bahrain'], ['1F1E7;&#x1F1EE', 'Burundi'], ['1F1E7;&#x1F1EF', 'Benin'], ['1F1E7;&#x1F1F1', 'St. Barthélemy'], ['1F1E7;&#x1F1F2', 'Bermuda'], ['1F1E7;&#x1F1F3', 'Brunei'], ['1F1E7;&#x1F1F4', 'Bolivia'], ['1F1E7;&#x1F1F6', 'Caribbean Netherlands'], ['1F1E7;&#x1F1F7', 'Brazil'], ['1F1E7;&#x1F1F8', 'Bahamas'], ['1F1E7;&#x1F1F9', 'Bhutan'], ['1F1E7;&#x1F1FB', 'Bouvet Island'], ['1F1E7;&#x1F1FC', 'Botswana'], ['1F1E7;&#x1F1FE', 'Belarus'], ['1F1E7;&#x1F1FF', 'Belize'], ['1F1E8;&#x1F1E6', 'Canada'], ['1F1E8;&#x1F1E8', 'Cocos (Keeling) Islands'], ['1F1E8;&#x1F1E9', 'Congo - Kinshasa'], ['1F1E8;&#x1F1EB', 'Central African Republic'], ['1F1E8;&#x1F1EC', 'Congo - Brazzaville'], ['1F1E8;&#x1F1ED', 'Switzerland'], ['1F1E8;&#x1F1EE', 'Côte D’Ivoire'], ['1F1E8;&#x1F1F0', 'Cook Islands'], ['1F1E8;&#x1F1F1', 'Chile'], ['1F1E8;&#x1F1F2', 'Cameroon'], ['1F1E8;&#x1F1F3', 'China'], ['1F1E8;&#x1F1F4', 'Colombia'], ['1F1E8;&#x1F1F5', 'Clipperton Island'], ['1F1E8;&#x1F1F7', 'Costa Rica'], ['1F1E8;&#x1F1FA', 'Cuba'], ['1F1E8;&#x1F1FB', 'Cape Verde'], ['1F1E8;&#x1F1FC', 'Curaçao'], ['1F1E8;&#x1F1FD', 'Christmas Island'], ['1F1E8;&#x1F1FE', 'Cyprus'], ['1F1E8;&#x1F1FF', 'Czechia'], ['1F1E9;&#x1F1EA', 'Germany'], ['1F1E9;&#x1F1EC', 'Diego Garcia'], ['1F1E9;&#x1F1EF', 'Djibouti'], ['1F1E9;&#x1F1F0', 'Denmark'], ['1F1E9;&#x1F1F2', 'Dominica'], ['1F1E9;&#x1F1F4', 'Dominican Republic'], ['1F1E9;&#x1F1FF', 'Algeria'], ['1F1EA;&#x1F1E6', 'Ceuta & Melilla'], ['1F1EA;&#x1F1E8', 'Ecuador'], ['1F1EA;&#x1F1EA', 'Estonia'], ['1F1EA;&#x1F1EC', 'Egypt'], ['1F1EA;&#x1F1ED', 'Western Sahara'], ['1F1EA;&#x1F1F7', 'Eritrea'], ['1F1EA;&#x1F1F8', 'Spain'], ['1F1EA;&#x1F1F9', 'Ethiopia'], ['1F1EA;&#x1F1FA', 'European Union'], ['1F1EB;&#x1F1EE', 'Finland'], ['1F1EB;&#x1F1EF', 'Fiji'], ['1F1EB;&#x1F1F0', 'Falkland Islands'], ['1F1EB;&#x1F1F2', 'Micronesia'], ['1F1EB;&#x1F1F4', 'Faroe Islands'], ['1F1EB;&#x1F1F7', 'France'], ['1F1EC;&#x1F1E6', 'Gabon'], ['1F1EC;&#x1F1E7', 'United Kingdom'], ['1F1EC;&#x1F1E9', 'Grenada'], ['1F1EC;&#x1F1EA', 'Georgia'], ['1F1EC;&#x1F1EB', 'French Guiana'], ['1F1EC;&#x1F1EC', 'Guernsey'], ['1F1EC;&#x1F1ED', 'Ghana'], ['1F1EC;&#x1F1EE', 'Gibraltar'], ['1F1EC;&#x1F1F1', 'Greenland'], ['1F1EC;&#x1F1F2', 'Gambia'], ['1F1EC;&#x1F1F3', 'Guinea'], ['1F1EC;&#x1F1F5', 'Guadeloupe'], ['1F1EC;&#x1F1F6', 'Equatorial Guinea'], ['1F1EC;&#x1F1F7', 'Greece'], ['1F1EC;&#x1F1F8', 'South Georgia & South Sandwich Islands'], ['1F1EC;&#x1F1F9', 'Guatemala'], ['1F1EC;&#x1F1FA', 'Guam'], ['1F1EC;&#x1F1FC', 'Guinea-Bissau'], ['1F1EC;&#x1F1FE', 'Guyana'], ['1F1ED;&#x1F1F0', 'Hong Kong Sar China'], ['1F1ED;&#x1F1F2', 'Heard & Mcdonald Islands'], ['1F1ED;&#x1F1F3', 'Honduras'], ['1F1ED;&#x1F1F7', 'Croatia'], ['1F1ED;&#x1F1F9', 'Haiti'], ['1F1ED;&#x1F1FA', 'Hungary'], ['1F1EE;&#x1F1E8', 'Canary Islands'], ['1F1EE;&#x1F1E9', 'Indonesia'], ['1F1EE;&#x1F1EA', 'Ireland'], ['1F1EE;&#x1F1F1', 'Israel'], ['1F1EE;&#x1F1F2', 'Isle of Man'], ['1F1EE;&#x1F1F3', 'India'], ['1F1EE;&#x1F1F4', 'British Indian Ocean Territory'], ['1F1EE;&#x1F1F6', 'Iraq'], ['1F1EE;&#x1F1F7', 'Iran'], ['1F1EE;&#x1F1F8', 'Iceland'], ['1F1EE;&#x1F1F9', 'Italy'], ['1F1EF;&#x1F1EA', 'Jersey'], ['1F1EF;&#x1F1F2', 'Jamaica'], ['1F1EF;&#x1F1F4', 'Jordan'], ['1F1EF;&#x1F1F5', 'Japan'], ['1F1F0;&#x1F1EA', 'Kenya'], ['1F1F0;&#x1F1EC', 'Kyrgyzstan'], ['1F1F0;&#x1F1ED', 'Cambodia'], ['1F1F0;&#x1F1EE', 'Kiribati'], ['1F1F0;&#x1F1F2', 'Comoros'], ['1F1F0;&#x1F1F3', 'St. Kitts & Nevis'], ['1F1F0;&#x1F1F5', 'North Korea'], ['1F1F0;&#x1F1F7', 'South Korea'], ['1F1F0;&#x1F1FC', 'Kuwait'], ['1F1F0;&#x1F1FE', 'Cayman Islands'], ['1F1F0;&#x1F1FF', 'Kazakhstan'], ['1F1F1;&#x1F1E6', 'Laos'], ['1F1F1;&#x1F1E7', 'Lebanon'], ['1F1F1;&#x1F1E8', 'St. Lucia'], ['1F1F1;&#x1F1EE', 'Liechtenstein'], ['1F1F1;&#x1F1F0', 'Sri Lanka'], ['1F1F1;&#x1F1F7', 'Liberia'], ['1F1F1;&#x1F1F8', 'Lesotho'], ['1F1F1;&#x1F1F9', 'Lithuania'], ['1F1F1;&#x1F1FA', 'Luxembourg'], ['1F1F1;&#x1F1FB', 'Latvia'], ['1F1F1;&#x1F1FE', 'Libya'], ['1F1F2;&#x1F1E6', 'Morocco'], ['1F1F2;&#x1F1E8', 'Monaco'], ['1F1F2;&#x1F1E9', 'Moldova'], ['1F1F2;&#x1F1EA', 'Montenegro'], ['1F1F2;&#x1F1EB', 'St. Martin'], ['1F1F2;&#x1F1EC', 'Madagascar'], ['1F1F2;&#x1F1ED', 'Marshall Islands'], ['1F1F2;&#x1F1F0', 'Macedonia'], ['1F1F2;&#x1F1F1', 'Mali'], ['1F1F2;&#x1F1F2', 'Myanmar (Burma)'], ['1F1F2;&#x1F1F3', 'Mongolia'], ['1F1F2;&#x1F1F4', 'Macau Sar China'], ['1F1F2;&#x1F1F5', 'Northern Mariana Islands'], ['1F1F2;&#x1F1F6', 'Martinique'], ['1F1F2;&#x1F1F7', 'Mauritania'], ['1F1F2;&#x1F1F8', 'Montserrat'], ['1F1F2;&#x1F1F9', 'Malta'], ['1F1F2;&#x1F1FA', 'Mauritius'], ['1F1F2;&#x1F1FB', 'Maldives'], ['1F1F2;&#x1F1FC', 'Malawi'], ['1F1F2;&#x1F1FD', 'Mexico'], ['1F1F2;&#x1F1FE', 'Malaysia'], ['1F1F2;&#x1F1FF', 'Mozambique'], ['1F1F3;&#x1F1E6', 'Namibia'], ['1F1F3;&#x1F1E8', 'New Caledonia'], ['1F1F3;&#x1F1EA', 'Niger'], ['1F1F3;&#x1F1EB', 'Norfolk Island'], ['1F1F3;&#x1F1EC', 'Nigeria'], ['1F1F3;&#x1F1EE', 'Nicaragua'], ['1F1F3;&#x1F1F1', 'Netherlands'], ['1F1F3;&#x1F1F4', 'Norway'], ['1F1F3;&#x1F1F5', 'Nepal'], ['1F1F3;&#x1F1F7', 'Nauru'], ['1F1F3;&#x1F1FA', 'Niue'], ['1F1F3;&#x1F1FF', 'New Zealand'], ['1F1F4;&#x1F1F2', 'Oman'], ['1F1F5;&#x1F1E6', 'Panama'], ['1F1F5;&#x1F1EA', 'Peru'], ['1F1F5;&#x1F1EB', 'French Polynesia'], ['1F1F5;&#x1F1EC', 'Papua New Guinea'], ['1F1F5;&#x1F1ED', 'Philippines'], ['1F1F5;&#x1F1F0', 'Pakistan'], ['1F1F5;&#x1F1F1', 'Poland'], ['1F1F5;&#x1F1F2', 'St. Pierre & Miquelon'], ['1F1F5;&#x1F1F3', 'Pitcairn Islands'], ['1F1F5;&#x1F1F7', 'Puerto Rico'], ['1F1F5;&#x1F1F8', 'Palestinian Territories'], ['1F1F5;&#x1F1F9', 'Portugal'], ['1F1F5;&#x1F1FC', 'Palau'], ['1F1F5;&#x1F1FE', 'Paraguay'], ['1F1F6;&#x1F1E6', 'Qatar'], ['1F1F7;&#x1F1EA', 'Réunion'], ['1F1F7;&#x1F1F4', 'Romania'], ['1F1F7;&#x1F1F8', 'Serbia'], ['1F1F7;&#x1F1FA', 'Russia'], ['1F1F7;&#x1F1FC', 'Rwanda'], ['1F1F8;&#x1F1E6', 'Saudi Arabia'], ['1F1F8;&#x1F1E7', 'Solomon Islands'], ['1F1F8;&#x1F1E8', 'Seychelles'], ['1F1F8;&#x1F1E9', 'Sudan'], ['1F1F8;&#x1F1EA', 'Sweden'], ['1F1F8;&#x1F1EC', 'Singapore'], ['1F1F8;&#x1F1ED', 'St. Helena'], ['1F1F8;&#x1F1EE', 'Slovenia'], ['1F1F8;&#x1F1EF', 'Svalbard & Jan Mayen'], ['1F1F8;&#x1F1F0', 'Slovakia'], ['1F1F8;&#x1F1F1', 'Sierra Leone'], ['1F1F8;&#x1F1F2', 'San Marino'], ['1F1F8;&#x1F1F3', 'Senegal'], ['1F1F8;&#x1F1F4', 'Somalia'], ['1F1F8;&#x1F1F7', 'Suriname'], ['1F1F8;&#x1F1F8', 'South Sudan'], ['1F1F8;&#x1F1F9', 'São Tomé & Príncipe'], ['1F1F8;&#x1F1FB', 'El Salvador'], ['1F1F8;&#x1F1FD', 'Sint Maarten'], ['1F1F8;&#x1F1FE', 'Syria'], ['1F1F8;&#x1F1FF', 'Swaziland'], ['1F1F9;&#x1F1E6', 'Tristan Da Cunha'], ['1F1F9;&#x1F1E8', 'Turks & Caicos Islands'], ['1F1F9;&#x1F1E9', 'Chad'], ['1F1F9;&#x1F1EB', 'French Southern Territories'], ['1F1F9;&#x1F1EC', 'Togo'], ['1F1F9;&#x1F1ED', 'Thailand'], ['1F1F9;&#x1F1EF', 'Tajikistan'], ['1F1F9;&#x1F1F0', 'Tokelau'], ['1F1F9;&#x1F1F1', 'Timor-Leste'], ['1F1F9;&#x1F1F2', 'Turkmenistan'], ['1F1F9;&#x1F1F3', 'Tunisia'], ['1F1F9;&#x1F1F4', 'Tonga'], ['1F1F9;&#x1F1F7', 'Turkey'], ['1F1F9;&#x1F1F9', 'Trinidad & Tobago'], ['1F1F9;&#x1F1FB', 'Tuvalu'], ['1F1F9;&#x1F1FC', 'Taiwan'], ['1F1F9;&#x1F1FF', 'Tanzania'], ['1F1FA;&#x1F1E6', 'Ukraine'], ['1F1FA;&#x1F1EC', 'Uganda'], ['1F1FA;&#x1F1F2', 'U.S. Outlying Islands'], ['1F1FA;&#x1F1F3', 'United Nations'], ['1F1FA;&#x1F1F8', 'United States'], ['1F1FA;&#x1F1FE', 'Uruguay'], ['1F1FA;&#x1F1FF', 'Uzbekistan'], ['1F1FB;&#x1F1E6', 'Vatican City'], ['1F1FB;&#x1F1E8', 'St. Vincent & Grenadines'], ['1F1FB;&#x1F1EA', 'Venezuela'], ['1F1FB;&#x1F1EC', 'British Virgin Islands'], ['1F1FB;&#x1F1EE', 'U.S. Virgin Islands'], ['1F1FB;&#x1F1F3', 'Vietnam'], ['1F1FB;&#x1F1FA', 'Vanuatu'], ['1F1FC;&#x1F1EB', 'Wallis & Futuna'], ['1F1FC;&#x1F1F8', 'Samoa'], ['1F1FD;&#x1F1F0', 'Kosovo'], ['1F1FE;&#x1F1EA', 'Yemen'], ['1F1FE;&#x1F1F9', 'Mayotte'], ['1F1FF;&#x1F1E6', 'South Africa'], ['1F1FF;&#x1F1F2', 'Zambia'], ['1F1FF;&#x1F1FC', 'Zimbabwe'], ['1F3F4;&#xE0067;&#xE0062;&#xE0065;&#xE006E;&#xE0067;&#xE007F', 'Flag for England (GB-ENG)'], ['1F3F4;&#xE0067;&#xE0062;&#xE0073;&#xE0063;&#xE0074;&#xE007F', 'Flag for Scotland (GB-SCT)'], ['1F3F4;&#xE0067;&#xE0062;&#xE0077;&#xE006C;&#xE0073;&#xE007F', 'Flag for Wales (GB-WLS)'], ['1F3F4;&#x200D;&#x2620;&#xFE0F', 'Pirate Flag']],
  536. dingbatsAndMore: (function () {
  537. var base = ['1f4b9', '1f4b2', '1f4b1', 'a9', 'ae', '1f518', '3030', '2588', '2318'], more = [[0x2600, 0x2647], [0x2654, 0x266F], [0xb0, 0xb0]], i;
  538.  
  539. more.forEach(function (range) {
  540. for (i = range[0]; i <= range[1]; i++) {
  541. base.push(i.toString(16));
  542. }
  543. });
  544.  
  545. return base;
  546. }())
  547. },
  548. events: {
  549. toolbar: function (e, parent) {
  550. e.preventDefault();
  551.  
  552. if (e.target.classList.contains('gm_emoji_selected')) {
  553. return;
  554. }
  555.  
  556. var css = '.gm_emoji_' + e.target.getAttribute('data-toolbar'), div = parent.querySelector(css);
  557.  
  558. Array.from(parent.querySelectorAll('.gm_emoji_list a.gm_emoji_hidden')).forEach(function (e) {
  559. e.classList.remove('gm_emoji_hidden');
  560. });
  561.  
  562. Array.from(parent.querySelectorAll('.gm_emoji_list:not(.gm_emoji_hidden):not(' + css + ')'))
  563. .forEach(function (e) { e.classList.add('gm_emoji_hidden'); });
  564.  
  565. Array.from(parent.querySelectorAll('.gm_emoji_toolbar_button'))
  566. .forEach(function (e) { e.classList.remove('gm_emoji_selected'); });
  567.  
  568. e.target.classList.add('gm_emoji_selected');
  569. div.classList.remove('gm_emoji_hidden');
  570. },
  571. search: function () {
  572. var title = this.value.trim(), reg, div = this.parentElement;
  573.  
  574. Array.from(div.querySelectorAll('.gm_emoji_list.gm_emoji_hidden'))
  575. .forEach(function (e) { e.classList.remove('gm_emoji_hidden'); });
  576.  
  577. if (title.length === 0) {
  578. Array.from(div.querySelectorAll('.gm_emoji_list a.gm_emoji_hidden')).forEach(function (e) {
  579. e.classList.remove('gm_emoji_hidden');
  580. });
  581. } else {
  582. reg = new RegExp(title, 'ig');
  583.  
  584. Array.from(div.querySelectorAll('.gm_emoji_list a')).forEach(function (e) { // a:not([title*='${title}' i])
  585. if (reg.test(e.title)) {
  586. // console.log(e.title);
  587. e.classList.remove('gm_emoji_hidden');
  588. } else {
  589. e.classList.add('gm_emoji_hidden');
  590. }
  591. });
  592.  
  593. div.querySelector('.gm_emoji_box').scrollTop = 0;
  594. }
  595. },
  596. insert: function (e, textarea) {
  597. var target = e.target.hasAttribute('data-char') ? e.target : e.target.parentElement, start, end, pre, post, point, emo, m;
  598. if (textarea.selectionStart < 0) { return; }
  599. if (target.hasAttribute('data-char')) {
  600. if (textarea.selectionEnd > textarea.value.length) {
  601. textarea.selectionEnd = textarea.value.length;
  602. }
  603. start = textarea.selectionStart || 0;
  604. end = textarea.selectionEnd || 0;
  605. pre = textarea.value.substring(0, start);
  606.  
  607. point = ucs2decode(pre);
  608. point = point[point.length - 1] || 0;
  609.  
  610. try {
  611. m = String.fromCodePoint(point).match(emoji.ranges);
  612. // console.log('pt:', String.fromCodePoint(point), point, m);
  613. emo = (point <= 32 || (m && m.length > 0) ? '' : ' ') + target.getAttribute('data-char');
  614. } catch (e) {
  615. // console.error(e);
  616. emo = (point <= 32 || point > 8500 ? '' : ' ') + target.getAttribute('data-char');
  617. }
  618.  
  619. post = emo + textarea.value.substring(end);
  620. // console.log('pre:', pre, 'emo:', emo, 'post:', textarea.value.substring(end));
  621. textarea.value = pre + post;
  622. textarea.focus();
  623. textarea.setSelectionRange(start + emo.length, start + emo.length);
  624. }
  625. },
  626. wire: function (div, textarea) {
  627. div.addEventListener('click', function (e) {
  628. if (e.target.hasAttribute('data-toolbar')) {
  629. emoji.events.toolbar(e, div);
  630. return;
  631. }
  632.  
  633. if (e.which === 1) {
  634. e.preventDefault();
  635. emoji.events.insert(e, textarea);
  636. }
  637. }, false);
  638. }
  639. },
  640. make: function (parent, textarea) {
  641. var div = $.e('div', { className: 'gm_emoji' });
  642. parent.appendChild(div);
  643. this.presentation.panel(div);
  644. this.events.wire(div, textarea);
  645. },
  646. css: (function () {
  647.  
  648. dom.css('.gm_emoji_selected { opacity: .7; } .gm_emoji_toolbar_button { cursor: pointer; }');
  649. dom.css('.gm_emoji_hidden { display: none !important; }');
  650. dom.css('.gm_emoji_toolbar > div { display: inline-block; }');
  651.  
  652. /* Windows Chromium Browser Emoji Fix
  653. * Emoji in Chrome-based browsers!
  654. *
  655. * Modifies font-family to include fonts that display Emoji
  656. *
  657. * Windows 7 Users: Install Symbola.ttf if you do not see all Emoji
  658. * Find it here: http://users.teilar.gr/~g1951d/
  659. */
  660. if (window.chrome) { // && /(?:win)/i.test(navigator.platform)
  661. dom.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; }');
  662. }
  663. }())
  664. };
  665.  
  666. // Quick Editor
  667. Editor = (function () {
  668. function Editor(el) {
  669. this.id = el.name.split('::')[2];
  670. this.show = 0;
  671. this.num = document.querySelector('a[href^="showpost.php?p=' + this.id + '&postcount="]').textContent; //.href.match(/(?:postcount=)(\d+)/)[1];
  672. this.dom(el);
  673. Editor.set[this.id] = this;
  674. }
  675.  
  676. Editor.set = {};
  677.  
  678. Editor.toggle = {
  679. displays : [
  680. ['block', 'none', 'none'],
  681. ['none', 'block', 'inline']
  682. ],
  683. spin : ['none', 'inherit']
  684. };
  685.  
  686. Editor.url = {
  687. gets : 'ajax.php?do=quickedit&p=',
  688. posts : 'editpost.php?do=updatepost&postid=undefined'
  689. };
  690.  
  691. Editor.fromLink = function (link) {
  692. link.textContent += '+';
  693. link.title = 'Quick Editor++';
  694. link.onclick = null;
  695. link.dataset.qqre = true;
  696. return new Editor(link);
  697. };
  698.  
  699. Editor.fromOldLink = function (link) {
  700. return new Editor(link);
  701. };
  702.  
  703. Editor.events = {
  704. quickEdit: function (e) {
  705. var qe;
  706. e.preventDefault();
  707.  
  708. if (e.target && e.target.hasAttribute('data-qqre')) {
  709. qe = Editor.set[e.target.dataset.qqre] || Editor.fromOldLink(e.target);
  710.  
  711. if (qe.x) {
  712. qe.toggle();
  713. } else {
  714. qe.gets();
  715. }
  716. }
  717. }
  718. };
  719.  
  720. Editor.prototype = {
  721. constuctor: Editor,
  722. dom: function (el) {
  723. this.el = el;
  724. this.el.dataset.qqre = this.id;
  725. // this.el.addEventListener('click', Editor.events.quickEdit.bind(this), false);
  726. this.spin = document.getElementById('progress_' + this.id);
  727. this.post = document.getElementById('post_message_' + this.id);
  728. this.div = $.e('div', {className : 'inpost'}, this.post.parentNode);
  729. },
  730. gets: function () {
  731. this.x = new XMLHttpRequest();
  732. this.x.onprogress = bond(this, this.spinner)(true);
  733. this.x.open('POST', Editor.url.gets + this.id, true);
  734. this.x.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded'); //charset=UTF-8
  735. this.x.setRequestHeader('X-Requested-With', 'XMLHttpRequest');
  736. this.x.onload = this.load.bind(this);
  737. this.x.onerror = this.error.bind(this);
  738. this.x.send('securitytoken=' + Form.token() + '&do=quickedit&p=' + this.id + '&editorid=vB_Editor_QE_1');
  739. },
  740. buildPost: function () {
  741. return [
  742. 'securitytoken=',
  743. Form.token(),
  744. '&do=updatepost&ajax=1&postid=',
  745. this.id,
  746. '&wysiwyg=0&message=',
  747. $.uriHtmlEncode(this.div.querySelector('textarea').value),
  748. '&reason=',
  749. $.uriHtmlEncode(this.div.querySelector('input[name="reason"]').value),
  750. '&postcount=',
  751. this.num
  752. ].join('');
  753. },
  754. posts : function (e) {
  755. e.preventDefault();
  756. var s = this.buildPost(); //.call(this);
  757. this.p = new XMLHttpRequest();
  758. this.p.open('POST', Editor.url.posts + this.id, true);
  759. this.p.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded; charset=UTF-8');
  760. this.p.setRequestHeader('X-Requested-With', 'XMLHttpRequest');
  761. this.p.setRequestHeader('Content-Length', s.length);
  762. this.p.onload = this.update.bind(this);
  763. this.p.onerror = this.error.bind(this);
  764. this.p.send(s);
  765. },
  766. toggle : function () {
  767. var toggle = Editor.toggle.displays[this.show];
  768. this.el.style.display = toggle[2];
  769. this.div.style.display = toggle[0];
  770. this.post.style.display = toggle[1];
  771. app.state.activeBox = this.show ? app.state.form.elements.message : this.text;
  772. this.show = Number(!this.show);
  773. },
  774. spinner : function (show) {
  775. this.spin.style.display = Editor.toggle.spin[Number(!!show)];
  776. },
  777. load: function () {
  778. var t = this.xmlhtml(this.x, 'editor', 'textarea'), f;
  779. if (t && t[0]) {
  780. f = new Form(this.id, this.div, t[0].value, false, true, t[1]);
  781.  
  782. this.extra = new Extra(this.id, true);
  783. this.text = this.div.querySelector('textarea');
  784.  
  785. this.div.querySelector('.cancel_button').addEventListener('click', this.toggle.bind(this), false);
  786. document.getElementById('vB_Editor_' + this.id + '_save').addEventListener('click', this.posts.bind(this), false);
  787.  
  788. dom.js(';(function () { $vb_hook(' + this.id + '); }());');
  789. }
  790. this.spinner();
  791. this.toggle();
  792. return f;
  793. },
  794. update: function () {
  795. var post = this.xmlhtml(this.p, 'postbit', '.post');
  796. try {
  797. this.post.innerHTML = post[0].innerHTML;
  798. this.toggle();
  799. } catch (e) {} //console.log(e);
  800. },
  801. error : function () {
  802. alert('QQRE: Request failed.');
  803. this.spin.style.display = 'none';
  804. this.x = null;
  805. },
  806. xmlhtml : function (attr, b, selector) { // b may be unnecessary
  807. try {
  808. // console.log(attr, this[attr].responseXML);
  809. var div = $.e('div', {innerHTML: attr.responseText}), r = '';
  810. // div.innerHTML = attr.responseText;
  811. div = div.querySelector(b);
  812. try {
  813. r = div.querySelector('editor[reason]').getAttribute('reason');
  814. } catch (er) {}
  815. return [div.querySelector(selector), r];
  816. } catch (e) {}
  817. }
  818. };
  819. return Editor;
  820. }());
  821.  
  822. // Form Maker
  823. Form = (function () {
  824.  
  825. function Form(id, parent, message, title, edit, reason) {
  826. if (!id || !parent) { return; }
  827.  
  828. this.id = id;
  829. this.el = parent;
  830. this.message = $.h(message || '');
  831. this.title = $.h(title || '');
  832. this.edit = !!edit;
  833. this.reason = $.h(reason || '');
  834.  
  835. this.elements = {};
  836.  
  837. this.make();
  838. }
  839.  
  840. Form.PRIMARY = 'quickreplyform_primary';
  841.  
  842. Form.POST_KEY = 'uQQR.post';
  843.  
  844. Form.STORAGE_KEY = 'uQQR';
  845.  
  846. Form.userStorage = strg.read(Form.STORAGE_KEY);
  847.  
  848. Form.threadId = null;
  849.  
  850. Form.postId = null;
  851.  
  852. Form.shared = {
  853. elements: {
  854. main: document.querySelector('#main .wrap'),
  855. closed: !!document.querySelector('a.large-button.disabled'),
  856. logged: document.querySelector('#usercptools'),
  857. postbits: function () { return document.querySelectorAll('.postbit:not(.ignored):not([data-qqre])'); },
  858. editButtons: function () { return document.querySelectorAll('a[href*="editpost"]:not([data-qqre])'); }
  859. },
  860. scriptVersion: (function () {
  861. try {
  862. return '?v=' + document.querySelector('script[src*="v="]').src.split('v=')[1];
  863. } catch (e) { return ''; }
  864. }()),
  865. reg: {
  866. '<blockquote>' : '[indent]',
  867. '<blockquote class\\="quote code">[^]+?<pre>([^]+?)<\\/pre>[^]+?<\\/blockquote>' : '[code]$1[/code]',
  868. '<\\/blockquote>' : '[/indent]',
  869. '<li>' : '[*]',
  870. '<\\/li>' : '',
  871. '<b>' : '[b]',
  872. '<\\/b>' : '[/b]',
  873. '<i>' : '[i]',
  874. '<\\/i>' : '[/i]',
  875. '<strike>' : '[strike]',
  876. '<\\/strike>' : '[/strike]',
  877. '<u>' : '[u]',
  878. '<\\/u>' : '[/u]',
  879. '<ul>' : '[list]',
  880. '<\\/ul>' : '[/list]\n',
  881. '<ol [^]+?>' : '[list=1]',
  882. '<\\/ol>' : '[/list]\n',
  883. '<img[^]+?src\\="data\\:image[^]+?>' : '',
  884. '<img[^]+?src\\="([^]+?)"[^]+?>' : '[img]$1[/img]',
  885. '<a href\\="mailto\\:([^]+?)">([^]+?)<\\/a>' : '[email="$1"]$2[/email]',
  886. '<a[^]+?href\\="([^]+?)"[^]+?>' : '[url="$1"]',
  887. '</a>' : '[/url]',
  888. '<span style\\="color\\: #e21212">([^]+?)<\\/span>' : '$1',
  889. '<span class\\="?spoiler".*?>([^]+?)<\\/span>' : '[spoiler]$1[/spoiler]',
  890. '<span class\\="?spoiler .*?".*?>([^]+?)<\\/span>' : '[spoiler]$1[/spoiler]',
  891. '<span class\\="highlight">([^]+?)<\\/span>' : '[highlight]$1[/highlight]',
  892. '<br.?\\/?>' : '',
  893. '<span [^]+?<\\/span>' : '',
  894. '<div [^]+?<\\/div>' : '',
  895. '<.*?>' : '',
  896. '\\[I\'m an idiot\\.\\]' : 'lol'
  897. },
  898. buttons: ['removeformat', 'space', 'undo', 'redo', 'space', 'bold', 'italic', 'underline', 'space', 'insertorderedlist', 'insertunorderedlist', 'outdent', 'indent', 'space', 'createlink', 'unlink', 'email', 'insertimage', 'space', 'quote', 'code', 'space', 'resize'],
  899. quote: {
  900. $replaceImgWithUrlP1: function (text) {
  901. return text.replace(/\[img\]/i, 'Link[/url] : [url="');
  902. },
  903. replaceImgWithUrl: function (text) {
  904. return text.replace(/\[url=".+?"\]\[img\]/ig, Form.shared.quote.$replaceImgWithUrlP1)
  905. .replace(/\[\/img\]\[\/url/ig, '"]Image[/url')
  906. .replace(/(?:\[(\/?)img\])/ig, '[$1url]');
  907. },
  908. add: function (post) {
  909. if (Form.userStorage.u) {
  910. post = Form.shared.quote.replaceImgWithUrl(post);
  911. }
  912.  
  913. app.state.activeBox.value += app.state.activeBox.value ? ('\n\n' + post) : post;
  914.  
  915. if (app.state.activeBox.id === app.state.form.elements.message.id) {
  916. Form.events.saveStorage();
  917. }
  918. },
  919. unquote: function (post) {
  920. var div = $.e('div', {innerHTML: post}), e = div.querySelectorAll('.quote:not(.code)'), i = e.length;
  921. while (i--) {
  922. try {
  923. div.removeChild(e[i]);
  924. } catch (er) {}
  925. }
  926. return div.innerHTML;
  927. }
  928. },
  929. loadIds: function () {
  930. try {
  931. Form.threadId = +document.querySelector('.left a[href*="showthread.php?t="]').href.match(/t=(\d+)/)[1];
  932. Form.postId = +document.querySelector('a[href^="newreply.php?do=newreply&noquote=1&p="]').href.match(/p=(\d+)/)[1];
  933. } catch (e) {}
  934. return (Form.threadId && Form.postId);
  935. },
  936. tokenFromPage: function () {
  937. try {
  938. return document.getElementsByTagName('head')[0].textContent.match(/SECURITYTOKEN = "(.+?)"/)[1];
  939. } catch (e) { /*alert('Quick Quote, Reply and Edit encountered an error.'); */ return ''; }
  940. },
  941. loadScript: function () {
  942. var s = $.e('script', {type: 'text/javascript', src: '/forum/clientscript/vbulletin_textedit.js' + Form.shared.scriptVersion});
  943. s.addEventListener('load', function () {
  944. embedScript(function () {
  945. window.$vb_hook = function (id) {
  946. window.vB_Editor['vB_Editor_' + id] = new window.vB_Text_Editor('vB_Editor_' + id, 0, 3, 1);
  947. };
  948.  
  949. var vbphrase = window.vbphrase || [];
  950. vbphrase.enter_list_item = 'Enter a list item.';
  951. vbphrase.message_too_short = 'Your message is too short.';
  952. vbphrase.enter_link_url = 'URL:';
  953. vbphrase.enter_image_url = 'Image URL:';
  954. vbphrase.enter_email_link = 'E-mail:';
  955. window.vbphrase = vbphrase;
  956.  
  957. window.$vb_hook('001');
  958. });
  959. }, false);
  960. document.body.appendChild(s);
  961. }
  962. };
  963.  
  964. Form.events = {
  965. clickDelegate: function (e) {
  966. if (e.target && e.target.title === 'Quick Quote') {
  967. Form.events.quickQuote(e);
  968. }
  969. if (e.target && e.target.title === 'Quick Editor++') {
  970. Editor.events.quickEdit(e);
  971. }
  972. },
  973. quickQuote: function (e) {
  974. // if (e.target && e.target.title === 'Quick Quote') {
  975. e = e.target;
  976. var user = e.getAttribute('data-user'), pid = e.getAttribute('data-post-id'), id = e.getAttribute('data-id'),
  977. i, re, post = Form.shared.quote.unquote(document.getElementById(pid).innerHTML);
  978. for (i in Form.shared.reg) {
  979. if (Form.shared.reg.hasOwnProperty(i)) {
  980. re = new RegExp(i, 'ig');
  981. post = post.replace(re, Form.shared.reg[i]);
  982. }
  983. }
  984. post = '[quote=' + user + ';' + id + ']' + $.h(post) + '[/quote]\n';
  985. e.className = 'multiquotelink quickquotes quickquoted';
  986. Form.shared.quote.add(post);
  987. // }
  988. },
  989. submit: function (e) {
  990. if (Form.shared.elements.logged) {
  991. strg.save(Form.POST_KEY, {threadId: Form.threadId});
  992. } else {
  993. e.preventDefault();
  994. document.location.hash = '#';
  995. alert('Please login!');
  996. document.location.hash = '#top';
  997. Form.events.saveStorage();
  998. }
  999. },
  1000. toggleImageQuotes: function (e) {
  1001. var k = String.fromCharCode(e.which).toLowerCase();
  1002. if (e.altKey && e.shiftKey && (k === 't' || k === '\u02C7')) {
  1003. e.preventDefault();
  1004. Extra.click(app.state.form.elements.convert);
  1005. alert('Notice: Converting IMG to URL ' + (app.state.form.elements.convert.checked ? 'enabled.' : 'disabled.'));
  1006. }
  1007. },
  1008. saveStorage: function () {
  1009. app.model.saveStorage();
  1010. },
  1011. clear: function () {
  1012. if (this.value === 'Clear') {
  1013. app.state.form.elements.title.value = app.state.form.elements.message.value = '';
  1014. app.state.form.elements.message.focus();
  1015. }
  1016. app.model.clearStorage();
  1017. },
  1018. textStore: function (e) {
  1019. sessionStorage.setItem('qqre-temp', JSON.stringify({
  1020. id: Form.threadId,
  1021. title: app.state.form.elements.title.value,
  1022. text: e.target.value,
  1023. z: +(new Date())
  1024. }));
  1025. }
  1026. };
  1027.  
  1028. Form.token = function () {
  1029. var s = greaseWindow.SECURITYTOKEN;
  1030. return typeof s === 'string' ? s : '';
  1031. };
  1032.  
  1033. Form.canAdd = function () {
  1034. return Form.shared.elements.main && Form.shared.loadIds();
  1035. };
  1036.  
  1037. Form.removePreviousPost = function () {
  1038. if (Form.hasJustPosted()) {
  1039. // console.log('I think you have just posted, so I will clear previous post . . .');
  1040. app.model.clearStorage();
  1041. }
  1042. };
  1043.  
  1044. Form.hasJustPosted = function () {
  1045. var id,
  1046. tid = +strg.grab(Form.POST_KEY).threadId,
  1047. hash = document.location.hash.substr(1),
  1048. last = (function (posts) { return posts.length > 0 ? posts[posts.length - 1] : null; }(document.querySelectorAll('.mypost')));
  1049.  
  1050. console.log('just posted?');
  1051. if (tid === Form.threadId && last) {
  1052. id = last.id;
  1053. console.log(id, hash, tid);
  1054.  
  1055. if (/(?:posted=1)/.test(document.location.href) && id === hash) {
  1056. console.log('just posted : 1');
  1057. return true;
  1058. }
  1059. if (/(?:post(\d+))/.test(document.location.hash) && id === hash) {
  1060. console.log('just posted : 2');
  1061. return true;
  1062. }
  1063. }
  1064. return false;
  1065. };
  1066.  
  1067. Form.prototype = {
  1068. constructor: Form,
  1069. lastTabIndex: function () {
  1070. var tabs = document.querySelectorAll('[tabindex]'), i, tabIndex = 0;
  1071.  
  1072. for (i = 0; i < tabs.length; i++) {
  1073. if (tabs[i].tabIndex > tabIndex) {
  1074. tabIndex = tabs[i].tabIndex;
  1075. }
  1076. }
  1077.  
  1078. return tabIndex;
  1079. },
  1080. make: function () {
  1081. 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),
  1082. d = $.e('div', {className: this.id === '001' ? 'alt1 newreplybox' : 'alt2 newreplybox'}, f),
  1083. p = $.e('p', null, d),
  1084. editor = $.e('div', {id: 'vB_Editor_' + this.id, className: 'vBulletin_editor'}, d),
  1085. container = $.e('div', {innerHTML: '<div class="text_emo_container"><div class="text_emo_container_text"></div><div class="text_emo_container_emo"></div></div>'}, d),
  1086. r = $.e('div', {className: 'quickreplyform_hotspot'}, d),
  1087. index = this.lastTabIndex() + 1;
  1088.  
  1089. this.elements.buttons = $.e('div', {className: 'quickreplyform_buttons'}, r);
  1090. 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});
  1091. this.elements.control = $.e('div', {id: 'vB_Editor_' + this.id + '_controls', className: 'control'}, editor);
  1092.  
  1093. container.querySelector('.text_emo_container_text').appendChild(this.elements.message);
  1094. emoji.make(container.querySelector('.text_emo_container_emo'), this.elements.message);
  1095.  
  1096. if (this.edit) {
  1097. this.elements.reason = $.e('input', {className: 'biginput', type: 'text', title: 'Optional', maxlength: 125, size: 50, name: 'reason', value: this.reason, placeholder: 'Reason for editing'}, p);
  1098. } else {
  1099. this.elements.title = $.e('input', {size: 50, name: 'title', className: 'biginput', type: 'text', value: this.title, placeholder: 'Title'}, p);
  1100. }
  1101.  
  1102. $.e('input', {type: 'hidden', name: 'securitytoken', id: '', value: Form.token()}, r);
  1103. $.e('input', {type: 'hidden', name: 'wysiwyg', id: '', value: 0}, r);
  1104. $.e('input', {type: 'hidden', name: 's', value: ''}, r);
  1105. $.e('input', {type: 'hidden', name: 'do', value: this.id === '001' ? 'postreply' : 'updatepost'}, r);
  1106. $.e('input', {type: 'hidden', name: 't', value: Form.threadId}, r);
  1107. $.e('input', {type: 'hidden', name: 'p', value: this.id === '001' ? Form.postId : this.id}, r);
  1108. $.e('input', {type: 'hidden', name: 'posthash', value: ''}, r);
  1109. $.e('input', {type: 'hidden', name: 'poststarttime', value: ''}, r);
  1110. $.e('input', {type: 'hidden', name: 'parseurl', value: 1}, r);
  1111.  
  1112. 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);
  1113.  
  1114. if (this.edit) {
  1115. 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);
  1116. }
  1117.  
  1118. 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);
  1119.  
  1120. $.e('div', {innerHTML: '<p style="text-align:center" onclick="this.nextSibling.style.display = this.nextSibling.style.display === \'block\' ? \'none\' : \'block\'"><span class="qqre_shortcuts" title="Click to see.">View <b class="qqreb">' + (Extra.mac ? '&#8984;' : 'CTRL') + '</b> Shortcuts</span></p><ul id="qqreshortcuts" style="display:none"><li><b class="qqreb">B</b> Bold</li><li><b class="qqreb">I</b> Italic</li><li><b class="qqreb">U</b> Underline</li><li><b class="qqreb">S</b> Strike</li><li><b class="qqreb">H</b> Highlight</li><li><b class="qqreb">P</b> Spoiler</li><li><b class="qqreb">L</b> URL</li><li><b class="qqreb">M</b> Image</li><li><b class="qqreb">E</b> Save*</li><li><b class="qqreb">D</b> Clear*</li><li>*Only for quick reply box.</li></ul>'}, r);
  1121.  
  1122. this.addButtons();
  1123.  
  1124. this.elements.body = f;
  1125. this.elements.hotspot = r;
  1126. },
  1127. makePrimary: function () {
  1128. this.elements.body.id = Form.PRIMARY;
  1129.  
  1130. 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'});
  1131. this.elements.buttons.insertBefore(this.elements.save, this.elements.submit.nextSibling);
  1132.  
  1133. 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);
  1134. this.elements.temp = this.elements.message;
  1135.  
  1136. this.elements.subscribe = $.e('div', {innerHTML: '<label><input type="checkbox"> Modify subscription<label><br>'}, this.elements.hotspot);
  1137. this.elements.subscription = $.e('select', {id: 'qqre_subs', name: 'emailupdate', innerHTML: '<option value="9999">Do not subscribe</option> <option value="0">No email notification</option> <option value="1">Instant email notification</option> <option value="2">Daily email notification</option> <option value="3">Weekly email notification</option>'});
  1138. this.elements.convert = $.e('label', {'for': 'qqre_img', innerHTML: '<input type="checkbox" id="qqre_img"> Convert [IMG] to [URL] tags on quoted posts (ALT+SHIFT+T)', title: 'Convert [IMG] to [URL]', id: 'qqre_labelimg2url'}, this.elements.hotspot);
  1139. this.elements.convert = app.state.form.elements.convert.querySelector('input');
  1140.  
  1141. this.wire();
  1142. },
  1143. addButtons: function () {
  1144. var i, j = Form.shared.buttons.length;
  1145.  
  1146. for (i = 0; i < j; i++) {
  1147. this.createButton(Form.shared.buttons[i]);
  1148. }
  1149. },
  1150. createButton: function (type) {
  1151. var buttonId;
  1152. switch (type) {
  1153. case 'space':
  1154. $.e('img', {width: 6, height: 20, alt: '', src: 'images/neogaf/editor/separator.gif'}, $.e('div', false, this.elements.control));
  1155. break;
  1156. case 'quote':
  1157. case 'code':
  1158. buttonId = 'vB_Editor_' + this.id + '_cmd_wrap0_';
  1159. break;
  1160. case 'resize':
  1161. $.a($.e('div', {className: 'imagebutton resize_merger'}, this.elements.control),
  1162. $.e('div', {className: 'imagebutton', id: 'vB_Editor_' + this.id + '_cmd_resize_0_100'}, $.e('img', {src: 'images/neogaf/editor/resize_0.gif', height: 9, width: 21, alt: 'Decrease Size'}), true),
  1163. $.e('div', {className: 'imagebutton', id: 'vB_Editor_' + this.id + '_cmd_resize_1_100'}, $.e('img', {src: 'images/neogaf/editor/resize_1.gif', width: 21, height: 9, alt: 'Increase Size'}), true));
  1164. break;
  1165. default:
  1166. buttonId = 'vB_Editor_' + this.id + '_cmd_';
  1167. break;
  1168. }
  1169. if (buttonId) {
  1170. $.e('img', {src: 'images/neogaf/editor/' + type + '.gif', width: 21, height: 20, alt: type}, $.e('div', {className: 'imagebutton', id: buttonId + type}, this.elements.control));
  1171. }
  1172. },
  1173. wire: function () {
  1174. var sub = this.elements.subscribe;
  1175.  
  1176. this.elements.message.addEventListener('input', throttle(Form.events.textStore, 250), false);
  1177. this.elements.submit.addEventListener('click', Form.events.submit, false);
  1178. this.elements.save.addEventListener('click', Form.events.saveStorage, false);
  1179. this.elements.clear.addEventListener('click', Form.events.clear, false);
  1180. this.elements.convert.addEventListener('click', Form.events.saveStorage, false);
  1181. sub.querySelector('input[type="checkbox"]').addEventListener('click', function () {
  1182. if (this.checked) {
  1183. sub.appendChild(app.state.form.elements.subscription);
  1184. } else {
  1185. sub.removeChild(app.state.form.elements.subscription);
  1186. }
  1187. }, false);
  1188.  
  1189. // document.addEventListener('click', Form.events.quickQuote, false);
  1190. document.addEventListener('click', Form.events.clickDelegate, false);
  1191. window.addEventListener(Extra.keyEvent, Form.events.toggleImageQuotes, false);
  1192. }
  1193. };
  1194.  
  1195. return Form;
  1196.  
  1197. }());
  1198.  
  1199. app = {
  1200. state: {
  1201. activeBox: null,
  1202. form: null
  1203. },
  1204. path: (function () {
  1205. document.location.pathname.match(/(\w+)\.php/);
  1206. return RegExp.lastParen;
  1207. }()),
  1208. main: function () {
  1209. if (Form.token()) {
  1210. app.go();
  1211. } else {
  1212. console.log('Token unavailable . . .');
  1213.  
  1214. if (window.postMessage) {
  1215. console.log('Adding message');
  1216.  
  1217. window.addEventListener('message', function (event) { // add messenger listener to this page
  1218. if (event.data.type && (event.data.type === 'QQRE_PAGE')) {
  1219. greaseWindow[event.data.prop] = event.data.val;
  1220. if (event.data.prop === 'SECURITYTOKEN') {
  1221. app.go();
  1222. }
  1223. }
  1224. }, false);
  1225.  
  1226. // post message from page scope
  1227. embedScript(function () {
  1228. window.postMessage({ type: 'QQRE_PAGE', prop: 'SECURITYTOKEN', val: window.SECURITYTOKEN }, '*');
  1229. });
  1230. } else {
  1231. Form.token = Form.shared.tokenFromPage;
  1232. if (Form.token()) {
  1233. app.go();
  1234. }
  1235. }
  1236. }
  1237. // console.log(app);
  1238. },
  1239. go: function () {
  1240. (this.actions[this.path] || this.actions.$default)();
  1241. },
  1242. actions: {
  1243. $default: function () {
  1244. try {
  1245. Form.threadId = +document.querySelector('[name="t"]').value;
  1246. } catch (e) {}
  1247.  
  1248. app.views.single(document.getElementById('vB_Editor_001_textarea'));
  1249. },
  1250. 'private': function () {
  1251. app.views.single(document.querySelector('#vB_Editor_QR_textarea, #vB_Editor_001_textarea'));
  1252. },
  1253. showthread: function () {
  1254. app.views.quickReply();
  1255. }
  1256. },
  1257. assets: {
  1258. button: ''
  1259. },
  1260. views: {
  1261. single: function (textarea) {
  1262. var parent, id, container;
  1263.  
  1264. if (!textarea) { return; }
  1265.  
  1266. parent = textarea.parentElement;
  1267. container = $.e('div', {innerHTML: '<div class="text_emo_container"><div class="text_emo_container_text"></div><div class="text_emo_container_emo"></div></div>'});
  1268.  
  1269. container.querySelector('.text_emo_container_text').appendChild(textarea);
  1270. textarea.removeAttribute('style');
  1271. parent.appendChild(container);
  1272. emoji.make(container.querySelector('.text_emo_container_emo'), textarea);
  1273.  
  1274. id = textarea.id.match(/vB_Editor_(\w+)_/)[1];
  1275.  
  1276. if (document.getElementById('vB_Editor_001_save')) {
  1277. document.getElementById('vB_Editor_001_save').addEventListener('click', Form.events.submit, false);
  1278. }
  1279.  
  1280. return new Extra(id);
  1281. },
  1282. quickReply: function () {
  1283. if (Form.token() && !Form.shared.elements.closed) {
  1284. dom.css('.quickreplyformp div.vBulletin_editor { background: transparent; border: 0; padding: 0 } #quickreplybox { position:fixed; top:0; left:0; width: 100%; background: transparent; text-align: center; } #quickreplybox span { background: #4aa4b7; color: #fff; padding: 3px; border-radius: 3px } #quickreplybox span:hover { background: #47a947; cursor: pointer} .hide { display: none; } .newreplybox { font-family: Arial, Verdana, sans-serif; max-width: 800px; margin: auto; padding: 9px 8px 6px} .inpost .newreplybox { width: 90% !important; border: 1px solid #ccc} .quickreplyformp textarea {font-size:110%; display:block; margin: auto} .newreplybox .vBulletin_editor { border: 0 none !important; } .newreplybox p { margin: .5em 0 .1em; text-align: center} .quickquotes, .quickquoted { cursor: pointer; background: url(' + app.assets.button + ') no-repeat; -moz-background-size: 16px 16px; background-size: 16px 16px; width:16px; height:16px; margin-right:2px; transition-duration: .2s; } .quickquotes:hover{opacity:.8} .quickquotes:active,.quickquoted,.quickquoted:hover{opacity:.5} #quickreplyformpoff { position: fixed; z-index: 1000; top: 28px; left: 150px; } #vB_Editor_001 { border: none; background: transparent; margin: 0; padding: 0 } \n.newreplybox .imagebutton { border: 0 !important; padding: 1px !important; margin: 0 2px;} .resize_merger.imagebutton { background: transparent !important; padding: 0 !important; vertical-align: middle; } .newreplybox .imagebutton:hover img { background-color: #C1D2EE; border-radius: 2px } .newreplybox .control { text-align:center; padding: 0px; margin: 4px auto 2px auto; } .newreplybox .control > div { display:inline-block } .newreplybox .large-button.submit:focus { outline: none;border-radius:3px; background-color:#01518E; border:0; color:#eee;} .newreplybox .large-button.submit:active{ outline: none; border-radius: 3px; background-color:#666; border:0; color:#fff; } \n.post { min-height: 108px } .editarea .newreplybox { width: 650px; margin: 12px 6px } .editarea textarea { width: 99% !important; } .editarea textarea, .biginput { border:1px solid #bbb } .newreplybox input[type="radio"],.newreplybox input[type="checkbox"]{width:13px;height:13px;padding:0;margin:0 0 0 4px;position:relative;top:-1px;}.inpost .newreplybox{width: 94%} .qqre_shortcuts {font-weight:bold;cursor:pointer} .qqre_shortcuts:hover {text-decoration:underline} #qqreshortcuts { margin: 5px auto; padding: 0; text-align: center; width:500px} #qqreshortcuts li {text-align:left; display: inline-block; padding: 0 3px; width:7em} #qqreshortcuts li:last-child{text-align:center;display:block;width:auto} #vB_Editor_001_cmd_wrap0_spoiler { visibility:hidden !important } .qqreb {line-height:1.2;color:#333;font-family:"Andale Mono",Consolas,monospace;background: none repeat scroll 0 0 rgba(248, 245, 245, 0.62); border: 1px solid #b1b1b1; border-radius: 3px; display: inline-block; padding: 0 5px; }');
  1285.  
  1286. app.views.editLinks();
  1287. if (Form.canAdd()) {
  1288. Form.removePreviousPost();
  1289. app.views.initialForm();
  1290. }
  1291. }
  1292. },
  1293. initialForm: function () {
  1294. if (!document.querySelector(Form.PRIMARY)) {
  1295. // console.log('creating new primary form');
  1296.  
  1297. app.state.form = new Form('001', Form.shared.elements.main);
  1298. app.state.form.makePrimary();
  1299.  
  1300. Form.shared.loadScript();
  1301.  
  1302. app.state.form.extra = new Extra(app.state.form.id, true);
  1303. app.state.form.extra.BB.clear = app.state.form.elements.clear;
  1304. app.state.form.extra.BB.save = app.state.form.elements.save;
  1305.  
  1306. app.model.init();
  1307.  
  1308. app.state.form.elements.convert.checked = Form.userStorage.u;
  1309.  
  1310. app.state.activeBox = app.state.form.elements.message;
  1311. app.views.addQuickButtons();
  1312.  
  1313. // document.dispatchEvent(new CustomEvent('LiveThreadUpdate'));
  1314. document.addEventListener('LiveThreadUpdate', app.events.liveUpdate, false);
  1315. }
  1316. },
  1317. editLinks: function () {
  1318. embedScript(function () {
  1319. // prevents Chrome from opening the default editor
  1320. var i, a = document.querySelectorAll('a.post-button');
  1321. for (i = 0; i < a.length; i++) {
  1322. a[i].onclick = null;
  1323. }
  1324. });
  1325.  
  1326. var buttons = Form.shared.elements.editButtons(), i = buttons.length;
  1327. while (i--) {
  1328. Editor.fromLink(buttons[i]);
  1329. }
  1330. },
  1331. addQuickButtons: function () {
  1332. return app.state.form && !Array.from(Form.shared.elements.postbits()).forEach(function (postbit) {
  1333. var post, button, user, quick;
  1334.  
  1335. postbit.dataset.qqre = true;
  1336. button = postbit.querySelector('.multiquotelink, .post-button');
  1337.  
  1338. if (button) {
  1339. post = postbit.querySelector('.post');
  1340. user = postbit.querySelector('.postbit-details-username a') || {};
  1341.  
  1342. quick = $.e('i', { title: 'Quick Quote', '$data-user': user.textContent, '$data-post-id': post.id, '$data-id': post.id.match(/\d+/)[0], className: 'multiquotelink quickquotes' });
  1343. button.parentElement.insertBefore(quick, button.parentElement.firstElementChild);
  1344. }
  1345. });
  1346. }
  1347. },
  1348. model: {
  1349. init: function () {
  1350. if (strg.zero(Form.userStorage)) {
  1351. app.model.saveStorage();
  1352. }
  1353.  
  1354. this.textLoad();
  1355. },
  1356. saveStorage: function () {
  1357. Form.userStorage = {
  1358. i: Form.threadId,
  1359. t: app.state.form.elements.title.value,
  1360. c: app.state.form.elements.message.value,
  1361. u: app.state.form.elements.convert.checked,
  1362. z: +(new Date())
  1363. };
  1364.  
  1365. strg.save(Form.STORAGE_KEY, Form.userStorage);
  1366. // console.log('saveStorage', Form.userStorage);
  1367. },
  1368. clearStorage: function () {
  1369. Form.userStorage = {i: 0, t: '', c: '', u: app.state.form ? app.state.form.elements.convert.checked : Form.userStorage.u};
  1370. strg.save(Form.STORAGE_KEY, Form.userStorage);
  1371. strg.wipe(Form.POST_KEY);
  1372. sessionStorage.removeItem('qqre-temp');
  1373. },
  1374. textLoad: function () {
  1375. var session = JSON.parse(sessionStorage.getItem('qqre-temp') || '{}'),
  1376. load = {title: '', text: ''};
  1377.  
  1378. if (Form.userStorage.i === Form.threadId) {
  1379. load.title = Form.userStorage.t;
  1380. load.text = Form.userStorage.c;
  1381. }
  1382.  
  1383. if (session.id === Form.threadId && (!Form.userStorage.z || Form.userStorage.z < session.z)) {
  1384. load.text = session.text;
  1385. load.title = session.title;
  1386. }
  1387.  
  1388. app.state.form.elements.title.value = load.title;
  1389. app.state.form.elements.message.value = load.text;
  1390. }
  1391. },
  1392. events: {
  1393. liveUpdate: function () {
  1394. console.log('qqre: live thread update');
  1395. app.views.editLinks();
  1396. app.views.addQuickButtons();
  1397. }
  1398. }
  1399. };
  1400.  
  1401. app.main();
  1402.  
  1403. }());