Skip to content

Commit c3f895d

Browse files
authored
feat: redesign carousel (#3177)
1 parent ded1c53 commit c3f895d

File tree

80 files changed

+1334
-1065
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

80 files changed

+1334
-1065
lines changed

bun.lockb

10.1 KB
Binary file not shown.
+93-4
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,94 @@
1-
[data-scope='carousel'][data-part='viewport'] {
2-
max-width: 600px;
3-
margin-top: 40px;
4-
overflow-x: hidden;
1+
[data-scope="carousel"][data-part="root"] {
2+
display: flex;
3+
align-items: flex-start;
4+
flex-direction: column;
5+
gap: 16px;
6+
}
7+
8+
[data-scope="carousel"][data-part="root"][data-orientation="horizontal"] {
9+
max-width: 400px;
10+
}
11+
12+
[data-scope="carousel"][data-part="root"][data-orientation="vertical"] {
13+
max-height: 400px;
14+
}
15+
16+
[data-scope="carousel"][data-part="control"] {
17+
display: flex;
18+
gap: 8px;
19+
align-self: stretch;
20+
}
21+
22+
[data-scope="carousel"][data-part="item-group"] {
23+
align-self: stretch;
24+
/* Hide scrollbar */
25+
scrollbar-width: none;
26+
-webkit-scrollbar-width: none;
27+
-ms-overflow-style: none;
28+
&::-webkit-scrollbar {
29+
display: none;
30+
}
31+
}
32+
33+
[data-scope="carousel"][data-part="item"] {
34+
display: flex;
35+
justify-content: center;
36+
align-items: center;
37+
font-size: 24px;
38+
border: 1px solid lightgray;
39+
}
40+
41+
[data-scope="carousel"][data-part="item"] img {
42+
margin: 0;
43+
object-fit: cover;
44+
}
45+
46+
[data-scope="carousel"][data-part="item"][data-orientation="horizontal"] img {
47+
height: 300px;
48+
width: 100%;
49+
}
50+
51+
[data-scope="carousel"][data-part="item"][data-orientation="vertical"] img {
52+
height: 100%;
53+
width: 400px;
54+
}
55+
56+
[data-scope="carousel"][data-part="indicator-group"] {
57+
display: flex;
58+
justify-content: center;
59+
align-items: center;
60+
margin-block: 8px;
61+
gap: 8px;
62+
}
63+
64+
[data-scope="carousel"][data-part="indicator"] {
65+
display: flex;
66+
border: none;
67+
justify-content: center;
68+
align-items: center;
69+
height: 12px;
70+
width: 12px;
71+
border-radius: 6px;
72+
cursor: pointer;
73+
background-color: lightgray;
74+
transition: background-color 0.2s ease-in-out;
75+
}
76+
77+
[data-scope="carousel"][data-part="indicator"][data-current] {
78+
background-color: blue;
79+
}
80+
81+
[data-scope="carousel"][data-part="autoplay-trigger"] {
82+
display: flex;
83+
justify-content: center;
84+
align-items: center;
85+
}
86+
87+
.carousel-spacer {
88+
flex-grow: 1;
89+
}
90+
91+
[data-scope="carousel"][data-part="autoplay-trigger"] svg {
92+
width: 12px;
93+
height: 12px;
594
}

packages/react/CHANGELOG.md

+9-2
Original file line numberDiff line numberDiff line change
@@ -6,9 +6,16 @@ description: All notable changes will be documented in this file.
66

77
## [Unreleased]
88

9-
### Fixed
9+
### Added
10+
11+
- **Carousel [Breaking]:** Redesigned the carousel for better touch handling and performance. See the [Carousel docs](https://ark-ui.com/docs/react/components/carousel) for more info.
12+
13+
### Fixed
1014

11-
- **Select**: Fixed regression where scroll restoration in overflowing select menus was not working.
15+
- **FileUpload:** Resolved an issue where the `accept` attribute wasn’t applied to the hidden input.
16+
- **NumberInput:** Fixed a bug where the input event wasn’t triggered on the first click of the increment/decrement controls.
17+
- **TreeView:** Addressed a limitation where React elements couldn’t be used in the tree view. The machine store has been revamped to support complex objects like React and Vue elements.
18+
- **Select:** Fixed a regression where scroll restoration didn’t work in overflowing select menus.
1219

1320
## [4.5.0] - 2024-12-12
1421

packages/react/package.json

+50-50
Original file line numberDiff line numberDiff line change
@@ -166,55 +166,55 @@
166166
"sideEffects": false,
167167
"dependencies": {
168168
"@internationalized/date": "3.6.0",
169-
"@zag-js/accordion": "0.78.3",
170-
"@zag-js/anatomy": "0.78.3",
171-
"@zag-js/auto-resize": "0.78.3",
172-
"@zag-js/avatar": "0.78.3",
173-
"@zag-js/carousel": "0.78.3",
174-
"@zag-js/checkbox": "0.78.3",
175-
"@zag-js/clipboard": "0.78.3",
176-
"@zag-js/collapsible": "0.78.3",
177-
"@zag-js/collection": "0.78.3",
178-
"@zag-js/color-picker": "0.78.3",
179-
"@zag-js/color-utils": "0.78.3",
180-
"@zag-js/combobox": "0.78.3",
181-
"@zag-js/core": "0.78.3",
182-
"@zag-js/date-picker": "0.78.3",
183-
"@zag-js/date-utils": "0.78.3",
184-
"@zag-js/dialog": "0.78.3",
185-
"@zag-js/dom-query": "0.78.3",
186-
"@zag-js/editable": "0.78.3",
187-
"@zag-js/file-upload": "0.78.3",
188-
"@zag-js/file-utils": "0.78.3",
189-
"@zag-js/highlight-word": "0.78.3",
190-
"@zag-js/hover-card": "0.78.3",
191-
"@zag-js/i18n-utils": "0.78.3",
192-
"@zag-js/menu": "0.78.3",
193-
"@zag-js/number-input": "0.78.3",
194-
"@zag-js/pagination": "0.78.3",
195-
"@zag-js/pin-input": "0.78.3",
196-
"@zag-js/popover": "0.78.3",
197-
"@zag-js/presence": "0.78.3",
198-
"@zag-js/progress": "0.78.3",
199-
"@zag-js/qr-code": "0.78.3",
200-
"@zag-js/radio-group": "0.78.3",
201-
"@zag-js/rating-group": "0.78.3",
202-
"@zag-js/react": "0.78.3",
203-
"@zag-js/select": "0.78.3",
204-
"@zag-js/signature-pad": "0.78.3",
205-
"@zag-js/slider": "0.78.3",
206-
"@zag-js/splitter": "0.78.3",
207-
"@zag-js/steps": "0.78.3",
208-
"@zag-js/switch": "0.78.3",
209-
"@zag-js/tabs": "0.78.3",
210-
"@zag-js/tags-input": "0.78.3",
211-
"@zag-js/time-picker": "0.78.3",
212-
"@zag-js/timer": "0.78.3",
213-
"@zag-js/toast": "0.78.3",
214-
"@zag-js/toggle-group": "0.78.3",
215-
"@zag-js/tooltip": "0.78.3",
216-
"@zag-js/tree-view": "0.78.3",
217-
"@zag-js/types": "0.78.3"
169+
"@zag-js/accordion": "0.79.1",
170+
"@zag-js/anatomy": "0.79.1",
171+
"@zag-js/auto-resize": "0.79.1",
172+
"@zag-js/avatar": "0.79.1",
173+
"@zag-js/carousel": "0.79.1",
174+
"@zag-js/checkbox": "0.79.1",
175+
"@zag-js/clipboard": "0.79.1",
176+
"@zag-js/collapsible": "0.79.1",
177+
"@zag-js/collection": "0.79.1",
178+
"@zag-js/color-picker": "0.79.1",
179+
"@zag-js/color-utils": "0.79.1",
180+
"@zag-js/combobox": "0.79.1",
181+
"@zag-js/core": "0.79.1",
182+
"@zag-js/date-picker": "0.79.1",
183+
"@zag-js/date-utils": "0.79.1",
184+
"@zag-js/dialog": "0.79.1",
185+
"@zag-js/dom-query": "0.79.1",
186+
"@zag-js/editable": "0.79.1",
187+
"@zag-js/file-upload": "0.79.1",
188+
"@zag-js/file-utils": "0.79.1",
189+
"@zag-js/highlight-word": "0.79.1",
190+
"@zag-js/hover-card": "0.79.1",
191+
"@zag-js/i18n-utils": "0.79.1",
192+
"@zag-js/menu": "0.79.1",
193+
"@zag-js/number-input": "0.79.1",
194+
"@zag-js/pagination": "0.79.1",
195+
"@zag-js/pin-input": "0.79.1",
196+
"@zag-js/popover": "0.79.1",
197+
"@zag-js/presence": "0.79.1",
198+
"@zag-js/progress": "0.79.1",
199+
"@zag-js/qr-code": "0.79.1",
200+
"@zag-js/radio-group": "0.79.1",
201+
"@zag-js/rating-group": "0.79.1",
202+
"@zag-js/react": "0.79.1",
203+
"@zag-js/select": "0.79.1",
204+
"@zag-js/signature-pad": "0.79.1",
205+
"@zag-js/slider": "0.79.1",
206+
"@zag-js/splitter": "0.79.1",
207+
"@zag-js/steps": "0.79.1",
208+
"@zag-js/switch": "0.79.1",
209+
"@zag-js/tabs": "0.79.1",
210+
"@zag-js/tags-input": "0.79.1",
211+
"@zag-js/time-picker": "0.79.1",
212+
"@zag-js/timer": "0.79.1",
213+
"@zag-js/toast": "0.79.1",
214+
"@zag-js/toggle-group": "0.79.1",
215+
"@zag-js/tooltip": "0.79.1",
216+
"@zag-js/tree-view": "0.79.1",
217+
"@zag-js/types": "0.79.1"
218218
},
219219
"devDependencies": {
220220
"@biomejs/biome": "1.9.4",
@@ -231,7 +231,7 @@
231231
"@types/react": "18.3.18",
232232
"@types/react-dom": "18.3.5",
233233
"@vitejs/plugin-react": "4.3.4",
234-
"@zag-js/stringify-state": "0.78.3",
234+
"@zag-js/stringify-state": "0.79.1",
235235
"clean-package": "2.2.0",
236236
"globby": "14.0.2",
237237
"jsdom": "25.0.1",
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
import { mergeProps } from '@zag-js/react'
2+
import { forwardRef } from 'react'
3+
import { type HTMLProps, type PolymorphicProps, ark } from '../factory'
4+
import { useCarouselContext } from './use-carousel-context'
5+
6+
export interface CarouselAutoplayTriggerBaseProps extends PolymorphicProps {}
7+
export interface CarouselAutoplayTriggerProps
8+
extends HTMLProps<'button'>,
9+
CarouselAutoplayTriggerBaseProps {}
10+
11+
export const CarouselAutoplayTrigger = forwardRef<HTMLButtonElement, CarouselAutoplayTriggerProps>(
12+
(props, ref) => {
13+
const carousel = useCarouselContext()
14+
const mergedProps = mergeProps(carousel.getAutoplayTriggerProps(), props)
15+
16+
return <ark.button {...mergedProps} ref={ref} />
17+
},
18+
)
19+
20+
CarouselAutoplayTrigger.displayName = 'CarouselAutoplayTrigger'
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,16 @@
1+
import { mergeProps } from '@zag-js/react'
12
import { forwardRef } from 'react'
23
import { type HTMLProps, type PolymorphicProps, ark } from '../factory'
3-
import { carouselAnatomy } from './carousel.anatomy'
4+
import { useCarouselContext } from './use-carousel-context'
45

56
export interface CarouselControlBaseProps extends PolymorphicProps {}
67
export interface CarouselControlProps extends HTMLProps<'div'>, CarouselControlBaseProps {}
78

8-
export const CarouselControl = forwardRef<HTMLDivElement, CarouselControlProps>((props, ref) => (
9-
<ark.div {...carouselAnatomy.build().control.attrs} {...props} ref={ref} />
10-
))
9+
export const CarouselControl = forwardRef<HTMLDivElement, CarouselControlProps>((props, ref) => {
10+
const carousel = useCarouselContext()
11+
const mergedProps = mergeProps(carousel.getControlProps(), props)
12+
13+
return <ark.div {...mergedProps} {...props} ref={ref} />
14+
})
1115

1216
CarouselControl.displayName = 'CarouselControl'

packages/react/src/components/carousel/carousel-item.tsx

+1-1
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ export interface CarouselItemBaseProps extends ItemProps, PolymorphicProps {}
99
export interface CarouselItemProps extends HTMLProps<'div'>, CarouselItemBaseProps {}
1010

1111
export const CarouselItem = forwardRef<HTMLDivElement, CarouselItemProps>((props, ref) => {
12-
const [itemProps, localProps] = createSplitProps<ItemProps>()(props, ['index'])
12+
const [itemProps, localProps] = createSplitProps<ItemProps>()(props, ['index', 'snapAlign'])
1313
const carousel = useCarouselContext()
1414
const mergedProps = mergeProps(carousel.getItemProps(itemProps), localProps)
1515

packages/react/src/components/carousel/carousel-root.tsx

+14-5
Original file line numberDiff line numberDiff line change
@@ -10,16 +10,25 @@ export interface CarouselRootProps extends HTMLProps<'div'>, CarouselRootBasePro
1010

1111
export const CarouselRoot = forwardRef<HTMLDivElement, CarouselRootProps>((props, ref) => {
1212
const [useCarouselProps, localProps] = createSplitProps<UseCarouselProps>()(props, [
13-
'align',
14-
'defaultIndex',
13+
'allowMouseDrag',
14+
'autoplay',
15+
'defaultPage',
1516
'id',
1617
'ids',
17-
'index',
18+
'inViewThreshold',
1819
'loop',
19-
'onIndexChange',
20+
'onAutoplayStatusChange',
21+
'onDragStatusChange',
22+
'onPageChange',
2023
'orientation',
21-
'slidesPerView',
24+
'padding',
25+
'page',
26+
'slideCount',
27+
'slidesPerMove',
28+
'slidesPerPage',
29+
'snapType',
2230
'spacing',
31+
'translations',
2332
])
2433
const carousel = useCarousel(useCarouselProps)
2534
const mergedProps = mergeProps(carousel.getRootProps(), localProps)

packages/react/src/components/carousel/carousel-viewport.tsx

-16
This file was deleted.
Original file line numberDiff line numberDiff line change
@@ -1,3 +1 @@
1-
import { anatomy } from '@zag-js/carousel'
2-
3-
export const carouselAnatomy = anatomy.extendWith('control')
1+
export { anatomy as carouselAnatomy } from '@zag-js/carousel'

packages/react/src/components/carousel/carousel.stories.tsx

+2-2
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@ const meta: Meta = {
66

77
export default meta
88

9+
export { Autoplay } from './examples/autoplay'
910
export { Basic } from './examples/basic'
10-
export { RootProvider } from './examples/root-provider'
1111
export { Controlled } from './examples/controlled'
12-
export { Customized } from './examples/customized'
12+
export { RootProvider } from './examples/root-provider'
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,47 @@
1+
import { cleanup, render, screen, waitFor } from '@testing-library/react/pure'
2+
import { axe } from 'vitest-axe'
3+
import { Carousel, carouselAnatomy } from '.'
4+
import { getExports, getParts } from '../../setup-test'
5+
import { Basic as ComponentUnderTest } from './examples/basic'
6+
7+
describe('Carousel / Parts & Exports', () => {
8+
afterAll(() => {
9+
cleanup()
10+
})
11+
12+
render(<ComponentUnderTest />)
13+
14+
const renderedParts = getParts(carouselAnatomy).filter(
15+
(part) => !part.includes('[data-part="autoplay-trigger"]'),
16+
)
17+
18+
it.each(renderedParts)('should render part %s', async (part) => {
19+
expect(document.querySelector(part)).toBeInTheDocument()
20+
})
21+
22+
it.each(getExports(carouselAnatomy))('should export %s', async (part) => {
23+
expect(Carousel[part]).toBeDefined()
24+
})
25+
})
26+
27+
describe('Carousel', () => {
28+
afterEach(() => {
29+
cleanup()
30+
})
31+
32+
it('should have no a11y violations', async () => {
33+
const { container } = render(<ComponentUnderTest />)
34+
const results = await axe(container)
35+
36+
expect(results).toHaveNoViolations()
37+
})
38+
39+
it('should have the correct disabled / enabled states for control buttons', async () => {
40+
render(<ComponentUnderTest />)
41+
const prevButton = screen.getByRole('button', { name: 'Previous slide' })
42+
const nextButton = screen.getByRole('button', { name: 'Next slide' })
43+
44+
await waitFor(() => expect(prevButton).toBeDisabled())
45+
await waitFor(() => expect(nextButton).toBeEnabled())
46+
})
47+
})

0 commit comments

Comments
 (0)