Redux-form:从状态设置表单值

Redux-form: Set form values from state

我有一个叫做 addContactForm 的 redux-form。此表单通过插件侦听我的应用程序中的一些操作。请参阅下面的代码以获取说明:

/reducers/index.js

import { combineReducers } from 'redux';
import { reducer as reduxFormReducer } from 'redux-form'
import AddContactFormPlugin from './add-contact-form-plugin';

const rootReducer = combineReducers({
    form:        reduxFormReducer.plugin({
        addContactForm: AddContactFormPlugin
    })
});

export default rootReducer;

/reducers/add-contact-form-plugin.js

import { SET_CURRENT_CONTACT, CLEAR_CURRENT_CONTACT } from '../constants';
export default function (state, {type, payload}) {
    switch (type) {
        case SET_CURRENT_CONTACT:
            // !!! MY PROBLEM IS ON THE NEXT LINE
            return {...state, values: { ...payload }};
        case CLEAR_CURRENT_CONTACT:
            return {...state, values: getEmptyValues()};
        default:
            return state;
    }
}

function getEmptyValues() {
    return {
        firstName: '',
        lastName: '',
        phone: ''
    }
}

为了阐明这里发生了什么:当类型为 SET_CURRENT_CONTACT 的操作流入此 reducer 时,我将表单值设置为用户选择的联系人的值。这部分工作正常。

但是,表单现在被标记为 !pristine && dirty。因此,我的提交按钮没有被禁用,实际上它应该被禁用,直到用户决定进行任何更改。

所以我的问题是:如何更新表单状态以便将其标记为原始状态?

(或者我猜是无效的,但原始感觉是更好的选择)

我试过简单地在状态对象中设置值,但这没有做任何事情: return {...state, values: { ...payload }, pristine: true };

如果我在这里使用了错误的方法,我也很感激被指出正确的方向。

好的,所以我现在找到了 'answer'。我基本上只是在 redux-form 网站 Initialize values from state 上跟着这个例子。但是,它对我来说不是开箱即用的。我不得不做出两个显着的改变:

  1. 在调用 reduxForm 时,我必须传递一个参数 enableReinitialize: true。现在在选择联系人后更新值可以正常工作并且表单被重置为原始状态。但是我的验证函数坏了...
  2. 原来是因为在某个时间点要验证的值未定义。为了避免这种情况,我添加了一个从验证函数提前退出的子句:if (!values) {return {};}

所以现在一切都按预期进行,尽管事件发生了变化。 2感觉有点不对...

在我工作代码最相关的部分下方(在 redux 形式的连接组件中)

const getInitialValues = (currContact) => {
    if (currContact) {
        const { firstName, lastName, phone } = currContact;
        return { firstName, lastName, phone };
    }
    return { firstName: '', lastName: '', phone: ''};
}

const reduxConnectedForm = reduxForm({
    form: 'addContactForm',
    validate,
    enableReinitialize: true
})(AddContactForm);


function mapStateToProps (state) {
    return {
        initialValues: getInitialValues(state.currContact)
    };
}

const connectedForm = connect(
    mapStateToProps, 
)(reduxConnectedForm);
export default connectedForm;