NOTICE: By continued use of this site you understand and agree to the binding Terms of Service and Privacy Policy.
// ==UserScript== // @name Twitch Chat Emotes // @namespace #Cletus // @version 2.1.5 // @description Adds a button to Twitch that allows you to "click-to-insert" an emote. // @copyright 2011+, Ryan Chatham <ryan.b.chatham@gmail.com> (https://github.com/cletusc) // @author Ryan Chatham <ryan.b.chatham@gmail.com> (https://github.com/cletusc) // @icon http://www.gravatar.com/avatar.php?gravatar_id=6875e83aa6c563790cb2da914aaba8b3&r=PG&s=48&default=identicon // @license MIT; http://opensource.org/licenses/MIT // @license CC BY-NC-SA 3.0; http://creativecommons.org/licenses/by-nc-sa/3.0/ // @homepage http://cletusc.github.io/Userscript--Twitch-Chat-Emotes/ // @supportURL https://github.com/cletusc/Userscript--Twitch-Chat-Emotes/issues // @contributionURL http://cletusc.github.io/Userscript--Twitch-Chat-Emotes/#donate // @grant none // @include http://*.twitch.tv/* // @include https://*.twitch.tv/* // @exclude http://api.twitch.tv/* // @exclude https://api.twitch.tv/* // @exclude http://tmi.twitch.tv/* // @exclude https://tmi.twitch.tv/* // @exclude http://*.twitch.tv/*/dashboard // @exclude https://*.twitch.tv/*/dashboard // @exclude http://chatdepot.twitch.tv/* // @exclude https://chatdepot.twitch.tv/* // @exclude http://im.twitch.tv/* // @exclude https://im.twitch.tv/* // @exclude http://platform.twitter.com/* // @exclude https://platform.twitter.com/* // @exclude http://www.facebook.com/* // @exclude https://www.facebook.com/* // ==/UserScript== /* Script compiled using build script. Script uses Browserify for CommonJS modules. */ (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){ var pkg = require('../package.json'); var publicApi = require('./modules/public-api'); var ember = require('./modules/ember-api'); var logger = require('./modules/logger'); var emotes = require('./modules/emotes'); var ui = require('./modules/ui'); logger.log('(v'+ pkg.version + ') Initial load on ' + location.href); // Only enable script if we have the right variables. //--------------------------------------------------- var initTimer = 0; (function init(time) { if (!time) { time = 0; } var objectsLoaded = ( window.Twitch !== undefined && window.jQuery !== undefined && ember.isLoaded() ); if (!objectsLoaded) { // Stops trying after 10 minutes. if (initTimer >= 600000) { logger.log('Taking too long to load, stopping. Refresh the page to try again. (' + initTimer + 'ms)'); return; } // Give an update every 10 seconds. if (initTimer % 10000) { logger.debug('Still waiting for objects to load. (' + initTimer + 'ms)'); } // Bump time up after 1s to reduce possible lag. time = time >= 1000 ? 1000 : time + 25; initTimer += time; setTimeout(init, time, time); return; } // Expose public api. if (typeof window.emoteMenu === 'undefined') { window.emoteMenu = publicApi; } ember.hook('route:channel', activate, deactivate); ember.hook('route:chat', activate, deactivate); activate(); })(); function activate() { ui.init(); emotes.init(); } function deactivate() { ui.hideMenu(); } },{"../package.json":7,"./modules/ember-api":8,"./modules/emotes":9,"./modules/logger":10,"./modules/public-api":11,"./modules/ui":15}],2:[function(require,module,exports){ (function (doc, cssText) { var id = "emote-menu-for-twitch-styles"; var styleEl = doc.getElementById(id); if (!styleEl) { styleEl = doc.createElement("style"); styleEl.id = id; doc.getElementsByTagName("head")[0].appendChild(styleEl); } if (styleEl.styleSheet) { if (!styleEl.styleSheet.disabled) { styleEl.styleSheet.cssText = cssText; } } else { try { styleEl.innerHTML = cssText; } catch (ignore) { styleEl.innerText = cssText; } } }(document, "/**\n" + " * Minified style.\n" + " * Original filename: \\node_modules\\jquery.scrollbar\\jquery.scrollbar.css\n" + " */\n" + ".scroll-wrapper{overflow:hidden!important;padding:0!important;position:relative}.scroll-wrapper>.scroll-content{border:none!important;-moz-box-sizing:content-box!important;box-sizing:content-box!important;height:auto;left:0;margin:0;max-height:none!important;max-width:none!important;overflow:scroll!important;padding:0;position:relative!important;top:0;width:auto!important}.scroll-wrapper>.scroll-content::-webkit-scrollbar{height:0;width:0}.scroll-element{display:none}.scroll-element,.scroll-element div{-moz-box-sizing:content-box;box-sizing:content-box}.scroll-element.scroll-x.scroll-scrollx_visible,.scroll-element.scroll-y.scroll-scrolly_visible{display:block}.scroll-element .scroll-arrow,.scroll-element .scroll-bar{cursor:default}.scroll-textarea{border:1px solid #ccc;border-top-color:#999}.scroll-textarea>.scroll-content{overflow:hidden!important}.scroll-textarea>.scroll-content>textarea{border:none!important;-moz-box-sizing:border-box;box-sizing:border-box;height:100%!important;margin:0;max-height:none!important;max-width:none!important;overflow:scroll!important;outline:0;padding:2px;position:relative!important;top:0;width:100%!important}.scroll-textarea>.scroll-content>textarea::-webkit-scrollbar{height:0;width:0}.scrollbar-inner>.scroll-element,.scrollbar-inner>.scroll-element div{border:none;margin:0;padding:0;position:absolute;z-index:10}.scrollbar-inner>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-inner>.scroll-element.scroll-x{bottom:2px;height:8px;left:0;width:100%}.scrollbar-inner>.scroll-element.scroll-y{height:100%;right:2px;top:0;width:8px}.scrollbar-inner>.scroll-element .scroll-element_outer{overflow:hidden}.scrollbar-inner>.scroll-element .scroll-bar,.scrollbar-inner>.scroll-element .scroll-element_outer,.scrollbar-inner>.scroll-element .scroll-element_track{border-radius:8px}.scrollbar-inner>.scroll-element .scroll-bar,.scrollbar-inner>.scroll-element .scroll-element_track{-ms-filter:\"progid:DXImageTransform.Microsoft.Alpha(Opacity=40)\";filter:alpha(opacity=40);opacity:.4}.scrollbar-inner>.scroll-element .scroll-element_track{background-color:#e0e0e0}.scrollbar-inner>.scroll-element .scroll-bar{background-color:#c2c2c2}.scrollbar-inner>.scroll-element.scroll-draggable .scroll-bar,.scrollbar-inner>.scroll-element:hover .scroll-bar{background-color:#919191}.scrollbar-inner>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_track{left:-12px}.scrollbar-inner>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_track{top:-12px}.scrollbar-inner>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-12px}.scrollbar-inner>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-12px}.scrollbar-outer>.scroll-element,.scrollbar-outer>.scroll-element div{border:none;margin:0;padding:0;position:absolute;z-index:10}.scrollbar-outer>.scroll-element{background-color:#fff}.scrollbar-outer>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-outer>.scroll-element.scroll-x{bottom:0;height:12px;left:0;width:100%}.scrollbar-outer>.scroll-element.scroll-y{height:100%;right:0;top:0;width:12px}.scrollbar-outer>.scroll-element.scroll-x .scroll-element_outer{height:8px;top:2px}.scrollbar-outer>.scroll-element.scroll-y .scroll-element_outer{left:2px;width:8px}.scrollbar-outer>.scroll-element .scroll-element_outer{overflow:hidden}.scrollbar-outer>.scroll-element .scroll-element_track{background-color:#eee}.scrollbar-outer>.scroll-element .scroll-bar,.scrollbar-outer>.scroll-element .scroll-element_outer,.scrollbar-outer>.scroll-element .scroll-element_track{border-radius:8px}.scrollbar-outer>.scroll-element .scroll-bar{background-color:#d9d9d9}.scrollbar-outer>.scroll-element .scroll-bar:hover{background-color:#c2c2c2}.scrollbar-outer>.scroll-element.scroll-draggable .scroll-bar{background-color:#919191}.scrollbar-outer>.scroll-content.scroll-scrolly_visible{left:-12px;margin-left:12px}.scrollbar-outer>.scroll-content.scroll-scrollx_visible{top:-12px;margin-top:12px}.scrollbar-outer>.scroll-element.scroll-x .scroll-bar{min-width:10px}.scrollbar-outer>.scroll-element.scroll-y .scroll-bar{min-height:10px}.scrollbar-outer>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_track{left:-14px}.scrollbar-outer>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_track{top:-14px}.scrollbar-outer>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-14px}.scrollbar-outer>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-14px}.scrollbar-macosx>.scroll-element,.scrollbar-macosx>.scroll-element div{background:0 0;border:none;margin:0;padding:0;position:absolute;z-index:10}.scrollbar-macosx>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-macosx>.scroll-element .scroll-element_track{display:none}.scrollbar-macosx>.scroll-element .scroll-bar{background-color:#6C6E71;display:block;-ms-filter:\"progid:DXImageTransform.Microsoft.Alpha(Opacity=0)\";filter:alpha(opacity=0);opacity:0;border-radius:7px;transition:opacity .2s linear}.scrollbar-macosx:hover>.scroll-element .scroll-bar,.scrollbar-macosx>.scroll-element.scroll-draggable .scroll-bar{-ms-filter:\"progid:DXImageTransform.Microsoft.Alpha(Opacity=70)\";filter:alpha(opacity=70);opacity:.7}.scrollbar-macosx>.scroll-element.scroll-x{bottom:0;height:0;left:0;min-width:100%;overflow:visible;width:100%}.scrollbar-macosx>.scroll-element.scroll-y{height:100%;min-height:100%;right:0;top:0;width:0}.scrollbar-macosx>.scroll-element.scroll-x .scroll-bar{height:7px;min-width:10px;top:-9px}.scrollbar-macosx>.scroll-element.scroll-y .scroll-bar{left:-9px;min-height:10px;width:7px}.scrollbar-macosx>.scroll-element.scroll-x .scroll-element_outer{left:2px}.scrollbar-macosx>.scroll-element.scroll-x .scroll-element_size{left:-4px}.scrollbar-macosx>.scroll-element.scroll-y .scroll-element_outer{top:2px}.scrollbar-macosx>.scroll-element.scroll-y .scroll-element_size{top:-4px}.scrollbar-macosx>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-11px}.scrollbar-macosx>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-11px}.scrollbar-light>.scroll-element,.scrollbar-light>.scroll-element div{border:none;margin:0;overflow:hidden;padding:0;position:absolute;z-index:10}.scrollbar-light>.scroll-element{background-color:#fff}.scrollbar-light>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-light>.scroll-element .scroll-element_outer{border-radius:10px}.scrollbar-light>.scroll-element .scroll-element_size{background:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiA/Pgo8c3ZnIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyIgd2lkdGg9IjEwMCUiIGhlaWdodD0iMTAwJSIgdmlld0JveD0iMCAwIDEgMSIgcHJlc2VydmVBc3BlY3RSYXRpbz0ibm9uZSI+CiAgPGxpbmVhckdyYWRpZW50IGlkPSJncmFkLXVjZ2ctZ2VuZXJhdGVkIiBncmFkaWVudFVuaXRzPSJ1c2VyU3BhY2VPblVzZSIgeDE9IjAlIiB5MT0iMCUiIHgyPSIxMDAlIiB5Mj0iMCUiPgogICAgPHN0b3Agb2Zmc2V0PSIwJSIgc3RvcC1jb2xvcj0iI2RiZGJkYiIgc3RvcC1vcGFjaXR5PSIxIi8+CiAgICA8c3RvcCBvZmZzZXQ9IjEwMCUiIHN0b3AtY29sb3I9IiNlOGU4ZTgiIHN0b3Atb3BhY2l0eT0iMSIvPgogIDwvbGluZWFyR3JhZGllbnQ+CiAgPHJlY3QgeD0iMCIgeT0iMCIgd2lkdGg9IjEiIGhlaWdodD0iMSIgZmlsbD0idXJsKCNncmFkLXVjZ2ctZ2VuZXJhdGVkKSIgLz4KPC9zdmc+);background:linear-gradient(to right,#dbdbdb 0,#e8e8e8 100%);border-radius:10px}.scrollbar-light>.scroll-element.scroll-x{bottom:0;height:17px;left:0;min-width:100%;width:100%}.scrollbar-light>.scroll-element.scroll-y{height:100%;min-height:100%;right:0;top:0;width:17px}.scrollbar-light>.scroll-element .scroll-bar{background:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiA/Pgo8c3ZnIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyIgd2lkdGg9IjEwMCUiIGhlaWdodD0iMTAwJSIgdmlld0JveD0iMCAwIDEgMSIgcHJlc2VydmVBc3BlY3RSYXRpbz0ibm9uZSI+CiAgPGxpbmVhckdyYWRpZW50IGlkPSJncmFkLXVjZ2ctZ2VuZXJhdGVkIiBncmFkaWVudFVuaXRzPSJ1c2VyU3BhY2VPblVzZSIgeDE9IjAlIiB5MT0iMCUiIHgyPSIxMDAlIiB5Mj0iMCUiPgogICAgPHN0b3Agb2Zmc2V0PSIwJSIgc3RvcC1jb2xvcj0iI2ZlZmVmZSIgc3RvcC1vcGFjaXR5PSIxIi8+CiAgICA8c3RvcCBvZmZzZXQ9IjEwMCUiIHN0b3AtY29sb3I9IiNmNWY1ZjUiIHN0b3Atb3BhY2l0eT0iMSIvPgogIDwvbGluZWFyR3JhZGllbnQ+CiAgPHJlY3QgeD0iMCIgeT0iMCIgd2lkdGg9IjEiIGhlaWdodD0iMSIgZmlsbD0idXJsKCNncmFkLXVjZ2ctZ2VuZXJhdGVkKSIgLz4KPC9zdmc+);background:linear-gradient(to right,#fefefe 0,#f5f5f5 100%);border:1px solid #dbdbdb;border-radius:10px}.scrollbar-light>.scroll-content.scroll-scrolly_visible{left:-17px;margin-left:17px}.scrollbar-light>.scroll-content.scroll-scrollx_visible{top:-17px;margin-top:17px}.scrollbar-light>.scroll-element.scroll-x .scroll-bar{height:10px;min-width:10px;top:0}.scrollbar-light>.scroll-element.scroll-y .scroll-bar{left:0;min-height:10px;width:10px}.scrollbar-light>.scroll-element.scroll-x .scroll-element_outer{height:12px;left:2px;top:2px}.scrollbar-light>.scroll-element.scroll-x .scroll-element_size{left:-4px}.scrollbar-light>.scroll-element.scroll-y .scroll-element_outer{left:2px;top:2px;width:12px}.scrollbar-light>.scroll-element.scroll-y .scroll-element_size{top:-4px}.scrollbar-light>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-19px}.scrollbar-light>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-19px}.scrollbar-light>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_track{left:-19px}.scrollbar-light>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_track{top:-19px}.scrollbar-rail>.scroll-element,.scrollbar-rail>.scroll-element div{border:none;margin:0;overflow:hidden;padding:0;position:absolute;z-index:10}.scrollbar-rail>.scroll-element{background-color:#fff}.scrollbar-rail>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-rail>.scroll-element .scroll-element_size{background-color:#999;background-color:rgba(0,0,0,.3)}.scrollbar-rail>.scroll-element .scroll-element_outer:hover .scroll-element_size{background-color:#666;background-color:rgba(0,0,0,.5)}.scrollbar-rail>.scroll-element.scroll-x{bottom:0;height:12px;left:0;min-width:100%;padding:3px 0 2px;width:100%}.scrollbar-rail>.scroll-element.scroll-y{height:100%;min-height:100%;padding:0 2px 0 3px;right:0;top:0;width:12px}.scrollbar-rail>.scroll-element .scroll-bar{background-color:#d0b9a0;border-radius:2px;box-shadow:1px 1px 3px rgba(0,0,0,.5)}.scrollbar-rail>.scroll-element .scroll-element_outer:hover .scroll-bar{box-shadow:1px 1px 3px rgba(0,0,0,.6)}.scrollbar-rail>.scroll-content.scroll-scrolly_visible{left:-17px;margin-left:17px}.scrollbar-rail>.scroll-content.scroll-scrollx_visible{margin-top:17px;top:-17px}.scrollbar-rail>.scroll-element.scroll-x .scroll-bar{height:10px;min-width:10px;top:1px}.scrollbar-rail>.scroll-element.scroll-y .scroll-bar{left:1px;min-height:10px;width:10px}.scrollbar-rail>.scroll-element.scroll-x .scroll-element_outer{height:15px;left:5px}.scrollbar-rail>.scroll-element.scroll-x .scroll-element_size{height:2px;left:-10px;top:5px}.scrollbar-rail>.scroll-element.scroll-y .scroll-element_outer{top:5px;width:15px}.scrollbar-rail>.scroll-element.scroll-y .scroll-element_size{left:5px;top:-10px;width:2px}.scrollbar-rail>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-25px}.scrollbar-rail>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-25px}.scrollbar-rail>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_track{left:-25px}.scrollbar-rail>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_track{top:-25px}.scrollbar-dynamic>.scroll-element,.scrollbar-dynamic>.scroll-element div{background:0 0;border:none;margin:0;padding:0;position:absolute;z-index:10}.scrollbar-dynamic>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-dynamic>.scroll-element.scroll-x{bottom:2px;height:7px;left:0;min-width:100%;width:100%}.scrollbar-dynamic>.scroll-element.scroll-y{height:100%;min-height:100%;right:2px;top:0;width:7px}.scrollbar-dynamic>.scroll-element .scroll-element_outer{opacity:.3;border-radius:12px}.scrollbar-dynamic>.scroll-element .scroll-element_size{background-color:#ccc;opacity:0;border-radius:12px;transition:opacity .2s}.scrollbar-dynamic>.scroll-element .scroll-bar{background-color:#6c6e71;border-radius:7px}.scrollbar-dynamic>.scroll-element.scroll-x .scroll-bar{bottom:0;height:7px;min-width:24px;top:auto}.scrollbar-dynamic>.scroll-element.scroll-y .scroll-bar{left:auto;min-height:24px;right:0;width:7px}.scrollbar-dynamic>.scroll-element.scroll-x .scroll-element_outer{bottom:0;top:auto;left:2px;transition:height .2s}.scrollbar-dynamic>.scroll-element.scroll-y .scroll-element_outer{left:auto;right:0;top:2px;transition:width .2s}.scrollbar-dynamic>.scroll-element.scroll-x .scroll-element_size{left:-4px}.scrollbar-dynamic>.scroll-element.scroll-y .scroll-element_size{top:-4px}.scrollbar-dynamic>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-11px}.scrollbar-dynamic>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-11px}.scrollbar-dynamic>.scroll-element.scroll-draggable .scroll-element_outer,.scrollbar-dynamic>.scroll-element:hover .scroll-element_outer{overflow:hidden;-ms-filter:\"progid:DXImageTransform.Microsoft.Alpha(Opacity=70)\";filter:alpha(opacity=70);opacity:.7}.scrollbar-dynamic>.scroll-element.scroll-draggable .scroll-element_outer .scroll-element_size,.scrollbar-dynamic>.scroll-element:hover .scroll-element_outer .scroll-element_size{opacity:1}.scrollbar-dynamic>.scroll-element.scroll-draggable .scroll-element_outer .scroll-bar,.scrollbar-dynamic>.scroll-element:hover .scroll-element_outer .scroll-bar{height:100%;width:100%;border-radius:12px}.scrollbar-dynamic>.scroll-element.scroll-x.scroll-draggable .scroll-element_outer,.scrollbar-dynamic>.scroll-element.scroll-x:hover .scroll-element_outer{height:20px;min-height:7px}.scrollbar-dynamic>.scroll-element.scroll-y.scroll-draggable .scroll-element_outer,.scrollbar-dynamic>.scroll-element.scroll-y:hover .scroll-element_outer{min-width:7px;width:20px}.scrollbar-chrome>.scroll-element,.scrollbar-chrome>.scroll-element div{border:none;margin:0;overflow:hidden;padding:0;position:absolute;z-index:10}.scrollbar-chrome>.scroll-element{background-color:#fff}.scrollbar-chrome>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-chrome>.scroll-element .scroll-element_track{background:#f1f1f1;border:1px solid #dbdbdb}.scrollbar-chrome>.scroll-element.scroll-x{bottom:0;height:16px;left:0;min-width:100%;width:100%}.scrollbar-chrome>.scroll-element.scroll-y{height:100%;min-height:100%;right:0;top:0;width:16px}.scrollbar-chrome>.scroll-element .scroll-bar{background-color:#d9d9d9;border:1px solid #bdbdbd;cursor:default;border-radius:2px}.scrollbar-chrome>.scroll-element .scroll-bar:hover{background-color:#c2c2c2;border-color:#a9a9a9}.scrollbar-chrome>.scroll-element.scroll-draggable .scroll-bar{background-color:#919191;border-color:#7e7e7e}.scrollbar-chrome>.scroll-content.scroll-scrolly_visible{left:-16px;margin-left:16px}.scrollbar-chrome>.scroll-content.scroll-scrollx_visible{top:-16px;margin-top:16px}.scrollbar-chrome>.scroll-element.scroll-x .scroll-bar{height:8px;min-width:10px;top:3px}.scrollbar-chrome>.scroll-element.scroll-y .scroll-bar{left:3px;min-height:10px;width:8px}.scrollbar-chrome>.scroll-element.scroll-x .scroll-element_outer{border-left:1px solid #dbdbdb}.scrollbar-chrome>.scroll-element.scroll-x .scroll-element_track{height:14px;left:-3px}.scrollbar-chrome>.scroll-element.scroll-x .scroll-element_size{height:14px;left:-4px}.scrollbar-chrome>.scroll-element.scroll-y .scroll-element_outer{border-top:1px solid #dbdbdb}.scrollbar-chrome>.scroll-element.scroll-y .scroll-element_track{top:-3px;width:14px}.scrollbar-chrome>.scroll-element.scroll-y .scroll-element_size{top:-4px;width:14px}.scrollbar-chrome>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-19px}.scrollbar-chrome>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-19px}.scrollbar-chrome>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_track{left:-19px}.scrollbar-chrome>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_track{top:-19px}\n" + "/**\n" + " * Minified style.\n" + " * Original filename: \\src\\styles\\style.css\n" + " */\n" + "@-webkit-keyframes spin{100%{-webkit-transform:rotate(360deg);transform:rotate(360deg)}}@keyframes spin{100%{-webkit-transform:rotate(360deg);transform:rotate(360deg)}}#emote-menu-button{background-image:url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABIAAAAQCAYAAAAbBi9cAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsMAAA7DAcdvqGQAAAKUSURBVDhPfZTNi1JRGMZvMIsWUZts5SIXFYK0CME/IGghxVC7WUoU1NBixI+mRSD4MQzmxziKO3XUBhRmUGZKdBG40XEGU6d0GFGZcT4qxW1hi7fzvNwZqKwDD5z7vs/vueeee+6VMJxO5wUhhdvtfuHz+T4tLS2NhegfGsMDLxiwHIIhLi57PJ75VCr1Y39/n4bDIY1Go4lCDx54wYCVYzjoVjQa/dxutyfCkwSvYJpgOSQf708tuBa1yWRy/L+V/Cl4wYBFhhTxfLhum/esiiJ1u12KRCJksVhofX2dTk5OzkHMUUMPHnjB2F55VpEhPde/Lbx8FqBEIkHpdJoMBgNptVrS6XRUqVTOg7a3t2lmZob0ej2p1Wr2ggGLDOnJ3QSZH4coHo/TysoKhygUCtJoNFQsFmkwGLAwR7hSqSSVSsVeMGCRIT29F6fXJi8Xy+Uymc1mmp6eJofDQfV6nU5PT1mY2+127uHxSqUSh4FFhhQLvrvtcrm+YpkHBwdUrVZpa2uLarUadTodOjw8ZGGOGnrwwAsGLDLw1i4uLrzRYeOOj49pb2+Pdnd3qdVq8StGAIQ5ao1Ggz3wggGLDD4C4izcEcWfR0dHbMrlcrSxscGbjVAIK8lms7S5ucmB/X6fXz9YDsEQFzdjsVit2Wzyqc1kMrwfVquVjEYjzc3NkclkIpvNRmtra+yBVzAfBXtDjuGgS8FgcFbc8QvuhjNSKBQoFAqR6LFEn/L5PPfggXd5eXkWrBzDQdC1QCBgFoeut7Ozw/tyBp2FQzhPwtOFFwzY34Yo4A9wRXzdD8LhcE48wncE9no9Fuaoid574bkPLxgZ/3uI5pTQVfFlP/L7/Wmhb7JSXq/3IXrwyHZ5SNIvGCnqyh+J7+gAAAAASUVORK5CYII=)!important;background-position:50%;background-repeat:no-repeat;cursor:pointer;height:30px;width:30px}#emote-menu-button:focus{box-shadow:none}#emote-menu-button.active{box-shadow:0 0 6px 0 #7d5bbe,inset 0 0 0 1px rgba(100,65,164,.5)}.emote-menu{padding:5px;z-index:1000;display:none;background-color:#202020;position:absolute}.emote-menu a{color:#fff}.emote-menu a:hover{cursor:pointer;text-decoration:underline;color:#ccc}.emote-menu .emotes-starred{height:38px}.emote-menu .draggable{background-image:repeating-linear-gradient(45deg,transparent,transparent 5px,rgba(255,255,255,.05) 5px,rgba(255,255,255,.05) 10px);cursor:move;height:7px;margin-bottom:3px}.emote-menu .draggable:hover{background-image:repeating-linear-gradient(45deg,transparent,transparent 5px,rgba(255,255,255,.1) 5px,rgba(255,255,255,.1) 10px)}.emote-menu .header-info{border-top:1px solid #000;box-shadow:0 1px 0 rgba(255,255,255,.05) inset;background-image:linear-gradient(to top,transparent,rgba(0,0,0,.5));padding:2px;color:#ddd;text-align:center;position:relative}.emote-menu .header-info img{margin-right:8px}.emote-menu .emote{display:inline-block;padding:2px;margin:1px;cursor:pointer;border-radius:5px;text-align:center;position:relative;width:30px;height:30px;transition:all .25s ease;border:1px solid transparent}.emote-menu.editing .emote{cursor:auto}.emote-menu .emote img{max-width:100%;max-height:100%;margin:auto;position:absolute;top:0;bottom:0;left:0;right:0}.emote-menu .single-row .emote-container{overflow:hidden;height:37px}.emote-menu .single-row .emote{display:inline-block;margin-bottom:100px}.emote-menu .emote:hover{background-color:rgba(255,255,255,.1)}.emote-menu .pull-left{float:left}.emote-menu .pull-right{float:right}.emote-menu .footer{text-align:center;border-top:1px solid #000;box-shadow:0 1px 0 rgba(255,255,255,.05) inset;padding:5px 0 2px;margin-top:5px;height:18px}.emote-menu .footer .pull-left{margin-right:5px}.emote-menu .footer .pull-right{margin-left:5px}.emote-menu .icon{height:16px;width:16px;opacity:.5;background-size:contain!important}.emote-menu .icon:hover{opacity:1}.emote-menu .icon-home{background:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iNjQiDQogICBoZWlnaHQ9IjY0Ig0KICAgdmlld0JveD0iMCAwIDY0IDY0Ig0KICAgaWQ9IkNhcGFfMSINCiAgIHhtbDpzcGFjZT0icHJlc2VydmUiPjxtZXRhZGF0YQ0KICAgaWQ9Im1ldGFkYXRhMzAwMSI+PHJkZjpSREY+PGNjOldvcmsNCiAgICAgICByZGY6YWJvdXQ9IiI+PGRjOmZvcm1hdD5pbWFnZS9zdmcreG1sPC9kYzpmb3JtYXQ+PGRjOnR5cGUNCiAgICAgICAgIHJkZjpyZXNvdXJjZT0iaHR0cDovL3B1cmwub3JnL2RjL2RjbWl0eXBlL1N0aWxsSW1hZ2UiIC8+PGRjOnRpdGxlPjwvZGM6dGl0bGU+PC9jYzpXb3JrPjwvcmRmOlJERj48L21ldGFkYXRhPjxkZWZzDQogICBpZD0iZGVmczI5OTkiIC8+DQo8cGF0aA0KICAgZD0ibSA1Ny4wNjIsMzEuMzk4IGMgMC45MzIsLTEuMDI1IDAuODQyLC0yLjU5NiAtMC4yMDEsLTMuNTA4IEwgMzMuODg0LDcuNzg1IEMgMzIuODQxLDYuODczIDMxLjE2OSw2Ljg5MiAzMC4xNDgsNy44MjggTCA3LjA5MywyOC45NjIgYyAtMS4wMjEsMC45MzYgLTEuMDcxLDIuNTA1IC0wLjExMSwzLjUwMyBsIDAuNTc4LDAuNjAyIGMgMC45NTksMC45OTggMi41MDksMS4xMTcgMy40NiwwLjI2NSBsIDEuNzIzLC0xLjU0MyB2IDIyLjU5IGMgMCwxLjM4NiAxLjEyMywyLjUwOCAyLjUwOCwyLjUwOCBoIDguOTg3IGMgMS4zODUsMCAyLjUwOCwtMS4xMjIgMi41MDgsLTIuNTA4IFYgMzguNTc1IGggMTEuNDYzIHYgMTUuODA0IGMgLTAuMDIsMS4zODUgMC45NzEsMi41MDcgMi4zNTYsMi41MDcgaCA5LjUyNCBjIDEuMzg1LDAgMi41MDgsLTEuMTIyIDIuNTA4LC0yLjUwOCBWIDMyLjEwNyBjIDAsMCAwLjQ3NiwwLjQxNyAxLjA2MywwLjkzMyAwLjU4NiwwLjUxNSAxLjgxNywwLjEwMiAyLjc0OSwtMC45MjQgbCAwLjY1MywtMC43MTggeiINCiAgIGlkPSJwYXRoMjk5NSINCiAgIHN0eWxlPSJmaWxsOiNmZmZmZmY7ZmlsbC1vcGFjaXR5OjEiIC8+DQo8L3N2Zz4=) 50% no-repeat}.emote-menu .icon-gear{background:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iMjEuNTkiDQogICBoZWlnaHQ9IjIxLjEzNjk5OSINCiAgIHZpZXdCb3g9IjAgMCAyMS41OSAyMS4xMzciDQogICBpZD0iQ2FwYV8xIg0KICAgeG1sOnNwYWNlPSJwcmVzZXJ2ZSI+PG1ldGFkYXRhDQogICBpZD0ibWV0YWRhdGEzOSI+PHJkZjpSREY+PGNjOldvcmsNCiAgICAgICByZGY6YWJvdXQ9IiI+PGRjOmZvcm1hdD5pbWFnZS9zdmcreG1sPC9kYzpmb3JtYXQ+PGRjOnR5cGUNCiAgICAgICAgIHJkZjpyZXNvdXJjZT0iaHR0cDovL3B1cmwub3JnL2RjL2RjbWl0eXBlL1N0aWxsSW1hZ2UiIC8+PGRjOnRpdGxlPjwvZGM6dGl0bGU+PC9jYzpXb3JrPjwvcmRmOlJERj48L21ldGFkYXRhPjxkZWZzDQogICBpZD0iZGVmczM3IiAvPg0KPHBhdGgNCiAgIGQ9Ik0gMTguNjIyLDguMTQ1IDE4LjA3Nyw2Ljg1IGMgMCwwIDEuMjY4LC0yLjg2MSAxLjE1NiwtMi45NzEgTCAxNy41NTQsMi4yNCBDIDE3LjQzOCwyLjEyNyAxNC41NzYsMy40MzMgMTQuNTc2LDMuNDMzIEwgMTMuMjU2LDIuOSBDIDEzLjI1NiwyLjkgMTIuMDksMCAxMS45MywwIEggOS41NjEgQyA5LjM5NiwwIDguMzE3LDIuOTA2IDguMzE3LDIuOTA2IEwgNi45OTksMy40NDEgYyAwLDAgLTIuOTIyLC0xLjI0MiAtMy4wMzQsLTEuMTMxIEwgMi4yODksMy45NTEgQyAyLjE3Myw0LjA2NCAzLjUwNyw2Ljg2NyAzLjUwNyw2Ljg2NyBMIDIuOTYyLDguMTYgQyAyLjk2Miw4LjE2IDAsOS4zMDEgMCw5LjQ1NSB2IDIuMzIyIGMgMCwwLjE2MiAyLjk2OSwxLjIxOSAyLjk2OSwxLjIxOSBsIDAuNTQ1LDEuMjkxIGMgMCwwIC0xLjI2OCwyLjg1OSAtMS4xNTcsMi45NjkgbCAxLjY3OCwxLjY0MyBjIDAuMTE0LDAuMTExIDIuOTc3LC0xLjE5NSAyLjk3NywtMS4xOTUgbCAxLjMyMSwwLjUzNSBjIDAsMCAxLjE2NiwyLjg5OCAxLjMyNywyLjg5OCBoIDIuMzY5IGMgMC4xNjQsMCAxLjI0NCwtMi45MDYgMS4yNDQsLTIuOTA2IGwgMS4zMjIsLTAuNTM1IGMgMCwwIDIuOTE2LDEuMjQyIDMuMDI5LDEuMTMzIGwgMS42NzgsLTEuNjQxIGMgMC4xMTcsLTAuMTE1IC0xLjIyLC0yLjkxNiAtMS4yMiwtMi45MTYgbCAwLjU0NCwtMS4yOTMgYyAwLDAgMi45NjMsLTEuMTQzIDIuOTYzLC0xLjI5OSBWIDkuMzYgQyAyMS41OSw5LjE5OSAxOC42MjIsOC4xNDUgMTguNjIyLDguMTQ1IHogbSAtNC4zNjYsMi40MjMgYyAwLDEuODY3IC0xLjU1MywzLjM4NyAtMy40NjEsMy4zODcgLTEuOTA2LDAgLTMuNDYxLC0xLjUyIC0zLjQ2MSwtMy4zODcgMCwtMS44NjcgMS41NTUsLTMuMzg1IDMuNDYxLC0zLjM4NSAxLjkwOSwwLjAwMSAzLjQ2MSwxLjUxOCAzLjQ2MSwzLjM4NSB6Ig0KICAgaWQ9InBhdGgzIg0KICAgc3R5bGU9ImZpbGw6I0ZGRkZGRiIgLz4NCjxnDQogICBpZD0iZzUiPg0KPC9nPg0KPGcNCiAgIGlkPSJnNyI+DQo8L2c+DQo8Zw0KICAgaWQ9Imc5Ij4NCjwvZz4NCjxnDQogICBpZD0iZzExIj4NCjwvZz4NCjxnDQogICBpZD0iZzEzIj4NCjwvZz4NCjxnDQogICBpZD0iZzE1Ij4NCjwvZz4NCjxnDQogICBpZD0iZzE3Ij4NCjwvZz4NCjxnDQogICBpZD0iZzE5Ij4NCjwvZz4NCjxnDQogICBpZD0iZzIxIj4NCjwvZz4NCjxnDQogICBpZD0iZzIzIj4NCjwvZz4NCjxnDQogICBpZD0iZzI1Ij4NCjwvZz4NCjxnDQogICBpZD0iZzI3Ij4NCjwvZz4NCjxnDQogICBpZD0iZzI5Ij4NCjwvZz4NCjxnDQogICBpZD0iZzMxIj4NCjwvZz4NCjxnDQogICBpZD0iZzMzIj4NCjwvZz4NCjwvc3ZnPg0K) 50% no-repeat}.emote-menu.editing .icon-gear{-webkit-animation:spin 4s linear infinite;animation:spin 4s linear infinite}.emote-menu .icon-resize-handle{background:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iMTYiDQogICBoZWlnaHQ9IjE2Ig0KICAgdmlld0JveD0iMCAwIDE2IDE2Ig0KICAgaWQ9IkNhcGFfMSINCiAgIHhtbDpzcGFjZT0icHJlc2VydmUiPjxtZXRhZGF0YQ0KICAgaWQ9Im1ldGFkYXRhNDM1NyI+PHJkZjpSREY+PGNjOldvcmsNCiAgICAgICByZGY6YWJvdXQ9IiI+PGRjOmZvcm1hdD5pbWFnZS9zdmcreG1sPC9kYzpmb3JtYXQ+PGRjOnR5cGUNCiAgICAgICAgIHJkZjpyZXNvdXJjZT0iaHR0cDovL3B1cmwub3JnL2RjL2RjbWl0eXBlL1N0aWxsSW1hZ2UiIC8+PGRjOnRpdGxlPjwvZGM6dGl0bGU+PC9jYzpXb3JrPjwvcmRmOlJERj48L21ldGFkYXRhPjxkZWZzDQogICBpZD0iZGVmczQzNTUiIC8+DQo8cGF0aA0KICAgZD0iTSAxMy41LDggQyAxMy4yMjUsOCAxMyw4LjIyNCAxMyw4LjUgdiAzLjc5MyBMIDMuNzA3LDMgSCA3LjUgQyA3Ljc3NiwzIDgsMi43NzYgOCwyLjUgOCwyLjIyNCA3Ljc3NiwyIDcuNSwyIGggLTUgTCAyLjMwOSwyLjAzOSAyLjE1LDIuMTQ0IDIuMTQ2LDIuMTQ2IDIuMTQzLDIuMTUyIDIuMDM5LDIuMzA5IDIsMi41IHYgNSBDIDIsNy43NzYgMi4yMjQsOCAyLjUsOCAyLjc3Niw4IDMsNy43NzYgMyw3LjUgViAzLjcwNyBMIDEyLjI5MywxMyBIIDguNSBDIDguMjI0LDEzIDgsMTMuMjI1IDgsMTMuNSA4LDEzLjc3NSA4LjIyNCwxNCA4LjUsMTQgaCA1IGwgMC4xOTEsLTAuMDM5IGMgMC4xMjEsLTAuMDUxIDAuMjIsLTAuMTQ4IDAuMjcsLTAuMjcgTCAxNCwxMy41MDIgViA4LjUgQyAxNCw4LjIyNCAxMy43NzUsOCAxMy41LDggeiINCiAgIGlkPSJwYXRoNDM1MSINCiAgIHN0eWxlPSJmaWxsOiNmZmZmZmY7ZmlsbC1vcGFjaXR5OjEiIC8+DQo8L3N2Zz4=) 50% no-repeat;cursor:nwse-resize!important}.emote-menu .icon-pin{background:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iMTYiDQogICBoZWlnaHQ9IjE2Ig0KICAgaWQ9InN2ZzMwMDUiPg0KICA8bWV0YWRhdGENCiAgICAgaWQ9Im1ldGFkYXRhMzAyMyI+DQogICAgPHJkZjpSREY+DQogICAgICA8Y2M6V29yaw0KICAgICAgICAgcmRmOmFib3V0PSIiPg0KICAgICAgICA8ZGM6Zm9ybWF0PmltYWdlL3N2Zyt4bWw8L2RjOmZvcm1hdD4NCiAgICAgICAgPGRjOnR5cGUNCiAgICAgICAgICAgcmRmOnJlc291cmNlPSJodHRwOi8vcHVybC5vcmcvZGMvZGNtaXR5cGUvU3RpbGxJbWFnZSIgLz4NCiAgICAgICAgPGRjOnRpdGxlPjwvZGM6dGl0bGU+DQogICAgICA8L2NjOldvcms+DQogICAgPC9yZGY6UkRGPg0KICA8L21ldGFkYXRhPg0KICA8ZGVmcw0KICAgICBpZD0iZGVmczMwMjEiIC8+DQogIDxnDQogICAgIHRyYW5zZm9ybT0ibWF0cml4KDAuNzkzMDc4MiwwLDAsMC43OTMwNzgyLC0yLjE3MDk4NSwtODE0LjY5Mjk5KSINCiAgICAgaWQ9ImczMDA3Ij4NCiAgICA8Zw0KICAgICAgIHRyYW5zZm9ybT0ibWF0cml4KDAuNzA3MTEsMC43MDcxMSwtMC43MDcxMSwwLjcwNzExLDczNy43MDc1NSwyOTUuNDg4MDgpIg0KICAgICAgIGlkPSJnMzAwOSI+DQogICAgICA8Zw0KICAgICAgICAgaWQ9ImczNzU1Ij4NCiAgICAgICAgPHBhdGgNCiAgICAgICAgICAgZD0iTSA5Ljc4MTI1LDAgQyA5LjQ3NDA1NjIsMC42ODkxMTIgOS41MjA2OCwxLjUyMzA4NTMgOS4zMTI1LDIuMTg3NSBMIDQuOTM3NSw2LjU5Mzc1IEMgMy45NTg5NjA4LDYuNDI5NDgzIDIuOTQ3NzU0OCw2LjUzMjc4OTkgMiw2LjgxMjUgTCA1LjAzMTI1LDkuODQzNzUgMC41NjI1LDE0LjMxMjUgMCwxNiBDIDAuNTY5Mjk2MjgsMTUuNzk1NjI2IDEuMTY3NzM3OCwxNS42NDAyMzcgMS43MTg3NSwxNS40MDYyNSBMIDYuMTU2MjUsMTAuOTY4NzUgOS4xODc1LDE0IGMgMC4yNzk2ODIzLC0wLjk0Nzc4MyAwLjM4MzE1MjgsLTEuOTU4OTM3IDAuMjE4NzUsLTIuOTM3NSAxLjUwMDAxMSwtMS40ODk1Nzk4IDMuMDAwMDAxLC0yLjk3OTE1OSA0LjUsLTQuNDY4NzUgMC42MDExMDIsLTAuMDMxMzYxIDEuODIyMTM4LC0wLjA5NjEzNyAyLC0wLjQ2ODc1IEMgMTMuODc5ODkyLDQuMDY5NDgwMyAxMS44NDI4NjUsMi4wMjAyMjgyIDkuNzgxMjUsMCB6Ig0KICAgICAgICAgICB0cmFuc2Zvcm09Im1hdHJpeCgwLjg5MTU5Mzc0LC0wLjg5MTU5Mzc0LDAuODkxNTkzNzQsMC44OTE1OTM3NCwtMi4yNjU1LDEwMzcuMTM0NSkiDQogICAgICAgICAgIGlkPSJwYXRoMzAxMSINCiAgICAgICAgICAgc3R5bGU9ImZpbGw6I2ZmZmZmZjtmaWxsLW9wYWNpdHk6MSIgLz4NCiAgICAgIDwvZz4NCiAgICA8L2c+DQogIDwvZz4NCjwvc3ZnPg0K) 50% no-repeat;transition:all .25s ease}.emote-menu .icon-pin:hover,.emote-menu.pinned .icon-pin{-webkit-transform:rotate(-45deg);transform:rotate(-45deg);opacity:1}.emote-menu .edit-tool{background-position:50%;background-repeat:no-repeat;background-size:14px;border-radius:4px;border:1px solid #000;cursor:pointer;display:none;height:14px;opacity:.25;position:absolute;transition:all .25s ease;width:14px;z-index:1}.emote-menu .edit-tool:hover,.emote-menu .emote:hover .edit-tool{opacity:1}.emote-menu .edit-visibility{background-color:#00c800;background-image:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iMTAwIg0KICAgaGVpZ2h0PSIxMDAiDQogICB2aWV3Qm94PSIwIDAgMTAwIDEwMCINCiAgIGlkPSJMYXllcl8xIg0KICAgeG1sOnNwYWNlPSJwcmVzZXJ2ZSI+PG1ldGFkYXRhDQogICBpZD0ibWV0YWRhdGE5Ij48cmRmOlJERj48Y2M6V29yaw0KICAgICAgIHJkZjphYm91dD0iIj48ZGM6Zm9ybWF0PmltYWdlL3N2Zyt4bWw8L2RjOmZvcm1hdD48ZGM6dHlwZQ0KICAgICAgICAgcmRmOnJlc291cmNlPSJodHRwOi8vcHVybC5vcmcvZGMvZGNtaXR5cGUvU3RpbGxJbWFnZSIgLz48ZGM6dGl0bGU+PC9kYzp0aXRsZT48L2NjOldvcms+PC9yZGY6UkRGPjwvbWV0YWRhdGE+PGRlZnMNCiAgIGlkPSJkZWZzNyIgLz4NCjxwYXRoDQogICBkPSJNIDk3Ljk2NCw0Ni41NDggQyA5Ny4wOTgsNDUuNTI4IDc2LjQyNywyMS42MDMgNTAsMjEuNjAzIGMgLTI2LjQyNywwIC00Ny4wOTgsMjMuOTI1IC00Ny45NjUsMjQuOTQ2IC0xLjcwMSwyIC0xLjcwMSw0LjkwMiAxMGUtNCw2LjkwMyAwLjg2NiwxLjAyIDIxLjUzNywyNC45NDUgNDcuOTY0LDI0Ljk0NSAyNi40MjcsMCA0Ny4wOTgsLTIzLjkyNiA0Ny45NjUsLTI0Ljk0NiAxLjcwMSwtMiAxLjcwMSwtNC45MDIgLTAuMDAxLC02LjkwMyB6IE0gNTguMDczLDM1Ljk3NSBjIDEuNzc3LC0wLjk3IDQuMjU1LDAuMTQzIDUuNTM0LDIuNDg1IDEuMjc5LDIuMzQzIDAuODc1LDUuMDI5IC0wLjkwMiw1Ljk5OSAtMS43NzcsMC45NzEgLTQuMjU1LC0wLjE0MyAtNS41MzUsLTIuNDg1IC0xLjI3OSwtMi4zNDMgLTAuODc1LC01LjAyOSAwLjkwMywtNS45OTkgeiBNIDUwLDY5LjcyOSBDIDMxLjU0LDY5LjcyOSAxNi4wMDUsNTUuNTUzIDEwLjYyOCw1MCAxNC4yNTksNDYuMjQ5IDIyLjUyNiwzOC41NzEgMzMuMTk1LDMzLjk3OSAzMS4xMTQsMzcuMTQ1IDI5Ljg5NCw0MC45MjggMjkuODk0LDQ1IGMgMCwxMS4xMDQgOS4wMDEsMjAuMTA1IDIwLjEwNSwyMC4xMDUgMTEuMTA0LDAgMjAuMTA2LC05LjAwMSAyMC4xMDYsLTIwLjEwNSAwLC00LjA3MiAtMS4yMTksLTcuODU1IC0zLjMsLTExLjAyMSBDIDc3LjQ3NCwzOC41NzIgODUuNzQxLDQ2LjI1IDg5LjM3Miw1MCA4My45OTUsNTUuNTU1IDY4LjQ2LDY5LjcyOSA1MCw2OS43MjkgeiINCiAgIGlkPSJwYXRoMyIgLz4NCjwvc3ZnPg==)}.emote-menu .edit-starred{background-color:#323232;background-image:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iNTAiDQogICBoZWlnaHQ9IjUwIg0KICAgdmlld0JveD0iMCAwIDUwIDUwIg0KICAgaWQ9IkxheWVyXzEiDQogICB4bWw6c3BhY2U9InByZXNlcnZlIj48bWV0YWRhdGENCiAgIGlkPSJtZXRhZGF0YTMwMDEiPjxyZGY6UkRGPjxjYzpXb3JrDQogICAgICAgcmRmOmFib3V0PSIiPjxkYzpmb3JtYXQ+aW1hZ2Uvc3ZnK3htbDwvZGM6Zm9ybWF0PjxkYzp0eXBlDQogICAgICAgICByZGY6cmVzb3VyY2U9Imh0dHA6Ly9wdXJsLm9yZy9kYy9kY21pdHlwZS9TdGlsbEltYWdlIiAvPjxkYzp0aXRsZT48L2RjOnRpdGxlPjwvY2M6V29yaz48L3JkZjpSREY+PC9tZXRhZGF0YT48ZGVmcw0KICAgaWQ9ImRlZnMyOTk5IiAvPg0KPHBhdGgNCiAgIGQ9Im0gNDMuMDQsMjIuNjk2IC03LjU2OCw3LjM3NyAxLjc4NywxMC40MTcgYyAwLjEyNywwLjc1IC0wLjE4MiwxLjUwOSAtMC43OTcsMS45NTcgLTAuMzQ4LDAuMjUzIC0wLjc2MiwwLjM4MiAtMS4xNzYsMC4zODIgLTAuMzE4LDAgLTAuNjM4LC0wLjA3NiAtMC45MzEsLTAuMjMgTCAyNSwzNy42ODEgMTUuNjQ1LDQyLjU5OSBjIC0wLjY3NCwwLjM1NSAtMS40OSwwLjI5NSAtMi4xMDcsLTAuMTUxIEMgMTIuOTIzLDQyIDEyLjYxNCw0MS4yNDIgMTIuNzQzLDQwLjQ5MSBMIDE0LjUzLDMwLjA3NCA2Ljk2MiwyMi42OTcgQyA2LjQxNSwyMi4xNjYgNi4yMjEsMjEuMzcxIDYuNDU0LDIwLjY0NyA2LjY5LDE5LjkyMyA3LjMxNSwxOS4zOTYgOC4wNjksMTkuMjg2IGwgMTAuNDU5LC0xLjUyMSA0LjY4LC05LjQ3OCBDIDIzLjU0Myw3LjYwMyAyNC4yMzksNy4xNzEgMjUsNy4xNzEgYyAwLjc2MywwIDEuNDU2LDAuNDMyIDEuNzkzLDEuMTE1IGwgNC42NzksOS40NzggMTAuNDYxLDEuNTIxIGMgMC43NTIsMC4xMDkgMS4zNzksMC42MzcgMS42MTIsMS4zNjEgMC4yMzcsMC43MjQgMC4wMzgsMS41MTkgLTAuNTA1LDIuMDUgeiINCiAgIGlkPSJwYXRoMjk5NSINCiAgIHN0eWxlPSJmaWxsOiNjY2NjY2M7ZmlsbC1vcGFjaXR5OjEiIC8+DQo8L3N2Zz4NCg==)}.emote-menu .emote>.edit-visibility{bottom:auto;left:auto;right:0;top:0}.emote-menu .emote>.edit-starred{bottom:auto;left:0;right:auto;top:0}.emote-menu .header-info>.edit-tool{margin-left:5px}.emote-menu.editing .edit-tool{display:inline-block}.emote-menu .emote-menu-hidden .edit-visibility{background-image:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iMTAwIg0KICAgaGVpZ2h0PSIxMDAiDQogICB2aWV3Qm94PSIwIDAgMTAwIDEwMCINCiAgIGlkPSJMYXllcl8zIg0KICAgeG1sOnNwYWNlPSJwcmVzZXJ2ZSI+PG1ldGFkYXRhDQogICBpZD0ibWV0YWRhdGExNSI+PHJkZjpSREY+PGNjOldvcmsNCiAgICAgICByZGY6YWJvdXQ9IiI+PGRjOmZvcm1hdD5pbWFnZS9zdmcreG1sPC9kYzpmb3JtYXQ+PGRjOnR5cGUNCiAgICAgICAgIHJkZjpyZXNvdXJjZT0iaHR0cDovL3B1cmwub3JnL2RjL2RjbWl0eXBlL1N0aWxsSW1hZ2UiIC8+PGRjOnRpdGxlPjwvZGM6dGl0bGU+PC9jYzpXb3JrPjwvcmRmOlJERj48L21ldGFkYXRhPjxkZWZzDQogICBpZD0iZGVmczEzIiAvPg0KPGcNCiAgIGlkPSJnMyI+DQoJPHBhdGgNCiAgIGQ9Ik0gNzAuMDgyLDQ1LjQ3NSA1MC40NzQsNjUuMDgyIEMgNjEuMTk4LDY0LjgzMSA2OS44MzEsNTYuMTk3IDcwLjA4Miw0NS40NzUgeiINCiAgIGlkPSJwYXRoNSINCiAgIHN0eWxlPSJmaWxsOiNGRkZGRkYiIC8+DQoJPHBhdGgNCiAgIGQ9Im0gOTcuOTY0LDQ2LjU0OCBjIC0wLjQ1LC0wLjUyOSAtNi4yNDUsLTcuMjMgLTE1LjQwMywtMTMuNTU0IGwgLTYuMiw2LjIgQyA4Mi4zNTEsNDMuMTQ4IDg2LjkyLDQ3LjQ2OSA4OS4zNzIsNTAgODMuOTk1LDU1LjU1NSA2OC40Niw2OS43MjkgNTAsNjkuNzI5IGMgLTEuMzM0LDAgLTIuNjUxLC0wLjA4MiAtMy45NTIsLTAuMjIyIGwgLTcuNDM5LDcuNDM5IGMgMy42MzksMC45MDkgNy40NDksMS40NSAxMS4zOTEsMS40NSAyNi40MjcsMCA0Ny4wOTgsLTIzLjkyNiA0Ny45NjUsLTI0Ljk0NiAxLjcwMSwtMS45OTkgMS43MDEsLTQuOTAxIC0wLjAwMSwtNi45MDIgeiINCiAgIGlkPSJwYXRoNyINCiAgIHN0eWxlPSJmaWxsOiNGRkZGRkYiIC8+DQoJPHBhdGgNCiAgIGQ9Im0gOTEuNDExLDE2LjY2IGMgMCwtMC4yNjYgLTAuMTA1LC0wLjUyIC0wLjI5MywtMC43MDcgbCAtNy4wNzEsLTcuMDcgYyAtMC4zOTEsLTAuMzkxIC0xLjAyMywtMC4zOTEgLTEuNDE0LDAgTCA2Ni44MDQsMjQuNzExIEMgNjEuNjAyLDIyLjgxOCA1NS45NDksMjEuNjAzIDUwLDIxLjYwMyBjIC0yNi40MjcsMCAtNDcuMDk4LDIzLjkyNiAtNDcuOTY1LDI0Ljk0NiAtMS43MDEsMiAtMS43MDEsNC45MDIgMTBlLTQsNi45MDMgMC41MTcsMC42MDcgOC4wODMsOS4zNTQgMTkuNzA3LDE2LjMyIEwgOC44ODMsODIuNjMyIEMgOC42OTUsODIuODIgOC41OSw4My4wNzMgOC41OSw4My4zMzkgYyAwLDAuMjY2IDAuMTA1LDAuNTIgMC4yOTMsMC43MDcgbCA3LjA3MSw3LjA3IGMgMC4xOTUsMC4xOTUgMC40NTEsMC4yOTMgMC43MDcsMC4yOTMgMC4yNTYsMCAwLjUxMiwtMC4wOTggMC43MDcsLTAuMjkzIGwgNzMuNzUsLTczLjc1IGMgMC4xODcsLTAuMTg2IDAuMjkzLC0wLjQ0IDAuMjkzLC0wLjcwNiB6IE0gMTAuNjI4LDUwIEMgMTQuMjU5LDQ2LjI0OSAyMi41MjYsMzguNTcxIDMzLjE5NSwzMy45NzkgMzEuMTE0LDM3LjE0NSAyOS44OTQsNDAuOTI4IDI5Ljg5NCw0NSBjIDAsNC42NjUgMS42MDEsOC45NDUgNC4yNywxMi4zNTEgTCAyOC4wNCw2My40NzUgQyAxOS44ODgsNTguOTU1IDEzLjY0OSw1My4xMiAxMC42MjgsNTAgeiINCiAgIGlkPSJwYXRoOSINCiAgIHN0eWxlPSJmaWxsOiNGRkZGRkYiIC8+DQo8L2c+DQo8L3N2Zz4NCg==);background-color:red}.emote-menu .emote-menu-starred .edit-starred{background-image:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iNTAiDQogICBoZWlnaHQ9IjUwIg0KICAgdmlld0JveD0iMCAwIDUwIDUwIg0KICAgaWQ9IkxheWVyXzEiDQogICB4bWw6c3BhY2U9InByZXNlcnZlIj48bWV0YWRhdGENCiAgIGlkPSJtZXRhZGF0YTMwMDEiPjxyZGY6UkRGPjxjYzpXb3JrDQogICAgICAgcmRmOmFib3V0PSIiPjxkYzpmb3JtYXQ+aW1hZ2Uvc3ZnK3htbDwvZGM6Zm9ybWF0PjxkYzp0eXBlDQogICAgICAgICByZGY6cmVzb3VyY2U9Imh0dHA6Ly9wdXJsLm9yZy9kYy9kY21pdHlwZS9TdGlsbEltYWdlIiAvPjxkYzp0aXRsZT48L2RjOnRpdGxlPjwvY2M6V29yaz48L3JkZjpSREY+PC9tZXRhZGF0YT48ZGVmcw0KICAgaWQ9ImRlZnMyOTk5IiAvPg0KPHBhdGgNCiAgIGQ9Im0gNDMuMDQsMjIuNjk2IC03LjU2OCw3LjM3NyAxLjc4NywxMC40MTcgYyAwLjEyNywwLjc1IC0wLjE4MiwxLjUwOSAtMC43OTcsMS45NTcgLTAuMzQ4LDAuMjUzIC0wLjc2MiwwLjM4MiAtMS4xNzYsMC4zODIgLTAuMzE4LDAgLTAuNjM4LC0wLjA3NiAtMC45MzEsLTAuMjMgTCAyNSwzNy42ODEgMTUuNjQ1LDQyLjU5OSBjIC0wLjY3NCwwLjM1NSAtMS40OSwwLjI5NSAtMi4xMDcsLTAuMTUxIEMgMTIuOTIzLDQyIDEyLjYxNCw0MS4yNDIgMTIuNzQzLDQwLjQ5MSBMIDE0LjUzLDMwLjA3NCA2Ljk2MiwyMi42OTcgQyA2LjQxNSwyMi4xNjYgNi4yMjEsMjEuMzcxIDYuNDU0LDIwLjY0NyA2LjY5LDE5LjkyMyA3LjMxNSwxOS4zOTYgOC4wNjksMTkuMjg2IGwgMTAuNDU5LC0xLjUyMSA0LjY4LC05LjQ3OCBDIDIzLjU0Myw3LjYwMyAyNC4yMzksNy4xNzEgMjUsNy4xNzEgYyAwLjc2MywwIDEuNDU2LDAuNDMyIDEuNzkzLDEuMTE1IGwgNC42NzksOS40NzggMTAuNDYxLDEuNTIxIGMgMC43NTIsMC4xMDkgMS4zNzksMC42MzcgMS42MTIsMS4zNjEgMC4yMzcsMC43MjQgMC4wMzgsMS41MTkgLTAuNTA1LDIuMDUgeiINCiAgIGlkPSJwYXRoMjk5NSINCiAgIHN0eWxlPSJmaWxsOiNmZmNjMDA7ZmlsbC1vcGFjaXR5OjEiIC8+DQo8L3N2Zz4NCg==)}.emote-menu .emote.emote-menu-starred{border-color:rgba(200,200,0,.5)}.emote-menu .emote.emote-menu-hidden{border-color:rgba(255,0,0,.5)}.emote-menu #starred-emotes-group .emote:not(.emote-menu-starred),.emote-menu:not(.editing) .emote-menu-hidden{display:none}.emote-menu:not(.editing) #starred-emotes-group .emote-menu-starred{border-color:transparent}.emote-menu #starred-emotes-group{text-align:center;color:#646464}.emote-menu #starred-emotes-group:empty:before{content:\"Use the edit mode to star an emote!\";position:relative;top:8px}.emote-menu .scrollable{height:calc(100% - 101px);overflow-y:auto}.emote-menu .sticky{position:absolute;bottom:0;width:100%}.emote-menu .emote-menu-inner{position:relative;max-height:100%;height:100%}")); },{}],3:[function(require,module,exports){ module.exports = (function() { var Hogan = require('hogan.js/lib/template.js'); var templates = {}; templates['emote'] = new Hogan.Template({code: function (c,p,i) { var t=this;t.b(i=i||"");t.b("<div class=\"emote");if(t.s(t.f("thirdParty",c,p,1),c,p,0,32,44,"{{ }}")){t.rs(c,p,function(c,p,t){t.b(" third-party");});c.pop();}if(!t.s(t.f("isVisible",c,p,1),c,p,1,0,0,"")){t.b(" emote-menu-hidden");};if(t.s(t.f("isStarred",c,p,1),c,p,0,119,138,"{{ }}")){t.rs(c,p,function(c,p,t){t.b(" emote-menu-starred");});c.pop();}t.b("\" data-emote=\"");t.b(t.v(t.f("text",c,p,0)));t.b("\" title=\"");t.b(t.v(t.f("text",c,p,0)));if(t.s(t.f("thirdParty",c,p,1),c,p,0,206,229,"{{ }}")){t.rs(c,p,function(c,p,t){t.b(" (from 3rd party addon)");});c.pop();}t.b("\">\r");t.b("\n" + i);t.b(" <img src=\"");t.b(t.t(t.f("url",c,p,0)));t.b("\">\r");t.b("\n" + i);t.b(" <div class=\"edit-tool edit-starred\" data-which=\"");t.b(t.v(t.f("text",c,p,0)));t.b("\" data-command=\"toggle-starred\" title=\"Star/unstar emote: ");t.b(t.v(t.f("text",c,p,0)));t.b("\"></div>\r");t.b("\n" + i);t.b(" <div class=\"edit-tool edit-visibility\" data-which=\"");t.b(t.v(t.f("text",c,p,0)));t.b("\" data-command=\"toggle-visibility\" title=\"Hide/show emote: ");t.b(t.v(t.f("text",c,p,0)));t.b("\"></div>\r");t.b("\n" + i);t.b("</div>\r");t.b("\n");return t.fl(); },partials: {}, subs: { }}); templates['emoteButton'] = new Hogan.Template({code: function (c,p,i) { var t=this;t.b(i=i||"");t.b("<button class=\"button button--icon-only float-left\" title=\"Emote Menu\" id=\"emote-menu-button\"></button>\r");t.b("\n");return t.fl(); },partials: {}, subs: { }}); templates['emoteGroupHeader'] = new Hogan.Template({code: function (c,p,i) { var t=this;t.b(i=i||"");t.b("<div class=\"group-header\" data-emote-channel=\"");t.b(t.v(t.f("channel",c,p,0)));t.b("\">\r");t.b("\n" + i);t.b(" <div class=\"header-info\">\r");t.b("\n" + i);t.b(" <img src=\"");t.b(t.v(t.f("badge",c,p,0)));t.b("\" />\r");t.b("\n" + i);t.b(" ");t.b(t.v(t.f("channelDisplayName",c,p,0)));t.b("\r");t.b("\n" + i);t.b(" <div class=\"edit-tool edit-visibility\" data-which=\"channel-");t.b(t.v(t.f("channel",c,p,0)));t.b("\" data-command=\"toggle-visibility\" title=\"Hide/show current emotes for ");t.b(t.v(t.f("channelDisplayName",c,p,0)));t.b(" (note: new emotes will still show up if they are added)\"></div>\r");t.b("\n" + i);t.b(" </div>\r");t.b("\n" + i);t.b(" <div class=\"emote-container\"></div>\r");t.b("\n" + i);t.b("</div>\r");t.b("\n");return t.fl(); },partials: {}, subs: { }}); templates['menu'] = new Hogan.Template({code: function (c,p,i) { var t=this;t.b(i=i||"");t.b("<div class=\"emote-menu\" id=\"emote-menu-for-twitch\">\r");t.b("\n" + i);t.b(" <div class=\"emote-menu-inner\">\r");t.b("\n" + i);t.b("\r");t.b("\n" + i);t.b(" <div class=\"draggable\"></div>\r");t.b("\n" + i);t.b("\r");t.b("\n" + i);t.b(" <div class=\"scrollable scrollbar-macosx\">\r");t.b("\n" + i);t.b(" <div class=\"group-container\" id=\"all-emotes-group\"></div>\r");t.b("\n" + i);t.b(" </div>\r");t.b("\n" + i);t.b("\r");t.b("\n" + i);t.b(" <div class=\"sticky\">\r");t.b("\n" + i);t.b(" <div class=\"group-header single-row\" id=\"starred-emotes-group\">\r");t.b("\n" + i);t.b(" <div class=\"header-info\">Favorite Emotes</div>\r");t.b("\n" + i);t.b(" <div class=\"emote-container\"></div>\r");t.b("\n" + i);t.b(" </div>\r");t.b("\n" + i);t.b("\r");t.b("\n" + i);t.b(" <div class=\"footer\">\r");t.b("\n" + i);t.b(" <a class=\"pull-left icon icon-home\" href=\"http://cletusc.github.io/Userscript--Twitch-Chat-Emotes\" target=\"_blank\" title=\"Visit the homepage where you can donate, post a review, or contact the developer\"></a>\r");t.b("\n" + i);t.b(" <a class=\"pull-left icon icon-gear\" data-command=\"toggle-editing\" title=\"Toggle edit mode\"></a>\r");t.b("\n" + i);t.b(" <a class=\"pull-right icon icon-resize-handle\" data-command=\"resize-handle\"></a>\r");t.b("\n" + i);t.b(" <a class=\"pull-right icon icon-pin\" data-command=\"toggle-pinned\" title=\"Pin/unpin the emote menu to the screen\"></a>\r");t.b("\n" + i);t.b(" </div>\r");t.b("\n" + i);t.b(" </div>\r");t.b("\n" + i);t.b("\r");t.b("\n" + i);t.b(" </div>\r");t.b("\n" + i);t.b("</div>\r");t.b("\n");return t.fl(); },partials: {}, subs: { }}); templates['newsMessage'] = new Hogan.Template({code: function (c,p,i) { var t=this;t.b(i=i||"");t.b("\r");t.b("\n" + i);t.b("<div class=\"twitch-chat-emotes-news\">\r");t.b("\n" + i);t.b(" [");t.b(t.v(t.f("scriptName",c,p,0)));t.b("] News: ");t.b(t.t(t.f("message",c,p,0)));t.b(" (<a href=\"#\" data-command=\"twitch-chat-emotes:dismiss-news\" data-news-id=\"");t.b(t.v(t.f("id",c,p,0)));t.b("\">Dismiss</a>)\r");t.b("\n" + i);t.b("</div>\r");t.b("\n");return t.fl(); },partials: {}, subs: { }}); return templates; })(); },{"hogan.js/lib/template.js":4}],4:[function(require,module,exports){ /* * Copyright 2011 Twitter, Inc. * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ var Hogan = {}; (function (Hogan) { Hogan.Template = function (codeObj, text, compiler, options) { codeObj = codeObj || {}; this.r = codeObj.code || this.r; this.c = compiler; this.options = options || {}; this.text = text || ''; this.partials = codeObj.partials || {}; this.subs = codeObj.subs || {}; this.buf = ''; } Hogan.Template.prototype = { // render: replaced by generated code. r: function (context, partials, indent) { return ''; }, // variable escaping v: hoganEscape, // triple stache t: coerceToString, render: function render(context, partials, indent) { return this.ri([context], partials || {}, indent); }, // render internal -- a hook for overrides that catches partials too ri: function (context, partials, indent) { return this.r(context, partials, indent); }, // ensurePartial ep: function(symbol, partials) { var partial = this.partials[symbol]; // check to see that if we've instantiated this partial before var template = partials[partial.name]; if (partial.instance && partial.base == template) { return partial.instance; } if (typeof template == 'string') { if (!this.c) { throw new Error("No compiler available."); } template = this.c.compile(template, this.options); } if (!template) { return null; } // We use this to check whether the partials dictionary has changed this.partials[symbol].base = template; if (partial.subs) { // Make sure we consider parent template now if (!partials.stackText) partials.stackText = {}; for (key in partial.subs) { if (!partials.stackText[key]) { partials.stackText[key] = (this.activeSub !== undefined && partials.stackText[this.activeSub]) ? partials.stackText[this.activeSub] : this.text; } } template = createSpecializedPartial(template, partial.subs, partial.partials, this.stackSubs, this.stackPartials, partials.stackText); } this.partials[symbol].instance = template; return template; }, // tries to find a partial in the current scope and render it rp: function(symbol, context, partials, indent) { var partial = this.ep(symbol, partials); if (!partial) { return ''; } return partial.ri(context, partials, indent); }, // render a section rs: function(context, partials, section) { var tail = context[context.length - 1]; if (!isArray(tail)) { section(context, partials, this); return; } for (var i = 0; i < tail.length; i++) { context.push(tail[i]); section(context, partials, this); context.pop(); } }, // maybe start a section s: function(val, ctx, partials, inverted, start, end, tags) { var pass; if (isArray(val) && val.length === 0) { return false; } if (typeof val == 'function') { val = this.ms(val, ctx, partials, inverted, start, end, tags); } pass = !!val; if (!inverted && pass && ctx) { ctx.push((typeof val == 'object') ? val : ctx[ctx.length - 1]); } return pass; }, // find values with dotted names d: function(key, ctx, partials, returnFound) { var found, names = key.split('.'), val = this.f(names[0], ctx, partials, returnFound), doModelGet = this.options.modelGet, cx = null; if (key === '.' && isArray(ctx[ctx.length - 2])) { val = ctx[ctx.length - 1]; } else { for (var i = 1; i < names.length; i++) { found = findInScope(names[i], val, doModelGet); if (found !== undefined) { cx = val; val = found; } else { val = ''; } } } if (returnFound && !val) { return false; } if (!returnFound && typeof val == 'function') { ctx.push(cx); val = this.mv(val, ctx, partials); ctx.pop(); } return val; }, // find values with normal names f: function(key, ctx, partials, returnFound) { var val = false, v = null, found = false, doModelGet = this.options.modelGet; for (var i = ctx.length - 1; i >= 0; i--) { v = ctx[i]; val = findInScope(key, v, doModelGet); if (val !== undefined) { found = true; break; } } if (!found) { return (returnFound) ? false : ""; } if (!returnFound && typeof val == 'function') { val = this.mv(val, ctx, partials); } return val; }, // higher order templates ls: function(func, cx, partials, text, tags) { var oldTags = this.options.delimiters; this.options.delimiters = tags; this.b(this.ct(coerceToString(func.call(cx, text)), cx, partials)); this.options.delimiters = oldTags; return false; }, // compile text ct: function(text, cx, partials) { if (this.options.disableLambda) { throw new Error('Lambda features disabled.'); } return this.c.compile(text, this.options).render(cx, partials); }, // template result buffering b: function(s) { this.buf += s; }, fl: function() { var r = this.buf; this.buf = ''; return r; }, // method replace section ms: function(func, ctx, partials, inverted, start, end, tags) { var textSource, cx = ctx[ctx.length - 1], result = func.call(cx); if (typeof result == 'function') { if (inverted) { return true; } else { textSource = (this.activeSub && this.subsText && this.subsText[this.activeSub]) ? this.subsText[this.activeSub] : this.text; return this.ls(result, cx, partials, textSource.substring(start, end), tags); } } return result; }, // method replace variable mv: function(func, ctx, partials) { var cx = ctx[ctx.length - 1]; var result = func.call(cx); if (typeof result == 'function') { return this.ct(coerceToString(result.call(cx)), cx, partials); } return result; }, sub: function(name, context, partials, indent) { var f = this.subs[name]; if (f) { this.activeSub = name; f(context, partials, this, indent); this.activeSub = false; } } }; //Find a key in an object function findInScope(key, scope, doModelGet) { var val; if (scope && typeof scope == 'object') { if (scope[key] !== undefined) { val = scope[key]; // try lookup with get for backbone or similar model data } else if (doModelGet && scope.get && typeof scope.get == 'function') { val = scope.get(key); } } return val; } function createSpecializedPartial(instance, subs, partials, stackSubs, stackPartials, stackText) { function PartialTemplate() {}; PartialTemplate.prototype = instance; function Substitutions() {}; Substitutions.prototype = instance.subs; var key; var partial = new PartialTemplate(); partial.subs = new Substitutions(); partial.subsText = {}; //hehe. substext. partial.buf = ''; stackSubs = stackSubs || {}; partial.stackSubs = stackSubs; partial.subsText = stackText; for (key in subs) { if (!stackSubs[key]) stackSubs[key] = subs[key]; } for (key in stackSubs) { partial.subs[key] = stackSubs[key]; } stackPartials = stackPartials || {}; partial.stackPartials = stackPartials; for (key in partials) { if (!stackPartials[key]) stackPartials[key] = partials[key]; } for (key in stackPartials) { partial.partials[key] = stackPartials[key]; } return partial; } var rAmp = /&/g, rLt = /</g, rGt = />/g, rApos = /\'/g, rQuot = /\"/g, hChars = /[&<>\"\']/; function coerceToString(val) { return String((val === null || val === undefined) ? '' : val); } function hoganEscape(str) { str = coerceToString(str); return hChars.test(str) ? str .replace(rAmp, '&') .replace(rLt, '<') .replace(rGt, '>') .replace(rApos, ''') .replace(rQuot, '"') : str; } var isArray = Array.isArray || function(a) { return Object.prototype.toString.call(a) === '[object Array]'; }; })(typeof exports !== 'undefined' ? exports : Hogan); },{}],5:[function(require,module,exports){ /** * jQuery CSS Customizable Scrollbar * * Copyright 2014, Yuriy Khabarov * Dual licensed under the MIT or GPL Version 2 licenses. * * If you found bug, please contact me via email <13real008@gmail.com> * * @author Yuriy Khabarov aka Gromo * @version 0.2.6 * @url https://github.com/gromo/jquery.scrollbar/ * */ (function(e,t,n){"use strict";function h(t){if(o.webkit&&!t){return{height:0,width:0}}if(!o.data.outer){var n={border:"none","box-sizing":"content-box",height:"200px",margin:"0",padding:"0",width:"200px"};o.data.inner=e("<div>").css(e.extend({},n));o.data.outer=e("<div>").css(e.extend({left:"-1000px",overflow:"scroll",position:"absolute",top:"-1000px"},n)).append(o.data.inner).appendTo("body")}o.data.outer.scrollLeft(1e3).scrollTop(1e3);return{height:Math.ceil(o.data.outer.offset().top-o.data.inner.offset().top||0),width:Math.ceil(o.data.outer.offset().left-o.data.inner.offset().left||0)}}function p(n,r){e(t).on({"blur.scrollbar":function(){e(t).add("body").off(".scrollbar");n&&n()},"dragstart.scrollbar":function(e){e.preventDefault();return false},"mouseup.scrollbar":function(){e(t).add("body").off(".scrollbar");n&&n()}});e("body").on({"selectstart.scrollbar":function(e){e.preventDefault();return false}});r&&r.preventDefault();return false}function d(){var e=h(true);return!(e.height||e.width)}function v(e){var t=e.originalEvent;if(t.axis&&t.axis===t.HORIZONTAL_AXIS)return false;if(t.wheelDeltaX)return false;return true}var r=false;var i=1,s="px";var o={data:{},macosx:n.navigator.platform.toLowerCase().indexOf("mac")!==-1,mobile:/Android|webOS|iPhone|iPad|iPod|BlackBerry/i.test(n.navigator.userAgent),overlay:null,scroll:null,scrolls:[],webkit:/WebKit/.test(n.navigator.userAgent),log:r?function(t,r){var i=t;if(r&&typeof t!="string"){i=[];e.each(t,function(e,t){i.push('"'+e+'": '+t)});i=i.join(", ")}if(n.console&&n.console.log){n.console.log(i)}else{alert(i)}}:function(){}};var u={autoScrollSize:true,autoUpdate:true,debug:false,disableBodyScroll:false,duration:200,ignoreMobile:true,ignoreOverlay:true,scrollStep:30,showArrows:false,stepScrolling:true,type:"simple",scrollx:null,scrolly:null,onDestroy:null,onInit:null,onScroll:null,onUpdate:null};var a=function(t,r){if(!o.scroll){o.log("Init jQuery Scrollbar v0.2.6");o.overlay=d();o.scroll=h();c();e(n).resize(function(){var e=false;if(o.scroll&&(o.scroll.height||o.scroll.width)){var t=h();if(t.height!=o.scroll.height||t.width!=o.scroll.width){o.scroll=t;e=true}}c(e)})}this.container=t;this.options=e.extend({},u,n.jQueryScrollbarOptions||{});this.scrollTo=null;this.scrollx={};this.scrolly={};this.init(r)};a.prototype={destroy:function(){if(!this.wrapper){return}var n=this.container.scrollLeft();var r=this.container.scrollTop();this.container.insertBefore(this.wrapper).css({height:"",margin:""}).removeClass("scroll-content").removeClass("scroll-scrollx_visible").removeClass("scroll-scrolly_visible").off(".scrollbar").scrollLeft(n).scrollTop(r);this.scrollx.scrollbar.removeClass("scroll-scrollx_visible").find("div").andSelf().off(".scrollbar");this.scrolly.scrollbar.removeClass("scroll-scrolly_visible").find("div").andSelf().off(".scrollbar");this.wrapper.remove();e(t).add("body").off(".scrollbar");if(e.isFunction(this.options.onDestroy))this.options.onDestroy.apply(this,[this.container])},getScrollbar:function(t){var n=this.options["scroll"+t];var r={advanced:'<div class="scroll-element_corner"></div>'+'<div class="scroll-arrow scroll-arrow_less"></div>'+'<div class="scroll-arrow scroll-arrow_more"></div>'+'<div class="scroll-element_outer">'+' <div class="scroll-element_size"></div>'+' <div class="scroll-element_inner-wrapper">'+' <div class="scroll-element_inner scroll-element_track">'+' <div class="scroll-element_inner-bottom"></div>'+" </div>"+" </div>"+' <div class="scroll-bar">'+' <div class="scroll-bar_body">'+' <div class="scroll-bar_body-inner"></div>'+" </div>"+' <div class="scroll-bar_bottom"></div>'+' <div class="scroll-bar_center"></div>'+" </div>"+"</div>",simple:'<div class="scroll-element_outer">'+' <div class="scroll-element_size"></div>'+' <div class="scroll-element_track"></div>'+' <div class="scroll-bar"></div>'+"</div>"};var i=r[this.options.type]?this.options.type:"advanced";if(n){if(typeof n=="string"){n=e(n).appendTo(this.wrapper)}else{n=e(n)}}else{n=e("<div>").addClass("scroll-element").html(r[i]).appendTo(this.wrapper)}if(this.options.showArrows){n.addClass("scroll-element_arrows_visible")}return n.addClass("scroll-"+t)},init:function(n){var r=this;var u=this.container;var a=this.containerWrapper||u;var f=e.extend(this.options,n||{});var l={x:this.scrollx,y:this.scrolly};var c=this.wrapper;var h={scrollLeft:u.scrollLeft(),scrollTop:u.scrollTop()};if(o.mobile&&f.ignoreMobile||o.overlay&&f.ignoreOverlay||o.macosx&&!o.webkit){return false}if(!c){this.wrapper=c=e("<div>").addClass("scroll-wrapper").addClass(u.attr("class")).css("position",u.css("position")=="absolute"?"absolute":"relative").insertBefore(u).append(u);if(u.is("textarea")){this.containerWrapper=a=e("<div>").insertBefore(u).append(u);c.addClass("scroll-textarea")}a.addClass("scroll-content").css({height:"","margin-bottom":o.scroll.height*-1+s,"margin-right":o.scroll.width*-1+s});u.on("scroll.scrollbar",function(t){if(e.isFunction(f.onScroll)){f.onScroll.call(r,{maxScroll:l.y.maxScrollOffset,scroll:u.scrollTop(),size:l.y.size,visible:l.y.visible},{maxScroll:l.x.maxScrollOffset,scroll:u.scrollLeft(),size:l.x.size,visible:l.x.visible})}l.x.isVisible&&l.x.scroller.css("left",u.scrollLeft()*l.x.kx+s);l.y.isVisible&&l.y.scroller.css("top",u.scrollTop()*l.y.kx+s)});c.on("scroll",function(){c.scrollTop(0).scrollLeft(0)});if(f.disableBodyScroll){var d=function(e){v(e)?l.y.isVisible&&l.y.mousewheel(e):l.x.isVisible&&l.x.mousewheel(e)};c.on({"MozMousePixelScroll.scrollbar":d,"mousewheel.scrollbar":d});if(o.mobile){c.on("touchstart.scrollbar",function(n){var r=n.originalEvent.touches&&n.originalEvent.touches[0]||n;var i={pageX:r.pageX,pageY:r.pageY};var s={left:u.scrollLeft(),top:u.scrollTop()};e(t).on({"touchmove.scrollbar":function(e){var t=e.originalEvent.targetTouches&&e.originalEvent.targetTouches[0]||e;u.scrollLeft(s.left+i.pageX-t.pageX);u.scrollTop(s.top+i.pageY-t.pageY);e.preventDefault()},"touchend.scrollbar":function(){e(t).off(".scrollbar")}})})}}if(e.isFunction(f.onInit))f.onInit.apply(this,[u])}else{a.css({height:"","margin-bottom":o.scroll.height*-1+s,"margin-right":o.scroll.width*-1+s})}e.each(l,function(n,s){var o=null;var a=1;var c=n=="x"?"scrollLeft":"scrollTop";var h=f.scrollStep;var d=function(){var e=u[c]();u[c](e+h);if(a==1&&e+h>=m)e=u[c]();if(a==-1&&e+h<=m)e=u[c]();if(u[c]()==e&&o){o()}};var m=0;if(!s.scrollbar){s.scrollbar=r.getScrollbar(n);s.scroller=s.scrollbar.find(".scroll-bar");s.mousewheel=function(e){if(!s.isVisible||n=="x"&&v(e)){return true}if(n=="y"&&!v(e)){l.x.mousewheel(e);return true}var t=e.originalEvent.wheelDelta*-1||e.originalEvent.detail;var i=s.size-s.visible-s.offset;if(!(m<=0&&t<0||m>=i&&t>0)){m=m+t;if(m<0)m=0;if(m>i)m=i;r.scrollTo=r.scrollTo||{};r.scrollTo[c]=m;setTimeout(function(){if(r.scrollTo){u.stop().animate(r.scrollTo,240,"linear",function(){m=u[c]()});r.scrollTo=null}},1)}e.preventDefault();return false};s.scrollbar.on({"MozMousePixelScroll.scrollbar":s.mousewheel,"mousewheel.scrollbar":s.mousewheel,"mouseenter.scrollbar":function(){m=u[c]()}});s.scrollbar.find(".scroll-arrow, .scroll-element_track").on("mousedown.scrollbar",function(t){if(t.which!=i)return true;a=1;var l={eventOffset:t[n=="x"?"pageX":"pageY"],maxScrollValue:s.size-s.visible-s.offset,scrollbarOffset:s.scroller.offset()[n=="x"?"left":"top"],scrollbarSize:s.scroller[n=="x"?"outerWidth":"outerHeight"]()};var v=0,g=0;if(e(this).hasClass("scroll-arrow")){a=e(this).hasClass("scroll-arrow_more")?1:-1;h=f.scrollStep*a;m=a>0?l.maxScrollValue:0}else{a=l.eventOffset>l.scrollbarOffset+l.scrollbarSize?1:l.eventOffset<l.scrollbarOffset?-1:0;h=Math.round(s.visible*.75)*a;m=l.eventOffset-l.scrollbarOffset-(f.stepScrolling?a==1?l.scrollbarSize:0:Math.round(l.scrollbarSize/2));m=u[c]()+m/s.kx}r.scrollTo=r.scrollTo||{};r.scrollTo[c]=f.stepScrolling?u[c]()+h:m;if(f.stepScrolling){o=function(){m=u[c]();clearInterval(g);clearTimeout(v);v=0;g=0};v=setTimeout(function(){g=setInterval(d,40)},f.duration+100)}setTimeout(function(){if(r.scrollTo){u.animate(r.scrollTo,f.duration);r.scrollTo=null}},1);return p(o,t)});s.scroller.on("mousedown.scrollbar",function(r){if(r.which!=i)return true;var o=r[n=="x"?"pageX":"pageY"];var a=u[c]();s.scrollbar.addClass("scroll-draggable");e(t).on("mousemove.scrollbar",function(e){var t=parseInt((e[n=="x"?"pageX":"pageY"]-o)/s.kx,10);u[c](a+t)});return p(function(){s.scrollbar.removeClass("scroll-draggable");m=u[c]()},r)})}});e.each(l,function(e,t){var n="scroll-scroll"+e+"_visible";var r=e=="x"?l.y:l.x;t.scrollbar.removeClass(n);r.scrollbar.removeClass(n);a.removeClass(n)});e.each(l,function(t,n){e.extend(n,t=="x"?{offset:parseInt(u.css("left"),10)||0,size:u.prop("scrollWidth"),visible:c.width()}:{offset:parseInt(u.css("top"),10)||0,size:u.prop("scrollHeight"),visible:c.height()})});var m=function(t,n){var r="scroll-scroll"+t+"_visible";var i=t=="x"?l.y:l.x;var f=parseInt(u.css(t=="x"?"left":"top"),10)||0;var h=n.size;var p=n.visible+f;n.isVisible=h-p>1;if(n.isVisible){n.scrollbar.addClass(r);i.scrollbar.addClass(r);a.addClass(r)}else{n.scrollbar.removeClass(r);i.scrollbar.removeClass(r);a.removeClass(r)}if(t=="y"&&(n.isVisible||n.size<n.visible)){a.css("height",p+o.scroll.height+s)}if(l.x.size!=u.prop("scrollWidth")||l.y.size!=u.prop("scrollHeight")||l.x.visible!=c.width()||l.y.visible!=c.height()||l.x.offset!=(parseInt(u.css("left"),10)||0)||l.y.offset!=(parseInt(u.css("top"),10)||0)){e.each(l,function(t,n){e.extend(n,t=="x"?{offset:parseInt(u.css("left"),10)||0,size:u.prop("scrollWidth"),visible:c.width()}:{offset:parseInt(u.css("top"),10)||0,size:u.prop("scrollHeight"),visible:c.height()})});m(t=="x"?"y":"x",i)}};e.each(l,m);if(e.isFunction(f.onUpdate))f.onUpdate.apply(this,[u]);e.each(l,function(e,t){var n=e=="x"?"left":"top";var r=e=="x"?"outerWidth":"outerHeight";var i=e=="x"?"width":"height";var o=parseInt(u.css(n),10)||0;var a=t.size;var l=t.visible+o;var c=t.scrollbar.find(".scroll-element_size");c=c[r]()+(parseInt(c.css(n),10)||0);if(f.autoScrollSize){t.scrollbarSize=parseInt(c*l/a,10);t.scroller.css(i,t.scrollbarSize+s)}t.scrollbarSize=t.scroller[r]();t.kx=(c-t.scrollbarSize)/(a-l)||1;t.maxScrollOffset=a-l});u.scrollLeft(h.scrollLeft).scrollTop(h.scrollTop).trigger("scroll")}};e.fn.scrollbar=function(t,n){var r=this;if(t==="get")r=null;this.each(function(){var i=e(this);if(i.hasClass("scroll-wrapper")||i.get(0).nodeName=="body"){return true}var s=i.data("scrollbar");if(s){if(t==="get"){r=s;return false}var u=typeof t=="string"&&s[t]?t:"init";s[u].apply(s,e.isArray(n)?n:[]);if(t==="destroy"){i.removeData("scrollbar");while(e.inArray(s,o.scrolls)>=0)o.scrolls.splice(e.inArray(s,o.scrolls),1)}}else{if(typeof t!="string"){s=new a(i,t);i.data("scrollbar",s);o.scrolls.push(s)}}return true});return r};e.fn.scrollbar.options=u;if(n.angular){(function(e){var t=e.module("jQueryScrollbar",[]);t.directive("jqueryScrollbar",function(){return{link:function(e,t){t.scrollbar(e.options).on("$destroy",function(){t.scrollbar("destroy")})},restring:"AC",scope:{options:"=jqueryScrollbar"}}})})(n.angular)}var f=0,l=0;var c=function(e){var t,n,i,s,u,a,h;for(t=0;t<o.scrolls.length;t++){s=o.scrolls[t];n=s.container;i=s.options;u=s.wrapper;a=s.scrollx;h=s.scrolly;if(e||i.autoUpdate&&u&&u.is(":visible")&&(n.prop("scrollWidth")!=a.size||n.prop("scrollHeight")!=h.size||u.width()!=a.visible||u.height()!=h.visible)){s.init();if(r){o.log({scrollHeight:n.prop("scrollHeight")+":"+s.scrolly.size,scrollWidth:n.prop("scrollWidth")+":"+s.scrollx.size,visibleHeight:u.height()+":"+s.scrolly.visible,visibleWidth:u.width()+":"+s.scrollx.visible},true);l++}}}if(r&&l>10){o.log("Scroll updates exceed 10");c=function(){}}else{clearTimeout(f);f=setTimeout(c,300)}}})(jQuery,document,window); },{}],6:[function(require,module,exports){ // Storage cache. var cache = {}; // The store handling expiration of data. var expiresStore = new Store({ namespace: '__storage-wrapper:expires' }); /** * Storage wrapper for making routine storage calls super easy. * @class Store * @constructor * @param {object} [options] The options for the store. Options not overridden will use the defaults. * @param {mixed} [options.namespace=''] See {{#crossLink "Store/setNamespace"}}Store#setNamespace{{/crossLink}} * @param {mixed} [options.storageType='local'] See {{#crossLink "Store/setStorageType"}}Store#setStorageType{{/crossLink}} */ function Store(options) { var settings = { namespace: '', storageType: 'local' }; /** * Sets the storage namespace. * @method setNamespace * @param {string|false|null} namespace The namespace to work under. To use no namespace (e.g. global namespace), pass in `false` or `null` or an empty string. */ this.setNamespace = function (namespace) { var validNamespace = /^[\w-:]+$/; // No namespace. if (namespace === false || namespace == null || namespace === '') { settings.namespace = ''; return; } if (typeof namespace !== 'string' || !validNamespace.test(namespace)) { throw new Error('Invalid namespace.'); } settings.namespace = namespace; }; /** * Gets the current storage namespace. * @method getNamespace * @return {string} The current namespace. */ this.getNamespace = function (includeSeparator) { if (includeSeparator && settings.namespace !== '') { return settings.namespace + ':'; } return settings.namespace; } /** * Sets the type of storage to use. * @method setStorageType * @param {string} type The type of storage to use. Use `session` for `sessionStorage` and `local` for `localStorage`. */ this.setStorageType = function (type) { if (['session', 'local'].indexOf(type) < 0) { throw new Error('Invalid storage type.'); } settings.storageType = type; }; /** * Get the type of storage being used. * @method getStorageType * @return {string} The type of storage being used. */ this.getStorageType = function () { return settings.storageType; }; // Override default settings. if (options) { for (var key in options) { switch (key) { case 'namespace': this.setNamespace(options[key]); break; case 'storageType': this.setStorageType(options[key]); break; } } } } /** * Gets the actual handler to use * @method getStorageHandler * @return {mixed} The storage handler. */ Store.prototype.getStorageHandler = function () { var handlers = { 'local': localStorage, 'session': sessionStorage }; return handlers[this.getStorageType()]; } /** * Gets the full storage name for a key, including the namespace, if any. * @method getStorageKey * @param {string} key The storage key name. * @return {string} The full storage name that is used by the storage methods. */ Store.prototype.getStorageKey = function (key) { if (!key || typeof key !== 'string' || key.length < 1) { throw new Error('Key must be a string.'); } return this.getNamespace(true) + key; }; /** * Gets a storage item from the current namespace. * @method get * @param {string} key The key that the data can be accessed under. * @param {mixed} defaultValue The default value to return in case the storage value is not set or `null`. * @return {mixed} The data for the storage. */ Store.prototype.get = function (key, defaultValue) { // Prevent recursion. Only check expire date if it isn't called from `expiresStore`. if (this !== expiresStore) { // Check if key is expired. var expireDate = expiresStore.get(this.getStorageKey(key)); if (expireDate !== null && expireDate.getTime() < Date.now()) { // Expired, remove it. this.remove(key); expiresStore.remove(this.getStorageKey(key)); } } // Cached, read from memory. if (cache[this.getStorageKey(key)] != null) { return cache[this.getStorageKey(key)]; } var val = this.getStorageHandler().getItem(this.getStorageKey(key)); // Value doesn't exist and we have a default, return default. if (val === null && typeof defaultValue !== 'undefined') { return defaultValue; } // Only pre-process strings. if (typeof val === 'string') { // Handle RegExps. if (val.indexOf('~RegExp:') === 0) { var matches = /^~RegExp:([gim]*?):(.*)/.exec(val); val = new RegExp(matches[2], matches[1]); } // Handle Dates. else if (val.indexOf('~Date:') === 0) { val = new Date(val.replace(/^~Date:/, '')); } // Handle numbers. else if (val.indexOf('~Number:') === 0) { val = parseInt(val.replace(/^~Number:/, ''), 10); } // Handle booleans. else if (val.indexOf('~Boolean:') === 0) { val = val.replace(/^~Boolean:/, '') === 'true'; } // Handle objects. else if (val.indexOf('~JSON:') === 0) { val = val.replace(/^~JSON:/, ''); // Try parsing it. try { val = JSON.parse(val); } // Parsing went wrong (invalid JSON), return default or null. catch (e) { if (typeof defaultValue !== 'undefined') { return defaultValue; } return null; } } } // Return it. cache[this.getStorageKey(key)] = val; return val; }; /** * Sets a storage item on the current namespace. * @method set * @param {string} key The key that the data can be accessed under. * @param {mixed} val The value to store. May be the following types of data: `RegExp`, `Date`, `Object`, `String`, `Boolean`, `Number` * @param {Date|number} [expires] The date in the future to expire, or relative number of milliseconds from `Date#now` to expire. * * Note: This converts special data types that normally can't be stored in the following way: * * - `RegExp`: prefixed with type, flags stored, and source stored as string. * - `Date`: prefixed with type, stored as string using `Date#toString`. * - `Object`: prefixed with "JSON" indicator, stored as string using `JSON#stringify`. */ Store.prototype.set = function (key, val, expires) { var parsedVal = null; // Handle RegExps. if (val instanceof RegExp) { var flags = [ val.global ? 'g' : '', val.ignoreCase ? 'i' : '', val.multiline ? 'm' : '', ].join(''); parsedVal = '~RegExp:' + flags + ':' + val.source; } // Handle Dates. else if (val instanceof Date) { parsedVal = '~Date:' + val.toString(); } // Handle objects. else if (val === Object(val)) { parsedVal = '~JSON:' + JSON.stringify(val); } // Handle numbers. else if (typeof val === 'number') { parsedVal = '~Number:' + val.toString(); } // Handle booleans. else if (typeof val === 'boolean') { parsedVal = '~Boolean:' + val.toString(); } // Handle strings. else if (typeof val === 'string') { parsedVal = val; } // Throw if we don't know what it is. else { throw new Error('Unable to store this value; wrong value type.'); } // Set expire date if needed. if (typeof expires !== 'undefined') { // Convert to a relative date. if (typeof expires === 'number') { expires = new Date(Date.now() + expires); } // Make sure it is a date. if (expires instanceof Date) { expiresStore.set(this.getStorageKey(key), expires); } else { throw new Error('Key expire must be a valid date or timestamp.'); } } // Save it. cache[this.getStorageKey(key)] = val; this.getStorageHandler().setItem(this.getStorageKey(key), parsedVal); }; /** * Gets all data for the current namespace. * @method getAll * @return {object} An object containing all data in the form of `{theKey: theData}` where `theData` is parsed using {{#crossLink "Store/get"}}Store#get{{/crossLink}}. */ Store.prototype.getAll = function () { var keys = this.listKeys(); var data = {}; keys.forEach(function (key) { data[key] = this.get(key); }, this); return data; }; /** * List all keys that are tied to the current namespace. * @method listKeys * @return {array} The storage keys. */ Store.prototype.listKeys = function () { var keys = []; var key = null; var storageLength = this.getStorageHandler().length; var prefix = new RegExp('^' + this.getNamespace(true)); for (var i = 0; i < storageLength; i++) { key = this.getStorageHandler().key(i) if (prefix.test(key)) { keys.push(key.replace(prefix, '')); } } return keys; }; /** * Removes a specific key and data from the current namespace. * @method remove * @param {string} key The key to remove the data for. */ Store.prototype.remove = function (key) { cache[this.getStorageKey(key)] = null; this.getStorageHandler().removeItem(this.getStorageKey(key)); }; /** * Removes all data and keys from the current namespace. * @method removeAll */ Store.prototype.removeAll = function () { this.listKeys().forEach(this.remove, this); }; /** * Removes namespaced items from the cache so your next {{#crossLink "Store/get"}}Store#get{{/crossLink}} will be fresh from the storage. * @method freshen * @param {string} key The key to remove the cache data for. */ Store.prototype.freshen = function (key) { var keys = key ? [key] : this.listKeys(); keys.forEach(function (key) { cache[this.getStorageKey(key)] = null; }, this); }; /** * Migrate data from a different namespace to current namespace. * @method migrate * @param {object} migration The migration object. * @param {string} migration.toKey The key name under your current namespace the old data should change to. * @param {string} migration.fromNamespace The old namespace that the old key belongs to. * @param {string} migration.fromKey The old key name to migrate from. * @param {string} [migration.fromStorageType] The storage type to migrate from. Defaults to same type as where you are migrating to. * @param {boolean} [migration.keepOldData=false] Whether old data should be kept after it has been migrated. * @param {boolean} [migration.overwriteNewData=false] Whether old data should overwrite currently stored data if it exists. * @param {function} [migration.transform] The function to pass the old key data through before migrating. * @example * * var Store = require('storage-wrapper'); * var store = new Store({ * namespace: 'myNewApp' * }); * * // Migrate from the old app. * store.migrate({ * toKey: 'new-key', * fromNamespace: 'myOldApp', * fromKey: 'old-key' * }); * * // Migrate from global data. Useful when moving from other storage wrappers or regular ol' `localStorage`. * store.migrate({ * toKey: 'other-new-key', * fromNamespace: '', * fromKey: 'other-old-key-on-global' * }); * * // Migrate some JSON data that was stored as a string. * store.migrate({ * toKey: 'new-json-key', * fromNamespace: 'myOldApp', * fromKey: 'old-json-key', * // Try converting some old JSON data. * transform: function (data) { * try { * return JSON.parse(data); * } * catch (e) { * return data; * } * } * }); */ Store.prototype.migrate = function (migration) { // Save our current namespace. var toNamespace = this.getNamespace(); var toStorageType = this.getStorageType(); // Create a temporary store to avoid changing namespace during actual get/sets. var store = new Store({ namespace: toNamespace, storageType: toStorageType }); var data = null; // Get data from old namespace. store.setNamespace(migration.fromNamespace); if (typeof migration.fromStorageType !== 'undefined') { store.setStorageType(migration.fromStorageType); } data = store.get(migration.fromKey); // Remove old if needed. if (!migration.keepOldData) { store.remove(migration.fromKey); } // No data, ignore this migration. if (data === null) { return; } // Transform data if needed. if (typeof migration.transform === 'function') { data = migration.transform(data); } else if (typeof migration.transform !== 'undefined') { throw new Error('Invalid transform callback.'); } // Go back to current namespace. store.setNamespace(toNamespace); store.setStorageType(toStorageType); // Only overwrite new data if it doesn't exist or it's requested. if (store.get(migration.toKey) === null || migration.overwriteNewData) { store.set(migration.toKey, data); } }; /** * Creates a substore that is nested in the current namespace. * @method createSubstore * @param {string} namespace The substore's namespace. * @return {Store} The substore. * @example * * var Store = require('storage-wrapper'); * // Create main store. * var store = new Store({ * namespace: 'myapp' * }); * * // Create substore. * var substore = store.createSubstore('things'); * substore.set('foo', 'bar'); * * substore.get('foo') === store.get('things:foo'); * // true */ Store.prototype.createSubstore = function (namespace) { return new Store({ namespace: this.getNamespace(true) + namespace, storageType: this.getStorageType() }); }; module.exports = Store; },{}],7:[function(require,module,exports){ module.exports={ "name": "twitch-chat-emotes", "version": "2.1.5", "homepage": "http://cletusc.github.io/Userscript--Twitch-Chat-Emotes/", "bugs": "https://github.com/cletusc/Userscript--Twitch-Chat-Emotes/issues", "author": "Ryan Chatham <ryan.b.chatham@gmail.com> (https://github.com/cletusc)", "repository": { "type": "git", "url": "https://github.com/cletusc/Userscript--Twitch-Chat-Emotes.git" }, "userscript": { "name": "Twitch Chat Emotes", "namespace": "#Cletus", "version": "{{{pkg.version}}}", "description": "Adds a button to Twitch that allows you to \"click-to-insert\" an emote.", "copyright": "2011+, {{{pkg.author}}}", "author": "{{{pkg.author}}}", "icon": "http://www.gravatar.com/avatar.php?gravatar_id=6875e83aa6c563790cb2da914aaba8b3&r=PG&s=48&default=identicon", "license": [ "MIT; http://opensource.org/licenses/MIT", "CC BY-NC-SA 3.0; http://creativecommons.org/licenses/by-nc-sa/3.0/" ], "homepage": "{{{pkg.homepage}}}", "supportURL": "{{{pkg.bugs}}}", "contributionURL": "http://cletusc.github.io/Userscript--Twitch-Chat-Emotes/#donate", "grant": "none", "include": [ "http://*.twitch.tv/*", "https://*.twitch.tv/*" ], "exclude": [ "http://api.twitch.tv/*", "https://api.twitch.tv/*", "http://tmi.twitch.tv/*", "https://tmi.twitch.tv/*", "http://*.twitch.tv/*/dashboard", "https://*.twitch.tv/*/dashboard", "http://chatdepot.twitch.tv/*", "https://chatdepot.twitch.tv/*", "http://im.twitch.tv/*", "https://im.twitch.tv/*", "http://platform.twitter.com/*", "https://platform.twitter.com/*", "http://www.facebook.com/*", "https://www.facebook.com/*" ] }, "devDependencies": { "browser-sync": "^1.3.2", "browserify": "^5.9.1", "generate-userscript-header": "^1.0.0", "gulp": "^3.8.3", "gulp-autoprefixer": "0.0.8", "gulp-beautify": "1.1.0", "gulp-changed": "^0.4.1", "gulp-concat": "^2.2.0", "gulp-conflict": "^0.1.2", "gulp-css-base64": "^1.1.0", "gulp-css2js": "^1.0.2", "gulp-header": "^1.0.2", "gulp-hogan-compile": "^0.2.1", "gulp-minify-css": "^0.3.5", "gulp-notify": "^1.4.1", "gulp-rename": "^1.2.0", "gulp-uglify": "^0.3.1", "gulp-util": "^3.0.0", "hogan.js": "^3.0.2", "jquery-ui": "^1.10.5", "jquery.scrollbar": "^0.2.7", "pretty-hrtime": "^0.2.1", "storage-wrapper": "cletusc/storage-wrapper#v0.1.1", "vinyl-map": "^1.0.1", "vinyl-source-stream": "^0.1.1", "watchify": "^1.0.1" } } },{}],8:[function(require,module,exports){ var logger = require('./logger'); var api = {}; var ember = null; var hookedFactories = {}; api.getEmber = function () { if (ember) { return ember; } if (window.App && window.App.__container__) { ember = window.App.__container__; return ember; } return false; }; api.isLoaded = function () { return Boolean(api.getEmber()); }; api.lookup = function (lookupFactory) { if (!api.isLoaded()) { logger.debug('Factory lookup failure, Ember not loaded.'); return false; } return api.getEmber().lookup(lookupFactory); }; api.hook = function (lookupFactory, activateCb, deactivateCb) { if (!api.isLoaded()) { logger.debug('Factory hook failure, Ember not loaded.'); return false; } if (hookedFactories[lookupFactory]) { logger.debug('Factory already hooked: ' + lookupFactory); return true; } var reopenOptions = {}; var factory = api.lookup(lookupFactory); if (!factory) { logger.debug('Factory hook failure, factory not found: ' + lookupFactory); return false; } if (activateCb) { reopenOptions.activate = function () { this._super(); activateCb.call(this); logger.debug('Hook run on activate: ' + lookupFactory); }; } if (deactivateCb) { reopenOptions.deactivate = function () { this._super(); deactivateCb.call(this); logger.debug('Hook run on deactivate: ' + lookupFactory); }; } try { factory.reopen(reopenOptions); hookedFactories[lookupFactory] = true; logger.debug('Factory hooked: ' + lookupFactory); return true; } catch (err) { logger.debug('Factory hook failure, unexpected error: ' + lookupFactory); logger.debug(err); return false; } }; api.get = function (lookupFactory, property) { if (!api.isLoaded()) { logger.debug('Factory get failure, Ember not loaded.'); return false; } var properties = property.split('.'); var getter = api.lookup(lookupFactory); properties.some(function (property) { // If getter fails, just exit, otherwise, keep looping. if (getter == null || typeof getter === 'undefined') { getter = null; return true; } if (getter[property] == null || typeof getter[property] === 'undefined') { getter = null; return true; } if (typeof getter.get === 'function') { getter = getter.get(property); if (getter == null || typeof getter === 'undefined') { getter = null; return true; } return false; } getter = getter[property]; }); return getter; }; module.exports = api; },{"./logger":10}],9:[function(require,module,exports){ var storage = require('./storage'); var logger = require('./logger'); var ui = require('./ui'); var api = {}; var emoteStore = new EmoteStore(); var $ = window.jQuery; /** * The entire emote storing system. */ function EmoteStore() { var getters = {}; var nativeEmotes = {}; var hasInitialized = false; /** * Get a list of usable emoticons. * @param {function} [filters] A filter method to limit what emotes are returned. Passed to Array#filter. * @param {function|string} [sortBy] How the emotes should be sorted. `function` will be passed to sort via Array#sort. `'channel'` sorts by channel name, globals first. All other values (or omitted) sort alphanumerically. * @param {string} [returnType] `'object'` will return in object format, e.g. `{'Kappa': Emote(...), ...}`. All other values (or omitted) return an array format, e.g. `[Emote(...), ...]`. * @return {object|array} See `returnType` param. */ this.getEmotes = function (filters, sortBy, returnType) { var twitchApi = require('./twitch-api'); // Get native emotes. var emotes = $.extend({}, nativeEmotes); // Parse the custom emotes provided by third party addons. Object.keys(getters).forEach(function (getterName) { // Try the getter. var results = null; try { results = getters[getterName](); } catch (err) { logger.debug('Emote getter `' + getterName + '` failed unexpectedly, skipping.', err.toString()); return; } if (!Array.isArray(results)) { logger.debug('Emote getter `' + getterName + '` must return an array, skipping.'); return; } // Override natives and previous getters. results.forEach(function (emote) { try { // Create the emote. var instance = new Emote(emote); // Force the getter. instance.setGetterName(getterName); // Force emotes without channels to the getter's name. if (!emote.channel) { instance.setChannelName(getterName); } // Add/override it. emotes[instance.getText()] = instance; } catch (err) { logger.debug('Emote parsing for getter `' + getterName + '` failed, skipping.', err.toString(), emote); } }); }); // Covert to array. emotes = Object.keys(emotes).map(function (emote) { return emotes[emote]; }); // Filter results. if (typeof filters === 'function') { emotes = emotes.filter(filters); } // Return as an object if requested. if (returnType === 'object') { var asObject = {}; emotes.forEach(function (emote) { asObject[emote.getText()] = emote; }); return asObject; } // Sort results. if (typeof sortBy === 'function') { emotes.sort(sortBy); } else if (sortBy === 'channel') { emotes.sort(sorting.allEmotesCategory); } else { emotes.sort(sorting.byText); } // Return the emotes in array format. return emotes; }; /** * Registers a 3rd party emote hook. * @param {string} name The name of the 3rd party registering the hook. * @param {function} getter The function called when looking for emotes. Must return an array of emote objects, e.g. `[emote, ...]`. See Emote class. */ this.registerGetter = function (name, getter) { if (typeof name !== 'string') { throw new Error('Name must be a string.'); } if (getters[name]) { throw new Error('Getter already exists.'); } if (typeof getter !== 'function') { throw new Error('Getter must be a function.'); } logger.debug('Getter registered: ' + name); getters[name] = getter; ui.updateEmotes(); }; /** * Registers a 3rd party emote hook. * @param {string} name The name of the 3rd party hook to deregister. */ this.deregisterGetter = function (name) { logger.debug('Getter unregistered: ' + name); delete getters[name]; ui.updateEmotes(); }; /** * Initializes the raw data from the API endpoints. Should be called at load and/or whenever the API may have changed. Populates internal objects with updated data. */ this.init = function () { if (hasInitialized) { logger.debug('Already initialized EmoteStore, stopping init.'); return; } logger.debug('Starting initialization.'); var twitchApi = require('./twitch-api'); var self = this; // Hash of emote set to forced channel. var forcedSetsToChannels = { // Globals. 0: '~global', // Bubble emotes. 33: 'turbo', // Monkey emotes. 42: 'turbo', // Hidden turbo emotes. 457: 'turbo', 793: 'turbo', 19151: 'twitch_prime', 19194: 'twitch_prime' }; logger.debug('Initializing emote set change listener.'); twitchApi.getEmotes(function (emoteSets) { logger.debug('Parsing emote sets.'); Object.keys(emoteSets).forEach(function (set) { var emotes = emoteSets[set]; set = Number(set); emotes.forEach(function (emote) { // Set some required info. emote.url = '//static-cdn.jtvnw.net/emoticons/v1/' + emote.id + '/1.0'; emote.text = getEmoteFromRegEx(emote.code); emote.set = set; // Hardcode the channels of certain sets. if (forcedSetsToChannels[set]) { emote.channel = forcedSetsToChannels[set]; } var instance = new Emote(emote); // Save the emote for use later. nativeEmotes[emote.text] = instance; }); }); logger.debug('Loading subscription data.'); // Get active subscriptions to find the channels. twitchApi.getTickets(function (tickets) { // Instances from each channel to preload channel data. var deferredChannelGets = {}; logger.debug('Tickets loaded from the API.', tickets); tickets.forEach(function (ticket) { var product = ticket.product; var channel = product.owner_name || product.short_name; // Get subscriptions with emotes only. if (!product.emoticons || !product.emoticons.length) { return; } // Set the channel on the emotes. product.emoticons.forEach(function (emote) { var instance = nativeEmotes[getEmoteFromRegEx(emote.regex)]; instance.setChannelName(channel); // Save instance for later, but only one instance per channel. if (!deferredChannelGets[channel]) { deferredChannelGets[channel] = instance; } }); }); // Preload channel data. Object.keys(deferredChannelGets).forEach(function (key) { var instance = deferredChannelGets[key]; instance.getChannelBadge(); instance.getChannelDisplayName(); }); ui.updateEmotes(); }); ui.updateEmotes(); }); hasInitialized = true; logger.debug('Finished EmoteStore initialization.'); }; }; /** * Gets a specific emote, if available. * @param {string} text The text of the emote to get. * @return {Emote|null} The Emote instance of the emote or `null` if it couldn't be found. */ EmoteStore.prototype.getEmote = function (text) { return this.getEmotes(null, null, 'object')[text] || null; }; /** * Emote object. * @param {object} details Object describing the emote. * @param {string} details.text The text to use in the chat box when emote is clicked. * @param {string} details.url The URL of the image for the emote. * @param {string} [details.badge] The URL of the badge for the emote. * @param {string} [details.channel] The channel the emote should be categorized under. * @param {string} [details.getterName] The 3rd party getter that registered the emote. Used internally only. */ function Emote(details) { var text = null; var url = null; var getterName = null; var channel = { name: null, displayName: null, badge: null }; /** * Gets the text of the emote. * @return {string} The emote text. */ this.getText = function () { return text; }; /** * Sets the text of the emote. * @param {string} theText The text to set. */ this.setText = function (theText) { if (typeof theText !== 'string' || theText.length < 1) { throw new Error('Invalid text'); } text = theText; }; /** * Gets the getter name this emote belongs to. * @return {string} The getter's name. */ this.getGetterName = function () { return getterName; }; /** * Sets the getter name this emote belongs to. * @param {string} theGetterName The getter's name. */ this.setGetterName = function (theGetterName) { if (typeof theGetterName !== 'string' || theGetterName.length < 1) { throw new Error('Invalid getter name'); } getterName = theGetterName; }; /** * Gets the emote's image URL. * @return {string} The emote image URL. */ this.getUrl = function () { return url; }; /** * Sets the emote's image URL. * @param {string} theUrl The image URL to set. */ this.setUrl = function (theUrl) { if (typeof theUrl !== 'string' || theUrl.length < 1) { throw new Error('Invalid URL'); } url = theUrl; }; /** * Gets the emote's channel name. * @return {string} The emote's channel or an empty string if it doesn't have one. */ this.getChannelName = function () { if (!channel.name) { channel.name = storage.channelNames.get(this.getText()); } return channel.name || ''; }; /** * Sets the emote's channel name. * @param {string} theChannel The channel name to set. */ this.setChannelName = function (theChannel) { if (typeof theChannel !== 'string' || theChannel.length < 1) { throw new Error('Invalid channel'); } // Only save the channel to storage if it's dynamic. if (theChannel !== '~global' && theChannel !== 'turbo' && theChannel !== 'twitch_prime') { storage.channelNames.set(this.getText(), theChannel); } channel.name = theChannel; }; /** * Gets the emote channel's badge image URL. * @return {string|null} The URL of the badge image for the emote's channel or `null` if it doesn't have a channel. */ this.getChannelBadge = function () { var twitchApi = require('./twitch-api'); var channelName = this.getChannelName(); var defaultBadge = '//static-cdn.jtvnw.net/jtv_user_pictures/subscriber-star.png'; // No channel. if (!channelName) { return null; } // Give globals a default badge. if (channelName === '~global') { return '/favicon.ico'; } // Already have one preset. if (channel.badge) { return channel.badge; } // Check storage. channel.badge = storage.badges.get(channelName); if (channel.badge !== null) { return channel.badge; } // Set default until API returns something. channel.badge = defaultBadge; // Get from API. logger.debug('Getting fresh badge for: ' + channelName); twitchApi.getBadges(channelName, function (badges) { var badge = null; // Save turbo badge while we are here. if (badges.turbo && badges.turbo.image) { badge = badges.turbo.image; storage.badges.set('turbo', badge, 86400000); // Turbo is actually what we wanted, so we are done. if (channelName === 'turbo') { channel.badge = badge; return; } } // Save turbo badge while we are here. if (badges.premium && badges.premium.image) { badge = badges.premium.image; storage.badges.set('twitch_prime', badge, 86400000); // Turbo is actually what we wanted, so we are done. if (channelName === 'twitch_prime') { channel.badge = badge; return; } } // Save subscriber badge in storage. if (badges.subscriber && badges.subscriber.image) { channel.badge = badges.subscriber.image; storage.badges.set(channelName, channel.badge, 86400000); ui.updateEmotes(); } // No subscriber badge. else { channel.badge = defaultBadge; logger.debug('Failed to get subscriber badge for: ' + channelName); } }); return channel.badge || defaultBadge; }; /** * Sets the emote's channel badge image URL. * @param {string} theBadge The badge image URL to set. */ this.setChannelBadge = function (theBadge) { if (typeof theBadge !== 'string' || theBadge.length < 1) { throw new Error('Invalid badge'); } channel.badge = theBadge; }; /** * Get a channel's display name. * @return {string} The channel's display name. May be equivalent to the channel the first time the API needs to be called. */ this.getChannelDisplayName = function () { var twitchApi = require('./twitch-api'); var channelName = this.getChannelName(); var self = this; var forcedChannelToDisplayNames = { '~global': 'Global', 'turbo': 'Twitch Turbo', 'twitch_prime': 'Twitch Prime' }; // No channel. if (!channelName) { return ''; } // Forced display name. if (forcedChannelToDisplayNames[channelName]) { return forcedChannelToDisplayNames[channelName]; } // Already have one preset. if (channel.displayName) { return channel.displayName; } // Look for obvious bad channel names that shouldn't hit the API or storage. Use channel name instead. if (/[^a-z0-9_]/.test(channelName)) { logger.debug('Unable to get display name due to obvious non-standard channel name for: ' + channelName); return channelName; } // Check storage. channel.displayName = storage.displayNames.get(channelName); if (channel.displayName !== null) { return channel.displayName; } // Get from API. else { // Set default until API returns something. channel.displayName = channelName; logger.debug('Getting fresh display name for: ' + channelName); twitchApi.getUser(channelName, function (user) { if (!user || !user.display_name) { logger.debug('Failed to get display name for: ' + channelName); return; } // Save it. self.setChannelDisplayName(user.display_name); ui.updateEmotes(); }); } return channel.displayName; }; /** * Sets the emote's channel badge image URL. * @param {string} theBadge The badge image URL to set. */ this.setChannelDisplayName = function (displayName) { if (typeof displayName !== 'string' || displayName.length < 1) { throw new Error('Invalid displayName'); } channel.displayName = displayName; storage.displayNames.set(this.getChannelName(), displayName, 86400000); }; /** * Initialize the details. */ // Required fields. this.setText(details.text); this.setUrl(details.url); // Optional fields. if (details.getterName) { this.setGetterName(details.getterName); } if (details.channel) { this.setChannelName(details.channel); } if (details.channelDisplayName) { this.setChannelDisplayName(details.channelDisplayName); } if (details.badge) { this.setChannelBadge(details.badge); } }; /** * State changers. */ /** * Toggle whether an emote should be a favorite. * @param {boolean} [force] `true` forces the emote to be a favorite, `false` forces the emote to not be a favorite. */ Emote.prototype.toggleFavorite = function (force) { if (typeof force !== 'undefined') { storage.starred.set(this.getText(), !!force); return; } storage.starred.set(this.getText(), !this.isFavorite()); }; /** * Toggle whether an emote should be visible out of editing mode. * @param {boolean} [force] `true` forces the emote to be visible, `false` forces the emote to be hidden. */ Emote.prototype.toggleVisibility = function (force) { if (typeof force !== 'undefined') { storage.visibility.set(this.getText(), !!force); return; } storage.visibility.set(this.getText(), !this.isVisible()); }; /** * State getters. */ /** * Whether the emote is from a 3rd party. * @return {boolean} Whether the emote is from a 3rd party. */ Emote.prototype.isThirdParty = function () { return !!this.getGetterName(); }; /** * Whether the emote was favorited. * @return {boolean} Whether the emote was favorited. */ Emote.prototype.isFavorite = function () { return storage.starred.get(this.getText(), false); }; /** * Whether the emote is visible outside of editing mode. * @return {boolean} Whether the emote is visible outside of editing mode. */ Emote.prototype.isVisible = function () { return storage.visibility.get(this.getText(), true); }; /** * Whether the emote is considered a simple smiley. * @return {boolean} Whether the emote is considered a simple smiley. */ Emote.prototype.isSmiley = function () { // The basic smiley emotes. var emotes = [':(', ':)', ':/', ':\\', ':D', ':o', ':p', ':z', ';)', ';p', '<3', '>(', 'B)', 'R)', 'o_o', 'O_O', '#/', ':7', ':>', ':S', '<]']; return emotes.indexOf(this.getText()) !== -1; }; /** * Property getters/setters. */ /** * Gets the usable emote text from a regex. */ function getEmoteFromRegEx(regex) { if (typeof regex === 'string') { regex = new RegExp(regex); } if (!regex) { throw new Error('`regex` must be a RegExp string or object.'); } return decodeURI(regex.source) // Replace HTML entity brackets with actual brackets. .replace('>\\;', '>') .replace('<\\;', '<') // Remove negative groups. // // / // \(\?! // (?! // [^)]* // any amount of characters that are not ) // \) // ) // /g .replace(/\(\?![^)]*\)/g, '') // Pick first option from a group. // // / // \( // ( // ([^|])* // any amount of characters that are not | // \|? // an optional | character // [^)]* // any amount of characters that are not ) // \) // ) // /g .replace(/\(([^|])*\|?[^)]*\)/g, '$1') // Pick first character from a character group. // // / // \[ // [ // ([^|\]\[])* // any amount of characters that are not |, [, or ] // \|? // an optional | character // [^\]]* // any amount of characters that are not [, or ] // \] // ] // /g .replace(/\[([^|\]\[])*\|?[^\]\[]*\]/g, '$1') // Remove optional characters. // // / // [^\\] // any character that is not \ // \? // ? // /g .replace(/[^\\]\?/g, '') // Remove boundaries at beginning and end. .replace(/^\\b|\\b$/g, '') // Unescape only single backslash, not multiple. // // / // \\ // \ // (?!\\) // look-ahead, any character that isn't \ // /g .replace(/\\(?!\\)/g, ''); } var sorting = {}; /** * Sort by alphanumeric in this order: symbols -> numbers -> AaBb... -> numbers */ sorting.byText = function (a, b) { textA = a.getText().toLowerCase(); textB = b.getText().toLowerCase(); if (textA < textB) { return -1; } if (textA > textB) { return 1; } return 0; } /** * Basic smilies before non-basic smilies. */ sorting.bySmiley = function (a, b) { if (a.isSmiley() && !b.isSmiley()) { return -1; } if (b.isSmiley() && !a.isSmiley()) { return 1; } return 0; }; /** * Globals before subscription emotes, subscriptions in alphabetical order. */ sorting.byChannelName = function (a, b) { var channelA = a.getChannelName(); var channelB = b.getChannelName(); // Both don't have channels. if (!channelA && !channelB) { return 0; } // "A" has channel, "B" doesn't. if (channelA && !channelB) { return 1; } // "B" has channel, "A" doesn't. if (channelB && !channelA) { return -1; } channelA = channelA.toLowerCase(); channelB = channelB.toLowerCase(); if (channelA < channelB) { return -1; } if (channelB > channelA) { return 1; } // All the same return 0; }; /** * The general sort order for the all emotes category. * Smileys -> Channel grouping -> alphanumeric */ sorting.allEmotesCategory = function (a, b) { var bySmiley = sorting.bySmiley(a, b); var byChannelName = sorting.byChannelName(a, b); var byText = sorting.byText(a, b); if (bySmiley !== 0) { return bySmiley; } if (byChannelName !== 0) { return byChannelName; } return byText; }; module.exports = emoteStore; },{"./logger":10,"./storage":12,"./twitch-api":14,"./ui":15}],10:[function(require,module,exports){ var api = {}; var instance = '[instance ' + (Math.floor(Math.random() * (999 - 100)) + 100) + '] '; var prefix = '[Emote Menu] '; var storage = require('./storage'); api.log = function () { if (typeof console.log === 'undefined') { return; } arguments = [].slice.call(arguments).map(function (arg) { if (typeof arg !== 'string') { return JSON.stringify(arg); } return arg; }); if (storage.global.get('debugMessagesEnabled', false)) { arguments.unshift(instance); } arguments.unshift(prefix); console.log.apply(console, arguments); }; api.debug = function () { if (!storage.global.get('debugMessagesEnabled', false)) { return; } arguments = [].slice.call(arguments); arguments.unshift('[DEBUG] '); api.log.apply(null, arguments); } module.exports = api; },{"./storage":12}],11:[function(require,module,exports){ var storage = require('./storage'); var logger = require('./logger'); var emotes = require('./emotes'); var api = {}; api.toggleDebug = function (forced) { if (typeof forced === 'undefined') { forced = !storage.global.get('debugMessagesEnabled', false); } else { forced = !!forced; } storage.global.set('debugMessagesEnabled', forced); logger.log('Debug messages are now ' + (forced ? 'enabled' : 'disabled')); }; api.registerEmoteGetter = emotes.registerGetter; api.deregisterEmoteGetter = emotes.deregisterGetter; module.exports = api; },{"./emotes":9,"./logger":10,"./storage":12}],12:[function(require,module,exports){ var Store = require('storage-wrapper'); var storage = {}; // General storage. storage.global = new Store({ namespace: 'emote-menu-for-twitch' }); // Emote visibility storage. storage.visibility = storage.global.createSubstore('visibility'); // Emote starred storage. storage.starred = storage.global.createSubstore('starred'); // Display name storage. storage.displayNames = storage.global.createSubstore('displayNames'); // Channel name storage. storage.channelNames = storage.global.createSubstore('channelNames'); // Badges storage. storage.badges = storage.global.createSubstore('badges'); module.exports = storage; },{"storage-wrapper":6}],13:[function(require,module,exports){ var templates = require('../../build/templates'); module.exports = (function () { var data = {}; var key = null; // Convert templates to their shorter "render" form. for (key in templates) { if (!templates.hasOwnProperty(key)) { continue; } data[key] = render(key); } // Shortcut the render function. All templates will be passed in as partials by default. function render(template) { template = templates[template]; return function (context, partials, indent) { return template.render(context, partials || templates, indent); }; } return data; })(); },{"../../build/templates":3}],14:[function(require,module,exports){ var twitchApi = window.Twitch.api; var jQuery = window.jQuery; var logger = require('./logger'); var api = {}; api.getBadges = function (username, callback) { if ( [ '~global', 'turbo', 'twitch_prime' ].indexOf(username) > -1 ) { if (!jQuery) { callback({}); } // Note: not a documented API endpoint. jQuery.getJSON('https://badges.twitch.tv/v1/badges/global/display') .done(function (api) { var badges = { turbo: { image: api.badge_sets.turbo.versions['1'].image_url_1x }, premium: { image: api.badge_sets.premium.versions['1'].image_url_1x } }; callback(badges); }) .fail(function () { callback({}); }); } else { twitchApi.get('chat/' + username + '/badges') .done(function (api) { callback(api); }) .fail(function () { callback({}); }); } }; api.getUser = function (username, callback) { // Note: not a documented API endpoint. twitchApi.get('users/' + username) .done(function (api) { callback(api); }) .fail(function () { callback({}); }); }; api.getTickets = function (callback) { // Note: not a documented API endpoint. twitchApi.get( '/api/users/:login/tickets', { offset: 0, limit: 100, unended: true } ) .done(function (api) { callback(api.tickets || []); }) .fail(function () { callback([]); }); }; api.getEmotes = function (callback) { twitchApi.get('users/:login/emotes') .done(function (response) { if (!response || !response.emoticon_sets) { logger.debug('getEmotes emoticon_sets empty'); callback({}); return; } callback(response.emoticon_sets); }) .fail(function () { logger.debug('getEmotes API call failed'); callback({}); }); }; module.exports = api; },{"./logger":10}],15:[function(require,module,exports){ var api = {}; var $ = jQuery = window.jQuery; var templates = require('./templates'); var storage = require('./storage'); var emotes = require('./emotes'); var logger = require('./logger'); var theMenu = new UIMenu(); var theMenuButton = new UIMenuButton(); api.init = function () { // Load CSS. require('../../build/styles'); // Load jQuery plugins. require('../plugins/resizable'); require('jquery.scrollbar'); theMenuButton.init(); theMenu.init(); }; api.hideMenu = function () { if (theMenu.dom && theMenu.dom.length) { theMenu.toggleDisplay(false); } }; api.updateEmotes = function () { theMenu.updateEmotes(); } function UIMenuButton() { this.dom = null; } UIMenuButton.prototype.init = function (timesFailed) { var self = this; var chatButton = $('.send-chat-button, .chat-buttons-container button'); var failCounter = timesFailed || 0; this.dom = $('#emote-menu-button'); // Element already exists. if (this.dom.length) { logger.debug('MenuButton already exists, stopping init.'); return this; } if (!chatButton.length) { failCounter += 1; if (failCounter === 1) { logger.log('MenuButton container missing, trying again.'); } if (failCounter >= 10) { logger.log('MenuButton container missing, MenuButton unable to be added, stopping init.'); return this; } setTimeout(function () { self.init(failCounter); }, 1000); return this; } // Create element. this.dom = $(templates.emoteButton()); this.dom.insertBefore(chatButton); // Hide then fade it in. this.dom.hide(); this.dom.fadeIn(); // Enable clicking. this.dom.on('click', function () { theMenu.toggleDisplay(); }); return this; }; UIMenuButton.prototype.toggleDisplay = function (forced) { var state = typeof forced !== 'undefined' ? !!forced : !this.isVisible(); if (state) { this.dom.addClass('active'); return this; } this.dom.removeClass('active'); return this; }; UIMenuButton.prototype.isVisible = function () { return this.dom.hasClass('active'); }; function UIMenu() { this.dom = null; this.groups = {}; this.emotes = []; this.offset = null; this.favorites = null; } UIMenu.prototype.init = function () { var logger = require('./logger'); var self = this; this.dom = $('#emote-menu-for-twitch'); // Element already exists. if (this.dom.length) { return this; } // Create element. this.dom = $(templates.menu()); $(document.body).append(this.dom); this.favorites = new UIFavoritesGroup(); // Enable dragging. this.dom.draggable({ handle: '.draggable', start: function () { self.togglePinned(true); self.toggleMovement(true); }, stop: function () { self.offset = self.dom.offset(); }, containment: $(document.body) }); // Enable resizing. this.dom.resizable({ handle: '[data-command="resize-handle"]', stop: function () { self.togglePinned(true); self.toggleMovement(true); }, alsoResize: self.dom.find('.scrollable'), containment: $(document.body), minHeight: 180, minWidth: 200 }); // Enable pinning. this.dom.find('[data-command="toggle-pinned"]').on('click', function () { self.togglePinned(); }); // Enable editing. this.dom.find('[data-command="toggle-editing"]').on('click', function () { self.toggleEditing(); }); this.dom.find('.scrollable').scrollbar() this.updateEmotes(); return this; }; UIMenu.prototype._detectOutsideClick = function (event) { // Not outside of the menu, ignore the click. if ($(event.target).is('#emote-menu-for-twitch, #emote-menu-for-twitch *')) { return; } // Clicked on the menu button, just remove the listener and let the normal listener handle it. if (!this.isVisible() || $(event.target).is('#emote-menu-button, #emote-menu-button *')) { $(document).off('mouseup', this._detectOutsideClick.bind(this)); return; } // Clicked outside, make sure the menu isn't pinned. if (!this.isPinned()) { // Menu wasn't pinned, remove listener. $(document).off('mouseup', this._detectOutsideClick.bind(this)); this.toggleDisplay(); } }; UIMenu.prototype.toggleDisplay = function (forced) { var state = typeof forced !== 'undefined' ? !!forced : !this.isVisible(); var loggedIn = window.Twitch && window.Twitch.user.isLoggedIn(); // Menu should be shown. if (state) { // Check if user is logged in. if (!loggedIn) { // Call native login form. $.login(); return this; } this.updateEmotes(); this.dom.show(); // Menu moved, move it back. if (this.hasMoved()) { this.dom.offset(this.offset); } // Never moved, make it the same size as the chat window. else { var chatContainer = $('.chat-messages'); // Adjust the size to be the same as the chat container. this.dom.height(chatContainer.outerHeight() - (this.dom.outerHeight() - this.dom.height())); this.dom.width(chatContainer.outerWidth() - (this.dom.outerWidth() - this.dom.width())); // Adjust the offset to be the same as the chat container. this.offset = chatContainer.offset(); this.dom.offset(this.offset); } // Listen for outside click. $(document).on('mouseup', this._detectOutsideClick.bind(this)); } // Menu should be hidden. else { this.dom.hide(); this.toggleEditing(false); this.togglePinned(false); } // Also toggle the menu button. theMenuButton.toggleDisplay(this.isVisible()); return this; }; UIMenu.prototype.isVisible = function () { return this.dom.is(':visible'); }; UIMenu.prototype.updateEmotes = function (which) { var emote = which ? this.getEmote(which) : null; var favoriteEmote = emote ? this.favorites.getEmote(which) : null; if (emote) { emote.update(); if (favoriteEmote) { favoriteEmote.update(); } return this; } var emotes = require('./emotes'); var theEmotes = emotes.getEmotes(); var theEmotesKeys = []; var self = this; theEmotes.forEach(function (emoteInstance) { self.addEmote(emoteInstance); theEmotesKeys.push(emoteInstance.getText()); }); // Difference the emotes and remove all non-valid emotes. this.emotes.forEach(function (oldEmote) { var text = oldEmote.getText() if (theEmotesKeys.indexOf(text) < 0) { logger.debug('Emote difference found, removing emote from UI: ' + text); self.removeEmote(text); } }); // Save the emotes for next differencing. this.emotes = theEmotes; //Update groups. Object.keys(this.groups).forEach(function (group) { self.getGroup(group).init(); }); return this; }; UIMenu.prototype.toggleEditing = function (forced) { var state = typeof forced !== 'undefined' ? !!forced : !this.isEditing(); this.dom.toggleClass('editing', state); return this; }; UIMenu.prototype.isEditing = function () { return this.dom.hasClass('editing'); }; UIMenu.prototype.togglePinned = function (forced) { var state = typeof forced !== 'undefined' ? !!forced : !this.isPinned(); this.dom.toggleClass('pinned', state); return this; }; UIMenu.prototype.isPinned = function () { return this.dom.hasClass('pinned'); }; UIMenu.prototype.toggleMovement = function (forced) { var state = typeof forced !== 'undefined' ? !!forced : !this.hasMoved(); this.dom.toggleClass('moved', state); return this; }; UIMenu.prototype.hasMoved = function () { return this.dom.hasClass('moved'); }; UIMenu.prototype.addGroup = function (emoteInstance) { var channel = emoteInstance.getChannelName(); var self = this; // Already added, don't add again. if (this.getGroup(channel)) { return this; } // Add to current menu groups. var group = new UIGroup(emoteInstance); this.groups[channel] = group; // Sort group names, get index of where this group should go. var keys = Object.keys(this.groups); keys.sort(function (a, b) { // Get the instances. a = self.groups[a].emoteInstance; b = self.groups[b].emoteInstance; // Get the channel name. var aChannel = a.getChannelName(); var bChannel = b.getChannelName(); // Get the channel display name. a = a.getChannelDisplayName().toLowerCase(); b = b.getChannelDisplayName().toLowerCase(); // Prime goes first, always. if (aChannel === 'twitch_prime' && bChannel !== 'twitch_prime') { return -1; } if (bChannel === 'twitch_prime' && aChannel !== 'twitch_prime') { return 1; } // Turbo goes after Prime, always. if (aChannel === 'turbo' && bChannel !== 'turbo') { return -1; } if (bChannel === 'turbo' && aChannel !== 'turbo') { return 1; } // Global goes after Turbo, always. if (aChannel === '~global' && bChannel !== '~global') { return -1; } if (bChannel === '~global' && aChannel !== '~global') { return 1; } // A goes first. if (a < b) { return -1; } // B goest first. if (a > b) { return 1; } // Both the same, doesn't matter. return 0; }); var index = keys.indexOf(channel); // First in the sort, place at the beginning of the menu. if (index === 0) { group.dom.prependTo(this.dom.find('#all-emotes-group')); } // Insert after the previous group in the sort. else { group.dom.insertAfter(this.getGroup(keys[index - 1]).dom); } return group; }; UIMenu.prototype.getGroup = function (name) { return this.groups[name] || null; }; UIMenu.prototype.addEmote = function (emoteInstance) { // Get the group, or add if needed. var group = this.getGroup(emoteInstance.getChannelName()) || this.addGroup(emoteInstance); group.addEmote(emoteInstance); group.toggleDisplay(group.isVisible(), true); this.favorites.addEmote(emoteInstance); return this; }; UIMenu.prototype.removeEmote = function (name) { var self = this; Object.keys(this.groups).forEach(function (groupName) { self.groups[groupName].removeEmote(name); }); this.favorites.removeEmote(name); return this; }; UIMenu.prototype.getEmote = function (name) { var groupName = null; var group = null; var emote = null; for (groupName in this.groups) { group = this.groups[groupName]; emote = group.getEmote(name); if (emote) { return emote; } } return null; }; function UIGroup(emoteInstance) { this.dom = null; this.emotes = {}; this.emoteInstance = emoteInstance; this.init(); } UIGroup.prototype.init = function () { var self = this; var emoteInstance = this.emoteInstance; // First init, create new DOM. if (this.dom === null) { this.dom = $(templates.emoteGroupHeader({ badge: emoteInstance.getChannelBadge(), channel: emoteInstance.getChannelName(), channelDisplayName: emoteInstance.getChannelDisplayName() })); } // Update DOM instead. else { this.dom.find('.header-info').replaceWith( $(templates.emoteGroupHeader({ badge: emoteInstance.getChannelBadge(), channel: emoteInstance.getChannelName(), channelDisplayName: emoteInstance.getChannelDisplayName() })) .find('.header-info') ); } // Enable emote hiding. this.dom.find('.header-info [data-command="toggle-visibility"]').on('click', function () { if (!theMenu.isEditing()) { return; } self.toggleDisplay(); }); this.toggleDisplay(this.isVisible(), true); }; UIGroup.prototype.toggleDisplay = function (forced, skipUpdatingEmoteDisplay) { var self = this; var state = typeof forced !== 'undefined' ? !forced : this.isVisible(); this.dom.toggleClass('emote-menu-hidden', state); // Update the display of all emotes. if (!skipUpdatingEmoteDisplay) { Object.keys(this.emotes).forEach(function (emoteName) { self.emotes[emoteName].toggleDisplay(!state); theMenu.updateEmotes(self.emotes[emoteName].instance.getText()); }); } return this; }; UIGroup.prototype.isVisible = function () { var self = this; // If any emote is visible, the group should be visible. return Object.keys(this.emotes).some(function (emoteName) { return self.emotes[emoteName].isVisible(); }); }; UIGroup.prototype.addEmote = function (emoteInstance) { var self = this; var emote = this.getEmote(emoteInstance.getText()); // Already added, update instead. if (emote) { emote.update(); return this; } // Add to current emotes. emote = new UIEmote(emoteInstance); this.emotes[emoteInstance.getText()] = emote; var keys = Object.keys(this.emotes); keys.sort(function (a, b) { // Get the emote instances. a = self.emotes[a].instance; b = self.emotes[b].instance; // A is a smiley, B isn't. A goes first. if (a.isSmiley() && !b.isSmiley()) { return -1; } // B is a smiley, A isn't. B goes first. if (b.isSmiley() && !a.isSmiley()) { return 1; } // Get the text of the emotes. a = a.getText().toLowerCase(); b = b.getText().toLowerCase(); // A goes first. if (a < b) { return -1; } // B goest first. if (a > b) { return 1; } // Both the same, doesn't matter. return 0; }); var index = keys.indexOf(emoteInstance.getText()); // First in the sort, place at the beginning of the group. if (index === 0) { emote.dom.prependTo(this.dom.find('.emote-container')); } // Insert after the previous emote in the sort. else { emote.dom.insertAfter(this.getEmote(keys[index - 1]).dom); } return this; }; UIGroup.prototype.getEmote = function (name) { return this.emotes[name] || null; }; UIGroup.prototype.removeEmote = function (name) { var emote = this.getEmote(name); if (!emote) { return this; } emote.dom.remove(); delete this.emotes[name]; return this; }; function UIFavoritesGroup() { this.dom = $('#starred-emotes-group'); this.emotes = {}; } UIFavoritesGroup.prototype.addEmote = UIGroup.prototype.addEmote; UIFavoritesGroup.prototype.getEmote = UIGroup.prototype.getEmote; UIFavoritesGroup.prototype.removeEmote = UIGroup.prototype.removeEmote; function UIEmote(emoteInstance) { this.dom = null; this.instance = emoteInstance; this.init(); } UIEmote.prototype.init = function () { var self = this; // Create element. this.dom = $(templates.emote({ url: this.instance.getUrl(), text: this.instance.getText(), thirdParty: this.instance.isThirdParty(), isVisible: this.instance.isVisible(), isStarred: this.instance.isFavorite() })); // Enable clicking. this.dom.on('click', function () { if (!theMenu.isEditing()) { self.addToChat(); // Close the menu if not pinned. if (!theMenu.isPinned()) { theMenu.toggleDisplay(); } } }); // Enable emote hiding. this.dom.find('[data-command="toggle-visibility"]').on('click', function () { if (!theMenu.isEditing()) { return; } self.toggleDisplay(); theMenu.updateEmotes(self.instance.getText()); }); // Enable emote favoriting. this.dom.find('[data-command="toggle-starred"]').on('click', function () { if (!theMenu.isEditing()) { return; } self.toggleFavorite(); theMenu.updateEmotes(self.instance.getText()); }); return this; }; UIEmote.prototype.toggleDisplay = function (forced, skipInstanceUpdate) { var state = typeof forced !== 'undefined' ? !forced : this.isVisible(); this.dom.toggleClass('emote-menu-hidden', state); if (!skipInstanceUpdate) { this.instance.toggleVisibility(!state); } var group = this.getGroup(); group.toggleDisplay(group.isVisible(), true); return this; }; UIEmote.prototype.isVisible = function () { return !this.dom.hasClass('emote-menu-hidden'); }; UIEmote.prototype.toggleFavorite = function (forced, skipInstanceUpdate) { var state = typeof forced !== 'undefined' ? !!forced : !this.isFavorite(); this.dom.toggleClass('emote-menu-starred', state); if (!skipInstanceUpdate) { this.instance.toggleFavorite(state); } return this; }; UIEmote.prototype.isFavorite = function () { return this.dom.hasClass('emote-menu-starred'); }; UIEmote.prototype.addToChat = function () { var ember = require('./ember-api'); // Get textarea element. var element = $('.chat-interface textarea').get(0); var text = this.instance.getText(); // Insert at cursor / replace selection. // https://developer.mozilla.org/en-US/docs/Code_snippets/Miscellaneous var selectionEnd = element.selectionStart + text.length; var currentValue = element.value; var beforeText = currentValue.substring(0, element.selectionStart); var afterText = currentValue.substring(element.selectionEnd, currentValue.length); // Smart padding, only put space at start if needed. if ( beforeText !== '' && beforeText.substr(-1) !== ' ' ) { text = ' ' + text; } // Always put space at end. text = beforeText + text + ' ' + afterText; // Set the text. ember.get('controller:chat', 'currentRoom').set('messageToSend', text); element.focus(); // Put cursor at end. selectionEnd = element.selectionStart + text.length; element.setSelectionRange(selectionEnd, selectionEnd); return this; }; UIEmote.prototype.getGroup = function () { return theMenu.getGroup(this.instance.getChannelName()); }; UIEmote.prototype.update = function () { this.toggleDisplay(this.instance.isVisible(), true); this.toggleFavorite(this.instance.isFavorite(), true); }; module.exports = api; },{"../../build/styles":2,"../plugins/resizable":16,"./ember-api":8,"./emotes":9,"./logger":10,"./storage":12,"./templates":13,"jquery.scrollbar":5}],16:[function(require,module,exports){ (function ($) { $.fn.resizable = function (options) { var settings = $.extend({ alsoResize: null, alsoResizeType: 'both', // `height`, `width`, `both` containment: null, create: null, destroy: null, handle: '.resize-handle', maxHeight: 9999, maxWidth: 9999, minHeight: 0, minWidth: 0, resize: null, resizeOnce: null, snapSize: 1, start: null, stop: null }, options); settings.element = $(this); function recalculateSize(evt) { var data = evt.data, resized = {}; data.diffX = Math.round((evt.pageX - data.pageX) / settings.snapSize) * settings.snapSize; data.diffY = Math.round((evt.pageY - data.pageY) / settings.snapSize) * settings.snapSize; if (Math.abs(data.diffX) > 0 || Math.abs(data.diffY) > 0) { if ( settings.element.height() !== data.height + data.diffY && data.height + data.diffY >= settings.minHeight && data.height + data.diffY <= settings.maxHeight && (settings.containment ? data.outerHeight + data.diffY + data.offset.top <= settings.containment.offset().top + settings.containment.outerHeight() : true) ) { settings.element.height(data.height + data.diffY); resized.height = true; } if ( settings.element.width() !== data.width + data.diffX && data.width + data.diffX >= settings.minWidth && data.width + data.diffX <= settings.maxWidth && (settings.containment ? data.outerWidth + data.diffX + data.offset.left <= settings.containment.offset().left + settings.containment.outerWidth() : true) ) { settings.element.width(data.width + data.diffX); resized.width = true; } if (resized.height || resized.width) { if (settings.resizeOnce) { settings.resizeOnce.bind(settings.element)(evt.data); settings.resizeOnce = null; } if (settings.resize) { settings.resize.bind(settings.element)(evt.data); } if (settings.alsoResize) { if (resized.height && (settings.alsoResizeType === 'height' || settings.alsoResizeType === 'both')) { settings.alsoResize.height(data.alsoResizeHeight + data.diffY); } if (resized.width && (settings.alsoResizeType === 'width' || settings.alsoResizeType === 'both')) { settings.alsoResize.width(data.alsoResizeWidth + data.diffX); } } } } } function start(evt) { evt.preventDefault(); if (settings.start) { settings.start.bind(settings.element)(); } var data = { alsoResizeHeight: settings.alsoResize ? settings.alsoResize.height() : 0, alsoResizeWidth: settings.alsoResize ? settings.alsoResize.width() : 0, height: settings.element.height(), offset: settings.element.offset(), outerHeight: settings.element.outerHeight(), outerWidth: settings.element.outerWidth(), pageX: evt.pageX, pageY: evt.pageY, width: settings.element.width() }; $(document).on('mousemove', '*', data, recalculateSize); $(document).on('mouseup', '*', stop); } function stop() { if (settings.stop) { settings.stop.bind(settings.element)(); } $(document).off('mousemove', '*', recalculateSize); $(document).off('mouseup', '*', stop); } if (settings.handle) { if (settings.alsoResize && ['both', 'height', 'width'].indexOf(settings.alsoResizeType) >= 0) { settings.alsoResize = $(settings.alsoResize); } if (settings.containment) { settings.containment = $(settings.containment); } settings.handle = $(settings.handle); settings.snapSize = settings.snapSize < 1 ? 1 : settings.snapSize; if (options === 'destroy') { settings.handle.off('mousedown', start); if (settings.destroy) { settings.destroy.bind(this)(); } return this; } settings.handle.on('mousedown', start); if (settings.create) { settings.create.bind(this)(); } } return this; }; })(jQuery); },{}]},{},[1]) //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["node_modules\\browserify\\node_modules\\browser-pack\\_prelude.js","./src/script.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/build/styles.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/build/templates.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/node_modules/hogan.js/lib/template.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/node_modules/jquery.scrollbar/jquery.scrollbar.min.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/node_modules/storage-wrapper/index.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/package.json","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/src/modules/ember-api.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/src/modules/emotes.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/src/modules/logger.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/src/modules/public-api.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/src/modules/storage.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/src/modules/templates.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/src/modules/twitch-api.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/src/modules/ui.js","C:/Users/Cletus/Projects/Userscript--Twitch-Chat-Emotes/src/plugins/resizable.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACTA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrVA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACbA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvbA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1GA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/uBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7rBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","var pkg = require('../package.json');\r\nvar publicApi = require('./modules/public-api');\r\nvar ember = require('./modules/ember-api');\r\nvar logger = require('./modules/logger');\r\nvar emotes = require('./modules/emotes');\r\nvar ui = require('./modules/ui');\r\n\r\nlogger.log('(v'+ pkg.version + ') Initial load on ' + location.href);\r\n\r\n// Only enable script if we have the right variables.\r\n//---------------------------------------------------\r\nvar initTimer = 0;\r\n(function init(time) {\t\r\n\tif (!time) {\r\n\t\ttime = 0;\r\n\t}\r\n\r\n\tvar objectsLoaded = (\r\n\t\twindow.Twitch !== undefined &&\r\n\t\twindow.jQuery !== undefined &&\r\n\t\tember.isLoaded()\r\n\t);\r\n\tif (!objectsLoaded) {\r\n\t\t// Stops trying after 10 minutes.\r\n\t\tif (initTimer >= 600000) {\r\n\t\t\tlogger.log('Taking too long to load, stopping. Refresh the page to try again. (' + initTimer + 'ms)');\r\n\t\t\treturn;\r\n\t\t}\r\n\r\n\t\t// Give an update every 10 seconds.\r\n\t\tif (initTimer % 10000) {\r\n\t\t\tlogger.debug('Still waiting for objects to load. (' + initTimer + 'ms)');\r\n\t\t}\r\n\r\n\t\t// Bump time up after 1s to reduce possible lag.\r\n\t\ttime = time >= 1000 ? 1000 : time + 25;\r\n\t\tinitTimer += time;\r\n\r\n\t\tsetTimeout(init, time, time);\r\n\t\treturn;\r\n\t}\r\n\t\r\n\t// Expose public api.\r\n\tif (typeof window.emoteMenu === 'undefined') {\r\n\t\twindow.emoteMenu = publicApi;\r\n\t}\r\n\r\n\tember.hook('route:channel', activate, deactivate);\r\n\tember.hook('route:chat', activate, deactivate);\r\n\r\n\tactivate();\r\n})();\r\n\r\nfunction activate() {\r\n\tui.init();\r\n\temotes.init();\r\n}\r\nfunction deactivate() {\r\n\tui.hideMenu();\r\n}\r\n","(function (doc, cssText) {\n    var id = \"emote-menu-for-twitch-styles\";\n    var styleEl = doc.getElementById(id);\n    if (!styleEl) {\n        styleEl = doc.createElement(\"style\");\n        styleEl.id = id;\n        doc.getElementsByTagName(\"head\")[0].appendChild(styleEl);\n    }\n    if (styleEl.styleSheet) {\n        if (!styleEl.styleSheet.disabled) {\n            styleEl.styleSheet.cssText = cssText;\n        }\n    } else {\n        try {\n            styleEl.innerHTML = cssText;\n        } catch (ignore) {\n            styleEl.innerText = cssText;\n        }\n    }\n}(document, \"/**\\n\" +\n\" * Minified style.\\n\" +\n\" * Original filename: \\\\node_modules\\\\jquery.scrollbar\\\\jquery.scrollbar.css\\n\" +\n\" */\\n\" +\n\".scroll-wrapper{overflow:hidden!important;padding:0!important;position:relative}.scroll-wrapper>.scroll-content{border:none!important;-moz-box-sizing:content-box!important;box-sizing:content-box!important;height:auto;left:0;margin:0;max-height:none!important;max-width:none!important;overflow:scroll!important;padding:0;position:relative!important;top:0;width:auto!important}.scroll-wrapper>.scroll-content::-webkit-scrollbar{height:0;width:0}.scroll-element{display:none}.scroll-element,.scroll-element div{-moz-box-sizing:content-box;box-sizing:content-box}.scroll-element.scroll-x.scroll-scrollx_visible,.scroll-element.scroll-y.scroll-scrolly_visible{display:block}.scroll-element .scroll-arrow,.scroll-element .scroll-bar{cursor:default}.scroll-textarea{border:1px solid #ccc;border-top-color:#999}.scroll-textarea>.scroll-content{overflow:hidden!important}.scroll-textarea>.scroll-content>textarea{border:none!important;-moz-box-sizing:border-box;box-sizing:border-box;height:100%!important;margin:0;max-height:none!important;max-width:none!important;overflow:scroll!important;outline:0;padding:2px;position:relative!important;top:0;width:100%!important}.scroll-textarea>.scroll-content>textarea::-webkit-scrollbar{height:0;width:0}.scrollbar-inner>.scroll-element,.scrollbar-inner>.scroll-element div{border:none;margin:0;padding:0;position:absolute;z-index:10}.scrollbar-inner>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-inner>.scroll-element.scroll-x{bottom:2px;height:8px;left:0;width:100%}.scrollbar-inner>.scroll-element.scroll-y{height:100%;right:2px;top:0;width:8px}.scrollbar-inner>.scroll-element .scroll-element_outer{overflow:hidden}.scrollbar-inner>.scroll-element .scroll-bar,.scrollbar-inner>.scroll-element .scroll-element_outer,.scrollbar-inner>.scroll-element .scroll-element_track{border-radius:8px}.scrollbar-inner>.scroll-element .scroll-bar,.scrollbar-inner>.scroll-element .scroll-element_track{-ms-filter:\\\"progid:DXImageTransform.Microsoft.Alpha(Opacity=40)\\\";filter:alpha(opacity=40);opacity:.4}.scrollbar-inner>.scroll-element .scroll-element_track{background-color:#e0e0e0}.scrollbar-inner>.scroll-element .scroll-bar{background-color:#c2c2c2}.scrollbar-inner>.scroll-element.scroll-draggable .scroll-bar,.scrollbar-inner>.scroll-element:hover .scroll-bar{background-color:#919191}.scrollbar-inner>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_track{left:-12px}.scrollbar-inner>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_track{top:-12px}.scrollbar-inner>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-12px}.scrollbar-inner>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-12px}.scrollbar-outer>.scroll-element,.scrollbar-outer>.scroll-element div{border:none;margin:0;padding:0;position:absolute;z-index:10}.scrollbar-outer>.scroll-element{background-color:#fff}.scrollbar-outer>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-outer>.scroll-element.scroll-x{bottom:0;height:12px;left:0;width:100%}.scrollbar-outer>.scroll-element.scroll-y{height:100%;right:0;top:0;width:12px}.scrollbar-outer>.scroll-element.scroll-x .scroll-element_outer{height:8px;top:2px}.scrollbar-outer>.scroll-element.scroll-y .scroll-element_outer{left:2px;width:8px}.scrollbar-outer>.scroll-element .scroll-element_outer{overflow:hidden}.scrollbar-outer>.scroll-element .scroll-element_track{background-color:#eee}.scrollbar-outer>.scroll-element .scroll-bar,.scrollbar-outer>.scroll-element .scroll-element_outer,.scrollbar-outer>.scroll-element .scroll-element_track{border-radius:8px}.scrollbar-outer>.scroll-element .scroll-bar{background-color:#d9d9d9}.scrollbar-outer>.scroll-element .scroll-bar:hover{background-color:#c2c2c2}.scrollbar-outer>.scroll-element.scroll-draggable .scroll-bar{background-color:#919191}.scrollbar-outer>.scroll-content.scroll-scrolly_visible{left:-12px;margin-left:12px}.scrollbar-outer>.scroll-content.scroll-scrollx_visible{top:-12px;margin-top:12px}.scrollbar-outer>.scroll-element.scroll-x .scroll-bar{min-width:10px}.scrollbar-outer>.scroll-element.scroll-y .scroll-bar{min-height:10px}.scrollbar-outer>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_track{left:-14px}.scrollbar-outer>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_track{top:-14px}.scrollbar-outer>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-14px}.scrollbar-outer>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-14px}.scrollbar-macosx>.scroll-element,.scrollbar-macosx>.scroll-element div{background:0 0;border:none;margin:0;padding:0;position:absolute;z-index:10}.scrollbar-macosx>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-macosx>.scroll-element .scroll-element_track{display:none}.scrollbar-macosx>.scroll-element .scroll-bar{background-color:#6C6E71;display:block;-ms-filter:\\\"progid:DXImageTransform.Microsoft.Alpha(Opacity=0)\\\";filter:alpha(opacity=0);opacity:0;border-radius:7px;transition:opacity .2s linear}.scrollbar-macosx:hover>.scroll-element .scroll-bar,.scrollbar-macosx>.scroll-element.scroll-draggable .scroll-bar{-ms-filter:\\\"progid:DXImageTransform.Microsoft.Alpha(Opacity=70)\\\";filter:alpha(opacity=70);opacity:.7}.scrollbar-macosx>.scroll-element.scroll-x{bottom:0;height:0;left:0;min-width:100%;overflow:visible;width:100%}.scrollbar-macosx>.scroll-element.scroll-y{height:100%;min-height:100%;right:0;top:0;width:0}.scrollbar-macosx>.scroll-element.scroll-x .scroll-bar{height:7px;min-width:10px;top:-9px}.scrollbar-macosx>.scroll-element.scroll-y .scroll-bar{left:-9px;min-height:10px;width:7px}.scrollbar-macosx>.scroll-element.scroll-x .scroll-element_outer{left:2px}.scrollbar-macosx>.scroll-element.scroll-x .scroll-element_size{left:-4px}.scrollbar-macosx>.scroll-element.scroll-y .scroll-element_outer{top:2px}.scrollbar-macosx>.scroll-element.scroll-y .scroll-element_size{top:-4px}.scrollbar-macosx>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-11px}.scrollbar-macosx>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-11px}.scrollbar-light>.scroll-element,.scrollbar-light>.scroll-element div{border:none;margin:0;overflow:hidden;padding:0;position:absolute;z-index:10}.scrollbar-light>.scroll-element{background-color:#fff}.scrollbar-light>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-light>.scroll-element .scroll-element_outer{border-radius:10px}.scrollbar-light>.scroll-element .scroll-element_size{background:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiA/Pgo8c3ZnIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyIgd2lkdGg9IjEwMCUiIGhlaWdodD0iMTAwJSIgdmlld0JveD0iMCAwIDEgMSIgcHJlc2VydmVBc3BlY3RSYXRpbz0ibm9uZSI+CiAgPGxpbmVhckdyYWRpZW50IGlkPSJncmFkLXVjZ2ctZ2VuZXJhdGVkIiBncmFkaWVudFVuaXRzPSJ1c2VyU3BhY2VPblVzZSIgeDE9IjAlIiB5MT0iMCUiIHgyPSIxMDAlIiB5Mj0iMCUiPgogICAgPHN0b3Agb2Zmc2V0PSIwJSIgc3RvcC1jb2xvcj0iI2RiZGJkYiIgc3RvcC1vcGFjaXR5PSIxIi8+CiAgICA8c3RvcCBvZmZzZXQ9IjEwMCUiIHN0b3AtY29sb3I9IiNlOGU4ZTgiIHN0b3Atb3BhY2l0eT0iMSIvPgogIDwvbGluZWFyR3JhZGllbnQ+CiAgPHJlY3QgeD0iMCIgeT0iMCIgd2lkdGg9IjEiIGhlaWdodD0iMSIgZmlsbD0idXJsKCNncmFkLXVjZ2ctZ2VuZXJhdGVkKSIgLz4KPC9zdmc+);background:linear-gradient(to right,#dbdbdb 0,#e8e8e8 100%);border-radius:10px}.scrollbar-light>.scroll-element.scroll-x{bottom:0;height:17px;left:0;min-width:100%;width:100%}.scrollbar-light>.scroll-element.scroll-y{height:100%;min-height:100%;right:0;top:0;width:17px}.scrollbar-light>.scroll-element .scroll-bar{background:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiA/Pgo8c3ZnIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyIgd2lkdGg9IjEwMCUiIGhlaWdodD0iMTAwJSIgdmlld0JveD0iMCAwIDEgMSIgcHJlc2VydmVBc3BlY3RSYXRpbz0ibm9uZSI+CiAgPGxpbmVhckdyYWRpZW50IGlkPSJncmFkLXVjZ2ctZ2VuZXJhdGVkIiBncmFkaWVudFVuaXRzPSJ1c2VyU3BhY2VPblVzZSIgeDE9IjAlIiB5MT0iMCUiIHgyPSIxMDAlIiB5Mj0iMCUiPgogICAgPHN0b3Agb2Zmc2V0PSIwJSIgc3RvcC1jb2xvcj0iI2ZlZmVmZSIgc3RvcC1vcGFjaXR5PSIxIi8+CiAgICA8c3RvcCBvZmZzZXQ9IjEwMCUiIHN0b3AtY29sb3I9IiNmNWY1ZjUiIHN0b3Atb3BhY2l0eT0iMSIvPgogIDwvbGluZWFyR3JhZGllbnQ+CiAgPHJlY3QgeD0iMCIgeT0iMCIgd2lkdGg9IjEiIGhlaWdodD0iMSIgZmlsbD0idXJsKCNncmFkLXVjZ2ctZ2VuZXJhdGVkKSIgLz4KPC9zdmc+);background:linear-gradient(to right,#fefefe 0,#f5f5f5 100%);border:1px solid #dbdbdb;border-radius:10px}.scrollbar-light>.scroll-content.scroll-scrolly_visible{left:-17px;margin-left:17px}.scrollbar-light>.scroll-content.scroll-scrollx_visible{top:-17px;margin-top:17px}.scrollbar-light>.scroll-element.scroll-x .scroll-bar{height:10px;min-width:10px;top:0}.scrollbar-light>.scroll-element.scroll-y .scroll-bar{left:0;min-height:10px;width:10px}.scrollbar-light>.scroll-element.scroll-x .scroll-element_outer{height:12px;left:2px;top:2px}.scrollbar-light>.scroll-element.scroll-x .scroll-element_size{left:-4px}.scrollbar-light>.scroll-element.scroll-y .scroll-element_outer{left:2px;top:2px;width:12px}.scrollbar-light>.scroll-element.scroll-y .scroll-element_size{top:-4px}.scrollbar-light>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-19px}.scrollbar-light>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-19px}.scrollbar-light>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_track{left:-19px}.scrollbar-light>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_track{top:-19px}.scrollbar-rail>.scroll-element,.scrollbar-rail>.scroll-element div{border:none;margin:0;overflow:hidden;padding:0;position:absolute;z-index:10}.scrollbar-rail>.scroll-element{background-color:#fff}.scrollbar-rail>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-rail>.scroll-element .scroll-element_size{background-color:#999;background-color:rgba(0,0,0,.3)}.scrollbar-rail>.scroll-element .scroll-element_outer:hover .scroll-element_size{background-color:#666;background-color:rgba(0,0,0,.5)}.scrollbar-rail>.scroll-element.scroll-x{bottom:0;height:12px;left:0;min-width:100%;padding:3px 0 2px;width:100%}.scrollbar-rail>.scroll-element.scroll-y{height:100%;min-height:100%;padding:0 2px 0 3px;right:0;top:0;width:12px}.scrollbar-rail>.scroll-element .scroll-bar{background-color:#d0b9a0;border-radius:2px;box-shadow:1px 1px 3px rgba(0,0,0,.5)}.scrollbar-rail>.scroll-element .scroll-element_outer:hover .scroll-bar{box-shadow:1px 1px 3px rgba(0,0,0,.6)}.scrollbar-rail>.scroll-content.scroll-scrolly_visible{left:-17px;margin-left:17px}.scrollbar-rail>.scroll-content.scroll-scrollx_visible{margin-top:17px;top:-17px}.scrollbar-rail>.scroll-element.scroll-x .scroll-bar{height:10px;min-width:10px;top:1px}.scrollbar-rail>.scroll-element.scroll-y .scroll-bar{left:1px;min-height:10px;width:10px}.scrollbar-rail>.scroll-element.scroll-x .scroll-element_outer{height:15px;left:5px}.scrollbar-rail>.scroll-element.scroll-x .scroll-element_size{height:2px;left:-10px;top:5px}.scrollbar-rail>.scroll-element.scroll-y .scroll-element_outer{top:5px;width:15px}.scrollbar-rail>.scroll-element.scroll-y .scroll-element_size{left:5px;top:-10px;width:2px}.scrollbar-rail>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-25px}.scrollbar-rail>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-25px}.scrollbar-rail>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_track{left:-25px}.scrollbar-rail>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_track{top:-25px}.scrollbar-dynamic>.scroll-element,.scrollbar-dynamic>.scroll-element div{background:0 0;border:none;margin:0;padding:0;position:absolute;z-index:10}.scrollbar-dynamic>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-dynamic>.scroll-element.scroll-x{bottom:2px;height:7px;left:0;min-width:100%;width:100%}.scrollbar-dynamic>.scroll-element.scroll-y{height:100%;min-height:100%;right:2px;top:0;width:7px}.scrollbar-dynamic>.scroll-element .scroll-element_outer{opacity:.3;border-radius:12px}.scrollbar-dynamic>.scroll-element .scroll-element_size{background-color:#ccc;opacity:0;border-radius:12px;transition:opacity .2s}.scrollbar-dynamic>.scroll-element .scroll-bar{background-color:#6c6e71;border-radius:7px}.scrollbar-dynamic>.scroll-element.scroll-x .scroll-bar{bottom:0;height:7px;min-width:24px;top:auto}.scrollbar-dynamic>.scroll-element.scroll-y .scroll-bar{left:auto;min-height:24px;right:0;width:7px}.scrollbar-dynamic>.scroll-element.scroll-x .scroll-element_outer{bottom:0;top:auto;left:2px;transition:height .2s}.scrollbar-dynamic>.scroll-element.scroll-y .scroll-element_outer{left:auto;right:0;top:2px;transition:width .2s}.scrollbar-dynamic>.scroll-element.scroll-x .scroll-element_size{left:-4px}.scrollbar-dynamic>.scroll-element.scroll-y .scroll-element_size{top:-4px}.scrollbar-dynamic>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-11px}.scrollbar-dynamic>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-11px}.scrollbar-dynamic>.scroll-element.scroll-draggable .scroll-element_outer,.scrollbar-dynamic>.scroll-element:hover .scroll-element_outer{overflow:hidden;-ms-filter:\\\"progid:DXImageTransform.Microsoft.Alpha(Opacity=70)\\\";filter:alpha(opacity=70);opacity:.7}.scrollbar-dynamic>.scroll-element.scroll-draggable .scroll-element_outer .scroll-element_size,.scrollbar-dynamic>.scroll-element:hover .scroll-element_outer .scroll-element_size{opacity:1}.scrollbar-dynamic>.scroll-element.scroll-draggable .scroll-element_outer .scroll-bar,.scrollbar-dynamic>.scroll-element:hover .scroll-element_outer .scroll-bar{height:100%;width:100%;border-radius:12px}.scrollbar-dynamic>.scroll-element.scroll-x.scroll-draggable .scroll-element_outer,.scrollbar-dynamic>.scroll-element.scroll-x:hover .scroll-element_outer{height:20px;min-height:7px}.scrollbar-dynamic>.scroll-element.scroll-y.scroll-draggable .scroll-element_outer,.scrollbar-dynamic>.scroll-element.scroll-y:hover .scroll-element_outer{min-width:7px;width:20px}.scrollbar-chrome>.scroll-element,.scrollbar-chrome>.scroll-element div{border:none;margin:0;overflow:hidden;padding:0;position:absolute;z-index:10}.scrollbar-chrome>.scroll-element{background-color:#fff}.scrollbar-chrome>.scroll-element div{display:block;height:100%;left:0;top:0;width:100%}.scrollbar-chrome>.scroll-element .scroll-element_track{background:#f1f1f1;border:1px solid #dbdbdb}.scrollbar-chrome>.scroll-element.scroll-x{bottom:0;height:16px;left:0;min-width:100%;width:100%}.scrollbar-chrome>.scroll-element.scroll-y{height:100%;min-height:100%;right:0;top:0;width:16px}.scrollbar-chrome>.scroll-element .scroll-bar{background-color:#d9d9d9;border:1px solid #bdbdbd;cursor:default;border-radius:2px}.scrollbar-chrome>.scroll-element .scroll-bar:hover{background-color:#c2c2c2;border-color:#a9a9a9}.scrollbar-chrome>.scroll-element.scroll-draggable .scroll-bar{background-color:#919191;border-color:#7e7e7e}.scrollbar-chrome>.scroll-content.scroll-scrolly_visible{left:-16px;margin-left:16px}.scrollbar-chrome>.scroll-content.scroll-scrollx_visible{top:-16px;margin-top:16px}.scrollbar-chrome>.scroll-element.scroll-x .scroll-bar{height:8px;min-width:10px;top:3px}.scrollbar-chrome>.scroll-element.scroll-y .scroll-bar{left:3px;min-height:10px;width:8px}.scrollbar-chrome>.scroll-element.scroll-x .scroll-element_outer{border-left:1px solid #dbdbdb}.scrollbar-chrome>.scroll-element.scroll-x .scroll-element_track{height:14px;left:-3px}.scrollbar-chrome>.scroll-element.scroll-x .scroll-element_size{height:14px;left:-4px}.scrollbar-chrome>.scroll-element.scroll-y .scroll-element_outer{border-top:1px solid #dbdbdb}.scrollbar-chrome>.scroll-element.scroll-y .scroll-element_track{top:-3px;width:14px}.scrollbar-chrome>.scroll-element.scroll-y .scroll-element_size{top:-4px;width:14px}.scrollbar-chrome>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_size{left:-19px}.scrollbar-chrome>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_size{top:-19px}.scrollbar-chrome>.scroll-element.scroll-x.scroll-scrolly_visible .scroll-element_track{left:-19px}.scrollbar-chrome>.scroll-element.scroll-y.scroll-scrollx_visible .scroll-element_track{top:-19px}\\n\" +\n\"/**\\n\" +\n\" * Minified style.\\n\" +\n\" * Original filename: \\\\src\\\\styles\\\\style.css\\n\" +\n\" */\\n\" +\n\"@-webkit-keyframes spin{100%{-webkit-transform:rotate(360deg);transform:rotate(360deg)}}@keyframes spin{100%{-webkit-transform:rotate(360deg);transform:rotate(360deg)}}#emote-menu-button{background-image:url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABIAAAAQCAYAAAAbBi9cAAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsMAAA7DAcdvqGQAAAKUSURBVDhPfZTNi1JRGMZvMIsWUZts5SIXFYK0CME/IGghxVC7WUoU1NBixI+mRSD4MQzmxziKO3XUBhRmUGZKdBG40XEGU6d0GFGZcT4qxW1hi7fzvNwZqKwDD5z7vs/vueeee+6VMJxO5wUhhdvtfuHz+T4tLS2NhegfGsMDLxiwHIIhLi57PJ75VCr1Y39/n4bDIY1Go4lCDx54wYCVYzjoVjQa/dxutyfCkwSvYJpgOSQf708tuBa1yWRy/L+V/Cl4wYBFhhTxfLhum/esiiJ1u12KRCJksVhofX2dTk5OzkHMUUMPHnjB2F55VpEhPde/Lbx8FqBEIkHpdJoMBgNptVrS6XRUqVTOg7a3t2lmZob0ej2p1Wr2ggGLDOnJ3QSZH4coHo/TysoKhygUCtJoNFQsFmkwGLAwR7hSqSSVSsVeMGCRIT29F6fXJi8Xy+Uymc1mmp6eJofDQfV6nU5PT1mY2+127uHxSqUSh4FFhhQLvrvtcrm+YpkHBwdUrVZpa2uLarUadTodOjw8ZGGOGnrwwAsGLDLw1i4uLrzRYeOOj49pb2+Pdnd3qdVq8StGAIQ5ao1Ggz3wggGLDD4C4izcEcWfR0dHbMrlcrSxscGbjVAIK8lms7S5ucmB/X6fXz9YDsEQFzdjsVit2Wzyqc1kMrwfVquVjEYjzc3NkclkIpvNRmtra+yBVzAfBXtDjuGgS8FgcFbc8QvuhjNSKBQoFAqR6LFEn/L5PPfggXd5eXkWrBzDQdC1QCBgFoeut7Ozw/tyBp2FQzhPwtOFFwzY34Yo4A9wRXzdD8LhcE48wncE9no9Fuaoid574bkPLxgZ/3uI5pTQVfFlP/L7/Wmhb7JSXq/3IXrwyHZ5SNIvGCnqyh+J7+gAAAAASUVORK5CYII=)!important;background-position:50%;background-repeat:no-repeat;cursor:pointer;height:30px;width:30px}#emote-menu-button:focus{box-shadow:none}#emote-menu-button.active{box-shadow:0 0 6px 0 #7d5bbe,inset 0 0 0 1px rgba(100,65,164,.5)}.emote-menu{padding:5px;z-index:1000;display:none;background-color:#202020;position:absolute}.emote-menu a{color:#fff}.emote-menu a:hover{cursor:pointer;text-decoration:underline;color:#ccc}.emote-menu .emotes-starred{height:38px}.emote-menu .draggable{background-image:repeating-linear-gradient(45deg,transparent,transparent 5px,rgba(255,255,255,.05) 5px,rgba(255,255,255,.05) 10px);cursor:move;height:7px;margin-bottom:3px}.emote-menu .draggable:hover{background-image:repeating-linear-gradient(45deg,transparent,transparent 5px,rgba(255,255,255,.1) 5px,rgba(255,255,255,.1) 10px)}.emote-menu .header-info{border-top:1px solid #000;box-shadow:0 1px 0 rgba(255,255,255,.05) inset;background-image:linear-gradient(to top,transparent,rgba(0,0,0,.5));padding:2px;color:#ddd;text-align:center;position:relative}.emote-menu .header-info img{margin-right:8px}.emote-menu .emote{display:inline-block;padding:2px;margin:1px;cursor:pointer;border-radius:5px;text-align:center;position:relative;width:30px;height:30px;transition:all .25s ease;border:1px solid transparent}.emote-menu.editing .emote{cursor:auto}.emote-menu .emote img{max-width:100%;max-height:100%;margin:auto;position:absolute;top:0;bottom:0;left:0;right:0}.emote-menu .single-row .emote-container{overflow:hidden;height:37px}.emote-menu .single-row .emote{display:inline-block;margin-bottom:100px}.emote-menu .emote:hover{background-color:rgba(255,255,255,.1)}.emote-menu .pull-left{float:left}.emote-menu .pull-right{float:right}.emote-menu .footer{text-align:center;border-top:1px solid #000;box-shadow:0 1px 0 rgba(255,255,255,.05) inset;padding:5px 0 2px;margin-top:5px;height:18px}.emote-menu .footer .pull-left{margin-right:5px}.emote-menu .footer .pull-right{margin-left:5px}.emote-menu .icon{height:16px;width:16px;opacity:.5;background-size:contain!important}.emote-menu .icon:hover{opacity:1}.emote-menu .icon-home{background:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iNjQiDQogICBoZWlnaHQ9IjY0Ig0KICAgdmlld0JveD0iMCAwIDY0IDY0Ig0KICAgaWQ9IkNhcGFfMSINCiAgIHhtbDpzcGFjZT0icHJlc2VydmUiPjxtZXRhZGF0YQ0KICAgaWQ9Im1ldGFkYXRhMzAwMSI+PHJkZjpSREY+PGNjOldvcmsNCiAgICAgICByZGY6YWJvdXQ9IiI+PGRjOmZvcm1hdD5pbWFnZS9zdmcreG1sPC9kYzpmb3JtYXQ+PGRjOnR5cGUNCiAgICAgICAgIHJkZjpyZXNvdXJjZT0iaHR0cDovL3B1cmwub3JnL2RjL2RjbWl0eXBlL1N0aWxsSW1hZ2UiIC8+PGRjOnRpdGxlPjwvZGM6dGl0bGU+PC9jYzpXb3JrPjwvcmRmOlJERj48L21ldGFkYXRhPjxkZWZzDQogICBpZD0iZGVmczI5OTkiIC8+DQo8cGF0aA0KICAgZD0ibSA1Ny4wNjIsMzEuMzk4IGMgMC45MzIsLTEuMDI1IDAuODQyLC0yLjU5NiAtMC4yMDEsLTMuNTA4IEwgMzMuODg0LDcuNzg1IEMgMzIuODQxLDYuODczIDMxLjE2OSw2Ljg5MiAzMC4xNDgsNy44MjggTCA3LjA5MywyOC45NjIgYyAtMS4wMjEsMC45MzYgLTEuMDcxLDIuNTA1IC0wLjExMSwzLjUwMyBsIDAuNTc4LDAuNjAyIGMgMC45NTksMC45OTggMi41MDksMS4xMTcgMy40NiwwLjI2NSBsIDEuNzIzLC0xLjU0MyB2IDIyLjU5IGMgMCwxLjM4NiAxLjEyMywyLjUwOCAyLjUwOCwyLjUwOCBoIDguOTg3IGMgMS4zODUsMCAyLjUwOCwtMS4xMjIgMi41MDgsLTIuNTA4IFYgMzguNTc1IGggMTEuNDYzIHYgMTUuODA0IGMgLTAuMDIsMS4zODUgMC45NzEsMi41MDcgMi4zNTYsMi41MDcgaCA5LjUyNCBjIDEuMzg1LDAgMi41MDgsLTEuMTIyIDIuNTA4LC0yLjUwOCBWIDMyLjEwNyBjIDAsMCAwLjQ3NiwwLjQxNyAxLjA2MywwLjkzMyAwLjU4NiwwLjUxNSAxLjgxNywwLjEwMiAyLjc0OSwtMC45MjQgbCAwLjY1MywtMC43MTggeiINCiAgIGlkPSJwYXRoMjk5NSINCiAgIHN0eWxlPSJmaWxsOiNmZmZmZmY7ZmlsbC1vcGFjaXR5OjEiIC8+DQo8L3N2Zz4=) 50% no-repeat}.emote-menu .icon-gear{background:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iMjEuNTkiDQogICBoZWlnaHQ9IjIxLjEzNjk5OSINCiAgIHZpZXdCb3g9IjAgMCAyMS41OSAyMS4xMzciDQogICBpZD0iQ2FwYV8xIg0KICAgeG1sOnNwYWNlPSJwcmVzZXJ2ZSI+PG1ldGFkYXRhDQogICBpZD0ibWV0YWRhdGEzOSI+PHJkZjpSREY+PGNjOldvcmsNCiAgICAgICByZGY6YWJvdXQ9IiI+PGRjOmZvcm1hdD5pbWFnZS9zdmcreG1sPC9kYzpmb3JtYXQ+PGRjOnR5cGUNCiAgICAgICAgIHJkZjpyZXNvdXJjZT0iaHR0cDovL3B1cmwub3JnL2RjL2RjbWl0eXBlL1N0aWxsSW1hZ2UiIC8+PGRjOnRpdGxlPjwvZGM6dGl0bGU+PC9jYzpXb3JrPjwvcmRmOlJERj48L21ldGFkYXRhPjxkZWZzDQogICBpZD0iZGVmczM3IiAvPg0KPHBhdGgNCiAgIGQ9Ik0gMTguNjIyLDguMTQ1IDE4LjA3Nyw2Ljg1IGMgMCwwIDEuMjY4LC0yLjg2MSAxLjE1NiwtMi45NzEgTCAxNy41NTQsMi4yNCBDIDE3LjQzOCwyLjEyNyAxNC41NzYsMy40MzMgMTQuNTc2LDMuNDMzIEwgMTMuMjU2LDIuOSBDIDEzLjI1NiwyLjkgMTIuMDksMCAxMS45MywwIEggOS41NjEgQyA5LjM5NiwwIDguMzE3LDIuOTA2IDguMzE3LDIuOTA2IEwgNi45OTksMy40NDEgYyAwLDAgLTIuOTIyLC0xLjI0MiAtMy4wMzQsLTEuMTMxIEwgMi4yODksMy45NTEgQyAyLjE3Myw0LjA2NCAzLjUwNyw2Ljg2NyAzLjUwNyw2Ljg2NyBMIDIuOTYyLDguMTYgQyAyLjk2Miw4LjE2IDAsOS4zMDEgMCw5LjQ1NSB2IDIuMzIyIGMgMCwwLjE2MiAyLjk2OSwxLjIxOSAyLjk2OSwxLjIxOSBsIDAuNTQ1LDEuMjkxIGMgMCwwIC0xLjI2OCwyLjg1OSAtMS4xNTcsMi45NjkgbCAxLjY3OCwxLjY0MyBjIDAuMTE0LDAuMTExIDIuOTc3LC0xLjE5NSAyLjk3NywtMS4xOTUgbCAxLjMyMSwwLjUzNSBjIDAsMCAxLjE2NiwyLjg5OCAxLjMyNywyLjg5OCBoIDIuMzY5IGMgMC4xNjQsMCAxLjI0NCwtMi45MDYgMS4yNDQsLTIuOTA2IGwgMS4zMjIsLTAuNTM1IGMgMCwwIDIuOTE2LDEuMjQyIDMuMDI5LDEuMTMzIGwgMS42NzgsLTEuNjQxIGMgMC4xMTcsLTAuMTE1IC0xLjIyLC0yLjkxNiAtMS4yMiwtMi45MTYgbCAwLjU0NCwtMS4yOTMgYyAwLDAgMi45NjMsLTEuMTQzIDIuOTYzLC0xLjI5OSBWIDkuMzYgQyAyMS41OSw5LjE5OSAxOC42MjIsOC4xNDUgMTguNjIyLDguMTQ1IHogbSAtNC4zNjYsMi40MjMgYyAwLDEuODY3IC0xLjU1MywzLjM4NyAtMy40NjEsMy4zODcgLTEuOTA2LDAgLTMuNDYxLC0xLjUyIC0zLjQ2MSwtMy4zODcgMCwtMS44NjcgMS41NTUsLTMuMzg1IDMuNDYxLC0zLjM4NSAxLjkwOSwwLjAwMSAzLjQ2MSwxLjUxOCAzLjQ2MSwzLjM4NSB6Ig0KICAgaWQ9InBhdGgzIg0KICAgc3R5bGU9ImZpbGw6I0ZGRkZGRiIgLz4NCjxnDQogICBpZD0iZzUiPg0KPC9nPg0KPGcNCiAgIGlkPSJnNyI+DQo8L2c+DQo8Zw0KICAgaWQ9Imc5Ij4NCjwvZz4NCjxnDQogICBpZD0iZzExIj4NCjwvZz4NCjxnDQogICBpZD0iZzEzIj4NCjwvZz4NCjxnDQogICBpZD0iZzE1Ij4NCjwvZz4NCjxnDQogICBpZD0iZzE3Ij4NCjwvZz4NCjxnDQogICBpZD0iZzE5Ij4NCjwvZz4NCjxnDQogICBpZD0iZzIxIj4NCjwvZz4NCjxnDQogICBpZD0iZzIzIj4NCjwvZz4NCjxnDQogICBpZD0iZzI1Ij4NCjwvZz4NCjxnDQogICBpZD0iZzI3Ij4NCjwvZz4NCjxnDQogICBpZD0iZzI5Ij4NCjwvZz4NCjxnDQogICBpZD0iZzMxIj4NCjwvZz4NCjxnDQogICBpZD0iZzMzIj4NCjwvZz4NCjwvc3ZnPg0K) 50% no-repeat}.emote-menu.editing .icon-gear{-webkit-animation:spin 4s linear infinite;animation:spin 4s linear infinite}.emote-menu .icon-resize-handle{background:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iMTYiDQogICBoZWlnaHQ9IjE2Ig0KICAgdmlld0JveD0iMCAwIDE2IDE2Ig0KICAgaWQ9IkNhcGFfMSINCiAgIHhtbDpzcGFjZT0icHJlc2VydmUiPjxtZXRhZGF0YQ0KICAgaWQ9Im1ldGFkYXRhNDM1NyI+PHJkZjpSREY+PGNjOldvcmsNCiAgICAgICByZGY6YWJvdXQ9IiI+PGRjOmZvcm1hdD5pbWFnZS9zdmcreG1sPC9kYzpmb3JtYXQ+PGRjOnR5cGUNCiAgICAgICAgIHJkZjpyZXNvdXJjZT0iaHR0cDovL3B1cmwub3JnL2RjL2RjbWl0eXBlL1N0aWxsSW1hZ2UiIC8+PGRjOnRpdGxlPjwvZGM6dGl0bGU+PC9jYzpXb3JrPjwvcmRmOlJERj48L21ldGFkYXRhPjxkZWZzDQogICBpZD0iZGVmczQzNTUiIC8+DQo8cGF0aA0KICAgZD0iTSAxMy41LDggQyAxMy4yMjUsOCAxMyw4LjIyNCAxMyw4LjUgdiAzLjc5MyBMIDMuNzA3LDMgSCA3LjUgQyA3Ljc3NiwzIDgsMi43NzYgOCwyLjUgOCwyLjIyNCA3Ljc3NiwyIDcuNSwyIGggLTUgTCAyLjMwOSwyLjAzOSAyLjE1LDIuMTQ0IDIuMTQ2LDIuMTQ2IDIuMTQzLDIuMTUyIDIuMDM5LDIuMzA5IDIsMi41IHYgNSBDIDIsNy43NzYgMi4yMjQsOCAyLjUsOCAyLjc3Niw4IDMsNy43NzYgMyw3LjUgViAzLjcwNyBMIDEyLjI5MywxMyBIIDguNSBDIDguMjI0LDEzIDgsMTMuMjI1IDgsMTMuNSA4LDEzLjc3NSA4LjIyNCwxNCA4LjUsMTQgaCA1IGwgMC4xOTEsLTAuMDM5IGMgMC4xMjEsLTAuMDUxIDAuMjIsLTAuMTQ4IDAuMjcsLTAuMjcgTCAxNCwxMy41MDIgViA4LjUgQyAxNCw4LjIyNCAxMy43NzUsOCAxMy41LDggeiINCiAgIGlkPSJwYXRoNDM1MSINCiAgIHN0eWxlPSJmaWxsOiNmZmZmZmY7ZmlsbC1vcGFjaXR5OjEiIC8+DQo8L3N2Zz4=) 50% no-repeat;cursor:nwse-resize!important}.emote-menu .icon-pin{background:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iMTYiDQogICBoZWlnaHQ9IjE2Ig0KICAgaWQ9InN2ZzMwMDUiPg0KICA8bWV0YWRhdGENCiAgICAgaWQ9Im1ldGFkYXRhMzAyMyI+DQogICAgPHJkZjpSREY+DQogICAgICA8Y2M6V29yaw0KICAgICAgICAgcmRmOmFib3V0PSIiPg0KICAgICAgICA8ZGM6Zm9ybWF0PmltYWdlL3N2Zyt4bWw8L2RjOmZvcm1hdD4NCiAgICAgICAgPGRjOnR5cGUNCiAgICAgICAgICAgcmRmOnJlc291cmNlPSJodHRwOi8vcHVybC5vcmcvZGMvZGNtaXR5cGUvU3RpbGxJbWFnZSIgLz4NCiAgICAgICAgPGRjOnRpdGxlPjwvZGM6dGl0bGU+DQogICAgICA8L2NjOldvcms+DQogICAgPC9yZGY6UkRGPg0KICA8L21ldGFkYXRhPg0KICA8ZGVmcw0KICAgICBpZD0iZGVmczMwMjEiIC8+DQogIDxnDQogICAgIHRyYW5zZm9ybT0ibWF0cml4KDAuNzkzMDc4MiwwLDAsMC43OTMwNzgyLC0yLjE3MDk4NSwtODE0LjY5Mjk5KSINCiAgICAgaWQ9ImczMDA3Ij4NCiAgICA8Zw0KICAgICAgIHRyYW5zZm9ybT0ibWF0cml4KDAuNzA3MTEsMC43MDcxMSwtMC43MDcxMSwwLjcwNzExLDczNy43MDc1NSwyOTUuNDg4MDgpIg0KICAgICAgIGlkPSJnMzAwOSI+DQogICAgICA8Zw0KICAgICAgICAgaWQ9ImczNzU1Ij4NCiAgICAgICAgPHBhdGgNCiAgICAgICAgICAgZD0iTSA5Ljc4MTI1LDAgQyA5LjQ3NDA1NjIsMC42ODkxMTIgOS41MjA2OCwxLjUyMzA4NTMgOS4zMTI1LDIuMTg3NSBMIDQuOTM3NSw2LjU5Mzc1IEMgMy45NTg5NjA4LDYuNDI5NDgzIDIuOTQ3NzU0OCw2LjUzMjc4OTkgMiw2LjgxMjUgTCA1LjAzMTI1LDkuODQzNzUgMC41NjI1LDE0LjMxMjUgMCwxNiBDIDAuNTY5Mjk2MjgsMTUuNzk1NjI2IDEuMTY3NzM3OCwxNS42NDAyMzcgMS43MTg3NSwxNS40MDYyNSBMIDYuMTU2MjUsMTAuOTY4NzUgOS4xODc1LDE0IGMgMC4yNzk2ODIzLC0wLjk0Nzc4MyAwLjM4MzE1MjgsLTEuOTU4OTM3IDAuMjE4NzUsLTIuOTM3NSAxLjUwMDAxMSwtMS40ODk1Nzk4IDMuMDAwMDAxLC0yLjk3OTE1OSA0LjUsLTQuNDY4NzUgMC42MDExMDIsLTAuMDMxMzYxIDEuODIyMTM4LC0wLjA5NjEzNyAyLC0wLjQ2ODc1IEMgMTMuODc5ODkyLDQuMDY5NDgwMyAxMS44NDI4NjUsMi4wMjAyMjgyIDkuNzgxMjUsMCB6Ig0KICAgICAgICAgICB0cmFuc2Zvcm09Im1hdHJpeCgwLjg5MTU5Mzc0LC0wLjg5MTU5Mzc0LDAuODkxNTkzNzQsMC44OTE1OTM3NCwtMi4yNjU1LDEwMzcuMTM0NSkiDQogICAgICAgICAgIGlkPSJwYXRoMzAxMSINCiAgICAgICAgICAgc3R5bGU9ImZpbGw6I2ZmZmZmZjtmaWxsLW9wYWNpdHk6MSIgLz4NCiAgICAgIDwvZz4NCiAgICA8L2c+DQogIDwvZz4NCjwvc3ZnPg0K) 50% no-repeat;transition:all .25s ease}.emote-menu .icon-pin:hover,.emote-menu.pinned .icon-pin{-webkit-transform:rotate(-45deg);transform:rotate(-45deg);opacity:1}.emote-menu .edit-tool{background-position:50%;background-repeat:no-repeat;background-size:14px;border-radius:4px;border:1px solid #000;cursor:pointer;display:none;height:14px;opacity:.25;position:absolute;transition:all .25s ease;width:14px;z-index:1}.emote-menu .edit-tool:hover,.emote-menu .emote:hover .edit-tool{opacity:1}.emote-menu .edit-visibility{background-color:#00c800;background-image:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iMTAwIg0KICAgaGVpZ2h0PSIxMDAiDQogICB2aWV3Qm94PSIwIDAgMTAwIDEwMCINCiAgIGlkPSJMYXllcl8xIg0KICAgeG1sOnNwYWNlPSJwcmVzZXJ2ZSI+PG1ldGFkYXRhDQogICBpZD0ibWV0YWRhdGE5Ij48cmRmOlJERj48Y2M6V29yaw0KICAgICAgIHJkZjphYm91dD0iIj48ZGM6Zm9ybWF0PmltYWdlL3N2Zyt4bWw8L2RjOmZvcm1hdD48ZGM6dHlwZQ0KICAgICAgICAgcmRmOnJlc291cmNlPSJodHRwOi8vcHVybC5vcmcvZGMvZGNtaXR5cGUvU3RpbGxJbWFnZSIgLz48ZGM6dGl0bGU+PC9kYzp0aXRsZT48L2NjOldvcms+PC9yZGY6UkRGPjwvbWV0YWRhdGE+PGRlZnMNCiAgIGlkPSJkZWZzNyIgLz4NCjxwYXRoDQogICBkPSJNIDk3Ljk2NCw0Ni41NDggQyA5Ny4wOTgsNDUuNTI4IDc2LjQyNywyMS42MDMgNTAsMjEuNjAzIGMgLTI2LjQyNywwIC00Ny4wOTgsMjMuOTI1IC00Ny45NjUsMjQuOTQ2IC0xLjcwMSwyIC0xLjcwMSw0LjkwMiAxMGUtNCw2LjkwMyAwLjg2NiwxLjAyIDIxLjUzNywyNC45NDUgNDcuOTY0LDI0Ljk0NSAyNi40MjcsMCA0Ny4wOTgsLTIzLjkyNiA0Ny45NjUsLTI0Ljk0NiAxLjcwMSwtMiAxLjcwMSwtNC45MDIgLTAuMDAxLC02LjkwMyB6IE0gNTguMDczLDM1Ljk3NSBjIDEuNzc3LC0wLjk3IDQuMjU1LDAuMTQzIDUuNTM0LDIuNDg1IDEuMjc5LDIuMzQzIDAuODc1LDUuMDI5IC0wLjkwMiw1Ljk5OSAtMS43NzcsMC45NzEgLTQuMjU1LC0wLjE0MyAtNS41MzUsLTIuNDg1IC0xLjI3OSwtMi4zNDMgLTAuODc1LC01LjAyOSAwLjkwMywtNS45OTkgeiBNIDUwLDY5LjcyOSBDIDMxLjU0LDY5LjcyOSAxNi4wMDUsNTUuNTUzIDEwLjYyOCw1MCAxNC4yNTksNDYuMjQ5IDIyLjUyNiwzOC41NzEgMzMuMTk1LDMzLjk3OSAzMS4xMTQsMzcuMTQ1IDI5Ljg5NCw0MC45MjggMjkuODk0LDQ1IGMgMCwxMS4xMDQgOS4wMDEsMjAuMTA1IDIwLjEwNSwyMC4xMDUgMTEuMTA0LDAgMjAuMTA2LC05LjAwMSAyMC4xMDYsLTIwLjEwNSAwLC00LjA3MiAtMS4yMTksLTcuODU1IC0zLjMsLTExLjAyMSBDIDc3LjQ3NCwzOC41NzIgODUuNzQxLDQ2LjI1IDg5LjM3Miw1MCA4My45OTUsNTUuNTU1IDY4LjQ2LDY5LjcyOSA1MCw2OS43MjkgeiINCiAgIGlkPSJwYXRoMyIgLz4NCjwvc3ZnPg==)}.emote-menu .edit-starred{background-color:#323232;background-image:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iNTAiDQogICBoZWlnaHQ9IjUwIg0KICAgdmlld0JveD0iMCAwIDUwIDUwIg0KICAgaWQ9IkxheWVyXzEiDQogICB4bWw6c3BhY2U9InByZXNlcnZlIj48bWV0YWRhdGENCiAgIGlkPSJtZXRhZGF0YTMwMDEiPjxyZGY6UkRGPjxjYzpXb3JrDQogICAgICAgcmRmOmFib3V0PSIiPjxkYzpmb3JtYXQ+aW1hZ2Uvc3ZnK3htbDwvZGM6Zm9ybWF0PjxkYzp0eXBlDQogICAgICAgICByZGY6cmVzb3VyY2U9Imh0dHA6Ly9wdXJsLm9yZy9kYy9kY21pdHlwZS9TdGlsbEltYWdlIiAvPjxkYzp0aXRsZT48L2RjOnRpdGxlPjwvY2M6V29yaz48L3JkZjpSREY+PC9tZXRhZGF0YT48ZGVmcw0KICAgaWQ9ImRlZnMyOTk5IiAvPg0KPHBhdGgNCiAgIGQ9Im0gNDMuMDQsMjIuNjk2IC03LjU2OCw3LjM3NyAxLjc4NywxMC40MTcgYyAwLjEyNywwLjc1IC0wLjE4MiwxLjUwOSAtMC43OTcsMS45NTcgLTAuMzQ4LDAuMjUzIC0wLjc2MiwwLjM4MiAtMS4xNzYsMC4zODIgLTAuMzE4LDAgLTAuNjM4LC0wLjA3NiAtMC45MzEsLTAuMjMgTCAyNSwzNy42ODEgMTUuNjQ1LDQyLjU5OSBjIC0wLjY3NCwwLjM1NSAtMS40OSwwLjI5NSAtMi4xMDcsLTAuMTUxIEMgMTIuOTIzLDQyIDEyLjYxNCw0MS4yNDIgMTIuNzQzLDQwLjQ5MSBMIDE0LjUzLDMwLjA3NCA2Ljk2MiwyMi42OTcgQyA2LjQxNSwyMi4xNjYgNi4yMjEsMjEuMzcxIDYuNDU0LDIwLjY0NyA2LjY5LDE5LjkyMyA3LjMxNSwxOS4zOTYgOC4wNjksMTkuMjg2IGwgMTAuNDU5LC0xLjUyMSA0LjY4LC05LjQ3OCBDIDIzLjU0Myw3LjYwMyAyNC4yMzksNy4xNzEgMjUsNy4xNzEgYyAwLjc2MywwIDEuNDU2LDAuNDMyIDEuNzkzLDEuMTE1IGwgNC42NzksOS40NzggMTAuNDYxLDEuNTIxIGMgMC43NTIsMC4xMDkgMS4zNzksMC42MzcgMS42MTIsMS4zNjEgMC4yMzcsMC43MjQgMC4wMzgsMS41MTkgLTAuNTA1LDIuMDUgeiINCiAgIGlkPSJwYXRoMjk5NSINCiAgIHN0eWxlPSJmaWxsOiNjY2NjY2M7ZmlsbC1vcGFjaXR5OjEiIC8+DQo8L3N2Zz4NCg==)}.emote-menu .emote>.edit-visibility{bottom:auto;left:auto;right:0;top:0}.emote-menu .emote>.edit-starred{bottom:auto;left:0;right:auto;top:0}.emote-menu .header-info>.edit-tool{margin-left:5px}.emote-menu.editing .edit-tool{display:inline-block}.emote-menu .emote-menu-hidden .edit-visibility{background-image:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iMTAwIg0KICAgaGVpZ2h0PSIxMDAiDQogICB2aWV3Qm94PSIwIDAgMTAwIDEwMCINCiAgIGlkPSJMYXllcl8zIg0KICAgeG1sOnNwYWNlPSJwcmVzZXJ2ZSI+PG1ldGFkYXRhDQogICBpZD0ibWV0YWRhdGExNSI+PHJkZjpSREY+PGNjOldvcmsNCiAgICAgICByZGY6YWJvdXQ9IiI+PGRjOmZvcm1hdD5pbWFnZS9zdmcreG1sPC9kYzpmb3JtYXQ+PGRjOnR5cGUNCiAgICAgICAgIHJkZjpyZXNvdXJjZT0iaHR0cDovL3B1cmwub3JnL2RjL2RjbWl0eXBlL1N0aWxsSW1hZ2UiIC8+PGRjOnRpdGxlPjwvZGM6dGl0bGU+PC9jYzpXb3JrPjwvcmRmOlJERj48L21ldGFkYXRhPjxkZWZzDQogICBpZD0iZGVmczEzIiAvPg0KPGcNCiAgIGlkPSJnMyI+DQoJPHBhdGgNCiAgIGQ9Ik0gNzAuMDgyLDQ1LjQ3NSA1MC40NzQsNjUuMDgyIEMgNjEuMTk4LDY0LjgzMSA2OS44MzEsNTYuMTk3IDcwLjA4Miw0NS40NzUgeiINCiAgIGlkPSJwYXRoNSINCiAgIHN0eWxlPSJmaWxsOiNGRkZGRkYiIC8+DQoJPHBhdGgNCiAgIGQ9Im0gOTcuOTY0LDQ2LjU0OCBjIC0wLjQ1LC0wLjUyOSAtNi4yNDUsLTcuMjMgLTE1LjQwMywtMTMuNTU0IGwgLTYuMiw2LjIgQyA4Mi4zNTEsNDMuMTQ4IDg2LjkyLDQ3LjQ2OSA4OS4zNzIsNTAgODMuOTk1LDU1LjU1NSA2OC40Niw2OS43MjkgNTAsNjkuNzI5IGMgLTEuMzM0LDAgLTIuNjUxLC0wLjA4MiAtMy45NTIsLTAuMjIyIGwgLTcuNDM5LDcuNDM5IGMgMy42MzksMC45MDkgNy40NDksMS40NSAxMS4zOTEsMS40NSAyNi40MjcsMCA0Ny4wOTgsLTIzLjkyNiA0Ny45NjUsLTI0Ljk0NiAxLjcwMSwtMS45OTkgMS43MDEsLTQuOTAxIC0wLjAwMSwtNi45MDIgeiINCiAgIGlkPSJwYXRoNyINCiAgIHN0eWxlPSJmaWxsOiNGRkZGRkYiIC8+DQoJPHBhdGgNCiAgIGQ9Im0gOTEuNDExLDE2LjY2IGMgMCwtMC4yNjYgLTAuMTA1LC0wLjUyIC0wLjI5MywtMC43MDcgbCAtNy4wNzEsLTcuMDcgYyAtMC4zOTEsLTAuMzkxIC0xLjAyMywtMC4zOTEgLTEuNDE0LDAgTCA2Ni44MDQsMjQuNzExIEMgNjEuNjAyLDIyLjgxOCA1NS45NDksMjEuNjAzIDUwLDIxLjYwMyBjIC0yNi40MjcsMCAtNDcuMDk4LDIzLjkyNiAtNDcuOTY1LDI0Ljk0NiAtMS43MDEsMiAtMS43MDEsNC45MDIgMTBlLTQsNi45MDMgMC41MTcsMC42MDcgOC4wODMsOS4zNTQgMTkuNzA3LDE2LjMyIEwgOC44ODMsODIuNjMyIEMgOC42OTUsODIuODIgOC41OSw4My4wNzMgOC41OSw4My4zMzkgYyAwLDAuMjY2IDAuMTA1LDAuNTIgMC4yOTMsMC43MDcgbCA3LjA3MSw3LjA3IGMgMC4xOTUsMC4xOTUgMC40NTEsMC4yOTMgMC43MDcsMC4yOTMgMC4yNTYsMCAwLjUxMiwtMC4wOTggMC43MDcsLTAuMjkzIGwgNzMuNzUsLTczLjc1IGMgMC4xODcsLTAuMTg2IDAuMjkzLC0wLjQ0IDAuMjkzLC0wLjcwNiB6IE0gMTAuNjI4LDUwIEMgMTQuMjU5LDQ2LjI0OSAyMi41MjYsMzguNTcxIDMzLjE5NSwzMy45NzkgMzEuMTE0LDM3LjE0NSAyOS44OTQsNDAuOTI4IDI5Ljg5NCw0NSBjIDAsNC42NjUgMS42MDEsOC45NDUgNC4yNywxMi4zNTEgTCAyOC4wNCw2My40NzUgQyAxOS44ODgsNTguOTU1IDEzLjY0OSw1My4xMiAxMC42MjgsNTAgeiINCiAgIGlkPSJwYXRoOSINCiAgIHN0eWxlPSJmaWxsOiNGRkZGRkYiIC8+DQo8L2c+DQo8L3N2Zz4NCg==);background-color:red}.emote-menu .emote-menu-starred .edit-starred{background-image:url(data:image/svg+xml;base64,PD94bWwgdmVyc2lvbj0iMS4wIiBlbmNvZGluZz0iVVRGLTgiIHN0YW5kYWxvbmU9Im5vIj8+DQo8IS0tIENyZWF0ZWQgd2l0aCBJbmtzY2FwZSAoaHR0cDovL3d3dy5pbmtzY2FwZS5vcmcvKSAtLT4NCg0KPHN2Zw0KICAgeG1sbnM6ZGM9Imh0dHA6Ly9wdXJsLm9yZy9kYy9lbGVtZW50cy8xLjEvIg0KICAgeG1sbnM6Y2M9Imh0dHA6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL25zIyINCiAgIHhtbG5zOnJkZj0iaHR0cDovL3d3dy53My5vcmcvMTk5OS8wMi8yMi1yZGYtc3ludGF4LW5zIyINCiAgIHhtbG5zOnN2Zz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciDQogICB2ZXJzaW9uPSIxLjEiDQogICB3aWR0aD0iNTAiDQogICBoZWlnaHQ9IjUwIg0KICAgdmlld0JveD0iMCAwIDUwIDUwIg0KICAgaWQ9IkxheWVyXzEiDQogICB4bWw6c3BhY2U9InByZXNlcnZlIj48bWV0YWRhdGENCiAgIGlkPSJtZXRhZGF0YTMwMDEiPjxyZGY6UkRGPjxjYzpXb3JrDQogICAgICAgcmRmOmFib3V0PSIiPjxkYzpmb3JtYXQ+aW1hZ2Uvc3ZnK3htbDwvZGM6Zm9ybWF0PjxkYzp0eXBlDQogICAgICAgICByZGY6cmVzb3VyY2U9Imh0dHA6Ly9wdXJsLm9yZy9kYy9kY21pdHlwZS9TdGlsbEltYWdlIiAvPjxkYzp0aXRsZT48L2RjOnRpdGxlPjwvY2M6V29yaz48L3JkZjpSREY+PC9tZXRhZGF0YT48ZGVmcw0KICAgaWQ9ImRlZnMyOTk5IiAvPg0KPHBhdGgNCiAgIGQ9Im0gNDMuMDQsMjIuNjk2IC03LjU2OCw3LjM3NyAxLjc4NywxMC40MTcgYyAwLjEyNywwLjc1IC0wLjE4MiwxLjUwOSAtMC43OTcsMS45NTcgLTAuMzQ4LDAuMjUzIC0wLjc2MiwwLjM4MiAtMS4xNzYsMC4zODIgLTAuMzE4LDAgLTAuNjM4LC0wLjA3NiAtMC45MzEsLTAuMjMgTCAyNSwzNy42ODEgMTUuNjQ1LDQyLjU5OSBjIC0wLjY3NCwwLjM1NSAtMS40OSwwLjI5NSAtMi4xMDcsLTAuMTUxIEMgMTIuOTIzLDQyIDEyLjYxNCw0MS4yNDIgMTIuNzQzLDQwLjQ5MSBMIDE0LjUzLDMwLjA3NCA2Ljk2MiwyMi42OTcgQyA2LjQxNSwyMi4xNjYgNi4yMjEsMjEuMzcxIDYuNDU0LDIwLjY0NyA2LjY5LDE5LjkyMyA3LjMxNSwxOS4zOTYgOC4wNjksMTkuMjg2IGwgMTAuNDU5LC0xLjUyMSA0LjY4LC05LjQ3OCBDIDIzLjU0Myw3LjYwMyAyNC4yMzksNy4xNzEgMjUsNy4xNzEgYyAwLjc2MywwIDEuNDU2LDAuNDMyIDEuNzkzLDEuMTE1IGwgNC42NzksOS40NzggMTAuNDYxLDEuNTIxIGMgMC43NTIsMC4xMDkgMS4zNzksMC42MzcgMS42MTIsMS4zNjEgMC4yMzcsMC43MjQgMC4wMzgsMS41MTkgLTAuNTA1LDIuMDUgeiINCiAgIGlkPSJwYXRoMjk5NSINCiAgIHN0eWxlPSJmaWxsOiNmZmNjMDA7ZmlsbC1vcGFjaXR5OjEiIC8+DQo8L3N2Zz4NCg==)}.emote-menu .emote.emote-menu-starred{border-color:rgba(200,200,0,.5)}.emote-menu .emote.emote-menu-hidden{border-color:rgba(255,0,0,.5)}.emote-menu #starred-emotes-group .emote:not(.emote-menu-starred),.emote-menu:not(.editing) .emote-menu-hidden{display:none}.emote-menu:not(.editing) #starred-emotes-group .emote-menu-starred{border-color:transparent}.emote-menu #starred-emotes-group{text-align:center;color:#646464}.emote-menu #starred-emotes-group:empty:before{content:\\\"Use the edit mode to star an emote!\\\";position:relative;top:8px}.emote-menu .scrollable{height:calc(100% - 101px);overflow-y:auto}.emote-menu .sticky{position:absolute;bottom:0;width:100%}.emote-menu .emote-menu-inner{position:relative;max-height:100%;height:100%}\"));\n","module.exports = (function() {\n    var Hogan = require('hogan.js/lib/template.js');\n    var templates = {};\n    templates['emote'] = new Hogan.Template({code: function (c,p,i) { var t=this;t.b(i=i||\"\");t.b(\"<div class=\\\"emote\");if(t.s(t.f(\"thirdParty\",c,p,1),c,p,0,32,44,\"{{ }}\")){t.rs(c,p,function(c,p,t){t.b(\" third-party\");});c.pop();}if(!t.s(t.f(\"isVisible\",c,p,1),c,p,1,0,0,\"\")){t.b(\" emote-menu-hidden\");};if(t.s(t.f(\"isStarred\",c,p,1),c,p,0,119,138,\"{{ }}\")){t.rs(c,p,function(c,p,t){t.b(\" emote-menu-starred\");});c.pop();}t.b(\"\\\" data-emote=\\\"\");t.b(t.v(t.f(\"text\",c,p,0)));t.b(\"\\\" title=\\\"\");t.b(t.v(t.f(\"text\",c,p,0)));if(t.s(t.f(\"thirdParty\",c,p,1),c,p,0,206,229,\"{{ }}\")){t.rs(c,p,function(c,p,t){t.b(\" (from 3rd party addon)\");});c.pop();}t.b(\"\\\">\\r\");t.b(\"\\n\" + i);t.b(\"\t<img src=\\\"\");t.b(t.t(t.f(\"url\",c,p,0)));t.b(\"\\\">\\r\");t.b(\"\\n\" + i);t.b(\"\t<div class=\\\"edit-tool edit-starred\\\" data-which=\\\"\");t.b(t.v(t.f(\"text\",c,p,0)));t.b(\"\\\" data-command=\\\"toggle-starred\\\" title=\\\"Star/unstar emote: \");t.b(t.v(t.f(\"text\",c,p,0)));t.b(\"\\\"></div>\\r\");t.b(\"\\n\" + i);t.b(\"\t<div class=\\\"edit-tool edit-visibility\\\" data-which=\\\"\");t.b(t.v(t.f(\"text\",c,p,0)));t.b(\"\\\" data-command=\\\"toggle-visibility\\\" title=\\\"Hide/show emote: \");t.b(t.v(t.f(\"text\",c,p,0)));t.b(\"\\\"></div>\\r\");t.b(\"\\n\" + i);t.b(\"</div>\\r\");t.b(\"\\n\");return t.fl(); },partials: {}, subs: {  }});\n    templates['emoteButton'] = new Hogan.Template({code: function (c,p,i) { var t=this;t.b(i=i||\"\");t.b(\"<button class=\\\"button button--icon-only float-left\\\" title=\\\"Emote Menu\\\" id=\\\"emote-menu-button\\\"></button>\\r\");t.b(\"\\n\");return t.fl(); },partials: {}, subs: {  }});\n    templates['emoteGroupHeader'] = new Hogan.Template({code: function (c,p,i) { var t=this;t.b(i=i||\"\");t.b(\"<div class=\\\"group-header\\\" data-emote-channel=\\\"\");t.b(t.v(t.f(\"channel\",c,p,0)));t.b(\"\\\">\\r\");t.b(\"\\n\" + i);t.b(\"\t<div class=\\\"header-info\\\">\\r\");t.b(\"\\n\" + i);t.b(\"\t\t<img src=\\\"\");t.b(t.v(t.f(\"badge\",c,p,0)));t.b(\"\\\" />\\r\");t.b(\"\\n\" + i);t.b(\"\t\t\");t.b(t.v(t.f(\"channelDisplayName\",c,p,0)));t.b(\"\\r\");t.b(\"\\n\" + i);t.b(\"\t\t<div class=\\\"edit-tool edit-visibility\\\" data-which=\\\"channel-\");t.b(t.v(t.f(\"channel\",c,p,0)));t.b(\"\\\" data-command=\\\"toggle-visibility\\\" title=\\\"Hide/show current emotes for \");t.b(t.v(t.f(\"channelDisplayName\",c,p,0)));t.b(\" (note: new emotes will still show up if they are added)\\\"></div>\\r\");t.b(\"\\n\" + i);t.b(\"\t</div>\\r\");t.b(\"\\n\" + i);t.b(\"\t<div class=\\\"emote-container\\\"></div>\\r\");t.b(\"\\n\" + i);t.b(\"</div>\\r\");t.b(\"\\n\");return t.fl(); },partials: {}, subs: {  }});\n    templates['menu'] = new Hogan.Template({code: function (c,p,i) { var t=this;t.b(i=i||\"\");t.b(\"<div class=\\\"emote-menu\\\" id=\\\"emote-menu-for-twitch\\\">\\r\");t.b(\"\\n\" + i);t.b(\"\t<div class=\\\"emote-menu-inner\\\">\\r\");t.b(\"\\n\" + i);t.b(\"\\r\");t.b(\"\\n\" + i);t.b(\"\t\t<div class=\\\"draggable\\\"></div>\\r\");t.b(\"\\n\" + i);t.b(\"\\r\");t.b(\"\\n\" + i);t.b(\"\t\t<div class=\\\"scrollable scrollbar-macosx\\\">\\r\");t.b(\"\\n\" + i);t.b(\"\t\t\t<div class=\\\"group-container\\\" id=\\\"all-emotes-group\\\"></div>\\r\");t.b(\"\\n\" + i);t.b(\"\t\t</div>\\r\");t.b(\"\\n\" + i);t.b(\"\\r\");t.b(\"\\n\" + i);t.b(\"\t\t<div class=\\\"sticky\\\">\\r\");t.b(\"\\n\" + i);t.b(\"\t\t\t<div class=\\\"group-header single-row\\\" id=\\\"starred-emotes-group\\\">\\r\");t.b(\"\\n\" + i);t.b(\"\t\t\t\t<div class=\\\"header-info\\\">Favorite Emotes</div>\\r\");t.b(\"\\n\" + i);t.b(\"\t\t\t\t<div class=\\\"emote-container\\\"></div>\\r\");t.b(\"\\n\" + i);t.b(\"\t\t\t</div>\\r\");t.b(\"\\n\" + i);t.b(\"\\r\");t.b(\"\\n\" + i);t.b(\"\t\t\t<div class=\\\"footer\\\">\\r\");t.b(\"\\n\" + i);t.b(\"\t\t\t\t<a class=\\\"pull-left icon icon-home\\\" href=\\\"http://cletusc.github.io/Userscript--Twitch-Chat-Emotes\\\" target=\\\"_blank\\\" title=\\\"Visit the homepage where you can donate, post a review, or contact the developer\\\"></a>\\r\");t.b(\"\\n\" + i);t.b(\"\t\t\t\t<a class=\\\"pull-left icon icon-gear\\\" data-command=\\\"toggle-editing\\\" title=\\\"Toggle edit mode\\\"></a>\\r\");t.b(\"\\n\" + i);t.b(\"\t\t\t\t<a class=\\\"pull-right icon icon-resize-handle\\\" data-command=\\\"resize-handle\\\"></a>\\r\");t.b(\"\\n\" + i);t.b(\"\t\t\t\t<a class=\\\"pull-right icon icon-pin\\\" data-command=\\\"toggle-pinned\\\" title=\\\"Pin/unpin the emote menu to the screen\\\"></a>\\r\");t.b(\"\\n\" + i);t.b(\"\t\t\t</div>\\r\");t.b(\"\\n\" + i);t.b(\"\t\t</div>\\r\");t.b(\"\\n\" + i);t.b(\"\\r\");t.b(\"\\n\" + i);t.b(\"\t</div>\\r\");t.b(\"\\n\" + i);t.b(\"</div>\\r\");t.b(\"\\n\");return t.fl(); },partials: {}, subs: {  }});\n    templates['newsMessage'] = new Hogan.Template({code: function (c,p,i) { var t=this;t.b(i=i||\"\");t.b(\"\\r\");t.b(\"\\n\" + i);t.b(\"<div class=\\\"twitch-chat-emotes-news\\\">\\r\");t.b(\"\\n\" + i);t.b(\"\t[\");t.b(t.v(t.f(\"scriptName\",c,p,0)));t.b(\"] News: \");t.b(t.t(t.f(\"message\",c,p,0)));t.b(\" (<a href=\\\"#\\\" data-command=\\\"twitch-chat-emotes:dismiss-news\\\" data-news-id=\\\"\");t.b(t.v(t.f(\"id\",c,p,0)));t.b(\"\\\">Dismiss</a>)\\r\");t.b(\"\\n\" + i);t.b(\"</div>\\r\");t.b(\"\\n\");return t.fl(); },partials: {}, subs: {  }});\n    return templates;\n})();","/*\n *  Copyright 2011 Twitter, Inc.\n *  Licensed under the Apache License, Version 2.0 (the \"License\");\n *  you may not use this file except in compliance with the License.\n *  You may obtain a copy of the License at\n *\n *  http://www.apache.org/licenses/LICENSE-2.0\n *\n *  Unless required by applicable law or agreed to in writing, software\n *  distributed under the License is distributed on an \"AS IS\" BASIS,\n *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n *  See the License for the specific language governing permissions and\n *  limitations under the License.\n */\n\nvar Hogan = {};\n\n(function (Hogan) {\n  Hogan.Template = function (codeObj, text, compiler, options) {\n    codeObj = codeObj || {};\n    this.r = codeObj.code || this.r;\n    this.c = compiler;\n    this.options = options || {};\n    this.text = text || '';\n    this.partials = codeObj.partials || {};\n    this.subs = codeObj.subs || {};\n    this.buf = '';\n  }\n\n  Hogan.Template.prototype = {\n    // render: replaced by generated code.\n    r: function (context, partials, indent) { return ''; },\n\n    // variable escaping\n    v: hoganEscape,\n\n    // triple stache\n    t: coerceToString,\n\n    render: function render(context, partials, indent) {\n      return this.ri([context], partials || {}, indent);\n    },\n\n    // render internal -- a hook for overrides that catches partials too\n    ri: function (context, partials, indent) {\n      return this.r(context, partials, indent);\n    },\n\n    // ensurePartial\n    ep: function(symbol, partials) {\n      var partial = this.partials[symbol];\n\n      // check to see that if we've instantiated this partial before\n      var template = partials[partial.name];\n      if (partial.instance && partial.base == template) {\n        return partial.instance;\n      }\n\n      if (typeof template == 'string') {\n        if (!this.c) {\n          throw new Error(\"No compiler available.\");\n        }\n        template = this.c.compile(template, this.options);\n      }\n\n      if (!template) {\n        return null;\n      }\n\n      // We use this to check whether the partials dictionary has changed\n      this.partials[symbol].base = template;\n\n      if (partial.subs) {\n        // Make sure we consider parent template now\n        if (!partials.stackText) partials.stackText = {};\n        for (key in partial.subs) {\n          if (!partials.stackText[key]) {\n            partials.stackText[key] = (this.activeSub !== undefined && partials.stackText[this.activeSub]) ? partials.stackText[this.activeSub] : this.text;\n          }\n        }\n        template = createSpecializedPartial(template, partial.subs, partial.partials,\n          this.stackSubs, this.stackPartials, partials.stackText);\n      }\n      this.partials[symbol].instance = template;\n\n      return template;\n    },\n\n    // tries to find a partial in the current scope and render it\n    rp: function(symbol, context, partials, indent) {\n      var partial = this.ep(symbol, partials);\n      if (!partial) {\n        return '';\n      }\n\n      return partial.ri(context, partials, indent);\n    },\n\n    // render a section\n    rs: function(context, partials, section) {\n      var tail = context[context.length - 1];\n\n      if (!isArray(tail)) {\n        section(context, partials, this);\n        return;\n      }\n\n      for (var i = 0; i < tail.length; i++) {\n        context.push(tail[i]);\n        section(context, partials, this);\n        context.pop();\n      }\n    },\n\n    // maybe start a section\n    s: function(val, ctx, partials, inverted, start, end, tags) {\n      var pass;\n\n      if (isArray(val) && val.length === 0) {\n        return false;\n      }\n\n      if (typeof val == 'function') {\n        val = this.ms(val, ctx, partials, inverted, start, end, tags);\n      }\n\n      pass = !!val;\n\n      if (!inverted && pass && ctx) {\n        ctx.push((typeof val == 'object') ? val : ctx[ctx.length - 1]);\n      }\n\n      return pass;\n    },\n\n    // find values with dotted names\n    d: function(key, ctx, partials, returnFound) {\n      var found,\n          names = key.split('.'),\n          val = this.f(names[0], ctx, partials, returnFound),\n          doModelGet = this.options.modelGet,\n          cx = null;\n\n      if (key === '.' && isArray(ctx[ctx.length - 2])) {\n        val = ctx[ctx.length - 1];\n      } else {\n        for (var i = 1; i < names.length; i++) {\n          found = findInScope(names[i], val, doModelGet);\n          if (found !== undefined) {\n            cx = val;\n            val = found;\n          } else {\n            val = '';\n          }\n        }\n      }\n\n      if (returnFound && !val) {\n        return false;\n      }\n\n      if (!returnFound && typeof val == 'function') {\n        ctx.push(cx);\n        val = this.mv(val, ctx, partials);\n        ctx.pop();\n      }\n\n      return val;\n    },\n\n    // find values with normal names\n    f: function(key, ctx, partials, returnFound) {\n      var val = false,\n          v = null,\n          found = false,\n          doModelGet = this.options.modelGet;\n\n      for (var i = ctx.length - 1; i >= 0; i--) {\n        v = ctx[i];\n        val = findInScope(key, v, doModelGet);\n        if (val !== undefined) {\n          found = true;\n          break;\n        }\n      }\n\n      if (!found) {\n        return (returnFound) ? false : \"\";\n      }\n\n      if (!returnFound && typeof val == 'function') {\n        val = this.mv(val, ctx, partials);\n      }\n\n      return val;\n    },\n\n    // higher order templates\n    ls: function(func, cx, partials, text, tags) {\n      var oldTags = this.options.delimiters;\n\n      this.options.delimiters = tags;\n      this.b(this.ct(coerceToString(func.call(cx, text)), cx, partials));\n      this.options.delimiters = oldTags;\n\n      return false;\n    },\n\n    // compile text\n    ct: function(text, cx, partials) {\n      if (this.options.disableLambda) {\n        throw new Error('Lambda features disabled.');\n      }\n      return this.c.compile(text, this.options).render(cx, partials);\n    },\n\n    // template result buffering\n    b: function(s) { this.buf += s; },\n\n    fl: function() { var r = this.buf; this.buf = ''; return r; },\n\n    // method replace section\n    ms: function(func, ctx, partials, inverted, start, end, tags) {\n      var textSource,\n          cx = ctx[ctx.length - 1],\n          result = func.call(cx);\n\n      if (typeof result == 'function') {\n        if (inverted) {\n          return true;\n        } else {\n          textSource = (this.activeSub && this.subsText && this.subsText[this.activeSub]) ? this.subsText[this.activeSub] : this.text;\n          return this.ls(result, cx, partials, textSource.substring(start, end), tags);\n        }\n      }\n\n      return result;\n    },\n\n    // method replace variable\n    mv: function(func, ctx, partials) {\n      var cx = ctx[ctx.length - 1];\n      var result = func.call(cx);\n\n      if (typeof result == 'function') {\n        return this.ct(coerceToString(result.call(cx)), cx, partials);\n      }\n\n      return result;\n    },\n\n    sub: function(name, context, partials, indent) {\n      var f = this.subs[name];\n      if (f) {\n        this.activeSub = name;\n        f(context, partials, this, indent);\n        this.activeSub = false;\n      }\n    }\n\n  };\n\n  //Find a key in an object\n  function findInScope(key, scope, doModelGet) {\n    var val;\n\n    if (scope && typeof scope == 'object') {\n\n      if (scope[key] !== undefined) {\n        val = scope[key];\n\n      // try lookup with get for backbone or similar model data\n      } else if (doModelGet && scope.get && typeof scope.get == 'function') {\n        val = scope.get(key);\n      }\n    }\n\n    return val;\n  }\n\n  function createSpecializedPartial(instance, subs, partials, stackSubs, stackPartials, stackText) {\n    function PartialTemplate() {};\n    PartialTemplate.prototype = instance;\n    function Substitutions() {};\n    Substitutions.prototype = instance.subs;\n    var key;\n    var partial = new PartialTemplate();\n    partial.subs = new Substitutions();\n    partial.subsText = {};  //hehe. substext.\n    partial.buf = '';\n\n    stackSubs = stackSubs || {};\n    partial.stackSubs = stackSubs;\n    partial.subsText = stackText;\n    for (key in subs) {\n      if (!stackSubs[key]) stackSubs[key] = subs[key];\n    }\n    for (key in stackSubs) {\n      partial.subs[key] = stackSubs[key];\n    }\n\n    stackPartials = stackPartials || {};\n    partial.stackPartials = stackPartials;\n    for (key in partials) {\n      if (!stackPartials[key]) stackPartials[key] = partials[key];\n    }\n    for (key in stackPartials) {\n      partial.partials[key] = stackPartials[key];\n    }\n\n    return partial;\n  }\n\n  var rAmp = /&/g,\n      rLt = /</g,\n      rGt = />/g,\n      rApos = /\\'/g,\n      rQuot = /\\\"/g,\n      hChars = /[&<>\\\"\\']/;\n\n  function coerceToString(val) {\n    return String((val === null || val === undefined) ? '' : val);\n  }\n\n  function hoganEscape(str) {\n    str = coerceToString(str);\n    return hChars.test(str) ?\n      str\n        .replace(rAmp, '&amp;')\n        .replace(rLt, '&lt;')\n        .replace(rGt, '&gt;')\n        .replace(rApos, '&#39;')\n        .replace(rQuot, '&quot;') :\n      str;\n  }\n\n  var isArray = Array.isArray || function(a) {\n    return Object.prototype.toString.call(a) === '[object Array]';\n  };\n\n})(typeof exports !== 'undefined' ? exports : Hogan);\n","/**\n * jQuery CSS Customizable Scrollbar\n *\n * Copyright 2014, Yuriy Khabarov\n * Dual licensed under the MIT or GPL Version 2 licenses.\n *\n * If you found bug, please contact me via email <13real008@gmail.com>\n *\n * @author Yuriy Khabarov aka Gromo\n * @version 0.2.6\n * @url https://github.com/gromo/jquery.scrollbar/\n *\n */\n(function(e,t,n){\"use strict\";function h(t){if(o.webkit&&!t){return{height:0,width:0}}if(!o.data.outer){var n={border:\"none\",\"box-sizing\":\"content-box\",height:\"200px\",margin:\"0\",padding:\"0\",width:\"200px\"};o.data.inner=e(\"<div>\").css(e.extend({},n));o.data.outer=e(\"<div>\").css(e.extend({left:\"-1000px\",overflow:\"scroll\",position:\"absolute\",top:\"-1000px\"},n)).append(o.data.inner).appendTo(\"body\")}o.data.outer.scrollLeft(1e3).scrollTop(1e3);return{height:Math.ceil(o.data.outer.offset().top-o.data.inner.offset().top||0),width:Math.ceil(o.data.outer.offset().left-o.data.inner.offset().left||0)}}function p(n,r){e(t).on({\"blur.scrollbar\":function(){e(t).add(\"body\").off(\".scrollbar\");n&&n()},\"dragstart.scrollbar\":function(e){e.preventDefault();return false},\"mouseup.scrollbar\":function(){e(t).add(\"body\").off(\".scrollbar\");n&&n()}});e(\"body\").on({\"selectstart.scrollbar\":function(e){e.preventDefault();return false}});r&&r.preventDefault();return false}function d(){var e=h(true);return!(e.height||e.width)}function v(e){var t=e.originalEvent;if(t.axis&&t.axis===t.HORIZONTAL_AXIS)return false;if(t.wheelDeltaX)return false;return true}var r=false;var i=1,s=\"px\";var o={data:{},macosx:n.navigator.platform.toLowerCase().indexOf(\"mac\")!==-1,mobile:/Android|webOS|iPhone|iPad|iPod|BlackBerry/i.test(n.navigator.userAgent),overlay:null,scroll:null,scrolls:[],webkit:/WebKit/.test(n.navigator.userAgent),log:r?function(t,r){var i=t;if(r&&typeof t!=\"string\"){i=[];e.each(t,function(e,t){i.push('\"'+e+'\": '+t)});i=i.join(\", \")}if(n.console&&n.console.log){n.console.log(i)}else{alert(i)}}:function(){}};var u={autoScrollSize:true,autoUpdate:true,debug:false,disableBodyScroll:false,duration:200,ignoreMobile:true,ignoreOverlay:true,scrollStep:30,showArrows:false,stepScrolling:true,type:\"simple\",scrollx:null,scrolly:null,onDestroy:null,onInit:null,onScroll:null,onUpdate:null};var a=function(t,r){if(!o.scroll){o.log(\"Init jQuery Scrollbar v0.2.6\");o.overlay=d();o.scroll=h();c();e(n).resize(function(){var e=false;if(o.scroll&&(o.scroll.height||o.scroll.width)){var t=h();if(t.height!=o.scroll.height||t.width!=o.scroll.width){o.scroll=t;e=true}}c(e)})}this.container=t;this.options=e.extend({},u,n.jQueryScrollbarOptions||{});this.scrollTo=null;this.scrollx={};this.scrolly={};this.init(r)};a.prototype={destroy:function(){if(!this.wrapper){return}var n=this.container.scrollLeft();var r=this.container.scrollTop();this.container.insertBefore(this.wrapper).css({height:\"\",margin:\"\"}).removeClass(\"scroll-content\").removeClass(\"scroll-scrollx_visible\").removeClass(\"scroll-scrolly_visible\").off(\".scrollbar\").scrollLeft(n).scrollTop(r);this.scrollx.scrollbar.removeClass(\"scroll-scrollx_visible\").find(\"div\").andSelf().off(\".scrollbar\");this.scrolly.scrollbar.removeClass(\"scroll-scrolly_visible\").find(\"div\").andSelf().off(\".scrollbar\");this.wrapper.remove();e(t).add(\"body\").off(\".scrollbar\");if(e.isFunction(this.options.onDestroy))this.options.onDestroy.apply(this,[this.container])},getScrollbar:function(t){var n=this.options[\"scroll\"+t];var r={advanced:'<div class=\"scroll-element_corner\"></div>'+'<div class=\"scroll-arrow scroll-arrow_less\"></div>'+'<div class=\"scroll-arrow scroll-arrow_more\"></div>'+'<div class=\"scroll-element_outer\">'+'    <div class=\"scroll-element_size\"></div>'+'    <div class=\"scroll-element_inner-wrapper\">'+'        <div class=\"scroll-element_inner scroll-element_track\">'+'            <div class=\"scroll-element_inner-bottom\"></div>'+\"        </div>\"+\"    </div>\"+'    <div class=\"scroll-bar\">'+'        <div class=\"scroll-bar_body\">'+'            <div class=\"scroll-bar_body-inner\"></div>'+\"        </div>\"+'        <div class=\"scroll-bar_bottom\"></div>'+'        <div class=\"scroll-bar_center\"></div>'+\"    </div>\"+\"</div>\",simple:'<div class=\"scroll-element_outer\">'+'    <div class=\"scroll-element_size\"></div>'+'    <div class=\"scroll-element_track\"></div>'+'    <div class=\"scroll-bar\"></div>'+\"</div>\"};var i=r[this.options.type]?this.options.type:\"advanced\";if(n){if(typeof n==\"string\"){n=e(n).appendTo(this.wrapper)}else{n=e(n)}}else{n=e(\"<div>\").addClass(\"scroll-element\").html(r[i]).appendTo(this.wrapper)}if(this.options.showArrows){n.addClass(\"scroll-element_arrows_visible\")}return n.addClass(\"scroll-\"+t)},init:function(n){var r=this;var u=this.container;var a=this.containerWrapper||u;var f=e.extend(this.options,n||{});var l={x:this.scrollx,y:this.scrolly};var c=this.wrapper;var h={scrollLeft:u.scrollLeft(),scrollTop:u.scrollTop()};if(o.mobile&&f.ignoreMobile||o.overlay&&f.ignoreOverlay||o.macosx&&!o.webkit){return false}if(!c){this.wrapper=c=e(\"<div>\").addClass(\"scroll-wrapper\").addClass(u.attr(\"class\")).css(\"position\",u.css(\"position\")==\"absolute\"?\"absolute\":\"relative\").insertBefore(u).append(u);if(u.is(\"textarea\")){this.containerWrapper=a=e(\"<div>\").insertBefore(u).append(u);c.addClass(\"scroll-textarea\")}a.addClass(\"scroll-content\").css({height:\"\",\"margin-bottom\":o.scroll.height*-1+s,\"margin-right\":o.scroll.width*-1+s});u.on(\"scroll.scrollbar\",function(t){if(e.isFunction(f.onScroll)){f.onScroll.call(r,{maxScroll:l.y.maxScrollOffset,scroll:u.scrollTop(),size:l.y.size,visible:l.y.visible},{maxScroll:l.x.maxScrollOffset,scroll:u.scrollLeft(),size:l.x.size,visible:l.x.visible})}l.x.isVisible&&l.x.scroller.css(\"left\",u.scrollLeft()*l.x.kx+s);l.y.isVisible&&l.y.scroller.css(\"top\",u.scrollTop()*l.y.kx+s)});c.on(\"scroll\",function(){c.scrollTop(0).scrollLeft(0)});if(f.disableBodyScroll){var d=function(e){v(e)?l.y.isVisible&&l.y.mousewheel(e):l.x.isVisible&&l.x.mousewheel(e)};c.on({\"MozMousePixelScroll.scrollbar\":d,\"mousewheel.scrollbar\":d});if(o.mobile){c.on(\"touchstart.scrollbar\",function(n){var r=n.originalEvent.touches&&n.originalEvent.touches[0]||n;var i={pageX:r.pageX,pageY:r.pageY};var s={left:u.scrollLeft(),top:u.scrollTop()};e(t).on({\"touchmove.scrollbar\":function(e){var t=e.originalEvent.targetTouches&&e.originalEvent.targetTouches[0]||e;u.scrollLeft(s.left+i.pageX-t.pageX);u.scrollTop(s.top+i.pageY-t.pageY);e.preventDefault()},\"touchend.scrollbar\":function(){e(t).off(\".scrollbar\")}})})}}if(e.isFunction(f.onInit))f.onInit.apply(this,[u])}else{a.css({height:\"\",\"margin-bottom\":o.scroll.height*-1+s,\"margin-right\":o.scroll.width*-1+s})}e.each(l,function(n,s){var o=null;var a=1;var c=n==\"x\"?\"scrollLeft\":\"scrollTop\";var h=f.scrollStep;var d=function(){var e=u[c]();u[c](e+h);if(a==1&&e+h>=m)e=u[c]();if(a==-1&&e+h<=m)e=u[c]();if(u[c]()==e&&o){o()}};var m=0;if(!s.scrollbar){s.scrollbar=r.getScrollbar(n);s.scroller=s.scrollbar.find(\".scroll-bar\");s.mousewheel=function(e){if(!s.isVisible||n==\"x\"&&v(e)){return true}if(n==\"y\"&&!v(e)){l.x.mousewheel(e);return true}var t=e.originalEvent.wheelDelta*-1||e.originalEvent.detail;var i=s.size-s.visible-s.offset;if(!(m<=0&&t<0||m>=i&&t>0)){m=m+t;if(m<0)m=0;if(m>i)m=i;r.scrollTo=r.scrollTo||{};r.scrollTo[c]=m;setTimeout(function(){if(r.scrollTo){u.stop().animate(r.scrollTo,240,\"linear\",function(){m=u[c]()});r.scrollTo=null}},1)}e.preventDefault();return false};s.scrollbar.on({\"MozMousePixelScroll.scrollbar\":s.mousewheel,\"mousewheel.scrollbar\":s.mousewheel,\"mouseenter.scrollbar\":function(){m=u[c]()}});s.scrollbar.find(\".scroll-arrow, .scroll-element_track\").on(\"mousedown.scrollbar\",function(t){if(t.which!=i)return true;a=1;var l={eventOffset:t[n==\"x\"?\"pageX\":\"pageY\"],maxScrollValue:s.size-s.visible-s.offset,scrollbarOffset:s.scroller.offset()[n==\"x\"?\"left\":\"top\"],scrollbarSize:s.scroller[n==\"x\"?\"outerWidth\":\"outerHeight\"]()};var v=0,g=0;if(e(this).hasClass(\"scroll-arrow\")){a=e(this).hasClass(\"scroll-arrow_more\")?1:-1;h=f.scrollStep*a;m=a>0?l.maxScrollValue:0}else{a=l.eventOffset>l.scrollbarOffset+l.scrollbarSize?1:l.eventOffset<l.scrollbarOffset?-1:0;h=Math.round(s.visible*.75)*a;m=l.eventOffset-l.scrollbarOffset-(f.stepScrolling?a==1?l.scrollbarSize:0:Math.round(l.scrollbarSize/2));m=u[c]()+m/s.kx}r.scrollTo=r.scrollTo||{};r.scrollTo[c]=f.stepScrolling?u[c]()+h:m;if(f.stepScrolling){o=function(){m=u[c]();clearInterval(g);clearTimeout(v);v=0;g=0};v=setTimeout(function(){g=setInterval(d,40)},f.duration+100)}setTimeout(function(){if(r.scrollTo){u.animate(r.scrollTo,f.duration);r.scrollTo=null}},1);return p(o,t)});s.scroller.on(\"mousedown.scrollbar\",function(r){if(r.which!=i)return true;var o=r[n==\"x\"?\"pageX\":\"pageY\"];var a=u[c]();s.scrollbar.addClass(\"scroll-draggable\");e(t).on(\"mousemove.scrollbar\",function(e){var t=parseInt((e[n==\"x\"?\"pageX\":\"pageY\"]-o)/s.kx,10);u[c](a+t)});return p(function(){s.scrollbar.removeClass(\"scroll-draggable\");m=u[c]()},r)})}});e.each(l,function(e,t){var n=\"scroll-scroll\"+e+\"_visible\";var r=e==\"x\"?l.y:l.x;t.scrollbar.removeClass(n);r.scrollbar.removeClass(n);a.removeClass(n)});e.each(l,function(t,n){e.extend(n,t==\"x\"?{offset:parseInt(u.css(\"left\"),10)||0,size:u.prop(\"scrollWidth\"),visible:c.width()}:{offset:parseInt(u.css(\"top\"),10)||0,size:u.prop(\"scrollHeight\"),visible:c.height()})});var m=function(t,n){var r=\"scroll-scroll\"+t+\"_visible\";var i=t==\"x\"?l.y:l.x;var f=parseInt(u.css(t==\"x\"?\"left\":\"top\"),10)||0;var h=n.size;var p=n.visible+f;n.isVisible=h-p>1;if(n.isVisible){n.scrollbar.addClass(r);i.scrollbar.addClass(r);a.addClass(r)}else{n.scrollbar.removeClass(r);i.scrollbar.removeClass(r);a.removeClass(r)}if(t==\"y\"&&(n.isVisible||n.size<n.visible)){a.css(\"height\",p+o.scroll.height+s)}if(l.x.size!=u.prop(\"scrollWidth\")||l.y.size!=u.prop(\"scrollHeight\")||l.x.visible!=c.width()||l.y.visible!=c.height()||l.x.offset!=(parseInt(u.css(\"left\"),10)||0)||l.y.offset!=(parseInt(u.css(\"top\"),10)||0)){e.each(l,function(t,n){e.extend(n,t==\"x\"?{offset:parseInt(u.css(\"left\"),10)||0,size:u.prop(\"scrollWidth\"),visible:c.width()}:{offset:parseInt(u.css(\"top\"),10)||0,size:u.prop(\"scrollHeight\"),visible:c.height()})});m(t==\"x\"?\"y\":\"x\",i)}};e.each(l,m);if(e.isFunction(f.onUpdate))f.onUpdate.apply(this,[u]);e.each(l,function(e,t){var n=e==\"x\"?\"left\":\"top\";var r=e==\"x\"?\"outerWidth\":\"outerHeight\";var i=e==\"x\"?\"width\":\"height\";var o=parseInt(u.css(n),10)||0;var a=t.size;var l=t.visible+o;var c=t.scrollbar.find(\".scroll-element_size\");c=c[r]()+(parseInt(c.css(n),10)||0);if(f.autoScrollSize){t.scrollbarSize=parseInt(c*l/a,10);t.scroller.css(i,t.scrollbarSize+s)}t.scrollbarSize=t.scroller[r]();t.kx=(c-t.scrollbarSize)/(a-l)||1;t.maxScrollOffset=a-l});u.scrollLeft(h.scrollLeft).scrollTop(h.scrollTop).trigger(\"scroll\")}};e.fn.scrollbar=function(t,n){var r=this;if(t===\"get\")r=null;this.each(function(){var i=e(this);if(i.hasClass(\"scroll-wrapper\")||i.get(0).nodeName==\"body\"){return true}var s=i.data(\"scrollbar\");if(s){if(t===\"get\"){r=s;return false}var u=typeof t==\"string\"&&s[t]?t:\"init\";s[u].apply(s,e.isArray(n)?n:[]);if(t===\"destroy\"){i.removeData(\"scrollbar\");while(e.inArray(s,o.scrolls)>=0)o.scrolls.splice(e.inArray(s,o.scrolls),1)}}else{if(typeof t!=\"string\"){s=new a(i,t);i.data(\"scrollbar\",s);o.scrolls.push(s)}}return true});return r};e.fn.scrollbar.options=u;if(n.angular){(function(e){var t=e.module(\"jQueryScrollbar\",[]);t.directive(\"jqueryScrollbar\",function(){return{link:function(e,t){t.scrollbar(e.options).on(\"$destroy\",function(){t.scrollbar(\"destroy\")})},restring:\"AC\",scope:{options:\"=jqueryScrollbar\"}}})})(n.angular)}var f=0,l=0;var c=function(e){var t,n,i,s,u,a,h;for(t=0;t<o.scrolls.length;t++){s=o.scrolls[t];n=s.container;i=s.options;u=s.wrapper;a=s.scrollx;h=s.scrolly;if(e||i.autoUpdate&&u&&u.is(\":visible\")&&(n.prop(\"scrollWidth\")!=a.size||n.prop(\"scrollHeight\")!=h.size||u.width()!=a.visible||u.height()!=h.visible)){s.init();if(r){o.log({scrollHeight:n.prop(\"scrollHeight\")+\":\"+s.scrolly.size,scrollWidth:n.prop(\"scrollWidth\")+\":\"+s.scrollx.size,visibleHeight:u.height()+\":\"+s.scrolly.visible,visibleWidth:u.width()+\":\"+s.scrollx.visible},true);l++}}}if(r&&l>10){o.log(\"Scroll updates exceed 10\");c=function(){}}else{clearTimeout(f);f=setTimeout(c,300)}}})(jQuery,document,window);","// Storage cache.\r\nvar cache = {};\r\n// The store handling expiration of data.\r\nvar expiresStore = new Store({\r\n\tnamespace: '__storage-wrapper:expires'\r\n});\r\n\r\n/**\r\n * Storage wrapper for making routine storage calls super easy.\r\n * @class Store\r\n * @constructor\r\n * @param {object} [options]                     The options for the store. Options not overridden will use the defaults.\r\n * @param {mixed}  [options.namespace='']        See {{#crossLink \"Store/setNamespace\"}}Store#setNamespace{{/crossLink}}\r\n * @param {mixed}  [options.storageType='local'] See {{#crossLink \"Store/setStorageType\"}}Store#setStorageType{{/crossLink}}\r\n */\r\nfunction Store(options) {\r\n\tvar settings = {\r\n\t\tnamespace: '',\r\n\t\tstorageType: 'local'\r\n\t};\r\n\r\n\t/**\r\n\t * Sets the storage namespace.\r\n\t * @method setNamespace\r\n\t * @param {string|false|null} namespace The namespace to work under. To use no namespace (e.g. global namespace), pass in `false` or `null` or an empty string.\r\n\t */\r\n\tthis.setNamespace = function (namespace) {\r\n\t\tvar validNamespace = /^[\\w-:]+$/;\r\n\t\t// No namespace.\r\n\t\tif (namespace === false || namespace == null || namespace === '') {\r\n\t\t\tsettings.namespace = '';\r\n\t\t\treturn;\r\n\t\t}\r\n\t\tif (typeof namespace !== 'string' || !validNamespace.test(namespace)) {\r\n\t\t\tthrow new Error('Invalid namespace.');\r\n\t\t}\r\n\t\tsettings.namespace = namespace;\r\n\t};\r\n\r\n\t/**\r\n\t * Gets the current storage namespace.\r\n\t * @method getNamespace\r\n\t * @return {string} The current namespace.\r\n\t */\r\n\tthis.getNamespace = function (includeSeparator) {\r\n\t\tif (includeSeparator && settings.namespace !== '') {\r\n\t\t\treturn settings.namespace + ':';\r\n\t\t}\r\n\t\treturn settings.namespace;\r\n\t}\r\n\r\n\t/**\r\n\t * Sets the type of storage to use.\r\n\t * @method setStorageType\r\n\t * @param {string} type The type of storage to use. Use `session` for `sessionStorage` and `local` for `localStorage`.\r\n\t */\r\n\tthis.setStorageType = function (type) {\r\n\t\tif (['session', 'local'].indexOf(type) < 0) {\r\n\t\t\tthrow new Error('Invalid storage type.');\r\n\t\t}\r\n\t\tsettings.storageType = type;\r\n\t};\r\n\t/**\r\n\t * Get the type of storage being used.\r\n\t * @method getStorageType\r\n\t * @return {string} The type of storage being used.\r\n\t */\r\n\tthis.getStorageType = function () {\r\n\t\treturn settings.storageType;\r\n\t};\r\n\r\n\t// Override default settings.\r\n\tif (options) {\r\n\t\tfor (var key in options) {\r\n\t\t\tswitch (key) {\r\n\t\t\t\tcase 'namespace':\r\n\t\t\t\t\tthis.setNamespace(options[key]);\r\n\t\t\t\t\tbreak;\r\n\t\t\t\tcase 'storageType':\r\n\t\t\t\t\tthis.setStorageType(options[key]);\r\n\t\t\t\t\tbreak;\r\n\t\t\t}\r\n\t\t}\r\n\t}\r\n}\r\n\r\n/**\r\n * Gets the actual handler to use\r\n * @method getStorageHandler\r\n * @return {mixed} The storage handler.\r\n */\r\nStore.prototype.getStorageHandler = function () {\r\n\tvar handlers = {\r\n\t\t'local': localStorage,\r\n\t\t'session': sessionStorage\r\n\t};\r\n\treturn handlers[this.getStorageType()];\r\n}\r\n\r\n/**\r\n * Gets the full storage name for a key, including the namespace, if any.\r\n * @method getStorageKey\r\n * @param  {string} key The storage key name.\r\n * @return {string}     The full storage name that is used by the storage methods.\r\n */\r\nStore.prototype.getStorageKey = function (key) {\r\n\tif (!key || typeof key !== 'string' || key.length < 1) {\r\n\t\tthrow new Error('Key must be a string.');\r\n\t}\r\n\treturn this.getNamespace(true) + key;\r\n};\r\n\r\n/**\r\n * Gets a storage item from the current namespace.\r\n * @method get\r\n * @param  {string} key          The key that the data can be accessed under.\r\n * @param  {mixed}  defaultValue The default value to return in case the storage value is not set or `null`.\r\n * @return {mixed}               The data for the storage.\r\n */\r\nStore.prototype.get = function (key, defaultValue) {\r\n\t// Prevent recursion. Only check expire date if it isn't called from `expiresStore`.\r\n\tif (this !== expiresStore) {\r\n\t\t// Check if key is expired.\r\n\t\tvar expireDate = expiresStore.get(this.getStorageKey(key));\r\n\t\tif (expireDate !== null && expireDate.getTime() < Date.now()) {\r\n\t\t\t// Expired, remove it.\r\n\t\t\tthis.remove(key);\r\n\t\t\texpiresStore.remove(this.getStorageKey(key));\r\n\t\t}\r\n\t}\r\n\r\n\t// Cached, read from memory.\r\n\tif (cache[this.getStorageKey(key)] != null) {\r\n\t\treturn cache[this.getStorageKey(key)];\r\n\t}\r\n\r\n\tvar val = this.getStorageHandler().getItem(this.getStorageKey(key));\r\n\r\n\t// Value doesn't exist and we have a default, return default.\r\n\tif (val === null && typeof defaultValue !== 'undefined') {\r\n\t\treturn defaultValue;\r\n\t}\r\n\r\n\t// Only pre-process strings.\r\n\tif (typeof val === 'string') {\r\n\t\t// Handle RegExps.\r\n\t\tif (val.indexOf('~RegExp:') === 0) {\r\n\t\t\tvar matches = /^~RegExp:([gim]*?):(.*)/.exec(val);\r\n\t\t\tval = new RegExp(matches[2], matches[1]);\r\n\t\t}\r\n\t\t// Handle Dates.\r\n\t\telse if (val.indexOf('~Date:') === 0) {\r\n\t\t\tval = new Date(val.replace(/^~Date:/, ''));\r\n\t\t}\r\n\t\t// Handle numbers.\r\n\t\telse if (val.indexOf('~Number:') === 0) {\r\n\t\t\tval = parseInt(val.replace(/^~Number:/, ''), 10);\r\n\t\t}\r\n\t\t// Handle booleans.\r\n\t\telse if (val.indexOf('~Boolean:') === 0) {\r\n\t\t\tval = val.replace(/^~Boolean:/, '') === 'true';\r\n\t\t}\r\n\t\t// Handle objects.\r\n\t\telse if (val.indexOf('~JSON:') === 0) {\r\n\t\t\tval = val.replace(/^~JSON:/, '');\r\n\t\t\t// Try parsing it.\r\n\t\t\ttry {\r\n\t\t\t\tval = JSON.parse(val);\r\n\t\t\t}\r\n\t\t\t// Parsing went wrong (invalid JSON), return default or null.\r\n\t\t\tcatch (e) {\r\n\t\t\t\tif (typeof defaultValue !== 'undefined') {\r\n\t\t\t\t\treturn defaultValue;\r\n\t\t\t\t}\r\n\t\t\t\treturn null;\r\n\t\t\t}\r\n\t\t}\r\n\t}\r\n\r\n\t// Return it.\r\n\tcache[this.getStorageKey(key)] = val;\r\n\treturn val;\r\n};\r\n\r\n/**\r\n * Sets a storage item on the current namespace.\r\n * @method set\r\n * @param {string}      key       The key that the data can be accessed under.\r\n * @param {mixed}       val       The value to store. May be the following types of data: `RegExp`, `Date`, `Object`, `String`, `Boolean`, `Number`\r\n * @param {Date|number} [expires] The date in the future to expire, or relative number of milliseconds from `Date#now` to expire.\r\n *\r\n * Note: This converts special data types that normally can't be stored in the following way:\r\n * \r\n * - `RegExp`: prefixed with type, flags stored, and source stored as string.\r\n * - `Date`: prefixed with type, stored as string using `Date#toString`.\r\n * - `Object`: prefixed with \"JSON\" indicator, stored as string using `JSON#stringify`.\r\n */\r\nStore.prototype.set = function (key, val, expires) {\r\n\tvar parsedVal = null;\r\n\t// Handle RegExps.\r\n\tif (val instanceof RegExp) {\r\n\t\tvar flags = [\r\n\t\t\tval.global ? 'g' : '',\r\n\t\t\tval.ignoreCase ? 'i' : '',\r\n\t\t\tval.multiline ? 'm' : '',\r\n\t\t].join('');\r\n\t\tparsedVal = '~RegExp:' + flags + ':' + val.source;\r\n\t}\r\n\t// Handle Dates.\r\n\telse if (val instanceof Date) {\r\n\t\tparsedVal = '~Date:' + val.toString();\r\n\t}\r\n\t// Handle objects.\r\n\telse if (val === Object(val)) {\r\n\t\tparsedVal = '~JSON:' + JSON.stringify(val);\r\n\t}\r\n\t// Handle numbers.\r\n\telse if (typeof val === 'number') {\r\n\t\tparsedVal = '~Number:' + val.toString();\r\n\t}\r\n\t// Handle booleans.\r\n\telse if (typeof val === 'boolean') {\r\n\t\tparsedVal = '~Boolean:' + val.toString();\r\n\t}\r\n\t// Handle strings.\r\n\telse if (typeof val === 'string') {\r\n\t\tparsedVal = val;\r\n\t}\r\n\t// Throw if we don't know what it is.\r\n\telse {\r\n\t\tthrow new Error('Unable to store this value; wrong value type.');\r\n\t}\r\n\t// Set expire date if needed.\r\n\tif (typeof expires !== 'undefined') {\r\n\t\t// Convert to a relative date.\r\n\t\tif (typeof expires === 'number') {\r\n\t\t\texpires = new Date(Date.now() + expires);\r\n\t\t}\r\n\t\t// Make sure it is a date.\r\n\t\tif (expires instanceof Date) {\r\n\t\t\texpiresStore.set(this.getStorageKey(key), expires);\r\n\t\t}\r\n\t\telse {\r\n\t\t\tthrow new Error('Key expire must be a valid date or timestamp.');\r\n\t\t}\r\n\t}\r\n\t// Save it.\r\n\tcache[this.getStorageKey(key)] = val;\r\n\tthis.getStorageHandler().setItem(this.getStorageKey(key), parsedVal);\r\n};\r\n\r\n/**\r\n * Gets all data for the current namespace.\r\n * @method getAll\r\n * @return {object} An object containing all data in the form of `{theKey: theData}` where `theData` is parsed using {{#crossLink \"Store/get\"}}Store#get{{/crossLink}}.\r\n */\r\nStore.prototype.getAll = function () {\r\n\tvar keys = this.listKeys();\r\n\tvar data = {};\r\n\tkeys.forEach(function (key) {\r\n\t\tdata[key] = this.get(key);\r\n\t}, this);\r\n\treturn data;\r\n};\r\n\r\n/**\r\n * List all keys that are tied to the current namespace.\r\n * @method listKeys\r\n * @return {array} The storage keys.\r\n */\r\nStore.prototype.listKeys = function () {\r\n\tvar keys = [];\r\n\tvar key = null;\r\n\tvar storageLength = this.getStorageHandler().length;\r\n\tvar prefix = new RegExp('^' + this.getNamespace(true));\r\n\tfor (var i = 0; i < storageLength; i++) {\r\n\t\tkey = this.getStorageHandler().key(i)\r\n\t\tif (prefix.test(key)) {\r\n\t\t\tkeys.push(key.replace(prefix, ''));\r\n\t\t}\r\n\t}\r\n\treturn keys;\r\n};\r\n\r\n/**\r\n * Removes a specific key and data from the current namespace.\r\n * @method remove\r\n * @param {string} key The key to remove the data for.\r\n */\r\nStore.prototype.remove = function (key) {\r\n\tcache[this.getStorageKey(key)] = null;\r\n\tthis.getStorageHandler().removeItem(this.getStorageKey(key));\r\n};\r\n\r\n/**\r\n * Removes all data and keys from the current namespace.\r\n * @method removeAll\r\n */\r\nStore.prototype.removeAll = function () {\r\n\tthis.listKeys().forEach(this.remove, this);\r\n};\r\n\r\n/**\r\n * Removes namespaced items from the cache so your next {{#crossLink \"Store/get\"}}Store#get{{/crossLink}} will be fresh from the storage.\r\n * @method freshen\r\n * @param {string} key The key to remove the cache data for.\r\n */\r\nStore.prototype.freshen = function (key) {\r\n\tvar keys = key ? [key] : this.listKeys();\r\n\tkeys.forEach(function (key) {\r\n\t\tcache[this.getStorageKey(key)] = null;\r\n\t}, this);\r\n};\r\n\r\n/**\r\n * Migrate data from a different namespace to current namespace.\r\n * @method migrate\r\n * @param {object}   migration                          The migration object.\r\n * @param {string}   migration.toKey                    The key name under your current namespace the old data should change to.\r\n * @param {string}   migration.fromNamespace            The old namespace that the old key belongs to.\r\n * @param {string}   migration.fromKey                  The old key name to migrate from.\r\n * @param {string}   [migration.fromStorageType]        The storage type to migrate from. Defaults to same type as where you are migrating to.\r\n * @param {boolean}  [migration.keepOldData=false]      Whether old data should be kept after it has been migrated.\r\n * @param {boolean}  [migration.overwriteNewData=false] Whether old data should overwrite currently stored data if it exists.\r\n * @param {function} [migration.transform]              The function to pass the old key data through before migrating.\r\n * @example\r\n * \r\n *     var Store = require('storage-wrapper');\r\n *     var store = new Store({\r\n *         namespace: 'myNewApp'\r\n *     });\r\n *\r\n *     // Migrate from the old app.\r\n *     store.migrate({\r\n *         toKey: 'new-key',\r\n *         fromNamespace: 'myOldApp',\r\n *         fromKey: 'old-key'\r\n *     });\r\n *     \r\n *     // Migrate from global data. Useful when moving from other storage wrappers or regular ol' `localStorage`.\r\n *     store.migrate({\r\n *         toKey: 'other-new-key',\r\n *         fromNamespace: '',\r\n *         fromKey: 'other-old-key-on-global'\r\n *     });\r\n *     \r\n *     // Migrate some JSON data that was stored as a string.\r\n *     store.migrate({\r\n *         toKey: 'new-json-key',\r\n *         fromNamespace: 'myOldApp',\r\n *         fromKey: 'old-json-key',\r\n *         // Try converting some old JSON data.\r\n *         transform: function (data) {\r\n *             try {\r\n *                 return JSON.parse(data);\r\n *             }\r\n *             catch (e) {\r\n *                 return data;\r\n *             }\r\n *         }\r\n *     });\r\n */\r\n\r\nStore.prototype.migrate = function (migration) {\r\n\t// Save our current namespace.\r\n\tvar toNamespace = this.getNamespace();\r\n\tvar toStorageType = this.getStorageType();\r\n\r\n\t// Create a temporary store to avoid changing namespace during actual get/sets.\r\n\tvar store = new Store({\r\n\t\tnamespace: toNamespace,\r\n\t\tstorageType: toStorageType\r\n\t});\r\n\r\n\tvar data = null;\r\n\r\n\t// Get data from old namespace.\r\n\tstore.setNamespace(migration.fromNamespace);\r\n\tif (typeof migration.fromStorageType !== 'undefined') {\r\n\t\tstore.setStorageType(migration.fromStorageType);\r\n\t}\r\n\tdata = store.get(migration.fromKey);\r\n\r\n\t// Remove old if needed.\r\n\tif (!migration.keepOldData) {\r\n\t\tstore.remove(migration.fromKey);\r\n\t}\r\n\t\r\n\t// No data, ignore this migration.\r\n\tif (data === null) {\r\n\t\treturn;\r\n\t}\r\n\r\n\t// Transform data if needed.\r\n\tif (typeof migration.transform === 'function') {\r\n\t\tdata = migration.transform(data);\r\n\t}\r\n\telse if (typeof migration.transform !== 'undefined') {\r\n\t\tthrow new Error('Invalid transform callback.');\r\n\t}\r\n\r\n\t// Go back to current namespace.\r\n\tstore.setNamespace(toNamespace);\r\n\tstore.setStorageType(toStorageType);\r\n\r\n\t// Only overwrite new data if it doesn't exist or it's requested.\r\n\tif (store.get(migration.toKey) === null || migration.overwriteNewData) {\r\n\t\tstore.set(migration.toKey, data);\r\n\t}\r\n};\r\n\r\n/**\r\n * Creates a substore that is nested in the current namespace.\r\n * @method createSubstore\r\n * @param  {string} namespace The substore's namespace.\r\n * @return {Store}            The substore.\r\n * @example\r\n * \r\n *     var Store = require('storage-wrapper');\r\n *     // Create main store.\r\n *     var store = new Store({\r\n *         namespace: 'myapp'\r\n *     });\r\n *\r\n *     // Create substore.\r\n *     var substore = store.createSubstore('things');\r\n *     substore.set('foo', 'bar');\r\n *\r\n *     substore.get('foo') === store.get('things:foo');\r\n *     // true\r\n */\r\nStore.prototype.createSubstore = function (namespace) {\r\n\treturn new Store({\r\n\t\tnamespace: this.getNamespace(true) + namespace,\r\n\t\tstorageType: this.getStorageType()\r\n\t});\r\n};\r\n\r\nmodule.exports = Store;\r\n","module.exports={\r\n  \"name\": \"twitch-chat-emotes\",\r\n  \"version\": \"2.1.5\",\r\n  \"homepage\": \"http://cletusc.github.io/Userscript--Twitch-Chat-Emotes/\",\r\n  \"bugs\": \"https://github.com/cletusc/Userscript--Twitch-Chat-Emotes/issues\",\r\n  \"author\": \"Ryan Chatham <ryan.b.chatham@gmail.com> (https://github.com/cletusc)\",\r\n  \"repository\": {\r\n    \"type\": \"git\",\r\n    \"url\": \"https://github.com/cletusc/Userscript--Twitch-Chat-Emotes.git\"\r\n  },\r\n  \"userscript\": {\r\n    \"name\": \"Twitch Chat Emotes\",\r\n    \"namespace\": \"#Cletus\",\r\n    \"version\": \"{{{pkg.version}}}\",\r\n    \"description\": \"Adds a button to Twitch that allows you to \\\"click-to-insert\\\" an emote.\",\r\n    \"copyright\": \"2011+, {{{pkg.author}}}\",\r\n    \"author\": \"{{{pkg.author}}}\",\r\n    \"icon\": \"http://www.gravatar.com/avatar.php?gravatar_id=6875e83aa6c563790cb2da914aaba8b3&r=PG&s=48&default=identicon\",\r\n    \"license\": [\r\n      \"MIT; http://opensource.org/licenses/MIT\",\r\n      \"CC BY-NC-SA 3.0; http://creativecommons.org/licenses/by-nc-sa/3.0/\"\r\n    ],\r\n    \"homepage\": \"{{{pkg.homepage}}}\",\r\n    \"supportURL\": \"{{{pkg.bugs}}}\",\r\n    \"contributionURL\": \"http://cletusc.github.io/Userscript--Twitch-Chat-Emotes/#donate\",\r\n    \"grant\": \"none\",\r\n    \"include\": [\r\n      \"http://*.twitch.tv/*\",\r\n      \"https://*.twitch.tv/*\"\r\n    ],\r\n    \"exclude\": [\r\n      \"http://api.twitch.tv/*\",\r\n      \"https://api.twitch.tv/*\",\r\n      \"http://tmi.twitch.tv/*\",\r\n      \"https://tmi.twitch.tv/*\",\r\n      \"http://*.twitch.tv/*/dashboard\",\r\n      \"https://*.twitch.tv/*/dashboard\",\r\n      \"http://chatdepot.twitch.tv/*\",\r\n      \"https://chatdepot.twitch.tv/*\",\r\n      \"http://im.twitch.tv/*\",\r\n      \"https://im.twitch.tv/*\",\r\n      \"http://platform.twitter.com/*\",\r\n      \"https://platform.twitter.com/*\",\r\n      \"http://www.facebook.com/*\",\r\n      \"https://www.facebook.com/*\"\r\n    ]\r\n  },\r\n  \"devDependencies\": {\r\n    \"browser-sync\": \"^1.3.2\",\r\n    \"browserify\": \"^5.9.1\",\r\n    \"generate-userscript-header\": \"^1.0.0\",\r\n    \"gulp\": \"^3.8.3\",\r\n    \"gulp-autoprefixer\": \"0.0.8\",\r\n    \"gulp-beautify\": \"1.1.0\",\r\n    \"gulp-changed\": \"^0.4.1\",\r\n    \"gulp-concat\": \"^2.2.0\",\r\n    \"gulp-conflict\": \"^0.1.2\",\r\n    \"gulp-css-base64\": \"^1.1.0\",\r\n    \"gulp-css2js\": \"^1.0.2\",\r\n    \"gulp-header\": \"^1.0.2\",\r\n    \"gulp-hogan-compile\": \"^0.2.1\",\r\n    \"gulp-minify-css\": \"^0.3.5\",\r\n    \"gulp-notify\": \"^1.4.1\",\r\n    \"gulp-rename\": \"^1.2.0\",\r\n    \"gulp-uglify\": \"^0.3.1\",\r\n    \"gulp-util\": \"^3.0.0\",\r\n    \"hogan.js\": \"^3.0.2\",\r\n    \"jquery-ui\": \"^1.10.5\",\r\n    \"jquery.scrollbar\": \"^0.2.7\",\r\n    \"pretty-hrtime\": \"^0.2.1\",\r\n    \"storage-wrapper\": \"cletusc/storage-wrapper#v0.1.1\",\r\n    \"vinyl-map\": \"^1.0.1\",\r\n    \"vinyl-source-stream\": \"^0.1.1\",\r\n    \"watchify\": \"^1.0.1\"\r\n  }\r\n}\r\n","var logger = require('./logger');\r\nvar api = {};\r\nvar ember = null;\r\nvar hookedFactories = {};\r\n\r\napi.getEmber = function () {\r\n\tif (ember) {\r\n\t\treturn ember;\r\n\t}\r\n\tif (window.App && window.App.__container__) {\r\n\t\tember = window.App.__container__;\r\n\t\treturn ember;\r\n\t}\r\n\treturn false;\r\n};\r\n\r\napi.isLoaded = function () {\r\n\treturn Boolean(api.getEmber());\r\n};\r\n\r\napi.lookup = function (lookupFactory) {\r\n\tif (!api.isLoaded()) {\r\n\t\tlogger.debug('Factory lookup failure, Ember not loaded.');\r\n\t\treturn false;\r\n\t}\r\n\treturn api.getEmber().lookup(lookupFactory);\r\n};\r\n\r\napi.hook = function (lookupFactory, activateCb, deactivateCb) {\r\n\tif (!api.isLoaded()) {\r\n\t\tlogger.debug('Factory hook failure, Ember not loaded.');\r\n\t\treturn false;\r\n\t}\r\n\tif (hookedFactories[lookupFactory]) {\r\n\t\tlogger.debug('Factory already hooked: ' + lookupFactory);\r\n\t\treturn true;\r\n\t}\r\n\tvar reopenOptions = {};\r\n\tvar factory = api.lookup(lookupFactory);\r\n\r\n\tif (!factory) {\r\n\t\tlogger.debug('Factory hook failure, factory not found: ' + lookupFactory);\r\n\t\treturn false;\r\n\t}\r\n\r\n\tif (activateCb) {\r\n\t\treopenOptions.activate = function () {\r\n\t\t\tthis._super();\r\n\t\t\tactivateCb.call(this);\r\n\t\t\tlogger.debug('Hook run on activate: ' + lookupFactory);\r\n\t\t};\r\n\t}\r\n\tif (deactivateCb) {\r\n\t\treopenOptions.deactivate = function () {\r\n\t\t\tthis._super();\r\n\t\t\tdeactivateCb.call(this);\r\n\t\t\tlogger.debug('Hook run on deactivate: ' + lookupFactory);\r\n\t\t};\r\n\t}\r\n\r\n\ttry {\r\n\t\tfactory.reopen(reopenOptions);\r\n\t\thookedFactories[lookupFactory] = true;\r\n\t\tlogger.debug('Factory hooked: ' + lookupFactory);\r\n\t\treturn true;\r\n\t}\r\n\tcatch (err) {\r\n\t\tlogger.debug('Factory hook failure, unexpected error: ' + lookupFactory);\r\n\t\tlogger.debug(err);\r\n\t\treturn false;\r\n\t}\r\n};\r\n\r\napi.get = function (lookupFactory, property) {\r\n\tif (!api.isLoaded()) {\r\n\t\tlogger.debug('Factory get failure, Ember not loaded.');\r\n\t\treturn false;\r\n\t}\r\n\tvar properties = property.split('.');\r\n\tvar getter = api.lookup(lookupFactory);\r\n\r\n\tproperties.some(function (property) {\r\n\t\t// If getter fails, just exit, otherwise, keep looping.\r\n\t\tif (getter == null || typeof getter === 'undefined') {\r\n\t\t\tgetter = null;\r\n\t\t\treturn true;\r\n\t\t}\r\n\t\tif (getter[property] == null || typeof getter[property] === 'undefined') {\r\n\t\t\tgetter = null;\r\n\t\t\treturn true;\r\n\t\t}\r\n\t\tif (typeof getter.get === 'function') {\r\n\t\t\tgetter = getter.get(property);\r\n\t\t\tif (getter == null || typeof getter === 'undefined') {\r\n\t\t\t\tgetter = null;\r\n\t\t\t\treturn true;\r\n\t\t\t}\r\n\t\t\treturn false;\r\n\t\t}\r\n\t\tgetter = getter[property];\r\n\t});\r\n\r\n\treturn getter;\r\n};\r\n\r\nmodule.exports = api;\r\n","var storage = require('./storage');\r\nvar logger = require('./logger');\r\nvar ui = require('./ui');\r\nvar api = {};\r\nvar emoteStore = new EmoteStore();\r\nvar $ = window.jQuery;\r\n\r\n/**\r\n * The entire emote storing system.\r\n */\r\nfunction EmoteStore() {\r\n\tvar getters = {};\r\n\tvar nativeEmotes = {};\r\n\tvar hasInitialized = false;\r\n\r\n\t/**\r\n\t * Get a list of usable emoticons.\r\n\t * @param  {function} [filters]       A filter method to limit what emotes are returned. Passed to Array#filter.\r\n\t * @param  {function|string} [sortBy] How the emotes should be sorted. `function` will be passed to sort via Array#sort. `'channel'` sorts by channel name, globals first. All other values (or omitted) sort alphanumerically.\r\n\t * @param  {string} [returnType]      `'object'` will return in object format, e.g. `{'Kappa': Emote(...), ...}`. All other values (or omitted) return an array format, e.g. `[Emote(...), ...]`.\r\n\t * @return {object|array}             See `returnType` param.\r\n\t */\r\n\tthis.getEmotes = function (filters, sortBy, returnType) {\r\n\t\tvar twitchApi = require('./twitch-api');\r\n\r\n\t\t// Get native emotes.\r\n\t\tvar emotes = $.extend({}, nativeEmotes);\r\n\r\n\t\t// Parse the custom emotes provided by third party addons.\r\n\t\tObject.keys(getters).forEach(function (getterName) {\r\n\t\t\t// Try the getter.\r\n\t\t\tvar results = null;\r\n\t\t\ttry {\r\n\t\t\t\tresults = getters[getterName]();\r\n\t\t\t}\r\n\t\t\tcatch (err) {\r\n\t\t\t\tlogger.debug('Emote getter `' + getterName + '` failed unexpectedly, skipping.', err.toString());\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\t\t\tif (!Array.isArray(results)) {\r\n\t\t\t\tlogger.debug('Emote getter `' + getterName + '` must return an array, skipping.');\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\r\n\t\t\t// Override natives and previous getters.\r\n\t\t\tresults.forEach(function (emote) {\r\n\t\t\t\ttry {\r\n\t\t\t\t\t// Create the emote.\r\n\t\t\t\t\tvar instance = new Emote(emote);\r\n\r\n\t\t\t\t\t// Force the getter.\r\n\t\t\t\t\tinstance.setGetterName(getterName);\r\n\r\n\t\t\t\t\t// Force emotes without channels to the getter's name.\r\n\t\t\t\t\tif (!emote.channel) {\r\n\t\t\t\t\t\tinstance.setChannelName(getterName);\r\n\t\t\t\t\t}\r\n\r\n\t\t\t\t\t// Add/override it.\r\n\t\t\t\t\temotes[instance.getText()] = instance;\r\n\t\t\t\t}\r\n\t\t\t\tcatch (err) {\r\n\t\t\t\t\tlogger.debug('Emote parsing for getter `' + getterName + '` failed, skipping.', err.toString(), emote);\r\n\t\t\t\t}\r\n\t\t\t});\r\n\t\t});\r\n\r\n\t\t// Covert to array.\r\n\t\temotes = Object.keys(emotes).map(function (emote) {\r\n\t\t\treturn emotes[emote];\r\n\t\t});\r\n\r\n\t\t// Filter results.\r\n\t\tif (typeof filters === 'function') {\r\n\t\t\temotes = emotes.filter(filters);\r\n\t\t}\r\n\t\t\r\n\t\t// Return as an object if requested.\r\n\t\tif (returnType === 'object') {\r\n\t\t\tvar asObject = {};\r\n\t\t\temotes.forEach(function (emote) {\r\n\t\t\t\tasObject[emote.getText()] = emote;\r\n\t\t\t});\r\n\t\t\treturn asObject;\r\n\t\t}\r\n\r\n\t\t// Sort results.\r\n\t\tif (typeof sortBy === 'function') {\r\n\t\t\temotes.sort(sortBy);\r\n\t\t}\r\n\t\telse if (sortBy === 'channel') {\r\n\t\t\temotes.sort(sorting.allEmotesCategory);\r\n\t\t}\r\n\t\telse {\r\n\t\t\temotes.sort(sorting.byText);\r\n\t\t}\r\n\r\n\t\t// Return the emotes in array format.\r\n\t\treturn emotes;\r\n\t};\r\n\r\n\t/**\r\n\t * Registers a 3rd party emote hook.\r\n\t * @param  {string} name   The name of the 3rd party registering the hook.\r\n\t * @param  {function} getter The function called when looking for emotes. Must return an array of emote objects, e.g. `[emote, ...]`. See Emote class.\r\n\t */\r\n\tthis.registerGetter = function (name, getter) {\r\n\t\tif (typeof name !== 'string') {\r\n\t\t\tthrow new Error('Name must be a string.');\r\n\t\t}\r\n\t\tif (getters[name]) {\r\n\t\t\tthrow new Error('Getter already exists.');\r\n\t\t}\r\n\t\tif (typeof getter !== 'function') {\r\n\t\t\tthrow new Error('Getter must be a function.');\r\n\t\t}\r\n\t\tlogger.debug('Getter registered: ' + name);\r\n\t\tgetters[name] = getter;\r\n\t\tui.updateEmotes();\r\n\t};\r\n\r\n\t/**\r\n\t * Registers a 3rd party emote hook.\r\n\t * @param  {string} name   The name of the 3rd party hook to deregister.\r\n\t */\r\n\tthis.deregisterGetter = function (name) {\r\n\t\tlogger.debug('Getter unregistered: ' + name);\r\n\t\tdelete getters[name];\r\n\t\tui.updateEmotes();\r\n\t};\r\n\r\n\t/**\r\n\t * Initializes the raw data from the API endpoints. Should be called at load and/or whenever the API may have changed. Populates internal objects with updated data.\r\n\t */\r\n\tthis.init = function () {\r\n\t\tif (hasInitialized) {\r\n\t\t\tlogger.debug('Already initialized EmoteStore, stopping init.');\r\n\t\t\treturn;\r\n\t\t}\r\n\r\n\t\tlogger.debug('Starting initialization.');\r\n\r\n\t\tvar twitchApi = require('./twitch-api');\r\n\t\tvar self = this;\r\n\r\n\t\t// Hash of emote set to forced channel.\r\n\t\tvar forcedSetsToChannels = {\r\n\t\t\t// Globals.\r\n\t\t\t0: '~global',\r\n\t\t\t// Bubble emotes.\r\n\t\t\t33: 'turbo',\r\n\t\t\t// Monkey emotes.\r\n\t\t\t42: 'turbo',\r\n\t\t\t// Hidden turbo emotes.\r\n\t\t\t457: 'turbo',\r\n\t\t\t793: 'turbo',\r\n\t\t\t19151: 'twitch_prime',\r\n\t\t\t19194: 'twitch_prime'\r\n\r\n\t\t};\r\n\r\n\t\tlogger.debug('Initializing emote set change listener.');\r\n\r\n\t\ttwitchApi.getEmotes(function (emoteSets) {\r\n\t\t\tlogger.debug('Parsing emote sets.');\r\n\r\n\t\t\tObject.keys(emoteSets).forEach(function (set) {\r\n\t\t\t\tvar emotes = emoteSets[set];\r\n\t\t\t\tset = Number(set);\r\n\t\t\t\temotes.forEach(function (emote) {\r\n\t\t\t\t\t// Set some required info.\r\n\t\t\t\t\temote.url = '//static-cdn.jtvnw.net/emoticons/v1/' + emote.id + '/1.0';\r\n\t\t\t\t\temote.text = getEmoteFromRegEx(emote.code);\r\n\t\t\t\t\temote.set = set;\r\n\r\n\t\t\t\t\t// Hardcode the channels of certain sets.\r\n\t\t\t\t\tif (forcedSetsToChannels[set]) {\r\n\t\t\t\t\t\temote.channel = forcedSetsToChannels[set];\r\n\t\t\t\t\t}\r\n\r\n\t\t\t\t\tvar instance = new Emote(emote);\r\n\r\n\t\t\t\t\t// Save the emote for use later.\r\n\t\t\t\t\tnativeEmotes[emote.text] = instance;\r\n\t\t\t\t});\r\n\t\t\t});\r\n\r\n\t\t\tlogger.debug('Loading subscription data.');\r\n\r\n\t\t\t// Get active subscriptions to find the channels.\r\n\t\t\ttwitchApi.getTickets(function (tickets) {\r\n\t\t\t\t// Instances from each channel to preload channel data.\r\n\t\t\t\tvar deferredChannelGets = {};\r\n\r\n\t\t\t\tlogger.debug('Tickets loaded from the API.', tickets);\r\n\t\t\t\ttickets.forEach(function (ticket) {\r\n\t\t\t\t\tvar product = ticket.product;\r\n\t\t\t\t\tvar channel = product.owner_name || product.short_name;\r\n\r\n\t\t\t\t\t// Get subscriptions with emotes only.\r\n\t\t\t\t\tif (!product.emoticons || !product.emoticons.length) {\r\n\t\t\t\t\t\treturn;\r\n\t\t\t\t\t}\r\n\t\t\t\t\t\r\n\t\t\t\t\t// Set the channel on the emotes.\r\n\t\t\t\t\tproduct.emoticons.forEach(function (emote) {\r\n\t\t\t\t\t\tvar instance = nativeEmotes[getEmoteFromRegEx(emote.regex)];\r\n\t\t\t\t\t\tinstance.setChannelName(channel);\r\n\r\n\t\t\t\t\t\t// Save instance for later, but only one instance per channel.\r\n\t\t\t\t\t\tif (!deferredChannelGets[channel]) {\r\n\t\t\t\t\t\t\tdeferredChannelGets[channel] = instance;\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t});\r\n\t\t\t\t});\r\n\r\n\t\t\t\t// Preload channel data.\r\n\t\t\t\tObject.keys(deferredChannelGets).forEach(function (key) {\r\n\t\t\t\t\tvar instance = deferredChannelGets[key];\r\n\t\t\t\t\tinstance.getChannelBadge();\r\n\t\t\t\t\tinstance.getChannelDisplayName();\r\n\t\t\t\t});\r\n\t\t\t\tui.updateEmotes();\r\n\t\t\t});\r\n\t\t\tui.updateEmotes();\r\n\t\t});\r\n\r\n\t\thasInitialized = true;\r\n\t\tlogger.debug('Finished EmoteStore initialization.');\r\n\t};\r\n};\r\n\r\n/**\r\n * Gets a specific emote, if available.\r\n * @param  {string}     text The text of the emote to get.\r\n * @return {Emote|null}      The Emote instance of the emote or `null` if it couldn't be found.\r\n */\r\nEmoteStore.prototype.getEmote = function (text) {\r\n\treturn this.getEmotes(null, null, 'object')[text] || null;\r\n};\r\n\r\n/**\r\n * Emote object.\r\n * @param {object} details              Object describing the emote.\r\n * @param {string} details.text         The text to use in the chat box when emote is clicked.\r\n * @param {string} details.url          The URL of the image for the emote.\r\n * @param {string} [details.badge]      The URL of the badge for the emote.\r\n * @param {string} [details.channel]    The channel the emote should be categorized under.\r\n * @param {string} [details.getterName] The 3rd party getter that registered the emote. Used internally only.\r\n */\r\nfunction Emote(details) {\r\n\tvar text = null;\r\n\tvar url = null;\r\n\tvar getterName = null;\r\n\tvar channel = {\r\n\t\tname: null,\r\n\t\tdisplayName: null,\r\n\t\tbadge: null\r\n\t};\r\n\r\n\t/**\r\n\t * Gets the text of the emote.\r\n\t * @return {string} The emote text.\r\n\t */\r\n\tthis.getText = function () {\r\n\t\treturn text;\r\n\t};\r\n\r\n\t/**\r\n\t * Sets the text of the emote.\r\n\t * @param {string} theText The text to set.\r\n\t */\r\n\tthis.setText = function (theText) {\r\n\t\tif (typeof theText !== 'string' || theText.length < 1) {\r\n\t\t\tthrow new Error('Invalid text');\r\n\t\t}\r\n\t\ttext = theText;\r\n\t};\r\n\r\n\t/**\r\n\t * Gets the getter name this emote belongs to.\r\n\t * @return {string} The getter's name.\r\n\t */\r\n\tthis.getGetterName = function () {\r\n\t\treturn getterName;\r\n\t};\r\n\r\n\t/**\r\n\t * Sets the getter name this emote belongs to.\r\n\t * @param {string} theGetterName The getter's name.\r\n\t */\r\n\tthis.setGetterName = function (theGetterName) {\r\n\t\tif (typeof theGetterName !== 'string' || theGetterName.length < 1) {\r\n\t\t\tthrow new Error('Invalid getter name');\r\n\t\t}\r\n\t\tgetterName = theGetterName;\r\n\t};\r\n\r\n\t/**\r\n\t * Gets the emote's image URL.\r\n\t * @return {string} The emote image URL.\r\n\t */\r\n\tthis.getUrl = function () {\r\n\t\treturn url;\r\n\t};\r\n\t/**\r\n\t * Sets the emote's image URL.\r\n\t * @param {string} theUrl The image URL to set.\r\n\t */\r\n\tthis.setUrl = function (theUrl) {\r\n\t\tif (typeof theUrl !== 'string' || theUrl.length < 1) {\r\n\t\t\tthrow new Error('Invalid URL');\r\n\t\t}\r\n\t\turl = theUrl;\r\n\t};\r\n\r\n\t/**\r\n\t * Gets the emote's channel name.\r\n\t * @return {string} The emote's channel or an empty string if it doesn't have one.\r\n\t */\r\n\tthis.getChannelName = function () {\r\n\t\tif (!channel.name) {\r\n\t\t\tchannel.name = storage.channelNames.get(this.getText());\r\n\t\t}\r\n\t\treturn channel.name || '';\r\n\t};\r\n\t/**\r\n\t * Sets the emote's channel name.\r\n\t * @param {string} theChannel The channel name to set.\r\n\t */\r\n\tthis.setChannelName = function (theChannel) {\r\n\t\tif (typeof theChannel !== 'string' || theChannel.length < 1) {\r\n\t\t\tthrow new Error('Invalid channel');\r\n\t\t}\r\n\r\n\t\t// Only save the channel to storage if it's dynamic.\r\n\t\tif (theChannel !== '~global' && theChannel !== 'turbo' && theChannel !== 'twitch_prime') {\r\n\t\t\tstorage.channelNames.set(this.getText(), theChannel);\r\n\t\t}\r\n\t\tchannel.name = theChannel;\r\n\t};\r\n\r\n\t/**\r\n\t * Gets the emote channel's badge image URL.\r\n\t * @return {string|null} The URL of the badge image for the emote's channel or `null` if it doesn't have a channel.\r\n\t */\r\n\tthis.getChannelBadge = function () {\r\n\t\tvar twitchApi = require('./twitch-api');\r\n\t\tvar channelName = this.getChannelName();\r\n\t\tvar defaultBadge = '//static-cdn.jtvnw.net/jtv_user_pictures/subscriber-star.png';\r\n\r\n\t\t// No channel.\r\n\t\tif (!channelName) {\r\n\t\t\treturn null;\r\n\t\t}\r\n\r\n\t\t// Give globals a default badge.\r\n\t\tif (channelName === '~global') {\r\n\t\t\treturn '/favicon.ico';\r\n\t\t}\r\n\r\n\t\t// Already have one preset.\r\n\t\tif (channel.badge) {\r\n\t\t\treturn channel.badge;\r\n\t\t}\r\n\r\n\t\t// Check storage.\r\n\t\tchannel.badge = storage.badges.get(channelName);\r\n\t\tif (channel.badge !== null) {\r\n\t\t\treturn channel.badge;\r\n\t\t}\r\n\r\n\t\t// Set default until API returns something.\r\n\t\tchannel.badge = defaultBadge;\r\n\r\n\t\t// Get from API.\r\n\t\tlogger.debug('Getting fresh badge for: ' + channelName);\r\n\t\ttwitchApi.getBadges(channelName, function (badges) {\r\n\t\t\tvar badge = null;\r\n\r\n\t\t\t// Save turbo badge while we are here.\r\n\t\t\tif (badges.turbo && badges.turbo.image) {\r\n\t\t\t\tbadge = badges.turbo.image;\r\n\t\t\t\tstorage.badges.set('turbo', badge, 86400000);\r\n\r\n\t\t\t\t// Turbo is actually what we wanted, so we are done.\r\n\t\t\t\tif (channelName === 'turbo') {\r\n\t\t\t\t\tchannel.badge = badge;\r\n\t\t\t\t\treturn;\r\n\t\t\t\t}\r\n\t\t\t}\r\n\r\n\t\t\t// Save turbo badge while we are here.\r\n\t\t\tif (badges.premium && badges.premium.image) {\r\n\t\t\t\tbadge = badges.premium.image;\r\n\t\t\t\tstorage.badges.set('twitch_prime', badge, 86400000);\r\n\r\n\t\t\t\t// Turbo is actually what we wanted, so we are done.\r\n\t\t\t\tif (channelName === 'twitch_prime') {\r\n\t\t\t\t\tchannel.badge = badge;\r\n\t\t\t\t\treturn;\r\n\t\t\t\t}\r\n\t\t\t}\r\n\r\n\t\t\t// Save subscriber badge in storage.\r\n\t\t\tif (badges.subscriber && badges.subscriber.image) {\r\n\t\t\t\tchannel.badge = badges.subscriber.image;\r\n\t\t\t\tstorage.badges.set(channelName, channel.badge, 86400000);\r\n\t\t\t\tui.updateEmotes();\r\n\t\t\t}\r\n\t\t\t// No subscriber badge.\r\n\t\t\telse {\r\n\t\t\t\tchannel.badge = defaultBadge;\r\n\t\t\t\tlogger.debug('Failed to get subscriber badge for: ' + channelName);\r\n\t\t\t}\r\n\t\t});\r\n\t\t\r\n\t\treturn channel.badge || defaultBadge;\r\n\t};\r\n\r\n\t/**\r\n\t * Sets the emote's channel badge image URL.\r\n\t * @param {string} theBadge The badge image URL to set.\r\n\t */\r\n\tthis.setChannelBadge = function (theBadge) {\r\n\t\tif (typeof theBadge !== 'string' || theBadge.length < 1) {\r\n\t\t\tthrow new Error('Invalid badge');\r\n\t\t}\r\n\t\tchannel.badge = theBadge;\r\n\t};\r\n\r\n\t/**\r\n\t * Get a channel's display name.\r\n\t * @return {string} The channel's display name. May be equivalent to the channel the first time the API needs to be called.\r\n\t */\r\n\tthis.getChannelDisplayName = function () {\r\n\t\tvar twitchApi = require('./twitch-api');\r\n\t\tvar channelName = this.getChannelName();\r\n\t\tvar self = this;\r\n\r\n\t\tvar forcedChannelToDisplayNames = {\r\n\t\t\t'~global': 'Global',\r\n\t\t\t'turbo': 'Twitch Turbo',\r\n\t\t\t'twitch_prime': 'Twitch Prime'\r\n\t\t};\r\n\r\n\t\t// No channel.\r\n\t\tif (!channelName) {\r\n\t\t\treturn '';\r\n\t\t}\r\n\r\n\t\t// Forced display name.\r\n\t\tif (forcedChannelToDisplayNames[channelName]) {\r\n\t\t\treturn forcedChannelToDisplayNames[channelName];\r\n\t\t}\r\n\r\n\t\t// Already have one preset.\r\n\t\tif (channel.displayName) {\r\n\t\t\treturn channel.displayName;\r\n\t\t}\r\n\r\n\t\t// Look for obvious bad channel names that shouldn't hit the API or storage. Use channel name instead.\r\n\t\tif (/[^a-z0-9_]/.test(channelName)) {\r\n\t\t\tlogger.debug('Unable to get display name due to obvious non-standard channel name for: ' + channelName);\r\n\t\t\treturn channelName;\r\n\t\t}\r\n\r\n\t\t// Check storage.\r\n\t\tchannel.displayName = storage.displayNames.get(channelName);\r\n\t\tif (channel.displayName !== null) {\r\n\t\t\treturn channel.displayName;\r\n\t\t}\r\n\t\t// Get from API.\r\n\t\telse {\r\n\t\t\t// Set default until API returns something.\r\n\t\t\tchannel.displayName = channelName;\r\n\r\n\t\t\tlogger.debug('Getting fresh display name for: ' + channelName);\r\n\t\t\ttwitchApi.getUser(channelName, function (user) {\r\n\t\t\t\tif (!user || !user.display_name) {\r\n\t\t\t\t\tlogger.debug('Failed to get display name for: ' + channelName);\r\n\t\t\t\t\treturn;\r\n\t\t\t\t}\r\n\r\n\t\t\t\t// Save it.\r\n\t\t\t\tself.setChannelDisplayName(user.display_name);\r\n\t\t\t\tui.updateEmotes();\r\n\t\t\t});\r\n\t\t}\r\n\t\t\r\n\t\treturn channel.displayName;\r\n\t};\r\n\r\n\t/**\r\n\t * Sets the emote's channel badge image URL.\r\n\t * @param {string} theBadge The badge image URL to set.\r\n\t */\r\n\tthis.setChannelDisplayName = function (displayName) {\r\n\t\tif (typeof displayName !== 'string' || displayName.length < 1) {\r\n\t\t\tthrow new Error('Invalid displayName');\r\n\t\t}\r\n\t\tchannel.displayName = displayName;\r\n\t\tstorage.displayNames.set(this.getChannelName(), displayName, 86400000);\r\n\t};\r\n\r\n\t/**\r\n\t * Initialize the details.\r\n\t */\r\n\t\r\n\t// Required fields.\r\n\tthis.setText(details.text);\r\n\tthis.setUrl(details.url);\r\n\r\n\t// Optional fields.\r\n\tif (details.getterName) {\r\n\t\tthis.setGetterName(details.getterName);\r\n\t}\r\n\tif (details.channel) {\r\n\t\tthis.setChannelName(details.channel);\r\n\t}\r\n\tif (details.channelDisplayName) {\r\n\t\tthis.setChannelDisplayName(details.channelDisplayName);\r\n\t}\r\n\tif (details.badge) {\r\n\t\tthis.setChannelBadge(details.badge);\r\n\t}\r\n};\r\n\r\n/**\r\n * State changers.\r\n */\r\n\r\n/**\r\n * Toggle whether an emote should be a favorite.\r\n * @param {boolean} [force] `true` forces the emote to be a favorite, `false` forces the emote to not be a favorite.\r\n */\r\nEmote.prototype.toggleFavorite = function (force) {\r\n\tif (typeof force !== 'undefined') {\r\n\t\tstorage.starred.set(this.getText(), !!force);\r\n\t\treturn;\r\n\t}\r\n\tstorage.starred.set(this.getText(), !this.isFavorite());\r\n};\r\n\r\n/**\r\n * Toggle whether an emote should be visible out of editing mode.\r\n * @param {boolean} [force] `true` forces the emote to be visible, `false` forces the emote to be hidden.\r\n */\r\nEmote.prototype.toggleVisibility = function (force) {\r\n\tif (typeof force !== 'undefined') {\r\n\t\tstorage.visibility.set(this.getText(), !!force);\r\n\t\treturn;\r\n\t}\r\n\tstorage.visibility.set(this.getText(), !this.isVisible());\r\n};\r\n\r\n/**\r\n * State getters.\r\n */\r\n\r\n/**\r\n * Whether the emote is from a 3rd party.\r\n * @return {boolean} Whether the emote is from a 3rd party.\r\n */\r\nEmote.prototype.isThirdParty = function () {\r\n\treturn !!this.getGetterName();\r\n};\r\n\r\n/**\r\n * Whether the emote was favorited.\r\n * @return {boolean} Whether the emote was favorited.\r\n */\r\nEmote.prototype.isFavorite = function () {\r\n\treturn storage.starred.get(this.getText(), false);\r\n};\r\n\r\n/**\r\n * Whether the emote is visible outside of editing mode.\r\n * @return {boolean} Whether the emote is visible outside of editing mode.\r\n */\r\nEmote.prototype.isVisible = function () {\r\n\treturn storage.visibility.get(this.getText(), true);\r\n};\r\n\r\n/**\r\n * Whether the emote is considered a simple smiley.\r\n * @return {boolean} Whether the emote is considered a simple smiley.\r\n */\r\nEmote.prototype.isSmiley = function () {\r\n\t// The basic smiley emotes.\r\n\tvar emotes = [':(', ':)', ':/', ':\\\\', ':D', ':o', ':p', ':z', ';)', ';p', '<3', '>(', 'B)', 'R)', 'o_o', 'O_O', '#/', ':7', ':>', ':S', '<]'];\r\n\treturn emotes.indexOf(this.getText()) !== -1;\r\n};\r\n\r\n/**\r\n * Property getters/setters.\r\n */\r\n\r\n/**\r\n * Gets the usable emote text from a regex.\r\n */\r\nfunction getEmoteFromRegEx(regex) {\r\n\tif (typeof regex === 'string') {\r\n\t\tregex = new RegExp(regex);\r\n\t}\r\n\tif (!regex) {\r\n\t\tthrow new Error('`regex` must be a RegExp string or object.');\r\n\t}\r\n\r\n\treturn decodeURI(regex.source)\r\n\r\n\t\t// Replace HTML entity brackets with actual brackets.\r\n\t\t.replace('&gt\\\\;', '>')\r\n\t\t.replace('&lt\\\\;', '<')\r\n\r\n\t\t// Remove negative groups.\r\n\t\t//\r\n\t\t// /\r\n\t\t//   \\(\\?!              // (?!\r\n\t\t//   [^)]*              // any amount of characters that are not )\r\n\t\t//   \\)                 // )\r\n\t\t// /g\r\n\t\t.replace(/\\(\\?![^)]*\\)/g, '')\r\n\r\n\t\t// Pick first option from a group.\r\n\t\t//\r\n\t\t// /\r\n\t\t//   \\(                 // (\r\n\t\t//   ([^|])*            // any amount of characters that are not |\r\n\t\t//   \\|?                // an optional | character\r\n\t\t//   [^)]*              // any amount of characters that are not )\r\n\t\t//   \\)                 // )\r\n\t\t// /g\r\n\t\t.replace(/\\(([^|])*\\|?[^)]*\\)/g, '$1')\r\n\r\n\t\t// Pick first character from a character group.\r\n\t\t//\r\n\t\t// /\r\n\t\t//   \\[                 // [\r\n\t\t//   ([^|\\]\\[])*        // any amount of characters that are not |, [, or ]\r\n\t\t//   \\|?                // an optional | character\r\n\t\t//   [^\\]]*             // any amount of characters that are not [, or ]\r\n\t\t//   \\]                 // ]\r\n\t\t// /g\r\n\t\t.replace(/\\[([^|\\]\\[])*\\|?[^\\]\\[]*\\]/g, '$1')\r\n\r\n\t\t// Remove optional characters.\r\n\t\t//\r\n\t\t// /\r\n\t\t//   [^\\\\]              // any character that is not \\\r\n\t\t//   \\?                 // ?\r\n\t\t// /g\r\n\t\t.replace(/[^\\\\]\\?/g, '')\r\n\r\n\t\t// Remove boundaries at beginning and end.\r\n\t\t.replace(/^\\\\b|\\\\b$/g, '') \r\n\r\n\t\t// Unescape only single backslash, not multiple.\r\n\t\t//\r\n\t\t// /\r\n\t\t//   \\\\                 // \\\r\n\t\t//   (?!\\\\)             // look-ahead, any character that isn't \\\r\n\t\t// /g\r\n\t\t.replace(/\\\\(?!\\\\)/g, '');\r\n}\r\n\r\nvar sorting = {};\r\n\r\n/**\r\n * Sort by alphanumeric in this order: symbols -> numbers -> AaBb... -> numbers\r\n */\r\nsorting.byText = function (a, b) {\r\n\ttextA = a.getText().toLowerCase();\r\n\ttextB = b.getText().toLowerCase();\r\n\r\n\tif (textA < textB) {\r\n\t\treturn -1;\r\n\t}\r\n\tif (textA > textB) {\r\n\t\treturn 1;\r\n\t}\r\n\treturn 0;\r\n}\r\n\r\n/**\r\n * Basic smilies before non-basic smilies.\r\n */\r\nsorting.bySmiley = function (a, b) {\r\n\tif (a.isSmiley() &&\t!b.isSmiley()) {\r\n\t\treturn -1;\r\n\t}\r\n\tif (b.isSmiley() &&\t!a.isSmiley()) {\r\n\t\treturn 1;\r\n\t}\r\n\treturn 0;\r\n};\r\n\r\n/**\r\n * Globals before subscription emotes, subscriptions in alphabetical order.\r\n */\r\nsorting.byChannelName = function (a, b) {\r\n\tvar channelA = a.getChannelName();\r\n\tvar channelB = b.getChannelName();\r\n\r\n\t// Both don't have channels.\r\n\tif (!channelA && !channelB) {\r\n\t\treturn 0;\r\n\t}\r\n\r\n\t// \"A\" has channel, \"B\" doesn't.\r\n\tif (channelA && !channelB) {\r\n\t\treturn 1;\r\n\t}\r\n\t// \"B\" has channel, \"A\" doesn't.\r\n\tif (channelB && !channelA) {\r\n\t\treturn -1;\r\n\t}\r\n\r\n\tchannelA = channelA.toLowerCase();\r\n\tchannelB = channelB.toLowerCase();\r\n\r\n\tif (channelA < channelB) {\r\n\t\treturn -1;\r\n\t}\r\n\tif (channelB > channelA) {\r\n\t\treturn 1;\r\n\t}\r\n\r\n\t// All the same\r\n\treturn 0;\r\n};\r\n\r\n/**\r\n * The general sort order for the all emotes category.\r\n * Smileys -> Channel grouping -> alphanumeric\r\n */\r\nsorting.allEmotesCategory = function (a, b) {\r\n\tvar bySmiley = sorting.bySmiley(a, b);\r\n\tvar byChannelName  = sorting.byChannelName(a, b);\r\n\tvar byText = sorting.byText(a, b);\r\n\r\n\tif (bySmiley !== 0) {\r\n\t\treturn bySmiley;\r\n\t}\r\n\tif (byChannelName !== 0) {\r\n\t\treturn byChannelName;\r\n\t}\r\n\treturn byText;\r\n};\r\n\r\nmodule.exports = emoteStore;\r\n","var api = {};\r\nvar instance = '[instance ' + (Math.floor(Math.random() * (999 - 100)) + 100) + '] ';\r\nvar prefix = '[Emote Menu] ';\r\nvar storage = require('./storage');\r\n\r\napi.log = function () {\r\n\tif (typeof console.log === 'undefined') {\r\n\t\treturn;\r\n\t}\r\n\targuments = [].slice.call(arguments).map(function (arg) {\r\n\t\tif (typeof arg !== 'string') {\r\n\t\t\treturn JSON.stringify(arg);\r\n\t\t}\r\n\t\treturn arg;\r\n\t});\r\n\tif (storage.global.get('debugMessagesEnabled', false)) {\r\n\t\targuments.unshift(instance);\r\n\t}\r\n\targuments.unshift(prefix);\r\n\tconsole.log.apply(console, arguments);\r\n};\r\n\r\napi.debug = function () {\r\n\tif (!storage.global.get('debugMessagesEnabled', false)) {\r\n\t\treturn;\r\n\t}\r\n\targuments = [].slice.call(arguments);\r\n\targuments.unshift('[DEBUG] ');\r\n\tapi.log.apply(null, arguments);\r\n}\r\n\r\nmodule.exports = api;\r\n","var storage = require('./storage');\r\nvar logger = require('./logger');\r\nvar emotes = require('./emotes');\r\nvar api = {};\r\n\r\napi.toggleDebug = function (forced) {\r\n\tif (typeof forced === 'undefined') {\r\n\t\tforced = !storage.global.get('debugMessagesEnabled', false);\r\n\t}\r\n\telse {\r\n\t\tforced = !!forced;\r\n\t}\r\n\tstorage.global.set('debugMessagesEnabled', forced);\r\n\tlogger.log('Debug messages are now ' + (forced ? 'enabled' : 'disabled'));\r\n};\r\n\r\napi.registerEmoteGetter = emotes.registerGetter;\r\napi.deregisterEmoteGetter = emotes.deregisterGetter;\r\n\r\nmodule.exports = api;\r\n","var Store = require('storage-wrapper');\r\nvar storage = {};\r\n\r\n// General storage.\r\nstorage.global = new Store({\r\n\tnamespace: 'emote-menu-for-twitch'\r\n});\r\n\r\n// Emote visibility storage.\r\nstorage.visibility = storage.global.createSubstore('visibility');\r\n// Emote starred storage.\r\nstorage.starred = storage.global.createSubstore('starred');\r\n// Display name storage.\r\nstorage.displayNames = storage.global.createSubstore('displayNames');\r\n// Channel name storage.\r\nstorage.channelNames = storage.global.createSubstore('channelNames');\r\n// Badges storage.\r\nstorage.badges = storage.global.createSubstore('badges');\r\n\r\nmodule.exports = storage;\r\n","var templates = require('../../build/templates');\r\n\r\nmodule.exports = (function () {\r\n\tvar data = {};\r\n\tvar key = null;\r\n\r\n\t// Convert templates to their shorter \"render\" form.\r\n\tfor (key in templates) {\r\n\t\tif (!templates.hasOwnProperty(key)) {\r\n\t\t\tcontinue;\r\n\t\t}\r\n\t\tdata[key] = render(key);\r\n\t}\r\n\r\n\t// Shortcut the render function. All templates will be passed in as partials by default.\r\n\tfunction render(template) {\r\n\t\ttemplate = templates[template];\r\n\t\treturn function (context, partials, indent) {\r\n\t\t\treturn template.render(context, partials || templates, indent);\r\n\t\t};\r\n\t}\r\n\r\n\treturn data;\r\n})();\r\n","var twitchApi = window.Twitch.api;\r\nvar jQuery = window.jQuery;\r\nvar logger = require('./logger');\r\nvar api = {};\r\n\r\napi.getBadges = function (username, callback) {\r\n\tif (\r\n\t\t[\r\n\t\t\t'~global',\r\n\t\t\t'turbo',\r\n\t\t\t'twitch_prime'\r\n\t\t].indexOf(username) > -1\r\n\t) {\r\n\t\tif (!jQuery) {\r\n\t\t\tcallback({});\r\n\t\t}\r\n\t\t// Note: not a documented API endpoint.\r\n\t\tjQuery.getJSON('https://badges.twitch.tv/v1/badges/global/display')\r\n\t\t\t.done(function (api) {\r\n\t\t\t\tvar badges = {\r\n\t\t\t\t\tturbo: {\r\n\t\t\t\t\t\timage: api.badge_sets.turbo.versions['1'].image_url_1x\r\n\t\t\t\t\t},\r\n\t\t\t\t\tpremium: {\r\n\t\t\t\t\t\timage: api.badge_sets.premium.versions['1'].image_url_1x\r\n\t\t\t\t\t}\r\n\t\t\t\t};\r\n\t\t\t\tcallback(badges);\r\n\t\t\t})\r\n\t\t\t.fail(function () {\r\n\t\t\t\tcallback({});\r\n\t\t\t});\r\n\t}\r\n\telse {\r\n\t\ttwitchApi.get('chat/' + username + '/badges')\r\n\t\t\t.done(function (api) {\r\n\t\t\t\tcallback(api);\r\n\t\t\t})\r\n\t\t\t.fail(function () {\r\n\t\t\t\tcallback({});\r\n\t\t\t});\r\n\t}\r\n};\r\n\r\napi.getUser = function (username, callback) {\r\n\t// Note: not a documented API endpoint.\r\n\ttwitchApi.get('users/' + username)\r\n\t\t.done(function (api) {\r\n\t\t\tcallback(api);\r\n\t\t})\r\n\t\t.fail(function () {\r\n\t\t\tcallback({});\r\n\t\t});\r\n};\r\n\r\napi.getTickets = function (callback) {\r\n\t// Note: not a documented API endpoint.\r\n\ttwitchApi.get(\r\n\t\t'/api/users/:login/tickets',\r\n\t\t{\r\n\t\t\toffset: 0,\r\n\t\t\tlimit: 100,\r\n\t\t\tunended: true\r\n\t\t}\r\n\t)\r\n\t\t.done(function (api) {\r\n\t\t\tcallback(api.tickets || []);\r\n\t\t})\r\n\t\t.fail(function () {\r\n\t\t\tcallback([]);\r\n\t\t});\r\n};\r\n\r\napi.getEmotes = function (callback) {\r\n\ttwitchApi.get('users/:login/emotes')\r\n\t\t.done(function (response) {\r\n\t\t\tif (!response || !response.emoticon_sets) {\r\n\t\t\t\tlogger.debug('getEmotes emoticon_sets empty');\r\n\t\t\t\tcallback({});\r\n\t\t\t\treturn;\r\n\t\t\t}\r\n\r\n\t\t\tcallback(response.emoticon_sets);\r\n\t\t})\r\n\t\t.fail(function () {\r\n\t\t\tlogger.debug('getEmotes API call failed');\r\n\t\t\tcallback({});\r\n\t\t});\r\n};\r\n\r\nmodule.exports = api;\r\n","var api = {};\r\nvar $ = jQuery = window.jQuery;\r\nvar templates = require('./templates');\r\nvar storage = require('./storage');\r\nvar emotes = require('./emotes');\r\nvar logger = require('./logger');\r\n\r\nvar theMenu = new UIMenu();\r\nvar theMenuButton = new UIMenuButton();\r\n\r\napi.init = function () {\r\n\t// Load CSS.\r\n\trequire('../../build/styles');\r\n\r\n\t// Load jQuery plugins.\r\n\trequire('../plugins/resizable');\r\n\trequire('jquery.scrollbar');\r\n\r\n\ttheMenuButton.init();\r\n\ttheMenu.init();\r\n};\r\n\r\napi.hideMenu = function () {\r\n\tif (theMenu.dom && theMenu.dom.length) {\r\n\t\ttheMenu.toggleDisplay(false);\r\n\t}\r\n};\r\n\r\napi.updateEmotes = function () {\r\n\ttheMenu.updateEmotes();\r\n}\r\n\r\nfunction UIMenuButton() {\r\n\tthis.dom = null;\r\n}\r\n\r\nUIMenuButton.prototype.init = function (timesFailed) {\r\n\tvar self = this;\r\n\tvar chatButton = $('.send-chat-button, .chat-buttons-container button');\r\n\tvar failCounter = timesFailed || 0;\r\n\tthis.dom = $('#emote-menu-button');\r\n\r\n\t// Element already exists.\r\n\tif (this.dom.length) {\r\n\t\tlogger.debug('MenuButton already exists, stopping init.');\r\n\t\treturn this;\r\n\t}\r\n\r\n\tif (!chatButton.length) {\r\n\t\tfailCounter += 1;\r\n\t\tif (failCounter === 1) {\r\n\t\t\tlogger.log('MenuButton container missing, trying again.');\r\n\t\t}\r\n\t\tif (failCounter >= 10) {\r\n\t\t\tlogger.log('MenuButton container missing, MenuButton unable to be added, stopping init.');\r\n\t\t\treturn this;\r\n\t\t}\r\n\t\tsetTimeout(function () {\r\n\t\t\tself.init(failCounter);\r\n\t\t}, 1000);\r\n\t\treturn this;\r\n\t}\r\n\r\n\t// Create element.\r\n\tthis.dom = $(templates.emoteButton());\r\n\tthis.dom.insertBefore(chatButton);\r\n\r\n\t// Hide then fade it in.\r\n\tthis.dom.hide();\r\n\tthis.dom.fadeIn();\r\n\r\n\t// Enable clicking.\r\n\tthis.dom.on('click', function () {\r\n\t\ttheMenu.toggleDisplay();\r\n\t});\r\n\r\n\treturn this;\r\n};\r\n\r\nUIMenuButton.prototype.toggleDisplay = function (forced) {\r\n\tvar state = typeof forced !== 'undefined' ? !!forced : !this.isVisible();\r\n\tif (state) {\r\n\t\tthis.dom.addClass('active');\r\n\t\treturn this;\r\n\t}\r\n\tthis.dom.removeClass('active');\r\n\r\n\treturn this;\r\n};\r\n\r\nUIMenuButton.prototype.isVisible = function () {\r\n\treturn this.dom.hasClass('active');\r\n};\r\n\r\nfunction UIMenu() {\r\n\tthis.dom = null;\r\n\tthis.groups = {};\r\n\tthis.emotes = [];\r\n\tthis.offset = null;\r\n\tthis.favorites = null;\r\n}\r\n\r\nUIMenu.prototype.init = function () {\r\n\tvar logger = require('./logger');\r\n\tvar self = this;\r\n\r\n\tthis.dom = $('#emote-menu-for-twitch');\r\n\r\n\t// Element already exists.\r\n\tif (this.dom.length) {\r\n\t\treturn this;\r\n\t}\r\n\r\n\t// Create element.\r\n\tthis.dom = $(templates.menu());\r\n\t$(document.body).append(this.dom);\r\n\r\n\tthis.favorites = new UIFavoritesGroup();\r\n\r\n\t// Enable dragging.\r\n\tthis.dom.draggable({\r\n\t\thandle: '.draggable',\r\n\t\tstart: function () {\r\n\t\t\tself.togglePinned(true);\r\n\t\t\tself.toggleMovement(true);\r\n\t\t},\r\n\t\tstop: function () {\r\n\t\t\tself.offset = self.dom.offset();\r\n\t\t},\r\n\t\tcontainment: $(document.body)\r\n\t});\r\n\r\n\t// Enable resizing.\r\n\tthis.dom.resizable({\r\n\t\thandle: '[data-command=\"resize-handle\"]',\r\n\t\tstop: function () {\r\n\t\t\tself.togglePinned(true);\r\n\t\t\tself.toggleMovement(true);\r\n\t\t},\r\n\t\talsoResize: self.dom.find('.scrollable'),\r\n\t\tcontainment: $(document.body),\r\n\t\tminHeight: 180,\r\n\t\tminWidth: 200\r\n\t});\r\n\r\n\t// Enable pinning.\r\n\tthis.dom.find('[data-command=\"toggle-pinned\"]').on('click', function () {\r\n\t\tself.togglePinned();\r\n\t});\r\n\r\n\t// Enable editing.\r\n\tthis.dom.find('[data-command=\"toggle-editing\"]').on('click', function () {\r\n\t\tself.toggleEditing();\r\n\t});\r\n\r\n\tthis.dom.find('.scrollable').scrollbar()\r\n\r\n\tthis.updateEmotes();\r\n\r\n\treturn this;\r\n};\r\n\r\nUIMenu.prototype._detectOutsideClick = function (event) {\r\n\t// Not outside of the menu, ignore the click.\r\n\tif ($(event.target).is('#emote-menu-for-twitch, #emote-menu-for-twitch *')) {\r\n\t\treturn;\r\n\t}\r\n\r\n\t// Clicked on the menu button, just remove the listener and let the normal listener handle it.\r\n\tif (!this.isVisible() || $(event.target).is('#emote-menu-button, #emote-menu-button *')) {\r\n\t\t$(document).off('mouseup', this._detectOutsideClick.bind(this));\r\n\t\treturn;\r\n\t}\r\n\r\n\t// Clicked outside, make sure the menu isn't pinned.\r\n\tif (!this.isPinned()) {\r\n\t\t// Menu wasn't pinned, remove listener.\r\n\t\t$(document).off('mouseup', this._detectOutsideClick.bind(this));\r\n\t\tthis.toggleDisplay();\r\n\t}\r\n};\r\n\r\nUIMenu.prototype.toggleDisplay = function (forced) {\r\n\tvar state = typeof forced !== 'undefined' ? !!forced : !this.isVisible();\r\n\tvar loggedIn = window.Twitch && window.Twitch.user.isLoggedIn();\r\n\r\n\t// Menu should be shown.\r\n\tif (state) {\r\n\t\t// Check if user is logged in.\r\n\t\tif (!loggedIn) {\r\n\t\t\t// Call native login form.\r\n\t\t\t$.login();\r\n\t\t\treturn this;\r\n\t\t}\r\n\r\n\t\tthis.updateEmotes();\r\n\t\tthis.dom.show();\r\n\r\n\t\t// Menu moved, move it back.\r\n\t\tif (this.hasMoved()) {\r\n\t\t\tthis.dom.offset(this.offset);\r\n\t\t}\r\n\t\t// Never moved, make it the same size as the chat window.\r\n\t\telse {\r\n\t\t\tvar chatContainer = $('.chat-messages');\r\n\t\t\t\r\n\t\t\t// Adjust the size to be the same as the chat container.\r\n\t\t\tthis.dom.height(chatContainer.outerHeight() - (this.dom.outerHeight() - this.dom.height()));\r\n\t\t\tthis.dom.width(chatContainer.outerWidth() - (this.dom.outerWidth() - this.dom.width()));\r\n\r\n\t\t\t// Adjust the offset to be the same as the chat container.\r\n\t\t\tthis.offset = chatContainer.offset();\r\n\t\t\tthis.dom.offset(this.offset);\r\n\t\t}\r\n\r\n\t\t// Listen for outside click.\r\n\t\t$(document).on('mouseup', this._detectOutsideClick.bind(this));\r\n\t}\r\n\t// Menu should be hidden.\r\n\telse {\r\n\t\tthis.dom.hide();\r\n\t\tthis.toggleEditing(false);\r\n\t\tthis.togglePinned(false);\r\n\t}\r\n\r\n\t// Also toggle the menu button.\r\n\ttheMenuButton.toggleDisplay(this.isVisible());\r\n\r\n\treturn this;\r\n};\r\n\r\nUIMenu.prototype.isVisible = function () {\r\n\treturn this.dom.is(':visible');\r\n};\r\n\r\nUIMenu.prototype.updateEmotes = function (which) {\r\n\tvar emote = which ? this.getEmote(which) : null;\r\n\tvar favoriteEmote = emote ? this.favorites.getEmote(which) : null;\r\n\tif (emote) {\r\n\t\temote.update();\r\n\t\tif (favoriteEmote) {\r\n\t\t\tfavoriteEmote.update();\r\n\t\t}\r\n\t\treturn this;\r\n\t}\r\n\tvar emotes = require('./emotes');\r\n\tvar theEmotes = emotes.getEmotes();\r\n\tvar theEmotesKeys = [];\r\n\tvar self = this;\r\n\r\n\ttheEmotes.forEach(function (emoteInstance) {\r\n\t\tself.addEmote(emoteInstance);\r\n\t\ttheEmotesKeys.push(emoteInstance.getText());\r\n\t});\r\n\r\n\t// Difference the emotes and remove all non-valid emotes.\r\n\tthis.emotes.forEach(function (oldEmote) {\r\n\t\tvar text = oldEmote.getText()\r\n\t\tif (theEmotesKeys.indexOf(text) < 0) {\r\n\t\t\tlogger.debug('Emote difference found, removing emote from UI: ' + text);\r\n\t\t\tself.removeEmote(text);\r\n\t\t}\r\n\t});\r\n\r\n\t// Save the emotes for next differencing.\r\n\tthis.emotes = theEmotes;\r\n\r\n\t//Update groups.\r\n\tObject.keys(this.groups).forEach(function (group) {\r\n\t\tself.getGroup(group).init();\r\n\t});\r\n\r\n\treturn this;\r\n};\r\n\r\nUIMenu.prototype.toggleEditing = function (forced) {\r\n\tvar state = typeof forced !== 'undefined' ? !!forced : !this.isEditing();\r\n\tthis.dom.toggleClass('editing', state);\r\n\treturn this;\r\n};\r\n\r\nUIMenu.prototype.isEditing = function () {\r\n\treturn this.dom.hasClass('editing');\r\n};\r\n\r\nUIMenu.prototype.togglePinned = function (forced) {\r\n\tvar state = typeof forced !== 'undefined' ? !!forced : !this.isPinned();\r\n\tthis.dom.toggleClass('pinned', state);\r\n\treturn this;\r\n};\r\n\r\nUIMenu.prototype.isPinned = function () {\r\n\treturn this.dom.hasClass('pinned');\r\n};\r\n\r\nUIMenu.prototype.toggleMovement = function (forced) {\r\n\tvar state = typeof forced !== 'undefined' ? !!forced : !this.hasMoved();\r\n\tthis.dom.toggleClass('moved', state);\r\n\treturn this;\r\n};\r\n\r\nUIMenu.prototype.hasMoved = function () {\r\n\treturn this.dom.hasClass('moved');\r\n};\r\n\r\nUIMenu.prototype.addGroup = function (emoteInstance) {\r\n\tvar channel = emoteInstance.getChannelName();\r\n\tvar self = this;\r\n\r\n\t// Already added, don't add again.\r\n\tif (this.getGroup(channel)) {\r\n\t\treturn this;\r\n\t}\r\n\r\n\t// Add to current menu groups.\r\n\tvar group = new UIGroup(emoteInstance);\r\n\tthis.groups[channel] = group;\r\n\r\n\t// Sort group names, get index of where this group should go.\r\n\tvar keys = Object.keys(this.groups);\r\n\tkeys.sort(function (a, b) {\r\n\t\t// Get the instances.\r\n\t\ta = self.groups[a].emoteInstance;\r\n\t\tb = self.groups[b].emoteInstance;\r\n\r\n\t\t// Get the channel name.\r\n\t\tvar aChannel = a.getChannelName();\r\n\t\tvar bChannel = b.getChannelName();\r\n\r\n\t\t// Get the channel display name.\r\n\t\ta = a.getChannelDisplayName().toLowerCase();\r\n\t\tb = b.getChannelDisplayName().toLowerCase();\r\n\r\n\t\t// Prime goes first, always.\r\n\t\tif (aChannel === 'twitch_prime' && bChannel !== 'twitch_prime') {\r\n\t\t\treturn -1;\r\n\t\t}\r\n\t\tif (bChannel === 'twitch_prime' && aChannel !== 'twitch_prime') {\r\n\t\t\treturn 1;\r\n\t\t}\r\n\r\n\t\t// Turbo goes after Prime, always.\r\n\t\tif (aChannel === 'turbo' && bChannel !== 'turbo') {\r\n\t\t\treturn -1;\r\n\t\t}\r\n\t\tif (bChannel === 'turbo' && aChannel !== 'turbo') {\r\n\t\t\treturn 1;\r\n\t\t}\r\n\r\n\t\t// Global goes after Turbo, always.\r\n\t\tif (aChannel === '~global' && bChannel !== '~global') {\r\n\t\t\treturn -1;\r\n\t\t}\r\n\t\tif (bChannel === '~global' && aChannel !== '~global') {\r\n\t\t\treturn 1;\r\n\t\t}\r\n\r\n\t\t// A goes first.\r\n\t\tif (a < b) {\r\n\t\t\treturn -1;\r\n\t\t}\r\n\t\t// B goest first.\r\n\t\tif (a > b) {\r\n\t\t\treturn 1;\r\n\t\t}\r\n\t\t// Both the same, doesn't matter.\r\n\t\treturn 0;\r\n\t});\r\n\r\n\tvar index = keys.indexOf(channel);\r\n\r\n\t// First in the sort, place at the beginning of the menu.\r\n\tif (index === 0) {\r\n\t\tgroup.dom.prependTo(this.dom.find('#all-emotes-group'));\r\n\t}\r\n\t// Insert after the previous group in the sort.\r\n\telse {\r\n\t\tgroup.dom.insertAfter(this.getGroup(keys[index - 1]).dom);\r\n\t}\r\n\r\n\treturn group;\r\n};\r\n\r\nUIMenu.prototype.getGroup = function (name) {\r\n\treturn this.groups[name] || null;\r\n};\r\n\r\nUIMenu.prototype.addEmote = function (emoteInstance) {\r\n\t// Get the group, or add if needed.\r\n\tvar group = this.getGroup(emoteInstance.getChannelName()) || this.addGroup(emoteInstance);\r\n\r\n\tgroup.addEmote(emoteInstance);\r\n\tgroup.toggleDisplay(group.isVisible(), true);\r\n\r\n\tthis.favorites.addEmote(emoteInstance);\r\n\r\n\treturn this;\r\n};\r\n\r\nUIMenu.prototype.removeEmote = function (name) {\r\n\tvar self = this;\r\n\tObject.keys(this.groups).forEach(function (groupName) {\r\n\t\tself.groups[groupName].removeEmote(name);\r\n\t});\r\n\tthis.favorites.removeEmote(name);\r\n\r\n\treturn this;\r\n};\r\n\r\nUIMenu.prototype.getEmote = function (name) {\r\n\tvar groupName = null;\r\n\tvar group = null;\r\n\tvar emote = null;\r\n\r\n\tfor (groupName in this.groups) {\r\n\t\tgroup = this.groups[groupName];\r\n\t\temote = group.getEmote(name);\r\n\r\n\t\tif (emote) {\r\n\t\t\treturn emote;\r\n\t\t}\r\n\t}\r\n\r\n\treturn null;\r\n};\r\n\r\nfunction UIGroup(emoteInstance) {\r\n\tthis.dom = null;\r\n\tthis.emotes = {};\r\n\tthis.emoteInstance = emoteInstance;\r\n\r\n\tthis.init();\r\n}\r\n\r\nUIGroup.prototype.init = function () {\r\n\tvar self = this;\r\n\tvar emoteInstance = this.emoteInstance;\r\n\r\n\t// First init, create new DOM.\r\n\tif (this.dom === null) {\r\n\t\tthis.dom = $(templates.emoteGroupHeader({\r\n\t\t\tbadge: emoteInstance.getChannelBadge(),\r\n\t\t\tchannel: emoteInstance.getChannelName(),\r\n\t\t\tchannelDisplayName: emoteInstance.getChannelDisplayName()\r\n\t\t}));\r\n\t}\r\n\t// Update DOM instead.\r\n\telse {\r\n\t\tthis.dom.find('.header-info').replaceWith(\r\n\t\t\t$(templates.emoteGroupHeader({\r\n\t\t\t\tbadge: emoteInstance.getChannelBadge(),\r\n\t\t\t\tchannel: emoteInstance.getChannelName(),\r\n\t\t\t\tchannelDisplayName: emoteInstance.getChannelDisplayName()\r\n\t\t\t}))\r\n\t\t\t.find('.header-info')\r\n\t\t);\r\n\t}\r\n\r\n\t// Enable emote hiding.\r\n\tthis.dom.find('.header-info [data-command=\"toggle-visibility\"]').on('click', function () {\r\n\t\tif (!theMenu.isEditing()) {\r\n\t\t\treturn;\r\n\t\t}\r\n\t\tself.toggleDisplay();\r\n\t});\r\n\r\n\tthis.toggleDisplay(this.isVisible(), true);\r\n};\r\n\r\nUIGroup.prototype.toggleDisplay = function (forced, skipUpdatingEmoteDisplay) {\r\n\tvar self = this;\r\n\tvar state = typeof forced !== 'undefined' ? !forced : this.isVisible();\r\n\r\n\tthis.dom.toggleClass('emote-menu-hidden', state);\r\n\r\n\t// Update the display of all emotes.\r\n\tif (!skipUpdatingEmoteDisplay) {\r\n\t\tObject.keys(this.emotes).forEach(function (emoteName) {\r\n\t\t\tself.emotes[emoteName].toggleDisplay(!state);\r\n\t\t\ttheMenu.updateEmotes(self.emotes[emoteName].instance.getText());\r\n\t\t});\r\n\t}\r\n\r\n\treturn this;\r\n};\r\n\r\nUIGroup.prototype.isVisible = function () {\r\n\tvar self = this;\r\n\r\n\t// If any emote is visible, the group should be visible.\r\n\treturn Object.keys(this.emotes).some(function (emoteName) {\r\n\t\treturn self.emotes[emoteName].isVisible();\r\n\t});\r\n};\r\n\r\nUIGroup.prototype.addEmote = function (emoteInstance) {\r\n\tvar self = this;\r\n\tvar emote = this.getEmote(emoteInstance.getText());\r\n\r\n\t// Already added, update instead.\r\n\tif (emote) {\r\n\t\temote.update();\r\n\t\treturn this;\r\n\t}\r\n\r\n\t// Add to current emotes.\r\n\temote = new UIEmote(emoteInstance);\r\n\tthis.emotes[emoteInstance.getText()] = emote;\r\n\r\n\tvar keys = Object.keys(this.emotes);\r\n\r\n\tkeys.sort(function (a, b) {\r\n\t\t// Get the emote instances.\r\n\t\ta = self.emotes[a].instance;\r\n\t\tb = self.emotes[b].instance;\r\n\r\n\t\t// A is a smiley, B isn't. A goes first.\r\n\t\tif (a.isSmiley() &&\t!b.isSmiley()) {\r\n\t\t\treturn -1;\r\n\t\t}\r\n\t\t// B is a smiley, A isn't. B goes first.\r\n\t\tif (b.isSmiley() &&\t!a.isSmiley()) {\r\n\t\t\treturn 1;\r\n\t\t}\r\n\r\n\t\t// Get the text of the emotes.\r\n\t\ta = a.getText().toLowerCase();\r\n\t\tb = b.getText().toLowerCase();\r\n\r\n\t\t// A goes first.\r\n\t\tif (a < b) {\r\n\t\t\treturn -1;\r\n\t\t}\r\n\t\t// B goest first.\r\n\t\tif (a > b) {\r\n\t\t\treturn 1;\r\n\t\t}\r\n\t\t// Both the same, doesn't matter.\r\n\t\treturn 0;\r\n\t});\r\n\r\n\tvar index = keys.indexOf(emoteInstance.getText());\r\n\r\n\t// First in the sort, place at the beginning of the group.\r\n\tif (index === 0) {\r\n\t\temote.dom.prependTo(this.dom.find('.emote-container'));\r\n\t}\r\n\t// Insert after the previous emote in the sort.\r\n\telse {\r\n\t\temote.dom.insertAfter(this.getEmote(keys[index - 1]).dom);\r\n\t}\r\n\r\n\treturn this;\r\n};\r\n\r\nUIGroup.prototype.getEmote = function (name) {\r\n\treturn this.emotes[name] || null;\r\n};\r\n\r\nUIGroup.prototype.removeEmote = function (name) {\r\n\tvar emote = this.getEmote(name);\r\n\tif (!emote) {\r\n\t\treturn this;\r\n\t}\r\n\temote.dom.remove();\r\n\tdelete this.emotes[name];\r\n\r\n\treturn this;\r\n};\r\n\r\nfunction UIFavoritesGroup() {\r\n\tthis.dom = $('#starred-emotes-group');\r\n\tthis.emotes = {};\r\n}\r\n\r\nUIFavoritesGroup.prototype.addEmote = UIGroup.prototype.addEmote;\r\nUIFavoritesGroup.prototype.getEmote = UIGroup.prototype.getEmote;\r\nUIFavoritesGroup.prototype.removeEmote = UIGroup.prototype.removeEmote;\r\n\r\nfunction UIEmote(emoteInstance) {\r\n\tthis.dom = null;\r\n\tthis.instance = emoteInstance;\r\n\tthis.init();\r\n}\r\n\r\nUIEmote.prototype.init = function () {\r\n\tvar self = this;\r\n\r\n\t// Create element.\r\n\tthis.dom = $(templates.emote({\r\n\t\turl: this.instance.getUrl(),\r\n\t\ttext: this.instance.getText(),\r\n\t\tthirdParty: this.instance.isThirdParty(),\r\n\t\tisVisible: this.instance.isVisible(),\r\n\t\tisStarred: this.instance.isFavorite()\r\n\t}));\r\n\r\n\t// Enable clicking.\r\n\tthis.dom.on('click', function () {\r\n\t\tif (!theMenu.isEditing()) {\r\n\t\t\tself.addToChat();\r\n\r\n\t\t\t// Close the menu if not pinned.\r\n\t\t\tif (!theMenu.isPinned()) {\r\n\t\t\t\ttheMenu.toggleDisplay();\r\n\t\t\t}\r\n\t\t}\r\n\t});\r\n\r\n\t// Enable emote hiding.\r\n\tthis.dom.find('[data-command=\"toggle-visibility\"]').on('click', function () {\r\n\t\tif (!theMenu.isEditing()) {\r\n\t\t\treturn;\r\n\t\t}\r\n\t\tself.toggleDisplay();\r\n\t\ttheMenu.updateEmotes(self.instance.getText());\r\n\t});\r\n\r\n\t// Enable emote favoriting.\r\n\tthis.dom.find('[data-command=\"toggle-starred\"]').on('click', function () {\r\n\t\tif (!theMenu.isEditing()) {\r\n\t\t\treturn;\r\n\t\t}\r\n\t\tself.toggleFavorite();\r\n\t\ttheMenu.updateEmotes(self.instance.getText());\r\n\t});\r\n\r\n\treturn this;\r\n};\r\n\r\nUIEmote.prototype.toggleDisplay = function (forced, skipInstanceUpdate) {\r\n\tvar state = typeof forced !== 'undefined' ? !forced : this.isVisible();\r\n\tthis.dom.toggleClass('emote-menu-hidden', state);\r\n\tif (!skipInstanceUpdate) {\r\n\t\tthis.instance.toggleVisibility(!state);\r\n\t}\r\n\r\n\tvar group = this.getGroup();\r\n\tgroup.toggleDisplay(group.isVisible(), true);\r\n\r\n\treturn this;\r\n};\r\n\r\nUIEmote.prototype.isVisible = function () {\r\n\treturn !this.dom.hasClass('emote-menu-hidden');\r\n};\r\n\r\nUIEmote.prototype.toggleFavorite = function (forced, skipInstanceUpdate) {\r\n\tvar state = typeof forced !== 'undefined' ? !!forced : !this.isFavorite();\r\n\tthis.dom.toggleClass('emote-menu-starred', state);\r\n\tif (!skipInstanceUpdate) {\r\n\t\tthis.instance.toggleFavorite(state);\r\n\t}\r\n\treturn this;\r\n};\r\n\r\nUIEmote.prototype.isFavorite = function () {\r\n\treturn this.dom.hasClass('emote-menu-starred');\r\n};\r\n\r\nUIEmote.prototype.addToChat = function () {\r\n\tvar ember = require('./ember-api');\r\n\t// Get textarea element.\r\n\tvar element = $('.chat-interface textarea').get(0);\r\n\tvar text = this.instance.getText();\r\n\r\n\t// Insert at cursor / replace selection.\r\n\t// https://developer.mozilla.org/en-US/docs/Code_snippets/Miscellaneous\r\n\tvar selectionEnd = element.selectionStart + text.length;\r\n\tvar currentValue = element.value;\r\n\tvar beforeText = currentValue.substring(0, element.selectionStart);\r\n\tvar afterText = currentValue.substring(element.selectionEnd, currentValue.length);\r\n\t// Smart padding, only put space at start if needed.\r\n\tif (\r\n\t\tbeforeText !== '' &&\r\n\t\tbeforeText.substr(-1) !== ' '\r\n\t) {\r\n\t\ttext = ' ' + text;\r\n\t}\r\n\t// Always put space at end.\r\n\ttext = beforeText + text + ' ' + afterText;\r\n\t// Set the text.\r\n\tember.get('controller:chat', 'currentRoom').set('messageToSend', text);\r\n\telement.focus();\r\n\t// Put cursor at end.\r\n\tselectionEnd = element.selectionStart + text.length;\r\n\telement.setSelectionRange(selectionEnd, selectionEnd);\r\n\r\n\treturn this;\r\n};\r\n\r\nUIEmote.prototype.getGroup = function () {\r\n\treturn theMenu.getGroup(this.instance.getChannelName());\r\n};\r\n\r\nUIEmote.prototype.update = function () {\r\n\tthis.toggleDisplay(this.instance.isVisible(), true);\r\n\tthis.toggleFavorite(this.instance.isFavorite(), true);\r\n};\r\n\r\nmodule.exports = api;\r\n","(function ($) {\r\n\t$.fn.resizable = function (options) {\r\n\t\tvar settings = $.extend({\r\n\t\t\talsoResize: null,\r\n\t\t\talsoResizeType: 'both', // `height`, `width`, `both`\r\n\t\t\tcontainment: null,\r\n\t\t\tcreate: null,\r\n\t\t\tdestroy: null,\r\n\t\t\thandle: '.resize-handle',\r\n\t\t\tmaxHeight: 9999,\r\n\t\t\tmaxWidth: 9999,\r\n\t\t\tminHeight: 0,\r\n\t\t\tminWidth: 0,\r\n\t\t\tresize: null,\r\n\t\t\tresizeOnce: null,\r\n\t\t\tsnapSize: 1,\r\n\t\t\tstart: null,\r\n\t\t\tstop: null\r\n\t\t}, options);\r\n\r\n\t\tsettings.element = $(this);\r\n\r\n\t\tfunction recalculateSize(evt) {\r\n\t\t\tvar data = evt.data,\r\n\t\t\t\tresized = {};\r\n\t\t\tdata.diffX = Math.round((evt.pageX - data.pageX) / settings.snapSize) * settings.snapSize;\r\n\t\t\tdata.diffY = Math.round((evt.pageY - data.pageY) / settings.snapSize) * settings.snapSize;\r\n\t\t\tif (Math.abs(data.diffX) > 0 || Math.abs(data.diffY) > 0) {\r\n\t\t\t\tif (\r\n\t\t\t\t\tsettings.element.height() !== data.height + data.diffY &&\r\n\t\t\t\t\tdata.height + data.diffY >= settings.minHeight &&\r\n\t\t\t\t\tdata.height + data.diffY <= settings.maxHeight &&\r\n\t\t\t\t\t(settings.containment ? data.outerHeight + data.diffY + data.offset.top <= settings.containment.offset().top + settings.containment.outerHeight() : true)\r\n\t\t\t\t) {\r\n\t\t\t\t\tsettings.element.height(data.height + data.diffY);\r\n\t\t\t\t\tresized.height = true;\r\n\t\t\t\t}\r\n\t\t\t\tif (\r\n\t\t\t\t\tsettings.element.width() !== data.width + data.diffX &&\r\n\t\t\t\t\tdata.width + data.diffX >= settings.minWidth &&\r\n\t\t\t\t\tdata.width + data.diffX <= settings.maxWidth &&\r\n\t\t\t\t\t(settings.containment ? data.outerWidth + data.diffX + data.offset.left <= settings.containment.offset().left + settings.containment.outerWidth() : true)\r\n\t\t\t\t) {\r\n\t\t\t\t\tsettings.element.width(data.width + data.diffX);\r\n\t\t\t\t\tresized.width = true;\r\n\t\t\t\t}\r\n\t\t\t\tif (resized.height || resized.width) {\r\n\t\t\t\t\tif (settings.resizeOnce) {\r\n\t\t\t\t\t\tsettings.resizeOnce.bind(settings.element)(evt.data);\r\n\t\t\t\t\t\tsettings.resizeOnce = null;\r\n\t\t\t\t\t}\r\n\t\t\t\t\tif (settings.resize) {\r\n\t\t\t\t\t\tsettings.resize.bind(settings.element)(evt.data);\r\n\t\t\t\t\t}\r\n\t\t\t\t\tif (settings.alsoResize) {\r\n\t\t\t\t\t\tif (resized.height && (settings.alsoResizeType === 'height' || settings.alsoResizeType === 'both')) {\r\n\t\t\t\t\t\t\tsettings.alsoResize.height(data.alsoResizeHeight + data.diffY);\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t\tif (resized.width && (settings.alsoResizeType === 'width' || settings.alsoResizeType === 'both')) {\r\n\t\t\t\t\t\t\tsettings.alsoResize.width(data.alsoResizeWidth + data.diffX);\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t}\r\n\t\t\t\t}\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\tfunction start(evt) {\r\n\t\t\tevt.preventDefault();\r\n\t\t\tif (settings.start) {\r\n\t\t\t\tsettings.start.bind(settings.element)();\r\n\t\t\t}\r\n\t\t\tvar data = {\r\n\t\t\t\talsoResizeHeight: settings.alsoResize ? settings.alsoResize.height() : 0,\r\n\t\t\t\talsoResizeWidth: settings.alsoResize ? settings.alsoResize.width() : 0,\r\n\t\t\t\theight: settings.element.height(),\r\n\t\t\t\toffset: settings.element.offset(),\r\n\t\t\t\touterHeight: settings.element.outerHeight(),\r\n\t\t\t\touterWidth: settings.element.outerWidth(),\r\n\t\t\t\tpageX: evt.pageX,\r\n\t\t\t\tpageY: evt.pageY,\r\n\t\t\t\twidth: settings.element.width()\r\n\t\t\t};\r\n\t\t\t$(document).on('mousemove', '*', data, recalculateSize);\r\n\t\t\t$(document).on('mouseup', '*', stop);\r\n\t\t}\r\n\r\n\t\tfunction stop() {\r\n\t\t\tif (settings.stop) {\r\n\t\t\t\tsettings.stop.bind(settings.element)();\r\n\t\t\t}\r\n\t\t\t$(document).off('mousemove', '*', recalculateSize);\r\n\t\t\t$(document).off('mouseup', '*', stop);\r\n\t\t}\r\n\r\n\t\tif (settings.handle) {\r\n\t\t\tif (settings.alsoResize && ['both', 'height', 'width'].indexOf(settings.alsoResizeType) >= 0) {\r\n\t\t\t\tsettings.alsoResize = $(settings.alsoResize);\r\n\t\t\t}\r\n\t\t\tif (settings.containment) {\r\n\t\t\t\tsettings.containment = $(settings.containment);\r\n\t\t\t}\r\n\t\t\tsettings.handle = $(settings.handle);\r\n\t\t\tsettings.snapSize = settings.snapSize < 1 ? 1 : settings.snapSize;\r\n\r\n\t\t\tif (options === 'destroy') {\r\n\t\t\t\tsettings.handle.off('mousedown', start);\r\n\r\n\t\t\t\tif (settings.destroy) {\r\n\t\t\t\t\tsettings.destroy.bind(this)();\r\n\t\t\t\t}\r\n\t\t\t\treturn this;\r\n\t\t\t}\r\n\r\n\t\t\tsettings.handle.on('mousedown', start);\r\n\r\n\t\t\tif (settings.create) {\r\n\t\t\t\tsettings.create.bind(this)();\r\n\t\t\t}\r\n\t\t}\r\n\t\treturn this;\r\n\t};\r\n})(jQuery);\r\n"]}