overleaf/services/web/frontend/js/shared/components/auto-expanding-text-area.tsx
Tim Down 38c673d057 Merge pull request #13720 from overleaf/td-review-panel-entry-pos
React review panel entry positioning

GitOrigin-RevId: c22617b1d3243b7d54b093426358aeb291421b9e
2023-07-19 08:03:52 +00:00

77 lines
1.9 KiB
TypeScript

import { useEffect, useRef } from 'react'
import { callFnsInSequence } from '../../utils/functions'
import { MergeAndOverride } from '../../../../types/utils'
export const resetHeight = (
e:
| React.ChangeEvent<HTMLTextAreaElement>
| React.KeyboardEvent<HTMLTextAreaElement>
) => {
const el = e.target as HTMLTextAreaElement
window.requestAnimationFrame(() => {
const curHeight = el.offsetHeight
const fitHeight = el.scrollHeight
// clear height if text area is empty
if (!el.value.length) {
el.style.removeProperty('height')
}
// otherwise expand to fit text
else if (fitHeight > curHeight) {
el.style.height = `${fitHeight}px`
}
})
}
type AutoExpandingTextAreaProps = MergeAndOverride<
React.ComponentProps<'textarea'>,
{
onResize?: () => void
}
>
function AutoExpandingTextArea({
onChange,
onResize,
...rest
}: AutoExpandingTextAreaProps) {
const ref = useRef<HTMLTextAreaElement>(null)
useEffect(() => {
if (!ref.current || !onResize || !('ResizeObserver' in window)) {
return
}
let isFirstResize = true
const resizeObserver = new ResizeObserver(() => {
// Ignore the resize that is triggered when the element is first
// inserted into the DOM
if (isFirstResize) {
isFirstResize = false
} else {
// Prevent errors like "ResizeObserver loop completed with undelivered
// notifications" that occur if onResize does something complicated.
// The cost of this is that onResize lags one frame behind, but it's
// unlikely to matter.
window.requestAnimationFrame(onResize)
}
})
resizeObserver.observe(ref.current)
return () => {
resizeObserver.disconnect()
}
}, [onResize])
return (
<textarea
onChange={callFnsInSequence(onChange, resetHeight)}
{...rest}
ref={ref}
/>
)
}
export default AutoExpandingTextArea