* feat(tools): add seed/solution restore script * chore(curriculum): remove empty sections' markers * chore(curriculum): add seed + solution to Chinese * chore: remove old formatter * fix: update getChallenges parse translated challenges separately, without reference to the source * chore(curriculum): add dashedName to English * chore(curriculum): add dashedName to Chinese * refactor: remove unused challenge property 'name' * fix: relax dashedName requirement * fix: stray tag Remove stray `pre` tag from challenge file. Signed-off-by: nhcarrigan <nhcarrigan@gmail.com> Co-authored-by: nhcarrigan <nhcarrigan@gmail.com>
		
			
				
	
	
		
			134 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Markdown
		
	
	
	
	
	
			
		
		
	
	
			134 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			Markdown
		
	
	
	
	
	
| ---
 | |
| id: 5a24c314108439a4d4036158
 | |
| title: Never Mutate State
 | |
| challengeType: 6
 | |
| forumTopicId: 301445
 | |
| dashedName: never-mutate-state
 | |
| ---
 | |
| 
 | |
| # --description--
 | |
| 
 | |
| These final challenges describe several methods of enforcing the key principle of state immutability in Redux. Immutable state means that you never modify state directly, instead, you return a new copy of state.
 | |
| 
 | |
| If you took a snapshot of the state of a Redux app over time, you would see something like `state 1`, `state 2`, `state 3`,`state 4`, `...` and so on where each state may be similar to the last, but each is a distinct piece of data. This immutability, in fact, is what provides such features as time-travel debugging that you may have heard about.
 | |
| 
 | |
| Redux does not actively enforce state immutability in its store or reducers, that responsibility falls on the programmer. Fortunately, JavaScript (especially ES6) provides several useful tools you can use to enforce the immutability of your state, whether it is a `string`, `number`, `array`, or `object`. Note that strings and numbers are primitive values and are immutable by nature. In other words, 3 is always 3. You cannot change the value of the number 3. An `array` or `object`, however, is mutable. In practice, your state will probably consist of an `array` or `object`, as these are useful data structures for representing many types of information.
 | |
| 
 | |
| # --instructions--
 | |
| 
 | |
| There is a `store` and `reducer` in the code editor for managing to-do items. Finish writing the `ADD_TO_DO` case in the reducer to append a new to-do to the state. There are a few ways to accomplish this with standard JavaScript or ES6. See if you can find a way to return a new array with the item from `action.todo` appended to the end.
 | |
| 
 | |
| # --hints--
 | |
| 
 | |
| The Redux store should exist and initialize with a state equal to the `todos` array in the code editor.
 | |
| 
 | |
| ```js
 | |
| assert(
 | |
|   (function () {
 | |
|     const todos = [
 | |
|       'Go to the store',
 | |
|       'Clean the house',
 | |
|       'Cook dinner',
 | |
|       'Learn to code'
 | |
|     ];
 | |
|     const initialState = store.getState();
 | |
|     return (
 | |
|       Array.isArray(initialState) && initialState.join(',') === todos.join(',')
 | |
|     );
 | |
|   })()
 | |
| );
 | |
| ```
 | |
| 
 | |
| `addToDo` and `immutableReducer` both should be functions.
 | |
| 
 | |
| ```js
 | |
| assert(typeof addToDo === 'function' && typeof immutableReducer === 'function');
 | |
| ```
 | |
| 
 | |
| Dispatching an action of type `ADD_TO_DO` on the Redux store should add a `todo` item and should NOT mutate state.
 | |
| 
 | |
| ```js
 | |
| assert(
 | |
|   (function () {
 | |
|     const initialState = store.getState();
 | |
|     const isFrozen = DeepFreeze(initialState);
 | |
|     store.dispatch(addToDo('__TEST__TO__DO__'));
 | |
|     const finalState = store.getState();
 | |
|     const expectedState = [
 | |
|       'Go to the store',
 | |
|       'Clean the house',
 | |
|       'Cook dinner',
 | |
|       'Learn to code',
 | |
|       '__TEST__TO__DO__'
 | |
|     ];
 | |
|     return isFrozen && DeepEqual(finalState, expectedState);
 | |
|   })()
 | |
| );
 | |
| ```
 | |
| 
 | |
| # --seed--
 | |
| 
 | |
| ## --seed-contents--
 | |
| 
 | |
| ```js
 | |
| const ADD_TO_DO = 'ADD_TO_DO';
 | |
| 
 | |
| // A list of strings representing tasks to do:
 | |
| const todos = [
 | |
|   'Go to the store',
 | |
|   'Clean the house',
 | |
|   'Cook dinner',
 | |
|   'Learn to code',
 | |
| ];
 | |
| 
 | |
| const immutableReducer = (state = todos, action) => {
 | |
|   switch(action.type) {
 | |
|     case ADD_TO_DO:
 | |
|       // Don't mutate state here or the tests will fail
 | |
|       return
 | |
|     default:
 | |
|       return state;
 | |
|   }
 | |
| };
 | |
| 
 | |
| const addToDo = (todo) => {
 | |
|   return {
 | |
|     type: ADD_TO_DO,
 | |
|     todo
 | |
|   }
 | |
| }
 | |
| 
 | |
| const store = Redux.createStore(immutableReducer);
 | |
| ```
 | |
| 
 | |
| # --solutions--
 | |
| 
 | |
| ```js
 | |
| const ADD_TO_DO = 'ADD_TO_DO';
 | |
| 
 | |
| const todos = [
 | |
|   'Go to the store',
 | |
|   'Clean the house',
 | |
|   'Cook dinner',
 | |
|   'Learn to code',
 | |
| ];
 | |
| 
 | |
| const immutableReducer = (state = todos, action) => {
 | |
|   switch(action.type) {
 | |
|     case ADD_TO_DO:
 | |
|       return state.concat(action.todo);
 | |
|     default:
 | |
|       return state;
 | |
|   }
 | |
| };
 | |
| 
 | |
| const addToDo = (todo) => {
 | |
|   return {
 | |
|     type: ADD_TO_DO,
 | |
|     todo
 | |
|   }
 | |
| }
 | |
| 
 | |
| const store = Redux.createStore(immutableReducer);
 | |
| ```
 |