React+Redux实现简单的待办事项列表ToDoList

使用Redux做了一个简单的ToDoList待办事项列表
这个例子也是源于Redux作者Dan Abramov的视频demo
还要特别说明一下
我还没有使用react-redux库进行解耦(可能以后加)
也没有拆分成多个文件等等优化
为了单纯的练习redux
适合初步学习redux的同学
本人学疏才浅,发现可以优化的地方或者问题还请大家指正,谢谢

功能样式

样子就是这样的
在输入框输入待办事项
功能很简单
鼠标点击Add或者键盘按下Enter输出
ShowAll显示全部待办事项
ShowActive显示未完成的待办事项(未划掉的)
ShowCrossed显示已完成的待办事项(划掉的)

配置文件

使用Webpack构建的文件夹如下

webpack.config.js配置文件

module.exports = {
    entry: {
        index: './src/js/entry.js'
    },output: {
        path: './static/dist/',publicPath: 'http://localhost:8080/static/dist/',filename: '[name].js'
    },module: {
        loaders: [
            {
                test: /\.js$/,loader: 'babel',exclude:/node_modules/,query: {
                    presets: ['react','es2015']
                }
            },{
                test: /.less$/,loader: 'style!css!less'
            }
        ]
    }
}

package.json的依赖项

{
  "name": "react-demo","version": "1.0.0","description": "","main": "webpack.config.js","scripts": { "test": "echo \"Error: no test specified\" && exit 1","diy": "webpack-dev-server --progress --colors --devtool sourcemap" },"author": "Payson","license": "ISC","devDependencies": { "babel-core": "^6.22.1","babel-loader": "^6.2.10","babel-preset-es2015": "^6.22.0","babel-preset-react": "^6.22.0","css-loader": "^0.26.1","jquery": "^3.1.1","less": "^2.7.2","less-loader": "^2.2.3","react": "^15.4.2","react-dom": "^15.4.2","react-redux": "^5.0.2","redux": "^3.6.0","style-loader": "^0.13.1","webpack": "^1.14.0","webpack-dev-server": "^1.16.2" } }

html文件

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>React</title>
</head>
<body>
    <div id="root"></div>
    <script src="http://localhost:8080/static/dist/index.js"></script>
</body> 
</html>

脚本文件

没有细拆文件
直接写在入口文件entry.js了
注释就写在代码里了

require('../less/index.less'); //行间样式受限制不能添加伪类伪元素,所以还是添加了less(css)控制样式
import React from 'react';
import {Component} from 'react'
import ReactDom from 'react-dom';
import {createStore,combineReducers} from 'redux';

class ToDoList extends Component {
  addHandler(){ //添加待办事项的listener
    let Inp = this.refs.Inp; //获取真实DOM的输入value
    if(!Inp.value){ //如果没有输入值,直接返回
      return;
    }
    store.dispatch( //dispatch一个添加项目的action,并传入输入数据
      {
        type: 'ADD_ITEM',newItem: Inp.value
      }
    )
    Inp.value = ''; //提交后,清空输入
    Inp.focus(); //重置输入焦点
  }
  toggleHandler(item){ //Action Creator:负责提交切换中划线的action
    store.dispatch(
      {
        type: 'TOGGLE_ITEM',changeID: item.ID
      }
    );
  }
  showAllHandler(){ //Action Creator:负责showAll的action
    store.dispatch(
      {
        type: 'SET_FILTER',filter: 'SHOW_ALL'
      }
    );
  }
  showActiveHandler(){ //Action Creator:负责showActive的action
    store.dispatch(
      {
        type: 'SET_FILTER',filter: 'SHOW_ACTIVE'
      }
    );
  }
  showCrossedHandler(){ //Action Creator:负责showCrossed的action
    store.dispatch(
      {
        type: 'SET_FILTER',filter: 'SHOW_CROSSED'
      }
    );
  }
  render(){ //渲染结构样式
    let _this = this; //缓存this
    let state = store.getState(); //缓存store的快照--state
    let {list,option} = state; //解构赋值获取两个子state
    //list是一个数组,内部数组元素是对象表示每一个列表项
    //option是一个字符串,表示当先选择的选项
    switch(option){ //通过判断当前的option字符串来决定是否过滤list数组
      case 'SHOW_ACTIVE':
        list = list.filter(function(item){
          return !item.del;
        });
        break;
      case 'SHOW_CROSSED':
        list = list.filter(function(item){
          return item.del;
        });
        break;
    }
    document.body.addEventListener('keydown',function(e){
      if(e.which == 13){
        _this.addHandler();
      }
    }); //绑定键盘enter事件
    return (
      <div>
        <input type="text" ref="Inp"/> //设置ref属性为了获取真实DOM节点
        <button onClick={_this.addHandler.bind(_this)}>Add</button>
        <ul className="option">
          <li onClick={_this.showAllHandler.bind(_this)}>
            <span style={{textDecoration: option!='SHOW_ALL' ? 'underline' : 'none'}}>ShowAll</span>
          </li>
          <li onClick={_this.showActiveHandler.bind(_this)}>
            <span style={{textDecoration: option!='SHOW_ACTIVE' ? 'underline' : 'none'}}>ShowActive</span>
          </li>
          <li onClick={_this.showCrossedHandler.bind(_this)}>
            <span style={{textDecoration: option!='SHOW_CROSSED' ? 'underline' : 'none'}}>ShowCrossed</span>
          </li> //判断option字符串来决定三个选项的样式
        </ul>
        <ul className="list">
          {
            list.map(function(item,index){ //通过list数组map映射为虚拟DOM节点
              return <li key={index}>
                        <span style={{textDecoration: item.del ? 'line-through': 'none'}} 
                        onClick={_this.toggleHandler.bind(_this,item)}>{item.item}</span>
                     </li>
            })
          }
        </ul>
      </div>
    )
  }
}
const list = (state = [],action) => { //list-reducer
  switch(action.type){
    case 'ADD_ITEM':
      return [
        ...state,{
          item: action.newItem,//列表项内容
          ID: state.length,//列表项ID
          del: false //列表项是否已划掉
        }
      ];
    case 'TOGGLE_ITEM':
      return state.map((item)=>{
        return Object.assign({},item,{
          del: action.changeID == item.ID ? !item.del : item.del
        });
      });
    default:
      return state;
  }
}
const option = (state = 'SHOW_ALL',action) => { //option-reducer
  switch(action.type){
    case 'SET_FILTER':
      return action.filter;
    default:
      return state;
  }
}
const reducer = combineReducers({list,option}); //利用redux库API-combineReducers()合并reducer
const store = createStore(reducer); //利用redux库API-createStore()创建store
const render = () => { //自定义的渲染函数
  ReactDom.render(
    <ToDoList/>,document.getElementById('root')
  );
}
store.subscribe(render); //绑定render函数,每次state更新时执行
render(); //首次渲染

样式文件

index.less文件加一些样式控制

.option {
    list-style-type: none;
    padding: 0;
    margin-top: 5px;
    font-size: 13px;
    li {
        float: left;
        margin-right: 15px;
        span {
            cursor: pointer;
            font-weight: bold;
        }
    }
    &::after {
        content: '';
        display: block;
        clear: both;
    }
}
.list {
    li {
        span {
            &:hover {
                color: #f40;
                cursor: pointer;
            }
            &::selection {
                color: #000;
                background-color: #fff;
            }
        }
    }
}

==主页传送门==

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


react 中的高阶组件主要是对于 hooks 之前的类组件来说的,如果组件之中有复用的代码,需要重新创建一个父类,父类中存储公共代码,返回子类,同时把公用属性...
我们上一节了解了组件的更新机制,但是只是停留在表层上,例如我们的 setState 函数式同步执行的,我们的事件处理直接绑定在了 dom 元素上,这些都跟 re...
我们上一节了解了 react 的虚拟 dom 的格式,如何把虚拟 dom 转为真实 dom 进行挂载。其实函数是组件和类组件也是在这个基础上包裹了一层,一个是调...
react 本身提供了克隆组件的方法,但是平时开发中可能很少使用,可能是不了解。我公司的项目就没有使用,但是在很多三方库中都有使用。本小节我们来学习下如果使用该...
mobx 是一个简单可扩展的状态管理库,中文官网链接。小编在接触 react 就一直使用 mobx 库,上手简单不复杂。
我们在平常的开发中不可避免的会有很多列表渲染逻辑,在 pc 端可以使用分页进行渲染数限制,在移动端可以使用下拉加载更多。但是对于大量的列表渲染,特别像有实时数据...
本小节开始前,我们先答复下一个同学的问题。上一小节发布后,有小伙伴后台来信问到:‘小编你只讲了类组件中怎么使用 ref,那在函数式组件中怎么使用呢?’。确实我们...
上一小节我们了解了固定高度的滚动列表实现,因为是固定高度所以容器总高度和每个元素的 size、offset 很容易得到,这种场景也适合我们常见的大部分场景,例如...
上一小节我们处理了 setState 的批量更新机制,但是我们有两个遗漏点,一个是源码中的 setState 可以传入函数,同时 setState 可以传入第二...
我们知道 react 进行页面渲染或者刷新的时候,会从根节点到子节点全部执行一遍,即使子组件中没有状态的改变,也会执行。这就造成了性能不必要的浪费。之前我们了解...
在平时工作中的某些场景下,你可能想在整个组件树中传递数据,但却不想手动地通过 props 属性在每一层传递属性,contextAPI 应用而生。
楼主最近入职新单位了,恰好新单位使用的技术栈是 react,因为之前一直进行的是 vue2/vue3 和小程序开发,对于这些技术栈实现机制也有一些了解,最少面试...
我们上一节了了解了函数式组件和类组件的处理方式,本质就是处理基于 babel 处理后的 type 类型,最后还是要处理虚拟 dom。本小节我们学习下组件的更新机...
前面几节我们学习了解了 react 的渲染机制和生命周期,本节我们正式进入基本面试必考的核心地带 -- diff 算法,了解如何优化和复用 dom 操作的,还有...
我们在之前已经学习过 react 生命周期,但是在 16 版本中 will 类的生命周期进行了废除,虽然依然可以用,但是需要加上 UNSAFE 开头,表示是不安...
上一小节我们学习了 react 中类组件的优化方式,对于 hooks 为主流的函数式编程,react 也提供了优化方式 memo 方法,本小节我们来了解下它的用...
开源不易,感谢你的支持,❤ star me if you like concent ^_^
hel-micro,模块联邦sdk化,免构建、热更新、工具链无关的微模块方案 ,欢迎关注与了解
本文主题围绕concent的setup和react的五把钩子来展开,既然提到了setup就离不开composition api这个关键词,准确的说setup是由...
ReactsetState的执行是异步还是同步官方文档是这么说的setState()doesnotalwaysimmediatelyupdatethecomponent.Itmaybatchordefertheupdateuntillater.Thismakesreadingthis.staterightaftercallingsetState()apotentialpitfall.Instead,usecom