import Component from '@ember/component'; import ShortcutsMixin from 'ghost-admin/mixins/shortcuts'; import ctrlOrCmd from 'ghost-admin/utils/ctrl-or-cmd'; import {and, equal, match, or} from '@ember/object/computed'; import {computed} from '@ember/object'; import {getOwner} from '@ember/application'; import {htmlSafe} from '@ember/template'; import {inject as service} from '@ember/service'; import {task} from 'ember-concurrency'; export default Component.extend(ShortcutsMixin, { billing: service(), config: service(), customViews: service(), feature: service(), ghostPaths: service(), navigation: service(), router: service(), session: service(), ui: service(), whatsNew: service(), membersStats: service(), settings: service(), tagName: '', iconStyle: '', iconClass: '', memberCountLoading: true, memberCount: 0, showSearchModal: false, shortcuts: null, isIntegrationRoute: match('router.currentRouteName', /^settings\.integration/), // HACK: {{link-to}} should be doing this automatically but there appears to // be a bug in Ember that's preventing it from working immediately after login isOnSite: equal('router.currentRouteName', 'site'), showTagsNavigation: or('session.user.isAdmin', 'session.user.isEditor'), showMenuExtension: and('config.clientExtensions.menu', 'session.user.isOwnerOnly'), showScriptExtension: and('config.clientExtensions.script', 'session.user.isOwnerOnly'), showBilling: computed.reads('config.hostSettings.billing.enabled'), isStripeConnected: computed.reads('settings.stripeConnectAccountId'), init() { this._super(...arguments); let shortcuts = {}; shortcuts[`${ctrlOrCmd}+k`] = {action: 'toggleSearchModal'}; this.shortcuts = shortcuts; }, // the menu has a rendering issue (#8307) when the the world is reloaded // during an import which we have worked around by not binding the icon // style directly. However we still need to keep track of changing icons // so that we can refresh when a new icon is uploaded didReceiveAttrs() { this._super(...arguments); this._setIconStyle(); this._loadMemberCountsTask.perform(); }, didInsertElement() { this._super(...arguments); this.registerShortcuts(); }, willDestroyElement() { this.removeShortcuts(); this._super(...arguments); }, actions: { transitionToOrRefreshSite() { let {currentRouteName} = this.router; if (currentRouteName === 'site') { getOwner(this).lookup(`route:${currentRouteName}`).refresh(); } else { this.router.transitionTo('site'); } }, toggleSearchModal() { this.toggleProperty('showSearchModal'); }, toggleBillingModal() { this.billing.openBillingWindow(this.router.currentURL); } }, _loadMemberCountsTask: task(function* () { try { this.set('memberCountLoading', true); const stats = yield this.membersStats.fetchCounts(); this.set('memberCountLoading', false); if (stats) { const statsDateObj = this.membersStats.fillCountDates(stats.data) || {}; const dateValues = Object.values(statsDateObj); this.set('memberCount', dateValues.length ? dateValues[dateValues.length - 1].total : 0); } } catch (e) { return false; } }), _setIconStyle() { let icon = this.icon; if (icon === this._icon) { return; } this._icon = icon; if (icon && icon.match(/^https?:\/\//i)) { this.set('iconClass', ''); this.set('iconStyle', htmlSafe(`background-image: url(${icon})`)); return; } let iconUrl = 'https://static.ghost.org/v4.0.0/images/ghost-orb-1.png'; this.set('iconStyle', htmlSafe(`background-image: url(${iconUrl})`)); this.set('iconClass', 'gh-nav-logo-default'); } });