// Copyright 2018 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import type { VirtualElement } from '@popperjs/core'; import React from 'react'; import { Manager, Popper, Reference } from 'react-popper'; import { createPortal } from 'react-dom'; import { showSettings } from '../shims/Whisper'; import { Avatar, AvatarSize } from './Avatar'; 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 = { areStoriesEnabled: boolean; avatarPath?: string; badge?: BadgeType; color?: AvatarColorType; hasPendingUpdate: boolean; i18n: LocalizerType; isMe?: boolean; isVerified?: boolean; name?: string; phoneNumber?: string; profileName?: string; theme: ThemeType; title: string; unreadStoriesCount: number; showArchivedConversations: () => void; startComposing: () => void; startUpdate: () => unknown; toggleProfileEditor: () => void; toggleStoriesView: () => unknown; }; type StateType = { 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), }; } export class MainHeader extends React.Component { public containerRef: React.RefObject = React.createRef(); constructor(props: PropsType) { super(props); this.state = { showingAvatarPopup: false, popperRoot: null, virtualElement: generateVirtualElement(0, 0), }; } 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; }, { containerElements: [popperRoot, this.containerRef], name: 'MainHeader.showAvatarPopup', } ); this.setState({ showingAvatarPopup: true, popperRoot, outsideClickDestructor, virtualElement: generateVirtualElement(ev.clientX, ev.clientY), }); }; public hideAvatarPopup = (): void => { const { popperRoot, outsideClickDestructor } = this.state; this.setState({ showingAvatarPopup: false, popperRoot: null, outsideClickDestructor: undefined, }); outsideClickDestructor?.(); if (popperRoot && document.body.contains(popperRoot)) { document.body.removeChild(popperRoot); } }; 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)) { document.body.removeChild(popperRoot); } } public handleGlobalKeyDown = (event: KeyboardEvent): void => { const { showingAvatarPopup } = this.state; const { key } = event; if (showingAvatarPopup && key === 'Escape') { this.hideAvatarPopup(); } }; public override render(): JSX.Element { const { areStoriesEnabled, avatarPath, badge, color, hasPendingUpdate, i18n, name, phoneNumber, profileName, showArchivedConversations, startComposing, startUpdate, theme, title, toggleProfileEditor, toggleStoriesView, unreadStoriesCount, } = this.props; const { showingAvatarPopup, popperRoot } = this.state; return (
{({ ref }) => (
` needs it to determine blurring. sharedGroupNames={[]} size={AvatarSize.TWENTY_EIGHT} innerRef={ref} onClick={this.showAvatarPopup} /> {hasPendingUpdate && (
)}
)} {showingAvatarPopup && popperRoot ? createPortal( {({ ref, style }) => ( { toggleProfileEditor(); this.hideAvatarPopup(); }} onViewPreferences={() => { showSettings(); this.hideAvatarPopup(); }} onViewArchive={() => { showArchivedConversations(); this.hideAvatarPopup(); }} /> )} , popperRoot ) : null}
{areStoriesEnabled && ( )}
); } }