使用 React、Redux 和 Immutable.js 创建模态弹出窗口

Creating modal popups with React, Redux, and Immutable.js

在我的项目中,我希望能够为诸如登录和注册帐户之类的事情发送模态弹出窗口。我遇到的问题是我希望它们能够用于任何目的,但我不确定如何在 "React way" 中做到这一点。一开始我是这么想的:

/constants/Modal.js

export const DISPATCH = 'MODAL/DISPATCH';
export const UPDATE = 'MODAL/UPDATE';
export const DISMISS = 'MODAL/DISMISS';

/actions/Modal.js

import {
    DISPATCH,
    UPDATE,
    DISMISS
} from '../constants/Modal';

export function dispatch(type, props) {
    return {
        type: DISPATCH,
        payload: {type, props}
    };
}

export function update(props) {
    return {
        type: UPDATE,
        payload: {props}
    };
}

export function dismiss() {
    return {type: DISMISS};
}

/reducers/Modal.js

import {Map} from 'immutable';

import {
    DISPATCH,
    UPDATE,
    DISMISS
} from '../constants/Modal';

const initialState = Map({
    type: '',
    props: Map(),
    isDispatched: false
});

export default function modalReducer(state = initialState, action) {
    switch (action.type) {
        case DISPATCH:
            return state.set('type', action.payload.type)
                .set('props', Map(action.payload.props))
                .set('isDispatched', true);
        case UPDATE:
            return state.set('props', Map(action.payload.props));
        case DISMISS:
            return state.set('isDispatched', false);
        default:
            return state;
    }
}

Components 然后会触发这些操作,然后 ModalContainer 组件会根据 state.modal.get('type') 呈现适当的弹出窗口并将 state.modal.get('props') 传递给 Modal 组件。这样做的问题是 props 最终会包括组件的子项和各种方法,这些在商店中没有业务。如何在不执行 React.render(<Modal {...props}>, document.getElementById('modal-container'))?

之类的操作的情况下,从不是 ModalContainer 组件子组件的组件中呈现弹出窗口

我描述了一种类似的模态对话方式
根据我的经验,我只是不通过有效负载传递不可序列化的道具。

例如,我不会传递子项,而是将行为封装在适当的模态组件本身中,例如<DeleteUserModal kind='sad' userId={42} hasErrors={false} />。然后根据需要 DeleteUserModal 连接到商店,检索它需要的数据,分派操作,并选择要呈现的子项。

或者你可以完全避免通过 Redux 路由模态,而只使用像 react-modal 这样的东西,它实际上为你做了这个:

doing something asinine like React.render(<Modal {...props}>, document.getElementById('modal-container'))?

(这不是傻瓜,人们称这种模式为“门户”。)