signal-desktop/ts/components/MainHeader.tsx

317 lines
8.7 KiB
TypeScript
Raw Normal View History

// Copyright 2018-2021 Signal Messenger, LLC
2020-10-30 15:34:04 -05:00
// SPDX-License-Identifier: AGPL-3.0-only
import React from 'react';
2019-10-17 11:22:07 -07:00
import { Manager, Popper, Reference } from 'react-popper';
import { createPortal } from 'react-dom';
2019-10-17 11:22:07 -07:00
import { showSettings } from '../shims/Whisper';
import { Avatar } from './Avatar';
2019-10-17 11:22:07 -07:00
import { AvatarPopup } from './AvatarPopup';
import type { LocalizerType, ThemeType } from '../types/Util';
import type { AvatarColorType } from '../types/Colors';
import type { ConversationType } from '../state/ducks/conversations';
2021-11-01 13:43:02 -05:00
import { LeftPaneSearchInput } from './LeftPaneSearchInput';
import type { BadgeType } from '../badges/types';
export type PropsType = {
2019-01-14 13:49:58 -08:00
searchTerm: string;
2021-11-01 13:43:02 -05:00
searchConversation: undefined | ConversationType;
2019-11-07 13:36:16 -08:00
startSearchCounter: number;
selectedConversation: undefined | ConversationType;
2019-01-14 13:49:58 -08:00
// For display
2020-07-23 18:35:32 -07:00
phoneNumber?: string;
isMe?: boolean;
name?: string;
2021-05-28 12:15:17 -04:00
color?: AvatarColorType;
disabled?: boolean;
2020-07-23 18:35:32 -07:00
isVerified?: boolean;
profileName?: string;
2020-07-23 18:35:32 -07:00
title: string;
avatarPath?: string;
badge?: BadgeType;
hasPendingUpdate: boolean;
theme: ThemeType;
2019-01-14 13:49:58 -08:00
i18n: LocalizerType;
2019-01-14 13:49:58 -08:00
updateSearchTerm: (searchTerm: string) => void;
startUpdate: () => unknown;
2019-08-09 16:12:29 -07:00
clearConversationSearch: () => void;
2019-01-14 13:49:58 -08:00
clearSearch: () => void;
2019-10-17 11:22:07 -07:00
showArchivedConversations: () => void;
startComposing: () => void;
2021-07-19 15:26:06 -04:00
toggleProfileEditor: () => void;
};
2019-10-17 11:22:07 -07:00
type StateType = {
2019-10-17 11:22:07 -07:00
showingAvatarPopup: boolean;
popperRoot: HTMLDivElement | null;
};
2019-10-17 11:22:07 -07:00
export class MainHeader extends React.Component<PropsType, StateType> {
2019-01-14 13:49:58 -08:00
private readonly inputRef: React.RefObject<HTMLInputElement>;
2019-08-09 16:12:29 -07:00
constructor(props: PropsType) {
2019-01-14 13:49:58 -08:00
super(props);
this.inputRef = React.createRef();
2019-10-17 11:22:07 -07:00
this.state = {
showingAvatarPopup: false,
popperRoot: null,
};
}
public override componentDidUpdate(prevProps: PropsType): void {
2021-11-01 13:43:02 -05:00
const { searchConversation, startSearchCounter } = this.props;
2019-01-14 13:49:58 -08:00
2019-08-09 16:12:29 -07:00
// When user chooses to search in a given conversation we focus the field for them
if (
2021-11-01 13:43:02 -05:00
searchConversation &&
searchConversation.id !== prevProps.searchConversation?.id
2019-08-09 16:12:29 -07:00
) {
this.setFocus();
}
2019-11-07 13:36:16 -08:00
// When user chooses to start a new search, we focus the field
if (startSearchCounter !== prevProps.startSearchCounter) {
this.setSelected();
}
2019-01-14 13:49:58 -08:00
}
2020-09-11 17:46:52 -07:00
public handleOutsideClick = ({ target }: MouseEvent): void => {
2019-10-17 11:22:07 -07:00
const { popperRoot, showingAvatarPopup } = this.state;
if (
showingAvatarPopup &&
popperRoot &&
!popperRoot.contains(target as Node)
) {
this.hideAvatarPopup();
}
};
2020-09-11 17:46:52 -07:00
public showAvatarPopup = (): void => {
const popperRoot = document.createElement('div');
document.body.appendChild(popperRoot);
2019-10-17 11:22:07 -07:00
this.setState({
showingAvatarPopup: true,
popperRoot,
2019-10-17 11:22:07 -07:00
});
document.addEventListener('click', this.handleOutsideClick);
};
2020-09-11 17:46:52 -07:00
public hideAvatarPopup = (): void => {
const { popperRoot } = this.state;
2019-10-17 11:22:07 -07:00
document.removeEventListener('click', this.handleOutsideClick);
2019-10-17 11:22:07 -07:00
this.setState({
showingAvatarPopup: false,
popperRoot: null,
2019-10-17 11:22:07 -07:00
});
if (popperRoot && document.body.contains(popperRoot)) {
document.body.removeChild(popperRoot);
}
2019-10-17 11:22:07 -07:00
};
public override componentDidMount(): void {
document.addEventListener('keydown', this.handleGlobalKeyDown);
}
public override componentWillUnmount(): void {
2019-10-17 11:22:07 -07:00
const { popperRoot } = this.state;
document.removeEventListener('click', this.handleOutsideClick);
document.removeEventListener('keydown', this.handleGlobalKeyDown);
if (popperRoot && document.body.contains(popperRoot)) {
2019-10-17 11:22:07 -07:00
document.body.removeChild(popperRoot);
}
}
2021-11-01 13:43:02 -05:00
private updateSearch = (searchTerm: string): void => {
2019-08-09 16:12:29 -07:00
const {
updateSearchTerm,
clearConversationSearch,
clearSearch,
2021-11-01 13:43:02 -05:00
searchConversation,
2019-08-09 16:12:29 -07:00
} = this.props;
2019-01-14 13:49:58 -08:00
if (!searchTerm) {
2021-11-01 13:43:02 -05:00
if (searchConversation) {
2019-08-09 16:12:29 -07:00
clearConversationSearch();
} else {
clearSearch();
}
2019-01-14 13:49:58 -08:00
return;
}
if (updateSearchTerm) {
updateSearchTerm(searchTerm);
}
2019-08-09 16:12:29 -07:00
};
2019-01-14 13:49:58 -08:00
2020-09-11 17:46:52 -07:00
public clearSearch = (): void => {
2019-01-14 13:49:58 -08:00
const { clearSearch } = this.props;
clearSearch();
this.setFocus();
2019-08-09 16:12:29 -07:00
};
2019-01-14 13:49:58 -08:00
private handleInputBlur = (): void => {
2021-11-01 13:43:02 -05:00
const { clearSearch, searchConversation, searchTerm } = this.props;
if (!searchConversation && !searchTerm) {
2019-01-14 13:49:58 -08:00
clearSearch();
}
2019-08-09 16:12:29 -07:00
};
public handleGlobalKeyDown = (event: KeyboardEvent): void => {
const { showingAvatarPopup } = this.state;
2021-11-01 13:43:02 -05:00
const { key } = event;
if (showingAvatarPopup && key === 'Escape') {
this.hideAvatarPopup();
}
};
2020-09-11 17:46:52 -07:00
public setFocus = (): void => {
2019-01-14 13:49:58 -08:00
if (this.inputRef.current) {
this.inputRef.current.focus();
}
2019-08-09 16:12:29 -07:00
};
2019-01-14 13:49:58 -08:00
2020-09-11 17:46:52 -07:00
public setSelected = (): void => {
2019-11-07 13:36:16 -08:00
if (this.inputRef.current) {
this.inputRef.current.select();
}
};
public override render(): JSX.Element {
const {
avatarPath,
badge,
color,
disabled,
hasPendingUpdate,
2019-08-09 16:12:29 -07:00
i18n,
name,
phoneNumber,
profileName,
2021-11-01 13:43:02 -05:00
searchConversation,
2019-08-09 16:12:29 -07:00
searchTerm,
2019-10-17 11:22:07 -07:00
showArchivedConversations,
startComposing,
startUpdate,
theme,
title,
2021-07-19 15:26:06 -04:00
toggleProfileEditor,
} = this.props;
2019-10-17 11:22:07 -07:00
const { showingAvatarPopup, popperRoot } = this.state;
2021-11-01 13:43:02 -05:00
const isSearching = Boolean(searchConversation || searchTerm.trim().length);
return (
<div className="module-main-header">
2019-10-17 11:22:07 -07:00
<Manager>
<Reference>
{({ ref }) => (
<div className="module-main-header__avatar--container">
<Avatar
acceptedMessageRequest
avatarPath={avatarPath}
badge={badge}
className="module-main-header__avatar"
color={color}
conversationType="direct"
i18n={i18n}
isMe
name={name}
phoneNumber={phoneNumber}
profileName={profileName}
theme={theme}
title={title}
// `sharedGroupNames` makes no sense for yourself, but
// `<Avatar>` needs it to determine blurring.
sharedGroupNames={[]}
size={28}
innerRef={ref}
onClick={this.showAvatarPopup}
/>
{hasPendingUpdate && (
<div className="module-main-header__avatar--badged" />
)}
</div>
2019-10-17 11:22:07 -07:00
)}
</Reference>
{showingAvatarPopup && popperRoot
? createPortal(
<Popper placement="bottom-end">
{({ ref, style }) => (
<AvatarPopup
2021-05-07 17:21:10 -05:00
acceptedMessageRequest
badge={badge}
2019-10-17 11:22:07 -07:00
innerRef={ref}
i18n={i18n}
2021-05-07 17:21:10 -05:00
isMe
2021-10-01 20:01:44 -04:00
style={{ ...style, zIndex: 10 }}
2019-10-17 11:22:07 -07:00
color={color}
conversationType="direct"
name={name}
phoneNumber={phoneNumber}
profileName={profileName}
theme={theme}
2020-07-23 18:35:32 -07:00
title={title}
2019-10-17 11:22:07 -07:00
avatarPath={avatarPath}
size={28}
hasPendingUpdate={hasPendingUpdate}
startUpdate={startUpdate}
2021-05-07 17:21:10 -05:00
// See the comment above about `sharedGroupNames`.
sharedGroupNames={[]}
2021-07-19 15:26:06 -04:00
onEditProfile={() => {
toggleProfileEditor();
this.hideAvatarPopup();
}}
2019-10-17 11:22:07 -07:00
onViewPreferences={() => {
showSettings();
this.hideAvatarPopup();
}}
onViewArchive={() => {
showArchivedConversations();
this.hideAvatarPopup();
}}
/>
)}
</Popper>,
popperRoot
)
: null}
</Manager>
2021-11-01 13:43:02 -05:00
<LeftPaneSearchInput
disabled={disabled}
i18n={i18n}
onBlur={this.handleInputBlur}
onChangeValue={this.updateSearch}
onClear={this.clearSearch}
ref={this.inputRef}
searchConversation={searchConversation}
value={searchTerm}
/>
{!isSearching && (
<button
aria-label={i18n('newConversation')}
className="module-main-header__compose-icon"
onClick={startComposing}
title={i18n('newConversation')}
type="button"
/>
)}
</div>
);
}
}