mirror of
https://github.com/CompassConnections/Compass.git
synced 2026-02-20 07:44:01 -05:00
104 lines
2.9 KiB
TypeScript
104 lines
2.9 KiB
TypeScript
import { DocumentTextIcon } from '@heroicons/react/outline'
|
|
import { JSONContent } from '@tiptap/react'
|
|
import clsx from 'clsx'
|
|
import { MouseEventHandler, ReactNode, useRef, useState } from 'react'
|
|
import { Button } from 'web/components/buttons/button'
|
|
import { Col } from '../layout/col'
|
|
import { Modal } from '../layout/modal'
|
|
import { Row } from '../layout/row'
|
|
import { Content } from './editor'
|
|
import { useSafeLayoutEffect } from 'web/hooks/use-safe-layout-effect'
|
|
import { SHOW_COLLAPSE_TRESHOLD } from 'web/components/widgets/collapsible-content'
|
|
|
|
export function ExpandButton(props: {
|
|
onClick?: MouseEventHandler<any> | undefined
|
|
className?: string
|
|
whatToRead?: string
|
|
}) {
|
|
const { onClick, className, whatToRead } = props
|
|
return (
|
|
<Button
|
|
color="none"
|
|
className={clsx(
|
|
'text-primary-500 hover:text-primary-700 z-10 select-none bg-inherit text-sm',
|
|
className
|
|
)}
|
|
onClick={onClick}
|
|
size={'xs'}
|
|
>
|
|
<Row className="items-center gap-1">
|
|
<DocumentTextIcon className="h-4 w-4" />
|
|
Read more{whatToRead ? ` ${whatToRead}` : ''}
|
|
</Row>
|
|
</Button>
|
|
)
|
|
}
|
|
|
|
export function ExpandableContent(props: {
|
|
content: JSONContent | string
|
|
modalContent: ReactNode
|
|
whatToRead?: string
|
|
className?: string
|
|
}) {
|
|
const { content, modalContent, whatToRead, className } = props
|
|
const [shouldAllowCollapseOfContent, setShouldAllowCollapseOfContent] =
|
|
useState(false)
|
|
const contentRef = useRef<HTMLDivElement>(null)
|
|
|
|
useSafeLayoutEffect(() => {
|
|
if (contentRef.current) {
|
|
if (contentRef.current.offsetHeight > SHOW_COLLAPSE_TRESHOLD) {
|
|
setShouldAllowCollapseOfContent(true)
|
|
}
|
|
}
|
|
}, [contentRef.current?.offsetHeight])
|
|
if (shouldAllowCollapseOfContent) {
|
|
return (
|
|
<div className={className}>
|
|
<ExpandsToModalContent
|
|
content={content}
|
|
modalContent={modalContent}
|
|
whatToRead={whatToRead}
|
|
/>
|
|
</div>
|
|
)
|
|
}
|
|
return (
|
|
<div ref={contentRef} className={className}>
|
|
<Content content={content} />
|
|
</div>
|
|
)
|
|
}
|
|
|
|
function ExpandsToModalContent(props: {
|
|
content: JSONContent | string
|
|
modalContent: ReactNode
|
|
whatToRead?: string
|
|
}) {
|
|
const { content, modalContent, whatToRead } = props
|
|
const [open, setOpen] = useState(false)
|
|
return (
|
|
<>
|
|
<Col className="relative gap-1">
|
|
<Content
|
|
content={content}
|
|
// Unfortunately line-clamp doesn't work with tiptap content
|
|
className={'relative max-h-52 w-full overflow-hidden'}
|
|
/>
|
|
<Row className="w-full justify-end">
|
|
<ExpandButton
|
|
className="bg-transparent"
|
|
onClick={() => {
|
|
setOpen(true)
|
|
}}
|
|
whatToRead={whatToRead}
|
|
/>
|
|
</Row>
|
|
</Col>
|
|
<Modal open={open} setOpen={setOpen} size="lg">
|
|
{modalContent}
|
|
</Modal>
|
|
</>
|
|
)
|
|
}
|