@akinon/projectzero 1.101.0 → 1.102.0-rc.77
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/CHANGELOG.md +233 -4
- package/app-template/.env.example +1 -0
- package/app-template/.github/instructions/routing.instructions.md +603 -0
- package/app-template/CHANGELOG.md +4980 -316
- package/app-template/README.md +25 -1
- package/app-template/package.json +19 -19
- package/app-template/public/locales/en/checkout.json +6 -0
- package/app-template/public/locales/en/common.json +48 -1
- package/app-template/public/locales/tr/checkout.json +6 -0
- package/app-template/public/locales/tr/common.json +48 -1
- package/app-template/src/app/[commerce]/[locale]/[currency]/basket/page.tsx +9 -82
- package/app-template/src/app/[commerce]/[locale]/[currency]/category/[pk]/page.tsx +17 -4
- package/app-template/src/app/[commerce]/[locale]/[currency]/flat-page/[pk]/page.tsx +12 -1
- package/app-template/src/app/[commerce]/[locale]/[currency]/group-product/[pk]/page.tsx +29 -11
- package/app-template/src/app/[commerce]/[locale]/[currency]/landing-page/[pk]/page.tsx +12 -1
- package/app-template/src/app/[commerce]/[locale]/[currency]/product/[pk]/loading.tsx +67 -0
- package/app-template/src/app/[commerce]/[locale]/[currency]/product/[pk]/page.tsx +28 -10
- package/app-template/src/app/[commerce]/[locale]/[currency]/special-page/[pk]/page.tsx +12 -1
- package/app-template/src/app/api/form/[...id]/route.ts +1 -7
- package/app-template/src/app/api/image-proxy/route.ts +1 -0
- package/app-template/src/app/api/similar-product-list/route.ts +1 -0
- package/app-template/src/app/api/similar-products/route.ts +1 -0
- package/app-template/src/assets/fonts/pz-icon.css +3 -0
- package/app-template/src/components/__tests__/link.test.tsx +2 -0
- package/app-template/src/components/accordion.tsx +22 -19
- package/app-template/src/components/currency-select.tsx +1 -0
- package/app-template/src/components/file-input.tsx +27 -7
- package/app-template/src/components/generate-form-fields.tsx +43 -4
- package/app-template/src/components/input.tsx +9 -2
- package/app-template/src/components/modal.tsx +32 -16
- package/app-template/src/components/pagination.tsx +1 -0
- package/app-template/src/components/price.tsx +1 -1
- package/app-template/src/components/select.tsx +38 -26
- package/app-template/src/components/types/index.ts +25 -1
- package/app-template/src/hooks/index.ts +2 -0
- package/app-template/src/hooks/use-product-cart.ts +77 -0
- package/app-template/src/hooks/use-stock-alert.ts +74 -0
- package/app-template/src/plugins.js +3 -1
- package/app-template/src/settings.js +8 -2
- package/app-template/src/types/index.ts +17 -0
- package/app-template/src/utils/variant-validation.ts +41 -0
- package/app-template/src/views/account/address-form.tsx +8 -4
- package/app-template/src/views/account/contact-form.tsx +1 -1
- package/app-template/src/views/account/content-header.tsx +2 -2
- package/app-template/src/views/account/faq/faq-tabs.tsx +8 -2
- package/app-template/src/views/basket/basket-content.tsx +106 -0
- package/app-template/src/views/basket/basket-item.tsx +22 -14
- package/app-template/src/views/basket/summary.tsx +10 -7
- package/app-template/src/views/breadcrumb.tsx +2 -2
- package/app-template/src/views/category/category-info.tsx +1 -0
- package/app-template/src/views/category/filters/index.tsx +1 -1
- package/app-template/src/views/checkout/steps/payment/options/store-credit.tsx +121 -0
- package/app-template/src/views/checkout/summary.tsx +10 -0
- package/app-template/src/views/guest-login/index.tsx +6 -1
- package/app-template/src/views/header/action-menu.tsx +1 -1
- package/app-template/src/views/header/search/index.tsx +17 -5
- package/app-template/src/views/product/product-actions.tsx +165 -0
- package/app-template/src/views/product/product-info.tsx +62 -263
- package/app-template/src/views/product/product-share.tsx +56 -0
- package/app-template/src/views/product/product-variants.tsx +26 -0
- package/app-template/src/views/product/slider.tsx +86 -73
- package/app-template/src/widgets/footer-menu.tsx +6 -2
- package/commands/plugins.ts +63 -16
- package/dist/commands/plugins.js +57 -16
- package/package.json +1 -1
|
@@ -0,0 +1,121 @@
|
|
|
1
|
+
import clsx from 'clsx';
|
|
2
|
+
import {
|
|
3
|
+
useGetCheckoutLoyaltyBalanceQuery,
|
|
4
|
+
usePayWithLoyaltyBalanceMutation
|
|
5
|
+
} from '@akinon/next/data/client/checkout';
|
|
6
|
+
import { useAppSelector } from '@akinon/next/redux/hooks';
|
|
7
|
+
import { useMemo, useState } from 'react';
|
|
8
|
+
import { useLocalization } from '@akinon/next/hooks';
|
|
9
|
+
import { twMerge } from 'tailwind-merge';
|
|
10
|
+
import { Icon, Price } from '@theme/components';
|
|
11
|
+
import { Trans } from '@akinon/next/components';
|
|
12
|
+
import { LoaderSpinner } from '@theme/components';
|
|
13
|
+
|
|
14
|
+
export const StoreCredits = () => {
|
|
15
|
+
const { t } = useLocalization();
|
|
16
|
+
|
|
17
|
+
const [payWithLoyaltyBalance, { isLoading: isPayWithLoyaltyBalanceLoading }] =
|
|
18
|
+
usePayWithLoyaltyBalanceMutation();
|
|
19
|
+
|
|
20
|
+
const { loyaltyBalance, preOrder } = useAppSelector(
|
|
21
|
+
(state) => state.checkout
|
|
22
|
+
);
|
|
23
|
+
|
|
24
|
+
const { isLoading: isLoyaltyBalanceLoading } =
|
|
25
|
+
useGetCheckoutLoyaltyBalanceQuery(undefined, {
|
|
26
|
+
refetchOnMountOrArgChange: true,
|
|
27
|
+
skip: !preOrder?.payment_option
|
|
28
|
+
});
|
|
29
|
+
|
|
30
|
+
const isLoyaltyBalanceUsed = useMemo(() => {
|
|
31
|
+
return parseFloat(preOrder?.loyalty_money ?? '0') > 0;
|
|
32
|
+
}, [preOrder?.loyalty_money]);
|
|
33
|
+
|
|
34
|
+
const [isLoading, setIsLoading] = useState(false);
|
|
35
|
+
|
|
36
|
+
const handleClick = async () => {
|
|
37
|
+
if (isLoading) return;
|
|
38
|
+
setIsLoading(true);
|
|
39
|
+
|
|
40
|
+
try {
|
|
41
|
+
await payWithLoyaltyBalance(isLoyaltyBalanceUsed ? '0' : loyaltyBalance);
|
|
42
|
+
} finally {
|
|
43
|
+
setIsLoading(false);
|
|
44
|
+
}
|
|
45
|
+
};
|
|
46
|
+
|
|
47
|
+
if (preOrder?.is_guest) {
|
|
48
|
+
return null;
|
|
49
|
+
}
|
|
50
|
+
|
|
51
|
+
if (isLoyaltyBalanceLoading) {
|
|
52
|
+
return (
|
|
53
|
+
<div className="mb-3 px-4 py-3 xs:px-0">
|
|
54
|
+
<LoaderSpinner />
|
|
55
|
+
</div>
|
|
56
|
+
);
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
if (parseFloat(loyaltyBalance) <= 0) {
|
|
60
|
+
return null;
|
|
61
|
+
}
|
|
62
|
+
return (
|
|
63
|
+
<div
|
|
64
|
+
className={twMerge(
|
|
65
|
+
'hidden flex-col w-full mb-4 border border-solid border-gray-400 px-0 md:p-4',
|
|
66
|
+
isPayWithLoyaltyBalanceLoading && 'pointer-events-none opacity-30',
|
|
67
|
+
parseFloat(loyaltyBalance) > 0 && 'block'
|
|
68
|
+
)}
|
|
69
|
+
>
|
|
70
|
+
<div className="flex w-full items-center">
|
|
71
|
+
<button onClick={handleClick}>
|
|
72
|
+
<span
|
|
73
|
+
className={clsx(
|
|
74
|
+
'flex h-5 w-5 items-center justify-center rounded border border-solid border-primary',
|
|
75
|
+
isLoyaltyBalanceUsed ? 'bg-primary' : 'bg-white'
|
|
76
|
+
)}
|
|
77
|
+
>
|
|
78
|
+
<Icon
|
|
79
|
+
name={isLoyaltyBalanceUsed ? 'check' : ''}
|
|
80
|
+
size={10}
|
|
81
|
+
className={clsx({ 'text-white': isLoyaltyBalanceUsed })}
|
|
82
|
+
/>
|
|
83
|
+
</span>
|
|
84
|
+
</button>
|
|
85
|
+
|
|
86
|
+
<div className="w-full pl-4">
|
|
87
|
+
<p className="cursor-pointer text-sm" onClick={handleClick}>
|
|
88
|
+
{t('checkout.payment.store_credit.use_my_store_credits')}
|
|
89
|
+
</p>
|
|
90
|
+
<p className="flex text-sm text-[#606060]">
|
|
91
|
+
{t('checkout.payment.store_credit.available_balance')}:
|
|
92
|
+
<Price
|
|
93
|
+
value={loyaltyBalance}
|
|
94
|
+
currencyCode={preOrder?.currency_type_label}
|
|
95
|
+
className="pe-1 font-bold"
|
|
96
|
+
/>
|
|
97
|
+
</p>
|
|
98
|
+
</div>
|
|
99
|
+
</div>
|
|
100
|
+
|
|
101
|
+
{isLoyaltyBalanceUsed && parseFloat(preOrder?.unpaid_amount) > 0 && (
|
|
102
|
+
<p className="my-4 text-[15px] font-light italic text-[#707070] max-xs:text-xs">
|
|
103
|
+
<Trans
|
|
104
|
+
i18nKey="checkout.payment.store_credit.insufficient_balance"
|
|
105
|
+
components={{
|
|
106
|
+
Amount: (
|
|
107
|
+
<div className="inline-flex">
|
|
108
|
+
<Price
|
|
109
|
+
value={preOrder?.unpaid_amount}
|
|
110
|
+
currencyCode={preOrder?.currency_type_label}
|
|
111
|
+
className="text-primary"
|
|
112
|
+
/>
|
|
113
|
+
</div>
|
|
114
|
+
)
|
|
115
|
+
}}
|
|
116
|
+
/>
|
|
117
|
+
</p>
|
|
118
|
+
)}
|
|
119
|
+
</div>
|
|
120
|
+
);
|
|
121
|
+
};
|
|
@@ -8,6 +8,7 @@ import PluginModule, { Component } from '@akinon/next/components/plugin-module';
|
|
|
8
8
|
import { twMerge } from 'tailwind-merge';
|
|
9
9
|
import { Image } from '@akinon/next/components/image';
|
|
10
10
|
import { Trans } from '@akinon/next/components/trans';
|
|
11
|
+
import { StoreCredits } from './steps/payment/options/store-credit';
|
|
11
12
|
|
|
12
13
|
export const Summary = () => {
|
|
13
14
|
const { t } = useLocalization();
|
|
@@ -38,6 +39,7 @@ export const Summary = () => {
|
|
|
38
39
|
'flex flex-col w-full mb-4 border border-solid border-gray-400'
|
|
39
40
|
}}
|
|
40
41
|
/>
|
|
42
|
+
<StoreCredits />
|
|
41
43
|
<div className="flex flex-col w-full border border-solid border-gray-400">
|
|
42
44
|
<div className="flex justify-between items-center flex-row border-b border-solid border-gray-400 px-4 py-2 sm:px-5 sm:py-4 sm:min-h-15">
|
|
43
45
|
<span className="text-black-800 text-xl font-light sm:text-2xl">
|
|
@@ -118,6 +120,14 @@ export const Summary = () => {
|
|
|
118
120
|
<Price value={preOrder?.shipping_amount} />
|
|
119
121
|
</span>
|
|
120
122
|
</div>
|
|
123
|
+
{parseFloat(preOrder?.loyalty_money) > 0 && (
|
|
124
|
+
<div className="flex items-center justify-between w-full text-xs text-black-800 py-1 px-4 sm:px-5">
|
|
125
|
+
<span>{t('checkout.summary.loyalty_money_total')}</span>
|
|
126
|
+
<span>
|
|
127
|
+
<Price value={preOrder?.loyalty_money} />
|
|
128
|
+
</span>
|
|
129
|
+
</div>
|
|
130
|
+
)}
|
|
121
131
|
<div className="flex items-center justify-between w-full text-xs text-black-800 py-1 px-4 sm:px-5">
|
|
122
132
|
<span>{t('checkout.summary.discounts_total')}</span>
|
|
123
133
|
<span>
|
|
@@ -51,9 +51,14 @@ const GuestLogin = () => {
|
|
|
51
51
|
).unwrap();
|
|
52
52
|
|
|
53
53
|
Object.keys(response?.errors || {}).forEach((key) => {
|
|
54
|
+
const errorValue = response?.errors[key];
|
|
55
|
+
const message = Array.isArray(errorValue)
|
|
56
|
+
? errorValue.join(', ')
|
|
57
|
+
: errorValue || '';
|
|
58
|
+
|
|
54
59
|
setError(key as keyof GuestLoginFormParams, {
|
|
55
60
|
type: 'custom',
|
|
56
|
-
message
|
|
61
|
+
message
|
|
57
62
|
});
|
|
58
63
|
});
|
|
59
64
|
} catch (error) {
|
|
@@ -4,11 +4,11 @@ import { useEffect, useRef, useState } from 'react';
|
|
|
4
4
|
import { useAppDispatch, useAppSelector } from '@akinon/next/redux/hooks';
|
|
5
5
|
import { closeSearch } from '@akinon/next/redux/reducers/header';
|
|
6
6
|
import clsx from 'clsx';
|
|
7
|
-
|
|
8
|
-
import { Icon } from '@theme/components';
|
|
7
|
+
import { Icon, Input } from '@theme/components';
|
|
9
8
|
import Results from './results';
|
|
10
9
|
import { ROUTES } from '@theme/routes';
|
|
11
10
|
import { useLocalization, useRouter } from '@akinon/next/hooks';
|
|
11
|
+
import PluginModule, { Component } from '@akinon/next/components/plugin-module';
|
|
12
12
|
|
|
13
13
|
export default function Search() {
|
|
14
14
|
const { t } = useLocalization();
|
|
@@ -41,6 +41,14 @@ export default function Search() {
|
|
|
41
41
|
};
|
|
42
42
|
}, [isSearchOpen, dispatch]);
|
|
43
43
|
|
|
44
|
+
const handleSearchTextChange = (e: React.ChangeEvent<HTMLInputElement>) => {
|
|
45
|
+
setSearchText(e.target.value);
|
|
46
|
+
};
|
|
47
|
+
|
|
48
|
+
const handleCloseSearch = () => {
|
|
49
|
+
dispatch(closeSearch());
|
|
50
|
+
};
|
|
51
|
+
|
|
44
52
|
return (
|
|
45
53
|
<>
|
|
46
54
|
<div
|
|
@@ -66,9 +74,9 @@ export default function Search() {
|
|
|
66
74
|
{t('common.search.results_for')}
|
|
67
75
|
</span>
|
|
68
76
|
<div className="flex items-center">
|
|
69
|
-
<
|
|
77
|
+
<Input
|
|
70
78
|
value={searchText}
|
|
71
|
-
onChange={
|
|
79
|
+
onChange={handleSearchTextChange}
|
|
72
80
|
onKeyDown={(e) => {
|
|
73
81
|
if (e.key === 'Enter' && searchText.trim() !== '') {
|
|
74
82
|
router.push(`${ROUTES.LIST}/?search_text=${searchText}`);
|
|
@@ -78,14 +86,18 @@ export default function Search() {
|
|
|
78
86
|
placeholder={t('common.search.placeholder')}
|
|
79
87
|
ref={inputRef}
|
|
80
88
|
/>
|
|
89
|
+
|
|
90
|
+
<PluginModule component={Component.HeaderImageSearchFeature} />
|
|
91
|
+
|
|
81
92
|
<Icon
|
|
82
93
|
name="close"
|
|
83
94
|
size={14}
|
|
84
|
-
onClick={
|
|
95
|
+
onClick={handleCloseSearch}
|
|
85
96
|
className="cursor-pointer"
|
|
86
97
|
/>
|
|
87
98
|
</div>
|
|
88
99
|
</div>
|
|
100
|
+
|
|
89
101
|
<Results searchText={searchText} />
|
|
90
102
|
</div>
|
|
91
103
|
</div>
|
|
@@ -0,0 +1,165 @@
|
|
|
1
|
+
import React from 'react';
|
|
2
|
+
import clsx from 'clsx';
|
|
3
|
+
import { Button, Icon, Modal } from '@theme/components';
|
|
4
|
+
import { useLocalization } from '@akinon/next/hooks';
|
|
5
|
+
import PluginModule, { Component } from '@akinon/next/components/plugin-module';
|
|
6
|
+
import { validateVariantSelection } from '../../utils/variant-validation';
|
|
7
|
+
import { VariantType } from '@akinon/next/types';
|
|
8
|
+
|
|
9
|
+
interface Product {
|
|
10
|
+
pk: number;
|
|
11
|
+
name: string;
|
|
12
|
+
[key: string]: any;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
interface ProductActionsProps {
|
|
16
|
+
product: Product;
|
|
17
|
+
variants: VariantType[];
|
|
18
|
+
inStock: boolean;
|
|
19
|
+
isVariantLoading: boolean;
|
|
20
|
+
onAddToCart: () => void;
|
|
21
|
+
onAddToStockAlert: () => void;
|
|
22
|
+
onClearError: () => void;
|
|
23
|
+
isAddToCartLoading: boolean;
|
|
24
|
+
isAddToStockAlertLoading: boolean;
|
|
25
|
+
productError: React.ReactNode | null;
|
|
26
|
+
isModalOpen: boolean;
|
|
27
|
+
stockAlertResponseMessage: React.ReactNode | null;
|
|
28
|
+
onCloseModal: () => void;
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
export const ProductActions: React.FC<ProductActionsProps> = ({
|
|
32
|
+
product,
|
|
33
|
+
variants,
|
|
34
|
+
inStock,
|
|
35
|
+
isVariantLoading,
|
|
36
|
+
onAddToCart,
|
|
37
|
+
onAddToStockAlert,
|
|
38
|
+
onClearError,
|
|
39
|
+
isAddToCartLoading,
|
|
40
|
+
isAddToStockAlertLoading,
|
|
41
|
+
productError,
|
|
42
|
+
isModalOpen,
|
|
43
|
+
stockAlertResponseMessage,
|
|
44
|
+
onCloseModal
|
|
45
|
+
}) => {
|
|
46
|
+
const { t } = useLocalization();
|
|
47
|
+
|
|
48
|
+
const checkoutProviderProps = {
|
|
49
|
+
product,
|
|
50
|
+
clearBasket: true,
|
|
51
|
+
addBeforeClick: () => validateVariantSelection(variants).isValid,
|
|
52
|
+
openMiniBasket: false,
|
|
53
|
+
className: clsx([
|
|
54
|
+
'py-2.5',
|
|
55
|
+
'bg-black',
|
|
56
|
+
'relative',
|
|
57
|
+
'hover:bg-black',
|
|
58
|
+
'before:content-[""]',
|
|
59
|
+
'before:w-6',
|
|
60
|
+
'before:h-6',
|
|
61
|
+
'before:bg-white',
|
|
62
|
+
'before:absolute',
|
|
63
|
+
'before:rounded-r-[18px]',
|
|
64
|
+
'before:left-0',
|
|
65
|
+
'after:content-[""]',
|
|
66
|
+
'after:absolute',
|
|
67
|
+
'after:w-3',
|
|
68
|
+
'after:h-3',
|
|
69
|
+
'after:bg-[#d02c2f]',
|
|
70
|
+
'after:rounded-xl',
|
|
71
|
+
'after:left-1'
|
|
72
|
+
]),
|
|
73
|
+
onError: (error: any) => {
|
|
74
|
+
const formattedError = error?.data?.non_field_errors ||
|
|
75
|
+
Object.keys(error?.data || {}).map(
|
|
76
|
+
(key) => `${key}: ${error?.data[key].join(', ')}`
|
|
77
|
+
);
|
|
78
|
+
// This would need to be handled by parent component
|
|
79
|
+
console.error('Checkout error:', formattedError);
|
|
80
|
+
}
|
|
81
|
+
};
|
|
82
|
+
|
|
83
|
+
const handleMainActionClick = () => {
|
|
84
|
+
onClearError();
|
|
85
|
+
|
|
86
|
+
if (inStock) {
|
|
87
|
+
onAddToCart();
|
|
88
|
+
} else {
|
|
89
|
+
onAddToStockAlert();
|
|
90
|
+
}
|
|
91
|
+
};
|
|
92
|
+
|
|
93
|
+
return (
|
|
94
|
+
<>
|
|
95
|
+
{productError && (
|
|
96
|
+
<div className="mt-4 text-xs text-center text-error">
|
|
97
|
+
{productError}
|
|
98
|
+
</div>
|
|
99
|
+
)}
|
|
100
|
+
|
|
101
|
+
<Button
|
|
102
|
+
disabled={isAddToCartLoading || isAddToStockAlertLoading || isVariantLoading}
|
|
103
|
+
className={clsx(
|
|
104
|
+
'fixed bottom-0 right-0 w-1/2 h-14 z-[20] flex items-center justify-center fill-primary-foreground',
|
|
105
|
+
'hover:fill-primary sm:relative sm:w-full sm:mt-3 sm:font-semibold sm:h-12'
|
|
106
|
+
)}
|
|
107
|
+
onClick={handleMainActionClick}
|
|
108
|
+
data-testid="product-add-to-cart"
|
|
109
|
+
>
|
|
110
|
+
{isVariantLoading ? (
|
|
111
|
+
<Icon
|
|
112
|
+
name="spinner"
|
|
113
|
+
size={20}
|
|
114
|
+
className="animate-spin mr-4 fill-primary"
|
|
115
|
+
/>
|
|
116
|
+
) : inStock ? (
|
|
117
|
+
<span>{t('product.add_to_cart')}</span>
|
|
118
|
+
) : (
|
|
119
|
+
<>
|
|
120
|
+
<Icon name="bell" size={20} className="mr-4" />
|
|
121
|
+
<span>{t('product.add_stock_alert')}</span>
|
|
122
|
+
</>
|
|
123
|
+
)}
|
|
124
|
+
</Button>
|
|
125
|
+
|
|
126
|
+
<PluginModule
|
|
127
|
+
component={Component.AkifastCheckoutButton}
|
|
128
|
+
props={{
|
|
129
|
+
...checkoutProviderProps,
|
|
130
|
+
isPdp: true
|
|
131
|
+
}}
|
|
132
|
+
/>
|
|
133
|
+
|
|
134
|
+
<PluginModule
|
|
135
|
+
component={Component.OneClickCheckoutButtons}
|
|
136
|
+
props={checkoutProviderProps}
|
|
137
|
+
/>
|
|
138
|
+
|
|
139
|
+
<Modal
|
|
140
|
+
portalId="stock-alert-modal"
|
|
141
|
+
open={isModalOpen}
|
|
142
|
+
setOpen={onCloseModal}
|
|
143
|
+
showCloseButton={false}
|
|
144
|
+
className="w-5/6 md:max-w-md"
|
|
145
|
+
>
|
|
146
|
+
<div className="flex flex-col items-center justify-center gap-4 px-6 py-9">
|
|
147
|
+
<Icon name="bell" size={48} />
|
|
148
|
+
<h2 className="text-xl font-semibold">
|
|
149
|
+
{t('product.stock_alert.title')}
|
|
150
|
+
</h2>
|
|
151
|
+
<div className="max-w-40 text-xs text-center leading-4">
|
|
152
|
+
<p>{stockAlertResponseMessage}</p>
|
|
153
|
+
</div>
|
|
154
|
+
<Button
|
|
155
|
+
onClick={onCloseModal}
|
|
156
|
+
appearance="outlined"
|
|
157
|
+
className="font-semibold px-10 h-12"
|
|
158
|
+
>
|
|
159
|
+
{t('product.stock_alert.close_button')}
|
|
160
|
+
</Button>
|
|
161
|
+
</div>
|
|
162
|
+
</Modal>
|
|
163
|
+
</>
|
|
164
|
+
);
|
|
165
|
+
};
|