signal-desktop/ts/components/Tooltip.tsx

128 lines
3.1 KiB
TypeScript
Raw Normal View History

// Copyright 2020-2021 Signal Messenger, LLC
2020-11-19 18:11:35 +00:00
// SPDX-License-Identifier: AGPL-3.0-only
import React from 'react';
import classNames from 'classnames';
import { noop } from 'lodash';
2020-11-19 18:11:35 +00:00
import { Manager, Reference, Popper } from 'react-popper';
import { Theme, themeClassName } from '../util/theme';
import { refMerger } from '../util/refMerger';
import { offsetDistanceModifier } from '../util/popperUtil';
2020-11-19 18:11:35 +00:00
type EventWrapperPropsType = {
children: React.ReactNode;
onHoverChanged: (_: boolean) => void;
};
// React doesn't reliably fire `onMouseLeave` or `onMouseOut` events if wrapping a
// disabled button. This uses native browser events to avoid that.
//
// See <https://lecstor.com/react-disabled-button-onmouseleave/>.
const TooltipEventWrapper = React.forwardRef<
HTMLSpanElement,
EventWrapperPropsType
>(({ onHoverChanged, children }, ref) => {
const wrapperRef = React.useRef<HTMLSpanElement | null>(null);
2021-03-01 20:08:37 +00:00
const on = React.useCallback(() => {
onHoverChanged(true);
}, [onHoverChanged]);
const off = React.useCallback(() => {
onHoverChanged(false);
}, [onHoverChanged]);
React.useEffect(() => {
const wrapperEl = wrapperRef.current;
if (!wrapperEl) {
return noop;
}
wrapperEl.addEventListener('mouseenter', on);
wrapperEl.addEventListener('mouseleave', off);
return () => {
wrapperEl.removeEventListener('mouseenter', on);
wrapperEl.removeEventListener('mouseleave', off);
};
2021-03-01 20:08:37 +00:00
}, [on, off]);
return (
<span
2021-03-01 20:08:37 +00:00
onFocus={on}
onBlur={off}
ref={refMerger<HTMLSpanElement>(ref, wrapperRef)}
>
{children}
</span>
);
});
2020-11-19 18:11:35 +00:00
export enum TooltipPlacement {
Top = 'top',
Right = 'right',
Bottom = 'bottom',
Left = 'left',
}
export type PropsType = {
content: string | JSX.Element;
className?: string;
2020-11-19 18:11:35 +00:00
direction?: TooltipPlacement;
sticky?: boolean;
theme?: Theme;
2020-11-19 18:11:35 +00:00
};
export const Tooltip: React.FC<PropsType> = ({
children,
className,
2020-11-19 18:11:35 +00:00
content,
direction,
sticky,
theme,
2020-11-19 18:11:35 +00:00
}) => {
const [isHovering, setIsHovering] = React.useState(false);
const showTooltip = isHovering || Boolean(sticky);
2020-11-19 18:11:35 +00:00
const tooltipThemeClassName = theme
? `module-tooltip--${themeClassName(theme)}`
: undefined;
2020-11-19 23:38:59 +00:00
2020-11-19 18:11:35 +00:00
return (
<Manager>
<Reference>
{({ ref }) => (
<TooltipEventWrapper ref={ref} onHoverChanged={setIsHovering}>
2020-11-19 18:11:35 +00:00
{children}
</TooltipEventWrapper>
2020-11-19 18:11:35 +00:00
)}
</Reference>
<Popper placement={direction} modifiers={[offsetDistanceModifier(12)]}>
2020-11-19 18:11:35 +00:00
{({ arrowProps, placement, ref, style }) =>
showTooltip && (
<div
className={classNames(
'module-tooltip',
tooltipThemeClassName,
className
)}
ref={ref}
style={style}
data-placement={placement}
>
{content}
2020-11-19 18:11:35 +00:00
<div
className="module-tooltip-arrow"
ref={arrowProps.ref}
style={arrowProps.style}
/>
2020-11-19 18:11:35 +00:00
</div>
)
}
</Popper>
</Manager>
);
};