move LeftPane items to hooks

pull/1540/head
Audric Ackermann 4 years ago
parent 51452c5406
commit 08ce55f1a6
No known key found for this signature in database
GPG Key ID: 999F434D76324AD4

@ -26,7 +26,7 @@ import { PubKey } from '../session/types';
import { ConversationType } from '../state/ducks/conversations';
export interface ConversationListItemProps extends ConversationType {
index: number; // used to force a refresh when one conversation is removed on top of the list
index?: number; // used to force a refresh when one conversation is removed on top of the list
memberAvatars?: Array<ConversationAvatar>; // this is added by usingClosedConversationDetails
}

@ -3,17 +3,29 @@ import React from 'react';
import { ActionsPanel, SectionType } from './session/ActionsPanel';
import { LeftPaneMessageSection } from './session/LeftPaneMessageSection';
import { ConversationListItemProps } from './ConversationListItem';
import { SearchResultsProps } from './SearchResults';
import { SearchOptions } from '../types/Search';
import { ConversationType } from '../state/ducks/conversations';
import { openConversationExternal } from '../state/ducks/conversations';
import { LeftPaneContactSection } from './session/LeftPaneContactSection';
import { LeftPaneSettingSection } from './session/LeftPaneSettingSection';
import { SessionTheme } from '../state/ducks/SessionTheme';
import { DefaultTheme } from 'styled-components';
import { SessionSettingCategory } from './session/settings/SessionSettings';
import { SessionOffline } from './session/network/SessionOffline';
import { SessionExpiredWarning } from './session/network/SessionExpiredWarning';
import { getFocusedSection } from '../state/selectors/section';
import { useDispatch, useSelector } from 'react-redux';
import {
getLeftPaneLists,
getOurPrimaryConversation,
getUnreadMessageCount,
} from '../state/selectors/conversations';
import {
getQuery,
getSearchResults,
isSearching,
} from '../state/selectors/search';
import { clearSearch, search, updateSearchTerm } from '../state/ducks/search';
import { showLeftPaneSection } from '../state/ducks/section';
import { getOurNumber } from '../state/selectors/user';
import { getTheme } from '../state/selectors/theme';
import { applyTheme, ThemeStateType } from '../state/ducks/theme';
// from https://github.com/bvaughn/react-virtualized/blob/fb3484ed5dcc41bffae8eab029126c0fb8f7abc0/source/List/types.js#L5
export type RowRendererParamsType = {
@ -25,132 +37,115 @@ export type RowRendererParamsType = {
style: Object;
};
interface Props {
ourPrimaryConversation: ConversationType;
conversations: Array<ConversationListItemProps>;
contacts: Array<ConversationType>;
unreadMessageCount: number;
searchResults?: SearchResultsProps;
searchTerm: string;
focusedSection: SectionType;
focusedSettingsSection?: SessionSettingCategory;
showLeftPaneSection: (section: SectionType) => void;
showSettingsSection: (section: SessionSettingCategory) => void;
type Props = {
isExpired: boolean;
};
openConversationExternal: (id: string, messageId?: string) => void;
updateSearchTerm: (searchTerm: string) => void;
search: (query: string, options: SearchOptions) => void;
clearSearch: () => void;
theme: DefaultTheme;
}
export class LeftPane extends React.Component<Props> {
public constructor(props: any) {
super(props);
this.handleSectionSelected = this.handleSectionSelected.bind(this);
}
const InnerLeftPaneMessageSection = (props: { isExpired: boolean }) => {
const dispatch = useDispatch();
const showSearch = useSelector(isSearching);
const searchTerm = useSelector(getQuery);
const searchResults = showSearch ? useSelector(getSearchResults) : undefined;
const lists = showSearch ? undefined : useSelector(getLeftPaneLists);
const theme = useSelector(getTheme);
// tslint:disable: use-simple-attributes
return (
<>
<SessionOffline />
{props.isExpired && <SessionExpiredWarning />}
<LeftPaneMessageSection
theme={theme}
openConversationExternal={(id, messageId) =>
dispatch(openConversationExternal(id, messageId))
}
conversations={lists?.conversations || []}
contacts={lists?.contacts || []}
searchResults={searchResults}
searchTerm={searchTerm}
updateSearchTerm={query => dispatch(updateSearchTerm(query))}
search={(query, options) => dispatch(search(query, options))}
clearSearch={() => dispatch(clearSearch())}
/>
</>
);
};
public handleSectionSelected(section: SectionType) {
this.props.clearSearch();
this.props.showLeftPaneSection(section);
}
const InnerLeftPaneContactSection = () => {
const dispatch = useDispatch();
const theme = useSelector(getTheme);
const showSearch = useSelector(isSearching);
const lists = showSearch ? undefined : useSelector(getLeftPaneLists);
const directContacts = lists?.contacts || [];
return (
<>
<SessionOffline />
<LeftPaneContactSection
openConversationExternal={(id, messageId) =>
dispatch(openConversationExternal(id, messageId))
}
directContacts={directContacts}
theme={theme}
/>
</>
);
};
public render(): JSX.Element {
return (
<SessionTheme theme={this.props.theme}>
<div className="module-left-pane-session">
<ActionsPanel
{...this.props}
selectedSection={this.props.focusedSection}
onSectionSelected={this.handleSectionSelected}
/>
<div className="module-left-pane">{this.renderSection()}</div>
</div>
</SessionTheme>
);
}
const LeftPaneSettingsSection = () => {
return <LeftPaneSettingSection />;
};
private renderSection(): JSX.Element | undefined {
switch (this.props.focusedSection) {
case SectionType.Message:
return this.renderMessageSection();
case SectionType.Contact:
return this.renderContactSection();
case SectionType.Settings:
return this.renderSettingSection();
default:
return undefined;
}
}
const LeftPaneSection = (props: { isExpired: boolean }) => {
const focusedSection = useSelector(getFocusedSection);
private renderMessageSection() {
const {
openConversationExternal,
conversations,
contacts,
searchResults,
searchTerm,
updateSearchTerm,
search,
clearSearch,
isExpired,
} = this.props;
return (
<>
<SessionOffline theme={this.props.theme} />
{isExpired && <SessionExpiredWarning theme={this.props.theme} />}
<LeftPaneMessageSection
theme={this.props.theme}
contacts={contacts}
openConversationExternal={openConversationExternal}
conversations={conversations}
searchResults={searchResults}
searchTerm={searchTerm}
updateSearchTerm={updateSearchTerm}
search={search}
clearSearch={clearSearch}
/>
</>
);
if (focusedSection === SectionType.Message) {
return <InnerLeftPaneMessageSection isExpired={props.isExpired} />;
}
private renderContactSection() {
const { openConversationExternal } = this.props;
const directContacts = this.getDirectContactsOnly();
return (
<>
<SessionOffline theme={this.props.theme} />
<LeftPaneContactSection
{...this.props}
openConversationExternal={openConversationExternal}
directContacts={directContacts}
/>
</>
);
if (focusedSection === SectionType.Contact) {
return <InnerLeftPaneContactSection />;
}
private getDirectContactsOnly() {
return this.props.contacts.filter(f => f.type === 'direct');
if (focusedSection === SectionType.Settings) {
return <LeftPaneSettingsSection />;
}
return <></>;
};
private renderSettingSection() {
const settingsCategory =
this.props.focusedSettingsSection || SessionSettingCategory.Appearance;
return (
<>
<LeftPaneSettingSection
{...this.props}
settingsCategory={settingsCategory}
export const LeftPane = (props: Props) => {
const theme = useSelector(getTheme);
const dispatch = useDispatch();
const focusedSection = useSelector(getFocusedSection);
const unreadMessageCount = useSelector(getUnreadMessageCount);
const ourPrimaryConversation = useSelector(getOurPrimaryConversation);
const ourNumber = useSelector(getOurNumber);
return (
<SessionTheme theme={theme}>
<div className="module-left-pane-session">
<ActionsPanel
selectedSection={focusedSection}
onSectionSelected={(section: SectionType) => {
dispatch(clearSearch());
dispatch(showLeftPaneSection(section));
}}
unreadMessageCount={unreadMessageCount}
ourPrimaryConversation={ourPrimaryConversation}
ourNumber={ourNumber}
theme={theme}
applyTheme={(newTheme: ThemeStateType) =>
dispatch(applyTheme(newTheme))
}
/>
</>
);
}
}
<div className="module-left-pane">
<LeftPaneSection isExpired={props.isExpired} />
</div>
</div>
</SessionTheme>
);
};

@ -1,22 +1,18 @@
import React from 'react';
import { useSelector } from 'react-redux';
import { getFocusedSettingsSection } from '../state/selectors/section';
import { SmartSessionConversation } from '../state/smart/SessionConversation';
import {
SessionSettingCategory,
SmartSettingsView,
} from './session/settings/SessionSettings';
import { SmartSettingsView } from './session/settings/SessionSettings';
const FilteredSettingsView = SmartSettingsView as any;
type Props = {
focusedSettingsSection?: SessionSettingCategory;
};
export const SessionMainPanel = (props: Props) => {
const isSettingsView = props.focusedSettingsSection !== undefined;
export const SessionMainPanel = () => {
const focusedSettingsSection = useSelector(getFocusedSettingsSection);
const isSettingsView = focusedSettingsSection !== undefined;
if (isSettingsView) {
return <FilteredSettingsView category={props.focusedSettingsSection} />;
return <FilteredSettingsView category={focusedSettingsSection} />;
}
return (
<div className="session-conversation">

@ -1,17 +1,11 @@
import React from 'react';
import { connect } from 'react-redux';
import { SessionIconButton, SessionIconSize, SessionIconType } from './icon';
import { Avatar } from '../Avatar';
import { darkTheme, lightTheme } from '../../state/ducks/SessionTheme';
import { SessionToastContainer } from './SessionToastContainer';
import { mapDispatchToProps } from '../../state/actions';
import { ConversationType } from '../../state/ducks/conversations';
import { DefaultTheme } from 'styled-components';
import { StateType } from '../../state/reducer';
import { ConversationController } from '../../session/conversations';
import { getFocusedSection } from '../../state/selectors/section';
import { getTheme } from '../../state/selectors/theme';
import { getOurNumber } from '../../state/selectors/user';
import { UserUtils } from '../../session/utils';
import { syncConfigurationIfNeeded } from '../../session/utils/syncUtils';
import { DAYS } from '../../session/utils/Number';
@ -35,12 +29,12 @@ export enum SectionType {
}
interface Props {
onSectionSelected: any;
onSectionSelected: (section: SectionType) => void;
selectedSection: SectionType;
unreadMessageCount: number;
ourPrimaryConversation: ConversationType;
ourNumber: string;
applyTheme?: any;
applyTheme: any;
theme: DefaultTheme;
}
@ -48,7 +42,7 @@ interface Props {
* ActionsPanel is the far left banner (not the left pane).
* The panel with buttons to switch between the message/contact/settings/theme views
*/
class ActionsPanelPrivate extends React.Component<Props> {
export class ActionsPanel extends React.Component<Props> {
private syncInterval: NodeJS.Timeout | null = null;
constructor(props: Props) {
@ -261,15 +255,3 @@ class ActionsPanelPrivate extends React.Component<Props> {
window.showResetSessionIdDialog();
}
}
const mapStateToProps = (state: StateType) => {
return {
section: getFocusedSection(state),
theme: getTheme(state),
ourNumber: getOurNumber(state),
};
};
const smart = connect(mapStateToProps, mapDispatchToProps);
export const ActionsPanel = smart(ActionsPanelPrivate);

@ -1,8 +1,6 @@
import React from 'react';
import classNames from 'classnames';
import { LeftPane } from '../LeftPane';
import {
SessionButton,
SessionButtonColor,
@ -10,14 +8,14 @@ import {
} from './SessionButton';
import { SessionIcon, SessionIconSize, SessionIconType } from './icon';
import { SessionSearchInput } from './SessionSearchInput';
import { SessionSettingCategory } from './settings/SessionSettings';
import { DefaultTheme, useTheme } from 'styled-components';
import { DefaultTheme } from 'styled-components';
import { LeftPaneSectionHeader } from './LeftPaneSectionHeader';
import { deleteAccount } from '../../util/accountManager';
import { useDispatch, useSelector } from 'react-redux';
import { showSettingsSection } from '../../state/ducks/section';
import { getFocusedSettingsSection } from '../../state/selectors/section';
import { getTheme } from '../../state/selectors/theme';
type Props = {
settingsCategory: SessionSettingCategory;
@ -59,7 +57,7 @@ const LeftPaneSettingsCategoryRow = (props: { item: any }) => {
const { item } = props;
const dispatch = useDispatch();
const theme = useTheme();
const theme = useSelector(getTheme);
const focusedSettingsSection = useSelector(getFocusedSettingsSection);
return (
@ -153,12 +151,14 @@ const LeftPaneBottomButtons = () => {
);
};
export const LeftPaneSettingSection = (props: Props) => {
export const LeftPaneSettingSection = () => {
const theme = useSelector(getTheme);
return (
<div className="left-pane-setting-section">
<LeftPaneSectionHeader
label={window.i18n('settingsHeader')}
theme={props.theme}
theme={theme}
/>
<div className="left-pane-setting-content">
<LeftPaneSettingsCategories />

@ -6,13 +6,12 @@ import { ConversationController } from '../../session/conversations';
import { UserUtils } from '../../session/utils';
import { createStore } from '../../state/createStore';
import { actions as conversationActions } from '../../state/ducks/conversations';
import { SmartLeftPane } from '../../state/smart/LeftPane';
import { SmartSessionMainPanel } from '../../state/smart/SessionMainPanel';
import { makeLookup } from '../../util';
import { LeftPane } from '../LeftPane';
import { SessionMainPanel } from '../SessionMainPanel';
// Workaround: A react component's required properties are filtering up through connect()
// https://github.com/DefinitelyTyped/DefinitelyTyped/issues/31363
const FilteredLeftPane = SmartLeftPane as any;
type State = {
isInitialLoadComplete: boolean;
@ -52,13 +51,13 @@ export class SessionInboxView extends React.Component<any, State> {
<div className="network-status-container" />
{this.renderLeftPane()}
</div>
<SmartSessionMainPanel />
<SessionMainPanel />
</Provider>
);
}
private renderLeftPane() {
return <FilteredLeftPane isExpired={this.state.isExpired} />;
return <LeftPane isExpired={this.state.isExpired} />;
}
private async setupLeftPane() {

@ -12,7 +12,7 @@ const SessionExpiredWarningLink = styled.a`
color: black;
`;
export const SessionExpiredWarning = (props: { theme: DefaultTheme }) => {
export const SessionExpiredWarning = () => {
return (
<SessionExpiredWarningContainer>
<div>{window.i18n('expiredWarning')}</div>

@ -1,5 +1,5 @@
import React from 'react';
import styled, { DefaultTheme } from 'styled-components';
import styled from 'styled-components';
import { useNetwork } from '../../../hooks/useNetwork';
type ContainerProps = {
@ -23,7 +23,7 @@ const OfflineTitle = styled.h3`
const OfflineMessage = styled.div``;
export const SessionOffline = (props: { theme: DefaultTheme }) => {
export const SessionOffline = () => {
const isOnline = useNetwork();
return (

@ -426,7 +426,7 @@ function conversationReset({
};
}
function openConversationExternal(
export function openConversationExternal(
id: string,
messageId?: string
): SelectedConversationChangedActionType {

@ -76,7 +76,7 @@ export const actions = {
updateSearchTerm,
};
function search(
export function search(
query: string,
options: SearchOptions
): SearchResultsKickoffActionType {
@ -125,13 +125,13 @@ async function doSearch(
messages: getMessageProps(filteredMessages) || [],
};
}
function clearSearch(): ClearSearchActionType {
export function clearSearch(): ClearSearchActionType {
return {
type: 'SEARCH_CLEAR',
payload: null,
};
}
function updateSearchTerm(query: string): UpdateSearchTermActionType {
export function updateSearchTerm(query: string): UpdateSearchTermActionType {
return {
type: 'SEARCH_UPDATE',
payload: {

@ -1,6 +1,7 @@
export const APPLY_THEME = 'APPLY_THEME';
export type ThemeStateType = typeof lightTheme;
export const applyTheme = (theme: any) => {
export const applyTheme = (theme: ThemeStateType) => {
return {
type: APPLY_THEME,
payload: theme,
@ -8,8 +9,6 @@ export const applyTheme = (theme: any) => {
};
import { lightTheme } from './SessionTheme';
export type ThemeStateType = typeof lightTheme;
const initialState = lightTheme;
export const reducer = (

@ -95,6 +95,7 @@ export const getConversationComparator = createSelector(
_getConversationComparator
);
// export only because we use it in some of our tests
export const _getLeftPaneLists = (
lookup: ConversationLookupType,
comparator: (left: ConversationType, right: ConversationType) => number,
@ -108,7 +109,7 @@ export const _getLeftPaneLists = (
const sorted = values.sort(comparator);
const conversations: Array<ConversationType> = [];
const allContacts: Array<ConversationType> = [];
const directConversations: Array<ConversationType> = [];
let index = 0;
@ -155,8 +156,8 @@ export const _getLeftPaneLists = (
continue;
}
if (conversation.activeAt !== undefined) {
allContacts.push(conversation);
if (conversation.activeAt !== undefined && conversation.type === 'direct') {
directConversations.push(conversation);
}
if (unreadCount < 9 && conversation.unreadCount > 0) {
@ -169,7 +170,7 @@ export const _getLeftPaneLists = (
return {
conversations,
contacts: allContacts,
contacts: directConversations,
unreadCount,
};
};
@ -223,3 +224,10 @@ export const getMe = createSelector(
return lookup[ourNumber];
}
);
export const getUnreadMessageCount = createSelector(
getLeftPaneLists,
(state): number => {
return state.unreadCount;
}
);

@ -28,7 +28,7 @@ export const isSearching = createSelector(
(state: SearchStateType) => {
const { query } = state;
return query && query.trim().length > 1;
return Boolean(query && query.trim().length > 1);
}
);

@ -1,45 +0,0 @@
import { connect } from 'react-redux';
import { LeftPane } from '../../components/LeftPane';
import { StateType } from '../reducer';
import { getQuery, getSearchResults, isSearching } from '../selectors/search';
import { getIntl, getOurNumber } from '../selectors/user';
import {
getLeftPaneLists,
getOurPrimaryConversation,
} from '../selectors/conversations';
import { mapDispatchToProps } from '../actions';
import {
getFocusedSection,
getFocusedSettingsSection,
} from '../selectors/section';
import { getTheme } from '../selectors/theme';
// Workaround: A react component's required properties are filtering up through connect()
// https://github.com/DefinitelyTyped/DefinitelyTyped/issues/3136k3
const mapStateToProps = (state: StateType) => {
const showSearch = isSearching(state);
const leftPaneList = getLeftPaneLists(state);
const lists = showSearch ? undefined : leftPaneList;
const searchResults = showSearch ? getSearchResults(state) : undefined;
const ourPrimaryConversation = getOurPrimaryConversation(state);
return {
...lists,
ourPrimaryConversation, // used in actionPanel
searchTerm: getQuery(state),
ourNumber: getOurNumber(state),
searchResults,
i18n: getIntl(state),
unreadMessageCount: leftPaneList.unreadCount,
theme: getTheme(state),
focusedSection: getFocusedSection(state),
focusedSettingsSection: getFocusedSettingsSection(state),
};
};
const smart = connect(mapStateToProps, mapDispatchToProps);
export const SmartLeftPane = smart(LeftPane);

@ -1,18 +0,0 @@
import { connect } from 'react-redux';
import { StateType } from '../reducer';
import { mapDispatchToProps } from '../actions';
import { getFocusedSettingsSection } from '../selectors/section';
import { getTheme } from '../selectors/theme';
import { SessionMainPanel } from '../../components/SessionMainPanel';
const mapStateToProps = (state: StateType) => {
return {
theme: getTheme(state),
focusedSettingsSection: getFocusedSettingsSection(state),
};
};
const smart = connect(mapStateToProps, mapDispatchToProps);
export const SmartSessionMainPanel = smart(SessionMainPanel);
Loading…
Cancel
Save