2020-11-05 10:22:05 +00:00
|
|
|
import { useRef, useState, useLayoutEffect } from 'react'
|
|
|
|
import classNames from 'classnames'
|
|
|
|
|
|
|
|
function useExpandCollapse({
|
|
|
|
initiallyExpanded = false,
|
2023-01-09 14:33:43 +00:00
|
|
|
collapsedSize = 0,
|
2020-11-05 10:22:05 +00:00
|
|
|
dimension = 'height',
|
2023-01-09 14:33:43 +00:00
|
|
|
classes = { container: '', containerCollapsed: '' },
|
2020-11-05 10:22:05 +00:00
|
|
|
} = {}) {
|
2023-01-09 14:33:43 +00:00
|
|
|
const ref = useRef<{ scrollHeight: number; scrollWidth: number }>()
|
2020-11-05 10:22:05 +00:00
|
|
|
const [isExpanded, setIsExpanded] = useState(initiallyExpanded)
|
2023-01-09 14:33:43 +00:00
|
|
|
const [sizing, setSizing] = useState<{
|
|
|
|
size: number | null
|
|
|
|
needsExpandCollapse: boolean | null
|
|
|
|
}>({
|
2020-12-02 10:03:03 +00:00
|
|
|
size: null,
|
2021-04-27 07:52:58 +00:00
|
|
|
needsExpandCollapse: null,
|
2020-12-02 10:03:03 +00:00
|
|
|
})
|
2020-11-05 10:22:05 +00:00
|
|
|
|
2020-12-15 10:23:54 +00:00
|
|
|
useLayoutEffect(() => {
|
|
|
|
const expandCollapseEl = ref.current
|
|
|
|
if (expandCollapseEl) {
|
|
|
|
const expandedSize =
|
|
|
|
dimension === 'height'
|
|
|
|
? expandCollapseEl.scrollHeight
|
|
|
|
: expandCollapseEl.scrollWidth
|
2020-12-02 10:03:03 +00:00
|
|
|
|
2020-12-15 10:23:54 +00:00
|
|
|
const needsExpandCollapse = expandedSize > collapsedSize
|
2020-12-02 10:03:03 +00:00
|
|
|
|
2020-12-15 10:23:54 +00:00
|
|
|
if (isExpanded) {
|
|
|
|
setSizing({ size: expandedSize, needsExpandCollapse })
|
|
|
|
} else {
|
|
|
|
setSizing({
|
|
|
|
size: needsExpandCollapse ? collapsedSize : expandedSize,
|
2021-04-27 07:52:58 +00:00
|
|
|
needsExpandCollapse,
|
2020-12-15 10:23:54 +00:00
|
|
|
})
|
2020-11-05 10:22:05 +00:00
|
|
|
}
|
2020-12-15 10:23:54 +00:00
|
|
|
}
|
2021-01-07 14:22:19 +00:00
|
|
|
}, [isExpanded, collapsedSize, dimension])
|
2020-11-05 10:22:05 +00:00
|
|
|
|
|
|
|
const expandableClasses = classNames(
|
|
|
|
'expand-collapse-container',
|
|
|
|
classes.container,
|
|
|
|
!isExpanded ? classes.containerCollapsed : null
|
|
|
|
)
|
|
|
|
|
|
|
|
function handleToggle() {
|
|
|
|
setIsExpanded(!isExpanded)
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
isExpanded,
|
2020-12-02 10:03:03 +00:00
|
|
|
needsExpandCollapse: sizing.needsExpandCollapse,
|
2020-11-05 10:22:05 +00:00
|
|
|
expandableProps: {
|
|
|
|
ref,
|
|
|
|
style: {
|
2021-04-27 07:52:58 +00:00
|
|
|
[dimension === 'height' ? 'height' : 'width']: `${sizing.size}px`,
|
2020-11-05 10:22:05 +00:00
|
|
|
},
|
2021-04-27 07:52:58 +00:00
|
|
|
className: expandableClasses,
|
2020-11-05 10:22:05 +00:00
|
|
|
},
|
|
|
|
toggleProps: {
|
2021-04-27 07:52:58 +00:00
|
|
|
onClick: handleToggle,
|
|
|
|
},
|
2020-11-05 10:22:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export default useExpandCollapse
|