- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有一个登录页面、一个仪表板页面和其他一些页面。我正在使用 firebase 作为后端。我在登录组件上执行了条件渲染以检查用户是否登录。 (如果已登录,则会重定向到仪表板)。如果用户未登录,我已将 ProtectedRoutes 用于仪表板和其他重定向到登录页面。
发生的事情是,当我重新加载仪表板页面或任何其他页面时,我会短暂地看到登录组件,然后它会执行 firebase“auth.onAuthStateChanged”方法并设置登录状态为真并将我重定向回仪表板。我明白为什么我会短暂地看到登录页面,但我不知道如何解决这个问题。
import React, { useState } from 'react'
import { connect } from "react-redux"
import { signin, autoSignin } from "../Redux/Actions/AuthActions"
import { Redirect } from 'react-router-dom'
const Login = (props) => {
const [form, setForm] = useState({})
const [disable, setDisable] = useState(false)
const handleChange = (e) => {
setForm({
...form,
[e.target.name]: e.target.value
})
}
const handleSubmit = (e) => {
setDisable(true)
e.preventDefault()
props.signin(form.email, form.password)
setDisable(false)
}
return (
props.auth.loggedin ?
<Redirect to="/dashboard" /> :
<div className="container-fluid login_screen">
<div className="row justify-content-center">
<div className="col-md-4 col-xs-12">
<img className="img-fluid login_screen_logo mt-4" src="/images/logo.png" alt="logo" />
<h1 className="mt-4 fs-3 heading">LOGIN</h1>
<p className="fs-6 mb-3">to access dashboard</p>
<form onSubmit={handleSubmit}>
<div className="form-floating mb-4 mt-4">
<input name="email" onChange={handleChange} required type="email" className="form-control shadow-sm" id="floatingInput" placeholder="room@adil.tower" />
<label htmlFor="floatingInput">Email id</label>
</div>
<div className="form-floating mt-4">
<input name="password" onChange={handleChange} required type="password" className="form-control shadow-sm" id="floatingPassword" placeholder="Password" />
<label htmlFor="floatingPassword">Password</label>
</div>
<div className="d-grid gap-2">
<button type="submit" className={`btn btn-custom mt-4 btn-lg shadow ${disable ? "disabled" : ""}`}>LOGIN</button>
</div>
</form>
</div>
</div>
</div>
)
}
const mapStateToProps = (state) => {
return {
auth: state.auth
}
}
const mapDispatchToProps = (dispatch) => {
return {
signin: (email, pass) => dispatch(signin(email, pass)),
autoSignin: (uid) => dispatch(autoSignin(uid))
}
}
export default connect(mapStateToProps, mapDispatchToProps)(Login)
import React from 'react'
const Dashboard = (props) => {
return (
<h1>
dashboard
</h1>
)
}
export default Dashboard
import React from 'react'
import { Redirect, Route } from "react-router-dom"
const ProtectedRoute = ({ loggedin, component: Component, ...rest }) => {
return (
<Route
{...rest}
render={
(props) => {
if (loggedin) {
return <Component {...props} />
} else {
return <Redirect to="/" />
}
}
}
/>
)
}
export default ProtectedRoute
import React, { useEffect } from 'react'
import { BrowserRouter as Router, Switch, Route } from "react-router-dom"
import ProtectedRoute from './ProtectedRoute'
import { auth } from "./FirebaseConfig"
// SCREENS
import Dashboard from './Dashboard/Dashboard'
import Login from "./Login/Login"
import Profile from './Profile/Profile'
import Emergency from './Emergency/Emergency'
import Notice from './Notice/Notice'
import Complains from "./Complains/Complains"
import Meetings from "./Meetings/Meetings"
// REDUX STORE & ACTIONS
import { connect } from "react-redux"
import { autoSignin } from "./Redux/Actions/AuthActions"
const App = (props) => {
useEffect(
() => {
auth.onAuthStateChanged(user => {
if (user) {
props.autoSignin(user.uid)
}
})
}, []
)
return (
<Router>
<Switch>
<ProtectedRoute loggedin={props.auth.loggedin} exact path="/dashboard" component={Dashboard}></ProtectedRoute>
<Route exact path="/"><Login /></Route>
<ProtectedRoute loggedin={props.auth.loggedin} exact path="/profile" component={Profile}></ProtectedRoute>
<ProtectedRoute loggedin={props.auth.loggedin} exact path="/emergency" component={Emergency}></ProtectedRoute>
<ProtectedRoute loggedin={props.auth.loggedin} exact path="/notices" component={Notice}></ProtectedRoute>
<ProtectedRoute loggedin={props.auth.loggedin} exact path="/complains" component={Complains}><Complains /> </ProtectedRoute>
<ProtectedRoute loggedin={props.auth.loggedin} exact path="/meetings" component={Meetings} ></ProtectedRoute>
</Switch>
</Router>
)
}
const mapStateToProps = (state) => {
return {
auth: state.auth
}
}
const mapDispatchToProps = (dispatch) => {
return {
autoSignin: (uid) => dispatch(autoSignin(uid))
}
}
export default connect(mapStateToProps, mapDispatchToProps)(App)
最佳答案
我不会编辑你的代码,只是解释一下基本原理:在呈现 protected 路线期间,您有 2 个选择:
要么等待身份验证解决(props.auth.loggedin 是真或假,但不是未定义的——假设这些值是您从 BE 收到的)。同时,显示加载屏幕。解决此问题后,您将显示您的页面,或者您重定向到登录。
第二个选项是继续并开始渲染私有(private)( protected )路由。用户将看不到任何私有(private)数据,因为这些数据来自后端,并且您的这些后端调用也受到后端身份验证的保护。这种方法将允许您直接开始渲染 protected 路由的内容,并在这样做时等待 props.auth.loggedin 从后端到达 - 在您的 App 组件中,设置 React.useEffect 监听 props.auth.loggedin .如果用户未通过身份验证,则将他重定向到登录页面。如果他是,则什么都不会改变(或者你也可以让你的状态管理知道这件事)。
我在我的代码中包含了我是如何做到这一点的,以激发您的灵感:
export const AppRoute: React.FC<RouteProps> = observer((props: RouteProps) => {
const { user } = useMst();
const { component, ...rest } = props;
if (!user.isAuthenticated) {
return <Redirect to={"/login"} />;
}
const Component = component as any;
return (
<Route
{...rest}
render={(props) => (
<Layout>
<Component {...props} />
</Layout>
)}
/>
);
});
然而,在这段代码中,我将有关用户是否已通过身份验证的信息存储在 Mobex 中,并监听 Mobex 状态变化 - 因此当 Mobex 意识到用户未通过身份验证时,它将触发该重定向语句的执行。
关于reactjs - React router 会短暂显示登录屏幕,然后在用户登录后重定向到仪表板,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/65187368/
尝试使用集成到 QTCreator 的表单编辑器,但即使我将插件放入 QtCreator.app/Contents/MacOS/designer 也不会显示。不过,相同的 dylib 文件确实适用于独
在此代码示例中。 “this.method2();”之后会读到什么?在返回returnedValue之前会跳转到method2()吗? public int method1(int returnedV
我的项目有通过gradle配置的依赖项。我想添加以下依赖项: compile group: 'org.restlet.jse', name: 'org.restlet.ext.apispark', v
我将把我们基于 Windows 的客户管理软件移植到基于 Web 的软件。我发现 polymer 可能是一种选择。 但是,对于我们的使用,我们找不到 polymer 组件具有表格 View 、下拉菜单
我的项目文件夹 Project 中有一个文件夹,比如 ED 文件夹,当我在 Eclipse 中指定在哪里查找我写入的文件时 File file = new File("ED/text.txt"); e
这是奇怪的事情,这个有效: $('#box').css({"backgroundPosition": "0px 250px"}); 但这不起作用,它只是不改变位置: $('#box').animate
这个问题在这里已经有了答案: Why does OR 0 round numbers in Javascript? (3 个答案) 关闭 5 年前。 Mozilla JavaScript Guide
这个问题在这里已经有了答案: Is the function strcmpi in the C standard libary of ISO? (3 个答案) 关闭 8 年前。 我有一个问题,为什么
我目前使用的是共享主机方案,我不确定它使用的是哪个版本的 MySQL,但它似乎不支持 DATETIMEOFFSET 类型。 是否存在支持 DATETIMEOFFSET 的 MySQL 版本?或者有计划
研究 Seam 3,我发现 Seam Solder 允许将 @Named 注释应用于包 - 在这种情况下,该包中的所有 bean 都将自动命名,就好像它们符合条件一样@Named 他们自己。我没有看到
我知道 .append 偶尔会增加数组的容量并形成数组的新副本,但 .removeLast 会逆转这种情况并减少容量通过复制到一个新的更小的数组来改变数组? 最佳答案 否(或者至少如果是,则它是一个错
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
noexcept 函数说明符是否旨在 boost 性能,因为生成的对象中可能没有记录异常的代码,因此应尽可能将其添加到函数声明和定义中?我首先想到了可调用对象的包装器,其中 noexcept 可能会产
我正在使用 Angularjs 1.3.7,刚刚发现 Promise.all 在成功响应后不会更新 angularjs View ,而 $q.all 会。由于 Promises 包含在 native
我最近发现了这段JavaScript代码: Math.random() * 0x1000000 10.12345 10.12345 >> 0 10 > 10.12345 >>> 0 10 我使用
我正在编写一个玩具(物理)矢量库,并且遇到了 GHC 坚持认为函数应该具有 Integer 的问题。是他们的类型。我希望向量乘以向量以及标量(仅使用 * ),虽然这可以通过仅使用 Vector 来实现
PHP 的 mail() 函数发送邮件正常,但 Swiftmailer 的 Swift_MailTransport 不起作用! 这有效: mail('user@example.com', 'test
我尝试通过 php 脚本转储我的数据,但没有命令行。所以我用 this script 创建了我的 .sql 文件然后我尝试使用我的脚本: $link = mysql_connect($host, $u
使用 python 2.6.4 中的 sqlite3 标准库,以下查询在 sqlite3 命令行上运行良好: select segmentid, node_t, start, number,title
我最近发现了这段JavaScript代码: Math.random() * 0x1000000 10.12345 10.12345 >> 0 10 > 10.12345 >>> 0 10 我使用
我是一名优秀的程序员,十分优秀!