import Component from '@glimmer/component'; import envConfig from 'ghost-admin/config/environment'; import {action} from '@ember/object'; import {currencies, getCurrencyOptions, getSymbol} from 'ghost-admin/utils/currency'; import {inject as service} from '@ember/service'; import {task} from 'ember-concurrency-decorators'; import {tracked} from '@glimmer/tracking'; const CURRENCIES = currencies.map((currency) => { return { value: currency.isoCode.toLowerCase(), label: `${currency.isoCode} - ${currency.name}`, isoCode: currency.isoCode }; }); export default class GhLaunchWizardSetPricingComponent extends Component { @service config; @service membersUtils; @service settings; @service store; currencies = CURRENCIES; @tracked stripeMonthlyAmount = 5; @tracked stripeYearlyAmount = 50; @tracked currency = 'usd'; @tracked isFreeChecked = true; @tracked isMonthlyChecked = true; @tracked isYearlyChecked = true; @tracked stripePlanError = ''; @tracked product; @tracked loadingProduct = false; get selectedCurrency() { return this.currencies.findBy('value', this.currency); } get allCurrencies() { return getCurrencyOptions(); } get isConnectDisallowed() { const siteUrl = this.config.get('blogUrl'); return envConfig.environment !== 'development' && !/^https:/.test(siteUrl); } get disabled() { return false; } get isPaidPriceDisabled() { return this.disabled || !this.membersUtils.isStripeEnabled; } get isFreeDisabled() { return this.disabled || this.settings.get('membersSignupAccess') !== 'all'; } getPrice(prices, type) { const monthlyPriceId = this.settings.get('membersMonthlyPriceId'); const yearlyPriceId = this.settings.get('membersYearlyPriceId'); if (type === 'monthly') { return ( prices.find(price => price.id === monthlyPriceId) || prices.find(price => price.nickname === 'Monthly') || prices.find(price => price.interval === 'month') ); } if (type === 'yearly') { return ( prices.find(price => price.id === yearlyPriceId) || prices.find(price => price.nickname === 'Yearly') || prices.find(price => price.interval === 'year') ); } return null; } @action setup() { this.fetchDefaultProduct.perform(); this.updatePreviewUrl(); } willDestroy() { // clear any unsaved settings changes when going back/forward/closing this.args.updatePreview(''); } @action backStep() { const product = this.product; const data = this.args.getData() || {}; this.args.storeData({ ...data, product, isFreeChecked: this.isFreeChecked, isMonthlyChecked: this.isMonthlyChecked, isYearlyChecked: this.isYearlyChecked, monthlyAmount: this.stripeMonthlyAmount, yearlyAmount: this.stripeYearlyAmount, currency: this.currency }); this.args.backStep(); } @action setStripePlansCurrency(event) { const newCurrency = event.value; this.currency = newCurrency; this.updatePreviewUrl(); } @action toggleFreePlan(event) { this.isFreeChecked = event.target.checked; this.updatePreviewUrl(); } @action toggleMonthlyPlan(event) { this.isMonthlyChecked = event.target.checked; this.updatePreviewUrl(); } @action toggleYearlyPlan(event) { this.isYearlyChecked = event.target.checked; this.updatePreviewUrl(); } @action validateStripePlans() { this.stripePlanError = undefined; try { const yearlyAmount = this.stripeYearlyAmount; const monthlyAmount = this.stripeMonthlyAmount; const symbol = getSymbol(this.currency); if (!yearlyAmount || yearlyAmount < 1 || !monthlyAmount || monthlyAmount < 1) { throw new TypeError(`Subscription amount must be at least ${symbol}1.00`); } this.updatePreviewUrl(); } catch (err) { this.stripePlanError = err.message; } } @task *saveAndContinue() { if (this.isConnectDisallowed) { this.args.nextStep(); } else { yield this.validateStripePlans(); if (this.stripePlanError) { return false; } const product = this.product; const data = this.args.getData() || {}; this.args.storeData({ ...data, product, isFreeChecked: this.isFreeChecked, isMonthlyChecked: this.isMonthlyChecked, isYearlyChecked: this.isYearlyChecked, monthlyAmount: this.stripeMonthlyAmount, yearlyAmount: this.stripeYearlyAmount, currency: this.currency }); this.args.nextStep(); } } @task({drop: true}) *fetchDefaultProduct() { const storedData = this.args.getData(); if (storedData?.product) { this.product = storedData.product; this.stripePrices = this.product.get('stripePrices') || []; if (storedData.isMonthlyChecked !== undefined) { this.isMonthlyChecked = storedData.isMonthlyChecked; } if (storedData.isYearlyChecked !== undefined) { this.isYearlyChecked = storedData.isYearlyChecked; } if (storedData.isFreeChecked !== undefined) { this.isFreeChecked = storedData.isFreeChecked; } if (storedData.currency !== undefined) { this.currency = storedData.currency; } this.stripeMonthlyAmount = storedData.monthlyAmount; this.stripeYearlyAmount = storedData.yearlyAmount; } else { const products = yield this.store.query('product', {include: 'stripe_prices'}); this.product = products.firstObject; let portalPlans = this.settings.get('portalPlans') || []; const currentMontlyPriceId = this.settings.get('membersMonthlyPriceId'); const currentYearlyPriceId = this.settings.get('membersYearlyPriceId'); this.isMonthlyChecked = false; this.isYearlyChecked = false; this.isFreeChecked = false; if (portalPlans.includes(currentMontlyPriceId)) { this.isMonthlyChecked = true; } if (portalPlans.includes(currentYearlyPriceId)) { this.isYearlyChecked = true; } if (portalPlans.includes('free')) { this.isFreeChecked = true; } this.stripePrices = this.product.get('stripePrices') || []; const activePrices = this.stripePrices.filter(price => !!price.active); const monthlyPrice = this.getPrice(activePrices, 'monthly'); const yearlyPrice = this.getPrice(activePrices, 'yearly'); if (monthlyPrice && monthlyPrice.amount) { this.stripeMonthlyAmount = (monthlyPrice.amount / 100); this.currency = monthlyPrice.currency; } if (yearlyPrice && yearlyPrice.amount) { this.stripeYearlyAmount = (yearlyPrice.amount / 100); } } this.updatePreviewUrl(); } updatePreviewUrl() { const options = { disableBackground: true, currency: this.selectedCurrency.value, monthlyPrice: this.stripeMonthlyAmount * 100, yearlyPrice: this.stripeYearlyAmount * 100, isMonthlyChecked: this.isMonthlyChecked, isYearlyChecked: this.isYearlyChecked, isFreeChecked: this.isFreeChecked, portalPlans: null }; const url = this.membersUtils.getPortalPreviewUrl(options); this.args.updatePreview(url); } }