session-desktop/ts/components/session/SessionInboxView.tsx
audric c2b5ac68d6 don't rely on scrollheight to calculate for ui Updates
as it causes a layout complete refresh
2021-07-22 10:20:09 +10:00

145 lines
4.8 KiB
TypeScript

import React from 'react';
import { Provider } from 'react-redux';
import { bindActionCreators } from 'redux';
import { ConversationModel } from '../../models/conversation';
import { getConversationController } from '../../session/conversations';
import { UserUtils } from '../../session/utils';
import { createStore } from '../../state/createStore';
import { actions as conversationActions } from '../../state/ducks/conversations';
import { initialDefaultRoomState } from '../../state/ducks/defaultRooms';
import { initialModalState } from '../../state/ducks/modalDialog';
import { initialOnionPathState } from '../../state/ducks/onion';
import { initialSearchState } from '../../state/ducks/search';
import { initialSectionState } from '../../state/ducks/section';
import { initialThemeState } from '../../state/ducks/theme';
import { initialUserConfigState } from '../../state/ducks/userConfig';
import { StateType } from '../../state/reducer';
import { makeLookup } from '../../util';
import { LeftPane } from '../LeftPane';
import { SessionMainPanel } from '../SessionMainPanel';
// tslint:disable-next-line: no-submodule-imports
import { PersistGate } from 'redux-persist/integration/react';
import { persistStore } from 'redux-persist';
import { TimerOptionsArray, TimerOptionsState } from '../../state/ducks/timerOptions';
// Workaround: A react component's required properties are filtering up through connect()
// https://github.com/DefinitelyTyped/DefinitelyTyped/issues/31363
type State = {
isInitialLoadComplete: boolean;
isExpired: boolean;
};
export class SessionInboxView extends React.Component<any, State> {
private store: any;
constructor(props: any) {
super(props);
this.state = {
isInitialLoadComplete: false,
isExpired: false,
};
void this.setupLeftPane();
// not reactified yet. this is a callback called once we were able to check for expiration of this Session version
window.extension.expired((expired: boolean) => {
if (expired) {
this.setState({
isExpired: true,
});
}
});
}
public render() {
if (!this.state.isInitialLoadComplete) {
return <></>;
}
const persistor = persistStore(this.store);
return (
<Provider store={this.store}>
<PersistGate loading={null} persistor={persistor}>
<div className="gutter">
<div className="network-status-container" />
{this.renderLeftPane()}
</div>
<SessionMainPanel />
</PersistGate>
</Provider>
);
}
private renderLeftPane() {
return <LeftPane isExpired={this.state.isExpired} />;
}
private async setupLeftPane() {
// Here we set up a full redux store with initial state for our LeftPane Root
const convoCollection = getConversationController().getConversations();
const conversations = convoCollection.map((conversation: ConversationModel) =>
conversation.getProps()
);
const filledConversations = conversations.map((conv: any) => {
return { ...conv, messages: [] };
});
const fullFilledConversations = await Promise.all(filledConversations);
const timerOptions: TimerOptionsArray = window.Whisper.ExpirationTimerOptions.map(
(item: any) => ({
name: item.getName(),
value: item.get('seconds'),
})
);
const initialState: StateType = {
conversations: {
conversationLookup: makeLookup(fullFilledConversations, 'id'),
messages: [],
showRightPanel: false,
messageDetailProps: undefined,
selectedMessageIds: [],
selectedConversation: undefined,
areMoreMessagesBeingFetched: false,
showScrollButton: false,
animateQuotedMessageId: undefined,
lightBox: undefined,
nextMessageToPlay: undefined,
quotedMessage: undefined,
mentionMembers: [],
firstUnreadMessageId: undefined,
},
user: {
ourNumber: UserUtils.getOurPubKeyStrFromCache(),
},
section: initialSectionState,
defaultRooms: initialDefaultRoomState,
search: initialSearchState,
theme: initialThemeState,
onionPaths: initialOnionPathState,
modals: initialModalState,
userConfig: initialUserConfigState,
timerOptions: {
timerOptions,
},
};
this.store = createStore(initialState);
window.inboxStore = this.store;
// Enables our redux store to be updated by backbone events in the outside world
const { messageExpired } = bindActionCreators(conversationActions, this.store.dispatch);
window.actionsCreators = conversationActions;
// messageExpired is currently inboked fropm js. So we link it to Redux that way
window.Whisper.events.on('messageExpired', messageExpired);
this.setState({ isInitialLoadComplete: true });
}
}