@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.js CHANGED
@@ -138,8 +138,8 @@ function createPaymentError(response, errorData) {
138
138
  }
139
139
  var ReevitAPIClient = class {
140
140
  constructor(config) {
141
- this.publicKey = config.publicKey;
142
- this.baseUrl = config.baseUrl || (isSandboxKey(config.publicKey) ? API_BASE_URL_SANDBOX : API_BASE_URL_PRODUCTION);
141
+ this.publicKey = config.publicKey || "";
142
+ this.baseUrl = config.baseUrl || (config.publicKey && isSandboxKey(config.publicKey) ? API_BASE_URL_SANDBOX : API_BASE_URL_PRODUCTION);
143
143
  this.timeout = config.timeout || DEFAULT_TIMEOUT;
144
144
  }
145
145
  /**
@@ -150,10 +150,12 @@ var ReevitAPIClient = class {
150
150
  const timeoutId = setTimeout(() => controller.abort(), this.timeout);
151
151
  const headers = {
152
152
  "Content-Type": "application/json",
153
- "X-Reevit-Key": this.publicKey,
154
153
  "X-Reevit-Client": "@reevit/react",
155
154
  "X-Reevit-Client-Version": "0.3.2"
156
155
  };
156
+ if (this.publicKey) {
157
+ headers["X-Reevit-Key"] = this.publicKey;
158
+ }
157
159
  if (method === "POST" || method === "PATCH" || method === "PUT") {
158
160
  headers["Idempotency-Key"] = `${Date.now()}-${Math.random().toString(36).substring(2, 15)}`;
159
161
  }
@@ -260,8 +262,9 @@ var ReevitAPIClient = class {
260
262
  * @param paymentId - The payment intent ID for Hubtel checkout
261
263
  * @returns Hubtel session with token, merchant account, and expiry information
262
264
  */
263
- async createHubtelSession(paymentId) {
264
- return this.request("POST", `/v1/payments/hubtel/sessions/${paymentId}`);
265
+ async createHubtelSession(paymentId, clientSecret) {
266
+ const query = clientSecret ? `?client_secret=${encodeURIComponent(clientSecret)}` : "";
267
+ return this.request("POST", `/v1/payments/hubtel/sessions/${paymentId}${query}`);
265
268
  }
266
269
  /**
267
270
  * Maps SDK payment method to backend format
@@ -291,6 +294,18 @@ var initialState = {
291
294
  error: null,
292
295
  result: null
293
296
  };
297
+ var DEFAULT_PUBLIC_API_BASE_URL = "https://api.reevit.io";
298
+ function buildPaymentLinkError(response, data) {
299
+ return {
300
+ code: data?.code || "payment_link_error",
301
+ message: data?.message || "Payment link request failed",
302
+ recoverable: true,
303
+ details: {
304
+ httpStatus: response.status,
305
+ ...data?.details || {}
306
+ }
307
+ };
308
+ }
294
309
  function reevitReducer(state, action) {
295
310
  switch (action.type) {
296
311
  case "INIT_START":
@@ -414,15 +429,48 @@ function useReevit(options) {
414
429
  const reference = config.reference || generateReference();
415
430
  const country = detectCountryFromCurrency(config.currency);
416
431
  const paymentMethod = method || config.paymentMethods?.[0] || "card";
417
- const { data, error } = await apiClient.createPaymentIntent(
418
- { ...config, reference },
419
- paymentMethod,
420
- country,
421
- {
422
- preferredProviders: options2?.preferredProvider ? [options2.preferredProvider] : void 0,
423
- allowedProviders: options2?.allowedProviders
432
+ let data;
433
+ let error;
434
+ if (config.paymentLinkCode) {
435
+ const response = await fetch(
436
+ `${apiBaseUrl || DEFAULT_PUBLIC_API_BASE_URL}/v1/pay/${config.paymentLinkCode}/pay`,
437
+ {
438
+ method: "POST",
439
+ headers: {
440
+ "Content-Type": "application/json",
441
+ "Idempotency-Key": `${Date.now()}-${Math.random().toString(36).substring(2, 15)}`
442
+ },
443
+ body: JSON.stringify({
444
+ amount: config.amount,
445
+ email: config.email || "",
446
+ name: config.customerName || "",
447
+ phone: config.phone || "",
448
+ method: paymentMethod,
449
+ country,
450
+ provider: options2?.preferredProvider || options2?.allowedProviders?.[0],
451
+ custom_fields: config.customFields
452
+ })
453
+ }
454
+ );
455
+ const responseData = await response.json().catch(() => ({}));
456
+ if (!response.ok) {
457
+ error = buildPaymentLinkError(response, responseData);
458
+ } else {
459
+ data = responseData;
424
460
  }
425
- );
461
+ } else {
462
+ const result = await apiClient.createPaymentIntent(
463
+ { ...config, reference },
464
+ paymentMethod,
465
+ country,
466
+ {
467
+ preferredProviders: options2?.preferredProvider ? [options2.preferredProvider] : void 0,
468
+ allowedProviders: options2?.allowedProviders
469
+ }
470
+ );
471
+ data = result.data;
472
+ error = result.error;
473
+ }
426
474
  if (error) {
427
475
  dispatch({ type: "INIT_ERROR", payload: error });
428
476
  onError?.(error);
@@ -587,14 +635,6 @@ function detectCountryFromCurrency(currency) {
587
635
  };
588
636
  return currencyToCountry[currency.toUpperCase()] || "GH";
589
637
  }
590
- var pspNames = {
591
- hubtel: "Hubtel",
592
- paystack: "Paystack",
593
- flutterwave: "Flutterwave",
594
- monnify: "Monnify",
595
- mpesa: "M-Pesa",
596
- stripe: "Stripe"
597
- };
598
638
  var methodConfig = {
599
639
  card: {
600
640
  label: "Card",
@@ -617,28 +657,31 @@ function PaymentMethodSelector({
617
657
  selectedMethod,
618
658
  onSelect,
619
659
  disabled = false,
620
- provider
660
+ provider,
661
+ layout = "list",
662
+ showLabel = true
621
663
  }) {
622
664
  const getMethodLabel = (method, psp) => {
623
665
  const config = methodConfig[method];
624
- if (psp?.toLowerCase().includes("hubtel") && method === "mobile_money") {
625
- return `Pay with ${pspNames[psp.toLowerCase()] || "Hubtel"}`;
626
- }
627
666
  return config.label;
628
667
  };
629
668
  const getMethodDescription = (method, psp) => {
630
669
  const config = methodConfig[method];
631
670
  if (psp?.toLowerCase().includes("hubtel")) {
632
- return "Card, Mobile Money, and Bank Transfer";
671
+ return config.description;
633
672
  }
634
673
  return config.description;
635
674
  };
636
- return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-method-selector", children: [
637
- /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-method-selector__label", children: "Select payment method" }),
638
- /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-method-selector__options", children: methods.map((method) => {
675
+ const isGrid = layout === "grid";
676
+ return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: cn("reevit-method-selector", isGrid && "reevit-method-selector--grid"), children: [
677
+ showLabel && /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-method-selector__label", children: "Select payment method" }),
678
+ /* @__PURE__ */ jsxRuntime.jsx("div", { className: cn(
679
+ "reevit-method-selector__options",
680
+ isGrid ? "reevit-method-selector__options--grid" : "reevit-method-selector__options--list"
681
+ ), children: methods.map((method, index) => {
639
682
  const config = methodConfig[method];
640
683
  const isSelected = selectedMethod === method;
641
- const methodLabel = getMethodLabel(method, provider);
684
+ const methodLabel = getMethodLabel(method);
642
685
  const methodDescription = getMethodDescription(method, provider);
643
686
  return /* @__PURE__ */ jsxRuntime.jsxs(
644
687
  "button",
@@ -646,28 +689,24 @@ function PaymentMethodSelector({
646
689
  type: "button",
647
690
  className: cn(
648
691
  "reevit-method-option",
692
+ isGrid ? "reevit-method-option--grid" : "reevit-method-option--list",
649
693
  isSelected && "reevit-method-option--selected",
650
694
  disabled && "reevit-method-option--disabled"
651
695
  ),
696
+ style: {
697
+ animationDelay: `${index * 0.05}s`
698
+ },
652
699
  onClick: () => onSelect(method),
653
700
  disabled,
654
701
  "aria-pressed": isSelected,
655
702
  children: [
656
- /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-method-option__icon", children: config.icon }),
703
+ /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-method-option__icon-wrapper", children: /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-method-option__icon", children: config.icon }) }),
657
704
  /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-method-option__content", children: [
658
705
  /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-method-option__label", children: methodLabel }),
659
- /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-method-option__description", children: methodDescription })
706
+ !isGrid && /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-method-option__description", children: methodDescription })
660
707
  ] }),
661
- isSelected && /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-method-option__check", children: /* @__PURE__ */ jsxRuntime.jsx("svg", { width: "20", height: "20", viewBox: "0 0 20 20", fill: "none", children: /* @__PURE__ */ jsxRuntime.jsx(
662
- "path",
663
- {
664
- d: "M16.667 5L7.5 14.167 3.333 10",
665
- stroke: "currentColor",
666
- strokeWidth: "2",
667
- strokeLinecap: "round",
668
- strokeLinejoin: "round"
669
- }
670
- ) }) })
708
+ !isGrid && isSelected && /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-method-option__check", children: /* @__PURE__ */ jsxRuntime.jsx("svg", { width: "20", height: "20", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "3", strokeLinecap: "round", strokeLinejoin: "round", children: /* @__PURE__ */ jsxRuntime.jsx("polyline", { points: "20 6 9 17 4 12" }) }) }),
709
+ !isGrid && !isSelected && /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-method-option__chevron", children: /* @__PURE__ */ jsxRuntime.jsx("svg", { width: "16", height: "16", viewBox: "0 0 24 24", fill: "none", stroke: "currentColor", strokeWidth: "2.5", strokeLinecap: "round", strokeLinejoin: "round", children: /* @__PURE__ */ jsxRuntime.jsx("polyline", { points: "9 18 15 12 9 6" }) }) })
671
710
  ]
672
711
  },
673
712
  method
@@ -684,7 +723,8 @@ function MobileMoneyForm({
684
723
  onSubmit,
685
724
  onCancel,
686
725
  isLoading = false,
687
- initialPhone = ""
726
+ initialPhone = "",
727
+ hideCancel = false
688
728
  }) {
689
729
  const [phone, setPhone] = react.useState(initialPhone);
690
730
  const [network, setNetwork] = react.useState(null);
@@ -769,7 +809,7 @@ function MobileMoneyForm({
769
809
  )) })
770
810
  ] }),
771
811
  /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-momo-form__actions", children: [
772
- /* @__PURE__ */ jsxRuntime.jsx(
812
+ !hideCancel && /* @__PURE__ */ jsxRuntime.jsx(
773
813
  "button",
774
814
  {
775
815
  type: "button",
@@ -811,11 +851,11 @@ var paystack_default = "
811
851
  // src/assets/providers/stripe.png
812
852
  var stripe_default = "";
813
853
  var providerMeta = {
814
- paystack: { logo: paystack_default, hint: "Card & MoMo" },
854
+ paystack: { logo: paystack_default, hint: "Card & Mobile Money" },
815
855
  stripe: { logo: stripe_default, hint: "Card payments" },
816
856
  flutterwave: { logo: flutterwave_default, hint: "Global methods" },
817
- hubtel: { logo: hubtel_default, hint: "Mobile money" },
818
- monnify: { logo: monnify_default, hint: "Bank & card" },
857
+ hubtel: { logo: hubtel_default, hint: "Mobile Money & Card" },
858
+ monnify: { logo: monnify_default, hint: "Bank & Card" },
819
859
  mpesa: { logo: mpesa_default, hint: "M-Pesa" }
820
860
  };
821
861
  var methodLabels = {
@@ -825,54 +865,100 @@ var methodLabels = {
825
865
  };
826
866
  function formatMethods(methods) {
827
867
  if (!methods.length) return "Payment methods";
828
- return methods.map((method) => methodLabels[method]).join(" \u2022 ");
868
+ return methods.map((method) => methodLabels[method]).join(", ");
869
+ }
870
+ function sanitizeMethods(providerId, methods) {
871
+ if (providerId.toLowerCase().includes("hubtel")) {
872
+ return methods.filter((method) => method === "card" || method === "mobile_money");
873
+ }
874
+ return methods;
829
875
  }
830
876
  function ProviderSelector({
831
877
  providers,
832
878
  selectedProvider,
833
879
  onSelect,
834
- disabled = false
880
+ disabled = false,
881
+ theme,
882
+ selectedMethod,
883
+ onMethodSelect,
884
+ renderMethodContent
835
885
  }) {
886
+ const useBorder = theme?.pspSelectorUseBorder ?? false;
887
+ const bgColor = theme?.pspSelectorBgColor || "#0a0a0a";
888
+ const textColor = theme?.pspSelectorTextColor || "#ffffff";
889
+ const borderColor = theme?.pspSelectorBorderColor || "#374151";
890
+ const getOptionStyle = (isSelected) => {
891
+ if (useBorder) {
892
+ return {
893
+ backgroundColor: "transparent",
894
+ border: `2px solid ${isSelected ? borderColor : "#374151"}`,
895
+ color: isSelected ? textColor : "var(--reevit-text)"
896
+ };
897
+ }
898
+ return {
899
+ backgroundColor: isSelected ? bgColor : "transparent",
900
+ border: `2px solid ${isSelected ? bgColor : "#374151"}`,
901
+ color: isSelected ? textColor : "var(--reevit-text)"
902
+ };
903
+ };
836
904
  return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-psp-selector", children: [
837
- /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-psp-selector__label", children: "Choose your payment provider" }),
905
+ /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-psp-selector__label", children: "Select payment provider" }),
838
906
  /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-psp-selector__options", children: providers.map((provider) => {
839
907
  const meta = providerMeta[provider.provider] || {
840
908
  logo: void 0,
841
909
  hint: "Payment methods"
842
910
  };
911
+ const providerMethods = sanitizeMethods(provider.provider, provider.methods);
843
912
  const isSelected = selectedProvider === provider.provider;
844
913
  const fallbackInitial = provider.name.slice(0, 1).toUpperCase();
845
- return /* @__PURE__ */ jsxRuntime.jsxs(
846
- "button",
847
- {
848
- type: "button",
849
- className: cn(
850
- "reevit-psp-option",
851
- isSelected && "reevit-psp-option--selected",
852
- disabled && "reevit-psp-option--disabled"
853
- ),
854
- onClick: () => onSelect(provider.provider),
855
- disabled,
856
- "aria-pressed": isSelected,
857
- children: [
858
- /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-psp-option__logo", "aria-hidden": "true", children: meta.logo ? /* @__PURE__ */ jsxRuntime.jsx(
859
- "img",
860
- {
861
- src: meta.logo,
862
- alt: "",
863
- className: "reevit-psp-option__logo-img",
864
- loading: "lazy"
865
- }
866
- ) : /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-psp-option__logo-fallback", children: fallbackInitial }) }),
867
- /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-psp-option__content", children: [
868
- /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-psp-option__name", children: provider.name }),
869
- /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-psp-option__methods", children: formatMethods(provider.methods) || meta.hint })
870
- ] }),
871
- /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-psp-option__check", "aria-hidden": "true", children: isSelected ? "Selected" : "Select" })
872
- ]
873
- },
874
- provider.provider
875
- );
914
+ const optionStyle = getOptionStyle(isSelected);
915
+ return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-psp-accordion", children: [
916
+ /* @__PURE__ */ jsxRuntime.jsxs(
917
+ "button",
918
+ {
919
+ type: "button",
920
+ className: cn(
921
+ "reevit-psp-option",
922
+ isSelected && "reevit-psp-option--selected",
923
+ disabled && "reevit-psp-option--disabled"
924
+ ),
925
+ style: optionStyle,
926
+ onClick: () => onSelect(provider.provider),
927
+ disabled,
928
+ "aria-expanded": isSelected,
929
+ children: [
930
+ /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-psp-option__logo", "aria-hidden": "true", children: meta.logo ? /* @__PURE__ */ jsxRuntime.jsx(
931
+ "img",
932
+ {
933
+ src: meta.logo,
934
+ alt: "",
935
+ className: "reevit-psp-option__logo-img",
936
+ loading: "lazy"
937
+ }
938
+ ) : /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-psp-option__logo-fallback", children: fallbackInitial }) }),
939
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-psp-option__content", children: [
940
+ /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-psp-option__name", children: provider.name }),
941
+ /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-psp-option__methods", children: formatMethods(providerMethods) || meta.hint })
942
+ ] })
943
+ ]
944
+ }
945
+ ),
946
+ isSelected && onMethodSelect && /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-psp-accordion__content", children: [
947
+ /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-psp-methods", children: /* @__PURE__ */ jsxRuntime.jsx(
948
+ PaymentMethodSelector,
949
+ {
950
+ methods: providerMethods,
951
+ selectedMethod: selectedMethod || null,
952
+ onSelect: onMethodSelect,
953
+ disabled,
954
+ provider: provider.provider,
955
+ layout: "list",
956
+ showLabel: false
957
+ }
958
+ ) }),
959
+ selectedMethod && renderMethodContent && /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-psp-accordion__method-content", children: renderMethodContent(provider.provider, selectedMethod) })
960
+ ] })
961
+ ] }, provider.provider);
876
962
  }) })
877
963
  ] });
878
964
  }
@@ -1004,8 +1090,11 @@ function HubtelBridge({
1004
1090
  phone,
1005
1091
  description = "Payment",
1006
1092
  callbackUrl,
1093
+ apiBaseUrl,
1094
+ clientSecret,
1007
1095
  hubtelSessionToken,
1008
1096
  basicAuth,
1097
+ preferredMethod,
1009
1098
  onSuccess,
1010
1099
  onError,
1011
1100
  onClose,
@@ -1015,13 +1104,17 @@ function HubtelBridge({
1015
1104
  const checkoutRef = react.useRef(null);
1016
1105
  const [authValue, setAuthValue] = react.useState("");
1017
1106
  const [isLoading, setIsLoading] = react.useState(false);
1107
+ const [resolvedMerchantAccount, setResolvedMerchantAccount] = react.useState(merchantAccount);
1108
+ react.useEffect(() => {
1109
+ setResolvedMerchantAccount(merchantAccount);
1110
+ }, [merchantAccount]);
1018
1111
  react.useEffect(() => {
1019
1112
  const fetchAuth = async () => {
1020
1113
  if (hubtelSessionToken) {
1021
1114
  setIsLoading(true);
1022
1115
  try {
1023
- const client = createReevitClient({ publicKey });
1024
- const { data, error } = await client.createHubtelSession(paymentId);
1116
+ const client = createReevitClient({ publicKey, baseUrl: apiBaseUrl });
1117
+ const { data, error } = await client.createHubtelSession(paymentId, clientSecret);
1025
1118
  if (error) {
1026
1119
  onError({
1027
1120
  code: "SESSION_ERROR",
@@ -1032,6 +1125,9 @@ function HubtelBridge({
1032
1125
  }
1033
1126
  if (data) {
1034
1127
  setAuthValue(data.token);
1128
+ if (data.merchantAccount) {
1129
+ setResolvedMerchantAccount(data.merchantAccount);
1130
+ }
1035
1131
  }
1036
1132
  } catch (err) {
1037
1133
  onError({
@@ -1048,7 +1144,7 @@ function HubtelBridge({
1048
1144
  }
1049
1145
  };
1050
1146
  fetchAuth();
1051
- }, [paymentId, publicKey, hubtelSessionToken, basicAuth, onError]);
1147
+ }, [paymentId, publicKey, apiBaseUrl, clientSecret, hubtelSessionToken, basicAuth, onError]);
1052
1148
  const startPayment = react.useCallback(async () => {
1053
1149
  if (isLoading || !authValue) {
1054
1150
  return;
@@ -1056,20 +1152,23 @@ function HubtelBridge({
1056
1152
  try {
1057
1153
  const checkout = new CheckoutSdk__default.default();
1058
1154
  checkoutRef.current = checkout;
1155
+ const methodPreference = preferredMethod === "mobile_money" ? "momo" : preferredMethod === "card" ? "card" : void 0;
1059
1156
  const purchaseInfo = {
1060
1157
  amount: amount / 100,
1061
1158
  // Convert from minor to major units
1062
1159
  purchaseDescription: description,
1063
1160
  customerPhoneNumber: phone || "",
1064
- clientReference: reference || `hubtel_${Date.now()}`
1161
+ clientReference: reference || `hubtel_${Date.now()}`,
1162
+ ...methodPreference ? { paymentMethod: methodPreference } : {}
1065
1163
  };
1066
1164
  const config = {
1067
1165
  branding: "enabled",
1068
1166
  callbackUrl: callbackUrl || window.location.href,
1069
- merchantAccount: typeof merchantAccount === "string" ? parseInt(merchantAccount, 10) : merchantAccount,
1167
+ merchantAccount: typeof resolvedMerchantAccount === "string" ? parseInt(resolvedMerchantAccount, 10) : resolvedMerchantAccount,
1070
1168
  // Use session token or basicAuth for authentication
1071
1169
  // Session tokens are base64-encoded credentials fetched securely from the server
1072
- basicAuth: authValue || ""
1170
+ basicAuth: authValue || "",
1171
+ ...methodPreference ? { paymentMethod: methodPreference } : {}
1073
1172
  };
1074
1173
  checkout.openModal({
1075
1174
  purchaseInfo,
@@ -1113,7 +1212,7 @@ function HubtelBridge({
1113
1212
  };
1114
1213
  onError(error);
1115
1214
  }
1116
- }, [merchantAccount, amount, reference, phone, description, callbackUrl, authValue, isLoading, onSuccess, onError, onClose]);
1215
+ }, [merchantAccount, amount, reference, phone, description, callbackUrl, authValue, isLoading, preferredMethod, onSuccess, onError, onClose]);
1117
1216
  react.useEffect(() => {
1118
1217
  if (autoStart && !initialized.current && !isLoading && authValue) {
1119
1218
  initialized.current = true;
@@ -1127,17 +1226,20 @@ function HubtelBridge({
1127
1226
  }
1128
1227
  function openHubtelPopup(config) {
1129
1228
  const checkout = new CheckoutSdk__default.default();
1229
+ const methodPreference = config.preferredMethod === "mobile_money" ? "momo" : config.preferredMethod === "card" ? "card" : void 0;
1130
1230
  const purchaseInfo = {
1131
1231
  amount: config.amount,
1132
1232
  purchaseDescription: config.description,
1133
1233
  customerPhoneNumber: config.customerPhoneNumber || "",
1134
- clientReference: config.clientReference || `hubtel_${Date.now()}`
1234
+ clientReference: config.clientReference || `hubtel_${Date.now()}`,
1235
+ ...methodPreference ? { paymentMethod: methodPreference } : {}
1135
1236
  };
1136
1237
  const checkoutConfig = {
1137
1238
  branding: "enabled",
1138
1239
  callbackUrl: config.callbackUrl || window.location.href,
1139
1240
  merchantAccount: typeof config.merchantAccount === "string" ? parseInt(config.merchantAccount, 10) : config.merchantAccount,
1140
- basicAuth: config.basicAuth || ""
1241
+ basicAuth: config.basicAuth || "",
1242
+ ...methodPreference ? { paymentMethod: methodPreference } : {}
1141
1243
  };
1142
1244
  checkout.openModal({
1143
1245
  purchaseInfo,
@@ -1830,6 +1932,14 @@ function useReevitContext() {
1830
1932
  }
1831
1933
  return context;
1832
1934
  }
1935
+ var pspNames = {
1936
+ hubtel: "Hubtel",
1937
+ paystack: "Paystack",
1938
+ flutterwave: "Flutterwave",
1939
+ monnify: "Monnify",
1940
+ mpesa: "M-Pesa",
1941
+ stripe: "Stripe"
1942
+ };
1833
1943
  function ReevitCheckout({
1834
1944
  // Config
1835
1945
  publicKey,
@@ -1837,8 +1947,11 @@ function ReevitCheckout({
1837
1947
  currency,
1838
1948
  email = "",
1839
1949
  phone = "",
1950
+ customerName,
1840
1951
  reference,
1841
1952
  metadata,
1953
+ customFields,
1954
+ paymentLinkCode,
1842
1955
  paymentMethods = ["card", "mobile_money"],
1843
1956
  initialPaymentIntent,
1844
1957
  // Callbacks
@@ -1889,8 +2002,11 @@ function ReevitCheckout({
1889
2002
  currency,
1890
2003
  email,
1891
2004
  phone,
2005
+ customerName,
1892
2006
  reference,
1893
2007
  metadata,
2008
+ customFields,
2009
+ paymentLinkCode,
1894
2010
  paymentMethods,
1895
2011
  initialPaymentIntent
1896
2012
  },
@@ -1918,7 +2034,8 @@ function ReevitCheckout({
1918
2034
  const options = available.length > 0 ? available : fallbackProvider;
1919
2035
  return options.map((provider) => {
1920
2036
  const methods = provider.methods && provider.methods.length > 0 ? provider.methods : paymentMethods;
1921
- const filteredMethods = methods.filter((method) => paymentMethods.includes(method));
2037
+ const sanitizedMethods = provider.provider.toLowerCase().includes("hubtel") ? methods.filter((method) => method === "card" || method === "mobile_money") : methods;
2038
+ const filteredMethods = sanitizedMethods.filter((method) => paymentMethods.includes(method));
1922
2039
  return {
1923
2040
  ...provider,
1924
2041
  methods: filteredMethods
@@ -1939,10 +2056,10 @@ function ReevitCheckout({
1939
2056
  if (selectedProvider && providerOptions.some((provider) => provider.provider === selectedProvider)) {
1940
2057
  return;
1941
2058
  }
1942
- const recommended = paymentIntent?.recommendedPsp;
1943
- const resolved = providerOptions.find((provider) => provider.provider === recommended)?.provider || providerOptions[0].provider;
1944
- setSelectedProvider(resolved);
1945
- }, [paymentIntent?.recommendedPsp, providerOptions, selectedProvider]);
2059
+ if (providerOptions.length === 1) {
2060
+ setSelectedProvider(providerOptions[0].provider);
2061
+ }
2062
+ }, [providerOptions, selectedProvider]);
1946
2063
  react.useEffect(() => {
1947
2064
  if (!activeProvider || !selectedMethod) {
1948
2065
  return;
@@ -1953,13 +2070,17 @@ function ReevitCheckout({
1953
2070
  }, [activeProvider, selectedMethod, selectMethod]);
1954
2071
  react.useEffect(() => {
1955
2072
  if (isOpen && selectedMethod && paymentIntent && !showPSPBridge) {
2073
+ const psp = (selectedProvider || paymentIntent.recommendedPsp || "paystack").toLowerCase();
2074
+ const needsPhone = psp.includes("mpesa");
1956
2075
  if (selectedMethod === "card") {
1957
2076
  setShowPSPBridge(true);
1958
- } else if (selectedMethod === "mobile_money" && (momoData?.phone || phone)) {
1959
- setShowPSPBridge(true);
2077
+ } else if (selectedMethod === "mobile_money") {
2078
+ if (!needsPhone || (momoData?.phone || phone)) {
2079
+ setShowPSPBridge(true);
2080
+ }
1960
2081
  }
1961
2082
  }
1962
- }, [isOpen, selectedMethod, showPSPBridge, paymentIntent, momoData, phone]);
2083
+ }, [isOpen, selectedMethod, showPSPBridge, paymentIntent, momoData, phone, selectedProvider]);
1963
2084
  const handleOpen = react.useCallback(() => {
1964
2085
  if (controlledIsOpen !== void 0) return;
1965
2086
  setIsOpen(true);
@@ -1983,6 +2104,10 @@ function ReevitCheckout({
1983
2104
  const handleProviderSelect = react.useCallback(
1984
2105
  (provider) => {
1985
2106
  if (provider === selectedProvider) {
2107
+ setSelectedProvider(null);
2108
+ reset();
2109
+ setShowPSPBridge(false);
2110
+ setMomoData(null);
1986
2111
  return;
1987
2112
  }
1988
2113
  const providerMethods = providerOptions.find((option) => option.provider === provider)?.methods || paymentMethods;
@@ -1999,8 +2124,14 @@ function ReevitCheckout({
1999
2124
  if (!selectedMethod) return;
2000
2125
  if (selectedMethod === "card") {
2001
2126
  setShowPSPBridge(true);
2127
+ } else if (selectedMethod === "mobile_money") {
2128
+ const psp = (selectedProvider || paymentIntent?.recommendedPsp || "paystack").toLowerCase();
2129
+ const needsPhone = psp.includes("mpesa");
2130
+ if (!needsPhone || (momoData?.phone || phone)) {
2131
+ setShowPSPBridge(true);
2132
+ }
2002
2133
  }
2003
- }, [selectedMethod]);
2134
+ }, [selectedMethod, selectedProvider, paymentIntent, momoData, phone]);
2004
2135
  const handleMomoSubmit = react.useCallback(
2005
2136
  (data) => {
2006
2137
  setMomoData(data);
@@ -2057,13 +2188,13 @@ function ReevitCheckout({
2057
2188
  ] }) : null;
2058
2189
  const renderContent = () => {
2059
2190
  if (status === "loading" || status === "processing") {
2060
- return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-loading", children: [
2191
+ return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-loading reevit-animate-fade-in", children: [
2061
2192
  /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-spinner" }),
2062
2193
  /* @__PURE__ */ jsxRuntime.jsx("p", { children: status === "loading" ? "Preparing checkout..." : "Processing payment..." })
2063
2194
  ] });
2064
2195
  }
2065
2196
  if (status === "success" && result) {
2066
- return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-success", children: [
2197
+ return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-success reevit-animate-scale-in", children: [
2067
2198
  /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-success__icon", children: "\u2713" }),
2068
2199
  /* @__PURE__ */ jsxRuntime.jsx("h3", { children: "Payment Successful" }),
2069
2200
  /* @__PURE__ */ jsxRuntime.jsxs("p", { children: [
@@ -2073,7 +2204,7 @@ function ReevitCheckout({
2073
2204
  ] });
2074
2205
  }
2075
2206
  if (status === "failed" && error && !error.recoverable) {
2076
- return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-error", children: [
2207
+ return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-error reevit-animate-fade-in", children: [
2077
2208
  /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-error__icon", children: "\u2715" }),
2078
2209
  /* @__PURE__ */ jsxRuntime.jsx("h3", { children: "Payment Failed" }),
2079
2210
  /* @__PURE__ */ jsxRuntime.jsx("p", { children: error.message }),
@@ -2081,17 +2212,16 @@ function ReevitCheckout({
2081
2212
  ] });
2082
2213
  }
2083
2214
  const psp = selectedProvider || paymentIntent?.recommendedPsp || "paystack";
2215
+ const pspLower = psp.toLowerCase();
2084
2216
  if (showPSPBridge) {
2085
- const pspKey = paymentIntent?.pspPublicKey || publicKey;
2217
+ const pspKey = paymentIntent?.pspPublicKey || publicKey || "";
2086
2218
  const bridgeMetadata = {
2087
2219
  ...metadata,
2088
- // Override with correct payment intent ID for webhook routing
2089
- // This ensures webhook includes the correct ID to find the payment
2090
2220
  payment_id: paymentIntent?.id,
2091
2221
  connection_id: paymentIntent?.connectionId ?? metadata?.connection_id,
2092
2222
  customer_phone: momoData?.phone || phone
2093
2223
  };
2094
- switch (psp) {
2224
+ switch (pspLower) {
2095
2225
  case "paystack":
2096
2226
  return /* @__PURE__ */ jsxRuntime.jsx(
2097
2227
  PaystackBridge,
@@ -2124,6 +2254,9 @@ function ReevitCheckout({
2124
2254
  phone: momoData?.phone || phone,
2125
2255
  description: `Payment ${paymentIntent?.reference || reference || ""}`,
2126
2256
  hubtelSessionToken: paymentIntent?.id ? paymentIntent.id : void 0,
2257
+ clientSecret: paymentIntent?.clientSecret,
2258
+ apiBaseUrl,
2259
+ preferredMethod: selectedMethod || void 0,
2127
2260
  onSuccess: handlePSPSuccess,
2128
2261
  onError: (err) => handlePSPError(err),
2129
2262
  onClose: handlePSPClose
@@ -2141,7 +2274,7 @@ function ReevitCheckout({
2141
2274
  phone: momoData?.phone || phone,
2142
2275
  metadata: bridgeMetadata,
2143
2276
  onSuccess: handlePSPSuccess,
2144
- onError: (err) => handlePSPError(err),
2277
+ onError: handlePSPError,
2145
2278
  onClose: handlePSPClose
2146
2279
  }
2147
2280
  );
@@ -2158,21 +2291,8 @@ function ReevitCheckout({
2158
2291
  customerEmail: email,
2159
2292
  customerPhone: momoData?.phone || phone,
2160
2293
  metadata: bridgeMetadata,
2161
- onSuccess: (result2) => handlePSPSuccess({
2162
- paymentId: result2.transactionReference,
2163
- reference: result2.paymentReference,
2164
- amount: result2.amount,
2165
- currency: paymentIntent?.currency ?? currency,
2166
- paymentMethod: selectedMethod || "card",
2167
- psp: "monnify",
2168
- pspReference: result2.transactionReference,
2169
- status: "success"
2170
- }),
2171
- onError: (err) => handlePSPError({
2172
- code: err.code,
2173
- message: err.message,
2174
- recoverable: true
2175
- }),
2294
+ onSuccess: (res) => handlePSPSuccess({ ...res, psp: "monnify" }),
2295
+ onError: handlePSPError,
2176
2296
  onClose: handlePSPClose
2177
2297
  }
2178
2298
  );
@@ -2186,22 +2306,9 @@ function ReevitCheckout({
2186
2306
  currency: paymentIntent?.currency ?? currency,
2187
2307
  reference: paymentIntent?.reference || reference || `mpesa_${Date.now()}`,
2188
2308
  description: `Payment ${paymentIntent?.reference || reference || ""}`,
2189
- headers: { "x-reevit-public-key": publicKey },
2190
- onSuccess: (result2) => handlePSPSuccess({
2191
- paymentId: result2.transactionId,
2192
- reference: result2.reference,
2193
- amount: paymentIntent?.amount ?? amount,
2194
- currency: paymentIntent?.currency ?? currency,
2195
- paymentMethod: "mobile_money",
2196
- psp: "mpesa",
2197
- pspReference: result2.transactionId,
2198
- status: "success"
2199
- }),
2200
- onError: (err) => handlePSPError({
2201
- code: err.code,
2202
- message: err.message,
2203
- recoverable: true
2204
- })
2309
+ headers: { "x-reevit-public-key": publicKey || "" },
2310
+ onSuccess: handlePSPSuccess,
2311
+ onError: handlePSPError
2205
2312
  }
2206
2313
  );
2207
2314
  case "stripe":
@@ -2212,58 +2319,60 @@ function ReevitCheckout({
2212
2319
  clientSecret: paymentIntent?.clientSecret || "",
2213
2320
  amount: paymentIntent?.amount ?? amount,
2214
2321
  currency: paymentIntent?.currency ?? currency,
2215
- onSuccess: (result2) => handlePSPSuccess({
2216
- paymentId: result2.paymentIntentId,
2217
- reference: paymentIntent?.reference || reference || result2.paymentIntentId,
2218
- amount: paymentIntent?.amount ?? amount,
2219
- currency: paymentIntent?.currency ?? currency,
2220
- paymentMethod: selectedMethod || "card",
2221
- psp: "stripe",
2222
- pspReference: result2.paymentIntentId,
2223
- status: result2.status === "succeeded" ? "success" : "pending"
2224
- }),
2225
- onError: (err) => handlePSPError({
2226
- code: err.code,
2227
- message: err.message,
2228
- recoverable: true
2229
- }),
2322
+ onSuccess: handlePSPSuccess,
2323
+ onError: handlePSPError,
2230
2324
  onCancel: handlePSPClose
2231
2325
  }
2232
2326
  );
2233
2327
  default:
2234
2328
  return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-error", children: [
2235
2329
  /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-error__icon", children: "\u26A0\uFE0F" }),
2236
- /* @__PURE__ */ jsxRuntime.jsx("h3", { children: "Payment Provider Not Supported" }),
2330
+ /* @__PURE__ */ jsxRuntime.jsx("h3", { children: "Provider Not Supported" }),
2237
2331
  /* @__PURE__ */ jsxRuntime.jsxs("p", { children: [
2238
- "The selected payment provider (",
2332
+ "Provider (",
2239
2333
  psp,
2240
- ") is not currently supported in this checkout."
2334
+ ") is not supported."
2241
2335
  ] }),
2242
2336
  /* @__PURE__ */ jsxRuntime.jsx("button", { className: "reevit-btn reevit-btn--primary", onClick: handleBack, children: "Go Back" })
2243
2337
  ] });
2244
2338
  }
2245
2339
  }
2246
- if (selectedMethod === "mobile_money" && !showPSPBridge) {
2247
- return /* @__PURE__ */ jsxRuntime.jsx(
2248
- MobileMoneyForm,
2249
- {
2250
- onSubmit: handleMomoSubmit,
2251
- onCancel: handleBack,
2252
- isLoading,
2253
- initialPhone: phone
2340
+ const renderMethodContent = (provider, method) => {
2341
+ const pspLower2 = provider.toLowerCase();
2342
+ const needsPhone = pspLower2.includes("mpesa");
2343
+ if (method === "card") {
2344
+ return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-inline-action reevit-animate-fade-in", children: [
2345
+ /* @__PURE__ */ jsxRuntime.jsx("p", { className: "reevit-inline-action__hint", children: "You'll be redirected to complete your card payment securely." }),
2346
+ /* @__PURE__ */ jsxRuntime.jsx("button", { className: "reevit-btn reevit-btn--primary", onClick: handleContinue, disabled: isLoading, children: "Pay with Card" })
2347
+ ] });
2348
+ }
2349
+ if (method === "mobile_money") {
2350
+ if (needsPhone && !phone) {
2351
+ return /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-inline-action reevit-animate-fade-in", children: /* @__PURE__ */ jsxRuntime.jsx(MobileMoneyForm, { onSubmit: handleMomoSubmit, onCancel: handleBack, isLoading, initialPhone: phone, hideCancel: true }) });
2254
2352
  }
2255
- );
2256
- }
2257
- return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-method-step", children: [
2258
- providerOptions.length > 1 && /* @__PURE__ */ jsxRuntime.jsx(
2353
+ return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-inline-action reevit-animate-fade-in", children: [
2354
+ /* @__PURE__ */ jsxRuntime.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)}.` }),
2355
+ /* @__PURE__ */ jsxRuntime.jsx("button", { className: "reevit-btn reevit-btn--primary", onClick: handleContinue, disabled: isLoading, children: pspLower2.includes("hubtel") ? "Continue with Hubtel" : "Pay with Mobile Money" })
2356
+ ] });
2357
+ }
2358
+ return null;
2359
+ };
2360
+ if (providerOptions.length > 1) {
2361
+ return /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-method-step reevit-animate-slide-up", children: /* @__PURE__ */ jsxRuntime.jsx(
2259
2362
  ProviderSelector,
2260
2363
  {
2261
2364
  providers: providerOptions,
2262
2365
  selectedProvider,
2263
2366
  onSelect: handleProviderSelect,
2264
- disabled: isLoading
2367
+ disabled: isLoading,
2368
+ theme: resolvedTheme,
2369
+ selectedMethod,
2370
+ onMethodSelect: handleMethodSelect,
2371
+ renderMethodContent
2265
2372
  }
2266
- ),
2373
+ ) });
2374
+ }
2375
+ return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-method-step reevit-animate-slide-up", children: [
2267
2376
  /* @__PURE__ */ jsxRuntime.jsx(
2268
2377
  PaymentMethodSelector,
2269
2378
  {
@@ -2271,21 +2380,26 @@ function ReevitCheckout({
2271
2380
  selectedMethod,
2272
2381
  onSelect: handleMethodSelect,
2273
2382
  disabled: isLoading,
2274
- provider: activeProvider?.provider || psp
2383
+ provider: psp,
2384
+ layout: "list",
2385
+ showLabel: false
2275
2386
  }
2276
2387
  ),
2277
- selectedMethod && selectedMethod !== "mobile_money" && /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-method-step__actions", children: /* @__PURE__ */ jsxRuntime.jsx(
2278
- "button",
2279
- {
2280
- className: "reevit-btn reevit-btn--primary",
2281
- onClick: handleContinue,
2282
- disabled: isLoading,
2283
- children: "Continue"
2284
- }
2285
- ) })
2388
+ selectedMethod && /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-method-step__actions reevit-animate-slide-up", children: selectedMethod === "mobile_money" && pspLower.includes("mpesa") && !phone ? /* @__PURE__ */ jsxRuntime.jsx(MobileMoneyForm, { onSubmit: handleMomoSubmit, onCancel: () => selectMethod(null), isLoading, initialPhone: phone }) : /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-card-info reevit-animate-fade-in", children: [
2389
+ /* @__PURE__ */ jsxRuntime.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)}.` }),
2390
+ /* @__PURE__ */ jsxRuntime.jsx(
2391
+ "button",
2392
+ {
2393
+ className: "reevit-btn reevit-btn--primary",
2394
+ onClick: handleContinue,
2395
+ disabled: isLoading,
2396
+ children: selectedMethod === "card" ? "Pay with Card" : pspLower.includes("hubtel") ? "Continue with Hubtel" : "Pay with Mobile Money"
2397
+ }
2398
+ )
2399
+ ] }) })
2286
2400
  ] });
2287
2401
  };
2288
- return /* @__PURE__ */ jsxRuntime.jsxs(ReevitContext.Provider, { value: { publicKey, amount, currency }, children: [
2402
+ return /* @__PURE__ */ jsxRuntime.jsxs(ReevitContext.Provider, { value: { publicKey: publicKey || "", amount, currency }, children: [
2289
2403
  trigger,
2290
2404
  isOpen && /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-overlay", onClick: handleClose, children: /* @__PURE__ */ jsxRuntime.jsxs(
2291
2405
  "div",
@@ -2298,23 +2412,8 @@ function ReevitCheckout({
2298
2412
  "aria-modal": "true",
2299
2413
  children: [
2300
2414
  /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-modal__header", children: [
2301
- /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-modal__branding", children: /* @__PURE__ */ jsxRuntime.jsx(
2302
- "img",
2303
- {
2304
- src: "https://i.imgur.com/bzUR5Lm.png",
2305
- alt: "Reevit",
2306
- className: "reevit-modal__logo"
2307
- }
2308
- ) }),
2309
- /* @__PURE__ */ jsxRuntime.jsx(
2310
- "button",
2311
- {
2312
- className: "reevit-modal__close",
2313
- onClick: handleClose,
2314
- "aria-label": "Close",
2315
- children: "\u2715"
2316
- }
2317
- )
2415
+ /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-modal__branding", children: /* @__PURE__ */ jsxRuntime.jsx("img", { src: resolvedTheme?.logoUrl || "https://i.imgur.com/bzUR5Lm.png", alt: "Checkout", className: "reevit-modal__logo" }) }),
2416
+ /* @__PURE__ */ jsxRuntime.jsx("button", { className: "reevit-modal__close", onClick: handleClose, "aria-label": "Close", children: "\u2715" })
2318
2417
  ] }),
2319
2418
  /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-modal__amount", children: [
2320
2419
  /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-modal__amount-label", children: "Amount" }),