* fix(files): Decouple files from challenges * feat(server/react): Remove action logger use redux remote devtools instead! * feat(Challenges): Disable js on edit, enable on execute * feat(Challenge/Preview): Show message when js is disabled * refactor(frameEpic): Reduce code by using lodash * feat(frameEpic): Disable js in preview by state * feat(frameEpic): Colocate epic in Challenges/redux * refactor(ExecuteChallengeEpic): CoLocated with Challenges * refactor(executeChallengesEpic): Separate tests from main logic * feat(Challenge/Preview): Update main on edit * feat(frameEpuc): Replace frame on edit/execute This allows for sandbox to work properly * fix(Challenges/Utils): Require utisl * revert(frameEpic): Hoist function to mount code in frame * fix(frameEpic): Ensure new frame is given classname * feat(executeChallenge): Update main on code unlocked * fix(frameEpic): Filter out empty test message * fix(Challenge/Preview): Remove unnessary quote in classname * feat(codeStorageEpic): Separate localstorage from solutions loading * fix(fetchUser): Merge user actions into one prefer many effects from one action over one action to one effect * fix(themes): Centralize theme utils and defs * fix(entities.user): Fix user reducer namespacing * feat(frame): Refactor frameEpic to util * feat(Challenges.redux): Should not attempt to update main from storage * fix(loadPreviousChallengeEpic): Refactor for RFR * fix(Challenges.Modern): Show preview plane
		
			
				
	
	
		
			180 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			180 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| import _ from 'lodash';
 | |
| import {
 | |
|   composeReducers,
 | |
|   createAction,
 | |
|   createTypes,
 | |
|   handleActions
 | |
| } from 'berkeleys-redux-utils';
 | |
| 
 | |
| import { themes } from '../../utils/themes';
 | |
| import { types as challenges } from '../routes/Challenges/redux';
 | |
| 
 | |
| export const ns = 'entities';
 | |
| export const getNS = state => state[ns];
 | |
| export const entitiesSelector = getNS;
 | |
| export const types = createTypes([
 | |
|   'updateTheme',
 | |
|   'updateUserFlag',
 | |
|   'updateUserEmail',
 | |
|   'updateUserLang',
 | |
|   'updateUserCurrentChallenge'
 | |
| ], ns);
 | |
| 
 | |
| // updateUserFlag(username: String, flag: String) => Action
 | |
| export const updateUserFlag = createAction(
 | |
|   types.updateUserFlag,
 | |
|   (username, flag) => ({ username, flag })
 | |
| );
 | |
| // updateUserEmail(username: String, email: String) => Action
 | |
| export const updateUserEmail = createAction(
 | |
|   types.updateUserEmail,
 | |
|   (username, email) => ({ username, email })
 | |
| );
 | |
| // updateUserLang(username: String, lang: String) => Action
 | |
| export const updateUserLang = createAction(
 | |
|   types.updateUserLang,
 | |
|   (username, lang) => ({ username, languageTag: lang })
 | |
| );
 | |
| 
 | |
| export const updateUserCurrentChallenge = createAction(
 | |
|   types.updateUserCurrentChallenge
 | |
| );
 | |
| 
 | |
| // entity meta creators
 | |
| const getEntityAction = _.property('meta.entitiesAction');
 | |
| export const updateThemeMetacreator = (username, theme) => ({
 | |
|   entitiesAction: {
 | |
|     type: types.updateTheme,
 | |
|     payload: {
 | |
|       username,
 | |
|       theme: !theme || theme === themes.default ? themes.default : themes.night
 | |
|     }
 | |
|   }
 | |
| });
 | |
| 
 | |
| const defaultState = {
 | |
|   superBlock: {},
 | |
|   block: {},
 | |
|   challenge: {},
 | |
|   user: {}
 | |
| };
 | |
| 
 | |
| export const challengeMapSelector = state => getNS(state).challenge || {};
 | |
| export function makeBlockSelector(block) {
 | |
|   return state => {
 | |
|     const blockMap = getNS(state).block || {};
 | |
|     return blockMap[block] || {};
 | |
|   };
 | |
| }
 | |
| export function makeSuperBlockSelector(name) {
 | |
|   return state => {
 | |
|     const superBlock = getNS(state).superBlock || {};
 | |
|     return superBlock[name] || {};
 | |
|   };
 | |
| }
 | |
| 
 | |
| export const isChallengeLoaded = (state, { dashedName }) =>
 | |
|   !!challengeMapSelector(state)[dashedName];
 | |
| 
 | |
| export default composeReducers(
 | |
|   ns,
 | |
|   function metaReducer(state = defaultState, action) {
 | |
|     if (action.meta && action.meta.entities) {
 | |
|       return {
 | |
|         ...state,
 | |
|         ...action.meta.entities
 | |
|       };
 | |
|     }
 | |
|     return state;
 | |
|   },
 | |
|   function(state = defaultState, action) {
 | |
|     if (getEntityAction(action)) {
 | |
|       const { payload: { username, theme } } = getEntityAction(action);
 | |
|       return {
 | |
|         ...state,
 | |
|         user: {
 | |
|           ...state.user,
 | |
|           [username]: {
 | |
|             ...state.user[username],
 | |
|             theme
 | |
|           }
 | |
|         }
 | |
|       };
 | |
|     }
 | |
|     return state;
 | |
|   },
 | |
|   handleActions(
 | |
|     () => ({
 | |
|       [
 | |
|         challenges.submitChallenge.complete
 | |
|       ]: (state, { payload: { username, points, challengeInfo } }) => ({
 | |
|         ...state,
 | |
|         user: {
 | |
|           ...state.user,
 | |
|           [username]: {
 | |
|             ...state.user[username],
 | |
|             points,
 | |
|             challengeMap: {
 | |
|               ...state.user[username].challengeMap,
 | |
|               [challengeInfo.id]: challengeInfo
 | |
|             }
 | |
|           }
 | |
|         }
 | |
|       }),
 | |
|       [types.updateUserFlag]: (state, { payload: { username, flag } }) => ({
 | |
|         ...state,
 | |
|         user: {
 | |
|           ...state.user,
 | |
|           [username]: {
 | |
|             ...state.user[username],
 | |
|             [flag]: !state.user[username][flag]
 | |
|           }
 | |
|         }
 | |
|       }),
 | |
|       [types.updateUserEmail]: (state, { payload: { username, email } }) => ({
 | |
|         ...state,
 | |
|         user: {
 | |
|           ...state.user,
 | |
|           [username]: {
 | |
|             ...state.user[username],
 | |
|             email
 | |
|           }
 | |
|         }
 | |
|       }),
 | |
|       [types.updateUserLang]:
 | |
|       (
 | |
|         state,
 | |
|         {
 | |
|           payload: { username, languageTag }
 | |
|         }
 | |
|       ) => ({
 | |
|         ...state,
 | |
|         user: {
 | |
|           ...state.user,
 | |
|           [username]: {
 | |
|             ...state.user[username],
 | |
|             languageTag
 | |
|           }
 | |
|         }
 | |
|       }),
 | |
|       [types.updateUserCurrentChallenge]:
 | |
|       (
 | |
|         state,
 | |
|         {
 | |
|           payload: { username, currentChallengeId }
 | |
|         }
 | |
|       ) => ({
 | |
|         ...state,
 | |
|         user: {
 | |
|           ...state.user,
 | |
|           [username]: {
 | |
|             ...state.user[username],
 | |
|             currentChallengeId
 | |
|           }
 | |
|         }
 | |
|       })
 | |
|     }),
 | |
|     defaultState
 | |
|   )
 | |
| );
 |