@akinon/next 1.13.1 → 1.14.0

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.
Files changed (133) hide show
  1. package/.editorconfig +7 -0
  2. package/.eslintrc.js +40 -40
  3. package/.prettierrc +13 -13
  4. package/CHANGELOG.md +13 -0
  5. package/api/auth.ts +231 -231
  6. package/api/cache.ts +44 -44
  7. package/api/client.ts +174 -174
  8. package/api/logout.ts +42 -42
  9. package/assets/styles/index.scss +28 -28
  10. package/bin/pz-check-dependencies.js +98 -98
  11. package/bin/pz-install-plugins.js +33 -33
  12. package/bin/pz-install-theme.js +58 -58
  13. package/bin/pz-postbuild.js +1 -1
  14. package/bin/pz-postdev.js +1 -1
  15. package/bin/pz-postinstall.js +6 -6
  16. package/bin/pz-poststart.js +1 -1
  17. package/bin/pz-prebuild.js +4 -4
  18. package/bin/pz-predev.js +4 -4
  19. package/bin/pz-prestart.js +1 -1
  20. package/bin/run-script.js +44 -44
  21. package/components/accordion.tsx +52 -0
  22. package/components/button.tsx +46 -0
  23. package/components/client-root.tsx +19 -19
  24. package/components/icon.tsx +18 -18
  25. package/components/image.tsx +133 -133
  26. package/components/index.ts +17 -1
  27. package/components/input.tsx +110 -0
  28. package/components/lazy-component.tsx +33 -33
  29. package/components/loader-spinner.tsx +23 -23
  30. package/components/mobile-app-toggler.tsx +26 -26
  31. package/components/oauth-login.tsx +24 -24
  32. package/components/plugin-module.tsx +11 -8
  33. package/components/price.tsx +55 -0
  34. package/components/pz-providers.tsx +24 -24
  35. package/components/pz-root.tsx +21 -21
  36. package/components/radio.tsx +18 -18
  37. package/components/react-portal.tsx +45 -45
  38. package/components/redirect-three-d/content/index.tsx +74 -74
  39. package/components/redirect-three-d/index.tsx +17 -17
  40. package/components/selected-payment-option-view.tsx +7 -0
  41. package/components/trans.tsx +39 -39
  42. package/data/client/account.ts +208 -208
  43. package/data/client/address.ts +107 -107
  44. package/data/client/api.ts +85 -84
  45. package/data/client/b2b.ts +106 -106
  46. package/data/client/basket.ts +82 -82
  47. package/data/client/checkout.ts +516 -479
  48. package/data/client/misc.ts +101 -101
  49. package/data/client/product.ts +89 -89
  50. package/data/client/user.ts +99 -99
  51. package/data/client/wishlist.ts +118 -88
  52. package/data/server/category.ts +132 -128
  53. package/data/server/flatpage.ts +21 -21
  54. package/data/server/form.ts +22 -22
  55. package/data/server/index.ts +10 -10
  56. package/data/server/landingpage.ts +24 -24
  57. package/data/server/list.ts +67 -62
  58. package/data/server/menu.ts +35 -35
  59. package/data/server/product.ts +86 -86
  60. package/data/server/seo.ts +48 -48
  61. package/data/server/special-page.ts +47 -47
  62. package/data/server/widget.ts +27 -27
  63. package/data/urls.ts +221 -210
  64. package/hocs/client/index.ts +1 -1
  65. package/hocs/client/with-segment-defaults.tsx +25 -25
  66. package/hocs/server/index.ts +1 -1
  67. package/hocs/server/with-segment-defaults.tsx +85 -85
  68. package/hooks/index.ts +10 -10
  69. package/hooks/use-captcha.tsx +76 -76
  70. package/hooks/use-common-product-attributes.ts +36 -36
  71. package/hooks/use-debounce.ts +20 -20
  72. package/hooks/use-localization.ts +78 -78
  73. package/hooks/use-media-query.ts +36 -36
  74. package/hooks/use-mobile-iframe-handler.ts +23 -23
  75. package/hooks/use-on-click-outside.tsx +28 -28
  76. package/hooks/use-payment-options.ts +2 -1
  77. package/hooks/use-router.ts +45 -45
  78. package/hooks/use-translation.ts +14 -14
  79. package/lib/cache.ts +215 -215
  80. package/localization/index.ts +5 -5
  81. package/localization/provider.tsx +58 -58
  82. package/middlewares/currency.ts +100 -100
  83. package/middlewares/default.ts +256 -256
  84. package/middlewares/index.ts +29 -29
  85. package/middlewares/locale.ts +68 -68
  86. package/middlewares/oauth-login.ts +79 -79
  87. package/middlewares/pretty-url.ts +104 -104
  88. package/middlewares/redirection-payment.ts +160 -160
  89. package/middlewares/three-d-redirection.ts +159 -159
  90. package/middlewares/url-redirection.ts +65 -65
  91. package/package.json +2 -2
  92. package/plugins.js +2 -1
  93. package/redux/hooks.ts +7 -7
  94. package/redux/middlewares/checkout.ts +265 -260
  95. package/redux/middlewares/index.ts +50 -50
  96. package/redux/reducers/checkout.ts +184 -171
  97. package/redux/reducers/config.ts +28 -28
  98. package/redux/reducers/header.ts +59 -59
  99. package/redux/reducers/index.ts +14 -14
  100. package/redux/reducers/root.ts +61 -61
  101. package/sentry/index.ts +27 -27
  102. package/tailwind/rtl.js +137 -137
  103. package/types/commerce/account.ts +64 -64
  104. package/types/commerce/address.ts +94 -94
  105. package/types/commerce/b2b.ts +117 -117
  106. package/types/commerce/basket.ts +43 -43
  107. package/types/commerce/category.ts +114 -114
  108. package/types/commerce/checkout.ts +143 -136
  109. package/types/commerce/flatpage.ts +7 -7
  110. package/types/commerce/form.ts +66 -66
  111. package/types/commerce/index.ts +12 -12
  112. package/types/commerce/landingpage.ts +7 -7
  113. package/types/commerce/misc.ts +127 -127
  114. package/types/commerce/order.ts +119 -119
  115. package/types/commerce/product.ts +109 -109
  116. package/types/commerce/widget.ts +28 -28
  117. package/types/gtm.ts +16 -16
  118. package/types/index.ts +274 -237
  119. package/types/metadata.ts +7 -7
  120. package/types/next-auth.d.ts +24 -24
  121. package/utils/app-fetch.ts +69 -69
  122. package/utils/deep-merge.js +24 -24
  123. package/utils/generate-commerce-search-params.ts +22 -22
  124. package/utils/get-currency.ts +29 -29
  125. package/utils/image-loader.ts +31 -31
  126. package/utils/index.ts +150 -150
  127. package/utils/localization.ts +29 -29
  128. package/utils/log.ts +138 -138
  129. package/utils/menu-generator.ts +27 -27
  130. package/utils/mobile-3d-iframe.ts +77 -77
  131. package/utils/server-translation.ts +57 -57
  132. package/utils/server-variables.ts +9 -9
  133. package/with-pz-config.js +94 -94
@@ -11,7 +11,8 @@ enum Plugin {
11
11
  CheckoutGiftPack = 'pz-checkout-gift-pack',
12
12
  GPay = 'pz-gpay',
13
13
  Otp = 'pz-otp',
14
- BKMExpress = 'pz-bkm'
14
+ BKMExpress = 'pz-bkm',
15
+ Masterpass = 'pz-masterpass'
15
16
  }
16
17
 
17
18
  export enum Component {
@@ -22,7 +23,8 @@ export enum Component {
22
23
  CheckoutGiftPack = 'CheckoutGiftPack',
23
24
  GPay = 'GPayOption',
24
25
  Otp = 'Otp',
25
- BKMExpress = 'BKMOption'
26
+ BKMExpress = 'BKMOption',
27
+ MasterpassCardList = 'MasterpassCardList'
26
28
  }
27
29
 
28
30
  const PluginComponents = new Map([
@@ -33,7 +35,8 @@ const PluginComponents = new Map([
33
35
  [Plugin.CheckoutGiftPack, [Component.CheckoutGiftPack]],
34
36
  [Plugin.GPay, [Component.GPay]],
35
37
  [Plugin.Otp, [Component.Otp]],
36
- [Plugin.BKMExpress, [Component.BKMExpress]]
38
+ [Plugin.BKMExpress, [Component.BKMExpress]],
39
+ [Plugin.Masterpass, [Component.MasterpassCardList]]
37
40
  ]);
38
41
 
39
42
  const getPlugin = (component: Component) => {
@@ -53,10 +56,6 @@ export default function PluginModule({
53
56
  }) {
54
57
  const plugin = getPlugin(component);
55
58
 
56
- if (!(plugins as string[]).includes(plugin)) {
57
- return null;
58
- }
59
-
60
59
  const Component = useMemo(
61
60
  () =>
62
61
  dynamic(
@@ -91,8 +90,12 @@ export default function PluginModule({
91
90
  },
92
91
  { ssr: false }
93
92
  ),
94
- [plugin]
93
+ [plugin, component]
95
94
  );
96
95
 
96
+ if (!(plugins as string[]).includes(plugin)) {
97
+ return null;
98
+ }
99
+
97
100
  return <Component {...props} />;
98
101
  }
@@ -0,0 +1,55 @@
1
+ import { useMemo } from 'react';
2
+ import NumberFormat, { NumberFormatProps } from 'react-number-format';
3
+ import { getCurrency } from '@akinon/next/utils';
4
+
5
+ import { useLocalization } from '@akinon/next/hooks';
6
+ import { PriceProps } from '../types';
7
+
8
+ export const Price = (props: NumberFormatProps & PriceProps) => {
9
+ const {
10
+ value,
11
+ currencyCode,
12
+ displayType = 'text',
13
+ useCurrencySymbol = false,
14
+ useCurrencyAfterPrice = true,
15
+ useCurrencySpace = true,
16
+ useNegative = false,
17
+ useNegativeSpace = true,
18
+ thousandSeparator = '.',
19
+ decimalScale = 2,
20
+ decimalSeparator = ',',
21
+ fixedDecimalScale = true,
22
+ ...rest
23
+ } = props;
24
+ const { currency: selectedCurrencyCode } = useLocalization();
25
+ const currencyCode_ = currencyCode || selectedCurrencyCode;
26
+
27
+ // TODO: This is very bad practice. It broke decimalScale.
28
+ const _value = value?.toString().replace('.', ',');
29
+
30
+ const currency = useMemo(
31
+ () =>
32
+ getCurrency({
33
+ currencyCode: currencyCode_,
34
+ useCurrencySymbol,
35
+ useCurrencyAfterPrice,
36
+ useCurrencySpace
37
+ }),
38
+ [currencyCode_, useCurrencySymbol, useCurrencyAfterPrice, useCurrencySpace]
39
+ );
40
+
41
+ return (
42
+ <NumberFormat
43
+ value={useNegative ? `-${useNegativeSpace}${_value}` : _value}
44
+ {...{
45
+ [useCurrencyAfterPrice ? 'suffix' : 'prefix']: currency
46
+ }}
47
+ displayType={displayType}
48
+ thousandSeparator={thousandSeparator}
49
+ decimalScale={decimalScale}
50
+ decimalSeparator={decimalSeparator}
51
+ fixedDecimalScale={fixedDecimalScale}
52
+ {...rest}
53
+ />
54
+ );
55
+ };
@@ -1,24 +1,24 @@
1
- 'use client';
2
-
3
- import LocalizationProvider from '@akinon/next/localization/provider';
4
- import { SessionProvider } from 'next-auth/react';
5
- import { Provider } from 'react-redux';
6
- import { store } from 'redux/store';
7
-
8
- export default function PzProviders({
9
- translations,
10
- children
11
- }: {
12
- translations: any;
13
- children: React.ReactNode;
14
- }) {
15
- return (
16
- <Provider store={store}>
17
- <SessionProvider refetchOnWindowFocus={false}>
18
- <LocalizationProvider translations={translations}>
19
- {children}
20
- </LocalizationProvider>
21
- </SessionProvider>
22
- </Provider>
23
- );
24
- }
1
+ 'use client';
2
+
3
+ import LocalizationProvider from '@akinon/next/localization/provider';
4
+ import { SessionProvider } from 'next-auth/react';
5
+ import { Provider } from 'react-redux';
6
+ import { store } from 'redux/store';
7
+
8
+ export default function PzProviders({
9
+ translations,
10
+ children
11
+ }: {
12
+ translations: any;
13
+ children: React.ReactNode;
14
+ }) {
15
+ return (
16
+ <Provider store={store}>
17
+ <SessionProvider refetchOnWindowFocus={false}>
18
+ <LocalizationProvider translations={translations}>
19
+ {children}
20
+ </LocalizationProvider>
21
+ </SessionProvider>
22
+ </Provider>
23
+ );
24
+ }
@@ -1,21 +1,21 @@
1
- import 'server-only';
2
- import ClientRoot from './client-root';
3
- import { cookies } from 'next/headers';
4
- import PzProviders from './pz-providers';
5
-
6
- export default function PzRoot({
7
- translations,
8
- children
9
- }: {
10
- translations: any;
11
- children: React.ReactNode;
12
- }) {
13
- const nextCookies = cookies();
14
- const sessionid = nextCookies.get('osessionid')?.value;
15
-
16
- return (
17
- <PzProviders translations={translations}>
18
- <ClientRoot sessionId={sessionid}>{children}</ClientRoot>
19
- </PzProviders>
20
- );
21
- }
1
+ import 'server-only';
2
+ import ClientRoot from './client-root';
3
+ import { cookies } from 'next/headers';
4
+ import PzProviders from './pz-providers';
5
+
6
+ export default function PzRoot({
7
+ translations,
8
+ children
9
+ }: {
10
+ translations: any;
11
+ children: React.ReactNode;
12
+ }) {
13
+ const nextCookies = cookies();
14
+ const sessionid = nextCookies.get('osessionid')?.value;
15
+
16
+ return (
17
+ <PzProviders translations={translations}>
18
+ <ClientRoot sessionId={sessionid}>{children}</ClientRoot>
19
+ </PzProviders>
20
+ );
21
+ }
@@ -1,18 +1,18 @@
1
- import { forwardRef } from 'react';
2
- import { RadioProps } from '../types/index';
3
- import { twMerge } from 'tailwind-merge';
4
-
5
- const Radio = forwardRef<HTMLInputElement, RadioProps>((props, ref) => {
6
- const { children, ...rest } = props;
7
-
8
- return (
9
- <label className={twMerge('flex items-center text-xs', props.className)}>
10
- <input type="radio" {...rest} ref={ref} className="w-4 h-4" />
11
- {children && <span className="text-xs ml-2">{children}</span>}
12
- </label>
13
- );
14
- });
15
-
16
- Radio.displayName = 'Radio';
17
-
18
- export { Radio };
1
+ import { forwardRef } from 'react';
2
+ import { RadioProps } from '../types/index';
3
+ import { twMerge } from 'tailwind-merge';
4
+
5
+ const Radio = forwardRef<HTMLInputElement, RadioProps>((props, ref) => {
6
+ const { children, ...rest } = props;
7
+
8
+ return (
9
+ <label className={twMerge('flex items-center text-xs', props.className)}>
10
+ <input type="radio" {...rest} ref={ref} className="w-4 h-4" />
11
+ {children && <span className="text-xs ml-2">{children}</span>}
12
+ </label>
13
+ );
14
+ });
15
+
16
+ Radio.displayName = 'Radio';
17
+
18
+ export { Radio };
@@ -1,45 +1,45 @@
1
- import { useState, useEffect } from 'react';
2
- import { createPortal } from 'react-dom';
3
-
4
- function createWrapperAndAppendToBody(wrapperId: string) {
5
- const wrapperElement = document.createElement('div');
6
- wrapperElement.setAttribute('id', wrapperId);
7
- document.body.appendChild(wrapperElement);
8
- return wrapperElement;
9
- }
10
-
11
- type Props = {
12
- children: React.ReactNode;
13
- wrapperId: string;
14
- };
15
-
16
- export const ReactPortal: React.FC<Props> = ({
17
- children,
18
- wrapperId = 'react-portal-wrapper'
19
- }) => {
20
- const [wrapperElement, setWrapperElement] = useState<HTMLElement | null>(
21
- null
22
- );
23
-
24
- useEffect(() => {
25
- let element = document.getElementById(wrapperId) as HTMLElement;
26
- let modalCreated = false;
27
-
28
- if (!element) {
29
- modalCreated = true;
30
- element = createWrapperAndAppendToBody(wrapperId);
31
- }
32
- setWrapperElement(element);
33
-
34
- return () => {
35
- // delete the programatically created element if it was created
36
- if (modalCreated && element.parentNode) {
37
- element.parentNode.removeChild(element);
38
- }
39
- };
40
- }, [wrapperId]);
41
-
42
- if (wrapperElement === null) return null;
43
-
44
- return createPortal(children, wrapperElement);
45
- };
1
+ import { useState, useEffect } from 'react';
2
+ import { createPortal } from 'react-dom';
3
+
4
+ function createWrapperAndAppendToBody(wrapperId: string) {
5
+ const wrapperElement = document.createElement('div');
6
+ wrapperElement.setAttribute('id', wrapperId);
7
+ document.body.appendChild(wrapperElement);
8
+ return wrapperElement;
9
+ }
10
+
11
+ type Props = {
12
+ children: React.ReactNode;
13
+ wrapperId: string;
14
+ };
15
+
16
+ export const ReactPortal: React.FC<Props> = ({
17
+ children,
18
+ wrapperId = 'react-portal-wrapper'
19
+ }) => {
20
+ const [wrapperElement, setWrapperElement] = useState<HTMLElement | null>(
21
+ null
22
+ );
23
+
24
+ useEffect(() => {
25
+ let element = document.getElementById(wrapperId) as HTMLElement;
26
+ let modalCreated = false;
27
+
28
+ if (!element) {
29
+ modalCreated = true;
30
+ element = createWrapperAndAppendToBody(wrapperId);
31
+ }
32
+ setWrapperElement(element);
33
+
34
+ return () => {
35
+ // delete the programatically created element if it was created
36
+ if (modalCreated && element.parentNode) {
37
+ element.parentNode.removeChild(element);
38
+ }
39
+ };
40
+ }, [wrapperId]);
41
+
42
+ if (wrapperElement === null) return null;
43
+
44
+ return createPortal(children, wrapperElement);
45
+ };
@@ -1,74 +1,74 @@
1
- 'use client';
2
-
3
- import { useEffect, useState } from 'react';
4
- import { ROUTES } from 'routes';
5
- import { useGet3dRedirectFormQuery } from '@akinon/next/data/client/checkout';
6
- import { LoaderSpinner } from 'components';
7
- import { useLocalization } from '../../../hooks/use-localization';
8
- import { getUrlPathWithLocale } from '../../../utils/localization';
9
- import { useSearchParams } from 'next/navigation';
10
-
11
- interface RedirectThreeDContentProps {
12
- sessionId: string;
13
- }
14
-
15
- export default function RedirectThreeDContent({
16
- sessionId
17
- }: RedirectThreeDContentProps) {
18
- const searchParams = useSearchParams();
19
- const { data } = useGet3dRedirectFormQuery();
20
- const [error, setError] = useState(null);
21
- const { locale } = useLocalization();
22
-
23
- useEffect(() => {
24
- if (data) {
25
- const fragment = document.createElement('fragment');
26
- fragment.innerHTML = data.result;
27
-
28
- const form = fragment.querySelector('form');
29
-
30
- // a way to determine if response includes a redirection form or not
31
- if (fragment.querySelector('link[rel="canonical"]') || !form) {
32
- setError('Redirecting to checkout page. Please wait...');
33
-
34
- setTimeout(() => {
35
- let checkoutUrl = `${ROUTES.CHECKOUT}`;
36
-
37
- // iframe param is used to prevent header and footer rendering
38
- if (searchParams.get('iframe') === 'true') {
39
- checkoutUrl = `${checkoutUrl}?iframe=true`;
40
- }
41
-
42
- // Use `window.location.href` instead of `router.push`
43
- // to capture the url change event in iframe
44
- location.href = getUrlPathWithLocale(checkoutUrl, locale);
45
- }, 3000);
46
- return;
47
- }
48
-
49
- const pzParamsInput = document.createElement('input');
50
- pzParamsInput.setAttribute('type', 'hidden');
51
- pzParamsInput.setAttribute('name', 'pzparams');
52
- pzParamsInput.setAttribute(
53
- 'value',
54
- encodeURIComponent(
55
- JSON.stringify({
56
- session: sessionId,
57
- locale
58
- })
59
- )
60
- );
61
- form.appendChild(pzParamsInput);
62
-
63
- form.style.display = 'none';
64
- document.body.appendChild(form);
65
- form.submit();
66
- }
67
- }, [data]);
68
-
69
- return (
70
- <div className="flex items-center justify-center py-20">
71
- {error ?? <LoaderSpinner />}
72
- </div>
73
- );
74
- }
1
+ 'use client';
2
+
3
+ import { useEffect, useState } from 'react';
4
+ import { ROUTES } from 'routes';
5
+ import { useGet3dRedirectFormQuery } from '@akinon/next/data/client/checkout';
6
+ import { LoaderSpinner } from 'components';
7
+ import { useLocalization } from '../../../hooks/use-localization';
8
+ import { getUrlPathWithLocale } from '../../../utils/localization';
9
+ import { useSearchParams } from 'next/navigation';
10
+
11
+ interface RedirectThreeDContentProps {
12
+ sessionId: string;
13
+ }
14
+
15
+ export default function RedirectThreeDContent({
16
+ sessionId
17
+ }: RedirectThreeDContentProps) {
18
+ const searchParams = useSearchParams();
19
+ const { data } = useGet3dRedirectFormQuery();
20
+ const [error, setError] = useState(null);
21
+ const { locale } = useLocalization();
22
+
23
+ useEffect(() => {
24
+ if (data) {
25
+ const fragment = document.createElement('fragment');
26
+ fragment.innerHTML = data.result;
27
+
28
+ const form = fragment.querySelector('form');
29
+
30
+ // a way to determine if response includes a redirection form or not
31
+ if (fragment.querySelector('link[rel="canonical"]') || !form) {
32
+ setError('Redirecting to checkout page. Please wait...');
33
+
34
+ setTimeout(() => {
35
+ let checkoutUrl = `${ROUTES.CHECKOUT}`;
36
+
37
+ // iframe param is used to prevent header and footer rendering
38
+ if (searchParams.get('iframe') === 'true') {
39
+ checkoutUrl = `${checkoutUrl}?iframe=true`;
40
+ }
41
+
42
+ // Use `window.location.href` instead of `router.push`
43
+ // to capture the url change event in iframe
44
+ location.href = getUrlPathWithLocale(checkoutUrl, locale);
45
+ }, 3000);
46
+ return;
47
+ }
48
+
49
+ const pzParamsInput = document.createElement('input');
50
+ pzParamsInput.setAttribute('type', 'hidden');
51
+ pzParamsInput.setAttribute('name', 'pzparams');
52
+ pzParamsInput.setAttribute(
53
+ 'value',
54
+ encodeURIComponent(
55
+ JSON.stringify({
56
+ session: sessionId,
57
+ locale
58
+ })
59
+ )
60
+ );
61
+ form.appendChild(pzParamsInput);
62
+
63
+ form.style.display = 'none';
64
+ document.body.appendChild(form);
65
+ form.submit();
66
+ }
67
+ }, [data]);
68
+
69
+ return (
70
+ <div className="flex items-center justify-center py-20">
71
+ {error ?? <LoaderSpinner />}
72
+ </div>
73
+ );
74
+ }
@@ -1,17 +1,17 @@
1
- import { cookies } from 'next/headers';
2
- import { redirect } from 'next/navigation';
3
- import RedirectThreeDContent from './content';
4
- import { ROUTES } from 'routes';
5
-
6
- const RedirectThreeD = async () => {
7
- const nextCookies = cookies();
8
- const sessionId = await nextCookies.get('osessionid')?.value;
9
-
10
- if (!sessionId) {
11
- return redirect(ROUTES.CHECKOUT);
12
- }
13
-
14
- return <RedirectThreeDContent sessionId={sessionId} />;
15
- };
16
-
17
- export default RedirectThreeD;
1
+ import { cookies } from 'next/headers';
2
+ import { redirect } from 'next/navigation';
3
+ import RedirectThreeDContent from './content';
4
+ import { ROUTES } from 'routes';
5
+
6
+ const RedirectThreeD = async () => {
7
+ const nextCookies = cookies();
8
+ const sessionId = await nextCookies.get('osessionid')?.value;
9
+
10
+ if (!sessionId) {
11
+ return redirect(ROUTES.CHECKOUT);
12
+ }
13
+
14
+ return <RedirectThreeDContent sessionId={sessionId} />;
15
+ };
16
+
17
+ export default RedirectThreeD;
@@ -50,7 +50,14 @@ export default function SelectedPaymentOptionView() {
50
50
  );
51
51
  } else if (payment_option.payment_type === 'loyalty_money') {
52
52
  promise = import(`views/checkout/steps/payment/options/loyalty`);
53
+ } else if (payment_option.payment_type === 'masterpass') {
54
+ promise = import(
55
+ `views/checkout/steps/payment/options/credit-card`
56
+ );
53
57
  }
58
+ // else if (payment_option.payment_type === 'credit_payment') {
59
+ // promise = import(`views/checkout/steps/payment/options/credit-payment`);
60
+ // }
54
61
  // else if (payment_option.payment_type === 'gpay') {
55
62
  // promise = import(`views/checkout/steps/payment/options/gpay`);
56
63
  // }
@@ -1,39 +1,39 @@
1
- 'use client';
2
-
3
- import { Fragment, useMemo } from 'react';
4
- import reactStringReplace from 'react-string-replace';
5
- import { useLocalization } from '@akinon/next/hooks';
6
-
7
- interface TransProps {
8
- i18nKey: string;
9
- components?: {
10
- [key: string]: React.ReactElement;
11
- };
12
- }
13
-
14
- export const Trans = ({ i18nKey, components }: TransProps) => {
15
- const { t } = useLocalization();
16
- const value = t(i18nKey);
17
-
18
- const replacedValue = useMemo(() => {
19
- if (!components) {
20
- return value;
21
- }
22
-
23
- let text;
24
-
25
- for (const [k] of Object.entries(components)) {
26
- text = reactStringReplace(
27
- text ?? value,
28
- new RegExp(`(<${k}/>|<${k} />)`),
29
- (_match, i) => (
30
- <Fragment key={`trans-${k}-${i}`}>{components[k]}</Fragment>
31
- )
32
- );
33
- }
34
-
35
- return text;
36
- }, [value, components]);
37
-
38
- return <>{replacedValue}</>;
39
- };
1
+ 'use client';
2
+
3
+ import { Fragment, useMemo } from 'react';
4
+ import reactStringReplace from 'react-string-replace';
5
+ import { useLocalization } from '@akinon/next/hooks';
6
+
7
+ interface TransProps {
8
+ i18nKey: string;
9
+ components?: {
10
+ [key: string]: React.ReactElement;
11
+ };
12
+ }
13
+
14
+ export const Trans = ({ i18nKey, components }: TransProps) => {
15
+ const { t } = useLocalization();
16
+ const value = t(i18nKey);
17
+
18
+ const replacedValue = useMemo(() => {
19
+ if (!components) {
20
+ return value;
21
+ }
22
+
23
+ let text;
24
+
25
+ for (const [k] of Object.entries(components)) {
26
+ text = reactStringReplace(
27
+ text ?? value,
28
+ new RegExp(`(<${k}/>|<${k} />)`),
29
+ (_match, i) => (
30
+ <Fragment key={`trans-${k}-${i}`}>{components[k]}</Fragment>
31
+ )
32
+ );
33
+ }
34
+
35
+ return text;
36
+ }, [value, components]);
37
+
38
+ return <>{replacedValue}</>;
39
+ };