2013-11-20 11:38:21 +08:00
|
|
|
(function() {
|
|
|
|
|
|
|
|
var emoji = <%= Dir.glob(File.expand_path("../../../public/images/*.png", __FILE__)).map{|f| File.basename(f).split(".")[0]}.inspect %>;
|
|
|
|
|
|
|
|
function imageFor(code) {
|
|
|
|
if (emoji.indexOf(code) !== -1) {
|
|
|
|
var url = Discourse.getURL('/plugins/emoji/images/' + code + '.png');
|
|
|
|
return ['img', {href: url, title: ':' + code + ':', 'class': 'emoji', alt: code}];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Also support default emotions
|
|
|
|
var translations = {
|
|
|
|
':)' : 'smile',
|
|
|
|
':-)' : 'smile',
|
|
|
|
':(' : 'frowning',
|
|
|
|
':-(' : 'frowning',
|
|
|
|
';)' : 'wink',
|
|
|
|
';-)' : 'wink',
|
|
|
|
':\'(' : 'cry',
|
|
|
|
':\'-(' : 'cry',
|
|
|
|
':-\'(' : 'cry',
|
|
|
|
':p' : 'stuck_out_tongue',
|
|
|
|
':P' : 'stuck_out_tongue',
|
|
|
|
':-P' : 'stuck_out_tongue',
|
|
|
|
':O' : 'open_mouth',
|
|
|
|
':-O' : 'open_mouth',
|
2013-11-27 05:34:44 +08:00
|
|
|
':D' : 'smiley',
|
|
|
|
':-D' : 'smiley',
|
2013-11-20 11:38:21 +08:00
|
|
|
':|' : 'expressionless',
|
|
|
|
':-|' : 'expressionless',
|
|
|
|
";P" : 'stuck_out_tongue_winking_eye',
|
|
|
|
";-P" : 'stuck_out_tongue_winking_eye',
|
|
|
|
';)' : 'wink',
|
|
|
|
';-)' : 'wink',
|
|
|
|
":$" : 'blush',
|
|
|
|
":-$" : 'blush'
|
|
|
|
};
|
|
|
|
|
2014-06-05 03:48:08 +08:00
|
|
|
function checkPrev(prev) {
|
|
|
|
if (prev && prev.length) {
|
|
|
|
var lastToken = prev[prev.length-1];
|
|
|
|
if (lastToken && lastToken.charAt) {
|
|
|
|
var lastChar = lastToken.charAt(lastToken.length-1);
|
2014-08-12 00:45:55 +08:00
|
|
|
if (lastChar !== ' ' && lastChar !== "\n") return false;
|
2014-06-05 03:48:08 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-11-20 11:38:21 +08:00
|
|
|
var translationsWithColon = {};
|
|
|
|
Object.keys(translations).forEach(function (t) {
|
|
|
|
if (t[0] === ':') {
|
|
|
|
translationsWithColon[t] = translations[t];
|
|
|
|
} else {
|
|
|
|
var replacement = translations[t];
|
2014-06-05 03:48:08 +08:00
|
|
|
Discourse.Dialect.inlineReplace(t, function (token, match, prev) {
|
|
|
|
return checkPrev(prev) ? imageFor(replacement) : token;
|
2013-11-20 11:38:21 +08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
function escapeRegExp(s) {
|
|
|
|
return s.replace(/[-/\\^$*+?.()|[\]{}]/g, '\\$&')
|
|
|
|
}
|
|
|
|
|
2014-06-10 02:44:34 +08:00
|
|
|
var translationColonRegexp = new RegExp(Object.keys(translationsWithColon).map(function (t) {
|
2013-11-20 11:38:21 +08:00
|
|
|
return "(" + escapeRegExp(t) + ")";
|
2014-06-10 02:44:34 +08:00
|
|
|
}).join("|"));
|
2013-11-20 11:38:21 +08:00
|
|
|
|
|
|
|
Discourse.Dialect.registerInline(':', function(text, match, prev) {
|
|
|
|
var endPos = text.indexOf(':', 1),
|
2014-01-22 02:32:59 +08:00
|
|
|
firstSpace = text.search(/\s/),
|
2013-11-20 11:38:21 +08:00
|
|
|
contents;
|
|
|
|
|
2014-06-05 03:48:08 +08:00
|
|
|
if (!checkPrev(prev)) { return; }
|
2014-04-01 23:00:51 +08:00
|
|
|
|
2013-11-20 11:38:21 +08:00
|
|
|
// If there is no trailing colon, check our translations that begin with colons
|
2014-01-22 02:32:59 +08:00
|
|
|
if (endPos === -1 || (firstSpace !== -1 && endPos > firstSpace)) {
|
2013-11-20 11:38:21 +08:00
|
|
|
translationColonRegexp.lastIndex = 0;
|
2014-06-10 02:44:34 +08:00
|
|
|
var m = translationColonRegexp.exec(text);
|
|
|
|
if (m && m[0] && text.indexOf(m[0]) === 0) {
|
|
|
|
contents = imageFor(translationsWithColon[m[0]]);
|
2013-11-20 11:38:21 +08:00
|
|
|
if (contents) {
|
2014-06-10 02:44:34 +08:00
|
|
|
return [m[0].length, contents];
|
2013-11-20 11:38:21 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Simple find and replace from our array
|
|
|
|
var between = text.slice(1, endPos);
|
|
|
|
contents = imageFor(between);
|
|
|
|
if (contents) {
|
|
|
|
return [endPos+1, contents];
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2014-06-25 04:40:48 +08:00
|
|
|
// TODO: Make this a proper ES6 import
|
|
|
|
var ComposerView = (Discourse && Discourse.ComposerView) || (typeof require !== "undefined" && require('discourse/views/composer').default);
|
|
|
|
if (ComposerView) {
|
|
|
|
ComposerView.on("initWmdEditor", function(event){
|
2013-11-20 11:38:21 +08:00
|
|
|
|
|
|
|
var baseUrl = Discourse.getURL("/");
|
|
|
|
|
|
|
|
template = Handlebars.compile("<div class='autocomplete'>" +
|
|
|
|
"<ul>" +
|
|
|
|
"{{#each options}}" +
|
|
|
|
"<li>" +
|
|
|
|
"<a href='#'>" +
|
|
|
|
"<img src='" + baseUrl + "plugins/emoji/images/{{this}}.png' class='emoji'> " +
|
|
|
|
"{{this}}</a>" +
|
|
|
|
"</li>" +
|
|
|
|
"{{/each}}" +
|
|
|
|
"</ul>" +
|
|
|
|
"</div>");
|
|
|
|
|
|
|
|
$('#wmd-input').autocomplete({
|
|
|
|
template: template,
|
|
|
|
key: ":",
|
|
|
|
transformComplete: function(v){ return v + ":"; },
|
|
|
|
dataSource: function(term){
|
|
|
|
|
|
|
|
var full = ":" + term;
|
|
|
|
term = term.toLowerCase();
|
|
|
|
|
|
|
|
if (term === "") {
|
|
|
|
return Ember.RSVP.resolve(["smile", "smiley", "wink", "sunny", "blush"]);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (translations[full]) {
|
|
|
|
return Ember.RSVP.resolve([translations[full]]);
|
|
|
|
}
|
|
|
|
|
|
|
|
var options = [];
|
|
|
|
var i;
|
|
|
|
for (i=0; i < emoji.length; i++) {
|
|
|
|
if (emoji[i].indexOf(term) === 0) {
|
|
|
|
options.push(emoji[i]);
|
|
|
|
if(options.length > 4) { break; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (options.length <= 4) {
|
|
|
|
for (i=0; i < emoji.length; i++) {
|
|
|
|
if (emoji[i].indexOf(term) > 0) {
|
|
|
|
options.push(emoji[i]);
|
|
|
|
if(options.length > 4) { break; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Ember.RSVP.resolve(options);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2014-07-04 04:54:56 +08:00
|
|
|
Discourse.Markdown.whiteListTag('img', 'class', 'emoji');
|
2013-11-20 11:38:21 +08:00
|
|
|
}).call(this);
|