2020-01-27 06:41:12 -05:00
|
|
|
// # {{price}} helper
|
|
|
|
//
|
|
|
|
// Usage: `{{price 2100}}`
|
2021-02-23 05:55:28 -05:00
|
|
|
// Usage: `{{price plan}}`
|
|
|
|
// Usage: `{{price plan numberFormat="long"}}`
|
|
|
|
// Usage: `{{price plan currencyFormat="code"}}`
|
|
|
|
// Usage: `{{price plan currencyFormat="name"}}`
|
|
|
|
// Usage: `{{price 500 currency="USD"}}`
|
|
|
|
// Usage: `{{price currency="USD"}}`
|
2020-01-27 06:41:12 -05:00
|
|
|
//
|
|
|
|
// Returns amount equal to the dominant denomintation of the currency.
|
|
|
|
// For example, if 2100 is passed, it will return 21.
|
2021-09-26 15:01:13 -05:00
|
|
|
const errors = require('@tryghost/errors');
|
|
|
|
const tpl = require('@tryghost/tpl');
|
2021-02-23 05:55:28 -05:00
|
|
|
const _ = require('lodash');
|
|
|
|
|
2021-09-23 14:46:22 -05:00
|
|
|
const messages = {
|
|
|
|
attrIsRequired: 'Attribute is required e.g. {{price plan.amount}}',
|
|
|
|
attrMustBeNumeric: 'Attribute value should be a number'
|
|
|
|
};
|
|
|
|
|
2021-02-23 05:55:28 -05:00
|
|
|
function formatter({amount, currency, numberFormat = 'short', currencyFormat = 'symbol', locale}) {
|
|
|
|
const formatterOptions = {
|
|
|
|
style: 'currency',
|
|
|
|
currency: currency,
|
|
|
|
currencyDisplay: currencyFormat
|
|
|
|
};
|
|
|
|
if (numberFormat === 'short') {
|
|
|
|
formatterOptions.minimumFractionDigits = 0;
|
|
|
|
}
|
2021-02-25 09:38:37 -05:00
|
|
|
if (_.isNumber(amount)) {
|
2021-02-23 05:55:28 -05:00
|
|
|
return new Intl.NumberFormat(locale, formatterOptions).format(amount);
|
|
|
|
} else {
|
|
|
|
const val = new Intl.NumberFormat('en', {
|
|
|
|
style: 'currency',
|
|
|
|
currency,
|
|
|
|
currencyDisplay: 'symbol',
|
|
|
|
minimumFractionDigits: 0,
|
|
|
|
maximumFractionDigits: 0
|
|
|
|
}).format(0);
|
|
|
|
return val.replace(/[\d\s.,]/g, '');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = function price(planOrAmount, options) {
|
|
|
|
let plan;
|
|
|
|
let amount;
|
|
|
|
if (arguments.length === 1) {
|
|
|
|
options = planOrAmount;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (arguments.length === 2) {
|
|
|
|
if (_.isNumber(planOrAmount)) {
|
|
|
|
amount = planOrAmount;
|
|
|
|
} else if (_.isObject(planOrAmount)) {
|
|
|
|
plan = planOrAmount;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
options = options || {};
|
|
|
|
options.hash = options.hash || {};
|
2022-05-12 09:07:05 -05:00
|
|
|
|
|
|
|
const {currency, numberFormat = 'short', currencyFormat = 'symbol', locale = _.get(options, 'data.site.locale', 'en')} = options.hash;
|
2021-02-23 05:55:28 -05:00
|
|
|
if (plan) {
|
|
|
|
return formatter({
|
|
|
|
amount: plan.amount && (plan.amount / 100),
|
|
|
|
currency: currency || plan.currency,
|
|
|
|
currencyFormat,
|
|
|
|
numberFormat,
|
|
|
|
locale
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
if (currency) {
|
|
|
|
return formatter({
|
|
|
|
amount: amount && (amount / 100),
|
|
|
|
currency: currency,
|
|
|
|
currencyFormat,
|
|
|
|
numberFormat,
|
|
|
|
locale
|
|
|
|
});
|
|
|
|
}
|
2020-01-27 06:41:12 -05:00
|
|
|
|
|
|
|
// CASE: if no amount is passed, e.g. `{{price}}` we throw an error
|
|
|
|
if (arguments.length < 2) {
|
|
|
|
throw new errors.IncorrectUsageError({
|
2021-09-23 14:46:22 -05:00
|
|
|
message: tpl(messages.attrIsRequired)
|
2020-01-27 06:41:12 -05:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
// CASE: if amount is passed, but it is undefined we throw an error
|
|
|
|
if (amount === undefined) {
|
|
|
|
throw new errors.IncorrectUsageError({
|
2021-09-23 14:46:22 -05:00
|
|
|
message: tpl(messages.attrIsRequired)
|
2020-01-27 06:41:12 -05:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-09-26 15:01:13 -05:00
|
|
|
if (!_.isNumber(amount)) {
|
2020-01-27 06:41:12 -05:00
|
|
|
throw new errors.IncorrectUsageError({
|
2021-09-23 14:46:22 -05:00
|
|
|
message: tpl(messages.attrMustBeNumeric)
|
2020-01-27 06:41:12 -05:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
return amount / 100;
|
|
|
|
};
|