2021-08-04 18:52:59 +09:00
|
|
|
import React, { useState, useEffect, useRef } from 'react';
|
2021-07-28 22:15:52 +09:00
|
|
|
import PropTypes from 'prop-types';
|
|
|
|
import './ChannelView.scss';
|
|
|
|
|
|
|
|
import EventEmitter from 'events';
|
|
|
|
|
|
|
|
import RoomTimeline from '../../../client/state/RoomTimeline';
|
|
|
|
|
|
|
|
import ScrollView from '../../atoms/scroll/ScrollView';
|
|
|
|
|
2021-08-04 18:52:59 +09:00
|
|
|
import ChannelViewHeader from './ChannelViewHeader';
|
|
|
|
import ChannelViewContent from './ChannelViewContent';
|
|
|
|
import ChannelViewFloating from './ChannelViewFloating';
|
|
|
|
import ChannelViewInput from './ChannelViewInput';
|
|
|
|
import ChannelViewCmdBar from './ChannelViewCmdBar';
|
2021-07-28 22:15:52 +09:00
|
|
|
|
2021-08-04 18:52:59 +09:00
|
|
|
import { scrollToBottom, isAtBottom, autoScrollToBottom } from './common';
|
2021-07-28 22:15:52 +09:00
|
|
|
|
2021-08-04 18:52:59 +09:00
|
|
|
const viewEvent = new EventEmitter();
|
2021-07-28 22:15:52 +09:00
|
|
|
|
|
|
|
let lastScrollTop = 0;
|
|
|
|
let lastScrollHeight = 0;
|
|
|
|
let isReachedBottom = true;
|
|
|
|
let isReachedTop = false;
|
|
|
|
function ChannelView({ roomId }) {
|
|
|
|
const [roomTimeline, updateRoomTimeline] = useState(null);
|
|
|
|
const timelineSVRef = useRef(null);
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
roomTimeline?.removeInternalListeners();
|
|
|
|
updateRoomTimeline(new RoomTimeline(roomId));
|
|
|
|
isReachedBottom = true;
|
|
|
|
isReachedTop = false;
|
|
|
|
}, [roomId]);
|
|
|
|
|
|
|
|
const timelineScroll = {
|
|
|
|
reachBottom() {
|
|
|
|
scrollToBottom(timelineSVRef);
|
|
|
|
},
|
|
|
|
autoReachBottom() {
|
|
|
|
autoScrollToBottom(timelineSVRef);
|
|
|
|
},
|
|
|
|
tryRestoringScroll() {
|
|
|
|
const sv = timelineSVRef.current;
|
|
|
|
const { scrollHeight } = sv;
|
|
|
|
|
|
|
|
if (lastScrollHeight === scrollHeight) return;
|
|
|
|
|
|
|
|
if (lastScrollHeight < scrollHeight) {
|
|
|
|
sv.scrollTop = lastScrollTop + (scrollHeight - lastScrollHeight);
|
|
|
|
} else {
|
|
|
|
timelineScroll.reachBottom();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
enableSmoothScroll() {
|
|
|
|
timelineSVRef.current.style.scrollBehavior = 'smooth';
|
|
|
|
},
|
|
|
|
disableSmoothScroll() {
|
|
|
|
timelineSVRef.current.style.scrollBehavior = 'auto';
|
|
|
|
},
|
|
|
|
isScrollable() {
|
|
|
|
const oHeight = timelineSVRef.current.offsetHeight;
|
|
|
|
const sHeight = timelineSVRef.current.scrollHeight;
|
|
|
|
if (sHeight > oHeight) return true;
|
|
|
|
return false;
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
function onTimelineScroll(e) {
|
|
|
|
const { scrollTop, scrollHeight, offsetHeight } = e.target;
|
|
|
|
const scrollBottom = scrollTop + offsetHeight;
|
|
|
|
lastScrollTop = scrollTop;
|
|
|
|
lastScrollHeight = scrollHeight;
|
|
|
|
|
|
|
|
const PLACEHOLDER_HEIGHT = 96;
|
|
|
|
const PLACEHOLDER_COUNT = 3;
|
|
|
|
|
|
|
|
const topPagKeyPoint = PLACEHOLDER_COUNT * PLACEHOLDER_HEIGHT;
|
|
|
|
const bottomPagKeyPoint = scrollHeight - (offsetHeight / 2);
|
|
|
|
|
|
|
|
if (!isReachedBottom && isAtBottom(timelineSVRef)) {
|
|
|
|
isReachedBottom = true;
|
|
|
|
viewEvent.emit('toggle-reached-bottom', true);
|
|
|
|
}
|
|
|
|
if (isReachedBottom && !isAtBottom(timelineSVRef)) {
|
|
|
|
isReachedBottom = false;
|
|
|
|
viewEvent.emit('toggle-reached-bottom', false);
|
|
|
|
}
|
|
|
|
// TOP of timeline
|
|
|
|
if (scrollTop < topPagKeyPoint && isReachedTop === false) {
|
|
|
|
isReachedTop = true;
|
|
|
|
viewEvent.emit('reached-top');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
isReachedTop = false;
|
|
|
|
|
|
|
|
// BOTTOM of timeline
|
|
|
|
if (scrollBottom > bottomPagKeyPoint) {
|
|
|
|
// TODO:
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div className="channel-view">
|
|
|
|
<ChannelViewHeader roomId={roomId} />
|
|
|
|
<div className="channel-view__content-wrapper">
|
|
|
|
<div className="channel-view__scrollable">
|
|
|
|
<ScrollView onScroll={onTimelineScroll} ref={timelineSVRef} autoHide>
|
|
|
|
{roomTimeline !== null && (
|
|
|
|
<ChannelViewContent
|
|
|
|
roomId={roomId}
|
|
|
|
roomTimeline={roomTimeline}
|
|
|
|
timelineScroll={timelineScroll}
|
2021-08-04 18:52:59 +09:00
|
|
|
viewEvent={viewEvent}
|
2021-07-28 22:15:52 +09:00
|
|
|
/>
|
|
|
|
)}
|
|
|
|
</ScrollView>
|
|
|
|
{roomTimeline !== null && (
|
2021-08-04 18:52:59 +09:00
|
|
|
<ChannelViewFloating
|
2021-07-28 22:15:52 +09:00
|
|
|
roomId={roomId}
|
|
|
|
roomTimeline={roomTimeline}
|
|
|
|
timelineScroll={timelineScroll}
|
2021-08-04 18:52:59 +09:00
|
|
|
viewEvent={viewEvent}
|
2021-07-28 22:15:52 +09:00
|
|
|
/>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
{roomTimeline !== null && (
|
2021-08-04 18:52:59 +09:00
|
|
|
<div className="channel-view__sticky">
|
|
|
|
<ChannelViewInput
|
2021-07-28 22:15:52 +09:00
|
|
|
roomId={roomId}
|
|
|
|
roomTimeline={roomTimeline}
|
|
|
|
timelineScroll={timelineScroll}
|
2021-08-04 18:52:59 +09:00
|
|
|
viewEvent={viewEvent}
|
2021-07-28 22:15:52 +09:00
|
|
|
/>
|
2021-08-04 18:52:59 +09:00
|
|
|
<ChannelViewCmdBar
|
2021-07-28 22:15:52 +09:00
|
|
|
roomId={roomId}
|
|
|
|
roomTimeline={roomTimeline}
|
2021-08-04 18:52:59 +09:00
|
|
|
viewEvent={viewEvent}
|
2021-07-28 22:15:52 +09:00
|
|
|
/>
|
2021-08-04 18:52:59 +09:00
|
|
|
</div>
|
2021-07-28 22:15:52 +09:00
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
ChannelView.propTypes = {
|
|
|
|
roomId: PropTypes.string.isRequired,
|
|
|
|
};
|
|
|
|
|
|
|
|
export default ChannelView;
|