mirror of
https://github.com/TryGhost/Ghost.git
synced 2025-01-07 11:30:55 +03:00
2c95cae248
no issue - remove `async` from non-async actions - tidy up template indentation/spacing
230 lines
7.0 KiB
JavaScript
230 lines
7.0 KiB
JavaScript
/* eslint-disable ghost/ember/alias-model-in-controller */
|
|
import $ from 'jquery';
|
|
import ModalComponent from 'ghost-admin/components/modal-base';
|
|
import RSVP from 'rsvp';
|
|
import {
|
|
IMAGE_EXTENSIONS,
|
|
IMAGE_MIME_TYPES
|
|
} from 'ghost-admin/components/gh-image-uploader';
|
|
import {computed} from '@ember/object';
|
|
import {htmlSafe} from '@ember/string';
|
|
import {run} from '@ember/runloop';
|
|
import {inject as service} from '@ember/service';
|
|
import {task} from 'ember-concurrency';
|
|
import {timeout} from 'ember-concurrency';
|
|
|
|
const ICON_EXTENSIONS = ['ico', 'png'];
|
|
|
|
export default ModalComponent.extend({
|
|
config: service(),
|
|
ghostPaths: service(),
|
|
notifications: service(),
|
|
session: service(),
|
|
settings: service(),
|
|
ajax: service(),
|
|
|
|
imageExtensions: IMAGE_EXTENSIONS,
|
|
imageMimeTypes: IMAGE_MIME_TYPES,
|
|
iconExtensions: null,
|
|
iconMimeTypes: 'image/png,image/x-icon',
|
|
|
|
previewGuid: (new Date()).valueOf(),
|
|
|
|
frontendUrl: computed('config.blogUrl', function () {
|
|
return `${this.get('config.blogUrl')}`;
|
|
}),
|
|
|
|
themePreviewUrl: computed(function () {
|
|
let origin = window.location.origin;
|
|
let subdir = this.ghostPaths.subdir;
|
|
let url = this.ghostPaths.url.join(origin, subdir);
|
|
|
|
return url.replace(/\/$/, '/ghost/preview/');
|
|
}),
|
|
|
|
accentColorPickerValue: computed('settings.accentColor', function () {
|
|
return this.get('settings.accentColor') || '#ffffff';
|
|
}),
|
|
|
|
accentColor: computed('settings.accentColor', function () {
|
|
let color = this.get('settings.accentColor');
|
|
if (color && color[0] === '#') {
|
|
return color.slice(1);
|
|
}
|
|
return color;
|
|
}),
|
|
|
|
accentColorBgStyle: computed(function () {
|
|
return htmlSafe(`background-color: ${this.accentColorPickerValue}`);
|
|
}),
|
|
|
|
getPreviewData: computed('settings.{accentColor,icon,logo,coverImage}', function () {
|
|
let string = `c=${encodeURIComponent(this.get('settings.accentColor'))}&icon=${encodeURIComponent(this.get('settings.icon'))}&logo=${encodeURIComponent(this.get('settings.logo'))}&cover=${encodeURIComponent(this.get('settings.coverImage'))}`;
|
|
|
|
return string;
|
|
}),
|
|
|
|
init() {
|
|
this._super(...arguments);
|
|
this.iconExtensions = ICON_EXTENSIONS;
|
|
},
|
|
|
|
didInsertElement() {
|
|
this._previewListener = run.bind(this, this.handlePreviewIframeMessage);
|
|
window.addEventListener('message', this._previewListener);
|
|
},
|
|
|
|
willDestroyElement() {
|
|
window.removeEventListener('message', this._previewListener);
|
|
// reset any unsaved changes when closing
|
|
this.settings.rollbackAttributes();
|
|
},
|
|
|
|
actions: {
|
|
save() {
|
|
this.save.perform();
|
|
},
|
|
|
|
removeImage(image) {
|
|
// setting `null` here will error as the server treats it as "null"
|
|
this.settings.set(image, '');
|
|
this.refreshPreview();
|
|
},
|
|
|
|
/**
|
|
* Opens a file selection dialog - Triggered by "Upload Image" buttons,
|
|
* searches for the hidden file input within the .gh-setting element
|
|
* containing the clicked button then simulates a click
|
|
* @param {MouseEvent} event - MouseEvent fired by the button click
|
|
*/
|
|
triggerFileDialog(event) {
|
|
// simulate click to open file dialog
|
|
// using jQuery because IE11 doesn't support MouseEvent
|
|
$(event.target)
|
|
.closest('.gh-uploadbutton-container')
|
|
.find('input[type="file"]')
|
|
.click();
|
|
},
|
|
|
|
/**
|
|
* Fired after an image upload completes
|
|
* @param {string} property - Property name to be set on `this.settings`
|
|
* @param {UploadResult[]} results - Array of UploadResult objects
|
|
* @return {string} The URL that was set on `this.settings.property`
|
|
*/
|
|
imageUploaded(property, results) {
|
|
if (results[0]) {
|
|
let result = this.settings.set(property, results[0].url);
|
|
this.refreshPreview();
|
|
return result;
|
|
}
|
|
},
|
|
|
|
updateAccentColor(event) {
|
|
this._updateAccentColor(event);
|
|
}
|
|
},
|
|
|
|
replacePreview() {
|
|
const ghostFrontendUrl = this.frontendUrl;
|
|
|
|
const options = {
|
|
contentType: 'text/html;charset=utf-8',
|
|
// ember-ajax will try and parse the response as JSON if not explicitly set
|
|
dataType: 'text',
|
|
headers: {
|
|
'x-ghost-preview': this.getPreviewData
|
|
}
|
|
};
|
|
|
|
this.ajax
|
|
.post(ghostFrontendUrl, options)
|
|
.then((response) => {
|
|
const iframe = this.getPreviewIframe();
|
|
if (iframe) {
|
|
iframe.contentWindow.postMessage(response, '*');
|
|
}
|
|
});
|
|
},
|
|
|
|
refreshPreview() {
|
|
// this.set('previewGuid',(new Date()).valueOf());
|
|
// REset the src and trigger a reload
|
|
this.getPreviewIframe().src = this.themePreviewUrl;
|
|
},
|
|
|
|
getPreviewIframe() {
|
|
return document.getElementById('site-frame');
|
|
},
|
|
|
|
handlePreviewIframeMessage(event) {
|
|
if (event && event.data && event.data === 'loaded') {
|
|
this.replacePreview();
|
|
}
|
|
},
|
|
|
|
debounceUpdateAccentColor: task(function* (event) {
|
|
yield timeout(500);
|
|
this._updateAccentColor(event);
|
|
}).restartable(),
|
|
|
|
saveTask: task(function* () {
|
|
let notifications = this.notifications;
|
|
let validationPromises = [];
|
|
|
|
try {
|
|
yield RSVP.all(validationPromises);
|
|
yield this.settings.save();
|
|
this.closeModal();
|
|
return true;
|
|
} catch (error) {
|
|
if (error) {
|
|
notifications.showAPIError(error);
|
|
throw error;
|
|
}
|
|
}
|
|
}),
|
|
|
|
async _updateAccentColor(event) {
|
|
let newColor = event.target.value;
|
|
let oldColor = this.get('settings.accentColor');
|
|
|
|
// reset errors and validation
|
|
this.get('settings.errors').remove('accentColor');
|
|
this.get('settings.hasValidated').removeObject('accentColor');
|
|
|
|
if (newColor === '') {
|
|
if (newColor === oldColor) {
|
|
return;
|
|
}
|
|
|
|
// clear out the accent color
|
|
this.settings.set('accentColor', '');
|
|
this.refreshPreview();
|
|
return;
|
|
}
|
|
|
|
// accentColor will be null unless the user has input something
|
|
if (!newColor) {
|
|
newColor = oldColor;
|
|
}
|
|
|
|
if (newColor[0] !== '#') {
|
|
newColor = `#${newColor}`;
|
|
}
|
|
|
|
if (newColor.match(/#[0-9A-Fa-f]{6}$/)) {
|
|
if (newColor === oldColor) {
|
|
return;
|
|
}
|
|
|
|
this.set('settings.accentColor', newColor);
|
|
this.refreshPreview();
|
|
} else {
|
|
this.get('settings.errors').add('accentColor', 'The colour should be in valid hex format');
|
|
this.get('settings.hasValidated').pushObject('accentColor');
|
|
return;
|
|
}
|
|
}
|
|
});
|