@luxfi/core 4.4.14 → 4.4.16

Sign up to get free protection for your applications and to get access to all the features.
Files changed (84) hide show
  1. package/components/access-code-input.tsx +71 -71
  2. package/components/auth/auth-listener.tsx +29 -29
  3. package/components/auth/auth-token/clear-auth-token.tsx +12 -12
  4. package/components/auth/auth-token/set-auth-token.tsx +16 -16
  5. package/components/auth/common-auth-domains.ts +16 -16
  6. package/components/auth/login-panel.tsx +103 -98
  7. package/components/chat-widget.tsx +77 -77
  8. package/components/commerce/bag-button.tsx +67 -67
  9. package/components/commerce/checkout-panel/close-button.tsx +26 -26
  10. package/components/commerce/checkout-panel/dt-bag-carousel.tsx +36 -36
  11. package/components/commerce/checkout-panel/dt-checkout-panel.tsx +66 -66
  12. package/components/commerce/checkout-panel/index.tsx +123 -123
  13. package/components/commerce/checkout-panel/links-row.tsx +21 -21
  14. package/components/commerce/checkout-panel/mb-checkout-panel.tsx +55 -55
  15. package/components/commerce/checkout-panel/steps-indicator.tsx +39 -39
  16. package/components/commerce/checkout-panel/thank-you.tsx +18 -18
  17. package/components/commerce/desktop-bag-popup.tsx +78 -78
  18. package/components/commerce/mobile-bag-drawer.tsx +51 -51
  19. package/components/commerce/mobile-menu-toggle-button.tsx +35 -35
  20. package/components/commerce/mobile-nav-menu.tsx +64 -64
  21. package/components/contact-dialog/contact-form.tsx +112 -112
  22. package/components/contact-dialog/disclaimer.tsx +13 -13
  23. package/components/contact-dialog/index.tsx +64 -64
  24. package/components/copyright.tsx +21 -21
  25. package/components/footer.tsx +77 -77
  26. package/components/header/desktop.tsx +54 -54
  27. package/components/header/index.tsx +26 -26
  28. package/components/header/mobile.tsx +161 -161
  29. package/components/header/theme-toggle.tsx +26 -26
  30. package/components/icons/bag-icon.tsx +10 -10
  31. package/components/icons/github.tsx +14 -14
  32. package/components/icons/index.tsx +35 -35
  33. package/components/icons/lux-logo.tsx +10 -10
  34. package/components/icons/secure-delivery.tsx +13 -13
  35. package/components/icons/social-icon.tsx +35 -35
  36. package/components/icons/social-svg.css +3 -3
  37. package/components/icons/youtube-logo.tsx +59 -59
  38. package/components/index.ts +27 -27
  39. package/components/logo.tsx +81 -81
  40. package/components/mini-chart/index.tsx +7 -7
  41. package/components/mini-chart/mini-chart-props.ts +43 -43
  42. package/components/mini-chart/mini-chart.tsx +85 -85
  43. package/components/mini-chart/wrapper.tsx +23 -23
  44. package/components/not-found/index.tsx +27 -27
  45. package/components/not-found/not-found-content.mdx +5 -5
  46. package/components/root-layout.tsx +71 -71
  47. package/components/scripts.tsx +23 -23
  48. package/conf/index.ts +50 -50
  49. package/environment.d.ts +5 -5
  50. package/next/analytics/fpixel.ts +15 -15
  51. package/next/analytics/google-analytics.ts +13 -13
  52. package/next/analytics/index.ts +3 -3
  53. package/next/analytics/pixel-analytics.tsx +54 -54
  54. package/next/determine-device-mw.ts +16 -16
  55. package/next/font/get-app-router-font-classes.ts +12 -12
  56. package/next/font/load-and-return-lux-next-fonts-on-import.ts +68 -68
  57. package/next/font/next-font-desc.ts +27 -27
  58. package/next/font/pages-router-font-vars.tsx +18 -18
  59. package/next/head-metadata/from-next/metadata-types.ts +158 -158
  60. package/next/head-metadata/from-next/opengraph-types.ts +267 -267
  61. package/next/head-metadata/from-next/twitter-types.ts +92 -92
  62. package/next/head-metadata/index.tsx +208 -208
  63. package/package.json +73 -73
  64. package/server-actions/firebase-app.ts +14 -14
  65. package/server-actions/index.ts +5 -5
  66. package/server-actions/store-contact.ts +51 -51
  67. package/site-def/footer/community.tsx +67 -67
  68. package/site-def/footer/company.ts +37 -37
  69. package/site-def/footer/ecosystem.ts +37 -37
  70. package/site-def/footer/index.tsx +26 -26
  71. package/site-def/footer/legal.ts +28 -28
  72. package/site-def/footer/network.ts +45 -45
  73. package/site-def/footer/svg/warpcast-logo.svg +11 -11
  74. package/site-def/index.ts +2 -2
  75. package/site-def/main-nav.ts +35 -35
  76. package/site-def/site-def.ts +36 -36
  77. package/style/lux-colors.css +85 -85
  78. package/style/lux-global.css +30 -30
  79. package/tailwind/fontFamily.tailwind.lux.ts +18 -18
  80. package/tailwind/index.ts +2 -2
  81. package/tailwind/lux-tw-fonts.ts +39 -39
  82. package/tailwind/tailwind.config.lux-preset.ts +10 -10
  83. package/tsconfig.json +10 -10
  84. package/types/contact-info.ts +10 -10
@@ -1,67 +1,67 @@
1
- 'use client'
2
- import React from 'react'
3
- import { observer } from 'mobx-react-lite'
4
-
5
- import { buttonVariants, type ButtonSizes } from '@hanzo/ui/primitives'
6
- import { cn } from '@hanzo/ui/util'
7
- import { useCommerce } from '@hanzo/commerce'
8
-
9
- import * as Icons from '../icons'
10
-
11
- const BagButton: React.FC<{
12
- showIfEmpty?: boolean
13
- noHoverEffects?: boolean
14
- size?: ButtonSizes
15
- className?: string
16
- iconClx?: string
17
- onClick?: () => void
18
- }> = observer(({
19
- showIfEmpty=false,
20
- noHoverEffects=false,
21
- size='default',
22
- className='',
23
- iconClx='',
24
- onClick
25
- }) => {
26
-
27
- const c = useCommerce()
28
-
29
- // undefined means context is not installed, ie commerce functions are not in use
30
- if (!c || (!showIfEmpty && c.cartEmpty)) {
31
- return <div /> // trigger code needs non-null
32
- }
33
-
34
- return (
35
- <div
36
- aria-label="Bag"
37
- role='button'
38
- onClick={onClick}
39
- className={cn(
40
- buttonVariants({ variant: 'ghost', size, rounded: 'md' }),
41
- // Overides the bg change on hover --not a "hover effect"
42
- 'relative group p-0 aspect-square hover:bg-background',
43
- className
44
- )}
45
- >
46
- {!c.cartEmpty && (
47
- <div className={
48
- 'z-above-content flex flex-col justify-center items-center ' +
49
- 'absolute left-0 right-0 top-0 bottom-0 ' +
50
- 'leading-none font-sans font-bold text-primary-fg text-xs '
51
- }>
52
- <div className='h-[3px] w-full' />
53
- <div>{c.cartQuantity}</div>
54
- </div>
55
- )}
56
- <Icons.bag className={cn(
57
- 'relative -top-[3px] fill-primary w-6 h-7 ',
58
- iconClx,
59
- (noHoverEffects ? '' : (
60
- 'group-hover:fill-primary-hover group-hover:scale-105 transition-scale transition-duration-300'
61
- ))
62
- )} aria-hidden="true" />
63
- </div>
64
- )
65
- })
66
-
67
- export default BagButton
1
+ 'use client'
2
+ import React from 'react'
3
+ import { observer } from 'mobx-react-lite'
4
+
5
+ import { buttonVariants, type ButtonSizes } from '@hanzo/ui/primitives'
6
+ import { cn } from '@hanzo/ui/util'
7
+ import { useCommerce } from '@hanzo/commerce'
8
+
9
+ import * as Icons from '../icons'
10
+
11
+ const BagButton: React.FC<{
12
+ showIfEmpty?: boolean
13
+ noHoverEffects?: boolean
14
+ size?: ButtonSizes
15
+ className?: string
16
+ iconClx?: string
17
+ onClick?: () => void
18
+ }> = observer(({
19
+ showIfEmpty=false,
20
+ noHoverEffects=false,
21
+ size='default',
22
+ className='',
23
+ iconClx='',
24
+ onClick
25
+ }) => {
26
+
27
+ const c = useCommerce()
28
+
29
+ // undefined means context is not installed, ie commerce functions are not in use
30
+ if (!c || (!showIfEmpty && c.cartEmpty)) {
31
+ return <div /> // trigger code needs non-null
32
+ }
33
+
34
+ return (
35
+ <div
36
+ aria-label="Bag"
37
+ role='button'
38
+ onClick={onClick}
39
+ className={cn(
40
+ buttonVariants({ variant: 'ghost', size, rounded: 'md' }),
41
+ // Overides the bg change on hover --not a "hover effect"
42
+ 'relative group p-0 aspect-square hover:bg-background',
43
+ className
44
+ )}
45
+ >
46
+ {!c.cartEmpty && (
47
+ <div className={
48
+ 'z-above-content flex flex-col justify-center items-center ' +
49
+ 'absolute left-0 right-0 top-0 bottom-0 ' +
50
+ 'leading-none font-sans font-bold text-primary-fg text-xs '
51
+ }>
52
+ <div className='h-[3px] w-full' />
53
+ <div>{c.cartQuantity}</div>
54
+ </div>
55
+ )}
56
+ <Icons.bag className={cn(
57
+ 'relative -top-[3px] fill-primary w-6 h-7 ',
58
+ iconClx,
59
+ (noHoverEffects ? '' : (
60
+ 'group-hover:fill-primary-hover group-hover:scale-105 transition-scale transition-duration-300'
61
+ ))
62
+ )} aria-hidden="true" />
63
+ </div>
64
+ )
65
+ })
66
+
67
+ export default BagButton
@@ -1,26 +1,26 @@
1
- 'use client'
2
- import React from 'react'
3
-
4
- import { cn } from '@hanzo/ui/util'
5
-
6
- import Logo from '../../logo'
7
- import type { TShirtSize } from '@hanzo/ui/types'
8
-
9
- const CloseButton: React.FC<{
10
- close: () => void
11
- size?: TShirtSize
12
- className?: string
13
- }> = ({
14
- close,
15
- size,
16
- className=''
17
- }) => (
18
- <div
19
- onClick={close}
20
- className={cn('md:self-start', className)}
21
- >
22
- <Logo layout='text-only' href='/' size={size}/>
23
- </div>
24
- )
25
-
26
- export default CloseButton
1
+ 'use client'
2
+ import React from 'react'
3
+
4
+ import { cn } from '@hanzo/ui/util'
5
+
6
+ import Logo from '../../logo'
7
+ import type { TShirtSize } from '@hanzo/ui/types'
8
+
9
+ const CloseButton: React.FC<{
10
+ close: () => void
11
+ size?: TShirtSize
12
+ className?: string
13
+ }> = ({
14
+ close,
15
+ size,
16
+ className=''
17
+ }) => (
18
+ <div
19
+ onClick={close}
20
+ className={cn('md:self-start', className)}
21
+ >
22
+ <Logo layout='text-only' href='/' size={size}/>
23
+ </div>
24
+ )
25
+
26
+ export default CloseButton
@@ -1,36 +1,36 @@
1
- 'use client'
2
- import React from 'react'
3
- import { observer } from 'mobx-react-lite'
4
-
5
- import {
6
- useCommerce,
7
- CarouselItemSelector,
8
- type CarouselItemSelectorPropsExt
9
- } from '@hanzo/commerce'
10
-
11
- const DesktopBagCarousel: React.FC<{
12
- constrainTo: {w: number, h: number}
13
- className?: string
14
- }> = observer(({
15
- constrainTo,
16
- className=''
17
-
18
- }) => {
19
- const cmmc = useCommerce()
20
- return (
21
- <CarouselItemSelector
22
- items={cmmc.cartItems}
23
- selectedItemRef={cmmc}
24
- scrollable={false} // ignored
25
- selectSku={cmmc.setCurrentItem.bind(cmmc)}
26
- clx={className}
27
- ext={{
28
- options: {loop: true},
29
- constrainTo,
30
- imageOnly: true
31
- } satisfies CarouselItemSelectorPropsExt}
32
- />
33
- )
34
- })
35
-
36
- export default DesktopBagCarousel
1
+ 'use client'
2
+ import React from 'react'
3
+ import { observer } from 'mobx-react-lite'
4
+
5
+ import {
6
+ useCommerce,
7
+ CarouselItemSelector,
8
+ type CarouselItemSelectorPropsExt
9
+ } from '@hanzo/commerce'
10
+
11
+ const DesktopBagCarousel: React.FC<{
12
+ constrainTo: {w: number, h: number}
13
+ className?: string
14
+ }> = observer(({
15
+ constrainTo,
16
+ className=''
17
+
18
+ }) => {
19
+ const cmmc = useCommerce()
20
+ return (
21
+ <CarouselItemSelector
22
+ items={cmmc.cartItems}
23
+ selectedItemRef={cmmc}
24
+ scrollable={false} // ignored
25
+ selectSku={cmmc.setCurrentItem.bind(cmmc)}
26
+ clx={className}
27
+ ext={{
28
+ options: {loop: true},
29
+ constrainTo,
30
+ imageOnly: true
31
+ } satisfies CarouselItemSelectorPropsExt}
32
+ />
33
+ )
34
+ })
35
+
36
+ export default DesktopBagCarousel
@@ -1,67 +1,67 @@
1
- 'use client'
2
- import React, { type PropsWithChildren } from 'react'
3
-
4
- import { ScrollArea, StepIndicator } from '@hanzo/ui/primitives'
5
- import { AuthWidget } from '@hanzo/auth/components'
6
- import { CartPanel } from '@hanzo/commerce'
7
-
8
- import * as Icons from '../../icons'
9
- import DesktopBagCarousel from './dt-bag-carousel'
10
- import CloseButton from './close-button'
11
- import { cn } from '@hanzo/ui/util'
12
- import LinksRow from './links-row'
13
-
14
- const DesktopCheckoutPanel: React.FC<PropsWithChildren & {
15
- index: number
16
- stepNames: string[]
17
- close:() => void
18
- className?: string
19
- }> = ({
20
- index,
21
- stepNames,
22
- close,
23
- className='',
24
- children
25
- }) => (
26
-
27
- <div /* id='CHECKOUT_PANEL' */ className={cn('grid grid-cols-2', className)}>
28
- <div className='w-full h-full bg-background flex flex-row items-start justify-end'>
29
- <div className='w-full max-w-[750px] relative flex flex-col items-center justify-start p-8'>
30
- <CloseButton close={close} size='md'/>
31
- <div className='w-full max-w-[550px] mx-auto flex flex-col gap-3'>
32
- <DesktopBagCarousel className='h-[260px] w-[360px] lg:w-[420px] mx-auto -mt-8' constrainTo={{w: 250, h: 250}}/>
33
- <CartPanel
34
- className='w-full border-none p-0'
35
- itemClx='mb-3'
36
- totalClx='sticky -bottom-1 p-1 bg-background'
37
- listClx='pr-3'
38
- scrollAfter={5}
39
- scrollHeightClx='h-[50vh]'
40
- showPromoCode
41
- showShipping
42
- />
43
- </div>
44
- </div>
45
- </div>
46
- <div className='w-full h-full flex flex-col bg-level-1 min-h-screen justify-between'>
47
- <ScrollArea className='w-full flex flex-row items-start justify-start overflow-y-auto'>
48
- <div className='h-full w-full max-w-[750px] relative flex flex-col items-center px-8 pt-0'>
49
- <div className='bg-level-1 sticky h-30 pb-8 w-full top-0 flex justify-center items-end'>
50
- <AuthWidget noLogin className='hidden md:flex absolute top-4 right-4 '/>
51
- <StepIndicator dotSizeRem={1.5} steps={stepNames} currentStep={index} className='gap-2 text-base w-pr-70' />
52
- </div>
53
- <div className='w-full max-w-[550px] mx-auto pb-10'>
54
- {children}
55
- </div>
56
- </div>
57
- </ScrollArea>
58
- <div className='w-full max-w-[750px] relative flex flex-col items-center px-8 pt-0'>
59
- <div className='w-full max-w-[550px] mx-auto flex flex-col items-center'>
60
- <LinksRow className='w-full' />
61
- </div>
62
- </div>
63
- </div>
64
- </div>
65
- )
66
-
1
+ 'use client'
2
+ import React, { type PropsWithChildren } from 'react'
3
+
4
+ import { ScrollArea, StepIndicator } from '@hanzo/ui/primitives'
5
+ import { AuthWidget } from '@hanzo/auth/components'
6
+ import { CartPanel } from '@hanzo/commerce'
7
+
8
+ import * as Icons from '../../icons'
9
+ import DesktopBagCarousel from './dt-bag-carousel'
10
+ import CloseButton from './close-button'
11
+ import { cn } from '@hanzo/ui/util'
12
+ import LinksRow from './links-row'
13
+
14
+ const DesktopCheckoutPanel: React.FC<PropsWithChildren & {
15
+ index: number
16
+ stepNames: string[]
17
+ close:() => void
18
+ className?: string
19
+ }> = ({
20
+ index,
21
+ stepNames,
22
+ close,
23
+ className='',
24
+ children
25
+ }) => (
26
+
27
+ <div /* id='CHECKOUT_PANEL' */ className={cn('grid grid-cols-2', className)}>
28
+ <div className='w-full h-full bg-background flex flex-row items-start justify-end'>
29
+ <div className='w-full max-w-[750px] relative flex flex-col items-center justify-start p-8'>
30
+ <CloseButton close={close} size='md'/>
31
+ <div className='w-full max-w-[550px] mx-auto flex flex-col gap-3'>
32
+ <DesktopBagCarousel className='h-[260px] w-[360px] lg:w-[420px] mx-auto -mt-8' constrainTo={{w: 250, h: 250}}/>
33
+ <CartPanel
34
+ className='w-full border-none p-0'
35
+ itemClx='mb-3'
36
+ totalClx='sticky -bottom-1 p-1 bg-background'
37
+ listClx='pr-3'
38
+ scrollAfter={5}
39
+ scrollHeightClx='h-[50vh]'
40
+ showPromoCode
41
+ showShipping
42
+ />
43
+ </div>
44
+ </div>
45
+ </div>
46
+ <div className='w-full h-full flex flex-col bg-level-1 min-h-screen justify-between'>
47
+ <ScrollArea className='w-full flex flex-row items-start justify-start overflow-y-auto'>
48
+ <div className='h-full w-full max-w-[750px] relative flex flex-col items-center px-8 pt-0'>
49
+ <div className='bg-level-1 sticky h-30 pb-8 w-full top-0 flex justify-center items-end'>
50
+ <AuthWidget noLogin className='hidden md:flex absolute top-4 right-4 '/>
51
+ <StepIndicator dotSizeRem={1.5} steps={stepNames} currentStep={index} className='gap-2 text-base w-pr-70' />
52
+ </div>
53
+ <div className='w-full max-w-[550px] mx-auto pb-10'>
54
+ {children}
55
+ </div>
56
+ </div>
57
+ </ScrollArea>
58
+ <div className='w-full max-w-[750px] relative flex flex-col items-center px-8 pt-0'>
59
+ <div className='w-full max-w-[550px] mx-auto flex flex-col items-center'>
60
+ <LinksRow className='w-full' />
61
+ </div>
62
+ </div>
63
+ </div>
64
+ </div>
65
+ )
66
+
67
67
  export default DesktopCheckoutPanel
@@ -1,124 +1,124 @@
1
- 'use client'
2
- import React, { useEffect, useRef, useState } from 'react'
3
-
4
- import { capitalize, cn } from '@hanzo/ui/util'
5
-
6
- import { useCommerce, ShippingStepForm, PaymentStepForm } from '@hanzo/commerce'
7
- import type { CheckoutStep } from '@hanzo/commerce/types'
8
-
9
- import ThankYou from './thank-you'
10
-
11
- const STEPS = [
12
- {
13
- name: 'payment',
14
- Comp: PaymentStepForm
15
- },
16
- {
17
- name: 'delivery',
18
- Comp: ShippingStepForm
19
- },
20
- {
21
- name: 'done',
22
- label: 'Done!',
23
- Comp: ThankYou
24
- }
25
- ] satisfies CheckoutStep[]
26
-
27
- const STEP_NAMES = STEPS.map((s) => (s.label ? s.label : capitalize(s.name)))
28
-
29
- import DesktopCP from './dt-checkout-panel'
30
- import MobileCP from './mb-checkout-panel'
31
-
32
- const CheckoutPanel: React.FC<{
33
- close: () => void
34
- className?: string
35
- }> = ({
36
- close,
37
- className=''
38
- }) => {
39
-
40
- const cmmc = useCommerce()
41
-
42
- // For sites that don't initialize cmmc
43
- if (!cmmc) {
44
- return <></>
45
- }
46
- const [stepIndex, setStepIndex] = useState<number>(0)
47
- const [orderId, setOrderId] = useState<string | undefined>(undefined)
48
-
49
- // Step.name or 'first' or 'next' or 'last'
50
- const setStep = (name: string): void => {
51
-
52
- if (name === 'first') {
53
- setStepIndex(0)
54
- }
55
- else if (name === 'last') {
56
- setStepIndex(STEPS.length - 1)
57
- }
58
- else if (name === 'next') {
59
- if (stepIndex <= STEPS.length - 2) {
60
- setStepIndex(stepIndex + 1)
61
- }
62
- else {
63
- throw new Error('CheckoutPanel.setStep(): Attempting to advance past last step!')
64
- }
65
- }
66
- else {
67
- const indexFound = STEPS.findIndex((el) => (el.name === name))
68
- if (indexFound !== -1) {
69
- setStepIndex(indexFound)
70
- }
71
- else {
72
- throw new Error('CheckoutPanel.setStep(): Step named ' + name + ' not found!')
73
- }
74
- }
75
- }
76
-
77
- const _close = () => {
78
- setStep('first')
79
- close()
80
- }
81
-
82
- // Determine if mobile or desktop based on visibility of desktopElement
83
- // https://stackoverflow.com/a/21696585/11378853
84
- const desktopElement = useRef<HTMLDivElement | null>(null)
85
- const [layout, setLayout] = useState<'mobile' | 'desktop' | undefined>()
86
- useEffect(() => {
87
- const checkLayout = () => {
88
- setLayout(!!desktopElement.current?.offsetParent ? 'desktop' : 'mobile')
89
- }
90
-
91
- // initial layout check
92
- checkLayout()
93
-
94
- window.addEventListener('resize', checkLayout)
95
- return () => {
96
- window.removeEventListener('resize', checkLayout)
97
- }
98
- }, [])
99
-
100
- const StepToRender = STEPS[stepIndex].Comp
101
-
102
- return (<>
103
- <DesktopCP
104
- className={cn('h-full', className, 'hidden md:flex')}
105
- close={_close}
106
- index={stepIndex}
107
- stepNames={STEP_NAMES}
108
- >
109
- {/* Element required to determine if DesktopCP is visible */}
110
- <div ref={desktopElement}/>
111
- {layout === 'desktop' && <StepToRender onDone={() => {setStep('next')}} orderId={orderId} setOrderId={setOrderId}/>}
112
- </DesktopCP>
113
- <MobileCP
114
- className={cn('h-full overflow-y-auto', className, 'md:hidden' )}
115
- close={_close}
116
- index={stepIndex}
117
- stepNames={STEP_NAMES}
118
- >
119
- {layout === 'mobile' && <StepToRender onDone={() => {setStep('next')}} orderId={orderId} setOrderId={setOrderId}/>}
120
- </MobileCP>
121
- </>)
122
- }
123
-
1
+ 'use client'
2
+ import React, { useEffect, useRef, useState } from 'react'
3
+
4
+ import { capitalize, cn } from '@hanzo/ui/util'
5
+
6
+ import { useCommerce, ShippingStepForm, PaymentStepForm } from '@hanzo/commerce'
7
+ import type { CheckoutStep } from '@hanzo/commerce/types'
8
+
9
+ import ThankYou from './thank-you'
10
+
11
+ const STEPS = [
12
+ {
13
+ name: 'payment',
14
+ Comp: PaymentStepForm
15
+ },
16
+ {
17
+ name: 'delivery',
18
+ Comp: ShippingStepForm
19
+ },
20
+ {
21
+ name: 'done',
22
+ label: 'Done!',
23
+ Comp: ThankYou
24
+ }
25
+ ] satisfies CheckoutStep[]
26
+
27
+ const STEP_NAMES = STEPS.map((s) => (s.label ? s.label : capitalize(s.name)))
28
+
29
+ import DesktopCP from './dt-checkout-panel'
30
+ import MobileCP from './mb-checkout-panel'
31
+
32
+ const CheckoutPanel: React.FC<{
33
+ close: () => void
34
+ className?: string
35
+ }> = ({
36
+ close,
37
+ className=''
38
+ }) => {
39
+
40
+ const cmmc = useCommerce()
41
+
42
+ // For sites that don't initialize cmmc
43
+ if (!cmmc) {
44
+ return <></>
45
+ }
46
+ const [stepIndex, setStepIndex] = useState<number>(0)
47
+ const [orderId, setOrderId] = useState<string | undefined>(undefined)
48
+
49
+ // Step.name or 'first' or 'next' or 'last'
50
+ const setStep = (name: string): void => {
51
+
52
+ if (name === 'first') {
53
+ setStepIndex(0)
54
+ }
55
+ else if (name === 'last') {
56
+ setStepIndex(STEPS.length - 1)
57
+ }
58
+ else if (name === 'next') {
59
+ if (stepIndex <= STEPS.length - 2) {
60
+ setStepIndex(stepIndex + 1)
61
+ }
62
+ else {
63
+ throw new Error('CheckoutPanel.setStep(): Attempting to advance past last step!')
64
+ }
65
+ }
66
+ else {
67
+ const indexFound = STEPS.findIndex((el) => (el.name === name))
68
+ if (indexFound !== -1) {
69
+ setStepIndex(indexFound)
70
+ }
71
+ else {
72
+ throw new Error('CheckoutPanel.setStep(): Step named ' + name + ' not found!')
73
+ }
74
+ }
75
+ }
76
+
77
+ const _close = () => {
78
+ setStep('first')
79
+ close()
80
+ }
81
+
82
+ // Determine if mobile or desktop based on visibility of desktopElement
83
+ // https://stackoverflow.com/a/21696585/11378853
84
+ const desktopElement = useRef<HTMLDivElement | null>(null)
85
+ const [layout, setLayout] = useState<'mobile' | 'desktop' | undefined>()
86
+ useEffect(() => {
87
+ const checkLayout = () => {
88
+ setLayout(!!desktopElement.current?.offsetParent ? 'desktop' : 'mobile')
89
+ }
90
+
91
+ // initial layout check
92
+ checkLayout()
93
+
94
+ window.addEventListener('resize', checkLayout)
95
+ return () => {
96
+ window.removeEventListener('resize', checkLayout)
97
+ }
98
+ }, [])
99
+
100
+ const StepToRender = STEPS[stepIndex].Comp
101
+
102
+ return (<>
103
+ <DesktopCP
104
+ className={cn('h-full', className, 'hidden md:flex')}
105
+ close={_close}
106
+ index={stepIndex}
107
+ stepNames={STEP_NAMES}
108
+ >
109
+ {/* Element required to determine if DesktopCP is visible */}
110
+ <div ref={desktopElement}/>
111
+ {layout === 'desktop' && <StepToRender onDone={() => {setStep('next')}} orderId={orderId} setOrderId={setOrderId}/>}
112
+ </DesktopCP>
113
+ <MobileCP
114
+ className={cn('h-full overflow-y-auto', className, 'md:hidden' )}
115
+ close={_close}
116
+ index={stepIndex}
117
+ stepNames={STEP_NAMES}
118
+ >
119
+ {layout === 'mobile' && <StepToRender onDone={() => {setStep('next')}} orderId={orderId} setOrderId={setOrderId}/>}
120
+ </MobileCP>
121
+ </>)
122
+ }
123
+
124
124
  export default CheckoutPanel