signal-desktop/ts/components/MainHeader.tsx

278 lines
8.2 KiB
TypeScript
Raw Normal View History

2023-01-03 19:55:46 +00:00
// Copyright 2018 Signal Messenger, LLC
2020-10-30 20:34:04 +00:00
// SPDX-License-Identifier: AGPL-3.0-only
import type { VirtualElement } from '@popperjs/core';
import React from 'react';
2019-10-17 18:22:07 +00:00
import { Manager, Popper, Reference } from 'react-popper';
import { createPortal } from 'react-dom';
2019-10-17 18:22:07 +00:00
import { showSettings } from '../shims/Whisper';
2022-12-09 20:37:45 +00:00
import { Avatar, AvatarSize } from './Avatar';
2019-10-17 18:22:07 +00:00
import { AvatarPopup } from './AvatarPopup';
import type { LocalizerType, ThemeType } from '../types/Util';
import type { AvatarColorType } from '../types/Colors';
import type { BadgeType } from '../badges/types';
import { handleOutsideClick } from '../util/handleOutsideClick';
export type PropsType = {
2022-03-04 21:14:52 +00:00
areStoriesEnabled: boolean;
avatarPath?: string;
badge?: BadgeType;
2022-01-27 22:12:26 +00:00
color?: AvatarColorType;
hasPendingUpdate: boolean;
2019-01-14 21:49:58 +00:00
i18n: LocalizerType;
2022-01-27 22:12:26 +00:00
isMe?: boolean;
isVerified?: boolean;
name?: string;
phoneNumber?: string;
profileName?: string;
theme: ThemeType;
title: string;
2023-02-07 19:33:04 +00:00
hasFailedStorySends?: boolean;
2022-07-20 23:06:15 +00:00
unreadStoriesCount: number;
2019-10-17 18:22:07 +00:00
showArchivedConversations: () => void;
startComposing: () => void;
2022-01-27 22:12:26 +00:00
startUpdate: () => unknown;
2021-07-19 19:26:06 +00:00
toggleProfileEditor: () => void;
2022-03-04 21:14:52 +00:00
toggleStoriesView: () => unknown;
};
2019-10-17 18:22:07 +00:00
type StateType = {
2019-10-17 18:22:07 +00:00
showingAvatarPopup: boolean;
popperRoot: HTMLDivElement | null;
outsideClickDestructor?: () => void;
virtualElement: {
getBoundingClientRect: () => DOMRect;
};
};
// https://popper.js.org/docs/v2/virtual-elements/
// Generating a virtual element here so that we can make the menu pop up
// right under the mouse cursor.
function generateVirtualElement(x: number, y: number): VirtualElement {
return {
getBoundingClientRect: () => new DOMRect(x, y),
};
}
2019-10-17 18:22:07 +00:00
export class MainHeader extends React.Component<PropsType, StateType> {
2022-06-15 17:53:08 +00:00
public containerRef: React.RefObject<HTMLDivElement> = React.createRef();
2019-08-09 23:12:29 +00:00
constructor(props: PropsType) {
2019-01-14 21:49:58 +00:00
super(props);
2019-10-17 18:22:07 +00:00
this.state = {
showingAvatarPopup: false,
popperRoot: null,
virtualElement: generateVirtualElement(0, 0),
2019-10-17 18:22:07 +00:00
};
}
public showAvatarPopup = (ev: React.MouseEvent): void => {
const popperRoot = document.createElement('div');
document.body.appendChild(popperRoot);
const outsideClickDestructor = handleOutsideClick(
() => {
const { showingAvatarPopup } = this.state;
if (!showingAvatarPopup) {
return false;
}
this.hideAvatarPopup();
return true;
},
2022-09-27 20:24:21 +00:00
{
containerElements: [popperRoot, this.containerRef],
name: 'MainHeader.showAvatarPopup',
}
);
2019-10-17 18:22:07 +00:00
this.setState({
showingAvatarPopup: true,
popperRoot,
outsideClickDestructor,
virtualElement: generateVirtualElement(ev.clientX, ev.clientY),
2019-10-17 18:22:07 +00:00
});
};
2020-09-12 00:46:52 +00:00
public hideAvatarPopup = (): void => {
const { popperRoot, outsideClickDestructor } = this.state;
2019-10-17 18:22:07 +00:00
this.setState({
showingAvatarPopup: false,
popperRoot: null,
outsideClickDestructor: undefined,
2019-10-17 18:22:07 +00:00
});
outsideClickDestructor?.();
if (popperRoot && document.body.contains(popperRoot)) {
document.body.removeChild(popperRoot);
}
2019-10-17 18:22:07 +00:00
};
public override componentDidMount(): void {
const useCapture = true;
document.addEventListener('keydown', this.handleGlobalKeyDown, useCapture);
}
public override componentWillUnmount(): void {
const { popperRoot, outsideClickDestructor } = this.state;
const useCapture = true;
outsideClickDestructor?.();
document.removeEventListener(
'keydown',
this.handleGlobalKeyDown,
useCapture
);
if (popperRoot && document.body.contains(popperRoot)) {
2019-10-17 18:22:07 +00:00
document.body.removeChild(popperRoot);
}
}
public handleGlobalKeyDown = (event: KeyboardEvent): void => {
const { showingAvatarPopup } = this.state;
2021-11-01 18:43:02 +00:00
const { key } = event;
if (showingAvatarPopup && key === 'Escape') {
this.hideAvatarPopup();
}
};
public override render(): JSX.Element {
const {
2022-03-04 21:14:52 +00:00
areStoriesEnabled,
avatarPath,
badge,
color,
2023-02-07 19:33:04 +00:00
hasFailedStorySends,
hasPendingUpdate,
2019-08-09 23:12:29 +00:00
i18n,
name,
phoneNumber,
profileName,
2019-10-17 18:22:07 +00:00
showArchivedConversations,
startComposing,
startUpdate,
theme,
title,
2021-07-19 19:26:06 +00:00
toggleProfileEditor,
2022-03-04 21:14:52 +00:00
toggleStoriesView,
2022-07-20 23:06:15 +00:00
unreadStoriesCount,
} = this.props;
2019-10-17 18:22:07 +00:00
const { showingAvatarPopup, popperRoot } = this.state;
return (
<div className="module-main-header">
2019-10-17 18:22:07 +00:00
<Manager>
<Reference>
{({ ref }) => (
2022-06-15 17:53:08 +00:00
<div
className="module-main-header__avatar--container"
ref={this.containerRef}
>
<Avatar
acceptedMessageRequest
avatarPath={avatarPath}
badge={badge}
className="module-main-header__avatar"
color={color}
conversationType="direct"
i18n={i18n}
isMe
phoneNumber={phoneNumber}
profileName={profileName}
theme={theme}
title={title}
// `sharedGroupNames` makes no sense for yourself, but
// `<Avatar>` needs it to determine blurring.
sharedGroupNames={[]}
2022-12-09 20:37:45 +00:00
size={AvatarSize.TWENTY_EIGHT}
innerRef={ref}
onClick={this.showAvatarPopup}
/>
{hasPendingUpdate && (
<div className="module-main-header__avatar--badged" />
)}
</div>
2019-10-17 18:22:07 +00:00
)}
</Reference>
{showingAvatarPopup && popperRoot
? createPortal(
<Popper referenceElement={this.state.virtualElement}>
2019-10-17 18:22:07 +00:00
{({ ref, style }) => (
<AvatarPopup
2021-05-07 22:21:10 +00:00
acceptedMessageRequest
badge={badge}
2019-10-17 18:22:07 +00:00
innerRef={ref}
i18n={i18n}
2021-05-07 22:21:10 +00:00
isMe
2021-10-02 00:01:44 +00:00
style={{ ...style, zIndex: 10 }}
2019-10-17 18:22:07 +00:00
color={color}
conversationType="direct"
name={name}
phoneNumber={phoneNumber}
profileName={profileName}
theme={theme}
2020-07-24 01:35:32 +00:00
title={title}
2019-10-17 18:22:07 +00:00
avatarPath={avatarPath}
hasPendingUpdate={hasPendingUpdate}
startUpdate={startUpdate}
2021-05-07 22:21:10 +00:00
// See the comment above about `sharedGroupNames`.
sharedGroupNames={[]}
2021-07-19 19:26:06 +00:00
onEditProfile={() => {
toggleProfileEditor();
this.hideAvatarPopup();
}}
2019-10-17 18:22:07 +00:00
onViewPreferences={() => {
showSettings();
this.hideAvatarPopup();
}}
onViewArchive={() => {
showArchivedConversations();
this.hideAvatarPopup();
}}
/>
)}
</Popper>,
popperRoot
)
: null}
</Manager>
2022-03-04 21:14:52 +00:00
<div className="module-main-header__icon-container">
{areStoriesEnabled && (
<button
2023-03-30 00:03:25 +00:00
aria-label={i18n('icu:stories')}
2022-03-04 21:14:52 +00:00
className="module-main-header__stories-icon"
onClick={toggleStoriesView}
2023-03-30 00:03:25 +00:00
title={i18n('icu:stories')}
2022-03-04 21:14:52 +00:00
type="button"
2022-07-20 23:06:15 +00:00
>
2023-02-07 19:33:04 +00:00
{hasFailedStorySends && (
<span className="module-main-header__stories-badge">!</span>
)}
{!hasFailedStorySends && unreadStoriesCount ? (
2022-07-20 23:06:15 +00:00
<span className="module-main-header__stories-badge">
{unreadStoriesCount}
</span>
) : undefined}
</button>
2022-03-04 21:14:52 +00:00
)}
<button
2023-03-30 00:03:25 +00:00
aria-label={i18n('icu:newConversation')}
2022-03-04 21:14:52 +00:00
className="module-main-header__compose-icon"
onClick={startComposing}
2023-03-30 00:03:25 +00:00
title={i18n('icu:newConversation')}
2022-03-04 21:14:52 +00:00
type="button"
/>
</div>
</div>
);
}
}