mirror of
https://github.com/d0zingcat/NotionNext.git
synced 2026-05-14 15:09:22 +00:00
112 lines
2.6 KiB
JavaScript
112 lines
2.6 KiB
JavaScript
import { useEffect, useImperativeHandle, useRef } from 'react'
|
|
|
|
/**
|
|
* 折叠面板组件,支持水平折叠、垂直折叠
|
|
* @param {type:['horizontal','vertical'], isOpen} props
|
|
* @returns
|
|
*/
|
|
const Collapse = ({
|
|
type = 'vertical',
|
|
isOpen = false,
|
|
children,
|
|
onHeightChange,
|
|
className,
|
|
collapseRef
|
|
}) => {
|
|
const ref = useRef(null)
|
|
|
|
useImperativeHandle(collapseRef, () => {
|
|
return {
|
|
/**
|
|
* 当子元素高度变化时,可调用此方法更新折叠组件的高度
|
|
* @param {*} param0
|
|
*/
|
|
updateCollapseHeight: ({ height, increase }) => {
|
|
if (isOpen) {
|
|
ref.current.style.height = ref.current.scrollHeight
|
|
ref.current.style.height = 'auto'
|
|
}
|
|
}
|
|
}
|
|
})
|
|
|
|
/**
|
|
* 折叠
|
|
* @param {*} element
|
|
*/
|
|
const collapseSection = element => {
|
|
const sectionHeight = element.scrollHeight
|
|
const sectionWidth = element.scrollWidth
|
|
|
|
requestAnimationFrame(function () {
|
|
switch (type) {
|
|
case 'horizontal':
|
|
element.style.width = sectionWidth + 'px'
|
|
requestAnimationFrame(function () {
|
|
element.style.width = 0 + 'px'
|
|
})
|
|
break
|
|
case 'vertical':
|
|
element.style.height = sectionHeight + 'px'
|
|
requestAnimationFrame(function () {
|
|
element.style.height = 0 + 'px'
|
|
})
|
|
}
|
|
})
|
|
}
|
|
|
|
/**
|
|
* 展开
|
|
* @param {*} element
|
|
*/
|
|
const expandSection = element => {
|
|
const sectionHeight = element.scrollHeight
|
|
const sectionWidth = element.scrollWidth
|
|
let clearTime = 0
|
|
switch (type) {
|
|
case 'horizontal':
|
|
element.style.width = sectionWidth + 'px'
|
|
clearTime = setTimeout(() => {
|
|
element.style.width = 'auto'
|
|
}, 400)
|
|
break
|
|
case 'vertical':
|
|
element.style.height = sectionHeight + 'px'
|
|
clearTime = setTimeout(() => {
|
|
element.style.height = 'auto'
|
|
}, 400)
|
|
}
|
|
|
|
clearTimeout(clearTime)
|
|
}
|
|
|
|
useEffect(() => {
|
|
if (isOpen) {
|
|
expandSection(ref.current)
|
|
} else {
|
|
collapseSection(ref.current)
|
|
}
|
|
// 通知父组件高度变化
|
|
onHeightChange &&
|
|
onHeightChange({
|
|
height: ref.current.scrollHeight,
|
|
increase: isOpen
|
|
})
|
|
}, [isOpen])
|
|
|
|
return (
|
|
<div
|
|
ref={ref}
|
|
style={
|
|
type === 'vertical'
|
|
? { height: '0px', willChange: 'height' }
|
|
: { width: '0px', willChange: 'width' }
|
|
}
|
|
className={`${className || ''} overflow-hidden duration-300`}>
|
|
{children}
|
|
</div>
|
|
)
|
|
}
|
|
|
|
export default Collapse
|