#javascript #reactjs #ecmascript-6
#javascript #reactjs #ecmascript-6
Вопрос:
Как новичок в Reacteur, я следовал этому руководству, чтобы создать простую форму входа в систему с помощью reactjs. Все работает нормально и работает так, как задумано. Весь учебник основан на скрипте signle.
Вот окончательный код (из tutrial):
import React from 'react'
import {
BrowserRouter as Router,
Route,
Link,
Redirect,
withRouter
} from 'react-router-dom'
const authenticator = {
isAuthenticated: false,
authenticate(cb) {
this.isAuthenticated = true
setTimeout(cb, 100)
},
signout(cb) {
this.isAuthenticated = false
setTimeout(cb, 100)
}
}
const Public = () => <h3>Public</h3>
const Protected = () => <h3>Protected</h3>
class Login extends React.Component {
state = {
redirectToReferrer: false
}
login = () => {
authenticator.authenticate(() => {
this.setState(() => ({
redirectToReferrer: true
}))
})
}
render() {
const { from } = this.props.location.state || { from: { pathname: '/' } }
const { redirectToReferrer } = this.state
if (redirectToReferrer === true) {
return <Redirect to={from} />
}
return (
<div>
<p>You must log in to view the page</p>
<button onClick={this.login}>Log in</button>
</div>
)
}
}
const PrivateRoute = ({ component: Component, ...rest }) => (
<Route {...rest} render={(props) => (
fakeAuth.isAuthenticated === true
? <Component {...props} />
: <Redirect to={{
pathname: '/login',
state: { from: props.location }
}} />
)} />
)
const AuthButton = withRouter(({ history }) => (
fakeAuth.isAuthenticated ? (
<p>
Welcome! <button onClick={() => {
fakeAuth.signout(() => history.push('/'))
}}>Sign out</button>
</p>
) : (
<p>You are not logged in.</p>
)
))
export default function AuthExample () {
return (
<Router>
<div>
<AuthButton/>
<ul>
<li><Link to="/public">Public Page</Link></li>
<li><Link to="/protected">Protected Page</Link></li>
</ul>
<Route path="/public" component={Public}/>
<Route path="/login" component={Login}/>
<PrivateRoute path='/protected' component={Protected} />
</div>
</Router>
)
}
Однако проблема заключается в том, что я пытаюсь поместить разные скрипты в разные файлы и импортировать их. Из приведенного выше кода я перенес следующий код в скрипт с именем Authenticator.js
, как показано ниже:
import React from 'react'
import { BrowserRouter as Route, Redirect } from "react-router-dom";
export const authenticator = {
isAuthenticated: false,
authenticate(cb) {
this.isAuthenticated = true
setTimeout(cb, 100) // fake async
},
signout(cb) {
this.isAuthenticated = false
setTimeout(cb, 100) // fake async
}
}
export const PrivateRoute = ({ component: Component, ...rest }) => (
<Route {...rest} render={(props) => (
authenticator.isAuthenticated === true
? <Component {...props} />
: <Redirect to={{
pathname: '/login',
state: { from: props.location }
}} />
)} />
)
и импортировал его в исходный файл как :
import { PrivateRoute, authenticator } from './login/Authenticator'
Я также переместил класс Login как компонент в Login.js
:
import React, { Component } from 'react'
import { authenticator } from './Authenticator'
import { Redirect } from "react-router-dom";
class Login extends Component {
state = {
redirectToReferrer: false
}
login = () => {
authenticator.authenticate(() => {
this.setState(() => ({
redirectToReferrer: true
}))
})
}
render() {
const { from } = this.props.location.state || { from: { pathname: '/' } }
const { redirectToReferrer } = this.state
if (redirectToReferrer === true) {
return <Redirect to={from} />
}
return (
<div>
<p>You must log in to view the page</p>
<button onClick={this.login}>Log in</button>
</div>
)
}
}
export default Login
И импортировал его в скрипт:
import Login from './login/Login'
И, конечно, я удалил эти методы из скрипта из учебника. В идеале, теперь, когда я перехожу по /protected
ссылке, не входя в систему, я должен перенаправляться на страницу входа в систему. Однако я ничего не вижу. Однако я могу получить доступ, /login
страница и логин сохраняются, как и ожидалось. Я считаю, что проблема связана с классом Authenticator, но я ничего не изменил, кроме перемещения кода в другой файл.
Вот мой окончательный основной сценарий с изменениями:
// import css
import 'bootstrap/dist/css/bootstrap.css'
import '@fortawesome/fontawesome-free/css/all.min.css'
import './css/ViewerApp.css'
// import js modules
import React, { Component } from 'react'
import { BrowserRouter as Router, Route, Link, withRouter } from "react-router-dom";
import 'jquery/dist/jquery.min.js'
import 'popper.js/dist/umd/popper.min.js'
import 'bootstrap/dist/js/bootstrap.min.js'
import Login from './login/Login'
import { PrivateRoute, authenticator } from './login/Authenticator'
// import TableView from './table/TableView'
const Public = () => <h3>Public</h3>
const Protected = () => <h3>Protected</h3>
class ViewerApp extends Component {
render() {
return (
<Router>
<ul style={{ marginTop:'122px' }}>
<li><Link to="/public">Public Page</Link></li>
<li><Link to="/protected">Protected Page</Link></li>
</ul>
<AuthButton/>
<Route path="/public" component={Public}/>
<Route path="/login" component={Login}/>
<PrivateRoute path='/protected' component={Protected} />
</Router>
)
}
}
const AuthButton = withRouter(({ history }) => (
authenticator.isAuthenticated ? (
<p>
Welcome! <button onClick={() => {
authenticator.signout(() => history.push('/'))
}}>Sign out</button>
</p>
) : (
<p>You are not logged in.</p>
)
))
export default ViewerApp;
Кто-нибудь может помочь? Заранее спасибо.
Комментарии:
1. куда вы поместили свой импортированный
<Login />
компонент?2. @fard непосредственно перед импортом
Authenticator.js
3. Вам нужно выполнить его где-нибудь в вашем основном файле
4. Я сделал, я просто добавил код с компонентом кнопки авторизации в основной скрипт
5. Какие-либо ошибки, зарегистрированные в консоли?
Ответ №1:
Похоже, у вас неправильный импорт в вашем файле аутентификации. Вы импортируете BrowserRouter
вместо Route
import { BrowserRouter as Route, Redirect } from "react-router-dom";
Это должно быть
import { Route, Redirect } from "react-router-dom";
Комментарии:
1. Спасибо, это сработало! Потрясающе, но можете ли вы объяснить, как это работает, потому что в основном скрипте я импортировал браузер-маршрутизатор, и почему он не работает для этого?