mirror of
https://github.com/TryGhost/Ghost.git
synced 2025-01-08 12:09:43 +03:00
c449e32121
closes https://github.com/TryGhost/Team/issues/678 closes https://github.com/TryGhost/Team/issues/681 The prices in "Add subscription" modal should follow the same ordering as on the product detail screen, ie. currency, amount. Also, we only want to allow adding subscriptions for active prices, so the list is filtered on that. Since a Stripe customer is not allowed to have subscriptions in multiple currencies, this also filters the available currency prices based on any active subscription for a member.
109 lines
2.8 KiB
JavaScript
109 lines
2.8 KiB
JavaScript
import ModalComponent from 'ghost-admin/components/modal-base';
|
|
import {action} from '@ember/object';
|
|
import {getNonDecimal, getSymbol} from 'ghost-admin/utils/currency';
|
|
import {inject as service} from '@ember/service';
|
|
import {task} from 'ember-concurrency-decorators';
|
|
import {tracked} from '@glimmer/tracking';
|
|
|
|
export default class ModalMemberProduct extends ModalComponent {
|
|
@service
|
|
store
|
|
|
|
@service
|
|
ghostPaths
|
|
|
|
@service
|
|
ajax
|
|
|
|
@tracked
|
|
price
|
|
|
|
@tracked
|
|
product
|
|
|
|
@tracked
|
|
products = []
|
|
|
|
constructor(...args) {
|
|
super(...args);
|
|
this.fetchProducts();
|
|
}
|
|
|
|
async fetchProducts() {
|
|
this.products = await this.store.query('product', {include: 'stripe_prices'});
|
|
this.product = this.products.firstObject;
|
|
this.price = this.prices ? this.prices[0] : null;
|
|
}
|
|
|
|
get member() {
|
|
return this.model;
|
|
}
|
|
|
|
get cannotAddPrice() {
|
|
return !this.price || this.price.amount !== 0;
|
|
}
|
|
|
|
get prices() {
|
|
if (!this.products || !this.products.length) {
|
|
return [];
|
|
}
|
|
if (this.product) {
|
|
let subscriptions = this.member.get('subscriptions') || [];
|
|
let activeCurrency;
|
|
if (subscriptions.length > 0) {
|
|
activeCurrency = subscriptions[0].price?.currency;
|
|
}
|
|
|
|
const product = this.products.find((_product) => {
|
|
return _product.id === this.product.id;
|
|
});
|
|
return product.stripePrices.sort((a, b) => {
|
|
return a.amount - b.amount;
|
|
}).filter((price) => {
|
|
return price.active;
|
|
}).filter((price) => {
|
|
if (activeCurrency) {
|
|
return price.currency?.toLowerCase() === activeCurrency.toLowerCase();
|
|
}
|
|
return true;
|
|
}).sort((a, b) => {
|
|
return a.currency.localeCompare(b.currency, undefined, {ignorePunctuation: true});
|
|
}).map((price) => {
|
|
return {
|
|
...price,
|
|
label: `${price.nickname} (${getSymbol(price.currency)}${getNonDecimal(price.amount)}/${price.interval})`
|
|
};
|
|
});
|
|
} else {
|
|
return [];
|
|
}
|
|
}
|
|
|
|
@action
|
|
setProduct(product) {
|
|
this.product = product;
|
|
}
|
|
|
|
@action
|
|
setPrice(price) {
|
|
this.price = price;
|
|
}
|
|
|
|
@task({
|
|
drop: true
|
|
})
|
|
*addPriceTask() {
|
|
let url = this.ghostPaths.url.api('members', this.member.get('id'), 'subscriptions');
|
|
|
|
let response = yield this.ajax.post(url, {
|
|
data: {
|
|
stripe_price_id: this.price.stripe_price_id
|
|
}
|
|
});
|
|
|
|
this.store.pushPayload('member', response);
|
|
this.closeModal();
|
|
return response;
|
|
}
|
|
}
|