From 38b5162746c17b8e21d54e7a53a06441101c167f Mon Sep 17 00:00:00 2001 From: Danny Rorabaugh Date: Thu, 26 Oct 2023 15:11:10 -0400 Subject: [PATCH 1/4] Try cobertura --- .github/workflows/frontend.yml | 4 ++-- package.json | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/.github/workflows/frontend.yml b/.github/workflows/frontend.yml index 0df946a33a..b6be810a31 100644 --- a/.github/workflows/frontend.yml +++ b/.github/workflows/frontend.yml @@ -72,7 +72,7 @@ jobs: with: if-no-files-found: error name: coverage - path: coverage/clover.xml + path: coverage/cobertura-coverage.xml retention-days: 7 upload_coverage: @@ -102,7 +102,7 @@ jobs: uses: codecov/codecov-action@eaaf4bedf32dbdc6b720b63067d99c4d77d6047d # v3.1.4 with: fail_ci_if_error: true - files: clover.xml + files: cobertura-coverage.xml flags: frontend name: Frontend diff --git a/package.json b/package.json index 7608c59fb4..fd6cb25f7e 100644 --- a/package.json +++ b/package.json @@ -31,7 +31,7 @@ "test-backend": " dotnet test Backend.Tests/Backend.Tests.csproj", "test-backend:coverage": "dotnet test Backend.Tests/Backend.Tests.csproj /p:CollectCoverage=true /p:CoverletOutputFormat=lcov /p:Threshold=77", "test-frontend": " react-scripts test", - "test-frontend:coverage": "react-scripts test --coverage --watchAll=false", + "test-frontend:coverage": "react-scripts test --coverage --coverageReporters=cobertura --watchAll=false", "test-frontend:debug": " react-scripts --inspect-brk test --runInBand --no-cache", "test:ci": "dotnet test Backend.Tests/Backend.Tests.csproj && CI=true react-scripts test --ci --all --testResultsProcessor jest-teamcity-reporter", "wordlist": "hunspell-reader words" From ad96cccf4acb670d0eab96af7df37011f51c658c Mon Sep 17 00:00:00 2001 From: Danny Rorabaugh Date: Thu, 26 Oct 2023 15:22:49 -0400 Subject: [PATCH 2/4] Revert "Try cobertura" This reverts commit 38b5162746c17b8e21d54e7a53a06441101c167f. --- .github/workflows/frontend.yml | 4 ++-- package.json | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/.github/workflows/frontend.yml b/.github/workflows/frontend.yml index b6be810a31..0df946a33a 100644 --- a/.github/workflows/frontend.yml +++ b/.github/workflows/frontend.yml @@ -72,7 +72,7 @@ jobs: with: if-no-files-found: error name: coverage - path: coverage/cobertura-coverage.xml + path: coverage/clover.xml retention-days: 7 upload_coverage: @@ -102,7 +102,7 @@ jobs: uses: codecov/codecov-action@eaaf4bedf32dbdc6b720b63067d99c4d77d6047d # v3.1.4 with: fail_ci_if_error: true - files: cobertura-coverage.xml + files: clover.xml flags: frontend name: Frontend diff --git a/package.json b/package.json index fd6cb25f7e..7608c59fb4 100644 --- a/package.json +++ b/package.json @@ -31,7 +31,7 @@ "test-backend": " dotnet test Backend.Tests/Backend.Tests.csproj", "test-backend:coverage": "dotnet test Backend.Tests/Backend.Tests.csproj /p:CollectCoverage=true /p:CoverletOutputFormat=lcov /p:Threshold=77", "test-frontend": " react-scripts test", - "test-frontend:coverage": "react-scripts test --coverage --coverageReporters=cobertura --watchAll=false", + "test-frontend:coverage": "react-scripts test --coverage --watchAll=false", "test-frontend:debug": " react-scripts --inspect-brk test --runInBand --no-cache", "test:ci": "dotnet test Backend.Tests/Backend.Tests.csproj && CI=true react-scripts test --ci --all --testResultsProcessor jest-teamcity-reporter", "wordlist": "hunspell-reader words" From 42b599c9ff6be3e174c0b0b7e21841f281120122 Mon Sep 17 00:00:00 2001 From: "D. Ror" Date: Mon, 30 Oct 2023 11:42:19 -0400 Subject: [PATCH 3/4] Port ExportProject to use redux-toolkit; Remove redux from CreateProject (#2747) --- src/components/App/DefaultState.ts | 5 - .../Redux/ExportProjectActions.ts | 63 +++++------ .../Redux/ExportProjectReducer.ts | 62 ++++++----- .../Redux/ExportProjectReduxTypes.ts | 5 - .../Redux/tests/ExportProjectActions.test.tsx | 85 +++++++++++++++ .../index.tsx => CreateProject.tsx} | 30 +++--- .../Redux/CreateProjectActions.ts | 100 ------------------ .../Redux/CreateProjectReducer.ts | 27 ----- .../Redux/CreateProjectReduxTypes.ts | 29 ----- .../tests/CreateProjectActions.test.tsx | 48 --------- .../tests/CreateProjectReducer.test.tsx | 47 -------- .../ProjectScreen/CreateProjectActions.ts | 54 ++++++++++ .../CreateProject.test.tsx} | 6 +- src/rootReducer.ts | 4 +- src/types/index.ts | 2 - 15 files changed, 225 insertions(+), 342 deletions(-) create mode 100644 src/components/ProjectExport/Redux/tests/ExportProjectActions.test.tsx rename src/components/ProjectScreen/{CreateProject/index.tsx => CreateProject.tsx} (94%) delete mode 100644 src/components/ProjectScreen/CreateProject/Redux/CreateProjectActions.ts delete mode 100644 src/components/ProjectScreen/CreateProject/Redux/CreateProjectReducer.ts delete mode 100644 src/components/ProjectScreen/CreateProject/Redux/CreateProjectReduxTypes.ts delete mode 100644 src/components/ProjectScreen/CreateProject/tests/CreateProjectActions.test.tsx delete mode 100644 src/components/ProjectScreen/CreateProject/tests/CreateProjectReducer.test.tsx create mode 100644 src/components/ProjectScreen/CreateProjectActions.ts rename src/components/ProjectScreen/{CreateProject/tests/CreateProjectComponent.test.tsx => tests/CreateProject.test.tsx} (95%) diff --git a/src/components/App/DefaultState.ts b/src/components/App/DefaultState.ts index 1290da98da..1294ddf45d 100644 --- a/src/components/App/DefaultState.ts +++ b/src/components/App/DefaultState.ts @@ -2,7 +2,6 @@ import { defaultState as goalTimelineState } from "components/GoalTimeline/Defau import { defaultState as loginState } from "components/Login/Redux/LoginReducer"; import { defaultState as currentProjectState } from "components/Project/ProjectReduxTypes"; import { defaultState as exportProjectState } from "components/ProjectExport/Redux/ExportProjectReduxTypes"; -import { defaultState as createProjectState } from "components/ProjectScreen/CreateProject/Redux/CreateProjectReduxTypes"; import { defaultState as pronunciationsState } from "components/Pronunciations/Redux/PronunciationsReduxTypes"; import { defaultState as treeViewState } from "components/TreeView/Redux/TreeViewReduxTypes"; import { defaultState as characterInventoryState } from "goals/CharacterInventory/Redux/CharacterInventoryReducer"; @@ -15,10 +14,6 @@ export const defaultState = { loginState: { ...loginState }, //project - createProjectState: { - ...createProjectState, - success: true, - }, currentProjectState: { ...currentProjectState }, exportProjectState: { ...exportProjectState }, diff --git a/src/components/ProjectExport/Redux/ExportProjectActions.ts b/src/components/ProjectExport/Redux/ExportProjectActions.ts index cfea4ba26a..8bd8488ff1 100644 --- a/src/components/ProjectExport/Redux/ExportProjectActions.ts +++ b/src/components/ProjectExport/Redux/ExportProjectActions.ts @@ -1,10 +1,39 @@ +import { Action, PayloadAction } from "@reduxjs/toolkit"; + import { deleteLift, downloadLift, exportLift } from "backend"; import { - ExportProjectAction, - ExportStatus, -} from "components/ProjectExport/Redux/ExportProjectReduxTypes"; + downloadingAction, + exportingAction, + failureAction, + resetAction, + successAction, +} from "components/ProjectExport/Redux//ExportProjectReducer"; import { StoreStateDispatch } from "types/Redux/actions"; +// Action Creation Functions + +export function exporting(projectId: string): PayloadAction { + return exportingAction(projectId); +} + +export function downloading(projectId: string): PayloadAction { + return downloadingAction(projectId); +} + +export function failure(projectId: string): PayloadAction { + return failureAction(projectId); +} + +export function reset(): Action { + return resetAction(); +} + +export function success(projectId: string): PayloadAction { + return successAction(projectId); +} + +// Dispatch Functions + export function asyncExportProject(projectId: string) { return async (dispatch: StoreStateDispatch) => { dispatch(exporting(projectId)); @@ -27,31 +56,3 @@ export function asyncResetExport() { await deleteLift(); }; } - -function exporting(projectId: string): ExportProjectAction { - return { - type: ExportStatus.Exporting, - projectId, - }; -} -function downloading(projectId: string): ExportProjectAction { - return { - type: ExportStatus.Downloading, - projectId, - }; -} -export function success(projectId: string): ExportProjectAction { - return { - type: ExportStatus.Success, - projectId, - }; -} -export function failure(projectId: string): ExportProjectAction { - return { - type: ExportStatus.Failure, - projectId, - }; -} -function reset(): ExportProjectAction { - return { type: ExportStatus.Default }; -} diff --git a/src/components/ProjectExport/Redux/ExportProjectReducer.ts b/src/components/ProjectExport/Redux/ExportProjectReducer.ts index 6306fe529a..d830a50508 100644 --- a/src/components/ProjectExport/Redux/ExportProjectReducer.ts +++ b/src/components/ProjectExport/Redux/ExportProjectReducer.ts @@ -1,29 +1,43 @@ +import { createSlice } from "@reduxjs/toolkit"; + import { defaultState, - ExportProjectAction, - ExportProjectState, ExportStatus, } from "components/ProjectExport/Redux/ExportProjectReduxTypes"; -import { StoreAction, StoreActionTypes } from "rootActions"; +import { StoreActionTypes } from "rootActions"; + +const exportProjectSlice = createSlice({ + name: "exportProjectState", + initialState: defaultState, + reducers: { + downloadingAction: (state, action) => { + state.projectId = action.payload; + state.status = ExportStatus.Downloading; + }, + exportingAction: (state, action) => { + state.projectId = action.payload; + state.status = ExportStatus.Exporting; + }, + failureAction: (state, action) => { + state.projectId = action.payload; + state.status = ExportStatus.Failure; + }, + resetAction: () => defaultState, + successAction: (state, action) => { + state.projectId = action.payload; + state.status = ExportStatus.Success; + }, + }, + extraReducers: (builder) => + builder.addCase(StoreActionTypes.RESET, () => defaultState), +}); + +export const { + downloadingAction, + exportingAction, + failureAction, + resetAction, + successAction, +} = exportProjectSlice.actions; -export const exportProjectReducer = ( - state: ExportProjectState = defaultState, - action: StoreAction | ExportProjectAction -): ExportProjectState => { - switch (action.type) { - case ExportStatus.Exporting: - case ExportStatus.Downloading: - case ExportStatus.Success: - case ExportStatus.Failure: - return { - ...defaultState, - projectId: action.projectId ?? "", - status: action.type, - }; - case ExportStatus.Default: - case StoreActionTypes.RESET: - return defaultState; - default: - return state; - } -}; +export default exportProjectSlice.reducer; diff --git a/src/components/ProjectExport/Redux/ExportProjectReduxTypes.ts b/src/components/ProjectExport/Redux/ExportProjectReduxTypes.ts index 6b910ca38a..3725b60e9d 100644 --- a/src/components/ProjectExport/Redux/ExportProjectReduxTypes.ts +++ b/src/components/ProjectExport/Redux/ExportProjectReduxTypes.ts @@ -6,11 +6,6 @@ export enum ExportStatus { Failure = "FAILURE", } -export interface ExportProjectAction { - type: ExportStatus; - projectId?: string; -} - export interface ExportProjectState { projectId: string; status: ExportStatus; diff --git a/src/components/ProjectExport/Redux/tests/ExportProjectActions.test.tsx b/src/components/ProjectExport/Redux/tests/ExportProjectActions.test.tsx new file mode 100644 index 0000000000..5ff440a77a --- /dev/null +++ b/src/components/ProjectExport/Redux/tests/ExportProjectActions.test.tsx @@ -0,0 +1,85 @@ +import { PreloadedState } from "redux"; + +import { defaultState } from "components/App/DefaultState"; +import { + asyncDownloadExport, + asyncExportProject, + asyncResetExport, +} from "components/ProjectExport/Redux/ExportProjectActions"; +import { ExportStatus } from "components/ProjectExport/Redux/ExportProjectReduxTypes"; +import { RootState, setupStore } from "store"; + +jest.mock("backend", () => ({ + deleteLift: jest.fn, + downloadLift: (...args: any[]) => mockDownloadList(...args), + exportLift: (...args: any[]) => mockExportLift(...args), +})); + +const mockDownloadList = jest.fn(); +const mockExportLift = jest.fn(); +const mockProjId = "project-id"; + +// Preloaded values for store when testing +const persistedDefaultState: PreloadedState = { + ...defaultState, + _persist: { version: 1, rehydrated: false }, +}; + +describe("ExportProjectActions", () => { + describe("asyncDownloadExport", () => { + it("correctly affects state on success", async () => { + const store = setupStore(); + mockDownloadList.mockResolvedValueOnce({}); + await store.dispatch(asyncDownloadExport(mockProjId)); + const { projectId, status } = store.getState().exportProjectState; + expect(projectId).toEqual(mockProjId); + expect(status).toEqual(ExportStatus.Downloading); + }); + + it("correctly affects state on failure", async () => { + const store = setupStore(); + mockDownloadList.mockRejectedValueOnce({}); + await store.dispatch(asyncDownloadExport(mockProjId)); + const { projectId, status } = store.getState().exportProjectState; + expect(projectId).toEqual(mockProjId); + expect(status).toEqual(ExportStatus.Failure); + }); + }); + + describe("asyncExportProject", () => { + it("correctly affects state on success", async () => { + const store = setupStore(); + mockExportLift.mockResolvedValueOnce({}); + await store.dispatch(asyncExportProject(mockProjId)); + const { projectId, status } = store.getState().exportProjectState; + expect(projectId).toEqual(mockProjId); + expect(status).toEqual(ExportStatus.Exporting); + }); + + it("correctly affects state on failure", async () => { + const store = setupStore(); + mockExportLift.mockRejectedValueOnce({}); + await store.dispatch(asyncExportProject(mockProjId)); + const { projectId, status } = store.getState().exportProjectState; + expect(projectId).toEqual(mockProjId); + expect(status).toEqual(ExportStatus.Failure); + }); + }); + + describe("asyncResetExport", () => { + it("correctly affects state", async () => { + const nonDefaultState = { + projectId: "nonempty-string", + status: ExportStatus.Success, + }; + const store = setupStore({ + ...persistedDefaultState, + exportProjectState: nonDefaultState, + }); + await store.dispatch(asyncResetExport()); + const { projectId, status } = store.getState().exportProjectState; + expect(projectId).toEqual(""); + expect(status).toEqual(ExportStatus.Default); + }); + }); +}); diff --git a/src/components/ProjectScreen/CreateProject/index.tsx b/src/components/ProjectScreen/CreateProject.tsx similarity index 94% rename from src/components/ProjectScreen/CreateProject/index.tsx rename to src/components/ProjectScreen/CreateProject.tsx index 6e4ab1a57c..8174810bae 100644 --- a/src/components/ProjectScreen/CreateProject/index.tsx +++ b/src/components/ProjectScreen/CreateProject.tsx @@ -11,7 +11,7 @@ import { Typography, } from "@mui/material"; import { LanguagePicker, languagePickerStrings_en } from "mui-language-picker"; -import React, { Fragment, ReactElement, useEffect, useState } from "react"; +import React, { Fragment, ReactElement, useState } from "react"; import { Trans, useTranslation } from "react-i18next"; import { WritingSystem } from "api/models"; @@ -20,10 +20,8 @@ import { FileInputButton, LoadingDoneButton } from "components/Buttons"; import { asyncCreateProject, asyncFinishProject, - reset, -} from "components/ProjectScreen/CreateProject/Redux/CreateProjectActions"; -import { StoreState } from "types"; -import { useAppDispatch, useAppSelector } from "types/hooks"; +} from "components/ProjectScreen/CreateProjectActions"; +import { useAppDispatch } from "types/hooks"; import theme from "types/theme"; import { newWritingSystem } from "types/writingSystem"; @@ -41,18 +39,12 @@ const undBcp47 = "und"; export default function CreateProject(): ReactElement { const dispatch = useAppDispatch(); - const { inProgress, success } = useAppSelector( - (state: StoreState) => state.createProjectState - ); - - useEffect(() => { - dispatch(reset()); - }, [dispatch]); - const [analysisLang, setAnalysisLang] = useState(newWritingSystem(undBcp47)); const [error, setError] = useState({ empty: false, nameTaken: false }); const [languageData, setLanguageData] = useState(); + const [loading, setLoading] = useState(false); const [name, setName] = useState(""); + const [success, setSuccess] = useState(false); const [vernLang, setVernLang] = useState(newWritingSystem(undBcp47)); const [vernLangIsOther, setVernLangIsOther] = useState(false); const [vernLangOptions, setVernLangOptions] = useState([]); @@ -178,10 +170,16 @@ export default function CreateProject(): ReactElement { return; } + setLoading(true); + if (languageData) { - dispatch(asyncFinishProject(name, vernLang)); + await dispatch(asyncFinishProject(name, vernLang)).then(() => + setSuccess(true) + ); } else { - dispatch(asyncCreateProject(name, vernLang, [analysisLang])); + await dispatch(asyncCreateProject(name, vernLang, [analysisLang])).then( + () => setSuccess(true) + ); } }; @@ -297,7 +295,7 @@ export default function CreateProject(): ReactElement { disabled={!vernLang.bcp47 || vernLang.bcp47 === undBcp47} done={success} doneText={t("createProject.success")} - loading={inProgress} + loading={loading} > {t("createProject.create")} diff --git a/src/components/ProjectScreen/CreateProject/Redux/CreateProjectActions.ts b/src/components/ProjectScreen/CreateProject/Redux/CreateProjectActions.ts deleted file mode 100644 index d0a593086d..0000000000 --- a/src/components/ProjectScreen/CreateProject/Redux/CreateProjectActions.ts +++ /dev/null @@ -1,100 +0,0 @@ -import { WritingSystem } from "api/models"; -import { createProject, finishUploadLift, getProject } from "backend"; -import router from "browserRouter"; -import { asyncCreateUserEdits } from "components/GoalTimeline/Redux/GoalActions"; -import { setNewCurrentProject } from "components/Project/ProjectActions"; -import { - CreateProjectAction, - CreateProjectActionTypes, -} from "components/ProjectScreen/CreateProject/Redux/CreateProjectReduxTypes"; -import { StoreStateDispatch } from "types/Redux/actions"; -import { Path } from "types/path"; -import { newProject } from "types/project"; - -/** thunk action creator for creating a project without an import. */ -export function asyncCreateProject( - name: string, - vernacularWritingSystem: WritingSystem, - analysisWritingSystems: WritingSystem[] -) { - return async (dispatch: StoreStateDispatch) => { - dispatch(inProgress()); - - const project = newProject(name); - project.vernacularWritingSystem = vernacularWritingSystem; - project.analysisWritingSystems = analysisWritingSystems; - - await createProject(project) - .then(async (createdProject) => { - dispatch(setNewCurrentProject(createdProject)); - dispatch(success()); - - // Manually pause so they have a chance to see the success message. - setTimeout(() => { - dispatch(asyncCreateUserEdits()); - router.navigate(Path.ProjSettings); - }, 1000); - }) - .catch((e) => { - dispatch(failure(e.response?.statusText)); - }); - }; -} - -/** thunk action creator for creating a project with a pre-uploaded import. */ -export function asyncFinishProject( - name: string, - vernacularWritingSystem: WritingSystem -) { - return async (dispatch: StoreStateDispatch) => { - dispatch(inProgress()); - - const project = newProject(name); - project.vernacularWritingSystem = vernacularWritingSystem; - - await createProject(project) - .then(async (createdProject) => { - await finishUploadLift(createdProject.id); - createdProject = await getProject(createdProject.id); - dispatch(setNewCurrentProject(createdProject)); - dispatch(success()); - - // Manually pause so they have a chance to see the success message. - setTimeout(() => { - dispatch(asyncCreateUserEdits()); - router.navigate(Path.ProjSettings); - }, 1000); - }) - .catch((e) => { - dispatch(failure(e.response?.statusText)); - }); - }; -} - -export function inProgress(): CreateProjectAction { - return { - type: CreateProjectActionTypes.CREATE_PROJECT_IN_PROGRESS, - payload: {}, - }; -} - -export function success(): CreateProjectAction { - return { - type: CreateProjectActionTypes.CREATE_PROJECT_SUCCESS, - payload: {}, - }; -} - -export function failure(errorMsg = ""): CreateProjectAction { - return { - type: CreateProjectActionTypes.CREATE_PROJECT_FAILURE, - payload: { errorMsg }, - }; -} - -export function reset(): CreateProjectAction { - return { - type: CreateProjectActionTypes.CREATE_PROJECT_RESET, - payload: {}, - }; -} diff --git a/src/components/ProjectScreen/CreateProject/Redux/CreateProjectReducer.ts b/src/components/ProjectScreen/CreateProject/Redux/CreateProjectReducer.ts deleted file mode 100644 index 31685a01d3..0000000000 --- a/src/components/ProjectScreen/CreateProject/Redux/CreateProjectReducer.ts +++ /dev/null @@ -1,27 +0,0 @@ -import { - CreateProjectAction, - CreateProjectActionTypes, - CreateProjectState, - defaultState, -} from "components/ProjectScreen/CreateProject/Redux/CreateProjectReduxTypes"; -import { StoreAction, StoreActionTypes } from "rootActions"; - -export const createProjectReducer = ( - state: CreateProjectState = defaultState, - action: CreateProjectAction | StoreAction -): CreateProjectState => { - switch (action.type) { - case CreateProjectActionTypes.CREATE_PROJECT_IN_PROGRESS: - return { ...defaultState, inProgress: true }; - case CreateProjectActionTypes.CREATE_PROJECT_SUCCESS: - return { ...defaultState, success: true }; - case CreateProjectActionTypes.CREATE_PROJECT_FAILURE: - return { ...defaultState, errorMsg: action.payload.errorMsg ?? "" }; - case CreateProjectActionTypes.CREATE_PROJECT_RESET: - return defaultState; - case StoreActionTypes.RESET: - return defaultState; - default: - return state; - } -}; diff --git a/src/components/ProjectScreen/CreateProject/Redux/CreateProjectReduxTypes.ts b/src/components/ProjectScreen/CreateProject/Redux/CreateProjectReduxTypes.ts deleted file mode 100644 index 39d2b6987a..0000000000 --- a/src/components/ProjectScreen/CreateProject/Redux/CreateProjectReduxTypes.ts +++ /dev/null @@ -1,29 +0,0 @@ -export enum CreateProjectActionTypes { - CREATE_PROJECT_FAILURE = "CREATE_PROJECT_FAILURE", - CREATE_PROJECT_IN_PROGRESS = "CREATE_PROJECT_IN_PROGRESS", - CREATE_PROJECT_RESET = "CREATE_PROJECT_RESET", - CREATE_PROJECT_SUCCESS = "CREATE_PROJECT_SUCCESS", -} - -type CreateProjectType = - | typeof CreateProjectActionTypes.CREATE_PROJECT_FAILURE - | typeof CreateProjectActionTypes.CREATE_PROJECT_IN_PROGRESS - | typeof CreateProjectActionTypes.CREATE_PROJECT_RESET - | typeof CreateProjectActionTypes.CREATE_PROJECT_SUCCESS; - -export interface CreateProjectAction { - type: CreateProjectType; - payload: { errorMsg?: string }; -} - -export interface CreateProjectState { - inProgress: boolean; - success: boolean; - errorMsg: string; -} - -export const defaultState: CreateProjectState = { - success: false, - inProgress: false, - errorMsg: "", -}; diff --git a/src/components/ProjectScreen/CreateProject/tests/CreateProjectActions.test.tsx b/src/components/ProjectScreen/CreateProject/tests/CreateProjectActions.test.tsx deleted file mode 100644 index 0c5af9eb1b..0000000000 --- a/src/components/ProjectScreen/CreateProject/tests/CreateProjectActions.test.tsx +++ /dev/null @@ -1,48 +0,0 @@ -import configureMockStore from "redux-mock-store"; -import thunk from "redux-thunk"; - -import * as action from "components/ProjectScreen/CreateProject/Redux/CreateProjectActions"; -import { defaultState } from "components/ProjectScreen/CreateProject/Redux/CreateProjectReduxTypes"; -import { newWritingSystem } from "types/writingSystem"; - -jest.mock("backend", () => ({ - createProject: () => Promise.reject({ response: "intentional failure" }), -})); - -const mockStore = configureMockStore([thunk])(defaultState); - -const project = { - name: "testProjectName", - vernacularLanguage: newWritingSystem("testVernCode", "testVernName"), - analysisLanguages: [newWritingSystem("testAnalysisCode", "testAnalysisName")], -}; - -beforeEach(() => { - mockStore.clearActions(); -}); - -describe("CreateProjectActions", () => { - test("asyncCreateProject correctly affects state", async () => { - await mockStore.dispatch( - action.asyncCreateProject( - project.name, - project.vernacularLanguage, - project.analysisLanguages - ) - ); - expect(mockStore.getActions()).toEqual([ - action.inProgress(), - action.failure(), // backend.createProject mocked to fail - ]); - }); - - test("asyncFinishProject correctly affects state", async () => { - await mockStore.dispatch( - action.asyncFinishProject(project.name, project.vernacularLanguage) - ); - expect(mockStore.getActions()).toEqual([ - action.inProgress(), - action.failure(), // backend.createProject mocked to fail - ]); - }); -}); diff --git a/src/components/ProjectScreen/CreateProject/tests/CreateProjectReducer.test.tsx b/src/components/ProjectScreen/CreateProject/tests/CreateProjectReducer.test.tsx deleted file mode 100644 index 89b80daff9..0000000000 --- a/src/components/ProjectScreen/CreateProject/tests/CreateProjectReducer.test.tsx +++ /dev/null @@ -1,47 +0,0 @@ -import { - failure, - inProgress, - reset, - success, -} from "components/ProjectScreen/CreateProject/Redux/CreateProjectActions"; -import * as reducer from "components/ProjectScreen/CreateProject/Redux/CreateProjectReducer"; -import { - CreateProjectState, - defaultState, -} from "components/ProjectScreen/CreateProject/Redux/CreateProjectReduxTypes"; -import { StoreAction, StoreActionTypes } from "rootActions"; - -describe("CreateProjectReducer", () => { - test("undefined state and reset action, expecting default state", () => { - expect(reducer.createProjectReducer(undefined, reset())).toEqual( - defaultState - ); - }); - - test("undefined state and failure action, expecting error message", () => { - const errorMsg = ""; - expect(reducer.createProjectReducer(undefined, failure(errorMsg))).toEqual({ - ...defaultState, - errorMsg, - }); - }); - - test("empty state and in-progress action, expecting inProgress", () => { - expect( - reducer.createProjectReducer({} as CreateProjectState, inProgress()) - ).toEqual({ ...defaultState, inProgress: true }); - }); - - test("empty state and success action, expecting success", () => { - expect( - reducer.createProjectReducer({} as CreateProjectState, success()) - ).toEqual({ ...defaultState, success: true }); - }); - - test("empty state and store reset action, expecting default state", () => { - const storeResetAction: StoreAction = { type: StoreActionTypes.RESET }; - expect( - reducer.createProjectReducer({} as CreateProjectState, storeResetAction) - ).toEqual(defaultState); - }); -}); diff --git a/src/components/ProjectScreen/CreateProjectActions.ts b/src/components/ProjectScreen/CreateProjectActions.ts new file mode 100644 index 0000000000..1c6a3c2e85 --- /dev/null +++ b/src/components/ProjectScreen/CreateProjectActions.ts @@ -0,0 +1,54 @@ +import { WritingSystem } from "api/models"; +import { createProject, finishUploadLift, getProject } from "backend"; +import router from "browserRouter"; +import { asyncCreateUserEdits } from "components/GoalTimeline/Redux/GoalActions"; +import { setNewCurrentProject } from "components/Project/ProjectActions"; +import { StoreStateDispatch } from "types/Redux/actions"; +import { Path } from "types/path"; +import { newProject } from "types/project"; + +// Dispatch Functions + +/*** Create a project without an import. */ +export function asyncCreateProject( + name: string, + vernacularWritingSystem: WritingSystem, + analysisWritingSystems: WritingSystem[] +) { + return async (dispatch: StoreStateDispatch) => { + const project = newProject(name); + project.vernacularWritingSystem = vernacularWritingSystem; + project.analysisWritingSystems = analysisWritingSystems; + + const createdProject = await createProject(project); + dispatch(setNewCurrentProject(createdProject)); + + // Manually pause so they have a chance to see the success message. + setTimeout(() => { + dispatch(asyncCreateUserEdits()); + router.navigate(Path.ProjSettings); + }, 1000); + }; +} + +/*** Create a project with a pre-uploaded import. */ +export function asyncFinishProject( + name: string, + vernacularWritingSystem: WritingSystem +) { + return async (dispatch: StoreStateDispatch) => { + const project = newProject(name); + project.vernacularWritingSystem = vernacularWritingSystem; + + const projId = (await createProject(project)).id; + await finishUploadLift(projId); + const createdProject = await getProject(projId); + dispatch(setNewCurrentProject(createdProject)); + + // Manually pause so they have a chance to see the success message. + setTimeout(() => { + dispatch(asyncCreateUserEdits()); + router.navigate(Path.ProjSettings); + }, 1000); + }; +} diff --git a/src/components/ProjectScreen/CreateProject/tests/CreateProjectComponent.test.tsx b/src/components/ProjectScreen/tests/CreateProject.test.tsx similarity index 95% rename from src/components/ProjectScreen/CreateProject/tests/CreateProjectComponent.test.tsx rename to src/components/ProjectScreen/tests/CreateProject.test.tsx index b39d3c42d2..f87d550a1b 100644 --- a/src/components/ProjectScreen/CreateProject/tests/CreateProjectComponent.test.tsx +++ b/src/components/ProjectScreen/tests/CreateProject.test.tsx @@ -17,7 +17,6 @@ import CreateProject, { formId, selectIdVern, } from "components/ProjectScreen/CreateProject"; -import { defaultState } from "components/ProjectScreen/CreateProject/Redux/CreateProjectReduxTypes"; import { newWritingSystem } from "types/writingSystem"; jest.mock("backend", () => ({ @@ -29,10 +28,7 @@ const mockProjectDuplicateCheck = jest.fn(); const mockUploadLiftAndGetWritingSystems = jest.fn(); const createMockStore = configureMockStore(); -const mockState = { - currentProjectState: { project: {} }, - createProjectState: defaultState, -}; +const mockState = { currentProjectState: { project: {} } }; const mockStore = createMockStore(mockState); const mockChangeEvent = ( diff --git a/src/rootReducer.ts b/src/rootReducer.ts index dabea8c886..b0863ed7f0 100644 --- a/src/rootReducer.ts +++ b/src/rootReducer.ts @@ -3,8 +3,7 @@ import { combineReducers, Reducer } from "redux"; import goalsReducer from "components/GoalTimeline/Redux/GoalReducer"; import { loginReducer } from "components/Login/Redux/LoginReducer"; import { projectReducer } from "components/Project/ProjectReducer"; -import { exportProjectReducer } from "components/ProjectExport/Redux/ExportProjectReducer"; -import { createProjectReducer } from "components/ProjectScreen/CreateProject/Redux/CreateProjectReducer"; +import exportProjectReducer from "components/ProjectExport/Redux/ExportProjectReducer"; import { pronunciationsReducer } from "components/Pronunciations/Redux/PronunciationsReducer"; import { treeViewReducer } from "components/TreeView/Redux/TreeViewReducer"; import { characterInventoryReducer } from "goals/CharacterInventory/Redux/CharacterInventoryReducer"; @@ -18,7 +17,6 @@ export const rootReducer: Reducer = combineReducers({ loginState: loginReducer, //project - createProjectState: createProjectReducer, currentProjectState: projectReducer, exportProjectState: exportProjectReducer, diff --git a/src/types/index.ts b/src/types/index.ts index a82aa20a00..114069a2de 100644 --- a/src/types/index.ts +++ b/src/types/index.ts @@ -1,7 +1,6 @@ import { LoginState } from "components/Login/Redux/LoginReduxTypes"; import { CurrentProjectState } from "components/Project/ProjectReduxTypes"; import { ExportProjectState } from "components/ProjectExport/Redux/ExportProjectReduxTypes"; -import { CreateProjectState } from "components/ProjectScreen/CreateProject/Redux/CreateProjectReduxTypes"; import { PronunciationsState } from "components/Pronunciations/Redux/PronunciationsReduxTypes"; import { TreeViewState } from "components/TreeView/Redux/TreeViewReduxTypes"; import { CharacterInventoryState } from "goals/CharacterInventory/Redux/CharacterInventoryReduxTypes"; @@ -16,7 +15,6 @@ export interface StoreState { loginState: LoginState; //project - createProjectState: CreateProjectState; currentProjectState: CurrentProjectState; exportProjectState: ExportProjectState; From bf5b4e91966aa04f0546b837a13527f4792761da Mon Sep 17 00:00:00 2001 From: "D. Ror" Date: Mon, 30 Oct 2023 12:22:29 -0400 Subject: [PATCH 4/4] Clean up keys; Fix up GoalTimeline tests (#2721) --- .../DataEntry/DataEntryTable/index.tsx | 7 +- src/components/GoalTimeline/GoalList.tsx | 49 ++++++---- .../GoalTimeline/tests/GoalRedux.test.tsx | 97 ++++++++++--------- .../ProjectSchedule/CalendarView.tsx | 2 +- .../ProjectSettings/ProjectSelect.tsx | 4 +- .../Statistics/DomainStatistics.tsx | 6 +- src/components/Statistics/UserStatistics.tsx | 15 +-- .../CharacterInventory/CharInvCompleted.tsx | 51 +++++----- .../ReviewEntriesComponent/CellColumns.tsx | 1 - .../CellComponents/AlignedList.tsx | 2 +- .../CellComponents/DefinitionCell.tsx | 2 +- .../CellComponents/DomainCell.tsx | 10 +- .../CellComponents/GlossCell.tsx | 2 +- .../CellComponents/NoteCell.tsx | 1 - .../CellComponents/PartOfSpeechCell.tsx | 5 +- .../CellComponents/SenseCell.tsx | 1 - .../CellComponents/VernacularCell.tsx | 1 - 17 files changed, 125 insertions(+), 131 deletions(-) diff --git a/src/components/DataEntry/DataEntryTable/index.tsx b/src/components/DataEntry/DataEntryTable/index.tsx index 5ebc1bc007..e3d37e5e0e 100644 --- a/src/components/DataEntry/DataEntryTable/index.tsx +++ b/src/components/DataEntry/DataEntryTable/index.tsx @@ -865,9 +865,12 @@ export default function DataEntryTable( {state.recentWords.map((wordAccess, index) => ( - + (); const tileSize = props.size / 3 - 1.25; + const id = (g: Goal): string => + props.completed ? `completed-goal-${g.guid}` : `new-goal-${g.name}`; + return ( setScrollVisible(props.scrollable)} onMouseLeave={() => setScrollVisible(false)} > - {props.data.length > 0 - ? props.data.map((g, i) => { - const buttonProps = { - id: props.completed - ? `completed-goal-${i}` - : `new-goal-${g.name}`, - onClick: () => props.handleChange(g), - }; - return makeGoalTile(tileSize, props.orientation, g, buttonProps); - }) - : makeGoalTile(tileSize, props.orientation)} + {props.data.length > 0 ? ( + props.data.map((g) => ( + props.handleChange(g) }} + goal={g} + key={g.guid || g.name} + orientation={props.orientation} + size={tileSize} + /> + )) + ) : ( + + )}
{ if (props.scrollToEnd && element) { @@ -93,19 +98,22 @@ function buttonStyle(orientation: Orientation, size: number): CSSProperties { } } -export function makeGoalTile( - size: number, - orientation: Orientation, - goal?: Goal, - buttonProps?: ButtonProps -): ReactElement { +interface GoalTileProps { + buttonProps?: ButtonProps; + goal?: Goal; + orientation: Orientation; + size: number; +} + +function GoalTile(props: GoalTileProps): ReactElement { + const goal = props.goal; return ( - +