Реагирует на редукцию роутера при смене реквизита
Я изо всех сил пытаюсь использовать реактивный маршрутизатор, и все в порядке, за исключением рендеринга на изменение реквизита.
index.js
import React from 'react';
import ReactDOM from 'react-dom';
import {Provider} from 'react-redux';
import {createStore, combineReducers, applyMiddleware} from 'redux';
import {Router, Route, browserHistory} from 'react-router';
import { syncHistoryWithStore, routerReducer, routerMiddleware } from 'react-router-redux';
import App from './app';
import Sign from '../sign/app';
import SignInfo from '../sign/redux/reducer';
let store = createStore(
combineReducers({
SignInfo,
routing: routerReducer
}),
);
const history = syncHistoryWithStore(browserHistory, store);
class Index extends React.Component {
constructor(props) {
super(props);
}
render() {
console.log('root')
return (
<Provider store={store}>
<Router history={history}>
<Route path="/" component={App}/>
<Route path="/sign" component={Sign}/>
</Router>
</Provider>
);
}
}
ReactDOM.render(<Index/>, document.getElementById('root'));
sign.js
import React from 'react';
import {connect} from 'react-redux';
import Belle from 'belle';
import {Grid, Row, Col} from 'react-flexbox-grid';
import { withRouter } from 'react-router';
import SignPaze1 from './signPaze1'
class App extends React.Component {
constructor(props) {
super(props);
}
render() {
const mainCardStyle = {
border: '1px solid #aaa',
}
return (
<div className="sign-page">
<SignPaze1/>
</div>
)
}
}
function mapStateToProps(state) {
return {
}
}
export default withRouter(connect(mapStateToProps)(App));
signPaze1.js
import React from 'react';
import { bindActionCreators } from 'redux';
import {connect} from 'react-redux';
import Belle from 'belle';
import { withRouter } from 'react-router';
import {Grid, Row, Col} from 'react-flexbox-grid';
import {handleSignInput} from './redux/action';
class SignPaze1 extends React.Component {
constructor(props) {
super(props);
this.handleInputValue = this.handleInputValue.bind(this);
}
handleInputValue(e) {
let {dispatch} = this.props;
console.log('on render')
const action = bindActionCreators(handleSignInput, dispatch)
action(e.target.id, e.target.value);
}
render() {
let signData = this.props.signData;
let mainCardStyle = {
border: '1px solid #aaa',
}
return (
<Grid>
<Row>
<Col xs={12}>
<Belle.Card style={mainCardStyle}>
<Row center="md">
<Col xs={12} sm={6} md={4} lg={3}
className="input-group">
<input type="text"
id="company_number"
value={signData.company_number}
onChange={this.handleInputValue}/>
</Col>
<Col xs={12} sm={6} md={4} lg={3}
mdOffset={1} lgOffset={2}
className="input-group">
<Belle.TextInput
onUpdate={this.handleInputValue.bind(this, 'data')}/>
</Col>
</Row>
<Row center="md">
<Col xs={12} sm={6} md={4} lg={3}
className="input-group">
<Belle.TextInput
onUpdate={this.handleInputValue.bind(this, 'da2')}/>
</Col>
<Col xs={12} sm={6} md={4} lg={3}
mdOffset={1} lgOffset={2}
className="input-group">
<Belle.TextInput
onUpdate={this.handleInputValue.bind(this, 'dat3')}/>
</Col>
</Row>
</Belle.Card>
</Col>
</Row>
</Grid>
)
}
}
function mapStateToProps(state) {
console.log(state.SignInfo);
return {
signData: state.SignInfo.signData,
}
}
export default withRouter(connect(mapStateToProps)(SignPaze1));
reducer.js
import {HANDLE_SIGN_INPUT} from './action';
const initialMainState = {
signData: {
company_number: 'asdas|',
company_name: '',
username: '',
email: '',
email_confirm: '',
password: '',
password_confirm: '',
}
}
function SignInfo(state = initialMainState, action) {
let newVal = {};
switch (action.type) {
case HANDLE_SIGN_INPUT:
newVal = Object.assign({}, state);
newVal.signData[action.target] = action.value;
return newVal;
default:
return state;
}
}
export default SignInfo;
кто-то говорит, что добавить withRouter
функция в строке экспорта исправит эту ошибку, но это не так. ( здесь). Тем не менее, обычно отображается снова, когда состояние является членом SignPaze1
изменено
Как я могу исправить эту проблему, вызванную реагировать-маршрутизатор-редуктор?
редактировать
action.js
export const HANDLE_SIGN_INPUT = 'HANDLE_SIGN_INPUT';
export function handleSignInput(target, value){
return {
type: HANDLE_SIGN_INPUT,
target: target,
value: value,
}
}
1 ответ
Я считаю, что проблема заключается в том, что вы должны использовать combineReducers
вместо одного редуктора, содержащего вложенные объекты.
Вам, вероятно, нужно создать редуктор специально для signData
и использовать Redux combineReducers
метод.
Новый файл: signDataReducer.js
import {HANDLE_SIGN_INPUT} from './action';
const initialMainState = {
company_number: 'asdas|',
company_name: '',
username: '',
email: '',
email_confirm: '',
password: '',
password_confirm: '',
};
export default function SignData (state = initialMainState, action) {
let newVal = {};
switch (action.type) {
case HANDLE_SIGN_INPUT:
newVal = Object.assign({}, state);
newVal[action.target] = action.value;
return newVal;
default:
return state;
}
}
reducer.js
import { combineReducers } from 'redux';
import signDataReducer from './signDataReducer';
export default combineReducers({
signDataReducer
});
Причина, по которой это не работало раньше, заключается в том, как Object.assign
работает, он копирует ссылку signData
новому объекту, созданному методом assign.
Тогда когда SignPaze1
реквизиты сравниваются путем поверхностного сравнения, signData
является точно таким же объектом, как и раньше, поэтому считается равным. И никакой визуализации не происходит.
т.е.
const initialMainState = {
signData: {
company_number: 'asdas|',
company_name: '',
username: '',
email: '',
email_confirm: '',
password: '',
password_confirm: '',
}
}
const newState = Object.assign({}, initialMainState);
newState.signData.company_name = 'Hello!';
// => "Hello!"
initialMainState.signData.company_name
// => "Hello!"
newState.signData === initialMainState.signData
// => true