mirror of
https://github.com/TryGhost/Ghost.git
synced 2024-12-23 10:53:34 +03:00
2cf55cb307
refs https://github.com/TryGhost/Team/issues/928 - it's possible to set a button with no content in the CTA card but the auto-removal when exiting edit mode was only checking for presence of content - updated to only remove the card when no content, button text, or button url has been set
151 lines
4.0 KiB
JavaScript
151 lines
4.0 KiB
JavaScript
import Browser from 'mobiledoc-kit/utils/browser';
|
|
import Component from '@glimmer/component';
|
|
import {action} from '@ember/object';
|
|
import {formatTextReplacementHtml} from './koenig-text-replacement-html-input';
|
|
import {isBlank} from '@ember/utils';
|
|
import {run} from '@ember/runloop';
|
|
import {inject as service} from '@ember/service';
|
|
import {set} from '@ember/object';
|
|
import {tracked} from '@glimmer/tracking';
|
|
|
|
export default class KoenigCardEmailCtaComponent extends Component {
|
|
@service ui;
|
|
|
|
@tracked buttonFocused = false;
|
|
@tracked contentFocused = false;
|
|
|
|
get formattedHtml() {
|
|
return formatTextReplacementHtml(this.args.payload.html);
|
|
}
|
|
|
|
get segments() {
|
|
return [{
|
|
name: 'free members',
|
|
filter: 'status:free'
|
|
}, {
|
|
name: 'paid members',
|
|
filter: 'status:-free'
|
|
}];
|
|
}
|
|
|
|
get selectedSegment() {
|
|
return this.segments.find(segment => segment.filter === this.args.payload.segment);
|
|
}
|
|
|
|
get toolbar() {
|
|
if (this.args.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.args.editCard)
|
|
}]
|
|
};
|
|
}
|
|
|
|
constructor() {
|
|
super(...arguments);
|
|
this.args.registerComponent(this);
|
|
if (!this.args.payload.segment) {
|
|
this._updatePayloadAttr('segment', 'status:free');
|
|
}
|
|
}
|
|
|
|
@action
|
|
updateHtml(html) {
|
|
this._updatePayloadAttr('html', html);
|
|
}
|
|
|
|
@action
|
|
setSegment(segment) {
|
|
this._updatePayloadAttr('segment', segment.filter);
|
|
}
|
|
|
|
@action
|
|
setButtonText(event) {
|
|
this._updatePayloadAttr('buttonText', event.target.value);
|
|
}
|
|
|
|
@action
|
|
setButtonUrl(event) {
|
|
this._updatePayloadAttr('buttonUrl', event.target.value);
|
|
}
|
|
|
|
@action
|
|
registerEditor(textReplacementEditor) {
|
|
let commands = {
|
|
'META+ENTER': run.bind(this, this._enter, 'meta'),
|
|
'CTRL+ENTER': run.bind(this, this._enter, 'ctrl')
|
|
};
|
|
|
|
Object.keys(commands).forEach((str) => {
|
|
textReplacementEditor.registerKeyCommand({
|
|
str,
|
|
run() {
|
|
return commands[str](textReplacementEditor, str);
|
|
}
|
|
});
|
|
});
|
|
|
|
this._textReplacementEditor = textReplacementEditor;
|
|
|
|
run.scheduleOnce('afterRender', this, this._placeCursorAtEnd);
|
|
}
|
|
|
|
@action
|
|
leaveEditMode() {
|
|
const {html, buttonText, buttonUrl} = this.args.payload;
|
|
|
|
if (isBlank(html) && isBlank(buttonText) && isBlank(buttonUrl)) {
|
|
// afterRender is required to avoid double modification of `isSelected`
|
|
// TODO: see if there's a way to avoid afterRender
|
|
run.scheduleOnce('afterRender', this, this.args.deleteCard);
|
|
}
|
|
}
|
|
|
|
@action
|
|
blurElement(event) {
|
|
event.preventDefault();
|
|
event.target.blur();
|
|
}
|
|
|
|
@action
|
|
focusElement(selector, event) {
|
|
event.preventDefault();
|
|
document.querySelector(selector)?.focus();
|
|
}
|
|
|
|
_updatePayloadAttr(attr, value) {
|
|
let payload = this.args.payload;
|
|
|
|
set(payload, attr, value);
|
|
|
|
// update the mobiledoc and stay in edit mode
|
|
this.args.saveCard?.(payload, false);
|
|
}
|
|
|
|
/* key commands ----------------------------------------------------------*/
|
|
|
|
_enter(modifier) {
|
|
if (this.isEditing && (modifier === 'meta' || (modifier === 'crtl' && Browser.isWin()))) {
|
|
this.args.editCard?.();
|
|
}
|
|
}
|
|
|
|
_placeCursorAtEnd() {
|
|
if (!this._textReplacementEditor) {
|
|
return;
|
|
}
|
|
|
|
let tailPosition = this._textReplacementEditor.post.tailPosition();
|
|
let rangeToSelect = tailPosition.toRange();
|
|
this._textReplacementEditor.selectRange(rangeToSelect);
|
|
}
|
|
}
|