2014-08-14 13:36:33 -07:00
|
|
|
/* global CodeMirror, moment, Showdown */
|
2014-10-24 21:09:50 +00:00
|
|
|
// jscs:disable disallowSpacesInsideParentheses
|
|
|
|
|
2014-06-21 12:58:06 -06:00
|
|
|
/** Set up a shortcut function to be called via router actions.
|
2014-11-03 20:31:10 -07:00
|
|
|
* See editor-base-route
|
2014-06-21 12:58:06 -06:00
|
|
|
*/
|
|
|
|
|
2014-07-31 08:02:15 -04:00
|
|
|
import titleize from 'ghost/utils/titleize';
|
|
|
|
|
2014-06-23 21:17:57 -04:00
|
|
|
function init() {
|
2014-07-31 12:21:48 -04:00
|
|
|
// remove predefined `ctrl+h` shortcut
|
|
|
|
delete CodeMirror.keyMap.emacsy['Ctrl-H'];
|
|
|
|
|
2014-10-24 21:09:50 +00:00
|
|
|
// Used for simple, noncomputational replace-and-go! shortcuts.
|
|
|
|
// See default case in shortcut function below.
|
2014-06-23 21:17:57 -04:00
|
|
|
CodeMirror.prototype.simpleShortcutSyntax = {
|
|
|
|
bold: '**$1**',
|
|
|
|
italic: '*$1*',
|
|
|
|
strike: '~~$1~~',
|
|
|
|
code: '`$1`',
|
|
|
|
link: '[$1](http://)',
|
|
|
|
image: '![$1](http://)',
|
|
|
|
blockquote: '> $1'
|
|
|
|
};
|
2014-10-24 21:09:50 +00:00
|
|
|
|
2014-06-23 21:17:57 -04:00
|
|
|
CodeMirror.prototype.shortcut = function (type) {
|
|
|
|
var text = this.getSelection(),
|
|
|
|
cursor = this.getCursor(),
|
|
|
|
line = this.getLine(cursor.line),
|
|
|
|
fromLineStart = {line: cursor.line, ch: 0},
|
2014-07-31 08:02:15 -04:00
|
|
|
toLineEnd = {line: cursor.line, ch: line.length},
|
2014-08-14 13:36:33 -07:00
|
|
|
md, letterCount, textIndex, position, converter,
|
2014-07-31 12:21:48 -04:00
|
|
|
generatedHTML, match, currentHeaderLevel, hashPrefix,
|
|
|
|
replacementLine;
|
2014-08-14 13:36:33 -07:00
|
|
|
|
2014-06-23 21:17:57 -04:00
|
|
|
switch (type) {
|
2014-07-31 12:21:48 -04:00
|
|
|
case 'cycleHeaderLevel':
|
|
|
|
match = line.match(/^#+/);
|
|
|
|
|
|
|
|
if (!match) {
|
|
|
|
currentHeaderLevel = 1;
|
|
|
|
} else {
|
|
|
|
currentHeaderLevel = match[0].length;
|
|
|
|
}
|
|
|
|
|
2014-10-24 21:09:50 +00:00
|
|
|
if (currentHeaderLevel > 2) {
|
|
|
|
currentHeaderLevel = 1;
|
|
|
|
}
|
2014-07-31 12:21:48 -04:00
|
|
|
|
|
|
|
hashPrefix = new Array(currentHeaderLevel + 2).join('#');
|
2014-10-24 21:09:50 +00:00
|
|
|
|
|
|
|
// jscs:disable
|
2014-07-31 12:21:48 -04:00
|
|
|
replacementLine = hashPrefix + ' ' + line.replace(/^#* /, '');
|
2014-10-24 21:09:50 +00:00
|
|
|
// jscs:enable
|
2014-07-31 12:21:48 -04:00
|
|
|
|
|
|
|
this.replaceRange(replacementLine, fromLineStart, toLineEnd);
|
|
|
|
this.setCursor(cursor.line, cursor.ch + replacementLine.length);
|
|
|
|
break;
|
2014-10-24 21:09:50 +00:00
|
|
|
|
2014-06-23 21:17:57 -04:00
|
|
|
case 'link':
|
|
|
|
md = this.simpleShortcutSyntax.link.replace('$1', text);
|
|
|
|
this.replaceSelection(md, 'end');
|
|
|
|
if (!text) {
|
|
|
|
this.setCursor(cursor.line, cursor.ch + 1);
|
|
|
|
} else {
|
|
|
|
textIndex = line.indexOf(text, cursor.ch - text.length);
|
|
|
|
position = textIndex + md.length - 1;
|
|
|
|
this.setSelection({
|
|
|
|
line: cursor.line,
|
|
|
|
ch: position - 7
|
|
|
|
}, {
|
|
|
|
line: cursor.line,
|
|
|
|
ch: position
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return;
|
2014-10-24 21:09:50 +00:00
|
|
|
|
2014-06-23 21:17:57 -04:00
|
|
|
case 'image':
|
|
|
|
md = this.simpleShortcutSyntax.image.replace('$1', text);
|
|
|
|
if (line !== '') {
|
|
|
|
md = '\n\n' + md;
|
|
|
|
}
|
|
|
|
this.replaceSelection(md, 'end');
|
|
|
|
cursor = this.getCursor();
|
|
|
|
this.setSelection({line: cursor.line, ch: cursor.ch - 8}, {line: cursor.line, ch: cursor.ch - 1});
|
|
|
|
return;
|
2014-10-24 21:09:50 +00:00
|
|
|
|
2014-06-23 21:17:57 -04:00
|
|
|
case 'list':
|
2014-10-24 21:09:50 +00:00
|
|
|
// jscs:disable
|
2014-06-23 21:17:57 -04:00
|
|
|
md = text.replace(/^(\s*)(\w\W*)/gm, '$1* $2');
|
2014-10-24 21:09:50 +00:00
|
|
|
// jscs:enable
|
2014-06-23 21:17:57 -04:00
|
|
|
this.replaceSelection(md, 'end');
|
|
|
|
return;
|
2014-10-24 21:09:50 +00:00
|
|
|
|
2014-06-23 21:17:57 -04:00
|
|
|
case 'currentDate':
|
|
|
|
md = moment(new Date()).format('D MMMM YYYY');
|
|
|
|
this.replaceSelection(md, 'end');
|
|
|
|
return;
|
2014-10-24 21:09:50 +00:00
|
|
|
|
2014-06-23 21:17:57 -04:00
|
|
|
case 'uppercase':
|
|
|
|
md = text.toLocaleUpperCase();
|
|
|
|
break;
|
2014-10-24 21:09:50 +00:00
|
|
|
|
2014-06-23 21:17:57 -04:00
|
|
|
case 'lowercase':
|
|
|
|
md = text.toLocaleLowerCase();
|
|
|
|
break;
|
2014-10-24 21:09:50 +00:00
|
|
|
|
2014-06-23 21:17:57 -04:00
|
|
|
case 'titlecase':
|
2014-07-31 08:02:15 -04:00
|
|
|
md = titleize(text);
|
2014-06-23 21:17:57 -04:00
|
|
|
break;
|
2014-10-24 21:09:50 +00:00
|
|
|
|
2014-06-23 21:17:57 -04:00
|
|
|
case 'copyHTML':
|
|
|
|
converter = new Showdown.converter();
|
2014-08-14 13:36:33 -07:00
|
|
|
|
2014-06-23 21:17:57 -04:00
|
|
|
if (text) {
|
2014-08-14 13:36:33 -07:00
|
|
|
generatedHTML = converter.makeHtml(text);
|
2014-06-23 21:17:57 -04:00
|
|
|
} else {
|
2014-08-14 13:36:33 -07:00
|
|
|
generatedHTML = converter.makeHtml(this.getValue());
|
2014-06-23 21:17:57 -04:00
|
|
|
}
|
2014-06-21 12:58:06 -06:00
|
|
|
|
2014-08-14 13:36:33 -07:00
|
|
|
// Talk to Ember
|
2014-10-24 21:09:50 +00:00
|
|
|
this.component.sendAction('openModal', 'copy-html', {generatedHTML: generatedHTML});
|
2014-07-31 12:21:48 -04:00
|
|
|
|
2014-06-23 21:17:57 -04:00
|
|
|
break;
|
2014-10-24 21:09:50 +00:00
|
|
|
|
2014-06-23 21:17:57 -04:00
|
|
|
default:
|
|
|
|
if (this.simpleShortcutSyntax[type]) {
|
|
|
|
md = this.simpleShortcutSyntax[type].replace('$1', text);
|
|
|
|
}
|
2014-06-21 12:58:06 -06:00
|
|
|
}
|
2014-06-23 21:17:57 -04:00
|
|
|
if (md) {
|
|
|
|
this.replaceSelection(md, 'end');
|
|
|
|
if (!text) {
|
|
|
|
letterCount = md.length;
|
|
|
|
this.setCursor({
|
|
|
|
line: cursor.line,
|
|
|
|
ch: cursor.ch + (letterCount / 2)
|
|
|
|
});
|
|
|
|
}
|
2014-06-21 12:58:06 -06:00
|
|
|
}
|
2014-06-23 21:17:57 -04:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
export default {
|
|
|
|
init: init
|
2014-07-30 21:18:00 -06:00
|
|
|
};
|