Мутация логина не работает последовательно в Material-UI. - PullRequest
0 голосов
/ 03 октября 2018

Я пытаюсь создать пример формы входа в систему в React, используя material-ui с помощью мутации graphql.Но форма входа не работает должным образом.

Я могу войти в систему только несколько раз, но даже в те времена, Я не вижу никаких данных, передаваемых через реквизиты loginMutation в консоли .Может кто-нибудь, пожалуйста, скажите мне, что я здесь делаю не так?

Вот компонент входа, который я пытаюсь создать

import React, { Component } from 'react';
import { AUTH_TOKEN } from '../constants';
import { graphql, compose } from 'react-apollo';
import {LOGIN_MUTATION} from "../gql/loginGQL";
import Paper from '@material-ui/core/Paper';
import Button from '@material-ui/core/Button';
import TextField from '@material-ui/core/TextField'

import PropTypes from 'prop-types';
import Avatar from '@material-ui/core/Avatar';
import CssBaseline from '@material-ui/core/CssBaseline';
import FormControl from '@material-ui/core/FormControl';
import FormControlLabel from '@material-ui/core/FormControlLabel';
import Checkbox from '@material-ui/core/Checkbox';
//import Input from '@material-ui/core/Input';
//import InputLabel from '@material-ui/core/InputLabel';
import LockIcon from '@material-ui/icons/LockOutlined';
import Typography from '@material-ui/core/Typography';
import withStyles from '@material-ui/core/styles/withStyles';

const styles = theme => ({
  layout: {
    width: 'auto',
    display: 'block', // Fix IE11 issue.
    marginLeft: theme.spacing.unit * 3,
    marginRight: theme.spacing.unit * 3,
    [theme.breakpoints.up(400 + theme.spacing.unit * 3 * 2)]: {
      width: 400,
      marginLeft: 'auto',
      marginRight: 'auto',
    },
  },
  paper: {
    marginTop: theme.spacing.unit * 8,
    display: 'flex',
    flexDirection: 'column',
    alignItems: 'center',
    padding: `${theme.spacing.unit * 2}px ${theme.spacing.unit * 3}px ${theme.spacing.unit * 3}px`,
  },
  avatar: {
    margin: theme.spacing.unit,
    backgroundColor: theme.palette.secondary.main,
  },
  form: {
    width: '100%', // Fix IE11 issue.
    marginTop: theme.spacing.unit,
  },
  submit: {
    marginTop: theme.spacing.unit * 3,
  },
});


class Login extends Component {
  state = {
    email: '',
    password: '',
    errors: null
  }

  render() {
    const { classes } = this.props;
    
    return (
      <React.Fragment>
        <CssBaseline />
        <main className={classes.layout}>
          <Paper className={classes.paper}>
            <Avatar className={classes.avatar}>
              <LockIcon />
            </Avatar>
            <Typography variant="headline">Sign in</Typography>
            <form className={classes.form}>
              <FormControl margin="normal" required fullWidth>

                <TextField
                  id='email'
                  value={this.state.email}
                  onChange={e => this.setState({ email: e.target.value })}
                  type='text'
                  label='Your email address'
                />
              </FormControl>
              <FormControl margin="normal" required fullWidth>
              <TextField
                id='password'
                value={this.state.password}
                onChange={e => this.setState({ password: e.target.value })}
                type='password'
                label='Password'
              />
              </FormControl>
              <FormControlLabel
                control={<Checkbox value="remember" color="primary" />}
                label="Remember me"
              />
              <Button
                id="submit"
                type="submit"
                fullWidth
                variant="raised"
                color="primary"
                className={classes.submit}
                onClick={() => this._confirm()}
              >
                Sign in
              </Button>
            </form>

          </Paper>
        </main>
      </React.Fragment>
    );
  }

  _confirm = async () => {
    const { email, password } = this.state
    try{
      const result = await this.props.loginMutation({
        variables: {
          email,
          password,
        },
      });
      
      console.log(result); // Here no data is being displayed !
      
      const { jwt } = result.data.signInUser;
      this._saveUserData(jwt);
      this.props.history.push(`/`)
    } catch(error) {
      const errors = error.graphQLErrors.map(error => error.message);
      this.setState({ errors });
      }

  }

  _saveUserData = (token) => {
    localStorage.setItem(AUTH_TOKEN, token)
  }
}

Login.propTypes = {
  classes: PropTypes.object.isRequired,
};

export default compose(
  graphql(LOGIN_MUTATION, { name: 'loginMutation' }),
  withStyles(styles),
)(Login)

1 Ответ

0 голосов
/ 04 октября 2018

Произошла ошибка из-за тега Button.Хотя он имел тип = "отправить", форма все еще не отправлялась, из-за чего входные данные не появлялись в "_confirm".Мне интересно, почему, я, должно быть, что-то здесь упустил.

Сейчас я создал тег div внутри кнопки и добавил там обработчик onClick вместо тега Button.

Итак, отредактированный кодпросто есть это небольшое изменение

<Button color="primary" className = {classes.submit} variant="raised" fullWidth>
  <div className="test" onClick={() => this._confirm()} >
    Sign in
  </div>
</Button>

Вот полный исходный код:

import React, { Component } from 'react';
import { AUTH_TOKEN } from '../constants';
import { graphql, compose } from 'react-apollo';
import {LOGIN_MUTATION} from "../gql/loginGQL";
import Paper from '@material-ui/core/Paper';
import Button from '@material-ui/core/Button';
import TextField from '@material-ui/core/TextField'

import PropTypes from 'prop-types';
import Avatar from '@material-ui/core/Avatar';
import CssBaseline from '@material-ui/core/CssBaseline';
import FormControl from '@material-ui/core/FormControl';
import FormControlLabel from '@material-ui/core/FormControlLabel';
import Checkbox from '@material-ui/core/Checkbox';
//import Input from '@material-ui/core/Input';
//import InputLabel from '@material-ui/core/InputLabel';
import LockIcon from '@material-ui/icons/LockOutlined';
import Typography from '@material-ui/core/Typography';
import withStyles from '@material-ui/core/styles/withStyles';

const styles = theme => ({
  layout: {
    width: 'auto',
    display: 'block', // Fix IE11 issue.
    marginLeft: theme.spacing.unit * 3,
    marginRight: theme.spacing.unit * 3,
    [theme.breakpoints.up(400 + theme.spacing.unit * 3 * 2)]: {
      width: 400,
      marginLeft: 'auto',
      marginRight: 'auto',
    },
  },
  paper: {
    marginTop: theme.spacing.unit * 8,
    display: 'flex',
    flexDirection: 'column',
    alignItems: 'center',
    padding: `${theme.spacing.unit * 2}px ${theme.spacing.unit * 3}px ${theme.spacing.unit * 3}px`,
  },
  avatar: {
    margin: theme.spacing.unit,
    backgroundColor: theme.palette.secondary.main,
  },
  form: {
    width: '100%', // Fix IE11 issue.
    marginTop: theme.spacing.unit,
  },
  submit: {
    marginTop: theme.spacing.unit * 3,
  },
});


class Login extends Component {
  state = {
    email: '',
    password: '',
    errors: null
  }

  render() {
    const { classes } = this.props;
    
    return (
      <React.Fragment>
        <CssBaseline />
        <main className={classes.layout}>
          <Paper className={classes.paper}>
            <Avatar className={classes.avatar}>
              <LockIcon />
            </Avatar>
            <Typography variant="headline">Sign in</Typography>
            <form className={classes.form}>
              <FormControl margin="normal" required fullWidth>

                <TextField
                  id='email'
                  value={this.state.email}
                  onChange={e => this.setState({ email: e.target.value })}
                  type='text'
                  label='Your email address'
                />
              </FormControl>
              <FormControl margin="normal" required fullWidth>
              <TextField
                id='password'
                value={this.state.password}
                onChange={e => this.setState({ password: e.target.value })}
                type='password'
                label='Password'
              />
              </FormControl>
              <FormControlLabel
                control={<Checkbox value="remember" color="primary" />}
                label="Remember me"
              />
            // Here is the change
              <Button color="primary" className = {classes.submit} 
                      variant="raised"fullWidth >
                <div className="test" onClick={() => this._confirm()} >
                   Sign in
                </div>
             </Button>
           // Change ends here
            </form>

          </Paper>
        </main>
      </React.Fragment>
    );
  }

  _confirm = async () => {
    const { email, password } = this.state
    try{
      const result = await this.props.loginMutation({
        variables: {
          email,
          password,
        },
      });
      
      console.log(result); // Here no data is being displayed !
      
      const { jwt } = result.data.signInUser;
      this._saveUserData(jwt);
      this.props.history.push(`/`)
    } catch(error) {
      const errors = error.graphQLErrors.map(error => error.message);
      this.setState({ errors });
      }

  }

  _saveUserData = (token) => {
    localStorage.setItem(AUTH_TOKEN, token)
  }
}

Login.propTypes = {
  classes: PropTypes.object.isRequired,
};

export default compose(
  graphql(LOGIN_MUTATION, { name: 'loginMutation' }),
  withStyles(styles),
)(Login)
...