1
0
mirror of https://github.com/kremalicious/metamask-extension.git synced 2024-11-22 18:00:18 +01:00
metamask-extension/ui/components/component-library/popover/popover.tsx
jainex 798a9b0e60
migration of Popover to use TS Box version (#20279)
* migration of Popover to use TS Box version

* fix lint error

* Updating docs

---------

Co-authored-by: georgewrmarshall <george.marshall@consensys.net>
2023-08-04 11:11:32 -07:00

149 lines
3.7 KiB
TypeScript

import React, { useState, useEffect } from 'react';
import { createPortal } from 'react-dom';
import { usePopper } from 'react-popper';
import classnames from 'classnames';
import {
AlignItems,
BackgroundColor,
BorderColor,
BorderRadius,
Display,
JustifyContent,
} from '../../../helpers/constants/design-system';
import { Box } from '..';
import type { BoxProps, PolymorphicRef } from '../box';
import {
PopoverProps,
PopoverComponent,
PopoverPosition,
PopoverRole,
} from './popover.types';
export const Popover: PopoverComponent = React.forwardRef(
<C extends React.ElementType = 'div'>(
{
children,
className = '',
position = PopoverPosition.Auto,
role = PopoverRole.Tooltip,
hasArrow = false,
matchWidth,
preventOverflow = false,
offset = [0, 8],
flip = false,
referenceHidden = true,
referenceElement,
isOpen,
title,
isPortal = false,
arrowProps,
onPressEscKey,
...props
}: PopoverProps<C>,
ref?: PolymorphicRef<C>,
) => {
const [popperElement, setPopperElement] = useState<HTMLElement | null>(
null,
);
const [arrowElement, setArrowElement] = useState<HTMLElement | null>(null);
// Define Popper options
const { styles, attributes } = usePopper(referenceElement, popperElement, {
placement: position,
modifiers: [
{
name: 'preventOverflow',
enabled: position === PopoverPosition.Auto ? true : preventOverflow,
},
{
name: 'flip',
enabled: position === PopoverPosition.Auto ? true : flip,
},
{
name: 'arrow',
enabled: hasArrow,
options: {
element: arrowElement,
},
},
{
name: 'offset',
options: {
offset,
},
},
],
});
// Define width to match reference element or auto
const contentStyle = {
width: matchWidth ? referenceElement?.clientWidth : 'auto',
};
// Esc key press
const handleEscKey = (event: KeyboardEvent) => {
if (event.key === 'Escape') {
// Close the popover when the "Esc" key is pressed
if (onPressEscKey) {
onPressEscKey();
}
}
};
useEffect(() => {
document.addEventListener('keydown', handleEscKey);
return () => {
document.removeEventListener('keydown', handleEscKey);
};
}, [onPressEscKey]);
const PopoverContent = (
<Box
borderColor={BorderColor.borderMuted}
borderRadius={BorderRadius.LG}
backgroundColor={BackgroundColor.backgroundDefault}
padding={4}
role={role}
className={classnames(
'mm-popover',
{
'mm-popover--open': Boolean(isOpen),
'mm-popover--reference-hidden': Boolean(referenceHidden),
},
className,
)}
ref={ref || setPopperElement}
{...attributes.popper}
{...(props as BoxProps<C>)}
style={{ ...styles.popper, ...contentStyle, ...props.style }}
>
{children}
{hasArrow && (
<Box
borderColor={BorderColor.borderMuted}
className={classnames('mm-popover__arrow')}
ref={setArrowElement}
display={Display.Flex}
justifyContent={JustifyContent.center}
alignItems={AlignItems.center}
style={styles.arrow}
{...attributes.arrow}
{...(arrowProps as BoxProps<'div'>)}
/>
)}
</Box>
);
return (
<>
{isPortal
? isOpen && createPortal(PopoverContent, document.body)
: isOpen && PopoverContent}
</>
);
},
);