2017-05-08 12:44:02 +03:00
|
|
|
import {
|
|
|
|
replaceWithHeaderSection,
|
|
|
|
replaceWithListSection
|
|
|
|
} from 'mobiledoc-kit/editor/text-input-handlers';
|
2018-04-03 15:43:08 +03:00
|
|
|
import {run} from '@ember/runloop';
|
2017-05-08 12:44:02 +03:00
|
|
|
|
|
|
|
// Text expansions watch text entry events and will look for matches, replacing
|
|
|
|
// the matches with additional markup, atoms, or cards
|
|
|
|
// https://github.com/bustlelabs/mobiledoc-kit#responding-to-text-input
|
2017-02-27 07:44:15 +03:00
|
|
|
|
2018-01-30 13:01:07 +03:00
|
|
|
// TODO: this was copied from our old Koenig editor, it could do with some
|
|
|
|
// comments, cleanup, and refactoring
|
|
|
|
|
2017-02-27 07:44:15 +03:00
|
|
|
export default function (editor) {
|
2017-05-08 12:44:02 +03:00
|
|
|
// We don't want to run all our content rules on every text entry event,
|
|
|
|
// instead we check to see if this text entry event could match a content
|
|
|
|
// rule, and only then run the rules. Right now we only want to match
|
|
|
|
// content ending with *, _, ), ~, and `. This could increase as we support
|
|
|
|
// more markdown.
|
2017-02-27 07:44:15 +03:00
|
|
|
|
2017-03-02 19:51:57 +03:00
|
|
|
editor.onTextInput({
|
|
|
|
name: 'inline_markdown',
|
|
|
|
match: /[*_)~`]$/,
|
2018-03-15 20:13:50 +03:00
|
|
|
run(editor, matches) {
|
|
|
|
let text = editor.range.head.section.textUntil(editor.range.head);
|
2017-03-02 19:51:57 +03:00
|
|
|
|
|
|
|
switch (matches[0]) {
|
|
|
|
case '*':
|
2018-04-03 15:43:08 +03:00
|
|
|
matchStrongStar(editor, text);
|
|
|
|
matchEmStar(editor, text);
|
2017-03-02 19:51:57 +03:00
|
|
|
break;
|
|
|
|
case '_':
|
2018-04-03 15:43:08 +03:00
|
|
|
matchStrongUnderscore(editor, text);
|
|
|
|
matchEmUnderscore(editor, text);
|
2017-03-02 19:51:57 +03:00
|
|
|
break;
|
|
|
|
case ')':
|
2018-04-03 15:43:08 +03:00
|
|
|
matchLink(editor, text);
|
|
|
|
matchImage(editor, text);
|
2017-03-02 19:51:57 +03:00
|
|
|
break;
|
|
|
|
case '~':
|
2018-04-03 15:43:08 +03:00
|
|
|
matchStrikethrough(editor, text);
|
2017-03-02 19:51:57 +03:00
|
|
|
break;
|
|
|
|
case '`':
|
2018-04-03 15:43:08 +03:00
|
|
|
matchCode(editor, text);
|
2017-03-02 19:51:57 +03:00
|
|
|
break;
|
2017-02-27 07:44:15 +03:00
|
|
|
}
|
|
|
|
}
|
2017-03-02 19:51:57 +03:00
|
|
|
});
|
2017-02-27 07:44:15 +03:00
|
|
|
|
2018-01-30 13:01:07 +03:00
|
|
|
/* block level markdown ------------------------------------------------- */
|
|
|
|
|
|
|
|
// mobiledoc-kit has `* ` already built-in so we only need to add `- `
|
2017-03-02 19:51:57 +03:00
|
|
|
editor.onTextInput({
|
2018-01-30 13:01:07 +03:00
|
|
|
name: 'md_ul',
|
2017-03-02 19:51:57 +03:00
|
|
|
match: /^- $/,
|
|
|
|
run(editor) {
|
|
|
|
replaceWithListSection(editor, 'ul');
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
editor.onTextInput({
|
2018-01-30 13:01:07 +03:00
|
|
|
name: 'md_blockquote',
|
2017-03-02 19:51:57 +03:00
|
|
|
match: /^> $/,
|
|
|
|
run(editor) {
|
|
|
|
replaceWithHeaderSection(editor, 'blockquote');
|
|
|
|
}
|
|
|
|
});
|
2017-02-27 07:44:15 +03:00
|
|
|
|
2018-01-30 18:19:30 +03:00
|
|
|
editor.onTextInput({
|
|
|
|
name: 'md_hr',
|
|
|
|
match: /^---$/,
|
|
|
|
run(editor) {
|
|
|
|
let {range: {head, head: {section}}} = editor;
|
|
|
|
|
|
|
|
// Skip if cursor is not at end of section
|
|
|
|
if (!head.isTail()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Skip if section is a list item
|
|
|
|
if (section.isListItem) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
editor.run((postEditor) => {
|
2018-02-01 14:26:24 +03:00
|
|
|
let card = postEditor.builder.createCardSection('hr');
|
2018-01-30 18:19:30 +03:00
|
|
|
let needsTrailingParagraph = !section.next;
|
|
|
|
|
|
|
|
postEditor.replaceSection(section, card);
|
|
|
|
|
|
|
|
// add an empty paragraph after if necessary so writing can continue
|
|
|
|
if (needsTrailingParagraph) {
|
|
|
|
let newSection = postEditor.builder.createMarkupSection('p');
|
|
|
|
postEditor.insertSectionAtEnd(newSection);
|
|
|
|
postEditor.setRange(newSection.tailPosition());
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2018-01-30 13:01:07 +03:00
|
|
|
/* inline markdown ------------------------------------------------------ */
|
|
|
|
|
2018-04-03 15:43:08 +03:00
|
|
|
function matchStrongStar(editor, text) {
|
2017-03-02 19:51:57 +03:00
|
|
|
let {range} = editor;
|
2018-03-15 20:13:50 +03:00
|
|
|
let matches = text.match(/(^|\s)\*\*([^\s*].+?[^\s*])\*\*/);
|
2017-03-02 19:51:57 +03:00
|
|
|
if (matches) {
|
2018-03-15 20:13:50 +03:00
|
|
|
let match = matches[0][0] === '*' ? matches[0] : matches[0].substr(1);
|
|
|
|
range = range.extend(-(match.length));
|
2018-04-03 15:43:08 +03:00
|
|
|
|
2017-03-02 19:51:57 +03:00
|
|
|
editor.run((postEditor) => {
|
2017-02-27 07:44:15 +03:00
|
|
|
let position = postEditor.deleteRange(range);
|
|
|
|
let bold = postEditor.builder.createMarkup('strong');
|
2018-04-03 15:43:08 +03:00
|
|
|
postEditor.insertTextWithMarkup(position, matches[2], [bold]);
|
2017-02-27 07:44:15 +03:00
|
|
|
});
|
|
|
|
}
|
2018-04-03 15:43:08 +03:00
|
|
|
|
|
|
|
// must be scheduled so that the toggle isn't reset automatically
|
|
|
|
run.schedule('actions', this, function () {
|
|
|
|
editor.toggleMarkup('strong');
|
|
|
|
});
|
2017-02-27 07:44:15 +03:00
|
|
|
}
|
2017-03-02 19:51:57 +03:00
|
|
|
|
2018-04-03 15:43:08 +03:00
|
|
|
function matchStrongUnderscore(editor, text) {
|
2017-03-02 19:51:57 +03:00
|
|
|
let {range} = editor;
|
2018-03-15 20:13:50 +03:00
|
|
|
let matches = text.match(/(^|\s)__([^\s_].+?[^\s_])__/);
|
2017-03-02 19:51:57 +03:00
|
|
|
if (matches) {
|
2018-03-15 20:13:50 +03:00
|
|
|
let match = matches[0][0] === '_' ? matches[0] : matches[0].substr(1);
|
|
|
|
range = range.extend(-(match.length));
|
2018-04-03 15:43:08 +03:00
|
|
|
|
2017-03-02 19:51:57 +03:00
|
|
|
editor.run((postEditor) => {
|
2017-02-27 07:44:15 +03:00
|
|
|
let position = postEditor.deleteRange(range);
|
|
|
|
let bold = postEditor.builder.createMarkup('strong');
|
2018-04-03 15:43:08 +03:00
|
|
|
postEditor.insertTextWithMarkup(position, matches[2], [bold]);
|
|
|
|
});
|
|
|
|
|
|
|
|
// must be scheduled so that the toggle isn't reset automatically
|
|
|
|
run.schedule('actions', this, function () {
|
|
|
|
editor.toggleMarkup('strong');
|
2017-02-27 07:44:15 +03:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
2017-03-02 19:51:57 +03:00
|
|
|
|
2018-04-03 15:43:08 +03:00
|
|
|
function matchEmStar(editor, text) {
|
2017-03-02 19:51:57 +03:00
|
|
|
let {range} = editor;
|
2018-03-15 20:13:50 +03:00
|
|
|
let matches = text.match(/(^|\s)\*([^\s*][^*]+?[^\s])\*/);
|
2017-03-02 19:51:57 +03:00
|
|
|
if (matches) {
|
2017-02-27 07:44:15 +03:00
|
|
|
let match = matches[0][0] === '*' ? matches[0] : matches[0].substr(1);
|
|
|
|
range = range.extend(-(match.length));
|
2018-04-03 15:43:08 +03:00
|
|
|
|
2017-03-02 19:51:57 +03:00
|
|
|
editor.run((postEditor) => {
|
2017-02-27 07:44:15 +03:00
|
|
|
let position = postEditor.deleteRange(range);
|
|
|
|
let em = postEditor.builder.createMarkup('em');
|
2018-04-03 15:43:08 +03:00
|
|
|
postEditor.insertTextWithMarkup(position, matches[2], [em]);
|
|
|
|
});
|
|
|
|
|
|
|
|
// must be scheduled so that the toggle isn't reset automatically
|
|
|
|
run.schedule('actions', this, function () {
|
|
|
|
editor.toggleMarkup('em');
|
2017-02-27 07:44:15 +03:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
2017-03-02 19:51:57 +03:00
|
|
|
|
2018-04-03 15:43:08 +03:00
|
|
|
function matchEmUnderscore(editor, text) {
|
2017-03-02 19:51:57 +03:00
|
|
|
let {range} = editor;
|
2018-03-15 20:13:50 +03:00
|
|
|
let matches = text.match(/(^|\s)_([^\s_][^_]+?[^\s])_/);
|
2017-03-02 19:51:57 +03:00
|
|
|
if (matches) {
|
2017-02-27 07:44:15 +03:00
|
|
|
let match = matches[0][0] === '_' ? matches[0] : matches[0].substr(1);
|
|
|
|
range = range.extend(-(match.length));
|
2018-04-03 15:43:08 +03:00
|
|
|
|
2017-03-02 19:51:57 +03:00
|
|
|
editor.run((postEditor) => {
|
2017-02-27 07:44:15 +03:00
|
|
|
let position = postEditor.deleteRange(range);
|
|
|
|
let em = postEditor.builder.createMarkup('em');
|
2018-04-03 15:43:08 +03:00
|
|
|
postEditor.insertTextWithMarkup(position, matches[2], [em]);
|
|
|
|
});
|
|
|
|
|
|
|
|
// must be scheduled so that the toggle isn't reset automatically
|
|
|
|
run.schedule('actions', this, function () {
|
|
|
|
editor.toggleMarkup('code');
|
2017-02-27 07:44:15 +03:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
2017-03-02 19:51:57 +03:00
|
|
|
|
2018-04-03 15:43:08 +03:00
|
|
|
function matchStrikethrough(editor, text) {
|
2018-03-15 20:13:50 +03:00
|
|
|
let {range} = editor;
|
|
|
|
let matches = text.match(/(^|\s)~([^\s~][^~]+?[^\s])~/);
|
|
|
|
if (matches) {
|
|
|
|
let match = matches[0][0] === '~' ? matches[0] : matches[0].substr(1);
|
|
|
|
range = range.extend(-(match.length));
|
2018-04-03 15:43:08 +03:00
|
|
|
|
2018-03-15 20:13:50 +03:00
|
|
|
editor.run((postEditor) => {
|
|
|
|
let position = postEditor.deleteRange(range);
|
|
|
|
let s = postEditor.builder.createMarkup('s');
|
2018-04-03 15:43:08 +03:00
|
|
|
postEditor.insertTextWithMarkup(position, matches[2], [s]);
|
|
|
|
});
|
|
|
|
|
|
|
|
// must be scheduled so that the toggle isn't reset automatically
|
|
|
|
run.schedule('actions', this, function () {
|
|
|
|
editor.toggleMarkup('s');
|
2018-03-15 20:13:50 +03:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-03 15:43:08 +03:00
|
|
|
function matchCode(editor, text) {
|
2018-03-15 20:13:50 +03:00
|
|
|
let {range} = editor;
|
|
|
|
let matches = text.match(/(^|\s)`([^\s`][^`]+?[^\s])`/);
|
|
|
|
if (matches) {
|
|
|
|
let match = matches[0][0] === '`' ? matches[0] : matches[0].substr(1);
|
|
|
|
range = range.extend(-(match.length));
|
2018-04-03 15:43:08 +03:00
|
|
|
|
2018-03-15 20:13:50 +03:00
|
|
|
editor.run((postEditor) => {
|
|
|
|
let position = postEditor.deleteRange(range);
|
|
|
|
let code = postEditor.builder.createMarkup('code');
|
2018-04-03 15:43:08 +03:00
|
|
|
postEditor.insertTextWithMarkup(position, matches[2], [code]);
|
|
|
|
});
|
|
|
|
|
|
|
|
// must be scheduled so that the toggle isn't reset automatically
|
|
|
|
run.schedule('actions', this, function () {
|
|
|
|
editor.toggleMarkup('code');
|
2018-03-15 20:13:50 +03:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-03 15:43:08 +03:00
|
|
|
function matchLink(editor, text) {
|
2017-03-02 19:51:57 +03:00
|
|
|
let {range} = editor;
|
2017-02-27 07:44:15 +03:00
|
|
|
let matches = text.match(/(^|[^!])\[(.*?)\]\((.*?)\)$/);
|
2017-03-02 19:51:57 +03:00
|
|
|
if (matches) {
|
2017-02-27 07:44:15 +03:00
|
|
|
let url = matches[3];
|
|
|
|
let text = matches[2];
|
|
|
|
let match = matches[0][0] === '[' ? matches[0] : matches[0].substr(1);
|
|
|
|
range = range.extend(-match.length);
|
2018-04-03 15:43:08 +03:00
|
|
|
|
2017-03-02 19:51:57 +03:00
|
|
|
editor.run((postEditor) => {
|
2017-02-27 07:44:15 +03:00
|
|
|
let position = postEditor.deleteRange(range);
|
|
|
|
let a = postEditor.builder.createMarkup('a', {href: url});
|
2018-04-03 15:43:08 +03:00
|
|
|
postEditor.insertTextWithMarkup(position, text, [a]);
|
|
|
|
});
|
|
|
|
|
|
|
|
// must be scheduled so that the toggle isn't reset automatically
|
|
|
|
run.schedule('actions', this, function () {
|
|
|
|
editor.toggleMarkup('code');
|
2017-02-27 07:44:15 +03:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
2017-03-02 19:51:57 +03:00
|
|
|
|
2018-03-15 20:54:15 +03:00
|
|
|
function matchImage(editor, text) {
|
2018-01-30 23:46:03 +03:00
|
|
|
let matches = text.match(/^!\[(.*?)\]\((.*?)\)$/);
|
|
|
|
if (matches) {
|
|
|
|
let {range: {head, head: {section}}} = editor;
|
|
|
|
let src = matches[2];
|
|
|
|
let alt = matches[1];
|
|
|
|
|
|
|
|
// skip if cursor is not at end of section
|
|
|
|
if (!head.isTail()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// mobiledoc lists don't support cards
|
|
|
|
if (section.isListItem) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
editor.run((postEditor) => {
|
2018-02-01 14:26:24 +03:00
|
|
|
let card = postEditor.builder.createCardSection('image', {src, alt});
|
2018-01-30 23:46:03 +03:00
|
|
|
// need to check the section before replacing else it will always
|
|
|
|
// add a trailing paragraph
|
|
|
|
let needsTrailingParagraph = !section.next;
|
|
|
|
|
|
|
|
editor.range.extend(-(matches[0].length));
|
|
|
|
postEditor.replaceSection(editor.range.headSection, card);
|
|
|
|
|
|
|
|
if (needsTrailingParagraph) {
|
|
|
|
let newSection = editor.builder.createMarkupSection('p');
|
|
|
|
postEditor.insertSectionAtEnd(newSection);
|
|
|
|
postEditor.setRange(newSection.tailPosition());
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
2017-02-27 07:44:15 +03:00
|
|
|
}
|