click to reveal text when very long

This commit is contained in:
keyan 2023-12-04 16:07:10 -06:00
parent 6c576bda84
commit 4127adb169
2 changed files with 45 additions and 2 deletions

View File

@ -6,7 +6,7 @@ import { LightAsync as SyntaxHighlighter } from 'react-syntax-highlighter'
import atomDark from 'react-syntax-highlighter/dist/cjs/styles/prism/atom-dark' import atomDark from 'react-syntax-highlighter/dist/cjs/styles/prism/atom-dark'
import mention from '../lib/remark-mention' import mention from '../lib/remark-mention'
import sub from '../lib/remark-sub' import sub from '../lib/remark-sub'
import React, { useState, memo, useRef, useCallback, useMemo } from 'react' import React, { useState, memo, useRef, useCallback, useMemo, useEffect } from 'react'
import GithubSlugger from 'github-slugger' import GithubSlugger from 'github-slugger'
import LinkIcon from '../svgs/link.svg' import LinkIcon from '../svgs/link.svg'
import Thumb from '../svgs/thumb-up-fill.svg' import Thumb from '../svgs/thumb-up-fill.svg'
@ -16,6 +16,7 @@ import ZoomableImage, { decodeOriginalUrl } from './image'
import { IMGPROXY_URL_REGEXP } from '../lib/url' import { IMGPROXY_URL_REGEXP } from '../lib/url'
import reactStringReplace from 'react-string-replace' import reactStringReplace from 'react-string-replace'
import { rehypeInlineCodeProperty } from '../lib/md' import { rehypeInlineCodeProperty } from '../lib/md'
import { Button } from 'react-bootstrap'
export function SearchText ({ text }) { export function SearchText ({ text }) {
return ( return (
@ -31,6 +32,17 @@ export function SearchText ({ text }) {
// this is one of the slowest components to render // this is one of the slowest components to render
export default memo(function Text ({ nofollow, imgproxyUrls, children, tab, ...outerProps }) { export default memo(function Text ({ nofollow, imgproxyUrls, children, tab, ...outerProps }) {
const [overflowing, setOverflowing] = useState(false)
const [show, setShow] = useState(false)
const containerRef = useRef(null)
useEffect(() => {
const container = containerRef.current
if (!container || overflowing) return
setOverflowing(container.scrollHeight > window.innerHeight)
}, [containerRef.current])
// all the reactStringReplace calls are to facilitate search highlighting // all the reactStringReplace calls are to facilitate search highlighting
const slugger = useRef(new GithubSlugger()) const slugger = useRef(new GithubSlugger())
@ -101,7 +113,7 @@ export default memo(function Text ({ nofollow, imgproxyUrls, children, tab, ...o
}, [imgproxyUrls, outerProps, tab]) }, [imgproxyUrls, outerProps, tab])
return ( return (
<div className={styles.text}> <div className={`${styles.text} ${overflowing && !show ? styles.textContained : ''}`} ref={containerRef}>
<ReactMarkdown <ReactMarkdown
components={{ components={{
h1: Heading, h1: Heading,
@ -155,6 +167,10 @@ export default memo(function Text ({ nofollow, imgproxyUrls, children, tab, ...o
> >
{children} {children}
</ReactMarkdown> </ReactMarkdown>
{overflowing && !show &&
<Button size='lg' variant='info' className={styles.textShowFull} onClick={() => setShow(true)}>
show full text
</Button>}
</div> </div>
) )
}) })

View File

@ -3,6 +3,32 @@
font-family: inherit; font-family: inherit;
word-break: break-word; word-break: break-word;
line-height: 140%; line-height: 140%;
overflow-y: hidden;
position: relative;
}
.textContained {
max-height: 200vh;
}
.textContained::before {
content: "";
position: absolute;
bottom: 0;
left: 0;
width: 100%;
height: 50vh;
pointer-events: none;
z-index: 1;
background: linear-gradient(rgba(255, 255, 255, 0), var(--bs-body-bg) 200%);
}
.textShowFull {
position: absolute;
bottom: 0;
left: 0;
border-radius: 0;
z-index: 2;
} }
@media screen and (min-width: 767px) { @media screen and (min-width: 767px) {
@ -47,6 +73,7 @@
} }
.text .p { .text .p {
display: block;
margin-bottom: .5rem; margin-bottom: .5rem;
white-space: pre-wrap; white-space: pre-wrap;
word-break: break-word; word-break: break-word;