У меня есть простое приложение React Hooks - список Todos - с React Router v4
В списке задач при нажатии на задачу мне нужно:
В предыдущей реализации на основе класса React я мог использовать this.context.history.push для перенаправления на другой маршрут.
Как бы я справился с этим, используя React Hooks в сочетании с React Router v4 (в коде ниже см. мой комментарий в функции editRow())?
Код ниже:
=====index.js=====
import React from 'react';
import ReactDOM from 'react-dom';
import { BrowserRouter} from "react-router-dom"
import App from './App';
ReactDOM.render(
<BrowserRouter>
<App />
</BrowserRouter>, document.getElementById('root'));
=====main.js=====
import React from 'react'
import { Switch, Route } from 'react-router-dom'
import TodosList from './todoslist'
import TodosEdit from './todosedit'
const Main = () => (
<main>
<Switch>
<Route exact path = "/todos" component = {TodosList}/>
<Route exact path = "/todos/:id" component = {TodosEdit} />
</Switch>
</main>
)
export default Main
=====app.js=====
import React, {useContext, useReducer} from 'react';
import Main from './main'
import TodosContext from './context'
import todosReducer from './reducer'
const App = () => {
const initialState = useContext(TodosContext);
const [state, dispatch] = useReducer(todosReducer, initialState);
return (
<div>
<TodosContext.Provider value = {{state, dispatch}}>
<Main/>
</TodosContext.Provider>
</div>
)
}
export default App;
=====TodosContext.js=====
import React from 'react'
const TodosContext = React.createContext({
todos: [
{id:1, text:'Get Grocery', complete:false},
{id:2, text:'Excercise', complete:false},
{id:3, text:'Drink Water', complete:true},
],
currentTodo: {}
})
export default TodosContext
=====reducer.js=====
import React from 'react'
export default function reducer(state, action){
switch(action.type){
case "GET_TODOS":
return {
...state,
todos: action.payload
}
case "SET_CURRENT_TODO":
return {
...state,
currentTodo: action.payload
}
default:
return state
}
}
=====Todos.js=====
import React, {useState, useContext, useEffect} from 'react';
import TodosContext from './context'
function Todos(){
const [todo, setTodo] = useState("")
const {state, dispatch} = useContext(TodosContext)
useEffect(()=>{
if (state.currentTodo.text){
setTodo(state.currentTodo.text)
} else {
setTodo("")
}
dispatch({
type: "GET_TODOS",
payload: state.todos
})
}, [state.currentTodo.id])
const editRow = event =>{
let destUrlEdit = `/todos/${event.id}`
let obj = {}
obj.id = event.id
obj.text = event.text
dispatch({type:"SET_CURRENT_TODO", payload: obj})
//after dispatch I would like to redirect to another route to do the actual edit
//destUrlEdit
}
return(
<div>
<h1>List of ToDos</h1>
<h4>{title}</h4>
<ul>
{state.todos.map(todo => (
<li key = {todo.id}>{todo.text}
<button onClick = {()=>{
editRow(todo)}}>
</button>
</li>
))}
</ul>
</div>
)
}
export default Todos;



![Безумие обратных вызовов в javascript [JS]](https://i.imgur.com/WsjO6zJb.png)


Ваша проблема связана с программной навигацией с использованием react-router-v4 вместо хуков,
В react-router-v4 вы получите историю из реквизитов, если компонент Todos отображается как дочерний элемент или маршрут, или от предка, который визуализирует форму маршрута, и он передал ему реквизиты маршрутизатора. Однако он не получает реквизиты маршрутизатора, вы можете использовать withRouter HOC от react-router, чтобы получить реквизиты маршрутизатора и вызвать props.history.push(destUrlEdit).
import React, {useState, useContext, useEffect} from 'react';
import TodosContext from './context'
import { withRouter } from 'react-router-dom';
function Todos(props){
const [todo, setTodo] = useState("")
const {state, dispatch} = useContext(TodosContext)
useEffect(()=>{
if (state.currentTodo.text){
setTodo(state.currentTodo.text)
} else {
setTodo("")
}
dispatch({
type: "GET_TODOS",
payload: state.todos
})
}, [state.currentTodo.id])
const editRow = event =>{
let destUrlEdit = `/todos/${event.id}`
let obj = {}
obj.id = event.id
obj.text = event.text
dispatch({type:"SET_CURRENT_TODO", payload: obj})
//after dispatch I would like to redirect to another route to do the actual edit
//destUrlEdit
props.history.push(destUrlEdit);
}
return(
<div>
<h1>List of ToDos</h1>
<h4>{title}</h4>
<ul>
{state.todos.map(todo => (
<li key = {todo.id}>{todo.text}
<button onClick = {()=>{
editRow(todo)}}>
</button>
</li>
))}
</ul>
</div>
)
}
export default withRouter(Todos);
Я написал об этом статью: dev.to/httpjunkie/… и пример песочницы здесь: 6wo1673y2r.codesandbox.io Может помочь!
Журналы React дают ощущение, что происходит рекурсивное обновление состояния, в результате чего компонент размонтируется, а затем снова монтируется, но, в конце концов, история размонтирует компонент до того, как состояние сохраняется, это так странно.
@EricBishard Я часами искал что-то настолько простое, как вы предлагаете в своей статье. Я думаю, вы должны опубликовать это как ответ здесь.
Использование реакции-редукции и подключенного-реагирующего-маршрутизатора...
import {useDispatch } from 'react-redux';
import { push } from 'connected-react-router';
export default () => {
const dispatch = useDispatch();
return (
<Button onClick = {() => dispatch(push('/login'))}>
Login
</Button>
);
};
Я получаю сообщение об ошибке Uncaught Could not find router reducer in state tree, it must be mounted under "router", хотя все настроено правильно до того, как я начал использовать хуки. Работали старые методы с использованием классов компонентов. Как только я переключился на этот метод, я получаю эту ошибку.
На самом деле это намного проще, чем другие ответы, React Router v5.1 предоставляет хук useHistory.
import React from 'react'
import { useHistory } from 'react-router-dom'
const MyComponent = () => {
const history = useHistory()
const handleButtonClick = (event) => {
history.push(event.target.value)
}
return (
<button
type = "button"
value = "/my/path"
onClick = {handleButtonClick}
>
Navigate Me!
</button>
)
}
Я думаю, что на самом деле это v5.1, которая представляет эту функцию: reacttraining.com/blog/react-router-v5-1
Да, ты прав! Я обновил версию в своем ответе.
history.push('/') по-прежнему отправит пользователя на старую итерацию страницы, однако вам все равно нужно будет либо поместить ее внутрь useEffect и предоставить useEffect переменную для проверки изменений, либо предоставить реквизиты в вашу историю через props.history.push('/'), где реквизиты передаются в функциональный компонент.
Спасибо за быстрый ответ!! Это указало мне правильное направление для изучения доступа к реквизитам React Router в компонентах класса React и функциональных компонентах React. К вашему ответу следует добавить одну вещь: поскольку <Route> по своей сути предоставляет реквизиты {history, match, location}, вам не нужен HOC withRouter, если маршрут уже определен и находится на основном уровне. Реквизит доступен как есть. Но если компонент глубоко вложен, то HOC withRouter становится полезным.