mirror of
https://github.com/TryGhost/Ghost.git
synced 2025-01-04 17:04:59 +03:00
9d67980a7e
refs #5798, closes #5018 - adds new `gh-fullscreen-modal` component - modals are now specified in-context so that they can have deeper interaction with their surrounding components/controller/route, i.e. a modal component can be a thin confirm/deny wrapper over the underlying controller action keeping all context-sensitive logic in one place - adds spin-buttons to all modals with async behaviour - adds/improves behaviour of inline-validation in modals - improves re-authenticate modal to properly handle validation and authentication errors
174 lines
4.8 KiB
JavaScript
174 lines
4.8 KiB
JavaScript
/* global moment, Showdown */
|
|
import Ember from 'ember';
|
|
import titleize from 'ghost/utils/titleize';
|
|
|
|
const {Mixin} = Ember;
|
|
|
|
// Used for simple, noncomputational replace-and-go! shortcuts.
|
|
// See default case in shortcut function below.
|
|
let simpleShortcutSyntax = {
|
|
bold: {
|
|
regex: '**|**',
|
|
cursor: '|'
|
|
},
|
|
italic: {
|
|
regex: '*|*',
|
|
cursor: '|'
|
|
|
|
},
|
|
strike: {
|
|
regex: '~~|~~',
|
|
cursor: '|'
|
|
},
|
|
code: {
|
|
regex: '`|`',
|
|
cursor: '|'
|
|
},
|
|
blockquote: {
|
|
regex: '> |',
|
|
cursor: '|',
|
|
newline: true
|
|
},
|
|
list: {
|
|
regex: '* |',
|
|
cursor: '|',
|
|
newline: true
|
|
},
|
|
link: {
|
|
regex: '[|](http://)',
|
|
cursor: 'http://'
|
|
},
|
|
image: {
|
|
regex: '![|](http://)',
|
|
cursor: 'http://',
|
|
newline: true
|
|
}
|
|
};
|
|
|
|
let shortcuts = {
|
|
simple(type, replacement, selection, line) {
|
|
let startIndex = 0;
|
|
let shortcut;
|
|
|
|
if (simpleShortcutSyntax.hasOwnProperty(type)) {
|
|
shortcut = simpleShortcutSyntax[type];
|
|
// insert the markdown
|
|
replacement.text = shortcut.regex.replace('|', selection.text);
|
|
|
|
// add a newline if needed
|
|
if (shortcut.newline && line.text !== '') {
|
|
startIndex = 1;
|
|
replacement.text = `\n${replacement.text}`;
|
|
}
|
|
|
|
// handle cursor position
|
|
if (selection.text === '' && shortcut.cursor === '|') {
|
|
// the cursor should go where | was
|
|
replacement.position = startIndex + replacement.start + shortcut.regex.indexOf(shortcut.cursor);
|
|
} else if (shortcut.cursor !== '|') {
|
|
// the cursor should select the string which matches shortcut.cursor
|
|
replacement.position = {
|
|
start: replacement.start + replacement.text.indexOf(shortcut.cursor)
|
|
};
|
|
replacement.position.end = replacement.position.start + shortcut.cursor.length;
|
|
}
|
|
}
|
|
|
|
return replacement;
|
|
},
|
|
|
|
cycleHeaderLevel(replacement, line) {
|
|
let match = line.text.match(/^#+/);
|
|
let currentHeaderLevel,
|
|
hashPrefix;
|
|
|
|
if (!match) {
|
|
currentHeaderLevel = 1;
|
|
} else {
|
|
currentHeaderLevel = match[0].length;
|
|
}
|
|
|
|
if (currentHeaderLevel > 2) {
|
|
currentHeaderLevel = 1;
|
|
}
|
|
|
|
hashPrefix = new Array(currentHeaderLevel + 2).join('#');
|
|
|
|
replacement.text = `${hashPrefix} ${line.text.replace(/^#* /, '')}`;
|
|
|
|
replacement.start = line.start;
|
|
replacement.end = line.end;
|
|
|
|
return replacement;
|
|
},
|
|
|
|
copyHTML(editor, selection) {
|
|
let converter = new Showdown.converter();
|
|
let generatedHTML;
|
|
|
|
if (selection.text) {
|
|
generatedHTML = converter.makeHtml(selection.text);
|
|
} else {
|
|
generatedHTML = converter.makeHtml(editor.getValue());
|
|
}
|
|
|
|
// Talk to the editor
|
|
editor.send('toggleCopyHTMLModal', generatedHTML);
|
|
},
|
|
|
|
currentDate(replacement) {
|
|
replacement.text = moment(new Date()).format('D MMMM YYYY');
|
|
return replacement;
|
|
},
|
|
|
|
uppercase(replacement, selection) {
|
|
replacement.text = selection.text.toLocaleUpperCase();
|
|
return replacement;
|
|
},
|
|
|
|
lowercase(replacement, selection) {
|
|
replacement.text = selection.text.toLocaleLowerCase();
|
|
return replacement;
|
|
},
|
|
|
|
titlecase(replacement, selection) {
|
|
replacement.text = titleize(selection.text);
|
|
return replacement;
|
|
}
|
|
};
|
|
|
|
export default Mixin.create({
|
|
shortcut(type) {
|
|
let selection = this.getSelection();
|
|
let replacement = {
|
|
start: selection.start,
|
|
end: selection.end,
|
|
position: 'collapseToEnd'
|
|
};
|
|
|
|
switch (type) {
|
|
// This shortcut is special as it needs to send an action
|
|
case 'copyHTML':
|
|
shortcuts.copyHTML(this, selection);
|
|
break;
|
|
case 'cycleHeaderLevel':
|
|
replacement = shortcuts.cycleHeaderLevel(replacement, this.getLine());
|
|
break;
|
|
// These shortcuts all process the basic information
|
|
case 'currentDate':
|
|
case 'uppercase':
|
|
case 'lowercase':
|
|
case 'titlecase':
|
|
replacement = shortcuts[type](replacement, selection, this.getLineToCursor());
|
|
break;
|
|
// All the of basic formatting shortcuts work with a regex
|
|
default:
|
|
replacement = shortcuts.simple(type, replacement, selection, this.getLineToCursor());
|
|
}
|
|
|
|
if (replacement.text) {
|
|
this.replaceSelection(replacement.text, replacement.start, replacement.end, replacement.position);
|
|
}
|
|
}
|
|
});
|