diff --git a/bower.json b/bower.json
index 06f2a69f7db0..1e6387389e7e 100644
--- a/bower.json
+++ b/bower.json
@@ -21,7 +21,8 @@
"backbone.typeahead": "mojotech/backbone.typeahead",
"blueimp-load-image": "~1.13.0",
"blueimp-canvas-to-blob": "~2.1.1",
- "twemoji": "~1.2.1"
+ "twemoji": "~1.2.1",
+ "emojijs": "iamcal/js-emoji"
},
"devDependencies": {
"mocha": "~2.0.1",
@@ -103,6 +104,9 @@
"72x72/*",
"36x36/*",
"26x26/*"
+ ],
+ "emojijs": [
+ "emoji.js"
]
},
"concat": {
@@ -122,7 +126,8 @@
"backbone.typeahead",
"blueimp-load-image",
"blueimp-canvas-to-blob",
- "twemoji"
+ "twemoji",
+ "emojijs"
],
"libtextsecure": [
"jquery",
diff --git a/components/emojijs/emoji.js b/components/emojijs/emoji.js
new file mode 100644
index 000000000000..caecbd7642a8
--- /dev/null
+++ b/components/emojijs/emoji.js
@@ -0,0 +1,1248 @@
+;(function() {
+
+/**
+ * @global
+ * @namespace
+ */
+function emoji(){}
+ /**
+ * The set of images to use got graphical emoji.
+ *
+ * @memberof emoji
+ * @type {string}
+ */
+ emoji.img_set = 'apple';
+
+ /**
+ * Configuration details for different image sets. This includes a path to a directory containing the
+ * individual images (`path`( and a URL to sprite sheets (`sheet`). All of these images can be found
+ * in the [emoji-data repository]{@link https://github.com/iamcal/emoji-data}. Using a CDN for these
+ * is not a bad idea.
+ *
+ * @memberof emoji
+ * @type {
+ */
+ emoji.img_sets = {
+ 'apple' : {'path' : '/emoji-data/img-apple-64/' , 'sheet' : '/emoji-data/sheet_apple_64.png' },
+ 'google' : {'path' : '/emoji-data/img-google-64/' , 'sheet' : '/emoji-data/sheet_google_64.png' },
+ 'twitter' : {'path' : '/emoji-data/img-twitter-64/' , 'sheet' : '/emoji-data/sheet_twitter_64.png' },
+ 'emojione' : {'path' : '/emoji-data/img-emojione-64/', 'sheet' : '/emoji-data/sheet_emojione_64.png' }
+ };
+
+ /**
+ * Use a CSS class instead of specifying a sprite or background image for
+ * the span representing the emoticon. This requires a CSS sheet with
+ * emoticon data-uris.
+ *
+ * @memberof emoji
+ * @type bool
+ * @todo document how to build the CSS stylesheet this requires.
+ */
+ emoji.use_css_imgs = false;
+
+ /**
+ * Instead of replacing emoticons with the appropriate representations,
+ * replace them with their colon string representation.
+ * @memberof emoji
+ * @type bool
+ */
+ emoji.colons_mode = false;
+ emoji.text_mode = false;
+
+ /**
+ * If true, sets the "title" property on the span or image that gets
+ * inserted for the emoticon.
+ * @memberof emoji
+ * @type bool
+ */
+ emoji.include_title = false;
+
+ /**
+ * If the platform supports native emoticons, use those instead
+ * of the fallbacks.
+ * @memberof emoji
+ * @type bool
+ */
+ emoji.allow_native = true;
+
+ /**
+ * Set to true to use CSS sprites instead of individual images on
+ * platforms that support it.
+ *
+ * @memberof emoji
+ * @type bool
+ */
+ emoji.use_sheet = false;
+
+ // Keeps track of what has been initialized.
+ /** @private */
+ emoji.inits = {};
+ emoji.map = {};
+
+ /**
+ * @memberof emoji
+ * @param {string} str A string potentially containing ascii emoticons
+ * (ie. `:)`)
+ *
+ * @returns {string} A new string with all emoticons in `str`
+ * replaced by a representatation that's supported by the current
+ * environtment.
+ */
+ emoji.replace_emoticons = function(str){
+ emoji.init_emoticons();
+ return str.replace(emoji.rx_emoticons, function(m, $1, $2){
+ var val = emoji.map.emoticons[$2];
+ return val ? $1+emoji.replacement(val, $2) : m;
+ });
+ };
+
+ /**
+ * @memberof emoji
+ * @param {string} str A string potentially containing ascii emoticons
+ * (ie. `:)`)
+ *
+ * @returns {string} A new string with all emoticons in `str`
+ * replaced by their colon string representations (ie. `:smile:`)
+ */
+ emoji.replace_emoticons_with_colons = function(str){
+ emoji.init_emoticons();
+ return str.replace(emoji.rx_emoticons, function(m, $1, $2){
+ var val = emoji.data[emoji.map.emoticons[$2]][3][0];
+ return val ? $1+':'+val+':' : m;
+ });
+ };
+
+ /**
+ * @memberof emoji
+ * @param {string} str A string potentially containing colon string
+ * representations of emoticons (ie. `:smile:`)
+ *
+ * @returns {string} A new string with all colon string emoticons replaced
+ * with the appropriate representation.
+ */
+ emoji.replace_colons = function(str){
+ emoji.init_colons();
+ return str.replace(emoji.rx_colons, function(m){
+ var idx = m.substr(1, m.length-2);
+ var val = emoji.map.colons[idx];
+ return val ? emoji.replacement(val, idx, ':') : m;
+ });
+ };
+
+ /**
+ * @memberof emoji
+ * @param {string} str A string potentially containing unified unicode
+ * emoticons. (ie. 😄)
+ *
+ * @returns {string} A new string with all unicode emoticons replaced with
+ * the appropriate representation for the current environment.
+ */
+ emoji.replace_unified = function(str){
+ emoji.init_unified();
+ return str.replace(emoji.rx_unified, function(m){
+ var val = emoji.map.unified[m];
+ return val ? emoji.replacement(val) : m;
+ });
+ };
+
+ // Does the actual replacement of a character with the appropriate
+ /** @private */
+ emoji.replacement = function(idx, actual, wrapper){
+ wrapper = wrapper || '';
+ if (emoji.colons_mode) return ':'+emoji.data[idx][3][0]+':';
+ var text_name = (actual) ? wrapper+actual+wrapper : emoji.data[idx][8] || wrapper+emoji.data[idx][3][0]+wrapper;
+ if (emoji.text_mode) return text_name;
+ emoji.init_env();
+ if (emoji.replace_mode == 'unified' && emoji.allow_native && emoji.data[idx][0][0]) return emoji.data[idx][0][0];
+ if (emoji.replace_mode == 'softbank' && emoji.allow_native && emoji.data[idx][1]) return emoji.data[idx][1];
+ if (emoji.replace_mode == 'google' && emoji.allow_native && emoji.data[idx][2]) return emoji.data[idx][2];
+ var img = emoji.data[idx][9] || emoji.img_sets[emoji.img_set].path+idx+'.png';
+ var title = emoji.include_title ? ' title="'+(actual || emoji.data[idx][3][0])+'"' : '';
+ var text = emoji.include_text ? wrapper+(actual || emoji.data[idx][3][0])+wrapper : '';
+ if (emoji.supports_css) {
+ var px = emoji.data[idx][4];
+ var py = emoji.data[idx][5];
+ if (emoji.use_sheet && px != null && py != null){
+ var mul = 100 / (emoji.sheet_size - 1);
+ var style = 'background: url('+emoji.img_sets[emoji.img_set].sheet+');background-position:'+(mul*px)+'% '+(mul*py)+'%;background-size:'+emoji.sheet_size+'00%';
+ return ''+text+'';
+ }else if (emoji.use_css_imgs){
+ return ''+text+'';
+ }else{
+ return ''+text+'';
+ }
+ }
+ return '';
+ };
+
+ // Initializes the text emoticon data
+ /** @private */
+ emoji.init_emoticons = function(){
+ if (emoji.inits.emoticons) return;
+ emoji.init_colons(); // we require this for the emoticons map
+ emoji.inits.emoticons = 1;
+
+ var a = [];
+ emoji.map.emoticons = {};
+ for (var i in emoji.emoticons_data){
+ // because we never see some characters in our text except as entities, we must do some replacing
+ var emoticon = i.replace(/\&/g, '&').replace(/\/g, '>');
+
+ if (!emoji.map.colons[emoji.emoticons_data[i]]) continue;
+
+ emoji.map.emoticons[emoticon] = emoji.map.colons[emoji.emoticons_data[i]];
+ a.push(emoji.escape_rx(emoticon));
+ }
+ emoji.rx_emoticons = new RegExp(('(^|\\s)('+a.join('|')+')(?=$|[\\s|\\?\\.,!])'), 'g');
+ };
+
+ // Initializes the colon string data
+ /** @private */
+ emoji.init_colons = function(){
+ if (emoji.inits.colons) return;
+ emoji.inits.colons = 1;
+ emoji.rx_colons = new RegExp('\:[a-zA-Z0-9-_+]+\:', 'g');
+ emoji.map.colons = {};
+ for (var i in emoji.data){
+ for (var j=0; j":"laughing",
+ ":->":"laughing",
+ ";)":"wink",
+ ";-)":"wink",
+ ":)":"blush",
+ "(:":"blush",
+ ":-)":"blush",
+ "8)":"sunglasses",
+ ":|":"neutral_face",
+ ":-|":"neutral_face",
+ ":\\\\":"confused",
+ ":-\\\\":"confused",
+ ":\/":"confused",
+ ":-\/":"confused",
+ ":p":"stuck_out_tongue",
+ ":-p":"stuck_out_tongue",
+ ":P":"stuck_out_tongue",
+ ":-P":"stuck_out_tongue",
+ ":b":"stuck_out_tongue",
+ ":-b":"stuck_out_tongue",
+ ";p":"stuck_out_tongue_winking_eye",
+ ";-p":"stuck_out_tongue_winking_eye",
+ ";b":"stuck_out_tongue_winking_eye",
+ ";-b":"stuck_out_tongue_winking_eye",
+ ";P":"stuck_out_tongue_winking_eye",
+ ";-P":"stuck_out_tongue_winking_eye",
+ "):":"disappointed",
+ ":(":"disappointed",
+ ":-(":"disappointed",
+ ">:(":"angry",
+ ">:-(":"angry",
+ ":'(":"cry",
+ "D:":"anguished",
+ ":o":"open_mouth",
+ ":-o":"open_mouth"
+ };
+
+ if (typeof exports === 'object'){
+ module.exports = emoji;
+ }else if (typeof define === 'function' && define.amd){
+ define(function() { return emoji; });
+ }else{
+ this.emoji = emoji;
+ }
+
+}).call(function(){
+ return this || (typeof window !== 'undefined' ? window : global);
+}());
diff --git a/js/components.js b/js/components.js
index badeb1a52cea..acd1c22fbb99 100644
--- a/js/components.js
+++ b/js/components.js
@@ -28417,4 +28417,1252 @@ var twemoji = (function (
return r.join(sep || '-');
}
-}());
\ No newline at end of file
+}());
+;(function() {
+
+/**
+ * @global
+ * @namespace
+ */
+function emoji(){}
+ /**
+ * The set of images to use got graphical emoji.
+ *
+ * @memberof emoji
+ * @type {string}
+ */
+ emoji.img_set = 'apple';
+
+ /**
+ * Configuration details for different image sets. This includes a path to a directory containing the
+ * individual images (`path`( and a URL to sprite sheets (`sheet`). All of these images can be found
+ * in the [emoji-data repository]{@link https://github.com/iamcal/emoji-data}. Using a CDN for these
+ * is not a bad idea.
+ *
+ * @memberof emoji
+ * @type {
+ */
+ emoji.img_sets = {
+ 'apple' : {'path' : '/emoji-data/img-apple-64/' , 'sheet' : '/emoji-data/sheet_apple_64.png' },
+ 'google' : {'path' : '/emoji-data/img-google-64/' , 'sheet' : '/emoji-data/sheet_google_64.png' },
+ 'twitter' : {'path' : '/emoji-data/img-twitter-64/' , 'sheet' : '/emoji-data/sheet_twitter_64.png' },
+ 'emojione' : {'path' : '/emoji-data/img-emojione-64/', 'sheet' : '/emoji-data/sheet_emojione_64.png' }
+ };
+
+ /**
+ * Use a CSS class instead of specifying a sprite or background image for
+ * the span representing the emoticon. This requires a CSS sheet with
+ * emoticon data-uris.
+ *
+ * @memberof emoji
+ * @type bool
+ * @todo document how to build the CSS stylesheet this requires.
+ */
+ emoji.use_css_imgs = false;
+
+ /**
+ * Instead of replacing emoticons with the appropriate representations,
+ * replace them with their colon string representation.
+ * @memberof emoji
+ * @type bool
+ */
+ emoji.colons_mode = false;
+ emoji.text_mode = false;
+
+ /**
+ * If true, sets the "title" property on the span or image that gets
+ * inserted for the emoticon.
+ * @memberof emoji
+ * @type bool
+ */
+ emoji.include_title = false;
+
+ /**
+ * If the platform supports native emoticons, use those instead
+ * of the fallbacks.
+ * @memberof emoji
+ * @type bool
+ */
+ emoji.allow_native = true;
+
+ /**
+ * Set to true to use CSS sprites instead of individual images on
+ * platforms that support it.
+ *
+ * @memberof emoji
+ * @type bool
+ */
+ emoji.use_sheet = false;
+
+ // Keeps track of what has been initialized.
+ /** @private */
+ emoji.inits = {};
+ emoji.map = {};
+
+ /**
+ * @memberof emoji
+ * @param {string} str A string potentially containing ascii emoticons
+ * (ie. `:)`)
+ *
+ * @returns {string} A new string with all emoticons in `str`
+ * replaced by a representatation that's supported by the current
+ * environtment.
+ */
+ emoji.replace_emoticons = function(str){
+ emoji.init_emoticons();
+ return str.replace(emoji.rx_emoticons, function(m, $1, $2){
+ var val = emoji.map.emoticons[$2];
+ return val ? $1+emoji.replacement(val, $2) : m;
+ });
+ };
+
+ /**
+ * @memberof emoji
+ * @param {string} str A string potentially containing ascii emoticons
+ * (ie. `:)`)
+ *
+ * @returns {string} A new string with all emoticons in `str`
+ * replaced by their colon string representations (ie. `:smile:`)
+ */
+ emoji.replace_emoticons_with_colons = function(str){
+ emoji.init_emoticons();
+ return str.replace(emoji.rx_emoticons, function(m, $1, $2){
+ var val = emoji.data[emoji.map.emoticons[$2]][3][0];
+ return val ? $1+':'+val+':' : m;
+ });
+ };
+
+ /**
+ * @memberof emoji
+ * @param {string} str A string potentially containing colon string
+ * representations of emoticons (ie. `:smile:`)
+ *
+ * @returns {string} A new string with all colon string emoticons replaced
+ * with the appropriate representation.
+ */
+ emoji.replace_colons = function(str){
+ emoji.init_colons();
+ return str.replace(emoji.rx_colons, function(m){
+ var idx = m.substr(1, m.length-2);
+ var val = emoji.map.colons[idx];
+ return val ? emoji.replacement(val, idx, ':') : m;
+ });
+ };
+
+ /**
+ * @memberof emoji
+ * @param {string} str A string potentially containing unified unicode
+ * emoticons. (ie. 😄)
+ *
+ * @returns {string} A new string with all unicode emoticons replaced with
+ * the appropriate representation for the current environment.
+ */
+ emoji.replace_unified = function(str){
+ emoji.init_unified();
+ return str.replace(emoji.rx_unified, function(m){
+ var val = emoji.map.unified[m];
+ return val ? emoji.replacement(val) : m;
+ });
+ };
+
+ // Does the actual replacement of a character with the appropriate
+ /** @private */
+ emoji.replacement = function(idx, actual, wrapper){
+ wrapper = wrapper || '';
+ if (emoji.colons_mode) return ':'+emoji.data[idx][3][0]+':';
+ var text_name = (actual) ? wrapper+actual+wrapper : emoji.data[idx][8] || wrapper+emoji.data[idx][3][0]+wrapper;
+ if (emoji.text_mode) return text_name;
+ emoji.init_env();
+ if (emoji.replace_mode == 'unified' && emoji.allow_native && emoji.data[idx][0][0]) return emoji.data[idx][0][0];
+ if (emoji.replace_mode == 'softbank' && emoji.allow_native && emoji.data[idx][1]) return emoji.data[idx][1];
+ if (emoji.replace_mode == 'google' && emoji.allow_native && emoji.data[idx][2]) return emoji.data[idx][2];
+ var img = emoji.data[idx][9] || emoji.img_sets[emoji.img_set].path+idx+'.png';
+ var title = emoji.include_title ? ' title="'+(actual || emoji.data[idx][3][0])+'"' : '';
+ var text = emoji.include_text ? wrapper+(actual || emoji.data[idx][3][0])+wrapper : '';
+ if (emoji.supports_css) {
+ var px = emoji.data[idx][4];
+ var py = emoji.data[idx][5];
+ if (emoji.use_sheet && px != null && py != null){
+ var mul = 100 / (emoji.sheet_size - 1);
+ var style = 'background: url('+emoji.img_sets[emoji.img_set].sheet+');background-position:'+(mul*px)+'% '+(mul*py)+'%;background-size:'+emoji.sheet_size+'00%';
+ return ''+text+'';
+ }else if (emoji.use_css_imgs){
+ return ''+text+'';
+ }else{
+ return ''+text+'';
+ }
+ }
+ return '';
+ };
+
+ // Initializes the text emoticon data
+ /** @private */
+ emoji.init_emoticons = function(){
+ if (emoji.inits.emoticons) return;
+ emoji.init_colons(); // we require this for the emoticons map
+ emoji.inits.emoticons = 1;
+
+ var a = [];
+ emoji.map.emoticons = {};
+ for (var i in emoji.emoticons_data){
+ // because we never see some characters in our text except as entities, we must do some replacing
+ var emoticon = i.replace(/\&/g, '&').replace(/\/g, '>');
+
+ if (!emoji.map.colons[emoji.emoticons_data[i]]) continue;
+
+ emoji.map.emoticons[emoticon] = emoji.map.colons[emoji.emoticons_data[i]];
+ a.push(emoji.escape_rx(emoticon));
+ }
+ emoji.rx_emoticons = new RegExp(('(^|\\s)('+a.join('|')+')(?=$|[\\s|\\?\\.,!])'), 'g');
+ };
+
+ // Initializes the colon string data
+ /** @private */
+ emoji.init_colons = function(){
+ if (emoji.inits.colons) return;
+ emoji.inits.colons = 1;
+ emoji.rx_colons = new RegExp('\:[a-zA-Z0-9-_+]+\:', 'g');
+ emoji.map.colons = {};
+ for (var i in emoji.data){
+ for (var j=0; j":"laughing",
+ ":->":"laughing",
+ ";)":"wink",
+ ";-)":"wink",
+ ":)":"blush",
+ "(:":"blush",
+ ":-)":"blush",
+ "8)":"sunglasses",
+ ":|":"neutral_face",
+ ":-|":"neutral_face",
+ ":\\\\":"confused",
+ ":-\\\\":"confused",
+ ":\/":"confused",
+ ":-\/":"confused",
+ ":p":"stuck_out_tongue",
+ ":-p":"stuck_out_tongue",
+ ":P":"stuck_out_tongue",
+ ":-P":"stuck_out_tongue",
+ ":b":"stuck_out_tongue",
+ ":-b":"stuck_out_tongue",
+ ";p":"stuck_out_tongue_winking_eye",
+ ";-p":"stuck_out_tongue_winking_eye",
+ ";b":"stuck_out_tongue_winking_eye",
+ ";-b":"stuck_out_tongue_winking_eye",
+ ";P":"stuck_out_tongue_winking_eye",
+ ";-P":"stuck_out_tongue_winking_eye",
+ "):":"disappointed",
+ ":(":"disappointed",
+ ":-(":"disappointed",
+ ">:(":"angry",
+ ">:-(":"angry",
+ ":'(":"cry",
+ "D:":"anguished",
+ ":o":"open_mouth",
+ ":-o":"open_mouth"
+ };
+
+ if (typeof exports === 'object'){
+ module.exports = emoji;
+ }else if (typeof define === 'function' && define.amd){
+ define(function() { return emoji; });
+ }else{
+ this.emoji = emoji;
+ }
+
+}).call(function(){
+ return this || (typeof window !== 'undefined' ? window : global);
+}());