Ghost/ghost/admin/lib/koenig-editor/addon/components/koenig-card-markdown.js
Kevin Ansfield 94c192041d Added guards for missing codemirror elements when destroying markdown cards
no issue

- if a markdown card is created and destroyed before the simplemde+codemirror instance is fully initialised we could throw errors due to simplemde calling methods on objects that don't yet exist during teardown
- we also had issues with the `_applyToolbarStyles()` method being called async by event handlers after the component was destroyed
2022-05-24 13:12:22 +01:00

242 lines
6.7 KiB
JavaScript

import Component from '@ember/component';
import classic from 'ember-classic-decorator';
import formatMarkdown from 'ghost-admin/utils/format-markdown';
import {action, computed, set} from '@ember/object';
import {utils as ghostHelperUtils} from '@tryghost/helpers';
import {htmlSafe} from '@ember/template';
import {isBlank} from '@ember/utils';
import {run} from '@ember/runloop';
import {task, timeout} from 'ember-concurrency';
const {countWords, countImages} = ghostHelperUtils;
const MIN_HEIGHT = 130;
@classic
export default class KoenigCardMarkdown extends Component {
// attrs
payload = null;
isSelected = false;
isEditing = false;
headerOffset = 0;
// internal attrs
bottomOffset = 0;
preventClick = false;
// closure actions
editCard() {}
saveCard() {}
selectCard() {}
deselectCard() {}
deleteCard() {}
registerComponent() {}
@computed('payload.markdown')
get isEmpty() {
return isBlank(this.payload.markdown);
}
@computed('renderedMarkdown')
get counts() {
return {
wordCount: countWords(this.renderedMarkdown),
imageCount: countImages(this.renderedMarkdown)
};
}
@computed('payload.markdown')
get renderedMarkdown() {
return htmlSafe(formatMarkdown(this.payload.markdown));
}
@computed('isEditing')
get toolbar() {
if (this.isEditing) {
return false;
}
return {
items: [{
buttonClass: 'fw4 flex items-center white',
icon: 'koenig/kg-edit',
iconClass: 'fill-white',
title: 'Edit',
text: '',
action: run.bind(this, this.editCard)
}]
};
}
init() {
super.init(...arguments);
if (!this.payload) {
this.set('payload', {});
}
// subtract toolbar height from MIN_HEIGHT so the trigger happens at
// the expected position without forcing the min height to be too small
this.set('bottomOffset', -MIN_HEIGHT);
this.registerComponent(this);
}
willDestroyElement() {
super.willDestroyElement(...arguments);
this._teardownResizeHandler();
}
@action
enterEditMode() {
this._preventAccidentalClick.perform();
}
@action
leaveEditMode() {
if (this.isEmpty) {
// afterRender is required to avoid double modification of `isSelected`
// TODO: see if there's a way to avoid afterRender
run.scheduleOnce('afterRender', this, this.deleteCard);
}
}
@action
updateMarkdown(markdown) {
let payload = this.payload;
let save = this.saveCard;
set(payload, 'markdown', markdown);
// update the mobiledoc and stay in edit mode
save(payload, false);
}
// fires if top comes into view 0 px from viewport top
// fires if top comes into view MIN_HEIGHTpx above viewport bottom
@action
topEntered() {
this._isTopVisible = true;
run.scheduleOnce('actions', this, this._applyToolbarStyles);
}
// fires if top leaves viewport 0 px from viewport top
// fires if top leaves viewport MIN_HEIGHTpx above viewport bottom
@action
topExited() {
let top = this._topElement.getBoundingClientRect().top;
this._isTopVisible = false;
this._isTopAbove = top < 0;
run.scheduleOnce('actions', this, this._applyToolbarStyles);
}
@action
bottomEntered() {
this._isBottomVisible = true;
run.scheduleOnce('actions', this, this._applyToolbarStyles);
}
@action
bottomExited() {
let top = this._bottomElement.getBoundingClientRect().top;
this._isBottomVisible = false;
this._isBottomBelow = top > window.innerHeight;
run.scheduleOnce('actions', this, this._applyToolbarStyles);
}
@action
registerTop(element) {
this._topElement = element;
}
@action
registerBottom(element) {
this._bottomElement = element;
}
_applyToolbarStyles() {
if (this.isDestroyed || this.isDestroying) {
return;
}
let toolbar = this.element?.querySelector('.editor-toolbar');
if (!toolbar) {
return;
}
let {left, width} = this._containerDimensions();
let style = '';
let stuckTop = `top: ${MIN_HEIGHT}px; bottom: auto`;
let fixedBottom = `position: fixed; left: ${left + 1}px; width: ${width - 2}px`;
let stuckBottom = '';
if (this._isTopVisible && this._isBottomVisible) {
style = stuckBottom;
}
if (this._isTopVisible && !this._isBottomVisible) {
style = fixedBottom;
}
if (!this._isTopVisible && !this._isTopAbove) {
style = stuckTop;
}
if (!this._isTopVisible && this._isBottomVisible) {
style = stuckBottom;
}
if (!this._isTopVisible && !this._isBottomVisible && this._isTopAbove && this._isBottomBelow) {
style = fixedBottom;
}
// set up resize watchers if in fixed position because we have to
// recalculate left position and width
if (!this._resizeHandler && style === fixedBottom) {
this._setupResizeHandler();
} else if (this._resizeHandler && style !== fixedBottom) {
this._teardownResizeHandler();
}
// account for the mobile nav bar when in fixed position
if (style === fixedBottom) {
let mobileNav = document.querySelector('.gh-mobile-nav-bar');
if (mobileNav.offsetHeight) {
style = `${style}; bottom: ${mobileNav.offsetHeight}px`;
}
}
toolbar.setAttribute('style', style);
}
_containerDimensions() {
return this.element.querySelector('.kg-card-selected').getBoundingClientRect();
}
_setupResizeHandler() {
if (this._resizeHandler) {
return;
}
this._resizeHandler = run.bind(this, this._applyToolbarStyles);
window.addEventListener('resize', this._resizeHandler);
}
_teardownResizeHandler() {
window.removeEventListener('resize', this._resizeHandler);
this._resizeHandler = null;
}
// when entering edit mode it can be easy to accidentally click where the
// toolbar is inserted. Setting `preventClick` to true adds an overlay, so
// we set that for half a second to stop double-clicks hitting the toolbar
@task(function* () {
this.set('preventClick', true);
yield timeout(500);
this.set('preventClick', false);
})
_preventAccidentalClick;
}