groupomania/client/src/components/ScrollToBottom.tsx

51 lines
1.7 KiB
TypeScript

import { ReactNode, useCallback, useEffect, useState } from 'react';
import { FaChevronDown } from 'react-icons/fa';
const ScrollToBottom = ({ children, className = '' }: { children: ReactNode; className?: string }) => {
const [node, setNode] = useState<HTMLDivElement | null>(null);
const [show, setShow] = useState(false);
const bottom = useCallback((node: HTMLDivElement) => {
if (node) {
node.scrollIntoView({ behavior: 'smooth' });
setNode(node);
}
}, []);
useEffect(() => {
const container = document.querySelector('main > div');
container?.addEventListener('scroll', () => {
if (node?.getBoundingClientRect() && node.getBoundingClientRect().y >= window.innerHeight) {
setShow(true);
} else {
setShow(false);
}
});
container?.addEventListener('DOMNodeInserted', (e) => {
if (!(e.target as Element).classList.contains('message')) return;
if (node?.getBoundingClientRect() && node.getBoundingClientRect().y >= window.innerHeight) {
node?.scrollIntoView({ behavior: 'smooth' });
}
});
}, [node]);
return (
<>
<div className={className}>
{children}
<div className="scroll-bottom" ref={bottom}></div>
</div>
<button
onClick={() => node?.scrollIntoView({ behavior: 'smooth' })}
className={'absolute right-3 transition-all' + (show ? ' bottom-3' : ' -bottom-10')}
>
<div className="popup-btn cursor-pointer rounded-full shadow-lg shadow-slate-900 bg-grey-dark hover:bg-grey-light transition-all">
<FaChevronDown className="fill-grey-light hover:fill-grey-dark transition-all text-xl w-10 h-10 p-2.5" />
</div>
</button>
</>
);
};
export default ScrollToBottom;