Reputation: 1
I am unable to bind actions to this component, I am using redux to bind actions, state of the component I am getting undefined, please help me thanks in advance.
actions and store and reducer I added below is the component, Action, reducer
import React from 'react';
import { Formik, Field, Form, ErrorMessage } from 'formik';
import { bindActionCreators } from 'redux'
import * as Yup from 'yup';
import { connect } from "react-redux";
import * as Actions from '../_actions'
//import { userActions } from '../_actions';
import { authenticationService } from '../_services';
export class LoginPage extends React.Component {
constructor(props) {
super(props);
// redirect to home if already logged in
if (authenticationService.currentUserValue) {
this.props.history.push('/');
}
}
render() {
return (
<div>
<div className="alert alert-info">
<strong>Normal User</strong> - U: user P: user<br />
<strong>Administrator</strong> - U: admin P: admin
</div>
<h2>Login</h2>
<Formik
initialValues={{
username: '',
password: ''
}}
validationSchema={Yup.object().shape({
username: Yup.string().required('Username is required'),
password: Yup.string().required('Password is required')
})}
onSubmit={((obj) => this.props.Actions.login(obj.username, obj.password))}
render={({ errors, status, touched, isSubmitting }) => (
<Form>
<div className="form-group">
<label htmlFor="username">Username</label>
<Field name="username" type="text" className={'form-control' + (errors.username && touched.username ? ' is-invalid' : '')} />
<ErrorMessage name="username" component="div" className="invalid-feedback" />
</div>
<div className="form-group">
<label htmlFor="password">Password</label>
<Field name="password" type="password" className={'form-control' + (errors.password && touched.password ? ' is-invalid' : '')} />
<ErrorMessage name="password" component="div" className="invalid-feedback" />
</div>
<div className="form-group">
<button type="submit" className="btn btn-primary" disabled={isSubmitting}>Login</button>
{isSubmitting &&
<img src=""
alt="login" />
}
</div>
{status &&
<div className={'alert alert-danger'}>{status}</div>
}
</Form>
)}
/>
</div>
)
}
}
function mapDispatch(dispatch) {
return bindActionCreators({ Actions }, dispatch)
}
function mapStateProps(props) {
return {
allprop: props
}
}
connect(mapStateProps, mapDispatch)(LoginPage);
action I have index.js file so I am adding action main code
import { authenticationConstants } from '../_constants';
import { authenticationService } from '../_services';
import { history } from '../_helpers';
import{alertActions} from './alert.actions'
export const userActions = {
login,
logout,
getAll,
};
function login(username, password) {
return dispatch => {
dispatch(request({ username }));
authenticationService.login(username, password)
.then(
user => {
dispatch(success(user));
history.push('/');
},
error => {
dispatch(failure(error));
dispatch(alertActions.error(error));
}
);
};
function request(user) { return { type: authenticationConstants.LOGIN_REQUEST, user } }
function success(user) { return { type: authenticationConstants.LOGIN_SUCCESS, user } }
function failure(error) { return { type: authenticationConstants.LOGIN_FAILURE, error } }
}
function logout() {
authenticationService.logout();
return { type: authenticationConstants.LOGOUT };
}
function getAll() {
return dispatch => {
dispatch(request());
authenticationService.getAll()
.then(
users => dispatch(success(users)),
error => dispatch(failure(error))
);
};
function request() { return { type: authenticationConstants.GETALL_REQUEST } }
function success(users) { return { type: authenticationConstants.GETALL_SUCCESS, users } }
function failure(error) { return { type: authenticationConstants.GETALL_FAILURE, error } }
}
I am adding reducer, please go through it
import { authenticationConstants } from '../_constants';
let user = JSON.parse(localStorage.getItem('user'));
const initialState = user ? { loggedIn: true, user } : {};
export function authentication(state = initialState, action) {
switch (action.type) {
case authenticationConstants.LOGIN_REQUEST:
return {
loggingIn: true,
user: action.user
};
case authenticationConstants.LOGIN_SUCCESS:
return {
loggedIn: true,
user: action.user
};
case authenticationConstants.LOGIN_FAILURE:
return {};
case authenticationConstants.LOGOUT:
return {};
case authenticationConstants.GETALL_REQUEST:
return {
loading: true
};
case authenticationConstants.GETALL_SUCCESS:
return {
items: action.users
};
case authenticationConstants.GETALL_FAILURE:
return {
error: action.error
};
default:
return state
}
}
Upvotes: 0
Views: 175
Reputation: 281894
You must pass Actions as the first argument to bindActionCreators instead of as an object
function mapDispatch(dispatch) {
return bindActionCreators(Actions, dispatch)
}
and then use the action like
this.props.login(obj.username, obj.password);
However you can simply use an object as the second argument instead using bindActionCreators like
connect(mapStateProps, Actions)(LoginPage);
Also you must export and use the connected component instead of exporting the normal component
So instead of this
export class LoginPage extends React.Component {
You should have
class LoginPage extends React.Component {
....
const LoginPageWithConnect = connect(mapStateProps, Actions)(LoginPage);
export { LoginPageWithConnect as LoginPage };
Upvotes: 0