Skip to content

Commit

Permalink
feat: headless applepay and googlepay (#454)
Browse files Browse the repository at this point in the history
  • Loading branch information
ArushKapoorJuspay authored Jun 24, 2024
1 parent 84de985 commit 238d806
Show file tree
Hide file tree
Showing 15 changed files with 1,194 additions and 503 deletions.
63 changes: 15 additions & 48 deletions public/icons/orca.svg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
96 changes: 85 additions & 11 deletions src/Components/SavedMethods.res
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ let make = (
~loadSavedCards: PaymentType.savedCardsLoadState,
~cvcProps,
~paymentType,
~sessions,
) => {
open CardUtils
open Utils
Expand All @@ -20,9 +21,21 @@ let make = (
loggerState.setLogError(~value=message, ~eventName=INVALID_FORMAT, ())
}
let (isSaveCardsChecked, setIsSaveCardsChecked) = React.useState(_ => false)
let {displaySavedPaymentMethodsCheckbox} = Recoil.useRecoilValueFromAtom(RecoilAtoms.optionAtom)
let {displaySavedPaymentMethodsCheckbox, readOnly} = Recoil.useRecoilValueFromAtom(
RecoilAtoms.optionAtom,
)
let isGuestCustomer = useIsGuestCustomer()

let {iframeId} = Recoil.useRecoilValueFromAtom(RecoilAtoms.keys)
let url = RescriptReactRouter.useUrl()
let componentName = CardUtils.getQueryParamsDictforKey(url.search, "componentName")

let dict = sessions->Utils.getDictFromJson
let sessionObj = React.useMemo(() => SessionsType.itemToObjMapper(dict, Others), [dict])

let isApplePayReady = Recoil.useRecoilValueFromAtom(RecoilAtoms.isApplePayReady)
let isGPayReady = Recoil.useRecoilValueFromAtom(RecoilAtoms.isGooglePayReady)

let intent = PaymentHelpers.usePaymentIntent(Some(loggerState), Card)
let savedCardlength = savedMethods->Array.length
let paymentMethodListValue = Recoil.useRecoilValueFromAtom(PaymentUtils.paymentMethodListValue)
Expand All @@ -46,6 +59,13 @@ let make = (

let bottomElement = {
savedMethods
->Array.filter(savedMethod => {
switch savedMethod.paymentMethodType {
| Some("apple_pay") => isApplePayReady
| Some("google_pay") => isGPayReady
| _ => true
}
})
->Array.mapWithIndex((obj, i) => {
let brandIcon = switch obj.paymentMethod {
| "wallet" => getWalletBrandIcon(obj)
Expand Down Expand Up @@ -101,6 +121,10 @@ let make = (

useHandlePostMessages(~complete, ~empty, ~paymentType, ~savedMethod=true)

GooglePayHelpers.useHandleGooglePayResponse(~connectors=[], ~intent)

ApplePayHelpers.useHandleApplePayResponse(~connectors=[], ~intent)

let submitCallback = React.useCallback((ev: Window.event) => {
let json = ev.data->JSON.parseExn
let confirm = json->getDictFromJson->ConfirmType.itemToObjMapper
Expand Down Expand Up @@ -137,16 +161,66 @@ let make = (
(!isCardPaymentMethod || isCardPaymentMethodValid) &&
confirm.confirmTimestamp >= confirm.readyTimestamp
) {
intent(
~bodyArr=savedPaymentMethodBody
->getJsonFromArrayOfJson
->flattenObject(true)
->mergeTwoFlattenedJsonDicts(requiredFieldsBody)
->getArrayOfTupleFromDict,
~confirmParam=confirm.confirmParams,
~handleUserError=false,
(),
)
switch customerMethod.paymentMethodType {
| Some("google_pay") => {
let gPayToken = SessionsType.getPaymentSessionObj(sessionObj.sessionsToken, Gpay)
switch gPayToken {
| OtherTokenOptional(optToken) =>
GooglePayHelpers.handleGooglePayClicked(
~sessionObj=optToken,
~componentName,
~iframeId,
~readOnly,
)
| _ =>
// TODO - To be replaced with proper error message
intent(
~bodyArr=savedPaymentMethodBody
->getJsonFromArrayOfJson
->flattenObject(true)
->mergeTwoFlattenedJsonDicts(requiredFieldsBody)
->getArrayOfTupleFromDict,
~confirmParam=confirm.confirmParams,
~handleUserError=false,
(),
)
}
}
| Some("apple_pay") => {
let applePaySessionObj = SessionsType.itemToObjMapper(dict, ApplePayObject)
let applePayToken = SessionsType.getPaymentSessionObj(
applePaySessionObj.sessionsToken,
ApplePay,
)
switch applePayToken {
| ApplePayTokenOptional(optToken) =>
ApplePayHelpers.handleApplePayButtonClicked(~sessionObj=optToken, ~componentName)
| _ =>
// TODO - To be replaced with proper error message
intent(
~bodyArr=savedPaymentMethodBody
->getJsonFromArrayOfJson
->flattenObject(true)
->mergeTwoFlattenedJsonDicts(requiredFieldsBody)
->getArrayOfTupleFromDict,
~confirmParam=confirm.confirmParams,
~handleUserError=false,
(),
)
}
}
| _ =>
intent(
~bodyArr=savedPaymentMethodBody
->getJsonFromArrayOfJson
->flattenObject(true)
->mergeTwoFlattenedJsonDicts(requiredFieldsBody)
->getArrayOfTupleFromDict,
~confirmParam=confirm.confirmParams,
~handleUserError=false,
(),
)
}
} else {
if isUnknownPaymentMethod || confirm.confirmTimestamp < confirm.readyTimestamp {
setUserError(localeString.selectPaymentMethodText)
Expand Down
17 changes: 13 additions & 4 deletions src/PaymentElement.res
Original file line number Diff line number Diff line change
Expand Up @@ -53,12 +53,13 @@ let make = (~cardProps, ~expiryProps, ~cvcProps, ~paymentType: CardThemeType.mod
}
| (_, LoadingSavedCards) => ()
| (_, LoadedSavedCards(savedPaymentMethods, isGuestCustomer)) => {
let displayDefaultSavedPaymentIcon = optionAtomValue.displayDefaultSavedPaymentIcon
let sortSavedPaymentMethods = (a, b) => {
let defaultCompareVal = compareLogic(
Date.fromString(a.lastUsedAt),
Date.fromString(b.lastUsedAt),
)
if optionAtomValue.displayDefaultSavedPaymentIcon {
if displayDefaultSavedPaymentIcon {
if a.defaultPaymentMethodSet {
-1.
} else if b.defaultPaymentMethodSet {
Expand All @@ -74,7 +75,15 @@ let make = (~cardProps, ~expiryProps, ~cvcProps, ~paymentType: CardThemeType.mod
let finalSavedPaymentMethods = savedPaymentMethods->Array.copy
finalSavedPaymentMethods->Array.sort(sortSavedPaymentMethods)

setSavedMethods(_ => finalSavedPaymentMethods)
let paymentOrder = paymentMethodOrder->getOptionalArr->removeDuplicate

let sortSavedMethodsBasedOnPriority =
finalSavedPaymentMethods->PaymentUtils.sortCustomerMethodsBasedOnPriority(
paymentOrder,
~displayDefaultSavedPaymentIcon,
)

setSavedMethods(_ => sortSavedMethodsBasedOnPriority)
setLoadSavedCards(_ =>
finalSavedPaymentMethods->Array.length == 0
? NoResult(isGuestCustomer)
Expand All @@ -89,7 +98,7 @@ let make = (~cardProps, ~expiryProps, ~cvcProps, ~paymentType: CardThemeType.mod
}

None
}, (customerPaymentMethods, displaySavedPaymentMethods))
}, (customerPaymentMethods, displaySavedPaymentMethods, optionAtomValue))

React.useEffect(() => {
let defaultSelectedPaymentMethod = optionAtomValue.displayDefaultSavedPaymentIcon
Expand Down Expand Up @@ -335,7 +344,7 @@ let make = (~cardProps, ~expiryProps, ~cvcProps, ~paymentType: CardThemeType.mod
</div>
</RenderIf>
<RenderIf condition={!showFields && displaySavedPaymentMethods}>
<SavedMethods paymentToken setPaymentToken savedMethods loadSavedCards cvcProps paymentType />
<SavedMethods paymentToken setPaymentToken savedMethods loadSavedCards cvcProps paymentType sessions />
</RenderIf>
<RenderIf
condition={(paymentOptions->Array.length > 0 || walletOptions->Array.length > 0) &&
Expand Down
Loading

0 comments on commit 238d806

Please sign in to comment.