mirror of
https://github.com/TryGhost/Ghost.git
synced 2025-01-05 09:50:34 +03:00
275 lines
9.5 KiB
JavaScript
275 lines
9.5 KiB
JavaScript
import Component from '@ember/component';
|
|
import getScrollParent from '../utils/get-scroll-parent';
|
|
import relativeToAbsolute from '../lib/relative-to-absolute';
|
|
import {TOOLBAR_MARGIN} from './koenig-toolbar';
|
|
import {computed} from '@ember/object';
|
|
import {getLinkMarkupFromRange} from '../utils/markup-utils';
|
|
import {htmlSafe} from '@ember/template';
|
|
import {run} from '@ember/runloop';
|
|
import {inject as service} from '@ember/service';
|
|
|
|
// pixels that should be added to the `left` property of the tick adjustment styles
|
|
// TODO: handle via CSS?
|
|
const TICK_ADJUSTMENT = 8;
|
|
|
|
export default Component.extend({
|
|
config: service(),
|
|
|
|
attributeBindings: ['style'],
|
|
classNames: ['kg-input-bar', 'absolute', 'z-999'],
|
|
|
|
// public attrs
|
|
editor: null,
|
|
linkRange: null,
|
|
linkRect: null,
|
|
selectedRange: null,
|
|
|
|
// internal properties
|
|
top: null,
|
|
left: null,
|
|
right: null,
|
|
_href: '',
|
|
|
|
// private properties
|
|
_selectedRange: null,
|
|
_windowRange: null,
|
|
_onMousedownHandler: null,
|
|
_onMouseupHandler: null,
|
|
|
|
// closure actions
|
|
cancel() {},
|
|
|
|
/* computed properties -------------------------------------------------- */
|
|
|
|
style: computed('top', 'left', 'right', function () {
|
|
let position = this.getProperties('top', 'left', 'right');
|
|
let styles = Object.keys(position).map((style) => {
|
|
if (position[style] !== null) {
|
|
return `${style}: ${position[style]}px`;
|
|
}
|
|
});
|
|
|
|
return htmlSafe(styles.compact().join('; '));
|
|
}),
|
|
|
|
/* lifecycle hooks ------------------------------------------------------ */
|
|
|
|
init() {
|
|
this._super(...arguments);
|
|
|
|
if (!this.source) {
|
|
// record the range now because the property is bound and will update
|
|
// as we make changes whilst calculating the link position
|
|
this._selectedRange = this.selectedRange;
|
|
this._linkRange = this.linkRange;
|
|
|
|
// grab a window range so that we can use getBoundingClientRect. Using
|
|
// document.createRange is more efficient than doing editor.setRange
|
|
// because it doesn't trigger all of the selection changing side-effects
|
|
// TODO: extract MobiledocRange->NativeRange into a util
|
|
let editor = this.editor;
|
|
let cursor = editor.cursor;
|
|
let {head, tail} = this._linkRange;
|
|
let {node: headNode, offset: headOffset} = cursor._findNodeForPosition(head);
|
|
let {node: tailNode, offset: tailOffset} = cursor._findNodeForPosition(tail);
|
|
let range = document.createRange();
|
|
range.setStart(headNode, headOffset);
|
|
range.setEnd(tailNode, tailOffset);
|
|
this._windowRange = range;
|
|
|
|
// grab an existing href value if there is one
|
|
this._getHrefFromMarkup();
|
|
}
|
|
|
|
// wait until rendered to position so that we have access to this.element
|
|
run.schedule('afterRender', this, function () {
|
|
this._positionToolbar();
|
|
this._focusInput();
|
|
});
|
|
|
|
// watch the window for mousedown events so that we can close the menu
|
|
// when we detect a click outside
|
|
this._onMousedownHandler = run.bind(this, this._handleMousedown);
|
|
window.addEventListener('mousedown', this._onMousedownHandler);
|
|
|
|
// watch for keydown events so that we can close the menu on Escape
|
|
this._onKeydownHandler = run.bind(this, this._handleKeydown);
|
|
window.addEventListener('keydown', this._onKeydownHandler);
|
|
},
|
|
|
|
didReceiveAttrs() {
|
|
if (this.source === 'direct' && this.href !== this._href) {
|
|
this.set('_href', this.href);
|
|
}
|
|
},
|
|
|
|
willDestroyElement() {
|
|
this._super(...arguments);
|
|
this._removeStyleElement();
|
|
window.removeEventListener('mousedown', this._onMousedownHandler);
|
|
window.removeEventListener('keydown', this._onKeydownHandler);
|
|
},
|
|
|
|
actions: {
|
|
inputKeydown(event) {
|
|
if (event.key === 'Enter') {
|
|
// prevent Enter from triggering in the editor and removing text
|
|
event.preventDefault();
|
|
|
|
let href = relativeToAbsolute(this._href, this.config.get('blogUrl'));
|
|
this.set('_href', href);
|
|
|
|
if (this.source === 'direct') {
|
|
this.update(href);
|
|
this.cancel();
|
|
return;
|
|
}
|
|
|
|
// create a single editor runloop here so that we don't get
|
|
// separate remove and replace ops pushed onto the undo stack
|
|
this.editor.run((postEditor) => {
|
|
if (href) {
|
|
this._replaceLink(href, postEditor);
|
|
} else {
|
|
this._removeLinks(postEditor);
|
|
}
|
|
});
|
|
|
|
this._cancelAndReselect();
|
|
}
|
|
},
|
|
|
|
clear() {
|
|
this.set('_href', '');
|
|
this._focusInput();
|
|
}
|
|
},
|
|
|
|
// if we have a single link or a slice of a single link selected, grab the
|
|
// href and adjust our linkRange to encompass the whole link
|
|
_getHrefFromMarkup() {
|
|
let linkMarkup = getLinkMarkupFromRange(this._linkRange);
|
|
if (linkMarkup) {
|
|
this.set('_href', linkMarkup.attributes.href);
|
|
this._linkRange = this._linkRange.expandByMarker(marker => !!marker.markups.includes(linkMarkup));
|
|
}
|
|
},
|
|
|
|
_replaceLink(href, postEditor) {
|
|
this._removeLinks(postEditor);
|
|
let linkMarkup = postEditor.builder.createMarkup('a', {href});
|
|
postEditor.toggleMarkup(linkMarkup, this._linkRange);
|
|
},
|
|
|
|
// loop over all markers that are touched by linkRange, removing any 'a'
|
|
// markups on them to clear all links
|
|
_removeLinks(postEditor) {
|
|
let {headMarker, tailMarker} = this.linkRange;
|
|
let curMarker = headMarker;
|
|
|
|
while (curMarker && curMarker !== tailMarker.next) {
|
|
curMarker.markups.filterBy('tagName', 'a').forEach((markup) => {
|
|
curMarker.removeMarkup(markup);
|
|
postEditor._markDirty(curMarker);
|
|
});
|
|
curMarker = curMarker.next;
|
|
}
|
|
},
|
|
|
|
_cancelAndReselect() {
|
|
this.cancel();
|
|
if (this._selectedRange) {
|
|
this.editor.selectRange(this._selectedRange);
|
|
}
|
|
},
|
|
|
|
_focusInput() {
|
|
let scrollParent = getScrollParent(this.element);
|
|
let scrollTop = scrollParent.scrollTop;
|
|
|
|
this.element.querySelector('input').focus();
|
|
|
|
// reset the scroll position to avoid jumps
|
|
// TODO: why does the input focus cause a scroll to the bottom of the doc?
|
|
scrollParent.scrollTop = scrollTop;
|
|
},
|
|
|
|
// TODO: largely shared with {{koenig-toolbar}} and {{koenig-snippet-input}} - extract to a shared util?
|
|
_positionToolbar() {
|
|
let containerRect = this.element.offsetParent.getBoundingClientRect();
|
|
let rangeRect = this.linkRect || this._windowRange.getBoundingClientRect();
|
|
let {width, height} = this.element.getBoundingClientRect();
|
|
let newPosition = {};
|
|
|
|
// rangeRect is relative to the viewport so we need to subtract the
|
|
// container measurements to get a position relative to the container
|
|
newPosition = {
|
|
top: rangeRect.top - containerRect.top - height - TOOLBAR_MARGIN,
|
|
left: rangeRect.left - containerRect.left + rangeRect.width / 2 - width / 2,
|
|
right: null
|
|
};
|
|
|
|
let tickPosition = 50;
|
|
// don't overflow left boundary
|
|
if (newPosition.left < 0) {
|
|
newPosition.left = 0;
|
|
|
|
// calculate the tick percentage position
|
|
let absTickPosition = rangeRect.left - containerRect.left + rangeRect.width / 2;
|
|
tickPosition = absTickPosition / width * 100;
|
|
if (tickPosition < 5) {
|
|
tickPosition = 5;
|
|
}
|
|
}
|
|
// same for right boundary
|
|
if (newPosition.left + width > containerRect.width) {
|
|
newPosition.left = null;
|
|
newPosition.right = 0;
|
|
|
|
// calculate the tick percentage position
|
|
let absTickPosition = rangeRect.right - containerRect.right - rangeRect.width / 2;
|
|
tickPosition = 100 + absTickPosition / width * 100;
|
|
if (tickPosition > 95) {
|
|
tickPosition = 95;
|
|
}
|
|
}
|
|
|
|
// the tick is a pseudo-element so we the only way we can affect it's
|
|
// style is by adding a style element to the head
|
|
this._removeStyleElement(); // reset to base styles
|
|
if (tickPosition !== 50) {
|
|
this._addStyleElement(`left: calc(${tickPosition}% - ${TICK_ADJUSTMENT}px)`);
|
|
}
|
|
|
|
// update the toolbar position
|
|
this.setProperties(newPosition);
|
|
},
|
|
|
|
_addStyleElement(styles) {
|
|
let styleElement = document.createElement('style');
|
|
styleElement.id = `${this.elementId}-style`;
|
|
styleElement.innerHTML = `#${this.elementId}:before, #${this.elementId}:after { ${styles} }`;
|
|
document.head.appendChild(styleElement);
|
|
},
|
|
|
|
_removeStyleElement() {
|
|
let styleElement = document.querySelector(`#${this.elementId}-style`);
|
|
if (styleElement) {
|
|
styleElement.remove();
|
|
}
|
|
},
|
|
|
|
_handleMousedown(event) {
|
|
if (!event.target.closest(`#${this.elementId}`)) {
|
|
// no need to re-select for mouse clicks
|
|
this.cancel();
|
|
}
|
|
},
|
|
|
|
_handleKeydown(event) {
|
|
if (event.key === 'Escape') {
|
|
this._cancelAndReselect();
|
|
}
|
|
}
|
|
});
|