@material-ui/core
Version:
React components that implement Google's Material Design.
409 lines (347 loc) • 11.8 kB
JavaScript
import _extends from "@babel/runtime/helpers/extends";
import _objectWithoutPropertiesLoose from "@babel/runtime/helpers/objectWithoutPropertiesLoose";
import React from 'react';
import ReactDOM from 'react-dom';
import PropTypes from 'prop-types';
import ownerDocument from '../utils/ownerDocument';
import Portal from '../Portal';
import { createChainedFunction } from '../utils/helpers';
import { setRef } from '../utils/reactHelpers';
import withForwardedRef from '../utils/withForwardedRef';
import { withTheme } from '@material-ui/styles';
import { elementAcceptingRef } from '@material-ui/utils';
import zIndex from '../styles/zIndex';
import ModalManager from './ModalManager';
import TrapFocus from './TrapFocus';
import SimpleBackdrop from './SimpleBackdrop';
import { ariaHidden } from './manageAriaHidden';
function getContainer(container) {
container = typeof container === 'function' ? container() : container;
return ReactDOM.findDOMNode(container);
}
function getHasTransition(props) {
return props.children ? props.children.props.hasOwnProperty('in') : false;
}
export const styles = theme => ({
/* Styles applied to the root element. */
root: {
position: 'fixed',
zIndex: theme.zIndex.modal,
right: 0,
bottom: 0,
top: 0,
left: 0
},
/* Styles applied to the root element if the `Modal` has exited. */
hidden: {
visibility: 'hidden'
}
});
/**
* Modal is a lower-level construct that is leveraged by the following components:
*
* - [Dialog](/api/dialog/)
* - [Drawer](/api/drawer/)
* - [Menu](/api/menu/)
* - [Popover](/api/popover/)
*
* If you are creating a modal dialog, you probably want to use the [Dialog](/api/dialog/) component
* rather than directly using Modal.
*
* This component shares many concepts with [react-overlays](https://react-bootstrap.github.io/react-overlays/#modals).
*/
class Modal extends React.Component {
constructor(props) {
super();
this.handleOpen = () => {
const container = getContainer(this.props.container) || this.getDoc().body;
this.props.manager.add(this, container);
if (this.modalRef) {
this.handleOpened();
}
};
this.handleRendered = () => {
if (this.props.onRendered) {
this.props.onRendered();
}
if (this.props.open) {
this.handleOpened();
} else {
ariaHidden(this.modalRef, true);
}
};
this.handleOpened = () => {
this.props.manager.mount(this); // Fix a bug on Chrome where the scroll isn't initially 0.
this.modalRef.scrollTop = 0;
};
this.handleClose = reason => {
const hasTransition = getHasTransition(this.props);
/* If the component does not have a transition or is unmounting remove the Modal
otherwise let the transition handle removing the style, this prevents elements
moving around when the Modal is closed. */
if (!(hasTransition && this.props.closeAfterTransition) || reason === 'unmount') {
this.props.manager.remove(this);
}
};
this.handleEnter = () => {
this.setState({
exited: false
});
};
this.handleExited = () => {
if (this.props.closeAfterTransition) {
this.props.manager.remove(this);
}
this.setState({
exited: true
});
};
this.handleBackdropClick = event => {
if (event.target !== event.currentTarget) {
return;
}
if (this.props.onBackdropClick) {
this.props.onBackdropClick(event);
}
if (!this.props.disableBackdropClick && this.props.onClose) {
this.props.onClose(event, 'backdropClick');
}
};
this.handleKeyDown = event => {
// We don't take event.defaultPrevented into account:
//
// event.preventDefault() is meant to stop default behaviours like
// clicking a checkbox to check it, hitting a button to submit a form,
// and hitting left arrow to move the cursor in a text input etc.
// Only special HTML elements have these default behaviors.
if (event.key !== 'Escape' || !this.isTopModal()) {
return;
} // Swallow the event, in case someone is listening for the escape key on the body.
event.stopPropagation();
if (this.props.onEscapeKeyDown) {
this.props.onEscapeKeyDown(event);
}
if (!this.props.disableEscapeKeyDown && this.props.onClose) {
this.props.onClose(event, 'escapeKeyDown');
}
};
this.handlePortalRef = ref => {
this.mountNode = ref;
};
this.handleModalRef = ref => {
this.modalRef = ref;
setRef(this.props.innerRef, ref);
};
this.isTopModal = () => {
return this.props.manager.isTopModal(this);
};
this.getDoc = () => {
return ownerDocument(this.mountNode);
};
this.state = {
exited: !props.open
};
}
componentDidMount() {
if (this.props.open) {
this.handleOpen();
}
}
componentDidUpdate(prevProps) {
if (prevProps.open && !this.props.open) {
this.handleClose();
} else if (!prevProps.open && this.props.open) {
this.handleOpen();
}
}
componentWillUnmount() {
if (this.props.open || getHasTransition(this.props) && !this.state.exited) {
this.handleClose('unmount');
}
}
render() {
const _this$props = this.props,
{
BackdropComponent,
BackdropProps,
children,
container,
disableAutoFocus,
disableEnforceFocus,
disablePortal,
disableRestoreFocus,
hideBackdrop,
keepMounted,
open,
theme
} = _this$props,
other = _objectWithoutPropertiesLoose(_this$props, ["BackdropComponent", "BackdropProps", "children", "closeAfterTransition", "container", "disableAutoFocus", "disableBackdropClick", "disableEnforceFocus", "disableEscapeKeyDown", "disablePortal", "disableRestoreFocus", "hideBackdrop", "innerRef", "keepMounted", "manager", "onBackdropClick", "onClose", "onEscapeKeyDown", "onRendered", "open", "theme"]);
const {
exited
} = this.state;
const hasTransition = getHasTransition(this.props);
if (!keepMounted && !open && (!hasTransition || exited)) {
return null;
}
const childProps = {}; // It's a Transition like component
if (hasTransition) {
childProps.onEnter = createChainedFunction(this.handleEnter, children.props.onEnter);
childProps.onExited = createChainedFunction(this.handleExited, children.props.onExited);
}
if (children.props.role === undefined) {
childProps.role = children.props.role || 'document';
}
if (children.props.tabIndex === undefined) {
childProps.tabIndex = children.props.tabIndex || '-1';
}
const stylesRender = styles(theme || {
zIndex
});
return React.createElement(Portal, {
ref: this.handlePortalRef,
container: container,
disablePortal: disablePortal,
onRendered: this.handleRendered
}, React.createElement("div", _extends({
ref: this.handleModalRef,
onKeyDown: this.handleKeyDown,
role: "presentation"
}, other, {
style: _extends({}, stylesRender.root, !open && exited ? stylesRender.hidden : {}, other.style)
}), hideBackdrop ? null : React.createElement(BackdropComponent, _extends({
open: open,
onClick: this.handleBackdropClick
}, BackdropProps)), React.createElement(TrapFocus, {
disableEnforceFocus: disableEnforceFocus,
disableAutoFocus: disableAutoFocus,
disableRestoreFocus: disableRestoreFocus,
getDoc: this.getDoc,
isEnabled: this.isTopModal,
open: open
}, React.cloneElement(children, childProps))));
}
}
process.env.NODE_ENV !== "production" ? Modal.propTypes = {
/**
* A backdrop component. This property enables custom backdrop rendering.
*/
BackdropComponent: PropTypes.elementType,
/**
* Properties applied to the [`Backdrop`](/api/backdrop/) element.
*/
BackdropProps: PropTypes.object,
/**
* A single child content element.
*/
children: elementAcceptingRef.isRequired,
/**
* When set to true the Modal waits until a nested Transition is completed before closing.
*/
closeAfterTransition: PropTypes.bool,
/**
* A node, component instance, or function that returns either.
* The `container` will have the portal children appended to it.
*/
container: PropTypes.oneOfType([PropTypes.object, PropTypes.func]),
/**
* If `true`, the modal will not automatically shift focus to itself when it opens, and
* replace it to the last focused element when it closes.
* This also works correctly with any modal children that have the `disableAutoFocus` prop.
*
* Generally this should never be set to `true` as it makes the modal less
* accessible to assistive technologies, like screen readers.
*/
disableAutoFocus: PropTypes.bool,
/**
* If `true`, clicking the backdrop will not fire any callback.
*/
disableBackdropClick: PropTypes.bool,
/**
* If `true`, the modal will not prevent focus from leaving the modal while open.
*
* Generally this should never be set to `true` as it makes the modal less
* accessible to assistive technologies, like screen readers.
*/
disableEnforceFocus: PropTypes.bool,
/**
* If `true`, hitting escape will not fire any callback.
*/
disableEscapeKeyDown: PropTypes.bool,
/**
* Disable the portal behavior.
* The children stay within it's parent DOM hierarchy.
*/
disablePortal: PropTypes.bool,
/**
* If `true`, the modal will not restore focus to previously focused element once
* modal is hidden.
*/
disableRestoreFocus: PropTypes.bool,
/**
* If `true`, the backdrop is not rendered.
*/
hideBackdrop: PropTypes.bool,
/**
* @ignore
* from `withForwardRef`
*/
innerRef: PropTypes.oneOfType([PropTypes.func, PropTypes.object]),
/**
* Always keep the children in the DOM.
* This property can be useful in SEO situation or
* when you want to maximize the responsiveness of the Modal.
*/
keepMounted: PropTypes.bool,
/**
* @ignore
*
* A modal manager used to track and manage the state of open
* Modals. This enables customizing how modals interact within a container.
*/
manager: PropTypes.object,
/**
* Callback fired when the backdrop is clicked.
*/
onBackdropClick: PropTypes.func,
/**
* Callback fired when the component requests to be closed.
* The `reason` parameter can optionally be used to control the response to `onClose`.
*
* @param {object} event The event source of the callback
* @param {string} reason Can be:`"escapeKeyDown"`, `"backdropClick"`
*/
onClose: PropTypes.func,
/**
* Callback fired when the escape key is pressed,
* `disableEscapeKeyDown` is false and the modal is in focus.
*/
onEscapeKeyDown: PropTypes.func,
/**
* Callback fired once the children has been mounted into the `container`.
* It signals that the `open={true}` property took effect.
*/
onRendered: PropTypes.func,
/**
* If `true`, the modal is open.
*/
open: PropTypes.bool.isRequired,
/**
* @ignore
*/
theme: PropTypes.object
} : void 0;
Modal.defaultProps = {
BackdropComponent: SimpleBackdrop,
closeAfterTransition: false,
disableAutoFocus: false,
disableBackdropClick: false,
disableEnforceFocus: false,
disableEscapeKeyDown: false,
disablePortal: false,
disableRestoreFocus: false,
hideBackdrop: false,
keepMounted: false,
// Modals don't open on the server so this won't conflict with concurrent requests.
manager: new ModalManager()
};
export default withTheme(withForwardedRef(Modal));