App.js 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154
  1. import React, { Component } from 'react';
  2. import logo from './logo.svg';
  3. import './App.css';
  4. import { connect, Provider } from 'react-redux';
  5. import { createStore, combineReducers } from 'redux';
  6. let todoList = [
  7. {
  8. title: 'todo one',
  9. text: 'todo one text',
  10. timestamp: (new Date()).getTime()
  11. },
  12. {
  13. title: 'todo two',
  14. text: 'todo two text',
  15. timestamp: (new Date()).getTime()
  16. },
  17. {
  18. title: 'todo three',
  19. text: 'todo three text',
  20. timestamp: (new Date()).getTime()
  21. }
  22. ];
  23. function todoReducer(state, action){
  24. if (typeof state === 'undefined'){
  25. return {items: JSON.parse(localStorage.todo)};
  26. }
  27. if (action.type == 'NEW_TODO'){
  28. console.log(state);
  29. return {items: [...state.items, {title: action.title, text: action.text, timestamp: (new Date()).getTime()}]}
  30. }
  31. if (action.type == 'DELETE_TODO'){
  32. let items = [];
  33. for (let i=0;i<state.items.length;i++){
  34. if (state.items[i].timestamp != action.timestamp){
  35. items.push(state.items[i]);
  36. }
  37. }
  38. return {items};
  39. }
  40. }
  41. const reducers = combineReducers({
  42. todo: todoReducer,
  43. })
  44. var store = createStore(reducers)
  45. store.subscribe(() => {
  46. let state = store.getState();
  47. localStorage.todo = JSON.stringify(state.todo.items);
  48. })
  49. class ToDoItem extends Component {
  50. constructor(props){
  51. super(props);
  52. this.delete = this.delete.bind(this);
  53. }
  54. delete() {
  55. store.dispatch({type: "DELETE_TODO", timestamp: this.props.item.timestamp})
  56. }
  57. render() {
  58. return (
  59. <div>
  60. <h3> { this.props.item.title } </h3>
  61. <p> { this.props.item.text }</p>
  62. <span> { (new Date(this.props.item.timestamp)).toLocaleString() } {this.props.item.timestamp} </span>
  63. <button onClick={this.delete} >x</button>
  64. </div>
  65. );
  66. }
  67. }
  68. class ToDoForm extends Component {
  69. constructor (props) {
  70. super(props);
  71. this.save = this.save.bind(this);
  72. this.state = {valid: true};
  73. }
  74. validator(){
  75. return this.title.value && this.text.value;
  76. }
  77. save() {
  78. let valid = this.validator();
  79. this.setState({valid})
  80. if (valid){
  81. store.dispatch({type: 'NEW_TODO',
  82. title: this.title.value,
  83. text: this.text.value})
  84. this.title.value = '';
  85. this.text.value = '';
  86. }
  87. }
  88. render (){
  89. let style = {
  90. backgroundColor: this.state.valid ? '' : 'red'
  91. }
  92. return (
  93. <div>
  94. <input style={style} placeholder='title' ref={ c => this.title = c}/>
  95. <input style={style} placeholder='text' ref={ c => this.text = c} />
  96. <button onClick = { this.save } >Save</button>
  97. </div>
  98. );
  99. }
  100. }
  101. class ToDoList extends Component {
  102. render (){
  103. return (
  104. <div> <h1>TODO LIST</h1>
  105. <ToDoForm />
  106. { this.props.items.map( (item, index) => <ToDoItem key={index} item={item} />) }
  107. </div>
  108. );
  109. }
  110. }
  111. const mapStateToProps = function(store){
  112. return {
  113. items: store.todo.items
  114. };
  115. }
  116. ToDoList = connect(mapStateToProps)(ToDoList);
  117. class App extends Component {
  118. render() {
  119. return (
  120. <div className="App">
  121. <Provider store={store} >
  122. <ToDoList />
  123. </Provider>
  124. </div>
  125. );
  126. }
  127. }
  128. export default App;