@reevit/react 0.4.2 → 0.4.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/index.d.mts +19 -4
- package/dist/index.d.ts +19 -4
- package/dist/index.js +212 -178
- package/dist/index.js.map +1 -1
- package/dist/index.mjs +212 -178
- package/dist/index.mjs.map +1 -1
- package/dist/styles.css +262 -67
- package/package.json +1 -1
package/dist/index.mjs
CHANGED
|
@@ -581,14 +581,6 @@ function detectCountryFromCurrency(currency) {
|
|
|
581
581
|
};
|
|
582
582
|
return currencyToCountry[currency.toUpperCase()] || "GH";
|
|
583
583
|
}
|
|
584
|
-
var pspNames = {
|
|
585
|
-
hubtel: "Hubtel",
|
|
586
|
-
paystack: "Paystack",
|
|
587
|
-
flutterwave: "Flutterwave",
|
|
588
|
-
monnify: "Monnify",
|
|
589
|
-
mpesa: "M-Pesa",
|
|
590
|
-
stripe: "Stripe"
|
|
591
|
-
};
|
|
592
584
|
var methodConfig = {
|
|
593
585
|
card: {
|
|
594
586
|
label: "Card",
|
|
@@ -611,28 +603,31 @@ function PaymentMethodSelector({
|
|
|
611
603
|
selectedMethod,
|
|
612
604
|
onSelect,
|
|
613
605
|
disabled = false,
|
|
614
|
-
provider
|
|
606
|
+
provider,
|
|
607
|
+
layout = "list",
|
|
608
|
+
showLabel = true
|
|
615
609
|
}) {
|
|
616
610
|
const getMethodLabel = (method, psp) => {
|
|
617
611
|
const config = methodConfig[method];
|
|
618
|
-
if (psp?.toLowerCase().includes("hubtel") && method === "mobile_money") {
|
|
619
|
-
return `Pay with ${pspNames[psp.toLowerCase()] || "Hubtel"}`;
|
|
620
|
-
}
|
|
621
612
|
return config.label;
|
|
622
613
|
};
|
|
623
614
|
const getMethodDescription = (method, psp) => {
|
|
624
615
|
const config = methodConfig[method];
|
|
625
616
|
if (psp?.toLowerCase().includes("hubtel")) {
|
|
626
|
-
return
|
|
617
|
+
return config.description;
|
|
627
618
|
}
|
|
628
619
|
return config.description;
|
|
629
620
|
};
|
|
630
|
-
|
|
631
|
-
|
|
632
|
-
/* @__PURE__ */ jsx("div", { className: "reevit-method-
|
|
621
|
+
const isGrid = layout === "grid";
|
|
622
|
+
return /* @__PURE__ */ jsxs("div", { className: cn("reevit-method-selector", isGrid && "reevit-method-selector--grid"), children: [
|
|
623
|
+
showLabel && /* @__PURE__ */ jsx("div", { className: "reevit-method-selector__label", children: "Select payment method" }),
|
|
624
|
+
/* @__PURE__ */ jsx("div", { className: cn(
|
|
625
|
+
"reevit-method-selector__options",
|
|
626
|
+
isGrid ? "reevit-method-selector__options--grid" : "reevit-method-selector__options--list"
|
|
627
|
+
), children: methods.map((method, index) => {
|
|
633
628
|
const config = methodConfig[method];
|
|
634
629
|
const isSelected = selectedMethod === method;
|
|
635
|
-
const methodLabel = getMethodLabel(method
|
|
630
|
+
const methodLabel = getMethodLabel(method);
|
|
636
631
|
const methodDescription = getMethodDescription(method, provider);
|
|
637
632
|
return /* @__PURE__ */ jsxs(
|
|
638
633
|
"button",
|
|
@@ -640,28 +635,24 @@ function PaymentMethodSelector({
|
|
|
640
635
|
type: "button",
|
|
641
636
|
className: cn(
|
|
642
637
|
"reevit-method-option",
|
|
638
|
+
isGrid ? "reevit-method-option--grid" : "reevit-method-option--list",
|
|
643
639
|
isSelected && "reevit-method-option--selected",
|
|
644
640
|
disabled && "reevit-method-option--disabled"
|
|
645
641
|
),
|
|
642
|
+
style: {
|
|
643
|
+
animationDelay: `${index * 0.05}s`
|
|
644
|
+
},
|
|
646
645
|
onClick: () => onSelect(method),
|
|
647
646
|
disabled,
|
|
648
647
|
"aria-pressed": isSelected,
|
|
649
648
|
children: [
|
|
650
|
-
/* @__PURE__ */ jsx("span", { className: "reevit-method-option__icon", children: config.icon }),
|
|
649
|
+
/* @__PURE__ */ jsx("span", { className: "reevit-method-option__icon-wrapper", children: /* @__PURE__ */ jsx("span", { className: "reevit-method-option__icon", children: config.icon }) }),
|
|
651
650
|
/* @__PURE__ */ jsxs("div", { className: "reevit-method-option__content", children: [
|
|
652
651
|
/* @__PURE__ */ jsx("span", { className: "reevit-method-option__label", children: methodLabel }),
|
|
653
|
-
/* @__PURE__ */ jsx("span", { className: "reevit-method-option__description", children: methodDescription })
|
|
652
|
+
!isGrid && /* @__PURE__ */ jsx("span", { className: "reevit-method-option__description", children: methodDescription })
|
|
654
653
|
] }),
|
|
655
|
-
isSelected && /* @__PURE__ */ jsx("span", { className: "reevit-method-option__check", children: /* @__PURE__ */ jsx("svg", { width: "20", height: "20", viewBox: "0 0
|
|
656
|
-
|
|
657
|
-
{
|
|
658
|
-
d: "M16.667 5L7.5 14.167 3.333 10",
|
|
659
|
-
stroke: "currentColor",
|
|
660
|
-
strokeWidth: "2",
|
|
661
|
-
strokeLinecap: "round",
|
|
662
|
-
strokeLinejoin: "round"
|
|
663
|
-
}
|
|
664
|
-
) }) })
|
|
654
|
+
!isGrid && isSelected && /* @__PURE__ */ jsx("span", { className: "reevit-method-option__check", children: /* @__PURE__ */ jsx("svg", { width: "20", height: "20", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "3", strokeLinecap: "round", strokeLinejoin: "round", children: /* @__PURE__ */ jsx("polyline", { points: "20 6 9 17 4 12" }) }) }),
|
|
655
|
+
!isGrid && !isSelected && /* @__PURE__ */ jsx("span", { className: "reevit-method-option__chevron", children: /* @__PURE__ */ jsx("svg", { width: "16", height: "16", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2.5", strokeLinecap: "round", strokeLinejoin: "round", children: /* @__PURE__ */ jsx("polyline", { points: "9 18 15 12 9 6" }) }) })
|
|
665
656
|
]
|
|
666
657
|
},
|
|
667
658
|
method
|
|
@@ -678,7 +669,8 @@ function MobileMoneyForm({
|
|
|
678
669
|
onSubmit,
|
|
679
670
|
onCancel,
|
|
680
671
|
isLoading = false,
|
|
681
|
-
initialPhone = ""
|
|
672
|
+
initialPhone = "",
|
|
673
|
+
hideCancel = false
|
|
682
674
|
}) {
|
|
683
675
|
const [phone, setPhone] = useState(initialPhone);
|
|
684
676
|
const [network, setNetwork] = useState(null);
|
|
@@ -763,7 +755,7 @@ function MobileMoneyForm({
|
|
|
763
755
|
)) })
|
|
764
756
|
] }),
|
|
765
757
|
/* @__PURE__ */ jsxs("div", { className: "reevit-momo-form__actions", children: [
|
|
766
|
-
/* @__PURE__ */ jsx(
|
|
758
|
+
!hideCancel && /* @__PURE__ */ jsx(
|
|
767
759
|
"button",
|
|
768
760
|
{
|
|
769
761
|
type: "button",
|
|
@@ -805,11 +797,11 @@ var paystack_default = "
|
|
|
805
797
|
// src/assets/providers/stripe.png
|
|
806
798
|
var stripe_default = "";
|
|
807
799
|
var providerMeta = {
|
|
808
|
-
paystack: { logo: paystack_default, hint: "Card &
|
|
800
|
+
paystack: { logo: paystack_default, hint: "Card & Mobile Money" },
|
|
809
801
|
stripe: { logo: stripe_default, hint: "Card payments" },
|
|
810
802
|
flutterwave: { logo: flutterwave_default, hint: "Global methods" },
|
|
811
|
-
hubtel: { logo: hubtel_default, hint: "Mobile
|
|
812
|
-
monnify: { logo: monnify_default, hint: "Bank &
|
|
803
|
+
hubtel: { logo: hubtel_default, hint: "Mobile Money & Card" },
|
|
804
|
+
monnify: { logo: monnify_default, hint: "Bank & Card" },
|
|
813
805
|
mpesa: { logo: mpesa_default, hint: "M-Pesa" }
|
|
814
806
|
};
|
|
815
807
|
var methodLabels = {
|
|
@@ -819,54 +811,100 @@ var methodLabels = {
|
|
|
819
811
|
};
|
|
820
812
|
function formatMethods(methods) {
|
|
821
813
|
if (!methods.length) return "Payment methods";
|
|
822
|
-
return methods.map((method) => methodLabels[method]).join("
|
|
814
|
+
return methods.map((method) => methodLabels[method]).join(", ");
|
|
815
|
+
}
|
|
816
|
+
function sanitizeMethods(providerId, methods) {
|
|
817
|
+
if (providerId.toLowerCase().includes("hubtel")) {
|
|
818
|
+
return methods.filter((method) => method === "card" || method === "mobile_money");
|
|
819
|
+
}
|
|
820
|
+
return methods;
|
|
823
821
|
}
|
|
824
822
|
function ProviderSelector({
|
|
825
823
|
providers,
|
|
826
824
|
selectedProvider,
|
|
827
825
|
onSelect,
|
|
828
|
-
disabled = false
|
|
826
|
+
disabled = false,
|
|
827
|
+
theme,
|
|
828
|
+
selectedMethod,
|
|
829
|
+
onMethodSelect,
|
|
830
|
+
renderMethodContent
|
|
829
831
|
}) {
|
|
832
|
+
const useBorder = theme?.pspSelectorUseBorder ?? false;
|
|
833
|
+
const bgColor = theme?.pspSelectorBgColor || "#0a0a0a";
|
|
834
|
+
const textColor = theme?.pspSelectorTextColor || "#ffffff";
|
|
835
|
+
const borderColor = theme?.pspSelectorBorderColor || "#374151";
|
|
836
|
+
const getOptionStyle = (isSelected) => {
|
|
837
|
+
if (useBorder) {
|
|
838
|
+
return {
|
|
839
|
+
backgroundColor: "transparent",
|
|
840
|
+
border: `2px solid ${isSelected ? borderColor : "#374151"}`,
|
|
841
|
+
color: isSelected ? textColor : "var(--reevit-text)"
|
|
842
|
+
};
|
|
843
|
+
}
|
|
844
|
+
return {
|
|
845
|
+
backgroundColor: isSelected ? bgColor : "transparent",
|
|
846
|
+
border: `2px solid ${isSelected ? bgColor : "#374151"}`,
|
|
847
|
+
color: isSelected ? textColor : "var(--reevit-text)"
|
|
848
|
+
};
|
|
849
|
+
};
|
|
830
850
|
return /* @__PURE__ */ jsxs("div", { className: "reevit-psp-selector", children: [
|
|
831
|
-
/* @__PURE__ */ jsx("div", { className: "reevit-psp-selector__label", children: "
|
|
851
|
+
/* @__PURE__ */ jsx("div", { className: "reevit-psp-selector__label", children: "Select payment provider" }),
|
|
832
852
|
/* @__PURE__ */ jsx("div", { className: "reevit-psp-selector__options", children: providers.map((provider) => {
|
|
833
853
|
const meta = providerMeta[provider.provider] || {
|
|
834
854
|
logo: void 0,
|
|
835
855
|
hint: "Payment methods"
|
|
836
856
|
};
|
|
857
|
+
const providerMethods = sanitizeMethods(provider.provider, provider.methods);
|
|
837
858
|
const isSelected = selectedProvider === provider.provider;
|
|
838
859
|
const fallbackInitial = provider.name.slice(0, 1).toUpperCase();
|
|
839
|
-
|
|
840
|
-
|
|
841
|
-
|
|
842
|
-
|
|
843
|
-
|
|
844
|
-
"
|
|
845
|
-
|
|
846
|
-
|
|
847
|
-
|
|
848
|
-
|
|
849
|
-
|
|
850
|
-
|
|
851
|
-
|
|
852
|
-
|
|
853
|
-
|
|
854
|
-
|
|
855
|
-
|
|
856
|
-
|
|
857
|
-
|
|
858
|
-
|
|
859
|
-
|
|
860
|
-
|
|
861
|
-
|
|
862
|
-
|
|
863
|
-
/* @__PURE__ */ jsx("span", { className: "reevit-psp-
|
|
864
|
-
|
|
865
|
-
|
|
866
|
-
|
|
867
|
-
|
|
868
|
-
|
|
869
|
-
|
|
860
|
+
const optionStyle = getOptionStyle(isSelected);
|
|
861
|
+
return /* @__PURE__ */ jsxs("div", { className: "reevit-psp-accordion", children: [
|
|
862
|
+
/* @__PURE__ */ jsxs(
|
|
863
|
+
"button",
|
|
864
|
+
{
|
|
865
|
+
type: "button",
|
|
866
|
+
className: cn(
|
|
867
|
+
"reevit-psp-option",
|
|
868
|
+
isSelected && "reevit-psp-option--selected",
|
|
869
|
+
disabled && "reevit-psp-option--disabled"
|
|
870
|
+
),
|
|
871
|
+
style: optionStyle,
|
|
872
|
+
onClick: () => onSelect(provider.provider),
|
|
873
|
+
disabled,
|
|
874
|
+
"aria-expanded": isSelected,
|
|
875
|
+
children: [
|
|
876
|
+
/* @__PURE__ */ jsx("span", { className: "reevit-psp-option__logo", "aria-hidden": "true", children: meta.logo ? /* @__PURE__ */ jsx(
|
|
877
|
+
"img",
|
|
878
|
+
{
|
|
879
|
+
src: meta.logo,
|
|
880
|
+
alt: "",
|
|
881
|
+
className: "reevit-psp-option__logo-img",
|
|
882
|
+
loading: "lazy"
|
|
883
|
+
}
|
|
884
|
+
) : /* @__PURE__ */ jsx("span", { className: "reevit-psp-option__logo-fallback", children: fallbackInitial }) }),
|
|
885
|
+
/* @__PURE__ */ jsxs("div", { className: "reevit-psp-option__content", children: [
|
|
886
|
+
/* @__PURE__ */ jsx("span", { className: "reevit-psp-option__name", children: provider.name }),
|
|
887
|
+
/* @__PURE__ */ jsx("span", { className: "reevit-psp-option__methods", children: formatMethods(providerMethods) || meta.hint })
|
|
888
|
+
] })
|
|
889
|
+
]
|
|
890
|
+
}
|
|
891
|
+
),
|
|
892
|
+
isSelected && onMethodSelect && /* @__PURE__ */ jsxs("div", { className: "reevit-psp-accordion__content", children: [
|
|
893
|
+
/* @__PURE__ */ jsx("div", { className: "reevit-psp-methods", children: /* @__PURE__ */ jsx(
|
|
894
|
+
PaymentMethodSelector,
|
|
895
|
+
{
|
|
896
|
+
methods: providerMethods,
|
|
897
|
+
selectedMethod: selectedMethod || null,
|
|
898
|
+
onSelect: onMethodSelect,
|
|
899
|
+
disabled,
|
|
900
|
+
provider: provider.provider,
|
|
901
|
+
layout: "list",
|
|
902
|
+
showLabel: false
|
|
903
|
+
}
|
|
904
|
+
) }),
|
|
905
|
+
selectedMethod && renderMethodContent && /* @__PURE__ */ jsx("div", { className: "reevit-psp-accordion__method-content", children: renderMethodContent(provider.provider, selectedMethod) })
|
|
906
|
+
] })
|
|
907
|
+
] }, provider.provider);
|
|
870
908
|
}) })
|
|
871
909
|
] });
|
|
872
910
|
}
|
|
@@ -1000,6 +1038,7 @@ function HubtelBridge({
|
|
|
1000
1038
|
callbackUrl,
|
|
1001
1039
|
hubtelSessionToken,
|
|
1002
1040
|
basicAuth,
|
|
1041
|
+
preferredMethod,
|
|
1003
1042
|
onSuccess,
|
|
1004
1043
|
onError,
|
|
1005
1044
|
onClose,
|
|
@@ -1050,12 +1089,14 @@ function HubtelBridge({
|
|
|
1050
1089
|
try {
|
|
1051
1090
|
const checkout = new CheckoutSdk();
|
|
1052
1091
|
checkoutRef.current = checkout;
|
|
1092
|
+
const methodPreference = preferredMethod === "mobile_money" ? "momo" : preferredMethod === "card" ? "card" : void 0;
|
|
1053
1093
|
const purchaseInfo = {
|
|
1054
1094
|
amount: amount / 100,
|
|
1055
1095
|
// Convert from minor to major units
|
|
1056
1096
|
purchaseDescription: description,
|
|
1057
1097
|
customerPhoneNumber: phone || "",
|
|
1058
|
-
clientReference: reference || `hubtel_${Date.now()}
|
|
1098
|
+
clientReference: reference || `hubtel_${Date.now()}`,
|
|
1099
|
+
...methodPreference ? { paymentMethod: methodPreference } : {}
|
|
1059
1100
|
};
|
|
1060
1101
|
const config = {
|
|
1061
1102
|
branding: "enabled",
|
|
@@ -1063,7 +1104,8 @@ function HubtelBridge({
|
|
|
1063
1104
|
merchantAccount: typeof merchantAccount === "string" ? parseInt(merchantAccount, 10) : merchantAccount,
|
|
1064
1105
|
// Use session token or basicAuth for authentication
|
|
1065
1106
|
// Session tokens are base64-encoded credentials fetched securely from the server
|
|
1066
|
-
basicAuth: authValue || ""
|
|
1107
|
+
basicAuth: authValue || "",
|
|
1108
|
+
...methodPreference ? { paymentMethod: methodPreference } : {}
|
|
1067
1109
|
};
|
|
1068
1110
|
checkout.openModal({
|
|
1069
1111
|
purchaseInfo,
|
|
@@ -1107,7 +1149,7 @@ function HubtelBridge({
|
|
|
1107
1149
|
};
|
|
1108
1150
|
onError(error);
|
|
1109
1151
|
}
|
|
1110
|
-
}, [merchantAccount, amount, reference, phone, description, callbackUrl, authValue, isLoading, onSuccess, onError, onClose]);
|
|
1152
|
+
}, [merchantAccount, amount, reference, phone, description, callbackUrl, authValue, isLoading, preferredMethod, onSuccess, onError, onClose]);
|
|
1111
1153
|
useEffect(() => {
|
|
1112
1154
|
if (autoStart && !initialized.current && !isLoading && authValue) {
|
|
1113
1155
|
initialized.current = true;
|
|
@@ -1121,17 +1163,20 @@ function HubtelBridge({
|
|
|
1121
1163
|
}
|
|
1122
1164
|
function openHubtelPopup(config) {
|
|
1123
1165
|
const checkout = new CheckoutSdk();
|
|
1166
|
+
const methodPreference = config.preferredMethod === "mobile_money" ? "momo" : config.preferredMethod === "card" ? "card" : void 0;
|
|
1124
1167
|
const purchaseInfo = {
|
|
1125
1168
|
amount: config.amount,
|
|
1126
1169
|
purchaseDescription: config.description,
|
|
1127
1170
|
customerPhoneNumber: config.customerPhoneNumber || "",
|
|
1128
|
-
clientReference: config.clientReference || `hubtel_${Date.now()}
|
|
1171
|
+
clientReference: config.clientReference || `hubtel_${Date.now()}`,
|
|
1172
|
+
...methodPreference ? { paymentMethod: methodPreference } : {}
|
|
1129
1173
|
};
|
|
1130
1174
|
const checkoutConfig = {
|
|
1131
1175
|
branding: "enabled",
|
|
1132
1176
|
callbackUrl: config.callbackUrl || window.location.href,
|
|
1133
1177
|
merchantAccount: typeof config.merchantAccount === "string" ? parseInt(config.merchantAccount, 10) : config.merchantAccount,
|
|
1134
|
-
basicAuth: config.basicAuth || ""
|
|
1178
|
+
basicAuth: config.basicAuth || "",
|
|
1179
|
+
...methodPreference ? { paymentMethod: methodPreference } : {}
|
|
1135
1180
|
};
|
|
1136
1181
|
checkout.openModal({
|
|
1137
1182
|
purchaseInfo,
|
|
@@ -1824,6 +1869,14 @@ function useReevitContext() {
|
|
|
1824
1869
|
}
|
|
1825
1870
|
return context;
|
|
1826
1871
|
}
|
|
1872
|
+
var pspNames = {
|
|
1873
|
+
hubtel: "Hubtel",
|
|
1874
|
+
paystack: "Paystack",
|
|
1875
|
+
flutterwave: "Flutterwave",
|
|
1876
|
+
monnify: "Monnify",
|
|
1877
|
+
mpesa: "M-Pesa",
|
|
1878
|
+
stripe: "Stripe"
|
|
1879
|
+
};
|
|
1827
1880
|
function ReevitCheckout({
|
|
1828
1881
|
// Config
|
|
1829
1882
|
publicKey,
|
|
@@ -1912,7 +1965,8 @@ function ReevitCheckout({
|
|
|
1912
1965
|
const options = available.length > 0 ? available : fallbackProvider;
|
|
1913
1966
|
return options.map((provider) => {
|
|
1914
1967
|
const methods = provider.methods && provider.methods.length > 0 ? provider.methods : paymentMethods;
|
|
1915
|
-
const
|
|
1968
|
+
const sanitizedMethods = provider.provider.toLowerCase().includes("hubtel") ? methods.filter((method) => method === "card" || method === "mobile_money") : methods;
|
|
1969
|
+
const filteredMethods = sanitizedMethods.filter((method) => paymentMethods.includes(method));
|
|
1916
1970
|
return {
|
|
1917
1971
|
...provider,
|
|
1918
1972
|
methods: filteredMethods
|
|
@@ -1933,10 +1987,10 @@ function ReevitCheckout({
|
|
|
1933
1987
|
if (selectedProvider && providerOptions.some((provider) => provider.provider === selectedProvider)) {
|
|
1934
1988
|
return;
|
|
1935
1989
|
}
|
|
1936
|
-
|
|
1937
|
-
|
|
1938
|
-
|
|
1939
|
-
}, [
|
|
1990
|
+
if (providerOptions.length === 1) {
|
|
1991
|
+
setSelectedProvider(providerOptions[0].provider);
|
|
1992
|
+
}
|
|
1993
|
+
}, [providerOptions, selectedProvider]);
|
|
1940
1994
|
useEffect(() => {
|
|
1941
1995
|
if (!activeProvider || !selectedMethod) {
|
|
1942
1996
|
return;
|
|
@@ -1947,13 +2001,17 @@ function ReevitCheckout({
|
|
|
1947
2001
|
}, [activeProvider, selectedMethod, selectMethod]);
|
|
1948
2002
|
useEffect(() => {
|
|
1949
2003
|
if (isOpen && selectedMethod && paymentIntent && !showPSPBridge) {
|
|
2004
|
+
const psp = (selectedProvider || paymentIntent.recommendedPsp || "paystack").toLowerCase();
|
|
2005
|
+
const needsPhone = psp.includes("mpesa");
|
|
1950
2006
|
if (selectedMethod === "card") {
|
|
1951
2007
|
setShowPSPBridge(true);
|
|
1952
|
-
} else if (selectedMethod === "mobile_money"
|
|
1953
|
-
|
|
2008
|
+
} else if (selectedMethod === "mobile_money") {
|
|
2009
|
+
if (!needsPhone || (momoData?.phone || phone)) {
|
|
2010
|
+
setShowPSPBridge(true);
|
|
2011
|
+
}
|
|
1954
2012
|
}
|
|
1955
2013
|
}
|
|
1956
|
-
}, [isOpen, selectedMethod, showPSPBridge, paymentIntent, momoData, phone]);
|
|
2014
|
+
}, [isOpen, selectedMethod, showPSPBridge, paymentIntent, momoData, phone, selectedProvider]);
|
|
1957
2015
|
const handleOpen = useCallback(() => {
|
|
1958
2016
|
if (controlledIsOpen !== void 0) return;
|
|
1959
2017
|
setIsOpen(true);
|
|
@@ -1977,6 +2035,10 @@ function ReevitCheckout({
|
|
|
1977
2035
|
const handleProviderSelect = useCallback(
|
|
1978
2036
|
(provider) => {
|
|
1979
2037
|
if (provider === selectedProvider) {
|
|
2038
|
+
setSelectedProvider(null);
|
|
2039
|
+
reset();
|
|
2040
|
+
setShowPSPBridge(false);
|
|
2041
|
+
setMomoData(null);
|
|
1980
2042
|
return;
|
|
1981
2043
|
}
|
|
1982
2044
|
const providerMethods = providerOptions.find((option) => option.provider === provider)?.methods || paymentMethods;
|
|
@@ -1993,8 +2055,14 @@ function ReevitCheckout({
|
|
|
1993
2055
|
if (!selectedMethod) return;
|
|
1994
2056
|
if (selectedMethod === "card") {
|
|
1995
2057
|
setShowPSPBridge(true);
|
|
2058
|
+
} else if (selectedMethod === "mobile_money") {
|
|
2059
|
+
const psp = (selectedProvider || paymentIntent?.recommendedPsp || "paystack").toLowerCase();
|
|
2060
|
+
const needsPhone = psp.includes("mpesa");
|
|
2061
|
+
if (!needsPhone || (momoData?.phone || phone)) {
|
|
2062
|
+
setShowPSPBridge(true);
|
|
2063
|
+
}
|
|
1996
2064
|
}
|
|
1997
|
-
}, [selectedMethod]);
|
|
2065
|
+
}, [selectedMethod, selectedProvider, paymentIntent, momoData, phone]);
|
|
1998
2066
|
const handleMomoSubmit = useCallback(
|
|
1999
2067
|
(data) => {
|
|
2000
2068
|
setMomoData(data);
|
|
@@ -2051,13 +2119,13 @@ function ReevitCheckout({
|
|
|
2051
2119
|
] }) : null;
|
|
2052
2120
|
const renderContent = () => {
|
|
2053
2121
|
if (status === "loading" || status === "processing") {
|
|
2054
|
-
return /* @__PURE__ */ jsxs("div", { className: "reevit-loading", children: [
|
|
2122
|
+
return /* @__PURE__ */ jsxs("div", { className: "reevit-loading reevit-animate-fade-in", children: [
|
|
2055
2123
|
/* @__PURE__ */ jsx("div", { className: "reevit-spinner" }),
|
|
2056
2124
|
/* @__PURE__ */ jsx("p", { children: status === "loading" ? "Preparing checkout..." : "Processing payment..." })
|
|
2057
2125
|
] });
|
|
2058
2126
|
}
|
|
2059
2127
|
if (status === "success" && result) {
|
|
2060
|
-
return /* @__PURE__ */ jsxs("div", { className: "reevit-success", children: [
|
|
2128
|
+
return /* @__PURE__ */ jsxs("div", { className: "reevit-success reevit-animate-scale-in", children: [
|
|
2061
2129
|
/* @__PURE__ */ jsx("div", { className: "reevit-success__icon", children: "\u2713" }),
|
|
2062
2130
|
/* @__PURE__ */ jsx("h3", { children: "Payment Successful" }),
|
|
2063
2131
|
/* @__PURE__ */ jsxs("p", { children: [
|
|
@@ -2067,7 +2135,7 @@ function ReevitCheckout({
|
|
|
2067
2135
|
] });
|
|
2068
2136
|
}
|
|
2069
2137
|
if (status === "failed" && error && !error.recoverable) {
|
|
2070
|
-
return /* @__PURE__ */ jsxs("div", { className: "reevit-error", children: [
|
|
2138
|
+
return /* @__PURE__ */ jsxs("div", { className: "reevit-error reevit-animate-fade-in", children: [
|
|
2071
2139
|
/* @__PURE__ */ jsx("div", { className: "reevit-error__icon", children: "\u2715" }),
|
|
2072
2140
|
/* @__PURE__ */ jsx("h3", { children: "Payment Failed" }),
|
|
2073
2141
|
/* @__PURE__ */ jsx("p", { children: error.message }),
|
|
@@ -2075,17 +2143,16 @@ function ReevitCheckout({
|
|
|
2075
2143
|
] });
|
|
2076
2144
|
}
|
|
2077
2145
|
const psp = selectedProvider || paymentIntent?.recommendedPsp || "paystack";
|
|
2146
|
+
const pspLower = psp.toLowerCase();
|
|
2078
2147
|
if (showPSPBridge) {
|
|
2079
2148
|
const pspKey = paymentIntent?.pspPublicKey || publicKey;
|
|
2080
2149
|
const bridgeMetadata = {
|
|
2081
2150
|
...metadata,
|
|
2082
|
-
// Override with correct payment intent ID for webhook routing
|
|
2083
|
-
// This ensures webhook includes the correct ID to find the payment
|
|
2084
2151
|
payment_id: paymentIntent?.id,
|
|
2085
2152
|
connection_id: paymentIntent?.connectionId ?? metadata?.connection_id,
|
|
2086
2153
|
customer_phone: momoData?.phone || phone
|
|
2087
2154
|
};
|
|
2088
|
-
switch (
|
|
2155
|
+
switch (pspLower) {
|
|
2089
2156
|
case "paystack":
|
|
2090
2157
|
return /* @__PURE__ */ jsx(
|
|
2091
2158
|
PaystackBridge,
|
|
@@ -2118,6 +2185,7 @@ function ReevitCheckout({
|
|
|
2118
2185
|
phone: momoData?.phone || phone,
|
|
2119
2186
|
description: `Payment ${paymentIntent?.reference || reference || ""}`,
|
|
2120
2187
|
hubtelSessionToken: paymentIntent?.id ? paymentIntent.id : void 0,
|
|
2188
|
+
preferredMethod: selectedMethod || void 0,
|
|
2121
2189
|
onSuccess: handlePSPSuccess,
|
|
2122
2190
|
onError: (err) => handlePSPError(err),
|
|
2123
2191
|
onClose: handlePSPClose
|
|
@@ -2135,7 +2203,7 @@ function ReevitCheckout({
|
|
|
2135
2203
|
phone: momoData?.phone || phone,
|
|
2136
2204
|
metadata: bridgeMetadata,
|
|
2137
2205
|
onSuccess: handlePSPSuccess,
|
|
2138
|
-
onError:
|
|
2206
|
+
onError: handlePSPError,
|
|
2139
2207
|
onClose: handlePSPClose
|
|
2140
2208
|
}
|
|
2141
2209
|
);
|
|
@@ -2152,21 +2220,8 @@ function ReevitCheckout({
|
|
|
2152
2220
|
customerEmail: email,
|
|
2153
2221
|
customerPhone: momoData?.phone || phone,
|
|
2154
2222
|
metadata: bridgeMetadata,
|
|
2155
|
-
onSuccess: (
|
|
2156
|
-
|
|
2157
|
-
reference: result2.paymentReference,
|
|
2158
|
-
amount: result2.amount,
|
|
2159
|
-
currency: paymentIntent?.currency ?? currency,
|
|
2160
|
-
paymentMethod: selectedMethod || "card",
|
|
2161
|
-
psp: "monnify",
|
|
2162
|
-
pspReference: result2.transactionReference,
|
|
2163
|
-
status: "success"
|
|
2164
|
-
}),
|
|
2165
|
-
onError: (err) => handlePSPError({
|
|
2166
|
-
code: err.code,
|
|
2167
|
-
message: err.message,
|
|
2168
|
-
recoverable: true
|
|
2169
|
-
}),
|
|
2223
|
+
onSuccess: (res) => handlePSPSuccess({ ...res, psp: "monnify" }),
|
|
2224
|
+
onError: handlePSPError,
|
|
2170
2225
|
onClose: handlePSPClose
|
|
2171
2226
|
}
|
|
2172
2227
|
);
|
|
@@ -2181,21 +2236,8 @@ function ReevitCheckout({
|
|
|
2181
2236
|
reference: paymentIntent?.reference || reference || `mpesa_${Date.now()}`,
|
|
2182
2237
|
description: `Payment ${paymentIntent?.reference || reference || ""}`,
|
|
2183
2238
|
headers: { "x-reevit-public-key": publicKey },
|
|
2184
|
-
onSuccess:
|
|
2185
|
-
|
|
2186
|
-
reference: result2.reference,
|
|
2187
|
-
amount: paymentIntent?.amount ?? amount,
|
|
2188
|
-
currency: paymentIntent?.currency ?? currency,
|
|
2189
|
-
paymentMethod: "mobile_money",
|
|
2190
|
-
psp: "mpesa",
|
|
2191
|
-
pspReference: result2.transactionId,
|
|
2192
|
-
status: "success"
|
|
2193
|
-
}),
|
|
2194
|
-
onError: (err) => handlePSPError({
|
|
2195
|
-
code: err.code,
|
|
2196
|
-
message: err.message,
|
|
2197
|
-
recoverable: true
|
|
2198
|
-
})
|
|
2239
|
+
onSuccess: handlePSPSuccess,
|
|
2240
|
+
onError: handlePSPError
|
|
2199
2241
|
}
|
|
2200
2242
|
);
|
|
2201
2243
|
case "stripe":
|
|
@@ -2206,58 +2248,60 @@ function ReevitCheckout({
|
|
|
2206
2248
|
clientSecret: paymentIntent?.clientSecret || "",
|
|
2207
2249
|
amount: paymentIntent?.amount ?? amount,
|
|
2208
2250
|
currency: paymentIntent?.currency ?? currency,
|
|
2209
|
-
onSuccess:
|
|
2210
|
-
|
|
2211
|
-
reference: paymentIntent?.reference || reference || result2.paymentIntentId,
|
|
2212
|
-
amount: paymentIntent?.amount ?? amount,
|
|
2213
|
-
currency: paymentIntent?.currency ?? currency,
|
|
2214
|
-
paymentMethod: selectedMethod || "card",
|
|
2215
|
-
psp: "stripe",
|
|
2216
|
-
pspReference: result2.paymentIntentId,
|
|
2217
|
-
status: result2.status === "succeeded" ? "success" : "pending"
|
|
2218
|
-
}),
|
|
2219
|
-
onError: (err) => handlePSPError({
|
|
2220
|
-
code: err.code,
|
|
2221
|
-
message: err.message,
|
|
2222
|
-
recoverable: true
|
|
2223
|
-
}),
|
|
2251
|
+
onSuccess: handlePSPSuccess,
|
|
2252
|
+
onError: handlePSPError,
|
|
2224
2253
|
onCancel: handlePSPClose
|
|
2225
2254
|
}
|
|
2226
2255
|
);
|
|
2227
2256
|
default:
|
|
2228
2257
|
return /* @__PURE__ */ jsxs("div", { className: "reevit-error", children: [
|
|
2229
2258
|
/* @__PURE__ */ jsx("div", { className: "reevit-error__icon", children: "\u26A0\uFE0F" }),
|
|
2230
|
-
/* @__PURE__ */ jsx("h3", { children: "
|
|
2259
|
+
/* @__PURE__ */ jsx("h3", { children: "Provider Not Supported" }),
|
|
2231
2260
|
/* @__PURE__ */ jsxs("p", { children: [
|
|
2232
|
-
"
|
|
2261
|
+
"Provider (",
|
|
2233
2262
|
psp,
|
|
2234
|
-
") is not
|
|
2263
|
+
") is not supported."
|
|
2235
2264
|
] }),
|
|
2236
2265
|
/* @__PURE__ */ jsx("button", { className: "reevit-btn reevit-btn--primary", onClick: handleBack, children: "Go Back" })
|
|
2237
2266
|
] });
|
|
2238
2267
|
}
|
|
2239
2268
|
}
|
|
2240
|
-
|
|
2241
|
-
|
|
2242
|
-
|
|
2243
|
-
|
|
2244
|
-
|
|
2245
|
-
|
|
2246
|
-
isLoading,
|
|
2247
|
-
|
|
2269
|
+
const renderMethodContent = (provider, method) => {
|
|
2270
|
+
const pspLower2 = provider.toLowerCase();
|
|
2271
|
+
const needsPhone = pspLower2.includes("mpesa");
|
|
2272
|
+
if (method === "card") {
|
|
2273
|
+
return /* @__PURE__ */ jsxs("div", { className: "reevit-inline-action reevit-animate-fade-in", children: [
|
|
2274
|
+
/* @__PURE__ */ jsx("p", { className: "reevit-inline-action__hint", children: "You'll be redirected to complete your card payment securely." }),
|
|
2275
|
+
/* @__PURE__ */ jsx("button", { className: "reevit-btn reevit-btn--primary", onClick: handleContinue, disabled: isLoading, children: "Pay with Card" })
|
|
2276
|
+
] });
|
|
2277
|
+
}
|
|
2278
|
+
if (method === "mobile_money") {
|
|
2279
|
+
if (needsPhone && !phone) {
|
|
2280
|
+
return /* @__PURE__ */ jsx("div", { className: "reevit-inline-action reevit-animate-fade-in", children: /* @__PURE__ */ jsx(MobileMoneyForm, { onSubmit: handleMomoSubmit, onCancel: handleBack, isLoading, initialPhone: phone, hideCancel: true }) });
|
|
2248
2281
|
}
|
|
2249
|
-
|
|
2250
|
-
|
|
2251
|
-
|
|
2252
|
-
|
|
2282
|
+
return /* @__PURE__ */ jsxs("div", { className: "reevit-inline-action reevit-animate-fade-in", children: [
|
|
2283
|
+
/* @__PURE__ */ jsx("p", { className: "reevit-inline-action__hint", children: pspLower2.includes("hubtel") ? "Opens the Hubtel checkout with Mobile Money selected." : `Continue to pay securely with Mobile Money via ${pspNames[pspLower2] || pspLower2.charAt(0).toUpperCase() + pspLower2.slice(1)}.` }),
|
|
2284
|
+
/* @__PURE__ */ jsx("button", { className: "reevit-btn reevit-btn--primary", onClick: handleContinue, disabled: isLoading, children: pspLower2.includes("hubtel") ? "Continue with Hubtel" : "Pay with Mobile Money" })
|
|
2285
|
+
] });
|
|
2286
|
+
}
|
|
2287
|
+
return null;
|
|
2288
|
+
};
|
|
2289
|
+
if (providerOptions.length > 1) {
|
|
2290
|
+
return /* @__PURE__ */ jsx("div", { className: "reevit-method-step reevit-animate-slide-up", children: /* @__PURE__ */ jsx(
|
|
2253
2291
|
ProviderSelector,
|
|
2254
2292
|
{
|
|
2255
2293
|
providers: providerOptions,
|
|
2256
2294
|
selectedProvider,
|
|
2257
2295
|
onSelect: handleProviderSelect,
|
|
2258
|
-
disabled: isLoading
|
|
2296
|
+
disabled: isLoading,
|
|
2297
|
+
theme: resolvedTheme,
|
|
2298
|
+
selectedMethod,
|
|
2299
|
+
onMethodSelect: handleMethodSelect,
|
|
2300
|
+
renderMethodContent
|
|
2259
2301
|
}
|
|
2260
|
-
)
|
|
2302
|
+
) });
|
|
2303
|
+
}
|
|
2304
|
+
return /* @__PURE__ */ jsxs("div", { className: "reevit-method-step reevit-animate-slide-up", children: [
|
|
2261
2305
|
/* @__PURE__ */ jsx(
|
|
2262
2306
|
PaymentMethodSelector,
|
|
2263
2307
|
{
|
|
@@ -2265,18 +2309,23 @@ function ReevitCheckout({
|
|
|
2265
2309
|
selectedMethod,
|
|
2266
2310
|
onSelect: handleMethodSelect,
|
|
2267
2311
|
disabled: isLoading,
|
|
2268
|
-
provider:
|
|
2312
|
+
provider: psp,
|
|
2313
|
+
layout: "list",
|
|
2314
|
+
showLabel: false
|
|
2269
2315
|
}
|
|
2270
2316
|
),
|
|
2271
|
-
selectedMethod &&
|
|
2272
|
-
"
|
|
2273
|
-
|
|
2274
|
-
|
|
2275
|
-
|
|
2276
|
-
|
|
2277
|
-
|
|
2278
|
-
|
|
2279
|
-
|
|
2317
|
+
selectedMethod && /* @__PURE__ */ jsx("div", { className: "reevit-method-step__actions reevit-animate-slide-up", children: selectedMethod === "mobile_money" && pspLower.includes("mpesa") && !phone ? /* @__PURE__ */ jsx(MobileMoneyForm, { onSubmit: handleMomoSubmit, onCancel: () => selectMethod(null), isLoading, initialPhone: phone }) : /* @__PURE__ */ jsxs("div", { className: "reevit-card-info reevit-animate-fade-in", children: [
|
|
2318
|
+
/* @__PURE__ */ jsx("p", { className: "reevit-info-text", children: selectedMethod === "card" ? "You will be redirected to complete your card payment securely." : pspLower.includes("hubtel") ? "Opens the Hubtel checkout with Mobile Money selected." : `Continue to pay securely via ${pspNames[pspLower] || pspLower.charAt(0).toUpperCase() + pspLower.slice(1)}.` }),
|
|
2319
|
+
/* @__PURE__ */ jsx(
|
|
2320
|
+
"button",
|
|
2321
|
+
{
|
|
2322
|
+
className: "reevit-btn reevit-btn--primary",
|
|
2323
|
+
onClick: handleContinue,
|
|
2324
|
+
disabled: isLoading,
|
|
2325
|
+
children: selectedMethod === "card" ? "Pay with Card" : pspLower.includes("hubtel") ? "Continue with Hubtel" : "Pay with Mobile Money"
|
|
2326
|
+
}
|
|
2327
|
+
)
|
|
2328
|
+
] }) })
|
|
2280
2329
|
] });
|
|
2281
2330
|
};
|
|
2282
2331
|
return /* @__PURE__ */ jsxs(ReevitContext.Provider, { value: { publicKey, amount, currency }, children: [
|
|
@@ -2292,23 +2341,8 @@ function ReevitCheckout({
|
|
|
2292
2341
|
"aria-modal": "true",
|
|
2293
2342
|
children: [
|
|
2294
2343
|
/* @__PURE__ */ jsxs("div", { className: "reevit-modal__header", children: [
|
|
2295
|
-
/* @__PURE__ */ jsx("div", { className: "reevit-modal__branding", children: /* @__PURE__ */ jsx(
|
|
2296
|
-
|
|
2297
|
-
{
|
|
2298
|
-
src: "https://i.imgur.com/bzUR5Lm.png",
|
|
2299
|
-
alt: "Reevit",
|
|
2300
|
-
className: "reevit-modal__logo"
|
|
2301
|
-
}
|
|
2302
|
-
) }),
|
|
2303
|
-
/* @__PURE__ */ jsx(
|
|
2304
|
-
"button",
|
|
2305
|
-
{
|
|
2306
|
-
className: "reevit-modal__close",
|
|
2307
|
-
onClick: handleClose,
|
|
2308
|
-
"aria-label": "Close",
|
|
2309
|
-
children: "\u2715"
|
|
2310
|
-
}
|
|
2311
|
-
)
|
|
2344
|
+
/* @__PURE__ */ jsx("div", { className: "reevit-modal__branding", children: /* @__PURE__ */ jsx("img", { src: resolvedTheme?.logoUrl || "https://i.imgur.com/bzUR5Lm.png", alt: "Checkout", className: "reevit-modal__logo" }) }),
|
|
2345
|
+
/* @__PURE__ */ jsx("button", { className: "reevit-modal__close", onClick: handleClose, "aria-label": "Close", children: "\u2715" })
|
|
2312
2346
|
] }),
|
|
2313
2347
|
/* @__PURE__ */ jsxs("div", { className: "reevit-modal__amount", children: [
|
|
2314
2348
|
/* @__PURE__ */ jsx("span", { className: "reevit-modal__amount-label", children: "Amount" }),
|