@akinon/projectzero 1.104.0-rc.83 → 1.104.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 (55) hide show
  1. package/CHANGELOG.md +5 -233
  2. package/app-template/.env.example +0 -1
  3. package/app-template/CHANGELOG.md +325 -4954
  4. package/app-template/README.md +1 -25
  5. package/app-template/package.json +19 -19
  6. package/app-template/public/locales/en/checkout.json +0 -6
  7. package/app-template/public/locales/en/common.json +1 -42
  8. package/app-template/public/locales/tr/checkout.json +0 -6
  9. package/app-template/public/locales/tr/common.json +1 -42
  10. package/app-template/src/app/[commerce]/[locale]/[currency]/basket/page.tsx +82 -9
  11. package/app-template/src/app/[commerce]/[locale]/[currency]/landing-page/[pk]/page.tsx +1 -12
  12. package/app-template/src/assets/fonts/pz-icon.css +0 -3
  13. package/app-template/src/components/__tests__/link.test.tsx +0 -2
  14. package/app-template/src/components/accordion.tsx +19 -22
  15. package/app-template/src/components/file-input.tsx +7 -27
  16. package/app-template/src/components/input.tsx +2 -9
  17. package/app-template/src/components/modal.tsx +16 -32
  18. package/app-template/src/components/pagination.tsx +0 -1
  19. package/app-template/src/components/price.tsx +1 -1
  20. package/app-template/src/components/select.tsx +26 -38
  21. package/app-template/src/components/types/index.ts +1 -25
  22. package/app-template/src/hooks/index.ts +0 -2
  23. package/app-template/src/plugins.js +1 -3
  24. package/app-template/src/settings.js +1 -6
  25. package/app-template/src/types/index.ts +0 -17
  26. package/app-template/src/views/account/address-form.tsx +4 -8
  27. package/app-template/src/views/account/contact-form.tsx +1 -1
  28. package/app-template/src/views/account/content-header.tsx +2 -2
  29. package/app-template/src/views/account/faq/faq-tabs.tsx +2 -8
  30. package/app-template/src/views/basket/basket-item.tsx +14 -22
  31. package/app-template/src/views/basket/summary.tsx +7 -10
  32. package/app-template/src/views/breadcrumb.tsx +2 -2
  33. package/app-template/src/views/category/category-info.tsx +0 -1
  34. package/app-template/src/views/category/filters/index.tsx +1 -1
  35. package/app-template/src/views/checkout/summary.tsx +0 -10
  36. package/app-template/src/views/guest-login/index.tsx +1 -6
  37. package/app-template/src/views/header/search/index.tsx +5 -17
  38. package/app-template/src/views/product/product-info.tsx +263 -62
  39. package/app-template/src/views/product/slider.tsx +73 -86
  40. package/app-template/src/widgets/footer-menu.tsx +2 -6
  41. package/commands/plugins.ts +16 -63
  42. package/dist/commands/plugins.js +16 -57
  43. package/package.json +1 -1
  44. package/app-template/src/app/[commerce]/[locale]/[currency]/product/[pk]/loading.tsx +0 -67
  45. package/app-template/src/app/api/image-proxy/route.ts +0 -1
  46. package/app-template/src/app/api/similar-product-list/route.ts +0 -1
  47. package/app-template/src/app/api/similar-products/route.ts +0 -1
  48. package/app-template/src/hooks/use-product-cart.ts +0 -77
  49. package/app-template/src/hooks/use-stock-alert.ts +0 -74
  50. package/app-template/src/utils/variant-validation.ts +0 -41
  51. package/app-template/src/views/basket/basket-content.tsx +0 -106
  52. package/app-template/src/views/checkout/steps/payment/options/store-credit.tsx +0 -121
  53. package/app-template/src/views/product/product-actions.tsx +0 -165
  54. package/app-template/src/views/product/product-share.tsx +0 -56
  55. package/app-template/src/views/product/product-variants.tsx +0 -26
@@ -1,121 +0,0 @@
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
- };
@@ -1,165 +0,0 @@
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
- };
@@ -1,56 +0,0 @@
1
- import React, { useState, useEffect } from 'react';
2
- import Share from '@theme/views/share';
3
- import { useLocalization } from '@akinon/next/hooks';
4
-
5
- interface ProductShareProps {
6
- productName: string;
7
- className?: string;
8
- }
9
-
10
- export const ProductShare: React.FC<ProductShareProps> = ({
11
- productName,
12
- className
13
- }) => {
14
- const { t } = useLocalization();
15
- const [currentUrl, setCurrentUrl] = useState<string | null>(null);
16
-
17
- useEffect(() => {
18
- setCurrentUrl(window.location.href);
19
- }, []);
20
-
21
- if (!currentUrl) {
22
- return null;
23
- }
24
-
25
- const shareItems = [
26
- {
27
- href: `https://www.facebook.com/sharer/sharer.php?u=${encodeURIComponent(
28
- currentUrl
29
- )}`,
30
- iconName: 'facebook' as const,
31
- iconSize: 22
32
- },
33
- {
34
- href: `https://twitter.com/intent/tweet?text=${encodeURIComponent(
35
- currentUrl
36
- )}`,
37
- iconName: 'twitter' as const,
38
- iconSize: 22
39
- },
40
- {
41
- href: `https://api.whatsapp.com/send?text=${productName}%20${encodeURIComponent(
42
- currentUrl
43
- )}`,
44
- iconName: 'whatsapp' as const,
45
- iconSize: 22
46
- }
47
- ];
48
-
49
- return (
50
- <Share
51
- className={className}
52
- buttonText={t('product.share')}
53
- items={shareItems}
54
- />
55
- );
56
- };
@@ -1,26 +0,0 @@
1
- import React from 'react';
2
- import { Variant } from '@theme/views/product';
3
- import { VariantType } from '@akinon/next/types';
4
-
5
- interface ProductVariantsProps {
6
- variants: VariantType[];
7
- onVariantChange: () => void;
8
- }
9
-
10
- export const ProductVariants: React.FC<ProductVariantsProps> = ({
11
- variants,
12
- onVariantChange
13
- }) => {
14
- return (
15
- <div className="flex flex-col">
16
- {variants.map((variant) => (
17
- <Variant
18
- key={variant.attribute_key}
19
- {...variant}
20
- className="items-center mt-8"
21
- onChange={onVariantChange}
22
- />
23
- ))}
24
- </div>
25
- );
26
- };