@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.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
- return this.request("POST", `/v1/payments/hubtel/sessions/${paymentId}`);
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
- const { data, error } = await apiClient.createPaymentIntent(
412
- { ...config, reference },
413
- paymentMethod,
414
- country,
415
- {
416
- preferredProviders: options2?.preferredProvider ? [options2.preferredProvider] : void 0,
417
- allowedProviders: options2?.allowedProviders
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 "Card, Mobile Money, and Bank Transfer";
665
+ return config.description;
627
666
  }
628
667
  return config.description;
629
668
  };
630
- return /* @__PURE__ */ jsxs("div", { className: "reevit-method-selector", children: [
631
- /* @__PURE__ */ jsx("div", { className: "reevit-method-selector__label", children: "Select payment method" }),
632
- /* @__PURE__ */ jsx("div", { className: "reevit-method-selector__options", children: methods.map((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, provider);
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 20 20", fill: "none", children: /* @__PURE__ */ jsx(
656
- "path",
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 & MoMo" },
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 money" },
812
- monnify: { logo: monnify_default, hint: "Bank & card" },
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(" \u2022 ");
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: "Choose your payment provider" }),
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
- return /* @__PURE__ */ jsxs(
840
- "button",
841
- {
842
- type: "button",
843
- className: cn(
844
- "reevit-psp-option",
845
- isSelected && "reevit-psp-option--selected",
846
- disabled && "reevit-psp-option--disabled"
847
- ),
848
- onClick: () => onSelect(provider.provider),
849
- disabled,
850
- "aria-pressed": isSelected,
851
- children: [
852
- /* @__PURE__ */ jsx("span", { className: "reevit-psp-option__logo", "aria-hidden": "true", children: meta.logo ? /* @__PURE__ */ jsx(
853
- "img",
854
- {
855
- src: meta.logo,
856
- alt: "",
857
- className: "reevit-psp-option__logo-img",
858
- loading: "lazy"
859
- }
860
- ) : /* @__PURE__ */ jsx("span", { className: "reevit-psp-option__logo-fallback", children: fallbackInitial }) }),
861
- /* @__PURE__ */ jsxs("div", { className: "reevit-psp-option__content", children: [
862
- /* @__PURE__ */ jsx("span", { className: "reevit-psp-option__name", children: provider.name }),
863
- /* @__PURE__ */ jsx("span", { className: "reevit-psp-option__methods", children: formatMethods(provider.methods) || meta.hint })
864
- ] }),
865
- /* @__PURE__ */ jsx("span", { className: "reevit-psp-option__check", "aria-hidden": "true", children: isSelected ? "Selected" : "Select" })
866
- ]
867
- },
868
- provider.provider
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 merchantAccount === "string" ? parseInt(merchantAccount, 10) : merchantAccount,
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 filteredMethods = methods.filter((method) => paymentMethods.includes(method));
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
- const recommended = paymentIntent?.recommendedPsp;
1937
- const resolved = providerOptions.find((provider) => provider.provider === recommended)?.provider || providerOptions[0].provider;
1938
- setSelectedProvider(resolved);
1939
- }, [paymentIntent?.recommendedPsp, providerOptions, selectedProvider]);
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" && (momoData?.phone || phone)) {
1953
- setShowPSPBridge(true);
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 (psp) {
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: (err) => handlePSPError(err),
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: (result2) => handlePSPSuccess({
2156
- paymentId: result2.transactionReference,
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: (result2) => handlePSPSuccess({
2185
- paymentId: result2.transactionId,
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: (result2) => handlePSPSuccess({
2210
- paymentId: result2.paymentIntentId,
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: "Payment Provider Not Supported" }),
2324
+ /* @__PURE__ */ jsx("h3", { children: "Provider Not Supported" }),
2231
2325
  /* @__PURE__ */ jsxs("p", { children: [
2232
- "The selected payment provider (",
2326
+ "Provider (",
2233
2327
  psp,
2234
- ") is not currently supported in this checkout."
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
- if (selectedMethod === "mobile_money" && !showPSPBridge) {
2241
- return /* @__PURE__ */ jsx(
2242
- MobileMoneyForm,
2243
- {
2244
- onSubmit: handleMomoSubmit,
2245
- onCancel: handleBack,
2246
- isLoading,
2247
- initialPhone: phone
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
- return /* @__PURE__ */ jsxs("div", { className: "reevit-method-step", children: [
2252
- providerOptions.length > 1 && /* @__PURE__ */ jsx(
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: activeProvider?.provider || psp
2377
+ provider: psp,
2378
+ layout: "list",
2379
+ showLabel: false
2269
2380
  }
2270
2381
  ),
2271
- selectedMethod && selectedMethod !== "mobile_money" && /* @__PURE__ */ jsx("div", { className: "reevit-method-step__actions", children: /* @__PURE__ */ jsx(
2272
- "button",
2273
- {
2274
- className: "reevit-btn reevit-btn--primary",
2275
- onClick: handleContinue,
2276
- disabled: isLoading,
2277
- children: "Continue"
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
- "img",
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" }),