在 Taro 中使用 Redux

前言

读了上篇文章,相信大家已经掌握了初步的 Taro 开发技能了,本文将带领大家结合 Redux 完善一个 Todolist。
首先,我们得对 Redux 有个初步的了解。Redux 是 JavaScript 状态容器,提供可预测的状态管理。一般来说,规模比较大的小程序,其页面状态和数据缓存等都需要管理很多的东西,这时候引入 Redux 可以方便的管理这些状态,同一数据,一次请求,应用全局共享

而 Taro 也非常友好地为开发者提供了可移植的 Redux 。

依赖

为了更方便地使用 Redux,Taro 提供了与 react-redux API 几乎一致的包 @tarojs/redux 来让开发人员获得更加良好的开发体验。

开发前需要安装 redux@tarojs/redux ,开发者可自行选择安装 Redux 中间件,本文以如下中间件为例:

  1. $ yarn add redux @tarojs/redux redux-logger
  2. # 或者使用 npm
  3. $ npm install --save redux @tarojs/redux redux-logger

示例

下面通过丰富上一篇文章的 Todolist 快速上手 Redux 。

目录结构

首先通过目录划分我们的store/reducers/actions

6基础篇 6:在 Taro 中使用 Redux - 图1

分别在三个文件夹里创建index.js,作为三个模块的入口文件。首先来看看store/index.js里面的内容。reducersactions里面的内容我们需要规划好功能之后再来处理。

  1. // store/index.js
  2. import { createStore, applyMiddleware } from 'redux'
  3. // 引入需要的中间件
  4. import thunkMiddleware from 'redux-thunk'
  5. import { createLogger } from 'redux-logger'
  6. // 引入根reducers
  7. import rootReducer from '../reducers'
  8. const middlewares = [
  9. thunkMiddleware,
  10. createLogger()
  11. ]
  12. // 创建 store
  13. export default function configStore () {
  14. const store = createStore(rootReducer, applyMiddleware(...middlewares))
  15. return store
  16. }

编写 Todos

首先,定义好store,然后在app.js中引入。使用@tarojs/redux中提供的Provider组件将前面写好的store接入应用中,这样一来,被Provider包裹的页面都能访问到应用的store

Provider 组件使组件层级中的 connect() 方法都能够获得 Redux store。

  1. import Taro, { Component } from '@tarojs/taro'
  2. import { Provider } from '@tarojs/redux'
  3. import configStore from './store'
  4. import Index from './pages/index'
  5. import './app.scss'
  6. const store = configStore()
  7. class App extends Component {
  8. ...
  9. render () {
  10. return (
  11. <Provider store={store}>
  12. <Index />
  13. </Provider>
  14. )
  15. }
  16. }

接下来我们正式开始规划 Todos 应用的主要功能。

首先我们可以新建constants文件夹来定义一系列所需的action type常量。例如 Todos 我们可以先增加ADDDELETE两个action type来区分新增和删除 Todo 指令。

  1. // src/constants/todos.js
  2. export const ADD = 'ADD'
  3. export const DELETE = 'DELETE'

然后开始创建处理这两个指令的reducer

  1. // src/reducers/index.js
  2. import { combineReducers } from 'redux'
  3. import { ADD, DELETE } from '../constants/todos'
  4. // 定义初始状态
  5. const INITIAL_STATE = {
  6. todos: [
  7. {id: 0, text: '第一条todo'}
  8. ]
  9. }
  10. function todos (state = INITIAL_STATE, action) {
  11. // 获取当前todos条数,用以id自增
  12. const todoNum = state.todos.length
  13. switch (action.type) {
  14. // 根据指令处理todos
  15. case ADD:
  16. return {
  17. ...state,
  18. todos: state.todos.concat({
  19. id: todoNum,
  20. text: action.data
  21. })
  22. }
  23. case DELETE:
  24. let newTodos = state.todos.filter(item => {
  25. return item.id !== action.id
  26. })
  27. return {
  28. ...state,
  29. todos: newTodos
  30. }
  31. default:
  32. return state
  33. }
  34. }
  35. export default combineReducers({
  36. todos
  37. })

接着在action中定义函数对应的指令。

  1. // src/actions/index.js
  2. import { ADD, DELETE } from '../constants/todos'
  3. export const add = (data) => {
  4. return {
  5. data,
  6. type: ADD
  7. }
  8. }
  9. export const del = (id) => {
  10. return {
  11. id,
  12. type: DELETE
  13. }
  14. }

完成上述三步之后,我们就可以在 Todos 应用的主页使用相应action修改并取得新的store数据了。来看一眼 Todos 的index.js

  1. // src/pages/index/index.js
  2. import Taro, { Component } from '@tarojs/taro'
  3. import { View, Input, Text } from '@tarojs/components'
  4. import { connect } from '@tarojs/redux'
  5. import './index.scss'
  6. import { add, del } from '../../actions/index'
  7. class Index extends Component {
  8. config = {
  9. navigationBarTitleText: '首页'
  10. }
  11. constructor () {
  12. super ()
  13. this.state = {
  14. newTodo: ''
  15. }
  16. }
  17. saveNewTodo (e) {
  18. let { newTodo } = this.state
  19. if (!e.detail.value || e.detail.value === newTodo) return
  20. this.setState({
  21. newTodo: e.detail.value
  22. })
  23. }
  24. addTodo () {
  25. let { newTodo } = this.state
  26. let { add } = this.props
  27. if (!newTodo) return
  28. add(newTodo)
  29. this.setState({
  30. newTodo: ''
  31. })
  32. }
  33. delTodo (id) {
  34. let { del } = this.props
  35. del(id)
  36. }
  37. render () {
  38. // 获取未经处理的todos并展示
  39. let { newTodo } = this.state
  40. let { todos, add, del } = this.props
  41. const todosJsx = todos.map(todo => {
  42. return (
  43. <View className='todos_item'><Text>{todo.text}</Text><View className='del' onClick={this.delTodo.bind(this, todo.id)}>-</View></View>
  44. )
  45. })
  46. return (
  47. <View className='index todos'>
  48. <View className='add_wrap'>
  49. <Input placeholder="填写新的todo" onBlur={this.saveNewTodo.bind(this)} value={newTodo} />
  50. <View className='add' onClick={this.addTodo.bind(this)}>+</View>
  51. </View>
  52. <View>{ todosJsx }</View>
  53. </View>
  54. )
  55. }
  56. }
  57. export default connect (({ todos }) => ({
  58. todos: todos.todos
  59. }), (dispatch) => ({
  60. add (data) {
  61. dispatch(add(data))
  62. },
  63. del (id) {
  64. dispatch(del(id))
  65. }
  66. }))(Index)

最后来看一眼实现的效果:

6基础篇 6:在 Taro 中使用 Redux - 图2

小结

本章我们结合 Redux 丰富了一个 Todolist,通过梳理文件目录结构,规划 Todolist 功能,再细化到每一个文件的具体代码,让读者们深入浅出地了解到如何在 Taro 内结合 Redux 开发应用。

诚然,该文只是提供一种选型建议,是否需要在应用中使用状态管理框架,是否选用 Redux 作为应用的状态管理框架,还需要具体问题具体分析。如果你是在搭建类似商城这样的大型应用,我们非常建议你采用 Redux 管理数据状态,而譬如开发单页应用这类小型的站点,使用 Redux 则有可能会增加你的工作量哦。