@reevit/react 0.4.2 → 0.4.3

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