diff --git a/ghost/admin/app/components/gh-alert.js b/ghost/admin/app/components/gh-alert.js index 0952a52657..ca22911459 100644 --- a/ghost/admin/app/components/gh-alert.js +++ b/ghost/admin/app/components/gh-alert.js @@ -6,17 +6,16 @@ export default class GhAlert extends Component { @service notifications; get typeClass() { - let type = this.args.message.type; - let classes = ''; - let typeMapping; - - typeMapping = { + const typeMapping = { success: 'green', error: 'red', warn: 'blue', info: 'blue' }; + const type = this.args.message.type; + + let classes = ''; if (typeMapping[type] !== undefined) { classes += `gh-alert-${typeMapping[type]}`; } @@ -26,6 +25,6 @@ export default class GhAlert extends Component { @action closeNotification() { - this.notifications.closeNotification(this.message); + this.notifications.closeNotification(this.args.message); } } diff --git a/ghost/admin/app/components/gh-notification.js b/ghost/admin/app/components/gh-notification.js index b040ce6631..806f775f22 100644 --- a/ghost/admin/app/components/gh-notification.js +++ b/ghost/admin/app/components/gh-notification.js @@ -5,8 +5,6 @@ import {inject as service} from '@ember/service'; export default class GhNotification extends Component { @service notifications; - message = null; - get typeClass() { const typeMapping = { error: 'red', diff --git a/ghost/admin/tests/integration/components/gh-alert-test.js b/ghost/admin/tests/integration/components/gh-alert-test.js index 90fdcb10a1..92c9612078 100644 --- a/ghost/admin/tests/integration/components/gh-alert-test.js +++ b/ghost/admin/tests/integration/components/gh-alert-test.js @@ -1,17 +1,28 @@ import hbs from 'htmlbars-inline-precompile'; import sinon from 'sinon'; -import {click, find, render} from '@ember/test-helpers'; +import {click, find, render, settled} from '@ember/test-helpers'; import {describe, it} from 'mocha'; import {expect} from 'chai'; import {setupRenderingTest} from 'ember-mocha'; +import {tracked} from '@glimmer/tracking'; + +class Message { + @tracked message; + @tracked type; + + constructor({message, type}) { + this.message = message; + this.type = type; + } +} describe('Integration: Component: gh-alert', function () { setupRenderingTest(); it('renders', async function () { - this.set('message', {message: 'Test message', type: 'success'}); + this.set('message', new Message({message: 'Test message', type: 'success'})); - await render(hbs`{{gh-alert message=message}}`); + await render(hbs``); let alert = this.element.querySelector('article.gh-alert'); expect(alert).to.exist; @@ -19,29 +30,33 @@ describe('Integration: Component: gh-alert', function () { }); it('maps message types to CSS classes', async function () { - this.set('message', {message: 'Test message', type: 'success'}); + this.set('message', new Message({message: 'Test message', type: 'success'})); - await render(hbs`{{gh-alert message=message}}`); + await render(hbs``); let alert = this.element.querySelector('article.gh-alert'); - this.set('message.type', 'success'); + this.message.type = 'success'; + await settled(); expect(alert, 'success class is green').to.have.class('gh-alert-green'); - this.set('message.type', 'error'); + this.message.type = 'error'; + await settled(); expect(alert, 'error class is red').to.have.class('gh-alert-red'); - this.set('message.type', 'warn'); + this.message.type = 'warn'; + await settled(); expect(alert, 'warn class is yellow').to.have.class('gh-alert-blue'); - this.set('message.type', 'info'); + this.message.type = 'info'; + await settled(); expect(alert, 'info class is blue').to.have.class('gh-alert-blue'); }); it('closes notification through notifications service', async function () { - let message = {message: 'Test close', type: 'success'}; + let message = new Message({message: 'Test close', type: 'success'}); this.set('message', message); - await render(hbs`{{gh-alert message=message}}`); + await render(hbs``); expect(find('article.gh-alert')).to.exist; let notifications = this.owner.lookup('service:notifications'); diff --git a/ghost/admin/tests/integration/components/gh-notification-test.js b/ghost/admin/tests/integration/components/gh-notification-test.js index 336c104aca..2adf7ee2d0 100644 --- a/ghost/admin/tests/integration/components/gh-notification-test.js +++ b/ghost/admin/tests/integration/components/gh-notification-test.js @@ -1,17 +1,28 @@ import hbs from 'htmlbars-inline-precompile'; import sinon from 'sinon'; -import {click, find, render} from '@ember/test-helpers'; +import {click, find, render, settled} from '@ember/test-helpers'; import {describe, it} from 'mocha'; import {expect} from 'chai'; import {setupRenderingTest} from 'ember-mocha'; +import {tracked} from '@glimmer/tracking'; + +class Message { + @tracked message; + @tracked type; + + constructor({message, type}) { + this.message = message; + this.type = type; + } +} describe('Integration: Component: gh-notification', function () { setupRenderingTest(); it('renders', async function () { - this.set('message', {message: 'Test message', type: 'success'}); + this.set('message', new Message({message: 'Test message', type: 'success'})); - await render(hbs`{{gh-notification message=message}}`); + await render(hbs``); expect(find('article.gh-notification')).to.exist; @@ -21,28 +32,30 @@ describe('Integration: Component: gh-notification', function () { }); it('maps message types to CSS classes', async function () { - this.set('message', {message: 'Test message', type: 'success'}); + this.set('message', new Message({message: 'Test message', type: 'success'})); - await render(hbs`{{gh-notification message=message}}`); + await render(hbs``); let notification = find('.gh-notification'); - this.set('message.type', 'error'); + this.message.type = 'error'; + await settled(); expect(notification, 'success class is red') .to.have.class('gh-notification-red'); - this.set('message.type', 'warn'); + this.message.type = 'warn'; + await settled(); expect(notification, 'success class is yellow') .to.have.class('gh-notification-yellow'); }); it('closes notification through notifications service', async function () { - let message = {message: 'Test close', type: 'success'}; + let message = new Message({message: 'Test close', type: 'success'}); this.set('message', message); let notifications = this.owner.lookup('service:notifications'); notifications.closeNotification = sinon.stub(); - await render(hbs`{{gh-notification message=message}}`); + await render(hbs``); expect(find('.gh-notification')).to.exist; await click('[data-test-button="close-notification"]'); diff --git a/ghost/admin/tests/unit/services/notifications-test.js b/ghost/admin/tests/unit/services/notifications-test.js index fdde06e775..32cd34a0cf 100644 --- a/ghost/admin/tests/unit/services/notifications-test.js +++ b/ghost/admin/tests/unit/services/notifications-test.js @@ -5,7 +5,6 @@ import {ServerUnreachableError} from 'ghost-admin/services/ajax'; import {describe, it} from 'mocha'; import {A as emberA} from '@ember/array'; import {expect} from 'chai'; -import {get} from '@ember/object'; import {run} from '@ember/runloop'; import {setupTest} from 'ember-mocha'; @@ -31,11 +30,11 @@ describe('Unit: Service: notifications', function () { notifications.showNotification('Notification'); }); - expect(notifications.get('alerts.length')).to.equal(1); - expect(notifications.get('alerts.firstObject.message')).to.equal('Alert'); + expect(notifications.alerts.length).to.equal(1); + expect(notifications.alerts.firstObject.message).to.equal('Alert'); - expect(notifications.get('notifications.length')).to.equal(1); - expect(notifications.get('notifications.firstObject.message')).to.equal('Notification'); + expect(notifications.notifications.length).to.equal(1); + expect(notifications.notifications.firstObject.message).to.equal('Notification'); }); it('#handleNotification deals with DS.Notification notifications', function () { @@ -44,10 +43,10 @@ describe('Unit: Service: notifications', function () { notifications.handleNotification(notification); - notification = notifications.get('alerts')[0]; + notification = notifications.alerts[0]; // alerts received from the server should be marked html safe - expect(notification.get('message')).to.have.property('toHTML'); + expect(notification.message).to.have.property('toHTML'); }); it('#handleNotification defaults to notification if no status supplied', function () { @@ -55,7 +54,7 @@ describe('Unit: Service: notifications', function () { notifications.handleNotification({message: 'Test'}, false); - expect(notifications.get('content')) + expect(notifications.content) .to.deep.include({message: 'Test', status: 'notification'}); }); @@ -66,7 +65,7 @@ describe('Unit: Service: notifications', function () { notifications.showAlert('Test Alert', {type: 'error'}); }); - expect(notifications.get('alerts')) + expect(notifications.alerts) .to.deep.include({message: 'Test Alert', status: 'alert', type: 'error', key: undefined, actions: undefined, description: undefined, icon: undefined}); }); @@ -77,7 +76,7 @@ describe('Unit: Service: notifications', function () { notifications.showNotification('Test Alert', {type: 'error', delayed: true}); }); - expect(notifications.get('delayedNotifications')) + expect(notifications.delayedNotifications) .to.deep.include({message: 'Test Alert', status: 'notification', type: 'error', key: undefined, actions: undefined, description: undefined, icon: undefined}); }); @@ -92,14 +91,14 @@ describe('Unit: Service: notifications', function () { notifications.showAlert('Duplicate', {key: 'duplicate.key.fail'}); }); - expect(notifications.get('alerts.length')).to.equal(2); + expect(notifications.alerts.length).to.equal(2); run(() => { notifications.showAlert('Duplicate with new message', {key: 'duplicate.key.success'}); }); - expect(notifications.get('alerts.length')).to.equal(2); - expect(notifications.get('alerts.lastObject.message')).to.equal('Duplicate with new message'); + expect(notifications.alerts.length).to.equal(2); + expect(notifications.alerts.lastObject.message).to.equal('Duplicate with new message'); }); it('#showAlert clears duplicates using message text', function () { @@ -109,8 +108,8 @@ describe('Unit: Service: notifications', function () { notifications.showAlert('Duplicate', {key: 'duplicate'}); notifications.showAlert('Duplicate'); - expect(notifications.get('alerts.length')).to.equal(2); - expect(notifications.get('alerts.lastObject.key')).to.not.exist; + expect(notifications.alerts.length).to.equal(2); + expect(notifications.alerts.lastObject.key).to.not.exist; }); it('#showNotification adds POJO notifications', function () { @@ -120,7 +119,7 @@ describe('Unit: Service: notifications', function () { notifications.showNotification('Test Notification', {type: 'success'}); }); - expect(notifications.get('notifications')) + expect(notifications.notifications) .to.deep.include({message: 'Test Notification', status: 'notification', type: 'success', key: undefined, actions: undefined, description: undefined, icon: undefined}); }); @@ -131,7 +130,7 @@ describe('Unit: Service: notifications', function () { notifications.showNotification('Test Notification', {delayed: true}); }); - expect(notifications.get('delayedNotifications')) + expect(notifications.delayedNotifications) .to.deep.include({message: 'Test Notification', status: 'notification', type: undefined, key: undefined, actions: undefined, description: undefined, icon: undefined}); }); @@ -143,11 +142,11 @@ describe('Unit: Service: notifications', function () { notifications.showAPIError(error); }); - let alert = notifications.get('alerts.firstObject'); - expect(get(alert, 'message')).to.equal('Single error'); - expect(get(alert, 'status')).to.equal('alert'); - expect(get(alert, 'type')).to.equal('error'); - expect(get(alert, 'key')).to.equal('api-error'); + let [alert] = notifications.alerts; + expect(alert.message).to.equal('Single error'); + expect(alert.status).to.equal('alert'); + expect(alert.type).to.equal('error'); + expect(alert.key).to.equal('api-error'); }); it('#showAPIError handles multiple json response errors', function () { @@ -161,8 +160,8 @@ describe('Unit: Service: notifications', function () { notifications.showAPIError(error); }); - expect(notifications.get('alerts.length')).to.equal(2); - let [alert1, alert2] = notifications.get('alerts'); + expect(notifications.alerts.length).to.equal(2); + let [alert1, alert2] = notifications.alerts; expect(alert1).to.deep.equal({message: 'First error message', status: 'alert', type: 'error', key: 'api-error.first-error', actions: undefined, description: undefined, icon: undefined}); expect(alert2).to.deep.equal({message: 'Second error message', status: 'alert', type: 'error', key: 'api-error.second-error', actions: undefined, description: undefined, icon: undefined}); }); @@ -174,7 +173,7 @@ describe('Unit: Service: notifications', function () { run(() => { notifications.showAPIError(resp); }); - expect(notifications.get('content').toArray()).to.deep.equal([ + expect(notifications.content).to.deep.equal([ {message: 'There was a problem on the server, please try again.', status: 'alert', type: 'error', key: 'api-error', actions: undefined, description: undefined, icon: undefined} ]); @@ -183,7 +182,7 @@ describe('Unit: Service: notifications', function () { run(() => { notifications.showAPIError(resp, {defaultErrorText: 'Overridden default'}); }); - expect(notifications.get('content').toArray()).to.deep.equal([ + expect(notifications.content).to.deep.equal([ {message: 'Overridden default', status: 'alert', type: 'error', key: 'api-error', actions: undefined, description: undefined, icon: undefined} ]); }); @@ -195,7 +194,7 @@ describe('Unit: Service: notifications', function () { notifications.showAPIError('Test', {key: 'test.alert'}); }); - expect(notifications.get('alerts.firstObject.key')).to.equal('api-error.test.alert'); + expect(notifications.alerts.firstObject.key).to.equal('api-error.test.alert'); }); it('#showAPIError sets correct key when not passed a key', function () { @@ -205,7 +204,7 @@ describe('Unit: Service: notifications', function () { notifications.showAPIError('Test'); }); - expect(notifications.get('alerts.firstObject.key')).to.equal('api-error'); + expect(notifications.alerts.firstObject.key).to.equal('api-error'); }); it('#showAPIError parses default ember-ajax errors correctly', function () { @@ -216,11 +215,11 @@ describe('Unit: Service: notifications', function () { notifications.showAPIError(error); }); - let notification = notifications.get('alerts.firstObject'); - expect(get(notification, 'message')).to.equal('Request was rejected because it was invalid'); - expect(get(notification, 'status')).to.equal('alert'); - expect(get(notification, 'type')).to.equal('error'); - expect(get(notification, 'key')).to.equal('api-error'); + let notification = notifications.alerts.firstObject; + expect(notification.message).to.equal('Request was rejected because it was invalid'); + expect(notification.status).to.equal('alert'); + expect(notification.type).to.equal('error'); + expect(notification.key).to.equal('api-error'); }); it('#showAPIError parses custom ember-ajax errors correctly', function () { @@ -231,11 +230,11 @@ describe('Unit: Service: notifications', function () { notifications.showAPIError(error); }); - let notification = notifications.get('alerts.firstObject'); - expect(get(notification, 'message')).to.equal('Server was unreachable'); - expect(get(notification, 'status')).to.equal('alert'); - expect(get(notification, 'type')).to.equal('error'); - expect(get(notification, 'key')).to.equal('api-error'); + let notification = notifications.alerts.firstObject; + expect(notification.message).to.equal('Server was unreachable'); + expect(notification.status).to.equal('alert'); + expect(notification.type).to.equal('error'); + expect(notification.key).to.equal('api-error'); }); it('#showAPIError adds error context to message if available', function () { @@ -249,11 +248,11 @@ describe('Unit: Service: notifications', function () { notifications.showAPIError(error); }); - let alert = notifications.get('alerts.firstObject'); - expect(get(alert, 'message')).to.equal('Authorization Error. Please sign in.'); - expect(get(alert, 'status')).to.equal('alert'); - expect(get(alert, 'type')).to.equal('error'); - expect(get(alert, 'key')).to.equal('api-error'); + let [alert] = notifications.alerts; + expect(alert.message).to.equal('Authorization Error. Please sign in.'); + expect(alert.status).to.equal('alert'); + expect(alert.type).to.equal('error'); + expect(alert.key).to.equal('api-error'); }); it('#displayDelayed moves delayed notifications into content', function () { @@ -266,7 +265,7 @@ describe('Unit: Service: notifications', function () { notifications.displayDelayed(); }); - expect(notifications.get('notifications')).to.deep.equal([ + expect(notifications.notifications).to.deep.equal([ {message: 'Third', status: 'notification', type: undefined, key: undefined, actions: undefined, description: undefined, icon: undefined}, {message: 'First', status: 'notification', type: undefined, key: undefined, actions: undefined, description: undefined, icon: undefined}, {message: 'Second', status: 'notification', type: undefined, key: undefined, actions: undefined, description: undefined, icon: undefined} @@ -281,14 +280,14 @@ describe('Unit: Service: notifications', function () { notifications.handleNotification(notification); }); - expect(notifications.get('notifications')) + expect(notifications.notifications) .to.include(notification); run(() => { notifications.closeNotification(notification); }); - expect(notifications.get('notifications')) + expect(notifications.notifications) .to.not.include(notification); }); @@ -311,7 +310,7 @@ describe('Unit: Service: notifications', function () { notifications.handleNotification(notification); }); - expect(notifications.get('alerts')).to.include(notification); + expect(notifications.alerts).to.include(notification); run(() => { notifications.closeNotification(notification); @@ -320,7 +319,7 @@ describe('Unit: Service: notifications', function () { expect(notification.deleteRecord.calledOnce).to.be.true; expect(notification.save.calledOnce).to.be.true; - expect(notifications.get('alerts')).to.not.include(notification); + expect(notifications.alerts).to.not.include(notification); }); it('#closeNotifications only removes notifications', function () { @@ -332,15 +331,15 @@ describe('Unit: Service: notifications', function () { notifications.showNotification('Second notification'); }); - expect(notifications.get('alerts.length'), 'alerts count').to.equal(1); - expect(notifications.get('notifications.length'), 'notifications count').to.equal(2); + expect(notifications.alerts.length, 'alerts count').to.equal(1); + expect(notifications.notifications.length, 'notifications count').to.equal(2); run(() => { notifications.closeNotifications(); }); - expect(notifications.get('alerts.length'), 'alerts count').to.equal(1); - expect(notifications.get('notifications.length'), 'notifications count').to.equal(0); + expect(notifications.alerts.length, 'alerts count').to.equal(1); + expect(notifications.notifications.length, 'notifications count').to.equal(0); }); it('#closeNotifications only closes notifications with specified key', function () { @@ -358,9 +357,9 @@ describe('Unit: Service: notifications', function () { notifications.closeNotifications('test.close'); }); - expect(notifications.get('notifications.length'), 'notifications count').to.equal(1); - expect(notifications.get('notifications.firstObject.message'), 'notification message').to.equal('Second notification'); - expect(notifications.get('alerts.length'), 'alerts count').to.equal(1); + expect(notifications.notifications.length, 'notifications count').to.equal(1); + expect(notifications.notifications.firstObject.message, 'notification message').to.equal('Second notification'); + expect(notifications.alerts.length, 'alerts count').to.equal(1); }); it('#clearAll removes everything without deletion', function () { @@ -383,7 +382,7 @@ describe('Unit: Service: notifications', function () { notifications.clearAll(); - expect(notifications.get('content')).to.be.empty; + expect(notifications.content).to.be.empty; expect(notificationModel.deleteRecord.called).to.be.false; expect(notificationModel.save.called).to.be.false; }); @@ -399,8 +398,8 @@ describe('Unit: Service: notifications', function () { notifications.closeAlerts(); }); - expect(notifications.get('alerts.length')).to.equal(0); - expect(notifications.get('notifications.length')).to.equal(1); + expect(notifications.alerts.length).to.equal(0); + expect(notifications.notifications.length).to.equal(1); }); it('#closeAlerts closes only alerts with specified key', function () { @@ -415,8 +414,8 @@ describe('Unit: Service: notifications', function () { notifications.closeAlerts('test.close'); }); - expect(notifications.get('alerts.length')).to.equal(1); - expect(notifications.get('alerts.firstObject.message')).to.equal('Second alert'); - expect(notifications.get('notifications.length')).to.equal(1); + expect(notifications.alerts.length).to.equal(1); + expect(notifications.alerts.firstObject.message).to.equal('Second alert'); + expect(notifications.notifications.length).to.equal(1); }); });