2023-02-02 09:15:04 +00:00
|
|
|
import {
|
|
|
|
useState,
|
|
|
|
useEffect,
|
|
|
|
useLayoutEffect,
|
|
|
|
useMemo,
|
|
|
|
useCallback,
|
|
|
|
useRef,
|
|
|
|
useContext,
|
|
|
|
createContext,
|
|
|
|
} from 'react'
|
2023-02-09 09:48:40 +00:00
|
|
|
import { currencies, CurrencyCode, CurrencySymbol } from '../data/currency'
|
2023-02-02 09:15:04 +00:00
|
|
|
import { useTranslation } from 'react-i18next'
|
|
|
|
import getMeta from '../../../utils/meta'
|
|
|
|
import * as eventTracking from '../../../infrastructure/event-tracking'
|
|
|
|
import {
|
|
|
|
PaymentContextValue,
|
|
|
|
PricingFormState,
|
|
|
|
} from './types/payment-context-value'
|
|
|
|
import { Plan } from '../../../../../types/subscription/plan'
|
|
|
|
import {
|
|
|
|
RecurlyOptions,
|
|
|
|
SubscriptionPricingStateTax,
|
|
|
|
} from 'recurly__recurly-js'
|
|
|
|
import { SubscriptionPricingInstanceCustom } from '../../../../../types/recurly/pricing/subscription'
|
|
|
|
|
|
|
|
function usePayment({ publicKey }: RecurlyOptions) {
|
|
|
|
const { t } = useTranslation()
|
|
|
|
const plan: Plan = getMeta('ol-plan')
|
|
|
|
const initialCountry: PricingFormState['country'] = getMeta(
|
|
|
|
'ol-countryCode',
|
|
|
|
''
|
|
|
|
)
|
|
|
|
const initialCouponCode: PricingFormState['coupon'] = getMeta(
|
|
|
|
'ol-couponCode',
|
|
|
|
''
|
|
|
|
)
|
2023-02-09 09:48:40 +00:00
|
|
|
const initiallySelectedCurrencyCode: CurrencyCode = getMeta(
|
2023-02-02 09:15:04 +00:00
|
|
|
'ol-recommendedCurrency'
|
|
|
|
)
|
2023-02-09 09:48:40 +00:00
|
|
|
const planCode: string = getMeta('ol-planCode')
|
2023-02-02 09:15:04 +00:00
|
|
|
|
2023-02-09 09:48:40 +00:00
|
|
|
const [planName, setPlanName] = useState(plan.name)
|
2023-02-02 09:15:04 +00:00
|
|
|
const [recurlyLoading, setRecurlyLoading] = useState(true)
|
|
|
|
const [recurlyLoadError, setRecurlyLoadError] = useState(false)
|
|
|
|
const [recurlyPrice, setRecurlyPrice] = useState<{
|
|
|
|
subtotal: string
|
|
|
|
plan: string
|
|
|
|
addons: string
|
|
|
|
setup_fee: string
|
|
|
|
discount: string
|
|
|
|
tax: string
|
|
|
|
total: string
|
|
|
|
}>()
|
|
|
|
const [monthlyBilling, setMonthlyBilling] = useState<boolean>()
|
|
|
|
const [taxes, setTaxes] = useState<SubscriptionPricingStateTax[]>([])
|
|
|
|
const [coupon, setCoupon] = useState<{
|
|
|
|
discountMonths?: number
|
|
|
|
discountRate?: number
|
|
|
|
singleUse: boolean
|
|
|
|
normalPrice: number
|
|
|
|
name: string
|
|
|
|
normalPriceWithoutTax: number
|
|
|
|
}>()
|
|
|
|
const [couponError, setCouponError] = useState('')
|
|
|
|
const [trialLength, setTrialLength] = useState<number>()
|
|
|
|
const [currencyCode, setCurrencyCode] = useState(
|
|
|
|
initiallySelectedCurrencyCode
|
|
|
|
)
|
|
|
|
const [pricingFormState, setPricingFormState] = useState<PricingFormState>({
|
|
|
|
first_name: '',
|
|
|
|
last_name: '',
|
|
|
|
postal_code: '',
|
|
|
|
address1: '',
|
|
|
|
address2: '',
|
|
|
|
state: '',
|
|
|
|
city: '',
|
|
|
|
company: '',
|
|
|
|
vat_number: '',
|
|
|
|
country: initialCountry,
|
|
|
|
coupon: initialCouponCode,
|
|
|
|
})
|
|
|
|
const pricing = useRef<SubscriptionPricingInstanceCustom>()
|
|
|
|
|
|
|
|
const limitedCurrencyCodes = Array.from(
|
2023-02-09 09:48:40 +00:00
|
|
|
new Set<CurrencyCode>([initiallySelectedCurrencyCode, 'USD', 'EUR', 'GBP'])
|
2023-02-02 09:15:04 +00:00
|
|
|
)
|
|
|
|
const limitedCurrencies = limitedCurrencyCodes.reduce((prev, cur) => {
|
2023-02-09 09:48:40 +00:00
|
|
|
return { ...prev, [cur]: currencies[cur] }
|
|
|
|
}, {} as Partial<typeof currencies>)
|
|
|
|
const currencySymbol = limitedCurrencies[currencyCode] as CurrencySymbol
|
2023-02-02 09:15:04 +00:00
|
|
|
|
|
|
|
useLayoutEffect(() => {
|
|
|
|
if (typeof recurly === 'undefined' || !recurly) {
|
|
|
|
setRecurlyLoadError(true)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-05-02 12:45:55 +00:00
|
|
|
eventTracking.sendMB('payment-page-view', {
|
|
|
|
plan: planCode,
|
|
|
|
currency: currencyCode,
|
|
|
|
})
|
2023-02-02 09:15:04 +00:00
|
|
|
eventTracking.send(
|
|
|
|
'subscription-funnel',
|
|
|
|
'subscription-form-viewed',
|
2023-02-09 09:48:40 +00:00
|
|
|
planCode
|
2023-02-02 09:15:04 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
recurly.configure({ publicKey })
|
|
|
|
pricing.current =
|
|
|
|
recurly.Pricing.Subscription() as SubscriptionPricingInstanceCustom
|
|
|
|
|
|
|
|
const setupPricing = () => {
|
|
|
|
setRecurlyLoading(true)
|
|
|
|
|
|
|
|
pricing.current
|
2023-02-09 09:48:40 +00:00
|
|
|
?.plan(planCode, { quantity: 1 })
|
2023-02-02 09:15:04 +00:00
|
|
|
.address({
|
|
|
|
first_name: '',
|
|
|
|
last_name: '',
|
|
|
|
country: initialCountry,
|
|
|
|
})
|
|
|
|
.tax({ tax_code: 'digital', vat_number: '' })
|
2023-05-04 15:29:36 +00:00
|
|
|
.currency(currencyCode)
|
2023-02-02 09:15:04 +00:00
|
|
|
.coupon(initialCouponCode)
|
|
|
|
.catch(function (err) {
|
2023-05-04 15:29:36 +00:00
|
|
|
if (currencyCode !== 'USD' && err.name === 'invalid-currency') {
|
2023-02-02 09:15:04 +00:00
|
|
|
setCurrencyCode('USD')
|
|
|
|
setupPricing()
|
|
|
|
} else if (err.name === 'api-error' && err.code === 'not-found') {
|
|
|
|
// not-found here should refer to the coupon code, plan_code should be valid
|
|
|
|
setCouponError(t('coupon_code_is_not_valid_for_selected_plan'))
|
|
|
|
} else {
|
|
|
|
// Bail out on other errors, form state will not be correct
|
|
|
|
setRecurlyLoadError(true)
|
|
|
|
throw err
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.done(() => {
|
|
|
|
setRecurlyLoading(false)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
setupPricing()
|
|
|
|
}, [
|
|
|
|
initialCountry,
|
|
|
|
initialCouponCode,
|
|
|
|
initiallySelectedCurrencyCode,
|
2023-02-09 09:48:40 +00:00
|
|
|
planCode,
|
2023-02-02 09:15:04 +00:00
|
|
|
publicKey,
|
2023-05-02 12:45:55 +00:00
|
|
|
currencyCode,
|
2023-02-02 09:15:04 +00:00
|
|
|
t,
|
|
|
|
])
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
pricing.current?.on('change', function () {
|
|
|
|
if (!pricing.current) return
|
|
|
|
|
2023-02-09 09:48:40 +00:00
|
|
|
const planName = pricing.current.items.plan?.name
|
|
|
|
if (planName) {
|
|
|
|
setPlanName(planName)
|
|
|
|
}
|
|
|
|
|
2023-02-02 09:15:04 +00:00
|
|
|
const trialLength = pricing.current.items.plan?.trial?.length
|
|
|
|
setTrialLength(trialLength)
|
|
|
|
|
|
|
|
const recurlyPrice = trialLength
|
|
|
|
? pricing.current.price.next
|
|
|
|
: pricing.current.price.now
|
|
|
|
setRecurlyPrice(recurlyPrice)
|
|
|
|
|
|
|
|
const monthlyBilling = pricing.current.items.plan?.period.length === 1
|
|
|
|
setMonthlyBilling(monthlyBilling)
|
|
|
|
|
|
|
|
setTaxes(pricing.current.price.taxes)
|
|
|
|
|
|
|
|
const couponData = (() => {
|
|
|
|
if (pricing.current.items.coupon?.discount.type === 'percent') {
|
|
|
|
const coupon = pricing.current.items.coupon
|
|
|
|
const basePrice = parseInt(pricing.current.price.base.plan.unit, 10)
|
|
|
|
const discountData =
|
|
|
|
coupon.applies_for_months > 0 && coupon.discount.rate
|
|
|
|
? {
|
|
|
|
discountMonths: coupon.applies_for_months,
|
|
|
|
discountRate: coupon.discount.rate * 100,
|
|
|
|
}
|
|
|
|
: {}
|
|
|
|
|
|
|
|
const couponData = {
|
|
|
|
singleUse: coupon.single_use,
|
|
|
|
normalPrice: basePrice,
|
|
|
|
name: coupon.name,
|
|
|
|
normalPriceWithoutTax: basePrice,
|
|
|
|
...discountData,
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pricing.current.price.taxes[0]?.rate) {
|
|
|
|
couponData.normalPrice +=
|
|
|
|
basePrice * parseFloat(pricing.current.price.taxes[0].rate)
|
|
|
|
}
|
|
|
|
|
|
|
|
return couponData
|
|
|
|
}
|
|
|
|
})()
|
|
|
|
setCoupon(couponData)
|
|
|
|
})
|
2023-05-04 15:29:36 +00:00
|
|
|
}, [currencyCode])
|
2023-02-02 09:15:04 +00:00
|
|
|
|
|
|
|
const addCoupon = useCallback(
|
|
|
|
(coupon: PricingFormState['coupon']) => {
|
|
|
|
setRecurlyLoading(true)
|
|
|
|
setCouponError('')
|
|
|
|
|
|
|
|
pricing.current
|
|
|
|
?.coupon(coupon)
|
|
|
|
.catch(function (err) {
|
|
|
|
if (err.name === 'api-error' && err.code === 'not-found') {
|
|
|
|
setCouponError(t('coupon_code_is_not_valid_for_selected_plan'))
|
|
|
|
} else {
|
|
|
|
setCouponError(
|
|
|
|
t('an_error_occurred_when_verifying_the_coupon_code')
|
|
|
|
)
|
|
|
|
throw err
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.done(() => {
|
|
|
|
setRecurlyLoading(false)
|
|
|
|
})
|
|
|
|
},
|
|
|
|
[t]
|
|
|
|
)
|
|
|
|
|
|
|
|
const updateCountry = useCallback(
|
|
|
|
(country: PricingFormState['country']) => {
|
|
|
|
setRecurlyLoading(true)
|
|
|
|
|
|
|
|
pricing.current
|
|
|
|
?.address({
|
|
|
|
country,
|
|
|
|
first_name: pricingFormState.first_name,
|
|
|
|
last_name: pricingFormState.last_name,
|
|
|
|
})
|
|
|
|
.done(() => {
|
|
|
|
setRecurlyLoading(false)
|
|
|
|
})
|
|
|
|
},
|
|
|
|
[pricingFormState.first_name, pricingFormState.last_name]
|
|
|
|
)
|
|
|
|
|
|
|
|
const applyVatNumber = useCallback(
|
|
|
|
(vatNumber: PricingFormState['vat_number']) => {
|
|
|
|
setRecurlyLoading(true)
|
|
|
|
|
|
|
|
pricing.current
|
|
|
|
?.tax({ tax_code: 'digital', vat_number: vatNumber })
|
|
|
|
.done(() => {
|
|
|
|
setRecurlyLoading(false)
|
|
|
|
})
|
|
|
|
},
|
|
|
|
[]
|
|
|
|
)
|
|
|
|
|
|
|
|
const changeCurrency = useCallback(
|
2023-02-09 09:48:40 +00:00
|
|
|
(newCurrency: CurrencyCode) => {
|
2023-02-02 09:15:04 +00:00
|
|
|
setRecurlyLoading(true)
|
|
|
|
setCurrencyCode(newCurrency)
|
|
|
|
|
|
|
|
pricing.current
|
|
|
|
?.currency(newCurrency)
|
|
|
|
.catch(function (err) {
|
|
|
|
if (currencyCode !== 'USD' && err.name === 'invalid-currency') {
|
|
|
|
setCurrencyCode('USD')
|
|
|
|
} else {
|
|
|
|
throw err
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.done(() => {
|
|
|
|
setRecurlyLoading(false)
|
|
|
|
})
|
|
|
|
},
|
|
|
|
[currencyCode]
|
|
|
|
)
|
|
|
|
|
|
|
|
const value = useMemo<PaymentContextValue>(
|
|
|
|
() => ({
|
|
|
|
currencyCode,
|
|
|
|
setCurrencyCode,
|
|
|
|
currencySymbol,
|
|
|
|
limitedCurrencies,
|
|
|
|
pricingFormState,
|
|
|
|
setPricingFormState,
|
|
|
|
plan,
|
2023-02-09 09:48:40 +00:00
|
|
|
planCode,
|
|
|
|
planName,
|
2023-02-02 09:15:04 +00:00
|
|
|
pricing,
|
|
|
|
recurlyLoading,
|
|
|
|
recurlyLoadError,
|
|
|
|
recurlyPrice,
|
|
|
|
monthlyBilling,
|
|
|
|
taxes,
|
|
|
|
coupon,
|
|
|
|
couponError,
|
|
|
|
trialLength,
|
|
|
|
addCoupon,
|
|
|
|
applyVatNumber,
|
|
|
|
changeCurrency,
|
|
|
|
updateCountry,
|
|
|
|
}),
|
|
|
|
[
|
|
|
|
currencyCode,
|
|
|
|
setCurrencyCode,
|
|
|
|
currencySymbol,
|
|
|
|
limitedCurrencies,
|
|
|
|
pricingFormState,
|
|
|
|
setPricingFormState,
|
|
|
|
plan,
|
2023-02-09 09:48:40 +00:00
|
|
|
planCode,
|
|
|
|
planName,
|
2023-02-02 09:15:04 +00:00
|
|
|
pricing,
|
|
|
|
recurlyLoading,
|
|
|
|
recurlyLoadError,
|
|
|
|
recurlyPrice,
|
|
|
|
monthlyBilling,
|
|
|
|
taxes,
|
|
|
|
coupon,
|
|
|
|
couponError,
|
|
|
|
trialLength,
|
|
|
|
addCoupon,
|
|
|
|
applyVatNumber,
|
|
|
|
changeCurrency,
|
|
|
|
updateCountry,
|
|
|
|
]
|
|
|
|
)
|
|
|
|
|
|
|
|
return { value }
|
|
|
|
}
|
|
|
|
|
2023-02-14 08:17:02 +00:00
|
|
|
export const PaymentContext = createContext<PaymentContextValue | undefined>(
|
|
|
|
undefined
|
|
|
|
)
|
2023-02-02 09:15:04 +00:00
|
|
|
|
|
|
|
type PaymentProviderProps = {
|
|
|
|
publicKey: string
|
2023-02-28 10:04:44 +00:00
|
|
|
children?: React.ReactNode
|
2023-02-02 09:15:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export function PaymentProvider({ publicKey, ...props }: PaymentProviderProps) {
|
|
|
|
const { value } = usePayment({ publicKey })
|
|
|
|
|
|
|
|
return <PaymentContext.Provider value={value} {...props} />
|
|
|
|
}
|
|
|
|
|
|
|
|
export function usePaymentContext() {
|
|
|
|
const context = useContext(PaymentContext)
|
|
|
|
if (!context) {
|
|
|
|
throw new Error('PaymentContext is only available inside PaymentProvider')
|
|
|
|
}
|
|
|
|
return context
|
|
|
|
}
|