mirror of
https://github.com/TryGhost/Ghost.git
synced 2025-01-01 15:29:19 +03:00
352c4af1d7
no issue - ran [es5-getter-ember-codemod](https://github.com/rondale-sc/es5-getter-ember-codemod) - [es5 getters RFC](https://github.com/emberjs/rfcs/blob/master/text/0281-es5-getters.md) - updates the majority of `object.get('property')` with `object.property` with exceptions: - `.get('nested.property')` - it's not possible to determine if this is relying on "safe" path chaining for when `nested` doesn't exist - `.get('config.x')` and `.get('settings.x')` - both our `config` and `settings` services are proxy objects which do not support es5 getters - this PR is not exhaustive, there are still a number of places where `.get('service.foo')` and similar could be replaced but it gets us a long way there in a quick and automated fashion
168 lines
5.5 KiB
JavaScript
168 lines
5.5 KiB
JavaScript
import $ from 'jquery';
|
|
import Component from '@ember/component';
|
|
import md5 from 'blueimp-md5';
|
|
import request from 'ember-ajax/request';
|
|
import validator from 'validator';
|
|
import {htmlSafe} from '@ember/string';
|
|
import {inject as service} from '@ember/service';
|
|
import {task, timeout} from 'ember-concurrency';
|
|
|
|
const ANIMATION_TIMEOUT = 1000;
|
|
|
|
/**
|
|
* A component to manage a user profile image. By default it just handles picture uploads,
|
|
* but if passed a bound 'email' property it will render the user's gravatar image
|
|
*
|
|
* Example: {{gh-profile-image email=controllerEmailProperty setImage="controllerActionName" debounce=500}}
|
|
*
|
|
* @param {int} size The size of the image to render
|
|
* @param {String} email Reference to a bound email object if gravatar image behavior is desired.
|
|
* @param {String|action} setImage The string name of the action on the controller to be called when an image is added.
|
|
* @param {int} debounce Period to wait after changes to email before attempting to load gravatar
|
|
* @property {Boolean} hasUploadedImage Whether or not the user has uploaded an image (whether or not to show the default image/gravatar image)
|
|
* @property {String} defaultImage String containing the background-image css property of the default user profile image
|
|
* @property {String} imageBackground String containing the background-image css property with the gravatar url
|
|
*/
|
|
export default Component.extend({
|
|
config: service(),
|
|
ghostPaths: service(),
|
|
|
|
email: '',
|
|
size: 180,
|
|
debounce: 300,
|
|
|
|
imageFile: null,
|
|
hasUploadedImage: false,
|
|
|
|
_defaultImageUrl: '',
|
|
|
|
// closure actions
|
|
setImage() {},
|
|
|
|
placeholderStyle: htmlSafe('background-image: url()'),
|
|
avatarStyle: htmlSafe('display: none'),
|
|
|
|
init() {
|
|
this._super(...arguments);
|
|
|
|
let defaultImage = '/img/user-image.png';
|
|
this._defaultImageUrl = this.get('ghostPaths.assetRoot').replace(/\/$/, '') + defaultImage;
|
|
this._setPlaceholderImage(this._defaultImageUrl);
|
|
},
|
|
|
|
didReceiveAttrs() {
|
|
this._super(...arguments);
|
|
|
|
if (this.get('config.useGravatar')) {
|
|
this.setGravatar.perform();
|
|
}
|
|
},
|
|
|
|
actions: {
|
|
imageSelected(fileList, resetInput) {
|
|
// eslint-disable-next-line
|
|
let imageFile = fileList[0];
|
|
|
|
if (imageFile) {
|
|
let reader = new FileReader();
|
|
|
|
this.set('imageFile', imageFile);
|
|
this.setImage(imageFile);
|
|
|
|
reader.addEventListener('load', () => {
|
|
let dataURL = reader.result;
|
|
this.set('previewDataURL', dataURL);
|
|
}, false);
|
|
|
|
reader.readAsDataURL(imageFile);
|
|
}
|
|
|
|
resetInput();
|
|
},
|
|
|
|
openFileDialog(event) {
|
|
// simulate click to open file dialog
|
|
// using jQuery because IE11 doesn't support MouseEvent
|
|
$(event.target)
|
|
.closest('figure')
|
|
.find('input[type="file"]')
|
|
.click();
|
|
}
|
|
},
|
|
|
|
dragOver(event) {
|
|
if (!event.dataTransfer) {
|
|
return;
|
|
}
|
|
|
|
// this is needed to work around inconsistencies with dropping files
|
|
// from Chrome's downloads bar
|
|
if (navigator.userAgent.indexOf('Chrome') > -1) {
|
|
let eA = event.dataTransfer.effectAllowed;
|
|
event.dataTransfer.dropEffect = (eA === 'move' || eA === 'linkMove') ? 'move' : 'copy';
|
|
}
|
|
|
|
event.stopPropagation();
|
|
event.preventDefault();
|
|
},
|
|
|
|
dragLeave(event) {
|
|
event.preventDefault();
|
|
},
|
|
|
|
drop(event) {
|
|
event.preventDefault();
|
|
|
|
if (event.dataTransfer.files) {
|
|
this.send('imageSelected', event.dataTransfer.files);
|
|
}
|
|
},
|
|
|
|
setGravatar: task(function* () {
|
|
yield timeout(this.debounce);
|
|
|
|
let email = this.email;
|
|
|
|
if (validator.isEmail(email || '')) {
|
|
let size = this.size;
|
|
let gravatarUrl = `//www.gravatar.com/avatar/${md5(email)}?s=${size}&d=404`;
|
|
|
|
try {
|
|
// HEAD request is needed otherwise jquery attempts to process
|
|
// binary data as JSON and throws an error
|
|
yield request(gravatarUrl, {type: 'HEAD'});
|
|
// gravatar exists so switch style and let browser load it
|
|
this._setAvatarImage(gravatarUrl);
|
|
// wait for fade-in animation to finish before removing placeholder
|
|
yield timeout(ANIMATION_TIMEOUT);
|
|
this._setPlaceholderImage('');
|
|
} catch (e) {
|
|
// gravatar doesn't exist so make sure we're still showing the placeholder
|
|
this._setPlaceholderImage(this._defaultImageUrl);
|
|
// then make sure the avatar isn't visible
|
|
this._setAvatarImage('');
|
|
}
|
|
}
|
|
}).restartable(),
|
|
|
|
_setPlaceholderImage(url) {
|
|
this.set('placeholderStyle', htmlSafe(`background-image: url(${url});`));
|
|
},
|
|
|
|
_setAvatarImage(url) {
|
|
let display = url ? 'block' : 'none';
|
|
this.set('avatarStyle', htmlSafe(`background-image: url(${url}); display: ${display}`));
|
|
},
|
|
|
|
queueFile(e, data) {
|
|
let fileName = data.files[0].name;
|
|
|
|
if ((/\.(gif|jpe?g|png|svg?z)$/i).test(fileName)) {
|
|
let action = this.setImage;
|
|
if (action) {
|
|
action(data);
|
|
}
|
|
}
|
|
}
|
|
});
|