Skip to content

Commit

Permalink
Update collapsible component
Browse files Browse the repository at this point in the history
  • Loading branch information
yigiterdev committed Feb 29, 2024
1 parent b7142e1 commit e4246cb
Show file tree
Hide file tree
Showing 7 changed files with 69 additions and 163 deletions.
27 changes: 27 additions & 0 deletions src/components/Collapsible/Collapsible.stories.tsx
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import { StoryObj, Meta } from '@storybook/react'
import { useEffect, useState } from 'react'

import { Text } from '~/components/Text'

Expand All @@ -11,6 +12,28 @@ export default {

type Story = StoryObj<typeof Collapsible>

const CollapsibleStory = () => {
const [isOpen, setIsOpen] = useState(false)

useEffect(() => {
setIsOpen(true)
}, [])

return (
<Collapsible
open={isOpen}
label="My Heading"
onOpenChange={open => setIsOpen(open)}
>
{[1, 2, 3, 4, 5].map(x => (
<Text variant="normal" as="p" color="text80" key={x}>
Item {x}
</Text>
))}
</Collapsible>
)
}

export const Default: Story = {
args: {
label: 'My Heading',
Expand All @@ -21,3 +44,7 @@ export const Default: Story = {
)),
},
}

export const Controlled: Story = {
render: () => <CollapsibleStory />,
}
25 changes: 25 additions & 0 deletions src/components/Collapsible/Collapsible.test.tsx
Original file line number Diff line number Diff line change
@@ -1,7 +1,22 @@
import { cleanup, render, screen, fireEvent } from '@testing-library/react'
import { useState } from 'react'

import { Collapsible } from './Collapsible'

const TestComponent = () => {
const [isOpen, setIsOpen] = useState(false)

return (
<Collapsible
open={isOpen}
onOpenChange={open => setIsOpen(open)}
label="Hello"
>
World
</Collapsible>
)
}

describe('<Collapsible />', () => {
afterEach(cleanup)

Expand All @@ -15,6 +30,16 @@ describe('<Collapsible />', () => {
expect(screen.getByText(/World/)).toBeInTheDocument()
})

it('controlled', () => {
render(<TestComponent />)
expect(screen.getByText(/Hello/)).toBeInTheDocument()
expect(screen.queryByText(/World/)).toBeNull()

fireEvent.click(screen.getByRole('button'))

expect(screen.getByText(/World/)).toBeInTheDocument()
})

it('with default open', () => {
render(
<Collapsible label="Hello" defaultOpen>
Expand Down
25 changes: 17 additions & 8 deletions src/components/Collapsible/Collapsible.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -18,9 +18,17 @@ type CollapsibleProps = BoxProps &
}

export const Collapsible = (props: CollapsibleProps) => {
const { className, children, defaultOpen, onOpenChange, label, ...rest } =
props
const {
className,
children,
defaultOpen,
open,
onOpenChange,
label,
...rest
} = props
const [expanded, toggleExpanded] = useState(defaultOpen)
const isOpen = open ?? expanded

const handleOpenChange = (open: boolean) => {
toggleExpanded(open)
Expand All @@ -29,15 +37,16 @@ export const Collapsible = (props: CollapsibleProps) => {

return (
<CollapsiblePrimitive.Root
open={isOpen}
defaultOpen={defaultOpen}
onOpenChange={handleOpenChange}
asChild
>
<Box
as={motion.div}
className={clsx(className, styles.root)}
initial={{ height: defaultOpen ? 'auto' : styles.COLLAPSED_HEIGHT }}
animate={{ height: expanded ? 'auto' : styles.COLLAPSED_HEIGHT }}
initial={{ height: isOpen ? 'auto' : styles.COLLAPSED_HEIGHT }}
animate={{ height: isOpen ? 'auto' : styles.COLLAPSED_HEIGHT }}
transition={{ ease: 'easeOut', duration: 0.3 }}
borderRadius="md"
background="backgroundSecondary"
Expand All @@ -55,22 +64,22 @@ export const Collapsible = (props: CollapsibleProps) => {
position="absolute"
right="0"
marginRight="4"
initial={{ rotate: defaultOpen ? 180 : 0 }}
animate={{ rotate: expanded ? 180 : 0 }}
initial={{ rotate: isOpen ? 180 : 0 }}
animate={{ rotate: isOpen ? 180 : 0 }}
transition={{ ease: 'linear', duration: 0.1 }}
>
<ChevronDownIcon className={styles.icon} color="text50" />
</Box>
</CollapsiblePrimitive.Trigger>
<AnimatePresence>
{expanded && (
{isOpen && (
<CollapsiblePrimitive.Content
className={styles.content}
asChild
forceMount
>
<motion.div
initial={{ opacity: defaultOpen ? 1 : 0 }}
initial={{ opacity: isOpen ? 1 : 0 }}
animate={{ opacity: 1 }}
exit={{ opacity: 0 }}
transition={{ ease: 'easeOut', duration: 0.3 }}
Expand Down
39 changes: 0 additions & 39 deletions src/components/Collapsible/ControlledCollapsible.stories.tsx

This file was deleted.

41 changes: 0 additions & 41 deletions src/components/Collapsible/ControlledCollapsible.test.tsx

This file was deleted.

74 changes: 0 additions & 74 deletions src/components/Collapsible/ControlledCollapsible.tsx

This file was deleted.

1 change: 0 additions & 1 deletion src/components/Collapsible/index.ts
Original file line number Diff line number Diff line change
@@ -1,2 +1 @@
export { CollapsiblePrimitive, Collapsible } from './Collapsible'
export { ControlledCollapsible } from './ControlledCollapsible'

0 comments on commit e4246cb

Please sign in to comment.