在成功的异步 redux 操作中转换到另一条路线
Transition to another route on successful async redux action
我有一组非常简单的 React 组件:
container
连接到 redux 并处理操作、商店订阅等
list
显示我的项目列表
new
这是一种将新项目添加到列表的形式
我有一些 react-router 路线如下:
<Route name='products' path='products' handler={ProductsContainer}>
<Route name='productNew' path='new' handler={ProductNew} />
<DefaultRoute handler={ProductsList} />
</Route>
以便显示 list
或 form
,但不会同时显示两者。
我想做的是在成功添加新项目后让应用程序重新路由回列表。
到目前为止我的解决方案是在异步 dispatch
:
之后有一个 .then()
dispatch(actions.addProduct(product)
.then(this.transitionTo('products'))
)
这是执行此操作的正确方法还是我应该以某种方式触发另一个操作来触发路由更改?
我最终创建了一个超级简单的中间件,大致如下所示:
import history from "../routes/history";
export default store => next => action => {
if ( ! action.redirect ) return next(action);
history.replaceState(null, action.redirect);
}
因此您只需确保您的 successful
操作具有 redirect
属性。另请注意,此中间件不会触发 next()
。这是故意的,因为路由转换应该是动作链的末端。
如果您不想使用更完整的解决方案,例如 Redux Router, you can use Redux History Transitions,您可以这样编写代码:
export function login() {
return {
type: LOGGED_IN,
payload: {
userId: 123
}
meta: {
transition: (state, action) => ({
path: `/logged-in/${action.payload.userId}`,
query: {
some: 'queryParam'
},
state: {
some: 'state'
}
})
}
};
}
这类似于 but a tiny bit more sophisticated. It still uses the same history 引擎盖下的库,因此它与 React Router 兼容。
对于那些使用中间件 API 层来抽象他们对 isomorphic-fetch, and also happen to be using redux-thunk 之类的用法的人,您可以简单地将 dispatch
Promise 链接到您的操作中,就像这样:
import { push } from 'react-router-redux';
const USER_ID = // imported from JWT;
function fetchUser(primaryKey, opts) {
// this prepares object for the API middleware
}
// this can be called from your container
export function updateUser(payload, redirectUrl) {
var opts = {
method: 'PUT',
headers: {
'Content-Type': 'application/json'
},
body: JSON.stringify(payload)
};
return (dispatch) => {
return dispatch(fetchUser(USER_ID, opts))
.then((action) => {
if (action.type === ActionTypes.USER_SUCCESS) {
dispatch(push(redirectUrl));
}
});
};
}
这减少了按照建议将库添加到代码中的需要 , and also nicely co-locates your actions with their redirects as done in redux-history-transitions。
这是我的商店的样子:
import { createStore, applyMiddleware } from 'redux';
import rootReducer from '../reducers';
import thunk from 'redux-thunk';
import api from '../middleware/api';
import { routerMiddleware } from 'react-router-redux';
export default function configureStore(initialState, browserHistory) {
const store = createStore(
rootReducer,
initialState,
applyMiddleware(thunk, api, routerMiddleware(browserHistory))
);
return store;
}
我知道我来晚了一点,因为 react-navigation 已经包含在 react-native 文档中,但它仍然对拥有 used/using Navigator api 的用户有用在他们的应用程序中。
我尝试的有点老套,我在 renderScene 发生后立即将导航器实例保存在对象中 -
renderScene(route, navigator) {
const Component = Routes[route.Name]
api.updateNavigator(navigator); //will allow us to access navigator anywhere within the app
return <Component route={route} navigator={navigator} data={route.data}/>
}
我的 api 文件是这样的
'use strict';
//this file includes all my global functions
import React, {Component} from 'react';
import {Linking, Alert, NetInfo, Platform} from 'react-native';
var api = {
navigator,
isAndroid(){
return (Platform.OS === 'android');
},
updateNavigator(_navigator){
if(_navigator)
this.navigator = _navigator;
},
}
module.exports = api;
现在在您的操作中,您只需调用
api.navigator.push({Name:'routeName',
data:WHATEVER_YOU_WANTED_TO_PASS)
您只需要从模块中导入 api。
如果您正在使用 react-redux and react-router, then I think this link 提供了一个很好的解决方案。
这是我使用的步骤:
- 通过在 react-router
<Route/>
组件内渲染组件或使用 withRouter
创建高阶组件,将 react-router history
prop 传递给您的组件.
- 接下来,创建您要重定向到的路由(我将其命名为
to
)。
- 第三,用
history
和 to
调用你的 redux 动作。
- 最后,当您想要重定向时(例如,当您的 redux 操作解析时),请调用
history.push(to)
。
我有一组非常简单的 React 组件:
container
连接到 redux 并处理操作、商店订阅等list
显示我的项目列表new
这是一种将新项目添加到列表的形式
我有一些 react-router 路线如下:
<Route name='products' path='products' handler={ProductsContainer}>
<Route name='productNew' path='new' handler={ProductNew} />
<DefaultRoute handler={ProductsList} />
</Route>
以便显示 list
或 form
,但不会同时显示两者。
我想做的是在成功添加新项目后让应用程序重新路由回列表。
到目前为止我的解决方案是在异步 dispatch
:
.then()
dispatch(actions.addProduct(product)
.then(this.transitionTo('products'))
)
这是执行此操作的正确方法还是我应该以某种方式触发另一个操作来触发路由更改?
我最终创建了一个超级简单的中间件,大致如下所示:
import history from "../routes/history";
export default store => next => action => {
if ( ! action.redirect ) return next(action);
history.replaceState(null, action.redirect);
}
因此您只需确保您的 successful
操作具有 redirect
属性。另请注意,此中间件不会触发 next()
。这是故意的,因为路由转换应该是动作链的末端。
如果您不想使用更完整的解决方案,例如 Redux Router, you can use Redux History Transitions,您可以这样编写代码:
export function login() {
return {
type: LOGGED_IN,
payload: {
userId: 123
}
meta: {
transition: (state, action) => ({
path: `/logged-in/${action.payload.userId}`,
query: {
some: 'queryParam'
},
state: {
some: 'state'
}
})
}
};
}
这类似于
对于那些使用中间件 API 层来抽象他们对 isomorphic-fetch, and also happen to be using redux-thunk 之类的用法的人,您可以简单地将 dispatch
Promise 链接到您的操作中,就像这样:
import { push } from 'react-router-redux';
const USER_ID = // imported from JWT;
function fetchUser(primaryKey, opts) {
// this prepares object for the API middleware
}
// this can be called from your container
export function updateUser(payload, redirectUrl) {
var opts = {
method: 'PUT',
headers: {
'Content-Type': 'application/json'
},
body: JSON.stringify(payload)
};
return (dispatch) => {
return dispatch(fetchUser(USER_ID, opts))
.then((action) => {
if (action.type === ActionTypes.USER_SUCCESS) {
dispatch(push(redirectUrl));
}
});
};
}
这减少了按照建议将库添加到代码中的需要
这是我的商店的样子:
import { createStore, applyMiddleware } from 'redux';
import rootReducer from '../reducers';
import thunk from 'redux-thunk';
import api from '../middleware/api';
import { routerMiddleware } from 'react-router-redux';
export default function configureStore(initialState, browserHistory) {
const store = createStore(
rootReducer,
initialState,
applyMiddleware(thunk, api, routerMiddleware(browserHistory))
);
return store;
}
我知道我来晚了一点,因为 react-navigation 已经包含在 react-native 文档中,但它仍然对拥有 used/using Navigator api 的用户有用在他们的应用程序中。 我尝试的有点老套,我在 renderScene 发生后立即将导航器实例保存在对象中 -
renderScene(route, navigator) {
const Component = Routes[route.Name]
api.updateNavigator(navigator); //will allow us to access navigator anywhere within the app
return <Component route={route} navigator={navigator} data={route.data}/>
}
我的 api 文件是这样的
'use strict';
//this file includes all my global functions
import React, {Component} from 'react';
import {Linking, Alert, NetInfo, Platform} from 'react-native';
var api = {
navigator,
isAndroid(){
return (Platform.OS === 'android');
},
updateNavigator(_navigator){
if(_navigator)
this.navigator = _navigator;
},
}
module.exports = api;
现在在您的操作中,您只需调用
api.navigator.push({Name:'routeName', data:WHATEVER_YOU_WANTED_TO_PASS)
您只需要从模块中导入 api。
如果您正在使用 react-redux and react-router, then I think this link 提供了一个很好的解决方案。
这是我使用的步骤:
- 通过在 react-router
<Route/>
组件内渲染组件或使用withRouter
创建高阶组件,将 react-routerhistory
prop 传递给您的组件. - 接下来,创建您要重定向到的路由(我将其命名为
to
)。 - 第三,用
history
和to
调用你的 redux 动作。 - 最后,当您想要重定向时(例如,当您的 redux 操作解析时),请调用
history.push(to)
。