TimelineQueueButtonHeader: convert to tsx
This commit is contained in:
parent
577c391417
commit
697c028c4a
2 changed files with 83 additions and 0 deletions
Binary file not shown.
83
app/soapbox/components/timeline_queue_button_header.tsx
Normal file
83
app/soapbox/components/timeline_queue_button_header.tsx
Normal file
|
@ -0,0 +1,83 @@
|
||||||
|
import classNames from 'classnames';
|
||||||
|
import { throttle } from 'lodash';
|
||||||
|
import React, { useState, useEffect, useCallback } from 'react';
|
||||||
|
import { useIntl, MessageDescriptor } from 'react-intl';
|
||||||
|
|
||||||
|
import Icon from 'soapbox/components/icon';
|
||||||
|
import { Text } from 'soapbox/components/ui';
|
||||||
|
import { useSettings } from 'soapbox/hooks';
|
||||||
|
|
||||||
|
interface ITimelineQueueButtonHeader {
|
||||||
|
onClick: () => void,
|
||||||
|
count?: number,
|
||||||
|
message: MessageDescriptor,
|
||||||
|
threshold?: number,
|
||||||
|
autoloadThreshold?: number,
|
||||||
|
}
|
||||||
|
|
||||||
|
const TimelineQueueButtonHeader: React.FC<ITimelineQueueButtonHeader> = ({
|
||||||
|
onClick,
|
||||||
|
count = 0,
|
||||||
|
message,
|
||||||
|
threshold = 400,
|
||||||
|
autoloadThreshold = 50,
|
||||||
|
}) => {
|
||||||
|
const intl = useIntl();
|
||||||
|
const settings = useSettings();
|
||||||
|
|
||||||
|
const [scrolled, setScrolled] = useState<boolean>(false);
|
||||||
|
const autoload = settings.get('autoloadTimelines') === true;
|
||||||
|
|
||||||
|
const handleScroll = useCallback(throttle(() => {
|
||||||
|
const { scrollTop } = (document.scrollingElement || document.documentElement);
|
||||||
|
|
||||||
|
if (autoload && scrollTop <= autoloadThreshold) {
|
||||||
|
onClick();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (scrollTop > threshold) {
|
||||||
|
setScrolled(true);
|
||||||
|
} else {
|
||||||
|
setScrolled(false);
|
||||||
|
}
|
||||||
|
}, 150, { trailing: true }), []);
|
||||||
|
|
||||||
|
const scrollUp = () => {
|
||||||
|
window.scrollTo({ top: 0, behavior: 'smooth' });
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleClick: React.MouseEventHandler = () => {
|
||||||
|
setTimeout(scrollUp, 10);
|
||||||
|
onClick();
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
window.addEventListener('scroll', handleScroll);
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
window.removeEventListener('scroll', handleScroll);
|
||||||
|
};
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const visible = count > 0 && scrolled;
|
||||||
|
|
||||||
|
const classes = classNames('left-1/2 -translate-x-1/2 fixed top-20 z-50', {
|
||||||
|
'hidden': !visible,
|
||||||
|
});
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className={classes}>
|
||||||
|
<a className='flex items-center bg-primary-600 hover:bg-primary-700 hover:scale-105 active:scale-100 transition-transform text-white rounded-full px-4 py-2 space-x-1.5 cursor-pointer whitespace-nowrap' onClick={handleClick}>
|
||||||
|
<Icon src={require('@tabler/icons/icons/arrow-bar-to-up.svg')} />
|
||||||
|
|
||||||
|
{(count > 0) && (
|
||||||
|
<Text theme='inherit' size='sm'>
|
||||||
|
{intl.formatMessage(message, { count })}
|
||||||
|
</Text>
|
||||||
|
)}
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default TimelineQueueButtonHeader;
|
Loading…
Reference in a new issue