Search code examples
reactjsreact-routerreact-reduxreact-router-reduxreact-modal

How to trigger react-modal from the layout container?


I'm working to use react-modal in my React+Redux+ReactRouter4 App.

I have a MainLayout container and a Home container.

The modal will only be used when the home container is rendered so I have ReactModal's logic inside the Home Container. And I can easily open the modal from the Home Container like so:

<button onClick={this.openModal}>Open Modal</button>

The problem is the MainLayout container has a navigation that also needs the ability to open the modal, but obviously, this.openModal does not exist there... How can I allow the MainLayout Container to open the modal in the Home container?

class Home extends React.Component {

  constructor(props) {
    super(props);
    this.state = {
      modalIsOpen: false
    };
    this.openModal = this.openModal.bind(this);
    this.closeModal = this.closeModal.bind(this);

  }

  openModal() {
    this.setState({modalIsOpen: true});
  }

  closeModal() {
    this.setState({modalIsOpen: false});
  }

  render() {
    return (
      <div>
        ....
        <button onClick={this.openModal}>Open Modal</button>

        <Modal
          isOpen={this.state.modalIsOpen}
          onAfterOpen={this.afterOpenModal}
          onRequestClose={this.closeModal}
          style={modalCustomStyles}
          contentLabel="Example Modal"
        >
          <h2 ref={subtitle => this.subtitle = subtitle}>Hi</h2>
          <button onClick={this.closeModal}>close</button>
          <div>I am a modal</div>
        </Modal>

      </div>
    )
  };

};

App.jsx

const WithMainLayout = ({component: Component, ...more}) => {
  return <Route {...more} render={props => {
    return (
      <MainLayout {...props}>
        <Component {...props} />
      </MainLayout>
    );
  }}/>;
};    
....
<WithMainLayout exact path="/" component={Home} />

Solution

  • What I would do is just move the modalOpenState into redux rather than keeping it in local state. Your initial state would be like this.

    export default {
      modalIsOpen: false
    };
    

    Then write an action to toggle the modal state in the store.

    export function toggleQuestionModal(isOpen) {
      return { type: types.TOGGLE_QUESTION_MODAL, payload: isOpen };
    }
    

    Your presentational component for modal should be something like this.

    import React, { Component, PropTypes } from 'react';
    import Modal from 'react-modal';
    
    const QuestionModal = ({ modalIsOpen, openModal, closeModal, afterOpenModal }) => {
      const customStyles = {
        overlay: {
          position: 'fixed',
          top: 0,
          left: 0,
          right: 0,
          bottom: 0,
          backgroundColor: 'rgba(0, 0, 0, 0.75)'
        },
    
        content: {
          top: '50%',
          left: '50%',
          right: 'auto',
          bottom: 'auto',
          marginRight: '-50%',
          height: '50%',
          width: '80%',
          transform: 'translate(-50%, -50%)'
        }
      };
    
      return (
        <div>
          <button onClick={openModal}>Open Modal</button>
          <Modal
            isOpen={modalIsOpen}
            onAfterOpen={afterOpenModal}
            onRequestClose={closeModal}
            style={customStyles}
            contentLabel="Create A Question"
            role="dialog"
          >
    
            <h2>Hello</h2>
            <button onClick={closeModal}>close</button>
            <div>I am a modal</div>
            <form>
              <input />
              <button>tab navigation</button>
              <button>stays</button>
              <button>inside</button>
              <button>the modal</button>
            </form>
          </Modal>
        </div>
      );
    };
    
    QuestionModal.propTypes = {
      modalIsOpen: PropTypes.bool.isRequired,
      openModal: PropTypes.func.isRequired,
      closeModal: PropTypes.func.isRequired,
      afterOpenModal: PropTypes.func.isRequired
    };
    
    export default QuestionModal;
    

    Finally here's your container component for the modal.

    import React, { Component, PropTypes } from 'react';
    import { connect } from 'react-redux';
    import { bindActionCreators } from 'redux';
    import { toggleQuestionModal, toggleConfirmation } from '../actions/questionActions';
    import QuestionModal from '../components/questionModal';
    
    class QuestionPage extends Component {
        constructor(props, context) {
            super(props, context);
            this.openModal = this.openModal.bind(this);
            this.closeModal = this.closeModal.bind(this);
            this.afterOpenModal = this.afterOpenModal.bind(this);
        }
    
    
        openModal() {
            this.props.toggleQuestionModal(true);
        }
    
        afterOpenModal() {
            // references are now sync'd and can be accessed. 
            // this.subtitle.style.color = '#f00';
        }
    
        closeModal() {
            this.props.toggleConfirmation(true);
        }
    
        render() {
            const { modalIsOpen } = this.props;
            return (
                <div>
                    <QuestionModal modalIsOpen={modalIsOpen} openModal={this.openModal} closeModal={this.closeModal} 
                    afterOpenModal={this.afterOpenModal} />
                </div>
            );
        }
    }
    
    QuestionPage.propTypes = {
        modalIsOpen: PropTypes.bool.isRequired,
        toggleQuestionModal: PropTypes.func.isRequired,
    };
    
    function mapStateToProps(state, ownProps) {
        return {
            modalIsOpen: state.question.modalIsOpen
        };
    }
    
    function mapDispatchToProps(dispatch) {
        return {
            toggleQuestionModal: bindActionCreators(toggleQuestionModal, dispatch),
        };
    }
    
    export default connect(mapStateToProps, mapDispatchToProps)(QuestionPage);
    

    When you want to open the modal from any component merely invoke the toggleQuestionModal action with a true value. This will change the state and render the modal. Redux recommends to keep everything in the state. I do practice that. Don't keep things local. Keeping everything in state makes it easier for you to do a time travel debug using tools. You may find sample implementation here. Hope this helps. Happy Coding !