3
3
import { PlusIcon } from '@heroicons/react/24/outline' ;
4
4
import clsx from 'clsx' ;
5
5
import { addItem } from 'components/cart/actions' ;
6
- import LoadingDots from 'components/loading-dots ' ;
6
+ import { useProduct } from 'components/product/product-context ' ;
7
7
import { Product , ProductVariant } from 'lib/shopware/types' ;
8
- import { useSearchParams } from 'next/navigation' ;
9
8
import { useActionState } from 'react' ;
10
- import { useFormStatus } from 'react-dom' ;
9
+ import { useCart } from './cart-context' ;
10
+
11
11
function SubmitButton ( {
12
12
availableForSale,
13
13
selectedVariantId
14
14
} : {
15
15
availableForSale : boolean ;
16
16
selectedVariantId : string | undefined ;
17
17
} ) {
18
- const { pending } = useFormStatus ( ) ;
19
18
const buttonClasses =
20
19
'relative flex w-full items-center justify-center rounded-full bg-blue-600 p-4 tracking-wide text-white' ;
21
20
const disabledClasses = 'cursor-not-allowed opacity-60 hover:opacity-60' ;
22
21
23
22
if ( ! availableForSale ) {
24
23
return (
25
- < button aria- disabled className = { clsx ( buttonClasses , disabledClasses ) } >
24
+ < button disabled className = { clsx ( buttonClasses , disabledClasses ) } >
26
25
Out Of Stock
27
26
</ button >
28
27
) ;
@@ -32,7 +31,7 @@ function SubmitButton({
32
31
return (
33
32
< button
34
33
aria-label = "Please select an option"
35
- aria- disabled
34
+ disabled
36
35
className = { clsx ( buttonClasses , disabledClasses ) }
37
36
>
38
37
< div className = "absolute left-0 ml-4" >
@@ -45,52 +44,44 @@ function SubmitButton({
45
44
46
45
return (
47
46
< button
48
- onClick = { ( e : React . FormEvent < HTMLButtonElement > ) => {
49
- if ( pending ) e . preventDefault ( ) ;
50
- } }
51
47
aria-label = "Add to cart"
52
- aria-disabled = { pending }
53
48
className = { clsx ( buttonClasses , {
54
- 'hover:opacity-90' : true ,
55
- [ disabledClasses ] : pending
49
+ 'hover:opacity-90' : true
56
50
} ) }
57
51
>
58
52
< div className = "absolute left-0 ml-4" >
59
- { pending ? < LoadingDots className = "mb-3 bg-white" /> : < PlusIcon className = "h-5" /> }
53
+ < PlusIcon className = "h-5" />
60
54
</ div >
61
55
Add To Cart
62
56
</ button >
63
57
) ;
64
58
}
65
59
66
- export function AddToCart ( {
67
- product,
68
- variants,
69
- availableForSale
70
- } : {
71
- product : Product ;
72
- variants : ProductVariant [ ] ;
73
- availableForSale : boolean ;
74
- } ) {
60
+ export function AddToCart ( { product } : { product : Product } ) {
61
+ const { variants, availableForSale } = product ;
62
+ const { addCartItem } = useCart ( ) ;
63
+ const { state } = useProduct ( ) ;
75
64
const [ message , formAction ] = useActionState ( addItem , null ) ;
76
- const searchParams = useSearchParams ( ) ;
77
- const defaultVariantId = variants . length === 1 ? variants [ 0 ] ?. id : product . id ;
78
- const variant = variants . find ( ( variant ) =>
79
- variant . selectedOptions . every (
80
- ( option ) => option . value === searchParams . get ( option . name . toLowerCase ( ) )
81
- )
65
+
66
+ const variant = variants . find ( ( variant : ProductVariant ) =>
67
+ variant . selectedOptions . every ( ( option ) => option . value === state [ option . name . toLowerCase ( ) ] )
82
68
) ;
69
+ const defaultVariantId = variants . length === 1 ? variants [ 0 ] ?. id : product . id ;
83
70
const selectedVariantId = variant ?. id || defaultVariantId ;
84
71
const actionWithVariant = formAction . bind ( null , selectedVariantId ) ;
72
+ const finalVariant = variants . find ( ( variant ) => variant . id === selectedVariantId ) ! || product ;
85
73
86
74
return (
87
- < form action = { actionWithVariant } >
75
+ < form
76
+ action = { async ( ) => {
77
+ addCartItem ( finalVariant , product ) ;
78
+ await actionWithVariant ( ) ;
79
+ } }
80
+ >
88
81
< SubmitButton availableForSale = { availableForSale } selectedVariantId = { selectedVariantId } />
89
- < div className = "flex items-center px-4 py-3 text-sm font-bold text-black" >
90
- < p aria-live = "polite" className = "h-6" role = "status" >
91
- { message }
92
- </ p >
93
- </ div >
82
+ < p aria-live = "polite" className = "sr-only" role = "status" >
83
+ { message }
84
+ </ p >
94
85
</ form >
95
86
) ;
96
87
}
0 commit comments