diff --git a/web/modals/modal.react.js b/web/modals/modal.react.js --- a/web/modals/modal.react.js +++ b/web/modals/modal.react.js @@ -1,7 +1,6 @@ // @flow import classNames from 'classnames'; -import invariant from 'invariant'; import * as React from 'react'; import css from './modal.css'; @@ -14,72 +13,77 @@ +size?: ModalSize, +fixedHeight?: boolean, }; -class Modal extends React.PureComponent { - static defaultProps: { +size: ModalSize, fixedHeight: boolean } = { - size: 'small', - fixedHeight: true, - }; - overlay: ?HTMLDivElement; - componentDidMount() { - invariant(this.overlay, 'overlay ref unset'); - this.overlay.focus(); - } +function Modal(props: Props): React.Node { + const { size = 'small', children, onClose, fixedHeight = true, name } = props; + const overlayRef = React.useRef(); - render(): React.Node { - const { size, children, onClose, fixedHeight, name } = this.props; + const onBackgroundClick = React.useCallback( + event => { + if (event.target === overlayRef.current) { + onClose(); + } + }, + [onClose], + ); - const overlayClasses = classNames(css['modal-overlay'], { - [css['resizable-modal-overlay']]: !fixedHeight, - }); - const modalContainerClasses = classNames(css['modal-container'], { - [css['large-modal-container']]: size === 'large', - }); - const modalClasses = classNames(css['modal'], { - [css['fixed-height-modal']]: fixedHeight, - }); - return ( -
-
-
-
- - × - -

{name}

-
- {children} -
-
-
- ); - } - - overlayRef: (overlay: ?HTMLDivElement) => void = overlay => { - this.overlay = overlay; - }; + const onKeyDown = React.useCallback( + event => { + if (event.keyCode === 27) { + onClose(); + } + }, + [onClose], + ); - onBackgroundClick: ( - event: SyntheticEvent, - ) => void = event => { - if (event.target === this.overlay) { - this.props.onClose(); + React.useEffect(() => { + if (overlayRef.current) { + overlayRef.current.focus(); } - }; + }, []); - onKeyDown: ( - event: SyntheticKeyboardEvent, - ) => void = event => { - if (event.keyCode === 27) { - this.props.onClose(); - } - }; + const overlayClasses = React.useMemo( + () => + classNames(css['modal-overlay'], { + [css['resizable-modal-overlay']]: !fixedHeight, + }), + [fixedHeight], + ); + const modalContainerClasses = React.useMemo( + () => + classNames(css['modal-container'], { + [css['large-modal-container']]: size === 'large', + }), + [size], + ); + const modalClasses = React.useMemo( + () => + classNames(css['modal'], { + [css['fixed-height-modal']]: fixedHeight, + }), + [fixedHeight], + ); + return ( +
+
+
+
+ + × + +

{name}

+
+ {children} +
+
+
+ ); } export default Modal;