@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.js CHANGED
@@ -587,14 +587,6 @@ function detectCountryFromCurrency(currency) {
587
587
  };
588
588
  return currencyToCountry[currency.toUpperCase()] || "GH";
589
589
  }
590
- var pspNames = {
591
- hubtel: "Hubtel",
592
- paystack: "Paystack",
593
- flutterwave: "Flutterwave",
594
- monnify: "Monnify",
595
- mpesa: "M-Pesa",
596
- stripe: "Stripe"
597
- };
598
590
  var methodConfig = {
599
591
  card: {
600
592
  label: "Card",
@@ -617,28 +609,31 @@ function PaymentMethodSelector({
617
609
  selectedMethod,
618
610
  onSelect,
619
611
  disabled = false,
620
- provider
612
+ provider,
613
+ layout = "list",
614
+ showLabel = true
621
615
  }) {
622
616
  const getMethodLabel = (method, psp) => {
623
617
  const config = methodConfig[method];
624
- if (psp?.toLowerCase().includes("hubtel") && method === "mobile_money") {
625
- return `Pay with ${pspNames[psp.toLowerCase()] || "Hubtel"}`;
626
- }
627
618
  return config.label;
628
619
  };
629
620
  const getMethodDescription = (method, psp) => {
630
621
  const config = methodConfig[method];
631
622
  if (psp?.toLowerCase().includes("hubtel")) {
632
- return "Card, Mobile Money, and Bank Transfer";
623
+ return config.description;
633
624
  }
634
625
  return config.description;
635
626
  };
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) => {
627
+ const isGrid = layout === "grid";
628
+ return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: cn("reevit-method-selector", isGrid && "reevit-method-selector--grid"), children: [
629
+ showLabel && /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-method-selector__label", children: "Select payment method" }),
630
+ /* @__PURE__ */ jsxRuntime.jsx("div", { className: cn(
631
+ "reevit-method-selector__options",
632
+ isGrid ? "reevit-method-selector__options--grid" : "reevit-method-selector__options--list"
633
+ ), children: methods.map((method, index) => {
639
634
  const config = methodConfig[method];
640
635
  const isSelected = selectedMethod === method;
641
- const methodLabel = getMethodLabel(method, provider);
636
+ const methodLabel = getMethodLabel(method);
642
637
  const methodDescription = getMethodDescription(method, provider);
643
638
  return /* @__PURE__ */ jsxRuntime.jsxs(
644
639
  "button",
@@ -646,28 +641,24 @@ function PaymentMethodSelector({
646
641
  type: "button",
647
642
  className: cn(
648
643
  "reevit-method-option",
644
+ isGrid ? "reevit-method-option--grid" : "reevit-method-option--list",
649
645
  isSelected && "reevit-method-option--selected",
650
646
  disabled && "reevit-method-option--disabled"
651
647
  ),
648
+ style: {
649
+ animationDelay: `${index * 0.05}s`
650
+ },
652
651
  onClick: () => onSelect(method),
653
652
  disabled,
654
653
  "aria-pressed": isSelected,
655
654
  children: [
656
- /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-method-option__icon", children: config.icon }),
655
+ /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-method-option__icon-wrapper", children: /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-method-option__icon", children: config.icon }) }),
657
656
  /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-method-option__content", children: [
658
657
  /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-method-option__label", children: methodLabel }),
659
- /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-method-option__description", children: methodDescription })
658
+ !isGrid && /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-method-option__description", children: methodDescription })
660
659
  ] }),
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
- ) }) })
660
+ !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" }) }) }),
661
+ !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
662
  ]
672
663
  },
673
664
  method
@@ -684,7 +675,8 @@ function MobileMoneyForm({
684
675
  onSubmit,
685
676
  onCancel,
686
677
  isLoading = false,
687
- initialPhone = ""
678
+ initialPhone = "",
679
+ hideCancel = false
688
680
  }) {
689
681
  const [phone, setPhone] = react.useState(initialPhone);
690
682
  const [network, setNetwork] = react.useState(null);
@@ -769,7 +761,7 @@ function MobileMoneyForm({
769
761
  )) })
770
762
  ] }),
771
763
  /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-momo-form__actions", children: [
772
- /* @__PURE__ */ jsxRuntime.jsx(
764
+ !hideCancel && /* @__PURE__ */ jsxRuntime.jsx(
773
765
  "button",
774
766
  {
775
767
  type: "button",
@@ -811,11 +803,11 @@ var paystack_default = "
811
803
  // src/assets/providers/stripe.png
812
804
  var stripe_default = "";
813
805
  var providerMeta = {
814
- paystack: { logo: paystack_default, hint: "Card & MoMo" },
806
+ paystack: { logo: paystack_default, hint: "Card & Mobile Money" },
815
807
  stripe: { logo: stripe_default, hint: "Card payments" },
816
808
  flutterwave: { logo: flutterwave_default, hint: "Global methods" },
817
- hubtel: { logo: hubtel_default, hint: "Mobile money" },
818
- monnify: { logo: monnify_default, hint: "Bank & card" },
809
+ hubtel: { logo: hubtel_default, hint: "Mobile Money & Card" },
810
+ monnify: { logo: monnify_default, hint: "Bank & Card" },
819
811
  mpesa: { logo: mpesa_default, hint: "M-Pesa" }
820
812
  };
821
813
  var methodLabels = {
@@ -825,54 +817,100 @@ var methodLabels = {
825
817
  };
826
818
  function formatMethods(methods) {
827
819
  if (!methods.length) return "Payment methods";
828
- return methods.map((method) => methodLabels[method]).join(" \u2022 ");
820
+ return methods.map((method) => methodLabels[method]).join(", ");
821
+ }
822
+ function sanitizeMethods(providerId, methods) {
823
+ if (providerId.toLowerCase().includes("hubtel")) {
824
+ return methods.filter((method) => method === "card" || method === "mobile_money");
825
+ }
826
+ return methods;
829
827
  }
830
828
  function ProviderSelector({
831
829
  providers,
832
830
  selectedProvider,
833
831
  onSelect,
834
- disabled = false
832
+ disabled = false,
833
+ theme,
834
+ selectedMethod,
835
+ onMethodSelect,
836
+ renderMethodContent
835
837
  }) {
838
+ const useBorder = theme?.pspSelectorUseBorder ?? false;
839
+ const bgColor = theme?.pspSelectorBgColor || "#0a0a0a";
840
+ const textColor = theme?.pspSelectorTextColor || "#ffffff";
841
+ const borderColor = theme?.pspSelectorBorderColor || "#374151";
842
+ const getOptionStyle = (isSelected) => {
843
+ if (useBorder) {
844
+ return {
845
+ backgroundColor: "transparent",
846
+ border: `2px solid ${isSelected ? borderColor : "#374151"}`,
847
+ color: isSelected ? textColor : "var(--reevit-text)"
848
+ };
849
+ }
850
+ return {
851
+ backgroundColor: isSelected ? bgColor : "transparent",
852
+ border: `2px solid ${isSelected ? bgColor : "#374151"}`,
853
+ color: isSelected ? textColor : "var(--reevit-text)"
854
+ };
855
+ };
836
856
  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" }),
857
+ /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-psp-selector__label", children: "Select payment provider" }),
838
858
  /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-psp-selector__options", children: providers.map((provider) => {
839
859
  const meta = providerMeta[provider.provider] || {
840
860
  logo: void 0,
841
861
  hint: "Payment methods"
842
862
  };
863
+ const providerMethods = sanitizeMethods(provider.provider, provider.methods);
843
864
  const isSelected = selectedProvider === provider.provider;
844
865
  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
- );
866
+ const optionStyle = getOptionStyle(isSelected);
867
+ return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-psp-accordion", children: [
868
+ /* @__PURE__ */ jsxRuntime.jsxs(
869
+ "button",
870
+ {
871
+ type: "button",
872
+ className: cn(
873
+ "reevit-psp-option",
874
+ isSelected && "reevit-psp-option--selected",
875
+ disabled && "reevit-psp-option--disabled"
876
+ ),
877
+ style: optionStyle,
878
+ onClick: () => onSelect(provider.provider),
879
+ disabled,
880
+ "aria-expanded": isSelected,
881
+ children: [
882
+ /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-psp-option__logo", "aria-hidden": "true", children: meta.logo ? /* @__PURE__ */ jsxRuntime.jsx(
883
+ "img",
884
+ {
885
+ src: meta.logo,
886
+ alt: "",
887
+ className: "reevit-psp-option__logo-img",
888
+ loading: "lazy"
889
+ }
890
+ ) : /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-psp-option__logo-fallback", children: fallbackInitial }) }),
891
+ /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-psp-option__content", children: [
892
+ /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-psp-option__name", children: provider.name }),
893
+ /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-psp-option__methods", children: formatMethods(providerMethods) || meta.hint })
894
+ ] })
895
+ ]
896
+ }
897
+ ),
898
+ isSelected && onMethodSelect && /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-psp-accordion__content", children: [
899
+ /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-psp-methods", children: /* @__PURE__ */ jsxRuntime.jsx(
900
+ PaymentMethodSelector,
901
+ {
902
+ methods: providerMethods,
903
+ selectedMethod: selectedMethod || null,
904
+ onSelect: onMethodSelect,
905
+ disabled,
906
+ provider: provider.provider,
907
+ layout: "list",
908
+ showLabel: false
909
+ }
910
+ ) }),
911
+ selectedMethod && renderMethodContent && /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-psp-accordion__method-content", children: renderMethodContent(provider.provider, selectedMethod) })
912
+ ] })
913
+ ] }, provider.provider);
876
914
  }) })
877
915
  ] });
878
916
  }
@@ -1006,6 +1044,7 @@ function HubtelBridge({
1006
1044
  callbackUrl,
1007
1045
  hubtelSessionToken,
1008
1046
  basicAuth,
1047
+ preferredMethod,
1009
1048
  onSuccess,
1010
1049
  onError,
1011
1050
  onClose,
@@ -1056,12 +1095,14 @@ function HubtelBridge({
1056
1095
  try {
1057
1096
  const checkout = new CheckoutSdk__default.default();
1058
1097
  checkoutRef.current = checkout;
1098
+ const methodPreference = preferredMethod === "mobile_money" ? "momo" : preferredMethod === "card" ? "card" : void 0;
1059
1099
  const purchaseInfo = {
1060
1100
  amount: amount / 100,
1061
1101
  // Convert from minor to major units
1062
1102
  purchaseDescription: description,
1063
1103
  customerPhoneNumber: phone || "",
1064
- clientReference: reference || `hubtel_${Date.now()}`
1104
+ clientReference: reference || `hubtel_${Date.now()}`,
1105
+ ...methodPreference ? { paymentMethod: methodPreference } : {}
1065
1106
  };
1066
1107
  const config = {
1067
1108
  branding: "enabled",
@@ -1069,7 +1110,8 @@ function HubtelBridge({
1069
1110
  merchantAccount: typeof merchantAccount === "string" ? parseInt(merchantAccount, 10) : merchantAccount,
1070
1111
  // Use session token or basicAuth for authentication
1071
1112
  // Session tokens are base64-encoded credentials fetched securely from the server
1072
- basicAuth: authValue || ""
1113
+ basicAuth: authValue || "",
1114
+ ...methodPreference ? { paymentMethod: methodPreference } : {}
1073
1115
  };
1074
1116
  checkout.openModal({
1075
1117
  purchaseInfo,
@@ -1113,7 +1155,7 @@ function HubtelBridge({
1113
1155
  };
1114
1156
  onError(error);
1115
1157
  }
1116
- }, [merchantAccount, amount, reference, phone, description, callbackUrl, authValue, isLoading, onSuccess, onError, onClose]);
1158
+ }, [merchantAccount, amount, reference, phone, description, callbackUrl, authValue, isLoading, preferredMethod, onSuccess, onError, onClose]);
1117
1159
  react.useEffect(() => {
1118
1160
  if (autoStart && !initialized.current && !isLoading && authValue) {
1119
1161
  initialized.current = true;
@@ -1127,17 +1169,20 @@ function HubtelBridge({
1127
1169
  }
1128
1170
  function openHubtelPopup(config) {
1129
1171
  const checkout = new CheckoutSdk__default.default();
1172
+ const methodPreference = config.preferredMethod === "mobile_money" ? "momo" : config.preferredMethod === "card" ? "card" : void 0;
1130
1173
  const purchaseInfo = {
1131
1174
  amount: config.amount,
1132
1175
  purchaseDescription: config.description,
1133
1176
  customerPhoneNumber: config.customerPhoneNumber || "",
1134
- clientReference: config.clientReference || `hubtel_${Date.now()}`
1177
+ clientReference: config.clientReference || `hubtel_${Date.now()}`,
1178
+ ...methodPreference ? { paymentMethod: methodPreference } : {}
1135
1179
  };
1136
1180
  const checkoutConfig = {
1137
1181
  branding: "enabled",
1138
1182
  callbackUrl: config.callbackUrl || window.location.href,
1139
1183
  merchantAccount: typeof config.merchantAccount === "string" ? parseInt(config.merchantAccount, 10) : config.merchantAccount,
1140
- basicAuth: config.basicAuth || ""
1184
+ basicAuth: config.basicAuth || "",
1185
+ ...methodPreference ? { paymentMethod: methodPreference } : {}
1141
1186
  };
1142
1187
  checkout.openModal({
1143
1188
  purchaseInfo,
@@ -1830,6 +1875,14 @@ function useReevitContext() {
1830
1875
  }
1831
1876
  return context;
1832
1877
  }
1878
+ var pspNames = {
1879
+ hubtel: "Hubtel",
1880
+ paystack: "Paystack",
1881
+ flutterwave: "Flutterwave",
1882
+ monnify: "Monnify",
1883
+ mpesa: "M-Pesa",
1884
+ stripe: "Stripe"
1885
+ };
1833
1886
  function ReevitCheckout({
1834
1887
  // Config
1835
1888
  publicKey,
@@ -1918,7 +1971,8 @@ function ReevitCheckout({
1918
1971
  const options = available.length > 0 ? available : fallbackProvider;
1919
1972
  return options.map((provider) => {
1920
1973
  const methods = provider.methods && provider.methods.length > 0 ? provider.methods : paymentMethods;
1921
- const filteredMethods = methods.filter((method) => paymentMethods.includes(method));
1974
+ const sanitizedMethods = provider.provider.toLowerCase().includes("hubtel") ? methods.filter((method) => method === "card" || method === "mobile_money") : methods;
1975
+ const filteredMethods = sanitizedMethods.filter((method) => paymentMethods.includes(method));
1922
1976
  return {
1923
1977
  ...provider,
1924
1978
  methods: filteredMethods
@@ -1939,10 +1993,10 @@ function ReevitCheckout({
1939
1993
  if (selectedProvider && providerOptions.some((provider) => provider.provider === selectedProvider)) {
1940
1994
  return;
1941
1995
  }
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]);
1996
+ if (providerOptions.length === 1) {
1997
+ setSelectedProvider(providerOptions[0].provider);
1998
+ }
1999
+ }, [providerOptions, selectedProvider]);
1946
2000
  react.useEffect(() => {
1947
2001
  if (!activeProvider || !selectedMethod) {
1948
2002
  return;
@@ -1953,13 +2007,17 @@ function ReevitCheckout({
1953
2007
  }, [activeProvider, selectedMethod, selectMethod]);
1954
2008
  react.useEffect(() => {
1955
2009
  if (isOpen && selectedMethod && paymentIntent && !showPSPBridge) {
2010
+ const psp = (selectedProvider || paymentIntent.recommendedPsp || "paystack").toLowerCase();
2011
+ const needsPhone = psp.includes("mpesa");
1956
2012
  if (selectedMethod === "card") {
1957
2013
  setShowPSPBridge(true);
1958
- } else if (selectedMethod === "mobile_money" && (momoData?.phone || phone)) {
1959
- setShowPSPBridge(true);
2014
+ } else if (selectedMethod === "mobile_money") {
2015
+ if (!needsPhone || (momoData?.phone || phone)) {
2016
+ setShowPSPBridge(true);
2017
+ }
1960
2018
  }
1961
2019
  }
1962
- }, [isOpen, selectedMethod, showPSPBridge, paymentIntent, momoData, phone]);
2020
+ }, [isOpen, selectedMethod, showPSPBridge, paymentIntent, momoData, phone, selectedProvider]);
1963
2021
  const handleOpen = react.useCallback(() => {
1964
2022
  if (controlledIsOpen !== void 0) return;
1965
2023
  setIsOpen(true);
@@ -1983,6 +2041,10 @@ function ReevitCheckout({
1983
2041
  const handleProviderSelect = react.useCallback(
1984
2042
  (provider) => {
1985
2043
  if (provider === selectedProvider) {
2044
+ setSelectedProvider(null);
2045
+ reset();
2046
+ setShowPSPBridge(false);
2047
+ setMomoData(null);
1986
2048
  return;
1987
2049
  }
1988
2050
  const providerMethods = providerOptions.find((option) => option.provider === provider)?.methods || paymentMethods;
@@ -1999,8 +2061,14 @@ function ReevitCheckout({
1999
2061
  if (!selectedMethod) return;
2000
2062
  if (selectedMethod === "card") {
2001
2063
  setShowPSPBridge(true);
2064
+ } else if (selectedMethod === "mobile_money") {
2065
+ const psp = (selectedProvider || paymentIntent?.recommendedPsp || "paystack").toLowerCase();
2066
+ const needsPhone = psp.includes("mpesa");
2067
+ if (!needsPhone || (momoData?.phone || phone)) {
2068
+ setShowPSPBridge(true);
2069
+ }
2002
2070
  }
2003
- }, [selectedMethod]);
2071
+ }, [selectedMethod, selectedProvider, paymentIntent, momoData, phone]);
2004
2072
  const handleMomoSubmit = react.useCallback(
2005
2073
  (data) => {
2006
2074
  setMomoData(data);
@@ -2057,13 +2125,13 @@ function ReevitCheckout({
2057
2125
  ] }) : null;
2058
2126
  const renderContent = () => {
2059
2127
  if (status === "loading" || status === "processing") {
2060
- return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-loading", children: [
2128
+ return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-loading reevit-animate-fade-in", children: [
2061
2129
  /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-spinner" }),
2062
2130
  /* @__PURE__ */ jsxRuntime.jsx("p", { children: status === "loading" ? "Preparing checkout..." : "Processing payment..." })
2063
2131
  ] });
2064
2132
  }
2065
2133
  if (status === "success" && result) {
2066
- return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-success", children: [
2134
+ return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-success reevit-animate-scale-in", children: [
2067
2135
  /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-success__icon", children: "\u2713" }),
2068
2136
  /* @__PURE__ */ jsxRuntime.jsx("h3", { children: "Payment Successful" }),
2069
2137
  /* @__PURE__ */ jsxRuntime.jsxs("p", { children: [
@@ -2073,7 +2141,7 @@ function ReevitCheckout({
2073
2141
  ] });
2074
2142
  }
2075
2143
  if (status === "failed" && error && !error.recoverable) {
2076
- return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-error", children: [
2144
+ return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-error reevit-animate-fade-in", children: [
2077
2145
  /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-error__icon", children: "\u2715" }),
2078
2146
  /* @__PURE__ */ jsxRuntime.jsx("h3", { children: "Payment Failed" }),
2079
2147
  /* @__PURE__ */ jsxRuntime.jsx("p", { children: error.message }),
@@ -2081,17 +2149,16 @@ function ReevitCheckout({
2081
2149
  ] });
2082
2150
  }
2083
2151
  const psp = selectedProvider || paymentIntent?.recommendedPsp || "paystack";
2152
+ const pspLower = psp.toLowerCase();
2084
2153
  if (showPSPBridge) {
2085
2154
  const pspKey = paymentIntent?.pspPublicKey || publicKey;
2086
2155
  const bridgeMetadata = {
2087
2156
  ...metadata,
2088
- // Override with correct payment intent ID for webhook routing
2089
- // This ensures webhook includes the correct ID to find the payment
2090
2157
  payment_id: paymentIntent?.id,
2091
2158
  connection_id: paymentIntent?.connectionId ?? metadata?.connection_id,
2092
2159
  customer_phone: momoData?.phone || phone
2093
2160
  };
2094
- switch (psp) {
2161
+ switch (pspLower) {
2095
2162
  case "paystack":
2096
2163
  return /* @__PURE__ */ jsxRuntime.jsx(
2097
2164
  PaystackBridge,
@@ -2124,6 +2191,7 @@ function ReevitCheckout({
2124
2191
  phone: momoData?.phone || phone,
2125
2192
  description: `Payment ${paymentIntent?.reference || reference || ""}`,
2126
2193
  hubtelSessionToken: paymentIntent?.id ? paymentIntent.id : void 0,
2194
+ preferredMethod: selectedMethod || void 0,
2127
2195
  onSuccess: handlePSPSuccess,
2128
2196
  onError: (err) => handlePSPError(err),
2129
2197
  onClose: handlePSPClose
@@ -2141,7 +2209,7 @@ function ReevitCheckout({
2141
2209
  phone: momoData?.phone || phone,
2142
2210
  metadata: bridgeMetadata,
2143
2211
  onSuccess: handlePSPSuccess,
2144
- onError: (err) => handlePSPError(err),
2212
+ onError: handlePSPError,
2145
2213
  onClose: handlePSPClose
2146
2214
  }
2147
2215
  );
@@ -2158,21 +2226,8 @@ function ReevitCheckout({
2158
2226
  customerEmail: email,
2159
2227
  customerPhone: momoData?.phone || phone,
2160
2228
  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
- }),
2229
+ onSuccess: (res) => handlePSPSuccess({ ...res, psp: "monnify" }),
2230
+ onError: handlePSPError,
2176
2231
  onClose: handlePSPClose
2177
2232
  }
2178
2233
  );
@@ -2187,21 +2242,8 @@ function ReevitCheckout({
2187
2242
  reference: paymentIntent?.reference || reference || `mpesa_${Date.now()}`,
2188
2243
  description: `Payment ${paymentIntent?.reference || reference || ""}`,
2189
2244
  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
- })
2245
+ onSuccess: handlePSPSuccess,
2246
+ onError: handlePSPError
2205
2247
  }
2206
2248
  );
2207
2249
  case "stripe":
@@ -2212,58 +2254,60 @@ function ReevitCheckout({
2212
2254
  clientSecret: paymentIntent?.clientSecret || "",
2213
2255
  amount: paymentIntent?.amount ?? amount,
2214
2256
  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
- }),
2257
+ onSuccess: handlePSPSuccess,
2258
+ onError: handlePSPError,
2230
2259
  onCancel: handlePSPClose
2231
2260
  }
2232
2261
  );
2233
2262
  default:
2234
2263
  return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-error", children: [
2235
2264
  /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-error__icon", children: "\u26A0\uFE0F" }),
2236
- /* @__PURE__ */ jsxRuntime.jsx("h3", { children: "Payment Provider Not Supported" }),
2265
+ /* @__PURE__ */ jsxRuntime.jsx("h3", { children: "Provider Not Supported" }),
2237
2266
  /* @__PURE__ */ jsxRuntime.jsxs("p", { children: [
2238
- "The selected payment provider (",
2267
+ "Provider (",
2239
2268
  psp,
2240
- ") is not currently supported in this checkout."
2269
+ ") is not supported."
2241
2270
  ] }),
2242
2271
  /* @__PURE__ */ jsxRuntime.jsx("button", { className: "reevit-btn reevit-btn--primary", onClick: handleBack, children: "Go Back" })
2243
2272
  ] });
2244
2273
  }
2245
2274
  }
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
2275
+ const renderMethodContent = (provider, method) => {
2276
+ const pspLower2 = provider.toLowerCase();
2277
+ const needsPhone = pspLower2.includes("mpesa");
2278
+ if (method === "card") {
2279
+ return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-inline-action reevit-animate-fade-in", children: [
2280
+ /* @__PURE__ */ jsxRuntime.jsx("p", { className: "reevit-inline-action__hint", children: "You'll be redirected to complete your card payment securely." }),
2281
+ /* @__PURE__ */ jsxRuntime.jsx("button", { className: "reevit-btn reevit-btn--primary", onClick: handleContinue, disabled: isLoading, children: "Pay with Card" })
2282
+ ] });
2283
+ }
2284
+ if (method === "mobile_money") {
2285
+ if (needsPhone && !phone) {
2286
+ 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
2287
  }
2255
- );
2256
- }
2257
- return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-method-step", children: [
2258
- providerOptions.length > 1 && /* @__PURE__ */ jsxRuntime.jsx(
2288
+ return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-inline-action reevit-animate-fade-in", children: [
2289
+ /* @__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)}.` }),
2290
+ /* @__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" })
2291
+ ] });
2292
+ }
2293
+ return null;
2294
+ };
2295
+ if (providerOptions.length > 1) {
2296
+ return /* @__PURE__ */ jsxRuntime.jsx("div", { className: "reevit-method-step reevit-animate-slide-up", children: /* @__PURE__ */ jsxRuntime.jsx(
2259
2297
  ProviderSelector,
2260
2298
  {
2261
2299
  providers: providerOptions,
2262
2300
  selectedProvider,
2263
2301
  onSelect: handleProviderSelect,
2264
- disabled: isLoading
2302
+ disabled: isLoading,
2303
+ theme: resolvedTheme,
2304
+ selectedMethod,
2305
+ onMethodSelect: handleMethodSelect,
2306
+ renderMethodContent
2265
2307
  }
2266
- ),
2308
+ ) });
2309
+ }
2310
+ return /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-method-step reevit-animate-slide-up", children: [
2267
2311
  /* @__PURE__ */ jsxRuntime.jsx(
2268
2312
  PaymentMethodSelector,
2269
2313
  {
@@ -2271,18 +2315,23 @@ function ReevitCheckout({
2271
2315
  selectedMethod,
2272
2316
  onSelect: handleMethodSelect,
2273
2317
  disabled: isLoading,
2274
- provider: activeProvider?.provider || psp
2318
+ provider: psp,
2319
+ layout: "list",
2320
+ showLabel: false
2275
2321
  }
2276
2322
  ),
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
- ) })
2323
+ 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: [
2324
+ /* @__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)}.` }),
2325
+ /* @__PURE__ */ jsxRuntime.jsx(
2326
+ "button",
2327
+ {
2328
+ className: "reevit-btn reevit-btn--primary",
2329
+ onClick: handleContinue,
2330
+ disabled: isLoading,
2331
+ children: selectedMethod === "card" ? "Pay with Card" : pspLower.includes("hubtel") ? "Continue with Hubtel" : "Pay with Mobile Money"
2332
+ }
2333
+ )
2334
+ ] }) })
2286
2335
  ] });
2287
2336
  };
2288
2337
  return /* @__PURE__ */ jsxRuntime.jsxs(ReevitContext.Provider, { value: { publicKey, amount, currency }, children: [
@@ -2298,23 +2347,8 @@ function ReevitCheckout({
2298
2347
  "aria-modal": "true",
2299
2348
  children: [
2300
2349
  /* @__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
- )
2350
+ /* @__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" }) }),
2351
+ /* @__PURE__ */ jsxRuntime.jsx("button", { className: "reevit-modal__close", onClick: handleClose, "aria-label": "Close", children: "\u2715" })
2318
2352
  ] }),
2319
2353
  /* @__PURE__ */ jsxRuntime.jsxs("div", { className: "reevit-modal__amount", children: [
2320
2354
  /* @__PURE__ */ jsxRuntime.jsx("span", { className: "reevit-modal__amount-label", children: "Amount" }),