signal-desktop/ts/components/LeftPaneDialog.tsx

195 lines
4.2 KiB
TypeScript
Raw Normal View History

2021-09-17 22:20:49 +00:00
// Copyright 2021 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import type { ReactChild, ReactNode } from 'react';
import React from 'react';
2021-09-17 22:20:49 +00:00
import classNames from 'classnames';
import { Tooltip, TooltipPlacement } from './Tooltip';
import { WidthBreakpoint } from './_util';
2021-09-17 22:20:49 +00:00
const BASE_CLASS_NAME = 'LeftPaneDialog';
const TOOLTIP_CLASS_NAME = `${BASE_CLASS_NAME}__tooltip`;
2021-09-17 22:20:49 +00:00
export type PropsType = {
type?: 'warning' | 'error';
icon?: 'update' | 'relink' | 'network' | 'warning' | ReactChild;
2021-09-17 22:20:49 +00:00
title?: string;
subtitle?: string;
children?: ReactNode;
hoverText?: string;
containerWidthBreakpoint: WidthBreakpoint;
2021-09-17 22:20:49 +00:00
} & (
| {
onClick?: undefined;
clickLabel?: undefined;
hasAction?: false;
}
| {
onClick: () => void;
clickLabel: string;
hasAction: true;
}
) &
(
| {
onClose?: undefined;
closeLabel?: undefined;
hasXButton?: false;
}
| {
onClose: () => void;
closeLabel: string;
hasXButton: true;
}
);
2022-11-18 00:45:19 +00:00
export function LeftPaneDialog({
icon = 'warning',
2021-09-17 22:20:49 +00:00
type,
onClick,
clickLabel,
title,
subtitle,
children,
hoverText,
hasAction,
containerWidthBreakpoint,
2021-09-17 22:20:49 +00:00
hasXButton,
onClose,
closeLabel,
2022-11-18 00:45:19 +00:00
}: PropsType): JSX.Element {
2021-09-17 22:20:49 +00:00
const onClickWrap = (e: React.MouseEvent) => {
e.preventDefault();
e.stopPropagation();
onClick?.();
};
2021-10-05 14:22:41 +00:00
const onKeyDownWrap = (e: React.KeyboardEvent) => {
if (e.key !== 'Enter' && e.key !== ' ') {
return;
}
e.preventDefault();
e.stopPropagation();
onClick?.();
};
2021-09-17 22:20:49 +00:00
const onCloseWrap = (e: React.MouseEvent) => {
e.preventDefault();
e.stopPropagation();
onClose?.();
};
const iconClassName =
typeof icon === 'string'
? classNames([
`${BASE_CLASS_NAME}__icon`,
`${BASE_CLASS_NAME}__icon--${icon}`,
])
: undefined;
let action: ReactNode;
if (hasAction) {
action = (
<button
title={clickLabel}
aria-label={clickLabel}
className={`${BASE_CLASS_NAME}__action-text`}
onClick={onClickWrap}
tabIndex={0}
type="button"
>
{clickLabel}
</button>
);
}
let xButton: ReactNode;
2023-09-11 20:55:51 +00:00
if (hasXButton && containerWidthBreakpoint !== WidthBreakpoint.Narrow) {
2021-09-17 22:20:49 +00:00
xButton = (
<div className={`${BASE_CLASS_NAME}__container-close`}>
<button
title={closeLabel}
aria-label={closeLabel}
className={`${BASE_CLASS_NAME}__close-button`}
onClick={onCloseWrap}
tabIndex={0}
type="button"
/>
</div>
);
}
2023-09-11 20:55:51 +00:00
const className = classNames(BASE_CLASS_NAME, {
[`${BASE_CLASS_NAME}--width-narrow`]:
containerWidthBreakpoint === WidthBreakpoint.Narrow,
[`${BASE_CLASS_NAME}--${type}`]: type != null,
[`${BASE_CLASS_NAME}--clickable`]: onClick != null,
});
2021-09-17 22:20:49 +00:00
const message = (
<>
{title === undefined ? undefined : <h3>{title}</h3>}
{subtitle === undefined ? undefined : <div>{subtitle}</div>}
{children}
{action}
</>
);
2021-09-17 22:20:49 +00:00
const content = (
<>
<div className={`${BASE_CLASS_NAME}__container`}>
{typeof icon === 'string' ? <div className={iconClassName} /> : icon}
2023-09-11 20:55:51 +00:00
{containerWidthBreakpoint !== WidthBreakpoint.Narrow && (
<div className={`${BASE_CLASS_NAME}__message`}>{message}</div>
)}
2021-09-17 22:20:49 +00:00
</div>
{xButton}
</>
);
let dialogNode: ReactChild;
2021-09-17 22:20:49 +00:00
if (onClick) {
dialogNode = (
2021-10-05 14:22:41 +00:00
<div
2021-09-17 22:20:49 +00:00
className={className}
2021-10-05 14:22:41 +00:00
role="button"
2021-09-17 22:20:49 +00:00
onClick={onClickWrap}
2021-10-05 14:22:41 +00:00
onKeyDown={onKeyDownWrap}
2021-09-17 22:20:49 +00:00
aria-label={clickLabel}
title={hoverText}
tabIndex={0}
>
{content}
2021-10-05 14:22:41 +00:00
</div>
2021-09-17 22:20:49 +00:00
);
} else {
dialogNode = (
<div className={className} title={hoverText}>
{content}
</div>
);
2021-09-17 22:20:49 +00:00
}
if (containerWidthBreakpoint === WidthBreakpoint.Narrow) {
return (
<Tooltip
content={message}
direction={TooltipPlacement.Right}
className={classNames(
TOOLTIP_CLASS_NAME,
type && `${TOOLTIP_CLASS_NAME}--${type}`
)}
>
{dialogNode}
</Tooltip>
);
}
return dialogNode;
2022-11-18 00:45:19 +00:00
}