#javascript #reactjs
Вопрос:
Я новичок в реагировании и пытаюсь визуализировать компонент, когда пользователь пытается войти в систему с неверными данными для входа. У меня есть страница Login.jsx, на которой я отображаю форму входа в систему и обрабатываю всю логику входа в асинхронную функцию. Все это прекрасно работает. Если данные для входа неверны, у меня есть ErrorModal
компонент, отображаемый в уже установленном div с идентификатором error-modal.
ReactDOM.render(<ErrorModal />, document.getElementById('error-modal'))
...
function Login() {
const userRef = React.useRef()
const passRef = React.useRef()
const loginErr = React.useRef()
const handleLogin = async (e) => {
e.preventDefault()
requestLogin(userRef.current.value, passRef.current.value, loginErr)
}
return (
<form className='login-form' onSubmit={handleLogin}>
<Field ref={userRef} label='Username:' type='text' />
<Field ref={passRef} label='Password:' type='password' />
<div className='button-wrapper'>
<button type='submit' className='btn'>Login</button>
</div>
<div id='error-modal'></div>
</form>
)
}
export default Login
Нужно ли мне, чтобы div с модальной ошибкой уже был в DOM, или я могу просто ErrorModal
каким-то образом добавить компонент после другого элемента DOM (например, внутри формы или сразу за ее пределами) и не беспокоиться о том, что div с модальной ошибкой «жестко закодирован»?
Комментарии:
1. Способ реагирования состоит в том, чтобы поместить его в DOM и использовать переменную состояния, чтобы показывать его при необходимости.
2. Нет необходимости безоговорочно отображать модальный код ошибки, который, кстати, вероятно , должен быть компонентом, а не div. Постарайтесь избавиться от привычки использовать DOM напрямую. Когда вы впервые начинаете с React
useRef
, это запах кода. ИспользуйтеuseState
и обновляйте соответствующие обработчики в своей форме/вводах.
Ответ №1:
Вы должны использовать Условный рендеринг.
Вот пример:
function Login() {
const [userCredentials, setUserCredentials] = useState({
email: "",
password: ""
});
const [error, setError] = useState(false);
const handleLogin = async (e) => {
e.preventDefault();
try {
const res = await requestLogin(userCredentials);
if (!res.ok) {
throw Error("Login failed...");
}
} catch (e) {
setError(true);
console.error(e);
}
};
const handleUserCredentialsChange = (e) => {
setUserCredentials({ ...userCredentials, [e.target.name]: e.target.value });
};
return (
<>
<form onSubmit={handleLogin}>
<input
placeholder="Email"
name="email"
value={userCredentials.email}
onChange={handleUserCredentialsChange}
/>
<input
placeholder="Password"
name="password"
value={userCredentials.password}
type="password"
onChange={handleUserCredentialsChange}
/>
<input type="submit" />
</form>
{error amp;amp; (
<div style={modalContainer}>
<div style={{ backgroundColor: "aliceblue", padding: 40 }}>
<h1>Something went wrong...</h1>
<button onClick={() => setError(false)}>Close</button>
</div>
</div>
)}
</>
);
}
function requestLogin(userCredentials) {
return fetch("api/login", {
method: "POST",
body: JSON.stringify(userCredentials),
headers: {
"Content-Type": "application/json"
}
});
}
const modalContainer = {
position: "absolute",
zIndex: 100,
top: 0,
bottom: 0,
left: 0,
right: 0,
display: "flex",
justifyContent: "center",
alignItems: "center",
backgroundColor: "rgba(0,0,0,0.5)"
};
Проверьте это в песочнице кода, если вы хотите запустить его в браузере.
Ответ №2:
вы можете сделать это, создав простое состояние, подобное этому
function Login() {
const [error,setError]=React.useState("")
const userRef = React.useRef()
const passRef = React.useRef()
const loginErr = React.useRef()
const handleLogin = async (e) => {
e.preventDefault()
const res=requestLogin(userRef.current.value, passRef.current.value, loginErr)
//assuming that above function does a post call via fetch
//and also assuming that in case of wrong credentials error was passed from backend
const data=res.json()
if(!res.ok) setError(data.error);
}
return (
<form className='login-form' onSubmit={handleLogin}>
<Field ref={userRef} label='Username:' type='text' />
<Field ref={passRef} label='Password:' type='password' />
<div className='button-wrapper'>
<button type='submit' className='btn'>Login</button>
</div>
<div id='error-modal'>{error.length>0 amp;amp; error}</div>
</form>
)
}
export default Login