chuckya/app/javascript/mastodon/components/status_list.js

139 lines
3.8 KiB
JavaScript
Raw Normal View History

import React from 'react';
import ImmutablePropTypes from 'react-immutable-proptypes';
import { ScrollContainer } from 'react-router-scroll';
import PropTypes from 'prop-types';
import StatusContainer from '../containers/status_container';
import LoadMore from './load_more';
import ImmutablePureComponent from 'react-immutable-pure-component';
import IntersectionObserverWrapper from '../features/ui/util/intersection_observer_wrapper';
class StatusList extends ImmutablePureComponent {
static propTypes = {
scrollKey: PropTypes.string.isRequired,
statusIds: ImmutablePropTypes.list.isRequired,
onScrollToBottom: PropTypes.func,
onScrollToTop: PropTypes.func,
onScroll: PropTypes.func,
shouldUpdateScroll: PropTypes.func,
isLoading: PropTypes.bool,
isUnread: PropTypes.bool,
hasMore: PropTypes.bool,
prepend: PropTypes.node,
emptyMessage: PropTypes.node,
};
static defaultProps = {
trackScroll: true,
};
intersectionObserverWrapper = new IntersectionObserverWrapper();
handleScroll = (e) => {
2016-09-22 08:08:35 +09:00
const { scrollTop, scrollHeight, clientHeight } = e.target;
2017-01-24 12:12:10 +09:00
const offset = scrollHeight - scrollTop - clientHeight;
this._oldScrollPosition = scrollHeight - scrollTop;
2017-01-24 12:12:10 +09:00
if (250 > offset && this.props.onScrollToBottom && !this.props.isLoading) {
2016-09-22 08:08:35 +09:00
this.props.onScrollToBottom();
} else if (scrollTop < 100 && this.props.onScrollToTop) {
this.props.onScrollToTop();
} else if (this.props.onScroll) {
this.props.onScroll();
2016-09-22 08:08:35 +09:00
}
}
2016-09-22 08:08:35 +09:00
2017-01-24 12:12:10 +09:00
componentDidMount () {
this.attachScrollListener();
this.attachIntersectionObserver();
}
2017-01-24 12:12:10 +09:00
componentDidUpdate (prevProps) {
if ((prevProps.statusIds.size < this.props.statusIds.size && prevProps.statusIds.first() !== this.props.statusIds.first() && !!this._oldScrollPosition) && this.node.scrollTop > 0) {
2017-01-24 12:12:10 +09:00
this.node.scrollTop = this.node.scrollHeight - this._oldScrollPosition;
}
}
2017-01-24 12:12:10 +09:00
componentWillUnmount () {
this.detachScrollListener();
this.detachIntersectionObserver();
}
attachIntersectionObserver () {
this.intersectionObserverWrapper.connect({
root: this.node,
rootMargin: '300% 0px',
});
}
detachIntersectionObserver () {
this.intersectionObserverWrapper.disconnect();
}
2017-01-24 12:12:10 +09:00
attachScrollListener () {
this.node.addEventListener('scroll', this.handleScroll);
}
2017-01-24 12:12:10 +09:00
detachScrollListener () {
this.node.removeEventListener('scroll', this.handleScroll);
}
2017-01-24 12:12:10 +09:00
setRef = (c) => {
2017-01-24 12:12:10 +09:00
this.node = c;
}
2017-01-24 12:12:10 +09:00
handleLoadMore = (e) => {
e.preventDefault();
this.props.onScrollToBottom();
}
render () {
const { statusIds, onScrollToBottom, scrollKey, shouldUpdateScroll, isLoading, isUnread, hasMore, prepend, emptyMessage } = this.props;
let loadMore = null;
let scrollableArea = null;
let unread = null;
if (!isLoading && statusIds.size > 0 && hasMore) {
loadMore = <LoadMore onClick={this.handleLoadMore} />;
}
2016-09-30 07:00:45 +09:00
if (isUnread) {
unread = <div className='status-list__unread-indicator' />;
}
if (isLoading || statusIds.size > 0 || !emptyMessage) {
scrollableArea = (
<div className='scrollable' ref={this.setRef}>
{unread}
<div className='status-list'>
{prepend}
{statusIds.map((statusId) => {
return <StatusContainer key={statusId} id={statusId} intersectionObserverWrapper={this.intersectionObserverWrapper} />;
})}
{loadMore}
</div>
</div>
);
} else {
scrollableArea = (
<div className='empty-column-indicator' ref={this.setRef}>
{emptyMessage}
</div>
);
}
return (
<ScrollContainer scrollKey={scrollKey} shouldUpdateScroll={shouldUpdateScroll}>
{scrollableArea}
</ScrollContainer>
);
}
}
export default StatusList;