feat: collapible component

This commit is contained in:
codytseng
2025-05-16 22:15:32 +08:00
parent 12f5d41811
commit e1fbda5efc
2 changed files with 77 additions and 56 deletions

View File

@@ -0,0 +1,74 @@
import { Button } from '@/components/ui/button'
import { useEffect, useRef, useState } from 'react'
import { useTranslation } from 'react-i18next'
export default function Collapsible({
alwaysExpand = false,
children,
threshold = 1000,
collapsedHeight = 600,
...props
}: {
alwaysExpand?: boolean
threshold?: number
collapsedHeight?: number
} & React.HTMLProps<HTMLDivElement>) {
const { t } = useTranslation()
const containerRef = useRef<HTMLDivElement>(null)
const [expanded, setExpanded] = useState(false)
const [shouldCollapse, setShouldCollapse] = useState(false)
useEffect(() => {
if (alwaysExpand || shouldCollapse) return
const contentEl = containerRef.current
if (!contentEl) return
const checkHeight = () => {
const fullHeight = contentEl.scrollHeight
if (fullHeight > threshold) {
setShouldCollapse(true)
}
}
checkHeight()
const observer = new ResizeObserver(() => {
checkHeight()
})
observer.observe(contentEl)
return () => {
observer.disconnect()
}
}, [alwaysExpand, shouldCollapse])
return (
<div ref={containerRef} {...props}>
<div
className="relative text-left overflow-hidden"
style={{
maxHeight: !shouldCollapse || expanded ? 'none' : `${collapsedHeight}px`
}}
>
{children}
{shouldCollapse && !expanded && (
<div className="absolute bottom-0 h-40 w-full bg-gradient-to-b from-transparent to-background/90 flex items-end justify-center pb-4">
<div className="bg-background rounded-md">
<Button
className="bg-foreground hover:bg-foreground/80"
onClick={(e) => {
e.stopPropagation()
setExpanded(!expanded)
}}
>
{t('Show more')}
</Button>
</div>
</div>
)}
</div>
</div>
)
}

View File

@@ -1,10 +1,8 @@
import { Button } from '@/components/ui/button'
import { Separator } from '@/components/ui/separator'
import { toNote } from '@/lib/link'
import { useSecondaryPage } from '@/PageManager'
import { Event } from 'nostr-tools'
import { useEffect, useRef, useState } from 'react'
import { useTranslation } from 'react-i18next'
import Collapsible from '../Collapsible'
import Note from '../Note'
import NoteStats from '../NoteStats'
import RepostDescription from './RepostDescription'
@@ -20,41 +18,10 @@ export default function MainNoteCard({
reposter?: string
embedded?: boolean
}) {
const { t } = useTranslation()
const { push } = useSecondaryPage()
const containerRef = useRef<HTMLDivElement>(null)
const [expanded, setExpanded] = useState(false)
const [shouldCollapse, setShouldCollapse] = useState(false)
useEffect(() => {
if (embedded || shouldCollapse) return
const contentEl = containerRef.current
if (!contentEl) return
const checkHeight = () => {
const fullHeight = contentEl.scrollHeight
if (fullHeight > 1000) {
setShouldCollapse(true)
}
}
checkHeight()
const observer = new ResizeObserver(() => {
checkHeight()
})
observer.observe(contentEl)
return () => {
observer.disconnect()
}
}, [embedded, shouldCollapse])
return (
<div
ref={containerRef}
className={className}
onClick={(e) => {
e.stopPropagation()
@@ -62,34 +29,14 @@ export default function MainNoteCard({
}}
>
<div className={`clickable ${embedded ? 'p-2 sm:p-3 border rounded-lg' : 'py-3'}`}>
<div
className="relative text-left overflow-hidden"
style={{
maxHeight: !shouldCollapse || expanded ? 'none' : '600px'
}}
>
<Collapsible alwaysExpand={embedded}>
<RepostDescription className={embedded ? '' : 'px-4'} reposter={reposter} />
<Note
className={embedded ? '' : 'px-4'}
size={embedded ? 'small' : 'normal'}
event={event}
/>
{shouldCollapse && !expanded && (
<div className="absolute bottom-0 h-40 w-full bg-gradient-to-b from-transparent to-background/90 flex items-end justify-center pb-4">
<div className="bg-background rounded-md">
<Button
className="bg-foreground hover:bg-foreground/80"
onClick={(e) => {
e.stopPropagation()
setExpanded(!expanded)
}}
>
{t('Show more')}
</Button>
</div>
</div>
)}
</div>
</Collapsible>
{!embedded && <NoteStats className="mt-3 px-4" event={event} />}
</div>
{!embedded && <Separator />}