@akinon/next 1.14.0 → 1.14.1
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/.eslintrc.js +40 -40
- package/.prettierrc +13 -13
- package/CHANGELOG.md +6 -0
- package/api/auth.ts +231 -231
- package/api/cache.ts +44 -44
- package/api/client.ts +174 -174
- package/api/logout.ts +42 -42
- package/bin/pz-check-dependencies.js +98 -98
- package/bin/pz-install-plugins.js +33 -33
- package/bin/pz-install-theme.js +58 -58
- package/bin/pz-postbuild.js +1 -1
- package/bin/pz-postdev.js +1 -1
- package/bin/pz-postinstall.js +6 -6
- package/bin/pz-poststart.js +1 -1
- package/bin/pz-prebuild.js +4 -4
- package/bin/pz-predev.js +4 -4
- package/bin/pz-prestart.js +1 -1
- package/bin/run-script.js +44 -44
- package/components/accordion.tsx +52 -52
- package/components/button.tsx +46 -46
- package/components/client-root.tsx +19 -19
- package/components/icon.tsx +18 -18
- package/components/image.tsx +133 -133
- package/components/index.ts +17 -17
- package/components/input.tsx +110 -110
- package/components/lazy-component.tsx +33 -33
- package/components/loader-spinner.tsx +23 -23
- package/components/mobile-app-toggler.tsx +26 -26
- package/components/oauth-login.tsx +24 -24
- package/components/price.tsx +55 -55
- package/components/pz-providers.tsx +24 -24
- package/components/pz-root.tsx +21 -21
- package/components/radio.tsx +18 -18
- package/components/react-portal.tsx +45 -45
- package/components/redirect-three-d/content/index.tsx +74 -74
- package/components/redirect-three-d/index.tsx +17 -17
- package/components/trans.tsx +39 -39
- package/data/client/account.ts +208 -208
- package/data/client/api.ts +85 -85
- package/data/client/basket.ts +82 -82
- package/data/client/misc.ts +101 -101
- package/data/client/product.ts +89 -89
- package/data/client/user.ts +99 -99
- package/data/client/wishlist.ts +118 -118
- package/data/server/category.ts +132 -132
- package/data/server/flatpage.ts +21 -21
- package/data/server/form.ts +22 -22
- package/data/server/index.ts +10 -10
- package/data/server/landingpage.ts +24 -24
- package/data/server/list.ts +67 -67
- package/data/server/menu.ts +35 -35
- package/data/server/product.ts +86 -86
- package/data/server/seo.ts +48 -48
- package/data/server/special-page.ts +47 -47
- package/data/server/widget.ts +27 -27
- package/data/urls.ts +221 -221
- package/hocs/client/index.ts +1 -1
- package/hocs/client/with-segment-defaults.tsx +25 -25
- package/hocs/server/index.ts +1 -1
- package/hocs/server/with-segment-defaults.tsx +85 -85
- package/hooks/index.ts +10 -10
- package/hooks/use-captcha.tsx +76 -76
- package/hooks/use-common-product-attributes.ts +36 -36
- package/hooks/use-debounce.ts +20 -20
- package/hooks/use-localization.ts +78 -78
- package/hooks/use-media-query.ts +36 -36
- package/hooks/use-mobile-iframe-handler.ts +23 -23
- package/hooks/use-on-click-outside.tsx +28 -28
- package/hooks/use-router.ts +45 -45
- package/hooks/use-translation.ts +14 -14
- package/lib/cache.ts +215 -215
- package/localization/index.ts +5 -5
- package/localization/provider.tsx +58 -58
- package/middlewares/currency.ts +100 -100
- package/middlewares/default.ts +256 -256
- package/middlewares/index.ts +29 -29
- package/middlewares/locale.ts +68 -68
- package/middlewares/oauth-login.ts +79 -79
- package/middlewares/pretty-url.ts +104 -104
- package/middlewares/redirection-payment.ts +160 -160
- package/middlewares/three-d-redirection.ts +159 -159
- package/middlewares/url-redirection.ts +65 -65
- package/package.json +2 -2
- package/redux/hooks.ts +7 -7
- package/redux/middlewares/index.ts +50 -50
- package/redux/reducers/checkout.ts +184 -184
- package/redux/reducers/config.ts +28 -28
- package/redux/reducers/header.ts +59 -59
- package/redux/reducers/root.ts +61 -61
- package/sentry/index.ts +27 -27
- package/tailwind/rtl.js +137 -137
- package/types/commerce/account.ts +64 -64
- package/types/commerce/address.ts +94 -94
- package/types/commerce/basket.ts +43 -43
- package/types/commerce/category.ts +114 -114
- package/types/commerce/checkout.ts +143 -143
- package/types/commerce/flatpage.ts +7 -7
- package/types/commerce/form.ts +66 -66
- package/types/commerce/index.ts +12 -12
- package/types/commerce/landingpage.ts +7 -7
- package/types/commerce/misc.ts +127 -127
- package/types/commerce/order.ts +119 -119
- package/types/commerce/product.ts +109 -109
- package/types/commerce/widget.ts +28 -28
- package/types/gtm.ts +16 -16
- package/types/index.ts +274 -274
- package/types/metadata.ts +7 -7
- package/types/next-auth.d.ts +24 -24
- package/utils/app-fetch.ts +69 -69
- package/utils/deep-merge.js +24 -24
- package/utils/image-loader.ts +31 -31
- package/utils/index.ts +150 -150
- package/utils/localization.ts +29 -29
- package/utils/log.ts +138 -138
- package/utils/menu-generator.ts +27 -27
- package/utils/mobile-3d-iframe.ts +77 -77
- package/utils/server-translation.ts +57 -57
- package/utils/server-variables.ts +9 -9
- package/with-pz-config.js +94 -94
|
@@ -1,26 +1,26 @@
|
|
|
1
|
-
'use client';
|
|
2
|
-
|
|
3
|
-
import { useAppSelector } from '../redux/hooks';
|
|
4
|
-
import { useState, useEffect } from 'react';
|
|
5
|
-
import { useSearchParams } from 'next/navigation';
|
|
6
|
-
|
|
7
|
-
export default function MobileAppToggler({
|
|
8
|
-
children
|
|
9
|
-
}: {
|
|
10
|
-
children: React.ReactNode;
|
|
11
|
-
}) {
|
|
12
|
-
const searchParams = useSearchParams();
|
|
13
|
-
const { isMobileApp } = useAppSelector((state) => state.root);
|
|
14
|
-
const [isInIframe, setIsInIframe] = useState(false);
|
|
15
|
-
|
|
16
|
-
useEffect(() => {
|
|
17
|
-
if (window.frameElement) {
|
|
18
|
-
setIsInIframe(true);
|
|
19
|
-
}
|
|
20
|
-
}, []);
|
|
21
|
-
|
|
22
|
-
if (isMobileApp || isInIframe || searchParams.get('iframe') === 'true')
|
|
23
|
-
return null;
|
|
24
|
-
|
|
25
|
-
return <>{children}</>;
|
|
26
|
-
}
|
|
1
|
+
'use client';
|
|
2
|
+
|
|
3
|
+
import { useAppSelector } from '../redux/hooks';
|
|
4
|
+
import { useState, useEffect } from 'react';
|
|
5
|
+
import { useSearchParams } from 'next/navigation';
|
|
6
|
+
|
|
7
|
+
export default function MobileAppToggler({
|
|
8
|
+
children
|
|
9
|
+
}: {
|
|
10
|
+
children: React.ReactNode;
|
|
11
|
+
}) {
|
|
12
|
+
const searchParams = useSearchParams();
|
|
13
|
+
const { isMobileApp } = useAppSelector((state) => state.root);
|
|
14
|
+
const [isInIframe, setIsInIframe] = useState(false);
|
|
15
|
+
|
|
16
|
+
useEffect(() => {
|
|
17
|
+
if (window.frameElement) {
|
|
18
|
+
setIsInIframe(true);
|
|
19
|
+
}
|
|
20
|
+
}, []);
|
|
21
|
+
|
|
22
|
+
if (isMobileApp || isInIframe || searchParams.get('iframe') === 'true')
|
|
23
|
+
return null;
|
|
24
|
+
|
|
25
|
+
return <>{children}</>;
|
|
26
|
+
}
|
|
@@ -1,24 +1,24 @@
|
|
|
1
|
-
'use client';
|
|
2
|
-
|
|
3
|
-
import { signIn, useSession } from 'next-auth/react';
|
|
4
|
-
import { useEffect } from 'react';
|
|
5
|
-
import { useSearchParams } from 'next/navigation';
|
|
6
|
-
|
|
7
|
-
export default function OauthLogin() {
|
|
8
|
-
const { status } = useSession();
|
|
9
|
-
const searchParams = useSearchParams();
|
|
10
|
-
|
|
11
|
-
useEffect(() => {
|
|
12
|
-
if (!searchParams) {
|
|
13
|
-
return;
|
|
14
|
-
}
|
|
15
|
-
|
|
16
|
-
if (status === 'unauthenticated') {
|
|
17
|
-
signIn('oauth', {
|
|
18
|
-
callbackUrl: searchParams.get('next') ?? '/'
|
|
19
|
-
});
|
|
20
|
-
}
|
|
21
|
-
}, [status, searchParams]);
|
|
22
|
-
|
|
23
|
-
return null;
|
|
24
|
-
}
|
|
1
|
+
'use client';
|
|
2
|
+
|
|
3
|
+
import { signIn, useSession } from 'next-auth/react';
|
|
4
|
+
import { useEffect } from 'react';
|
|
5
|
+
import { useSearchParams } from 'next/navigation';
|
|
6
|
+
|
|
7
|
+
export default function OauthLogin() {
|
|
8
|
+
const { status } = useSession();
|
|
9
|
+
const searchParams = useSearchParams();
|
|
10
|
+
|
|
11
|
+
useEffect(() => {
|
|
12
|
+
if (!searchParams) {
|
|
13
|
+
return;
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
if (status === 'unauthenticated') {
|
|
17
|
+
signIn('oauth', {
|
|
18
|
+
callbackUrl: searchParams.get('next') ?? '/'
|
|
19
|
+
});
|
|
20
|
+
}
|
|
21
|
+
}, [status, searchParams]);
|
|
22
|
+
|
|
23
|
+
return null;
|
|
24
|
+
}
|
package/components/price.tsx
CHANGED
|
@@ -1,55 +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
|
+
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
|
+
}
|
package/components/pz-root.tsx
CHANGED
|
@@ -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
|
+
}
|
package/components/radio.tsx
CHANGED
|
@@ -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;
|