Skip to content

Commit

Permalink
Feature/ntgl 404 persisting prev data (#792)
Browse files Browse the repository at this point in the history
* Changed the default storage object to include a timetable array and added a useState for selected Timetable index

* Create handlers for changing and deleting timetables

* fix: updateTimetableEvents now fetches current timetable properly

* Fixed main idea for deleting timetables. Created timetables component to automatically rerender

* Fixed multiple timetables functionality

Fixed multiple timetables functionality + fixed issue with not allowing user to select classes

* refactor(client): NTGL-384 Move logic of timetable tabs to separate file

* feat(client): NTGL-384 Initial creation of styled-components for timetable tabs bar

* fix(client): NTGL-384 Change add timetable click handler to work on entire tab instead of just the icon

* style(client): NTGL-384 Add shadow and border styling for tabs

* Added timetable renaming functionality

* feat(client): NTGL-384 Add delete modal to menu

* fix(client): NTGL-384 Fix bug where displayTimetables[selectedTimetable].name would throw an error

* Fixed deleting timetables functionality

* Fixed issue with timetable data being passed to other timetables

* Fixed bugs with deleting and adding new timetables

* Added error checking to timetable rename and changed style

* Added brief documentation

* Changed timetable rename dialog style

* Added tab duplication, themes, and improved speed of switching between tabs

* feat(client): NTGL-384 Add alert for adding or copying a timetable when at timetable limit

* feat(client): NTGL-384 Add right click tab handler to switch tabs and bring up menu

* Added hotkeys to quick delete timetables

* Fixed bugs with enter keybind

* Added hotkey shortcut for creating new timetables

* Fixed duplicate timetables moving both events

* Fixed deep clone issue with duplicate timetables

* Added drag and drop to sort timetable tabs + fixed some styling with tabs

* Added hotkey to allow pressing enter to confirm timetable rename

* Forgot to push the timetabledata interface

* Added tool tips for adding and deleting timetable tabs

* fixed scrolling on timetable tabs, add timetable button sticks to scroll bar for ease of access

* fixed animation issue with custom events and switching between duplicate timetables

* feat(client): NTGL-384 Add core history functionality to support multiple timetables (WIP)

* feat(client): NTGL-384 Add modal supporting clearing current and all timetables, improvements to history logic

* style(client): NTGL-384 connect tab nav bar to timetable

* made timetable tabs similar to google tabs interface (changed border radius)

* Minor changes to tab UI and tab clearing feature

* Attempt at fixing history.ts for multiple timetables

* feat(client): NTGL-384 Add undo alert after deleting timetable

* feat(client): NTGL-384 Change history.ts to use hashmap/object instead of 2D array

* NTGL-384 Generate timetableId function checks for collisions with displayTimetables, duplicated timetables now given ids

* NTGL 384 Undo/redo works with timetable renaming

* fix(client): NTGL-384 Rearranging timetable will successfully trigger save to local storage

* Changed new tab button look

* fix(client): NTGL-384 resetAll resets timetables to default

* feat(client): NTGL-384 Re-wrote defaultReset logic to disable buttons on modal, and handle clearOne/clearAll seperately

* style/refactor(client): Cleaned up code, move tab styling to a separate file

* style(client): Minor style changes to history.ts

* Moved timetable tab popups to separate file + tidied up some of the logic for timetable tabs

* Fixed styling of rename timetable popup modal

* Updated the delete timetable popup modal

* Updated the style of clear timetables modal

* Ran prettier on files

* Context menu for rename, duplicate, delete is consistent with events. Got rid of inline styling for timetableTabs. Still yet to get rid of inline styling in the EditTabPopups. Fixed bug from context menu, now changes color when changing from light to dark mode

* Changed EditTabPopups component to TimetableTabContextMenu for consistency with events

* Tidied up rename timetable dialog

* Refactored the repeated code for timetable states

* made dialog for delete timetable consistent with discard changes for events

* Made dialog for clear timetable(s) consistent with  discard changes to event info

* Cleaned up code and fixed tooltips and keyboard shortcuts for context menu

* Added snackbar alert message for restoring deleted timetable - need to fix up the colour scheme for this

* Fixed colour styling for restore timetable snackbar alert

* Minor changes to restoreTimetable action component to be consistent with dialog messages

* NTGL-384 refactor(client): Move/create helper functions related to creating timetables/history, simplify logic in history.ts

* NTGL-384 refactor(client): minor changes to TimetableTab (move event duplicate function to utils) and History (fix logic)

* NTGL-384 bugfix(client): fixed bug when history.ts creates a new timetableActions array for timetable, and added back disabling reset button functionality

* Added styling to delete timetable button, fixed behaviour and bugs with undo/redo buttons

* Edited padding of ExpandedClassView, ExpandedEventView and Rename timetable dialog

* fix(client): NTGL-384 timetable time bounds properly reset upon timetable switching

* fix(client): NTGL-384 timetable tab and course select wrapper aligned with timetable layout

* fix(client): NTGL-384 close the edit timetable menu when they click out of edit timetable name

* fix(client): NTGL-384 Location no longer a necessary field for CustomEvent creation/editing

* Created CourseData object in defaults.ts

* changed courseData object structure

* feat: NTGL-402 Added color picker menu to ExpandedClassView, users can select colors for courses

* added subtype property

* disabled dragging of tutorial and laboratory classes

* disabled edit on tutoring class cards

* disabled edit icon on tutoring class card

* changed context menu of tutoring card to only have a delete option

* increasing timeout value for recieving course info

* Added basic layout for switching between terms

* Added basic implementation for storage of term data for switching terms

* feat: colors for courses are independent for each timetable. Color choices are stored locally.

* feat: duplicating timetables does not break custom colors

* feat: History stores the assigned colors when reversing deletion of a course

* resolved conflicts

* disabled event link in tutoring and lab events

* testing: add debugging code to App.tsx

* fixed usage of createEventObj in timetableHelpers.ts

* removed edit icon tutoring event

* fixed styling

* fixed syntax errors from merge

* Added todos for jeremy, brainstormed/drafted mock behaviour for switching terms

* fix: bug where initially loaded timetable resets to default assigned colors FIXED!

* fix: minor changes to history to fix unwanted behaviour

* feat: add save button to expanded class view

* minor fix: disable save button functionality fix

* able to change colors on tutorial card

* bugfix: History branching bug fixed - reference to newly branched timetable is properly made

* fix: older stored timetables will not break upon updating to assignedColors

* created a new TutorialColorPicker file, that is the exact same as ColorPicker except with a Save button

* Fixed errors with handling terms n history.ts

* changed props of ColorPick to accept Save function and display save button only for tutorial events

* fix: increased timeout value for fetching course info + refactored fetchTermData to fix issue with restoring previously saved data

* styled colour picker in class cards to be consistent with tutoring classes

* changed prop name of Color Picker from handleSaveNewTutorialColor to handleSaveNewColor

* Fixed behaviour of preserving prev calendar data

* Border outline of select term stays white and aligned, padded component

* forcing tabs to only be dragged horizontally

* chore: remove CTF artifacts

* style: move inline styling of divider and list item into styled components

* fix: add subtype to createLinkEvent to silence errors

* refactoring and cleaning up code for termSelect

* re-ordered terms in term select

* removed comment and fixing linting errors

* fixed undefined errors with rendering data on first load

* testing fix to populate timetable data when moving to multiple terms version

* debugged issues with migration to multiple terms version

* removed unused imports and variables

* NTGL-402 Fix minor styling issues

* Fix other issues

* style(event share, period): added new line and removed unused styled component

* bugfix - colors retained on delete. Also removed code to prevent timetables on assigned color migration

* Remove code that supported migration to assignedcolors

* resolving comments, improved typing and removed unnecessary data

* bugfix: loading timetable colours when switching terms

* updated calls to scraper with generic path for fetching previous term data

* fix: FIXED OUTLINE DISAPPEARING WHEN FOCUSED

* removed redundant styles

* updated changelog

---------

Co-authored-by: eklavya-joshi <[email protected]>
Co-authored-by: Michael Siu <[email protected]>
Co-authored-by: manhualu <[email protected]>
Co-authored-by: Jasmine Tran <[email protected]>
Co-authored-by: Jeremy Le <[email protected]>
  • Loading branch information
6 people authored May 14, 2024
1 parent 4078b21 commit b974aff
Show file tree
Hide file tree
Showing 14 changed files with 407 additions and 161 deletions.
62 changes: 45 additions & 17 deletions client/src/App.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -34,13 +34,14 @@ import {
ClassData,
CourseCode,
CourseData,
DisplayTimetablesMap,
InInventory,
SelectedClasses,
TimetableData,
} from './interfaces/Periods';
import { setDropzoneRange, useDrag } from './utils/Drag';
import { downloadIcsFile } from './utils/generateICS';
import storage from './utils/storage';
import { createDefaultTimetable } from './utils/timetableHelpers';

const StyledApp = styled(Box)`
height: 100%;
Expand Down Expand Up @@ -109,6 +110,7 @@ const App: React.FC = () => {
firstDayOfTerm,
setFirstDayOfTerm,
setTermName,
setTermsData,
setTermNumber,
setCoursesList,
setLastUpdated,
Expand Down Expand Up @@ -163,13 +165,31 @@ const App: React.FC = () => {
* Retrieves term data from the scraper backend
*/
const fetchTermData = async () => {
const termData = await getAvailableTermDetails();
const { term, termName, termNumber, year, firstDayOfTerm } = termData;
const { term, termName, termNumber, year, firstDayOfTerm, termsData } = await getAvailableTermDetails();
setTerm(term);
setTermName(termName);
setTermNumber(termNumber);
setYear(year);
setFirstDayOfTerm(firstDayOfTerm);
setTermsData(termsData);
const oldData = storage.get('timetables');

// avoid overwriting data from previous save
const newTimetableTerms: DisplayTimetablesMap = {
...{
[termsData.prevTerm.term]: oldData.hasOwnProperty(termsData.prevTerm.term)
? oldData[termsData.prevTerm.term]
: createDefaultTimetable(),
},
...{
[termsData.newTerm.term]: oldData.hasOwnProperty(termsData.newTerm.term)
? oldData[termsData.newTerm.term]
: createDefaultTimetable(),
},
};

setDisplayTimetables(newTimetableTerms);
storage.set('timetables', newTimetableTerms);
};

fetchReliably(fetchTermData);
Expand All @@ -186,11 +206,11 @@ const App: React.FC = () => {
};

if (year !== invalidYearFormat) fetchReliably(fetchCoursesList);
}, [year]);
}, [term, year]);

// Fetching the saved timetables from local storage
useEffect(() => {
const savedTimetables: TimetableData[] = storage.get('timetables');
const savedTimetables: DisplayTimetablesMap = storage.get('timetables');
if (savedTimetables) {
setDisplayTimetables(savedTimetables);
}
Expand Down Expand Up @@ -280,7 +300,7 @@ const App: React.FC = () => {
const newSelectedCourses = [...selectedCourses];
const newCourseData = courseData;

// Update the existing courses with the new data (for changing timezones).
// Update the existing courses with the new data (for changing timezone).
addedCourses.forEach((addedCourse) => {
if (newSelectedCourses.find((x) => x.code === addedCourse.code)) {
const index = newSelectedCourses.findIndex((x) => x.code === addedCourse.code);
Expand All @@ -295,11 +315,10 @@ const App: React.FC = () => {
newCourseData.map.push(addedCourse);
}
});

setSelectedCourses(newSelectedCourses);
setCourseData(newCourseData);

if (displayTimetables.length > 0) {
if (displayTimetables[term].length > 0) {
setAssignedColors(
useColorMapper(
newSelectedCourses.map((course) => course.code),
Expand All @@ -323,7 +342,7 @@ const App: React.FC = () => {
setSelectedCourses(newSelectedCourses);
const newCourseData = courseData;
newCourseData.map = courseData.map.filter(() => {
for (const timetable of displayTimetables) {
for (const timetable of displayTimetables[term]) {
for (const course of timetable.selectedCourses) {
if (course.code.localeCompare(courseCode)) {
return true;
Expand All @@ -348,11 +367,20 @@ const App: React.FC = () => {
* Populate selected courses, classes and created events with the data saved in local storage
*/
const updateTimetableEvents = () => {
if (!storage.get('timetables')[term]) {
// data stored in local storage not up to date with current term
const updatedWithTerms = { [term]: storage.get('timetables') };

storage.set('timetables', updatedWithTerms);
setDisplayTimetables(updatedWithTerms);
}

handleSelectCourse(
storage.get('timetables')[selectedTimetable].selectedCourses.map((course: CourseData) => course.code),
storage.get('timetables')[term][selectedTimetable].selectedCourses.map((course: CourseData) => course.code),
true,
(newSelectedCourses) => {
const timetableSelectedClasses: SelectedClasses = storage.get('timetables')[selectedTimetable].selectedClasses;
const timetableSelectedClasses: SelectedClasses =
storage.get('timetables')[term][selectedTimetable].selectedClasses;

const savedClasses: SavedClasses = {};

Expand Down Expand Up @@ -391,8 +419,8 @@ const App: React.FC = () => {
setSelectedClasses(newSelectedClasses);
},
);
setCreatedEvents(storage.get('timetables')[selectedTimetable].createdEvents);
setAssignedColors(storage.get('timetables')[selectedTimetable].assignedColors);
setCreatedEvents(storage.get('timetables')[term][selectedTimetable].createdEvents);
setAssignedColors(storage.get('timetables')[term][selectedTimetable].assignedColors);
};

useEffect(() => {
Expand All @@ -401,27 +429,27 @@ const App: React.FC = () => {

// The following three useUpdateEffects update local storage whenever a change is made to the timetable
useUpdateEffect(() => {
displayTimetables[selectedTimetable].selectedCourses = selectedCourses;
displayTimetables[term][selectedTimetable].selectedCourses = selectedCourses;
const newCourseData = courseData;
storage.set('courseData', newCourseData);
storage.set('timetables', displayTimetables);
setDisplayTimetables(displayTimetables);
}, [selectedCourses]);

useUpdateEffect(() => {
displayTimetables[selectedTimetable].selectedClasses = selectedClasses;
displayTimetables[term][selectedTimetable].selectedClasses = selectedClasses;
storage.set('timetables', displayTimetables);
setDisplayTimetables(displayTimetables);
}, [selectedClasses]);

useUpdateEffect(() => {
displayTimetables[selectedTimetable].createdEvents = createdEvents;
displayTimetables[term][selectedTimetable].createdEvents = createdEvents;
storage.set('timetables', displayTimetables);
setDisplayTimetables(displayTimetables);
}, [createdEvents]);

useUpdateEffect(() => {
displayTimetables[selectedTimetable].assignedColors = assignedColors;
displayTimetables[term][selectedTimetable].assignedColors = assignedColors;
storage.set('timetables', displayTimetables);
setDisplayTimetables(displayTimetables);
}, [assignedColors]);
Expand Down
7 changes: 4 additions & 3 deletions client/src/api/getCourseInfo.ts
Original file line number Diff line number Diff line change
Expand Up @@ -85,16 +85,17 @@ const getCourseInfo = async (
isConvertToLocalTimezone: boolean,
): Promise<CourseData> => {
const baseURL = `${API_URL.timetable}/terms/${year}-${term}`;
const COURSE_API_TIMEOUT = 2000;
try {
const data = await timeoutPromise(1000, fetch(`${baseURL}/courses/${courseCode}/`));
const data = await timeoutPromise(COURSE_API_TIMEOUT, fetch(`${baseURL}/courses/${courseCode}/`));

// Remove any leftover courses from localStorage if they are not offered in the current term
// which is why a 400 error is returned
if (data.status === 400) {
const selectedCourses = storage.get('timetables')[0].selectedCourses;
const selectedCourses = storage.get('timetables')[term][0].selectedCourses;
if (selectedCourses.includes(courseCode)) {
delete selectedCourses[courseCode];
storage.set('timetables[0].selectedCourses', selectedCourses);
storage.set(`timetables[${term}][0].selectedCourses`, selectedCourses);
} else {
throw new NetworkError('Internal server error');
}
Expand Down
15 changes: 6 additions & 9 deletions client/src/components/EventShareModal.tsx
Original file line number Diff line number Diff line change
@@ -1,16 +1,17 @@
import { useState, useContext, useEffect } from 'react';
import { useNavigate, useParams } from 'react-router-dom';
import { Add, Close, LocationOn } from '@mui/icons-material';
import { Add, Close } from '@mui/icons-material';
import { Card, CardProps, Dialog, IconButton } from '@mui/material';
import { styled } from '@mui/system';
import isBase64 from 'is-base64';
import { useContext, useEffect, useState } from 'react';
import { useNavigate, useParams } from 'react-router-dom';

import { AppContext } from '../context/AppContext';
import { CourseContext } from '../context/CourseContext';
import { StyledDialogTitle, StyledTitleContainer, StyledTopIcons } from '../styles/ControlStyles';
import { ExecuteButton, StyledListItemText, StyledLocationIcon } from '../styles/CustomEventStyles';
import { StyledCardName } from '../styles/DroppedCardStyles';
import { createEventObj } from '../utils/createEvent';
import { resizeWeekArray } from '../utils/eventTimes';
import isBase64 from 'is-base64';

const PreviewCard = styled(Card)<CardProps & { bgColour: string }>`
padding: 24px 16px;
Expand All @@ -26,12 +27,8 @@ const PreviewCard = styled(Card)<CardProps & { bgColour: string }>`
color: white;
`;

const StyledLocationOn = styled(LocationOn)`
font-size: 12px;
`;

const EventShareModal = () => {
let { encrypted } = useParams();
const { encrypted } = useParams();
const navigate = useNavigate();

const [open, setOpen] = useState(true);
Expand Down
67 changes: 43 additions & 24 deletions client/src/components/controls/History.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ import React, { useContext, useEffect, useRef, useState } from 'react';

import { AppContext } from '../../context/AppContext';
import { CourseContext } from '../../context/CourseContext';
import { CourseData, CreatedEvents, SelectedClasses, TimetableData } from '../../interfaces/Periods';
import { CourseData, CreatedEvents, DisplayTimetablesMap, SelectedClasses } from '../../interfaces/Periods';
import { StyledDialogButtons, StyledDialogContent, StyledTitleContainer } from '../../styles/ControlStyles';
import {
ActionsPointer,
Expand All @@ -28,7 +28,7 @@ const History: React.FC = () => {

const { selectedCourses, setSelectedCourses, selectedClasses, setSelectedClasses, createdEvents, setCreatedEvents } =
useContext(CourseContext);
const { isDrag, setIsDrag, selectedTimetable, setSelectedTimetable, displayTimetables, setDisplayTimetables } =
const { isDrag, setIsDrag, selectedTimetable, setSelectedTimetable, displayTimetables, setDisplayTimetables, term } =
useContext(AppContext);

const timetableActions = useRef<TimetableActions>({});
Expand All @@ -41,8 +41,8 @@ const History: React.FC = () => {
courses: CourseData[],
classes: SelectedClasses,
events: CreatedEvents,
timetableArg: TimetableData[] | ((prev: TimetableData[]) => void),
selected?: number
timetableArg: DisplayTimetablesMap,
selected?: number,
) => {
setSelectedCourses(courses);
setSelectedClasses(classes);
Expand All @@ -59,7 +59,7 @@ const History: React.FC = () => {
* @param direction Which way to update (1 for increment, -1 for decrement)
*/
const incrementActionsPointer = (direction: number) => {
const timetableId = displayTimetables[selectedTimetable].id;
const timetableId = displayTimetables[term][selectedTimetable].id;
actionsPointer.current[timetableId] += direction;
setDisableLeft(actionsPointer.current[timetableId] < 1);
setDisableRight(actionsPointer.current[timetableId] + 1 >= timetableActions.current[timetableId].length);
Expand All @@ -75,11 +75,16 @@ const History: React.FC = () => {
return; // Prevents adding change induced by clicking redo/undo
}

if (selectedTimetable >= displayTimetables.length) {
if (selectedTimetable >= displayTimetables[term]?.length) {
return;
}

const currentTimetable = displayTimetables[selectedTimetable];
// waiting for timetable data to render
if (!(term in displayTimetables)) {
return;
}

const currentTimetable = displayTimetables[term][selectedTimetable];

// Create object if it doesn't exist
if (!timetableActions.current[currentTimetable.id]) {
Expand All @@ -99,16 +104,16 @@ const History: React.FC = () => {
}

timetableActions.current[currentTimetable.id].push({
name: displayTimetables[selectedTimetable].name,
name: displayTimetables[term][selectedTimetable].name,
courses: [...selectedCourses],
classes: duplicateClasses(selectedClasses),
events: { ...createdEvents }
events: { ...createdEvents },
});

incrementActionsPointer(1);
}, [selectedClasses, isDrag, createdEvents, displayTimetables]);

//Disables reset timetable button when there is no courses, classes and events selected.
// Disables reset timetable button when there is no courses, classes and events selected.
useEffect(() => {
if (!isMounted.current) {
isMounted.current = true;
Expand All @@ -118,12 +123,12 @@ const History: React.FC = () => {
const disableStatus = { current: true, all: true };

// More than one timetable is resetAll-able
if (displayTimetables.length > 1) {
if (displayTimetables[term].length > 1) {
disableStatus.all = false;
}

// Current timetable being non-empty is resetAll and resetOne-able
const currentTimetable = displayTimetables[selectedTimetable];
const currentTimetable = displayTimetables[term][selectedTimetable];
// if new timetable has been created then set reset to be true since no courses, classes or events selected
if (actionsPointer.current[currentTimetable.id] < 1) {
disableStatus.current = true;
Expand All @@ -149,11 +154,12 @@ const History: React.FC = () => {
}, [selectedTimetable, selectedCourses, selectedClasses, createdEvents, displayTimetables]);

useEffect(() => {
if (displayTimetables.length < 1) {
// waiting for timetable data to render with valid term data
if (!(term in displayTimetables) || displayTimetables[term].length < 1) {
return;
}

const timetableId = displayTimetables[selectedTimetable].id;
const timetableId = displayTimetables[term][selectedTimetable].id;
const currentPointer = actionsPointer.current[timetableId];
setDisableLeft(currentPointer === undefined || currentPointer < 1);
setDisableRight(currentPointer === undefined || currentPointer + 1 >= timetableActions.current[timetableId].length);
Expand All @@ -167,24 +173,37 @@ const History: React.FC = () => {
incrementActionsPointer(direction);
dontAdd.current = true;

const timetableId = displayTimetables[selectedTimetable].id;
const modifyTimetableName = (prev: TimetableData[]) => {
return prev.map((timetable) => {
return timetable.id === timetableId
? { ...timetable, name: timetableActions.current[timetableId][actionsPointer.current[timetableId]].name }
const modifiedTimetableId = displayTimetables[term][selectedTimetable].id;
const modifiedTimetableName = {
...displayTimetables,
[term]: displayTimetables[term].map((timetable) => {
return timetable.id === modifiedTimetableId
? {
...timetable,
name: timetableActions.current[modifiedTimetableId][actionsPointer.current[modifiedTimetableId]].name,
}
: timetable;
});
}),
};

const { courses, classes, events } = extractHistoryInfo(timetableId, timetableActions.current, actionsPointer.current);
setTimetableState(courses, classes, events, modifyTimetableName);
const { courses, classes, events } = extractHistoryInfo(
modifiedTimetableId,
timetableActions.current,
actionsPointer.current,
);

setTimetableState(courses, classes, events, modifiedTimetableName);
};

/**
* Resets all timetables - leave one as default
*/
const clearAll = () => {
setTimetableState([], {}, {}, createDefaultTimetable(), 0);
const newDisplayTimetables: DisplayTimetablesMap = {
...displayTimetables,
[term]: createDefaultTimetable(),
};
setTimetableState([], {}, {}, newDisplayTimetables, 0);
};

/**
Expand All @@ -195,7 +214,7 @@ const History: React.FC = () => {
// event.metaKey corresponds to the Cmd key on Mac
if (!(event.ctrlKey || event.metaKey) || !(event.key === 'z' || event.key === 'y' || event.key === 'd')) return;

const currentTimetable = displayTimetables[selectedTimetable];
const currentTimetable = displayTimetables[term][selectedTimetable];

event.preventDefault();

Expand Down
4 changes: 4 additions & 0 deletions client/src/components/navbar/Changelog.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,10 @@ import React from 'react';
type Change = { date: string; changes: string[] };

const changelog: Change[] = [
{
date: 'May 14, 2024',
changes: ['Users can access data across multiple terms'],
},
{
date: 'February 16, 2024',
changes: ['Footer redesign with icons and sponsor logos'],
Expand Down
Loading

0 comments on commit b974aff

Please sign in to comment.