mirror of
https://github.com/TryGhost/Ghost.git
synced 2024-12-23 02:41:50 +03:00
715fe52793
closes https://github.com/TryGhost/Ghost/issues/8163 Previously when you created a link the toolbar component would re-render and dissapear. This update ensures that the logic that must happen on first render does not happen on subsequent renders.
158 lines
5.1 KiB
JavaScript
158 lines
5.1 KiB
JavaScript
import Component from 'ember-component';
|
|
import computed from 'ember-computed';
|
|
import run from 'ember-runloop';
|
|
import $ from 'jquery';
|
|
import layout from '../templates/components/koenig-toolbar';
|
|
|
|
import Tools from '../options/default-tools';
|
|
|
|
export default Component.extend({
|
|
layout,
|
|
classNames: ['gh-toolbar'],
|
|
classNameBindings: ['isVisible'],
|
|
isVisible: false,
|
|
tools: [],
|
|
hasRendered: false,
|
|
isLink: computed({
|
|
get() {
|
|
return this._isLink;
|
|
},
|
|
set(_, value) {
|
|
this._isLink = value;
|
|
return this._isLink;
|
|
}
|
|
}),
|
|
|
|
toolbar: computed('tools.@each.selected', function () {
|
|
// TODO if a block section other than a primary section is selected then
|
|
// the returned list removes one of the primary sections to compensate,
|
|
// so that there are only ever four primary sections.
|
|
let visibleTools = [];
|
|
|
|
this.tools.forEach((tool) => {
|
|
if (tool.type === 'markup') {
|
|
visibleTools.push(tool);
|
|
}
|
|
});
|
|
return visibleTools;
|
|
}),
|
|
|
|
toolbarBlocks: computed('tools.@each.selected', function () {
|
|
// TODO if a block section other than a primary section is selected then
|
|
// the returned list removes one of the primary sections to compensate,
|
|
// so that there are only ever four primary sections.
|
|
let visibleTools = [];
|
|
|
|
this.tools.forEach((tool) => {
|
|
if (tool.toolbar) {
|
|
visibleTools.push(tool);
|
|
}
|
|
});
|
|
return visibleTools;
|
|
}),
|
|
|
|
init() {
|
|
this._super(...arguments);
|
|
this.tools = new Tools(this.get('editor'), this);
|
|
this.iconURL = `${this.get('assetPath')}/tools/`;
|
|
},
|
|
|
|
didRender() {
|
|
if (this.get('hasRendered')) {
|
|
return;
|
|
}
|
|
let toolbar = this.$();
|
|
let {editor} = this;
|
|
let $editor = $(this.get('containerSelector')); // TODO - this element is part of ghost-admin, we need to separate them more.
|
|
let isMousedown = false;
|
|
|
|
$editor.mousedown(() => isMousedown = true);
|
|
$editor.mouseup(() => {
|
|
isMousedown = false;
|
|
updateToolbarToRange(this, toolbar, $editor, isMousedown);
|
|
});
|
|
editor.cursorDidChange(() => updateToolbarToRange(this, toolbar, $editor, isMousedown));
|
|
this.set('hasRendered', true);
|
|
},
|
|
|
|
willDestroyElement() {
|
|
this.editor.destroy();
|
|
},
|
|
|
|
actions: {
|
|
linkKeyDown(event) {
|
|
// if escape close link
|
|
if (event.keyCode === 27) {
|
|
this.set('isLink', false);
|
|
}
|
|
},
|
|
|
|
linkKeyPress(event) {
|
|
// if enter run link
|
|
if (event.keyCode === 13) {
|
|
this.set('isLink', false);
|
|
this.set('isVisible', false);
|
|
this.editor.run((postEditor) => {
|
|
let markup = postEditor.builder.createMarkup('a', {href: event.target.value});
|
|
postEditor.addMarkupToRange(this.get('linkRange'), markup);
|
|
});
|
|
|
|
this.set('linkRange', null);
|
|
event.stopPropagation();
|
|
}
|
|
}
|
|
},
|
|
doLink(range) {
|
|
this.set('isLink', true);
|
|
this.set('linkRange', range);
|
|
}
|
|
});
|
|
|
|
// update the location of the toolbar and display it if the range is visible.
|
|
function updateToolbarToRange(self, $holder, $editor, isMouseDown) {
|
|
// if there is no cursor:
|
|
let {editor} = self;
|
|
if (!editor.range || editor.range.head.isBlank || isMouseDown) {
|
|
if (!self.get('isLink')) {
|
|
self.set('isVisible', false);
|
|
}
|
|
return;
|
|
}
|
|
self.propertyWillChange('toolbar');
|
|
self.propertyWillChange('toolbarBlocks');
|
|
|
|
if (!editor.range.isCollapsed) {
|
|
// if we have a selection, then the toolbar appears just below said selection:
|
|
|
|
let range = window.getSelection().getRangeAt(0); // get the actual range within the DOM.
|
|
let position = range.getBoundingClientRect();
|
|
let edOffset = $editor.offset();
|
|
|
|
self.set('isVisible', true);
|
|
run.schedule('afterRender', this,
|
|
() => {
|
|
$holder.css('top', position.top + $editor.scrollTop() - $holder.height() - 20); // - edOffset.top+10
|
|
$holder.css('left', position.left + (position.width / 2) + $editor.scrollLeft() - edOffset.left - ($holder.width() / 2));
|
|
}
|
|
);
|
|
|
|
self.set('isLink', false);
|
|
|
|
self.tools.forEach((tool) => {
|
|
if (tool.hasOwnProperty('checkElements')) {
|
|
// if its a list we want to know what type
|
|
let sectionTagName = editor.activeSection._tagName === 'li' ? editor.activeSection.parent._tagName : editor.activeSection._tagName;
|
|
tool.checkElements(editor.activeMarkups.concat([{tagName: sectionTagName}]));
|
|
}
|
|
});
|
|
} else {
|
|
if (self.isVisible) {
|
|
self.set('isVisible', false);
|
|
self.set('isLink', false);
|
|
}
|
|
}
|
|
|
|
self.propertyDidChange('toolbar');
|
|
self.propertyDidChange('toolbarBlocks');
|
|
}
|