Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Enable flexible number of colors for themes #3482

Open
wants to merge 19 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
19 commits
Select commit Hold shift + click to select a range
f77edcf
feat: Enable flexible number of colors for themes
joeng03 Aug 3, 2023
b0392c7
fix: venue timetable renders white color lessons
joeng03 Aug 4, 2023
c8f0b38
Merge branch 'master' into joeng03/support-flexible-num-of-colors-in-…
joeng03 Aug 4, 2023
8563458
fix: linting errors
joeng03 Aug 4, 2023
0888fe2
Merge branch 'joeng03/support-flexible-num-of-colors-in-theme' of htt…
joeng03 Aug 4, 2023
bf7e006
refactor: refactor test cases to reflect current code modifications
joeng03 Aug 6, 2023
aeafe7c
feat: add action reassignAllModulesColor to reassign the color of the…
joeng03 Aug 7, 2023
1885a5c
Merge branch 'master' of https://github.com/joeng03/nusmods into joen…
joeng03 Aug 7, 2023
ef26201
fix(SettingsContainer): eslint.no-shadow
joeng03 Aug 7, 2023
2c34989
Merge branch 'master' into joeng03/support-flexible-num-of-colors-in-…
joeng03 Aug 7, 2023
6ec7da5
Merge branch 'master' into joeng03/support-flexible-num-of-colors-in-…
joeng03 Aug 9, 2023
52b4a53
Merge branch 'master' into joeng03/support-flexible-num-of-colors-in-…
joeng03 Aug 10, 2023
b97aa26
fix: lint error
joeng03 Aug 10, 2023
7ec797f
Merge branch 'master' into joeng03/support-flexible-num-of-colors-in-…
zwliew Aug 10, 2023
cbb938b
Merge branch 'master' into joeng03/support-flexible-num-of-colors-in-…
zwliew Aug 11, 2023
a4304f4
refactor: address changes requested
joeng03 Aug 13, 2023
927e7c9
merge changes from master
joeng03 Aug 13, 2023
a7ac61e
Merge branch 'master' into joeng03/support-flexible-num-of-colors-in-…
joeng03 Aug 14, 2023
0699b0a
Merge branch 'master' into joeng03/support-flexible-num-of-colors-in-…
zwliew Nov 26, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 5 additions & 1 deletion website/src/actions/__snapshots__/theme.test.ts.snap
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,11 @@ exports[`theme should dispatch a cycle of theme 2`] = `

exports[`theme should dispatch a select of theme 1`] = `
{
"payload": "test",
"payload": {
"id": "test",
"name": "Test",
"numOfColors": 8,
},
"type": "SELECT_THEME",
}
`;
Expand Down
7 changes: 6 additions & 1 deletion website/src/actions/theme.test.ts
Original file line number Diff line number Diff line change
@@ -1,8 +1,13 @@
import * as actions from 'actions/theme';
import { Theme } from 'types/settings';

describe('theme', () => {
test('should dispatch a select of theme', () => {
const theme = 'test';
const theme: Theme = {
id: 'test',
name: 'Test',
numOfColors: 8,
};
expect(actions.selectTheme(theme)).toMatchSnapshot();
});

Expand Down
4 changes: 3 additions & 1 deletion website/src/actions/theme.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,7 @@
import { Theme } from 'types/settings';

export const SELECT_THEME = 'SELECT_THEME' as const;
export function selectTheme(theme: string) {
export function selectTheme(theme: Theme) {
return {
type: SELECT_THEME,
payload: theme,
Expand Down
21 changes: 19 additions & 2 deletions website/src/actions/timetables.ts
Original file line number Diff line number Diff line change
Expand Up @@ -33,13 +33,19 @@
};
},

addModule(semester: Semester, moduleCode: ModuleCode, moduleLessonConfig: ModuleLessonConfig) {
addModule(
semester: Semester,
moduleCode: ModuleCode,
moduleLessonConfig: ModuleLessonConfig,
numOfColors: number,
) {

Check warning on line 41 in website/src/actions/timetables.ts

View check run for this annotation

Codecov / codecov/patch

website/src/actions/timetables.ts#L41

Added line #L41 was not covered by tests
return {
type: ADD_MODULE,
payload: {
semester,
moduleCode,
moduleLessonConfig,
numOfColors,
},
};
},
Expand Down Expand Up @@ -67,8 +73,9 @@

const lessons = getModuleTimetable(module, semester);
const moduleLessonConfig = randomModuleLessonConfig(lessons);
const { numOfColors } = getState().theme;

Check warning on line 76 in website/src/actions/timetables.ts

View check run for this annotation

Codecov / codecov/patch

website/src/actions/timetables.ts#L76

Added line #L76 was not covered by tests

dispatch(Internal.addModule(semester, moduleCode, moduleLessonConfig));
dispatch(Internal.addModule(semester, moduleCode, moduleLessonConfig, numOfColors));

Check warning on line 78 in website/src/actions/timetables.ts

View check run for this annotation

Codecov / codecov/patch

website/src/actions/timetables.ts#L78

Added line #L78 was not covered by tests
});
}

Expand Down Expand Up @@ -238,6 +245,16 @@
};
}

export const REASSIGN_ALL_MODULES_COLOR = 'REASSIGN_ALL_MODULES_COLOR' as const;
export function reassignAllModulesColor(numOfColors: number) {
return {

Check warning on line 250 in website/src/actions/timetables.ts

View check run for this annotation

Codecov / codecov/patch

website/src/actions/timetables.ts#L249-L250

Added lines #L249 - L250 were not covered by tests
type: REASSIGN_ALL_MODULES_COLOR,
payload: {
numOfColors,
},
};
}

export const HIDE_LESSON_IN_TIMETABLE = 'HIDE_LESSON_IN_TIMETABLE' as const;
export function hideLessonInTimetable(semester: Semester, moduleCode: ModuleCode) {
return {
Expand Down
41 changes: 29 additions & 12 deletions website/src/data/themes.json
Original file line number Diff line number Diff line change
@@ -1,50 +1,67 @@
[
{
"id": "ashes",
"name": " Ashes"
"name": " Ashes",
"numOfColors": 8
},
{
"id": "chalk",
"name": "Chalk"
"name": "Chalk",
"numOfColors": 8
},
{
"id": "eighties",
"name": "Eighties"
"name": "Eighties",
"numOfColors": 8
},
{
"id": "google",
"name": "Google"
"name": "Google",
"numOfColors": 8
},
{
"id": "mocha",
"name": "Mocha"
"name": "Mocha",
"numOfColors": 8
},
{
"id": "monokai",
"name": "Monokai"
"name": "Monokai",
"numOfColors": 8
},
{
"id": "ocean",
"name": "Ocean"
"name": "Ocean",
"numOfColors": 8
},
{
"id": "oceanic-next",
"name": "Oceanic Next"
"name": "Oceanic Next",
"numOfColors": 8
},
{
"id": "paraiso",
"name": "Paraiso"
"name": "Paraiso",
"numOfColors": 8
},
{
"id": "railscasts",
"name": "Railscasts"
"name": "Railscasts",
"numOfColors": 8
},
{
"id": "tequila",
"name": "Tequila",
"numOfColors": 10
},
{
"id": "tomorrow",
"name": "Tomorrow"
"name": "Tomorrow",
"numOfColors": 8
},
{
"id": "twilight",
"name": "Twilight"
"name": "Twilight",
"numOfColors": 8
}
]
6 changes: 3 additions & 3 deletions website/src/entry/export/TimetableOnly.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -29,15 +29,15 @@

override render() {
const { store } = this.props;
const theme = store.getState().theme.id;
const { theme } = store.getState();

Check warning on line 32 in website/src/entry/export/TimetableOnly.tsx

View check run for this annotation

Codecov / codecov/patch

website/src/entry/export/TimetableOnly.tsx#L32

Added line #L32 was not covered by tests

const { semester, timetable, colors } = this.state;
const timetableColors = fillColorMapping(timetable, colors);
const timetableColors = fillColorMapping(timetable, colors, theme.numOfColors);

Check warning on line 35 in website/src/entry/export/TimetableOnly.tsx

View check run for this annotation

Codecov / codecov/patch

website/src/entry/export/TimetableOnly.tsx#L35

Added line #L35 was not covered by tests

return (
<MemoryRouter initialEntries={['https://nusmods.com']}>
<Provider store={store}>
<div id="timetable-only" className={`theme-${theme}`}>
<div id="timetable-only" className={`theme-${theme.id}`}>
<TimetableContent
header={null}
semester={semester}
Expand Down
2 changes: 2 additions & 0 deletions website/src/reducers/index.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@ const exportData: ExportData = {
hidden: ['PC1222'],
theme: {
id: 'google',
numOfColors: 8,
timetableOrientation: VERTICAL,
showTitle: true,
},
Expand Down Expand Up @@ -84,6 +85,7 @@ test('reducers should set export data state', () => {

expect(state.theme).toEqual({
id: 'google',
numOfColors: 8,
timetableOrientation: VERTICAL,
showTitle: true,
});
Expand Down
9 changes: 7 additions & 2 deletions website/src/reducers/theme.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,10 +2,15 @@ import { ThemeState, VERTICAL } from 'types/reducers';

import * as actions from 'actions/theme';
import reducer, { defaultThemeState, themeIds } from 'reducers/theme';
import { Theme } from 'types/settings';

const themeInitialState: ThemeState = defaultThemeState;
const googleTheme = 'google';
const themeWithEighties: ThemeState = { ...themeInitialState, id: googleTheme };
const googleTheme: Theme = {
id: 'google',
name: 'Google',
numOfColors: 8,
};
const themeWithEighties: ThemeState = { ...themeInitialState, id: googleTheme.id };
const themeWithFirstTheme: ThemeState = { ...themeInitialState, id: themeIds[0] };
const themeWithLastTheme: ThemeState = { ...themeInitialState, id: themeIds[themeIds.length - 1] };
const themeWithVerticalOrientation: ThemeState = {
Expand Down
11 changes: 7 additions & 4 deletions website/src/reducers/theme.ts
Original file line number Diff line number Diff line change
Expand Up @@ -15,31 +15,34 @@ import { DIMENSIONS, withTracker } from 'bootstrapping/matomo';
export const defaultThemeState: ThemeState = {
// Available themes are defined in `themes.scss`
id: 'eighties',
numOfColors: 8,
timetableOrientation: HORIZONTAL,
showTitle: false,
};
export const themeIds = themes.map((obj: Theme) => obj.id);

function theme(state: ThemeState = defaultThemeState, action: Actions): ThemeState {
function setTheme(newTheme: string): ThemeState {
function setTheme(newTheme: Theme): ThemeState {
// Update theme analytics info
withTracker((tracker) => tracker.setCustomDimension(DIMENSIONS.theme, newTheme));
withTracker((tracker) => tracker.setCustomDimension(DIMENSIONS.theme, newTheme.id));

return {
...state,
id: newTheme,
id: newTheme.id,
numOfColors: newTheme.numOfColors,
Copy link
Member

@zwliew zwliew Aug 11, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Since this affects local storage schema, we should be more careful here. In the long run, I don't think we'll want to store numOfColors in ThemeState as it only contains user customizable data (i.e. current theme ID, should show title, orientation of timetable).

Since numOfColors is a static property of a theme, instead of storing it in Redux, we should store it in a predefined readonly map like the one that you've defined in themes.json. This has the added benefit of you no longer having to pass numOfColors around everywhere as you can just read it from the map.

That said, if/when we add user-defined themes, they should be stored in Redux, but likely in a different place instead of ThemeState.

};
}

switch (action.type) {
case SELECT_THEME:
// Reassign all modules' color when changing theme
return setTheme(action.payload);

case CYCLE_THEME: {
const newThemeIndex =
(themeIds.indexOf(state.id) + themeIds.length + action.payload) % themeIds.length;

return setTheme(themeIds[newThemeIndex]);
return setTheme(themes[newThemeIndex]);
}
case TOGGLE_TIMETABLE_ORIENTATION:
return {
Expand Down
5 changes: 5 additions & 0 deletions website/src/reducers/timetables.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,9 @@ import {
import { TimetablesState } from 'types/reducers';
import config from 'config';

// use 8 different colors for testing
const NUM_DIFFERENT_COLORS = 8;

const initialState = defaultTimetableState;

jest.mock('config');
Expand All @@ -28,6 +31,7 @@ describe('color reducers', () => {
semester: 1,
moduleCode: 'CS1010S',
moduleLessonConfig: {},
numOfColors: NUM_DIFFERENT_COLORS,
},
}).colors,
).toHaveProperty('1.CS1010S');
Expand All @@ -39,6 +43,7 @@ describe('color reducers', () => {
semester: 2,
moduleCode: 'CS3216',
moduleLessonConfig: {},
numOfColors: NUM_DIFFERENT_COLORS,
},
}).colors,
).toHaveProperty('2.CS3216');
Expand Down
22 changes: 20 additions & 2 deletions website/src/reducers/timetables.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,14 +5,15 @@
import { PersistConfig } from 'storage/persistReducer';
import { ModuleCode } from 'types/modules';
import { ModuleLessonConfig, SemTimetableConfig } from 'types/timetables';
import { ColorMapping, TimetablesState } from 'types/reducers';
import { ColorMapping, SemesterColorMap, TimetablesState } from 'types/reducers';

import config from 'config';
import {
ADD_MODULE,
CHANGE_LESSON,
HIDDEN_IMPORTED_SEM,
HIDE_LESSON_IN_TIMETABLE,
REASSIGN_ALL_MODULES_COLOR,
REMOVE_MODULE,
RESET_TIMETABLE,
SELECT_MODULE_COLOR,
Expand Down Expand Up @@ -134,7 +135,7 @@
case ADD_MODULE:
return {
...state,
[moduleCode]: getNewColor(values(state)),
[moduleCode]: getNewColor(values(state), action.payload.numOfColors),
};

case REMOVE_MODULE:
Expand All @@ -151,6 +152,18 @@
}
}

function recolor(curSemesterColorMap: SemesterColorMap, numOfColors: number): SemesterColorMap {
const newSemesterColorMap: SemesterColorMap = {};
Object.entries(curSemesterColorMap).forEach(([semester, colorMapping]) => {
const newColorMapping: ColorMapping = {};
Object.entries(colorMapping).forEach(([moduleCode, colorIndex]) => {
newColorMapping[moduleCode] = colorIndex % numOfColors;

Check warning on line 160 in website/src/reducers/timetables.ts

View check run for this annotation

Codecov / codecov/patch

website/src/reducers/timetables.ts#L155-L160

Added lines #L155 - L160 were not covered by tests
});
newSemesterColorMap[semester] = newColorMapping;

Check warning on line 162 in website/src/reducers/timetables.ts

View check run for this annotation

Codecov / codecov/patch

website/src/reducers/timetables.ts#L162

Added line #L162 was not covered by tests
});
return newSemesterColorMap;

Check warning on line 164 in website/src/reducers/timetables.ts

View check run for this annotation

Codecov / codecov/patch

website/src/reducers/timetables.ts#L164

Added line #L164 was not covered by tests
}

// Map of semester to list of hidden modules
const DEFAULT_HIDDEN_STATE: ModuleCode[] = [];
function semHiddenModules(state = DEFAULT_HIDDEN_STATE, action: Actions) {
Expand Down Expand Up @@ -236,6 +249,11 @@
hidden: { [semester]: hidden },
};
}
case REASSIGN_ALL_MODULES_COLOR:
return {

Check warning on line 253 in website/src/reducers/timetables.ts

View check run for this annotation

Codecov / codecov/patch

website/src/reducers/timetables.ts#L252-L253

Added lines #L252 - L253 were not covered by tests
...state,
colors: recolor(state.colors, action.payload.numOfColors),
};

case SET_HIDDEN_IMPORTED: {
const { semester, hiddenModules } = action.payload;
Expand Down
12 changes: 12 additions & 0 deletions website/src/styles/constants.scss
Original file line number Diff line number Diff line change
Expand Up @@ -145,6 +145,18 @@ $nusmods-theme-colors: (
#b6b3eb,
#bc9458
),
tequila: (
#e7553e,
#ff9f1e,
#ffc7c7,
#f1baa1,
#ffd700,
#8febdb,
#a9daf5,
#80b9f3,
#b9848c,
#b249f2,
),
tomorrow: (
#c66,
#de935f,
Expand Down
1 change: 1 addition & 0 deletions website/src/types/reducers.ts
Original file line number Diff line number Diff line change
Expand Up @@ -79,6 +79,7 @@ export const HORIZONTAL: TimetableOrientation = 'HORIZONTAL';

export type ThemeState = Readonly<{
id: string;
numOfColors: number;
timetableOrientation: TimetableOrientation;
showTitle: boolean;
}>;
Expand Down
1 change: 1 addition & 0 deletions website/src/types/settings.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ export type ThemeId = string;
export type Theme = {
readonly id: ThemeId;
readonly name: string;
readonly numOfColors: number;
};

/**
Expand Down
Loading