Package not found. Please check the package name and try again.
@reevit/react 0.4.2 → 0.4.4
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 +33 -10
- package/dist/index.d.ts +33 -10
- package/dist/index.js +297 -198
- package/dist/index.js.map +1 -1
- package/dist/index.mjs +297 -198
- package/dist/index.mjs.map +1 -1
- package/dist/styles.css +268 -69
- package/package.json +1 -1
package/dist/index.mjs
CHANGED
|
@@ -132,8 +132,8 @@ function createPaymentError(response, errorData) {
|
|
|
132
132
|
}
|
|
133
133
|
var ReevitAPIClient = class {
|
|
134
134
|
constructor(config) {
|
|
135
|
-
this.publicKey = config.publicKey;
|
|
136
|
-
this.baseUrl = config.baseUrl || (isSandboxKey(config.publicKey) ? API_BASE_URL_SANDBOX : API_BASE_URL_PRODUCTION);
|
|
135
|
+
this.publicKey = config.publicKey || "";
|
|
136
|
+
this.baseUrl = config.baseUrl || (config.publicKey && isSandboxKey(config.publicKey) ? API_BASE_URL_SANDBOX : API_BASE_URL_PRODUCTION);
|
|
137
137
|
this.timeout = config.timeout || DEFAULT_TIMEOUT;
|
|
138
138
|
}
|
|
139
139
|
/**
|
|
@@ -144,10 +144,12 @@ var ReevitAPIClient = class {
|
|
|
144
144
|
const timeoutId = setTimeout(() => controller.abort(), this.timeout);
|
|
145
145
|
const headers = {
|
|
146
146
|
"Content-Type": "application/json",
|
|
147
|
-
"X-Reevit-Key": this.publicKey,
|
|
148
147
|
"X-Reevit-Client": "@reevit/react",
|
|
149
148
|
"X-Reevit-Client-Version": "0.3.2"
|
|
150
149
|
};
|
|
150
|
+
if (this.publicKey) {
|
|
151
|
+
headers["X-Reevit-Key"] = this.publicKey;
|
|
152
|
+
}
|
|
151
153
|
if (method === "POST" || method === "PATCH" || method === "PUT") {
|
|
152
154
|
headers["Idempotency-Key"] = `${Date.now()}-${Math.random().toString(36).substring(2, 15)}`;
|
|
153
155
|
}
|
|
@@ -254,8 +256,9 @@ var ReevitAPIClient = class {
|
|
|
254
256
|
* @param paymentId - The payment intent ID for Hubtel checkout
|
|
255
257
|
* @returns Hubtel session with token, merchant account, and expiry information
|
|
256
258
|
*/
|
|
257
|
-
async createHubtelSession(paymentId) {
|
|
258
|
-
|
|
259
|
+
async createHubtelSession(paymentId, clientSecret) {
|
|
260
|
+
const query = clientSecret ? `?client_secret=${encodeURIComponent(clientSecret)}` : "";
|
|
261
|
+
return this.request("POST", `/v1/payments/hubtel/sessions/${paymentId}${query}`);
|
|
259
262
|
}
|
|
260
263
|
/**
|
|
261
264
|
* Maps SDK payment method to backend format
|
|
@@ -285,6 +288,18 @@ var initialState = {
|
|
|
285
288
|
error: null,
|
|
286
289
|
result: null
|
|
287
290
|
};
|
|
291
|
+
var DEFAULT_PUBLIC_API_BASE_URL = "https://api.reevit.io";
|
|
292
|
+
function buildPaymentLinkError(response, data) {
|
|
293
|
+
return {
|
|
294
|
+
code: data?.code || "payment_link_error",
|
|
295
|
+
message: data?.message || "Payment link request failed",
|
|
296
|
+
recoverable: true,
|
|
297
|
+
details: {
|
|
298
|
+
httpStatus: response.status,
|
|
299
|
+
...data?.details || {}
|
|
300
|
+
}
|
|
301
|
+
};
|
|
302
|
+
}
|
|
288
303
|
function reevitReducer(state, action) {
|
|
289
304
|
switch (action.type) {
|
|
290
305
|
case "INIT_START":
|
|
@@ -408,15 +423,48 @@ function useReevit(options) {
|
|
|
408
423
|
const reference = config.reference || generateReference();
|
|
409
424
|
const country = detectCountryFromCurrency(config.currency);
|
|
410
425
|
const paymentMethod = method || config.paymentMethods?.[0] || "card";
|
|
411
|
-
|
|
412
|
-
|
|
413
|
-
|
|
414
|
-
|
|
415
|
-
|
|
416
|
-
|
|
417
|
-
|
|
426
|
+
let data;
|
|
427
|
+
let error;
|
|
428
|
+
if (config.paymentLinkCode) {
|
|
429
|
+
const response = await fetch(
|
|
430
|
+
`${apiBaseUrl || DEFAULT_PUBLIC_API_BASE_URL}/v1/pay/${config.paymentLinkCode}/pay`,
|
|
431
|
+
{
|
|
432
|
+
method: "POST",
|
|
433
|
+
headers: {
|
|
434
|
+
"Content-Type": "application/json",
|
|
435
|
+
"Idempotency-Key": `${Date.now()}-${Math.random().toString(36).substring(2, 15)}`
|
|
436
|
+
},
|
|
437
|
+
body: JSON.stringify({
|
|
438
|
+
amount: config.amount,
|
|
439
|
+
email: config.email || "",
|
|
440
|
+
name: config.customerName || "",
|
|
441
|
+
phone: config.phone || "",
|
|
442
|
+
method: paymentMethod,
|
|
443
|
+
country,
|
|
444
|
+
provider: options2?.preferredProvider || options2?.allowedProviders?.[0],
|
|
445
|
+
custom_fields: config.customFields
|
|
446
|
+
})
|
|
447
|
+
}
|
|
448
|
+
);
|
|
449
|
+
const responseData = await response.json().catch(() => ({}));
|
|
450
|
+
if (!response.ok) {
|
|
451
|
+
error = buildPaymentLinkError(response, responseData);
|
|
452
|
+
} else {
|
|
453
|
+
data = responseData;
|
|
418
454
|
}
|
|
419
|
-
|
|
455
|
+
} else {
|
|
456
|
+
const result = await apiClient.createPaymentIntent(
|
|
457
|
+
{ ...config, reference },
|
|
458
|
+
paymentMethod,
|
|
459
|
+
country,
|
|
460
|
+
{
|
|
461
|
+
preferredProviders: options2?.preferredProvider ? [options2.preferredProvider] : void 0,
|
|
462
|
+
allowedProviders: options2?.allowedProviders
|
|
463
|
+
}
|
|
464
|
+
);
|
|
465
|
+
data = result.data;
|
|
466
|
+
error = result.error;
|
|
467
|
+
}
|
|
420
468
|
if (error) {
|
|
421
469
|
dispatch({ type: "INIT_ERROR", payload: error });
|
|
422
470
|
onError?.(error);
|
|
@@ -581,14 +629,6 @@ function detectCountryFromCurrency(currency) {
|
|
|
581
629
|
};
|
|
582
630
|
return currencyToCountry[currency.toUpperCase()] || "GH";
|
|
583
631
|
}
|
|
584
|
-
var pspNames = {
|
|
585
|
-
hubtel: "Hubtel",
|
|
586
|
-
paystack: "Paystack",
|
|
587
|
-
flutterwave: "Flutterwave",
|
|
588
|
-
monnify: "Monnify",
|
|
589
|
-
mpesa: "M-Pesa",
|
|
590
|
-
stripe: "Stripe"
|
|
591
|
-
};
|
|
592
632
|
var methodConfig = {
|
|
593
633
|
card: {
|
|
594
634
|
label: "Card",
|
|
@@ -611,28 +651,31 @@ function PaymentMethodSelector({
|
|
|
611
651
|
selectedMethod,
|
|
612
652
|
onSelect,
|
|
613
653
|
disabled = false,
|
|
614
|
-
provider
|
|
654
|
+
provider,
|
|
655
|
+
layout = "list",
|
|
656
|
+
showLabel = true
|
|
615
657
|
}) {
|
|
616
658
|
const getMethodLabel = (method, psp) => {
|
|
617
659
|
const config = methodConfig[method];
|
|
618
|
-
if (psp?.toLowerCase().includes("hubtel") && method === "mobile_money") {
|
|
619
|
-
return `Pay with ${pspNames[psp.toLowerCase()] || "Hubtel"}`;
|
|
620
|
-
}
|
|
621
660
|
return config.label;
|
|
622
661
|
};
|
|
623
662
|
const getMethodDescription = (method, psp) => {
|
|
624
663
|
const config = methodConfig[method];
|
|
625
664
|
if (psp?.toLowerCase().includes("hubtel")) {
|
|
626
|
-
return
|
|
665
|
+
return config.description;
|
|
627
666
|
}
|
|
628
667
|
return config.description;
|
|
629
668
|
};
|
|
630
|
-
|
|
631
|
-
|
|
632
|
-
/* @__PURE__ */ jsx("div", { className: "reevit-method-
|
|
669
|
+
const isGrid = layout === "grid";
|
|
670
|
+
return /* @__PURE__ */ jsxs("div", { className: cn("reevit-method-selector", isGrid && "reevit-method-selector--grid"), children: [
|
|
671
|
+
showLabel && /* @__PURE__ */ jsx("div", { className: "reevit-method-selector__label", children: "Select payment method" }),
|
|
672
|
+
/* @__PURE__ */ jsx("div", { className: cn(
|
|
673
|
+
"reevit-method-selector__options",
|
|
674
|
+
isGrid ? "reevit-method-selector__options--grid" : "reevit-method-selector__options--list"
|
|
675
|
+
), children: methods.map((method, index) => {
|
|
633
676
|
const config = methodConfig[method];
|
|
634
677
|
const isSelected = selectedMethod === method;
|
|
635
|
-
const methodLabel = getMethodLabel(method
|
|
678
|
+
const methodLabel = getMethodLabel(method);
|
|
636
679
|
const methodDescription = getMethodDescription(method, provider);
|
|
637
680
|
return /* @__PURE__ */ jsxs(
|
|
638
681
|
"button",
|
|
@@ -640,28 +683,24 @@ function PaymentMethodSelector({
|
|
|
640
683
|
type: "button",
|
|
641
684
|
className: cn(
|
|
642
685
|
"reevit-method-option",
|
|
686
|
+
isGrid ? "reevit-method-option--grid" : "reevit-method-option--list",
|
|
643
687
|
isSelected && "reevit-method-option--selected",
|
|
644
688
|
disabled && "reevit-method-option--disabled"
|
|
645
689
|
),
|
|
690
|
+
style: {
|
|
691
|
+
animationDelay: `${index * 0.05}s`
|
|
692
|
+
},
|
|
646
693
|
onClick: () => onSelect(method),
|
|
647
694
|
disabled,
|
|
648
695
|
"aria-pressed": isSelected,
|
|
649
696
|
children: [
|
|
650
|
-
/* @__PURE__ */ jsx("span", { className: "reevit-method-option__icon", children: config.icon }),
|
|
697
|
+
/* @__PURE__ */ jsx("span", { className: "reevit-method-option__icon-wrapper", children: /* @__PURE__ */ jsx("span", { className: "reevit-method-option__icon", children: config.icon }) }),
|
|
651
698
|
/* @__PURE__ */ jsxs("div", { className: "reevit-method-option__content", children: [
|
|
652
699
|
/* @__PURE__ */ jsx("span", { className: "reevit-method-option__label", children: methodLabel }),
|
|
653
|
-
/* @__PURE__ */ jsx("span", { className: "reevit-method-option__description", children: methodDescription })
|
|
700
|
+
!isGrid && /* @__PURE__ */ jsx("span", { className: "reevit-method-option__description", children: methodDescription })
|
|
654
701
|
] }),
|
|
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
|
-
) }) })
|
|
702
|
+
!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" }) }) }),
|
|
703
|
+
!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
704
|
]
|
|
666
705
|
},
|
|
667
706
|
method
|
|
@@ -678,7 +717,8 @@ function MobileMoneyForm({
|
|
|
678
717
|
onSubmit,
|
|
679
718
|
onCancel,
|
|
680
719
|
isLoading = false,
|
|
681
|
-
initialPhone = ""
|
|
720
|
+
initialPhone = "",
|
|
721
|
+
hideCancel = false
|
|
682
722
|
}) {
|
|
683
723
|
const [phone, setPhone] = useState(initialPhone);
|
|
684
724
|
const [network, setNetwork] = useState(null);
|
|
@@ -763,7 +803,7 @@ function MobileMoneyForm({
|
|
|
763
803
|
)) })
|
|
764
804
|
] }),
|
|
765
805
|
/* @__PURE__ */ jsxs("div", { className: "reevit-momo-form__actions", children: [
|
|
766
|
-
/* @__PURE__ */ jsx(
|
|
806
|
+
!hideCancel && /* @__PURE__ */ jsx(
|
|
767
807
|
"button",
|
|
768
808
|
{
|
|
769
809
|
type: "button",
|
|
@@ -805,11 +845,11 @@ var paystack_default = "
|
|
|
805
845
|
// src/assets/providers/stripe.png
|
|
806
846
|
var stripe_default = "";
|
|
807
847
|
var providerMeta = {
|
|
808
|
-
paystack: { logo: paystack_default, hint: "Card &
|
|
848
|
+
paystack: { logo: paystack_default, hint: "Card & Mobile Money" },
|
|
809
849
|
stripe: { logo: stripe_default, hint: "Card payments" },
|
|
810
850
|
flutterwave: { logo: flutterwave_default, hint: "Global methods" },
|
|
811
|
-
hubtel: { logo: hubtel_default, hint: "Mobile
|
|
812
|
-
monnify: { logo: monnify_default, hint: "Bank &
|
|
851
|
+
hubtel: { logo: hubtel_default, hint: "Mobile Money & Card" },
|
|
852
|
+
monnify: { logo: monnify_default, hint: "Bank & Card" },
|
|
813
853
|
mpesa: { logo: mpesa_default, hint: "M-Pesa" }
|
|
814
854
|
};
|
|
815
855
|
var methodLabels = {
|
|
@@ -819,54 +859,100 @@ var methodLabels = {
|
|
|
819
859
|
};
|
|
820
860
|
function formatMethods(methods) {
|
|
821
861
|
if (!methods.length) return "Payment methods";
|
|
822
|
-
return methods.map((method) => methodLabels[method]).join("
|
|
862
|
+
return methods.map((method) => methodLabels[method]).join(", ");
|
|
863
|
+
}
|
|
864
|
+
function sanitizeMethods(providerId, methods) {
|
|
865
|
+
if (providerId.toLowerCase().includes("hubtel")) {
|
|
866
|
+
return methods.filter((method) => method === "card" || method === "mobile_money");
|
|
867
|
+
}
|
|
868
|
+
return methods;
|
|
823
869
|
}
|
|
824
870
|
function ProviderSelector({
|
|
825
871
|
providers,
|
|
826
872
|
selectedProvider,
|
|
827
873
|
onSelect,
|
|
828
|
-
disabled = false
|
|
874
|
+
disabled = false,
|
|
875
|
+
theme,
|
|
876
|
+
selectedMethod,
|
|
877
|
+
onMethodSelect,
|
|
878
|
+
renderMethodContent
|
|
829
879
|
}) {
|
|
880
|
+
const useBorder = theme?.pspSelectorUseBorder ?? false;
|
|
881
|
+
const bgColor = theme?.pspSelectorBgColor || "#0a0a0a";
|
|
882
|
+
const textColor = theme?.pspSelectorTextColor || "#ffffff";
|
|
883
|
+
const borderColor = theme?.pspSelectorBorderColor || "#374151";
|
|
884
|
+
const getOptionStyle = (isSelected) => {
|
|
885
|
+
if (useBorder) {
|
|
886
|
+
return {
|
|
887
|
+
backgroundColor: "transparent",
|
|
888
|
+
border: `2px solid ${isSelected ? borderColor : "#374151"}`,
|
|
889
|
+
color: isSelected ? textColor : "var(--reevit-text)"
|
|
890
|
+
};
|
|
891
|
+
}
|
|
892
|
+
return {
|
|
893
|
+
backgroundColor: isSelected ? bgColor : "transparent",
|
|
894
|
+
border: `2px solid ${isSelected ? bgColor : "#374151"}`,
|
|
895
|
+
color: isSelected ? textColor : "var(--reevit-text)"
|
|
896
|
+
};
|
|
897
|
+
};
|
|
830
898
|
return /* @__PURE__ */ jsxs("div", { className: "reevit-psp-selector", children: [
|
|
831
|
-
/* @__PURE__ */ jsx("div", { className: "reevit-psp-selector__label", children: "
|
|
899
|
+
/* @__PURE__ */ jsx("div", { className: "reevit-psp-selector__label", children: "Select payment provider" }),
|
|
832
900
|
/* @__PURE__ */ jsx("div", { className: "reevit-psp-selector__options", children: providers.map((provider) => {
|
|
833
901
|
const meta = providerMeta[provider.provider] || {
|
|
834
902
|
logo: void 0,
|
|
835
903
|
hint: "Payment methods"
|
|
836
904
|
};
|
|
905
|
+
const providerMethods = sanitizeMethods(provider.provider, provider.methods);
|
|
837
906
|
const isSelected = selectedProvider === provider.provider;
|
|
838
907
|
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
|
-
|
|
908
|
+
const optionStyle = getOptionStyle(isSelected);
|
|
909
|
+
return /* @__PURE__ */ jsxs("div", { className: "reevit-psp-accordion", children: [
|
|
910
|
+
/* @__PURE__ */ jsxs(
|
|
911
|
+
"button",
|
|
912
|
+
{
|
|
913
|
+
type: "button",
|
|
914
|
+
className: cn(
|
|
915
|
+
"reevit-psp-option",
|
|
916
|
+
isSelected && "reevit-psp-option--selected",
|
|
917
|
+
disabled && "reevit-psp-option--disabled"
|
|
918
|
+
),
|
|
919
|
+
style: optionStyle,
|
|
920
|
+
onClick: () => onSelect(provider.provider),
|
|
921
|
+
disabled,
|
|
922
|
+
"aria-expanded": isSelected,
|
|
923
|
+
children: [
|
|
924
|
+
/* @__PURE__ */ jsx("span", { className: "reevit-psp-option__logo", "aria-hidden": "true", children: meta.logo ? /* @__PURE__ */ jsx(
|
|
925
|
+
"img",
|
|
926
|
+
{
|
|
927
|
+
src: meta.logo,
|
|
928
|
+
alt: "",
|
|
929
|
+
className: "reevit-psp-option__logo-img",
|
|
930
|
+
loading: "lazy"
|
|
931
|
+
}
|
|
932
|
+
) : /* @__PURE__ */ jsx("span", { className: "reevit-psp-option__logo-fallback", children: fallbackInitial }) }),
|
|
933
|
+
/* @__PURE__ */ jsxs("div", { className: "reevit-psp-option__content", children: [
|
|
934
|
+
/* @__PURE__ */ jsx("span", { className: "reevit-psp-option__name", children: provider.name }),
|
|
935
|
+
/* @__PURE__ */ jsx("span", { className: "reevit-psp-option__methods", children: formatMethods(providerMethods) || meta.hint })
|
|
936
|
+
] })
|
|
937
|
+
]
|
|
938
|
+
}
|
|
939
|
+
),
|
|
940
|
+
isSelected && onMethodSelect && /* @__PURE__ */ jsxs("div", { className: "reevit-psp-accordion__content", children: [
|
|
941
|
+
/* @__PURE__ */ jsx("div", { className: "reevit-psp-methods", children: /* @__PURE__ */ jsx(
|
|
942
|
+
PaymentMethodSelector,
|
|
943
|
+
{
|
|
944
|
+
methods: providerMethods,
|
|
945
|
+
selectedMethod: selectedMethod || null,
|
|
946
|
+
onSelect: onMethodSelect,
|
|
947
|
+
disabled,
|
|
948
|
+
provider: provider.provider,
|
|
949
|
+
layout: "list",
|
|
950
|
+
showLabel: false
|
|
951
|
+
}
|
|
952
|
+
) }),
|
|
953
|
+
selectedMethod && renderMethodContent && /* @__PURE__ */ jsx("div", { className: "reevit-psp-accordion__method-content", children: renderMethodContent(provider.provider, selectedMethod) })
|
|
954
|
+
] })
|
|
955
|
+
] }, provider.provider);
|
|
870
956
|
}) })
|
|
871
957
|
] });
|
|
872
958
|
}
|
|
@@ -998,8 +1084,11 @@ function HubtelBridge({
|
|
|
998
1084
|
phone,
|
|
999
1085
|
description = "Payment",
|
|
1000
1086
|
callbackUrl,
|
|
1087
|
+
apiBaseUrl,
|
|
1088
|
+
clientSecret,
|
|
1001
1089
|
hubtelSessionToken,
|
|
1002
1090
|
basicAuth,
|
|
1091
|
+
preferredMethod,
|
|
1003
1092
|
onSuccess,
|
|
1004
1093
|
onError,
|
|
1005
1094
|
onClose,
|
|
@@ -1009,13 +1098,17 @@ function HubtelBridge({
|
|
|
1009
1098
|
const checkoutRef = useRef(null);
|
|
1010
1099
|
const [authValue, setAuthValue] = useState("");
|
|
1011
1100
|
const [isLoading, setIsLoading] = useState(false);
|
|
1101
|
+
const [resolvedMerchantAccount, setResolvedMerchantAccount] = useState(merchantAccount);
|
|
1102
|
+
useEffect(() => {
|
|
1103
|
+
setResolvedMerchantAccount(merchantAccount);
|
|
1104
|
+
}, [merchantAccount]);
|
|
1012
1105
|
useEffect(() => {
|
|
1013
1106
|
const fetchAuth = async () => {
|
|
1014
1107
|
if (hubtelSessionToken) {
|
|
1015
1108
|
setIsLoading(true);
|
|
1016
1109
|
try {
|
|
1017
|
-
const client = createReevitClient({ publicKey });
|
|
1018
|
-
const { data, error } = await client.createHubtelSession(paymentId);
|
|
1110
|
+
const client = createReevitClient({ publicKey, baseUrl: apiBaseUrl });
|
|
1111
|
+
const { data, error } = await client.createHubtelSession(paymentId, clientSecret);
|
|
1019
1112
|
if (error) {
|
|
1020
1113
|
onError({
|
|
1021
1114
|
code: "SESSION_ERROR",
|
|
@@ -1026,6 +1119,9 @@ function HubtelBridge({
|
|
|
1026
1119
|
}
|
|
1027
1120
|
if (data) {
|
|
1028
1121
|
setAuthValue(data.token);
|
|
1122
|
+
if (data.merchantAccount) {
|
|
1123
|
+
setResolvedMerchantAccount(data.merchantAccount);
|
|
1124
|
+
}
|
|
1029
1125
|
}
|
|
1030
1126
|
} catch (err) {
|
|
1031
1127
|
onError({
|
|
@@ -1042,7 +1138,7 @@ function HubtelBridge({
|
|
|
1042
1138
|
}
|
|
1043
1139
|
};
|
|
1044
1140
|
fetchAuth();
|
|
1045
|
-
}, [paymentId, publicKey, hubtelSessionToken, basicAuth, onError]);
|
|
1141
|
+
}, [paymentId, publicKey, apiBaseUrl, clientSecret, hubtelSessionToken, basicAuth, onError]);
|
|
1046
1142
|
const startPayment = useCallback(async () => {
|
|
1047
1143
|
if (isLoading || !authValue) {
|
|
1048
1144
|
return;
|
|
@@ -1050,20 +1146,23 @@ function HubtelBridge({
|
|
|
1050
1146
|
try {
|
|
1051
1147
|
const checkout = new CheckoutSdk();
|
|
1052
1148
|
checkoutRef.current = checkout;
|
|
1149
|
+
const methodPreference = preferredMethod === "mobile_money" ? "momo" : preferredMethod === "card" ? "card" : void 0;
|
|
1053
1150
|
const purchaseInfo = {
|
|
1054
1151
|
amount: amount / 100,
|
|
1055
1152
|
// Convert from minor to major units
|
|
1056
1153
|
purchaseDescription: description,
|
|
1057
1154
|
customerPhoneNumber: phone || "",
|
|
1058
|
-
clientReference: reference || `hubtel_${Date.now()}
|
|
1155
|
+
clientReference: reference || `hubtel_${Date.now()}`,
|
|
1156
|
+
...methodPreference ? { paymentMethod: methodPreference } : {}
|
|
1059
1157
|
};
|
|
1060
1158
|
const config = {
|
|
1061
1159
|
branding: "enabled",
|
|
1062
1160
|
callbackUrl: callbackUrl || window.location.href,
|
|
1063
|
-
merchantAccount: typeof
|
|
1161
|
+
merchantAccount: typeof resolvedMerchantAccount === "string" ? parseInt(resolvedMerchantAccount, 10) : resolvedMerchantAccount,
|
|
1064
1162
|
// Use session token or basicAuth for authentication
|
|
1065
1163
|
// Session tokens are base64-encoded credentials fetched securely from the server
|
|
1066
|
-
basicAuth: authValue || ""
|
|
1164
|
+
basicAuth: authValue || "",
|
|
1165
|
+
...methodPreference ? { paymentMethod: methodPreference } : {}
|
|
1067
1166
|
};
|
|
1068
1167
|
checkout.openModal({
|
|
1069
1168
|
purchaseInfo,
|
|
@@ -1107,7 +1206,7 @@ function HubtelBridge({
|
|
|
1107
1206
|
};
|
|
1108
1207
|
onError(error);
|
|
1109
1208
|
}
|
|
1110
|
-
}, [merchantAccount, amount, reference, phone, description, callbackUrl, authValue, isLoading, onSuccess, onError, onClose]);
|
|
1209
|
+
}, [merchantAccount, amount, reference, phone, description, callbackUrl, authValue, isLoading, preferredMethod, onSuccess, onError, onClose]);
|
|
1111
1210
|
useEffect(() => {
|
|
1112
1211
|
if (autoStart && !initialized.current && !isLoading && authValue) {
|
|
1113
1212
|
initialized.current = true;
|
|
@@ -1121,17 +1220,20 @@ function HubtelBridge({
|
|
|
1121
1220
|
}
|
|
1122
1221
|
function openHubtelPopup(config) {
|
|
1123
1222
|
const checkout = new CheckoutSdk();
|
|
1223
|
+
const methodPreference = config.preferredMethod === "mobile_money" ? "momo" : config.preferredMethod === "card" ? "card" : void 0;
|
|
1124
1224
|
const purchaseInfo = {
|
|
1125
1225
|
amount: config.amount,
|
|
1126
1226
|
purchaseDescription: config.description,
|
|
1127
1227
|
customerPhoneNumber: config.customerPhoneNumber || "",
|
|
1128
|
-
clientReference: config.clientReference || `hubtel_${Date.now()}
|
|
1228
|
+
clientReference: config.clientReference || `hubtel_${Date.now()}`,
|
|
1229
|
+
...methodPreference ? { paymentMethod: methodPreference } : {}
|
|
1129
1230
|
};
|
|
1130
1231
|
const checkoutConfig = {
|
|
1131
1232
|
branding: "enabled",
|
|
1132
1233
|
callbackUrl: config.callbackUrl || window.location.href,
|
|
1133
1234
|
merchantAccount: typeof config.merchantAccount === "string" ? parseInt(config.merchantAccount, 10) : config.merchantAccount,
|
|
1134
|
-
basicAuth: config.basicAuth || ""
|
|
1235
|
+
basicAuth: config.basicAuth || "",
|
|
1236
|
+
...methodPreference ? { paymentMethod: methodPreference } : {}
|
|
1135
1237
|
};
|
|
1136
1238
|
checkout.openModal({
|
|
1137
1239
|
purchaseInfo,
|
|
@@ -1824,6 +1926,14 @@ function useReevitContext() {
|
|
|
1824
1926
|
}
|
|
1825
1927
|
return context;
|
|
1826
1928
|
}
|
|
1929
|
+
var pspNames = {
|
|
1930
|
+
hubtel: "Hubtel",
|
|
1931
|
+
paystack: "Paystack",
|
|
1932
|
+
flutterwave: "Flutterwave",
|
|
1933
|
+
monnify: "Monnify",
|
|
1934
|
+
mpesa: "M-Pesa",
|
|
1935
|
+
stripe: "Stripe"
|
|
1936
|
+
};
|
|
1827
1937
|
function ReevitCheckout({
|
|
1828
1938
|
// Config
|
|
1829
1939
|
publicKey,
|
|
@@ -1831,8 +1941,11 @@ function ReevitCheckout({
|
|
|
1831
1941
|
currency,
|
|
1832
1942
|
email = "",
|
|
1833
1943
|
phone = "",
|
|
1944
|
+
customerName,
|
|
1834
1945
|
reference,
|
|
1835
1946
|
metadata,
|
|
1947
|
+
customFields,
|
|
1948
|
+
paymentLinkCode,
|
|
1836
1949
|
paymentMethods = ["card", "mobile_money"],
|
|
1837
1950
|
initialPaymentIntent,
|
|
1838
1951
|
// Callbacks
|
|
@@ -1883,8 +1996,11 @@ function ReevitCheckout({
|
|
|
1883
1996
|
currency,
|
|
1884
1997
|
email,
|
|
1885
1998
|
phone,
|
|
1999
|
+
customerName,
|
|
1886
2000
|
reference,
|
|
1887
2001
|
metadata,
|
|
2002
|
+
customFields,
|
|
2003
|
+
paymentLinkCode,
|
|
1888
2004
|
paymentMethods,
|
|
1889
2005
|
initialPaymentIntent
|
|
1890
2006
|
},
|
|
@@ -1912,7 +2028,8 @@ function ReevitCheckout({
|
|
|
1912
2028
|
const options = available.length > 0 ? available : fallbackProvider;
|
|
1913
2029
|
return options.map((provider) => {
|
|
1914
2030
|
const methods = provider.methods && provider.methods.length > 0 ? provider.methods : paymentMethods;
|
|
1915
|
-
const
|
|
2031
|
+
const sanitizedMethods = provider.provider.toLowerCase().includes("hubtel") ? methods.filter((method) => method === "card" || method === "mobile_money") : methods;
|
|
2032
|
+
const filteredMethods = sanitizedMethods.filter((method) => paymentMethods.includes(method));
|
|
1916
2033
|
return {
|
|
1917
2034
|
...provider,
|
|
1918
2035
|
methods: filteredMethods
|
|
@@ -1933,10 +2050,10 @@ function ReevitCheckout({
|
|
|
1933
2050
|
if (selectedProvider && providerOptions.some((provider) => provider.provider === selectedProvider)) {
|
|
1934
2051
|
return;
|
|
1935
2052
|
}
|
|
1936
|
-
|
|
1937
|
-
|
|
1938
|
-
|
|
1939
|
-
}, [
|
|
2053
|
+
if (providerOptions.length === 1) {
|
|
2054
|
+
setSelectedProvider(providerOptions[0].provider);
|
|
2055
|
+
}
|
|
2056
|
+
}, [providerOptions, selectedProvider]);
|
|
1940
2057
|
useEffect(() => {
|
|
1941
2058
|
if (!activeProvider || !selectedMethod) {
|
|
1942
2059
|
return;
|
|
@@ -1947,13 +2064,17 @@ function ReevitCheckout({
|
|
|
1947
2064
|
}, [activeProvider, selectedMethod, selectMethod]);
|
|
1948
2065
|
useEffect(() => {
|
|
1949
2066
|
if (isOpen && selectedMethod && paymentIntent && !showPSPBridge) {
|
|
2067
|
+
const psp = (selectedProvider || paymentIntent.recommendedPsp || "paystack").toLowerCase();
|
|
2068
|
+
const needsPhone = psp.includes("mpesa");
|
|
1950
2069
|
if (selectedMethod === "card") {
|
|
1951
2070
|
setShowPSPBridge(true);
|
|
1952
|
-
} else if (selectedMethod === "mobile_money"
|
|
1953
|
-
|
|
2071
|
+
} else if (selectedMethod === "mobile_money") {
|
|
2072
|
+
if (!needsPhone || (momoData?.phone || phone)) {
|
|
2073
|
+
setShowPSPBridge(true);
|
|
2074
|
+
}
|
|
1954
2075
|
}
|
|
1955
2076
|
}
|
|
1956
|
-
}, [isOpen, selectedMethod, showPSPBridge, paymentIntent, momoData, phone]);
|
|
2077
|
+
}, [isOpen, selectedMethod, showPSPBridge, paymentIntent, momoData, phone, selectedProvider]);
|
|
1957
2078
|
const handleOpen = useCallback(() => {
|
|
1958
2079
|
if (controlledIsOpen !== void 0) return;
|
|
1959
2080
|
setIsOpen(true);
|
|
@@ -1977,6 +2098,10 @@ function ReevitCheckout({
|
|
|
1977
2098
|
const handleProviderSelect = useCallback(
|
|
1978
2099
|
(provider) => {
|
|
1979
2100
|
if (provider === selectedProvider) {
|
|
2101
|
+
setSelectedProvider(null);
|
|
2102
|
+
reset();
|
|
2103
|
+
setShowPSPBridge(false);
|
|
2104
|
+
setMomoData(null);
|
|
1980
2105
|
return;
|
|
1981
2106
|
}
|
|
1982
2107
|
const providerMethods = providerOptions.find((option) => option.provider === provider)?.methods || paymentMethods;
|
|
@@ -1993,8 +2118,14 @@ function ReevitCheckout({
|
|
|
1993
2118
|
if (!selectedMethod) return;
|
|
1994
2119
|
if (selectedMethod === "card") {
|
|
1995
2120
|
setShowPSPBridge(true);
|
|
2121
|
+
} else if (selectedMethod === "mobile_money") {
|
|
2122
|
+
const psp = (selectedProvider || paymentIntent?.recommendedPsp || "paystack").toLowerCase();
|
|
2123
|
+
const needsPhone = psp.includes("mpesa");
|
|
2124
|
+
if (!needsPhone || (momoData?.phone || phone)) {
|
|
2125
|
+
setShowPSPBridge(true);
|
|
2126
|
+
}
|
|
1996
2127
|
}
|
|
1997
|
-
}, [selectedMethod]);
|
|
2128
|
+
}, [selectedMethod, selectedProvider, paymentIntent, momoData, phone]);
|
|
1998
2129
|
const handleMomoSubmit = useCallback(
|
|
1999
2130
|
(data) => {
|
|
2000
2131
|
setMomoData(data);
|
|
@@ -2051,13 +2182,13 @@ function ReevitCheckout({
|
|
|
2051
2182
|
] }) : null;
|
|
2052
2183
|
const renderContent = () => {
|
|
2053
2184
|
if (status === "loading" || status === "processing") {
|
|
2054
|
-
return /* @__PURE__ */ jsxs("div", { className: "reevit-loading", children: [
|
|
2185
|
+
return /* @__PURE__ */ jsxs("div", { className: "reevit-loading reevit-animate-fade-in", children: [
|
|
2055
2186
|
/* @__PURE__ */ jsx("div", { className: "reevit-spinner" }),
|
|
2056
2187
|
/* @__PURE__ */ jsx("p", { children: status === "loading" ? "Preparing checkout..." : "Processing payment..." })
|
|
2057
2188
|
] });
|
|
2058
2189
|
}
|
|
2059
2190
|
if (status === "success" && result) {
|
|
2060
|
-
return /* @__PURE__ */ jsxs("div", { className: "reevit-success", children: [
|
|
2191
|
+
return /* @__PURE__ */ jsxs("div", { className: "reevit-success reevit-animate-scale-in", children: [
|
|
2061
2192
|
/* @__PURE__ */ jsx("div", { className: "reevit-success__icon", children: "\u2713" }),
|
|
2062
2193
|
/* @__PURE__ */ jsx("h3", { children: "Payment Successful" }),
|
|
2063
2194
|
/* @__PURE__ */ jsxs("p", { children: [
|
|
@@ -2067,7 +2198,7 @@ function ReevitCheckout({
|
|
|
2067
2198
|
] });
|
|
2068
2199
|
}
|
|
2069
2200
|
if (status === "failed" && error && !error.recoverable) {
|
|
2070
|
-
return /* @__PURE__ */ jsxs("div", { className: "reevit-error", children: [
|
|
2201
|
+
return /* @__PURE__ */ jsxs("div", { className: "reevit-error reevit-animate-fade-in", children: [
|
|
2071
2202
|
/* @__PURE__ */ jsx("div", { className: "reevit-error__icon", children: "\u2715" }),
|
|
2072
2203
|
/* @__PURE__ */ jsx("h3", { children: "Payment Failed" }),
|
|
2073
2204
|
/* @__PURE__ */ jsx("p", { children: error.message }),
|
|
@@ -2075,17 +2206,16 @@ function ReevitCheckout({
|
|
|
2075
2206
|
] });
|
|
2076
2207
|
}
|
|
2077
2208
|
const psp = selectedProvider || paymentIntent?.recommendedPsp || "paystack";
|
|
2209
|
+
const pspLower = psp.toLowerCase();
|
|
2078
2210
|
if (showPSPBridge) {
|
|
2079
|
-
const pspKey = paymentIntent?.pspPublicKey || publicKey;
|
|
2211
|
+
const pspKey = paymentIntent?.pspPublicKey || publicKey || "";
|
|
2080
2212
|
const bridgeMetadata = {
|
|
2081
2213
|
...metadata,
|
|
2082
|
-
// Override with correct payment intent ID for webhook routing
|
|
2083
|
-
// This ensures webhook includes the correct ID to find the payment
|
|
2084
2214
|
payment_id: paymentIntent?.id,
|
|
2085
2215
|
connection_id: paymentIntent?.connectionId ?? metadata?.connection_id,
|
|
2086
2216
|
customer_phone: momoData?.phone || phone
|
|
2087
2217
|
};
|
|
2088
|
-
switch (
|
|
2218
|
+
switch (pspLower) {
|
|
2089
2219
|
case "paystack":
|
|
2090
2220
|
return /* @__PURE__ */ jsx(
|
|
2091
2221
|
PaystackBridge,
|
|
@@ -2118,6 +2248,9 @@ function ReevitCheckout({
|
|
|
2118
2248
|
phone: momoData?.phone || phone,
|
|
2119
2249
|
description: `Payment ${paymentIntent?.reference || reference || ""}`,
|
|
2120
2250
|
hubtelSessionToken: paymentIntent?.id ? paymentIntent.id : void 0,
|
|
2251
|
+
clientSecret: paymentIntent?.clientSecret,
|
|
2252
|
+
apiBaseUrl,
|
|
2253
|
+
preferredMethod: selectedMethod || void 0,
|
|
2121
2254
|
onSuccess: handlePSPSuccess,
|
|
2122
2255
|
onError: (err) => handlePSPError(err),
|
|
2123
2256
|
onClose: handlePSPClose
|
|
@@ -2135,7 +2268,7 @@ function ReevitCheckout({
|
|
|
2135
2268
|
phone: momoData?.phone || phone,
|
|
2136
2269
|
metadata: bridgeMetadata,
|
|
2137
2270
|
onSuccess: handlePSPSuccess,
|
|
2138
|
-
onError:
|
|
2271
|
+
onError: handlePSPError,
|
|
2139
2272
|
onClose: handlePSPClose
|
|
2140
2273
|
}
|
|
2141
2274
|
);
|
|
@@ -2152,21 +2285,8 @@ function ReevitCheckout({
|
|
|
2152
2285
|
customerEmail: email,
|
|
2153
2286
|
customerPhone: momoData?.phone || phone,
|
|
2154
2287
|
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
|
-
}),
|
|
2288
|
+
onSuccess: (res) => handlePSPSuccess({ ...res, psp: "monnify" }),
|
|
2289
|
+
onError: handlePSPError,
|
|
2170
2290
|
onClose: handlePSPClose
|
|
2171
2291
|
}
|
|
2172
2292
|
);
|
|
@@ -2180,22 +2300,9 @@ function ReevitCheckout({
|
|
|
2180
2300
|
currency: paymentIntent?.currency ?? currency,
|
|
2181
2301
|
reference: paymentIntent?.reference || reference || `mpesa_${Date.now()}`,
|
|
2182
2302
|
description: `Payment ${paymentIntent?.reference || reference || ""}`,
|
|
2183
|
-
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
|
-
})
|
|
2303
|
+
headers: { "x-reevit-public-key": publicKey || "" },
|
|
2304
|
+
onSuccess: handlePSPSuccess,
|
|
2305
|
+
onError: handlePSPError
|
|
2199
2306
|
}
|
|
2200
2307
|
);
|
|
2201
2308
|
case "stripe":
|
|
@@ -2206,58 +2313,60 @@ function ReevitCheckout({
|
|
|
2206
2313
|
clientSecret: paymentIntent?.clientSecret || "",
|
|
2207
2314
|
amount: paymentIntent?.amount ?? amount,
|
|
2208
2315
|
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
|
-
}),
|
|
2316
|
+
onSuccess: handlePSPSuccess,
|
|
2317
|
+
onError: handlePSPError,
|
|
2224
2318
|
onCancel: handlePSPClose
|
|
2225
2319
|
}
|
|
2226
2320
|
);
|
|
2227
2321
|
default:
|
|
2228
2322
|
return /* @__PURE__ */ jsxs("div", { className: "reevit-error", children: [
|
|
2229
2323
|
/* @__PURE__ */ jsx("div", { className: "reevit-error__icon", children: "\u26A0\uFE0F" }),
|
|
2230
|
-
/* @__PURE__ */ jsx("h3", { children: "
|
|
2324
|
+
/* @__PURE__ */ jsx("h3", { children: "Provider Not Supported" }),
|
|
2231
2325
|
/* @__PURE__ */ jsxs("p", { children: [
|
|
2232
|
-
"
|
|
2326
|
+
"Provider (",
|
|
2233
2327
|
psp,
|
|
2234
|
-
") is not
|
|
2328
|
+
") is not supported."
|
|
2235
2329
|
] }),
|
|
2236
2330
|
/* @__PURE__ */ jsx("button", { className: "reevit-btn reevit-btn--primary", onClick: handleBack, children: "Go Back" })
|
|
2237
2331
|
] });
|
|
2238
2332
|
}
|
|
2239
2333
|
}
|
|
2240
|
-
|
|
2241
|
-
|
|
2242
|
-
|
|
2243
|
-
|
|
2244
|
-
|
|
2245
|
-
|
|
2246
|
-
isLoading,
|
|
2247
|
-
|
|
2334
|
+
const renderMethodContent = (provider, method) => {
|
|
2335
|
+
const pspLower2 = provider.toLowerCase();
|
|
2336
|
+
const needsPhone = pspLower2.includes("mpesa");
|
|
2337
|
+
if (method === "card") {
|
|
2338
|
+
return /* @__PURE__ */ jsxs("div", { className: "reevit-inline-action reevit-animate-fade-in", children: [
|
|
2339
|
+
/* @__PURE__ */ jsx("p", { className: "reevit-inline-action__hint", children: "You'll be redirected to complete your card payment securely." }),
|
|
2340
|
+
/* @__PURE__ */ jsx("button", { className: "reevit-btn reevit-btn--primary", onClick: handleContinue, disabled: isLoading, children: "Pay with Card" })
|
|
2341
|
+
] });
|
|
2342
|
+
}
|
|
2343
|
+
if (method === "mobile_money") {
|
|
2344
|
+
if (needsPhone && !phone) {
|
|
2345
|
+
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
2346
|
}
|
|
2249
|
-
|
|
2250
|
-
|
|
2251
|
-
|
|
2252
|
-
|
|
2347
|
+
return /* @__PURE__ */ jsxs("div", { className: "reevit-inline-action reevit-animate-fade-in", children: [
|
|
2348
|
+
/* @__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)}.` }),
|
|
2349
|
+
/* @__PURE__ */ jsx("button", { className: "reevit-btn reevit-btn--primary", onClick: handleContinue, disabled: isLoading, children: pspLower2.includes("hubtel") ? "Continue with Hubtel" : "Pay with Mobile Money" })
|
|
2350
|
+
] });
|
|
2351
|
+
}
|
|
2352
|
+
return null;
|
|
2353
|
+
};
|
|
2354
|
+
if (providerOptions.length > 1) {
|
|
2355
|
+
return /* @__PURE__ */ jsx("div", { className: "reevit-method-step reevit-animate-slide-up", children: /* @__PURE__ */ jsx(
|
|
2253
2356
|
ProviderSelector,
|
|
2254
2357
|
{
|
|
2255
2358
|
providers: providerOptions,
|
|
2256
2359
|
selectedProvider,
|
|
2257
2360
|
onSelect: handleProviderSelect,
|
|
2258
|
-
disabled: isLoading
|
|
2361
|
+
disabled: isLoading,
|
|
2362
|
+
theme: resolvedTheme,
|
|
2363
|
+
selectedMethod,
|
|
2364
|
+
onMethodSelect: handleMethodSelect,
|
|
2365
|
+
renderMethodContent
|
|
2259
2366
|
}
|
|
2260
|
-
)
|
|
2367
|
+
) });
|
|
2368
|
+
}
|
|
2369
|
+
return /* @__PURE__ */ jsxs("div", { className: "reevit-method-step reevit-animate-slide-up", children: [
|
|
2261
2370
|
/* @__PURE__ */ jsx(
|
|
2262
2371
|
PaymentMethodSelector,
|
|
2263
2372
|
{
|
|
@@ -2265,21 +2374,26 @@ function ReevitCheckout({
|
|
|
2265
2374
|
selectedMethod,
|
|
2266
2375
|
onSelect: handleMethodSelect,
|
|
2267
2376
|
disabled: isLoading,
|
|
2268
|
-
provider:
|
|
2377
|
+
provider: psp,
|
|
2378
|
+
layout: "list",
|
|
2379
|
+
showLabel: false
|
|
2269
2380
|
}
|
|
2270
2381
|
),
|
|
2271
|
-
selectedMethod &&
|
|
2272
|
-
"
|
|
2273
|
-
|
|
2274
|
-
|
|
2275
|
-
|
|
2276
|
-
|
|
2277
|
-
|
|
2278
|
-
|
|
2279
|
-
|
|
2382
|
+
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: [
|
|
2383
|
+
/* @__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)}.` }),
|
|
2384
|
+
/* @__PURE__ */ jsx(
|
|
2385
|
+
"button",
|
|
2386
|
+
{
|
|
2387
|
+
className: "reevit-btn reevit-btn--primary",
|
|
2388
|
+
onClick: handleContinue,
|
|
2389
|
+
disabled: isLoading,
|
|
2390
|
+
children: selectedMethod === "card" ? "Pay with Card" : pspLower.includes("hubtel") ? "Continue with Hubtel" : "Pay with Mobile Money"
|
|
2391
|
+
}
|
|
2392
|
+
)
|
|
2393
|
+
] }) })
|
|
2280
2394
|
] });
|
|
2281
2395
|
};
|
|
2282
|
-
return /* @__PURE__ */ jsxs(ReevitContext.Provider, { value: { publicKey, amount, currency }, children: [
|
|
2396
|
+
return /* @__PURE__ */ jsxs(ReevitContext.Provider, { value: { publicKey: publicKey || "", amount, currency }, children: [
|
|
2283
2397
|
trigger,
|
|
2284
2398
|
isOpen && /* @__PURE__ */ jsx("div", { className: "reevit-overlay", onClick: handleClose, children: /* @__PURE__ */ jsxs(
|
|
2285
2399
|
"div",
|
|
@@ -2292,23 +2406,8 @@ function ReevitCheckout({
|
|
|
2292
2406
|
"aria-modal": "true",
|
|
2293
2407
|
children: [
|
|
2294
2408
|
/* @__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
|
-
)
|
|
2409
|
+
/* @__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" }) }),
|
|
2410
|
+
/* @__PURE__ */ jsx("button", { className: "reevit-modal__close", onClick: handleClose, "aria-label": "Close", children: "\u2715" })
|
|
2312
2411
|
] }),
|
|
2313
2412
|
/* @__PURE__ */ jsxs("div", { className: "reevit-modal__amount", children: [
|
|
2314
2413
|
/* @__PURE__ */ jsx("span", { className: "reevit-modal__amount-label", children: "Amount" }),
|