From b9252e26fcca511a29c4057866c068c145bc3735 Mon Sep 17 00:00:00 2001 From: "D. Ror" Date: Thu, 2 Nov 2023 17:13:47 -0400 Subject: [PATCH 1/2] [DataEntry] Prevent RecentEntry re-renders when typing in NewEntry (#2752) --- .../DataEntry/DataEntryTable/RecentEntry.tsx | 26 +- .../DataEntry/DataEntryTable/index.tsx | 403 +++++++++--------- .../DataEntryTable/tests/RecentEntry.test.tsx | 6 +- .../DataEntryTable/tests/index.test.tsx | 4 +- 4 files changed, 234 insertions(+), 205 deletions(-) diff --git a/src/components/DataEntry/DataEntryTable/RecentEntry.tsx b/src/components/DataEntry/DataEntryTable/RecentEntry.tsx index 95165d2514..de27a2b4d3 100644 --- a/src/components/DataEntry/DataEntryTable/RecentEntry.tsx +++ b/src/components/DataEntry/DataEntryTable/RecentEntry.tsx @@ -1,5 +1,5 @@ import { Grid } from "@mui/material"; -import { ReactElement, useState } from "react"; +import { ReactElement, memo, useState } from "react"; import { Word, WritingSystem } from "api/models"; import { @@ -19,10 +19,10 @@ export interface RecentEntryProps { rowIndex: number; entry: Word; senseGuid: string; - updateGloss: (gloss: string) => void; - updateNote: (newText: string) => Promise; - updateVern: (newVernacular: string, targetWordId?: string) => void; - removeEntry: () => void; + updateGloss: (index: number, gloss: string) => void; + updateNote: (index: number, newText: string) => Promise; + updateVern: (index: number, newVern: string, targetWordId?: string) => void; + removeEntry: (index: number) => void; addAudioToWord: (wordId: string, audioFile: File) => void; deleteAudioFromWord: (wordId: string, fileName: string) => void; focusNewEntry: () => void; @@ -34,7 +34,7 @@ export interface RecentEntryProps { /** * Displays a recently entered word that a user can still edit */ -export default function RecentEntry(props: RecentEntryProps): ReactElement { +export function RecentEntry(props: RecentEntryProps): ReactElement { const sense = props.entry.senses.find((s) => s.guid === props.senseGuid)!; if (sense.glosses.length < 1) { sense.glosses.push(newGloss("", props.analysisLang.bcp47)); @@ -44,20 +44,24 @@ export default function RecentEntry(props: RecentEntryProps): ReactElement { function conditionallyUpdateGloss(): void { if (firstGlossText(sense) !== gloss) { - props.updateGloss(gloss); + props.updateGloss(props.rowIndex, gloss); } } function conditionallyUpdateVern(): void { if (vernacular.trim()) { if (props.entry.vernacular !== vernacular) { - props.updateVern(vernacular); + props.updateVern(props.rowIndex, vernacular); } } else { setVernacular(props.entry.vernacular); } } + const handleRemoveEntry = (): void => props.removeEntry(props.rowIndex); + const handleUpdateNote = (noteText: string): Promise => + props.updateNote(props.rowIndex, noteText); + return ( )} @@ -152,7 +156,7 @@ export default function RecentEntry(props: RecentEntryProps): ReactElement { > {!props.disabled && ( props.removeEntry()} + removeEntry={handleRemoveEntry} buttonId={`${idAffix}-${props.rowIndex}-delete`} confirmId={"addWords.deleteRowWarning"} wordId={props.entry.id} @@ -162,3 +166,5 @@ export default function RecentEntry(props: RecentEntryProps): ReactElement { ); } + +export default memo(RecentEntry); diff --git a/src/components/DataEntry/DataEntryTable/index.tsx b/src/components/DataEntry/DataEntryTable/index.tsx index e3d37e5e0e..607c3bde3f 100644 --- a/src/components/DataEntry/DataEntryTable/index.tsx +++ b/src/components/DataEntry/DataEntryTable/index.tsx @@ -249,26 +249,29 @@ export default function DataEntryTable( * to make sure that word doesn't get edited by two different functions. * Use this with newId to specify the replacement of a defunct word. */ - const defunctWord = (oldId: string, newId?: string): void => { - if (!newId && state.defunctWordIds[oldId]) { - return; - } - if (!newId) { + const defunctWord = useCallback( + (oldId: string, newId?: string): void => { + if (!newId && state.defunctWordIds[oldId]) { + return; + } + if (!newId) { + setState((prevState) => { + const defunctWordIds = { ...prevState.defunctWordIds }; + defunctWordIds[oldId] = DefunctStatus.Recent; + return { ...prevState, defunctWordIds }; + }); + return; + } setState((prevState) => { + const defunctUpdates = { ...prevState.defunctUpdates }; const defunctWordIds = { ...prevState.defunctWordIds }; + defunctUpdates[oldId] = newId; defunctWordIds[oldId] = DefunctStatus.Recent; - return { ...prevState, defunctWordIds }; + return { ...prevState, defunctUpdates, defunctWordIds }; }); - return; - } - setState((prevState) => { - const defunctUpdates = { ...prevState.defunctUpdates }; - const defunctWordIds = { ...prevState.defunctWordIds }; - defunctUpdates[oldId] = newId; - defunctWordIds[oldId] = DefunctStatus.Recent; - return { ...prevState, defunctUpdates, defunctWordIds }; - }); - }; + }, + [state.defunctWordIds] + ); /*** Update a recent entry to a different sense of the same word. */ const switchSense = useCallback( @@ -291,18 +294,21 @@ export default function DataEntryTable( ); /*** Add to recent entries every sense of the word with the current semantic domain. */ - const addAllSensesToDisplay = (word: Word): void => { - const domId = props.semanticDomain.id; - setState((prevState) => { - const recentWords = [...prevState.recentWords]; - word.senses.forEach((s) => { - if (s.semanticDomains.find((dom) => dom.id === domId)) { - recentWords.push({ word, senseGuid: s.guid }); - } + const addAllSensesToDisplay = useCallback( + (word: Word): void => { + const domId = props.semanticDomain.id; + setState((prevState) => { + const recentWords = [...prevState.recentWords]; + word.senses.forEach((s) => { + if (s.semanticDomains.find((dom) => dom.id === domId)) { + recentWords.push({ word, senseGuid: s.guid }); + } + }); + return { ...prevState, recentWords }; }); - return { ...prevState, recentWords }; - }); - }; + }, + [props.semanticDomain.id] + ); /*** Add one-sense word to the display of recent entries. */ const addToDisplay = (wordAccess: WordAccess, insertIndex?: number): void => { @@ -558,98 +564,103 @@ export default function DataEntryTable( //////////////////////////////////// /*** Given an array of audio file urls, add them all to specified word. */ - const addAudiosToBackend = async ( - oldId: string, - audioURLs: string[] - ): Promise => { - if (!audioURLs.length) { - return oldId; - } - defunctWord(oldId); - let newId = oldId; - for (const audioURL of audioURLs) { - newId = await uploadFileFromUrl(newId, audioURL); - } - defunctWord(oldId, newId); - return newId; - }; + const addAudiosToBackend = useCallback( + async (oldId: string, audioURLs: string[]): Promise => { + if (!audioURLs.length) { + return oldId; + } + defunctWord(oldId); + let newId = oldId; + for (const audioURL of audioURLs) { + newId = await uploadFileFromUrl(newId, audioURL); + } + defunctWord(oldId, newId); + return newId; + }, + [defunctWord] + ); /*** Given a single audio file, add to specified word. */ - const addAudioFileToWord = async ( - oldId: string, - audioFile: File - ): Promise => { - defunctWord(oldId); - const newId = await backend.uploadAudio(oldId, audioFile); - defunctWord(oldId, newId); - }; + const addAudioFileToWord = useCallback( + async (oldId: string, audioFile: File): Promise => { + defunctWord(oldId); + const newId = await backend.uploadAudio(oldId, audioFile); + defunctWord(oldId, newId); + }, + [defunctWord] + ); /*** Add a word determined to be a duplicate. * Ensures the updated word has representation in the display. * Note: Only for use after backend.getDuplicateId(). */ - const addDuplicateWord = async ( - word: Word, - audioURLs: string[], - oldId: string - ): Promise => { - const isInDisplay = - state.recentWords.findIndex((w) => w.word.id === oldId) > -1; + const addDuplicateWord = useCallback( + async (word: Word, audioURLs: string[], oldId: string): Promise => { + const isInDisplay = + state.recentWords.findIndex((w) => w.word.id === oldId) > -1; - defunctWord(oldId); - const newWord = await backend.updateDuplicate(oldId, word); - defunctWord(oldId, newWord.id); + defunctWord(oldId); + const newWord = await backend.updateDuplicate(oldId, word); + defunctWord(oldId, newWord.id); - const newId = await addAudiosToBackend(newWord.id, audioURLs); + const newId = await addAudiosToBackend(newWord.id, audioURLs); - if (!isInDisplay) { - addAllSensesToDisplay(await backend.getWord(newId)); - } - }; + if (!isInDisplay) { + addAllSensesToDisplay(await backend.getWord(newId)); + } + }, + [addAllSensesToDisplay, addAudiosToBackend, defunctWord, state.recentWords] + ); /*** Deletes specified audio file from specified word. */ - const deleteAudioFromWord = async ( - oldId: string, - fileName: string - ): Promise => { - defunctWord(oldId); - const newId = await backend.deleteAudio(oldId, fileName); - defunctWord(oldId, newId); - }; + const deleteAudioFromWord = useCallback( + async (oldId: string, fileName: string): Promise => { + defunctWord(oldId); + const newId = await backend.deleteAudio(oldId, fileName); + defunctWord(oldId, newId); + }, + [defunctWord] + ); /*** Updates word. */ - const updateWordInBackend = async (word: Word): Promise => { - defunctWord(word.id); - const newWord = await backend.updateWord(word); - defunctWord(word.id, newWord.id); - return newWord; - }; + const updateWordInBackend = useCallback( + async (word: Word): Promise => { + defunctWord(word.id); + const newWord = await backend.updateWord(word); + defunctWord(word.id, newWord.id); + return newWord; + }, + [defunctWord] + ); ///////////////////////////////// // General async functions. ///////////////////////////////// /*** Add a new word to the project, or update if new word is a duplicate. */ - const addNewWord = async ( - wordToAdd: Word, - audioURLs: string[], - insertIndex?: number - ): Promise => { - wordToAdd.note.language = analysisLang.bcp47; - - // Check if word is duplicate to existing word. - const dupId = await backend.getDuplicateId(wordToAdd); - if (dupId) { - return await addDuplicateWord(wordToAdd, audioURLs, dupId); - } + const addNewWord = useCallback( + async ( + wordToAdd: Word, + audioURLs: string[], + insertIndex?: number + ): Promise => { + wordToAdd.note.language = analysisLang.bcp47; + + // Check if word is duplicate to existing word. + const dupId = await backend.getDuplicateId(wordToAdd); + if (dupId) { + return await addDuplicateWord(wordToAdd, audioURLs, dupId); + } - let word = await backend.createWord(wordToAdd); - const wordId = await addAudiosToBackend(word.id, audioURLs); - if (wordId !== word.id) { - word = await backend.getWord(wordId); - } - addToDisplay({ word, senseGuid: word.senses[0].guid }, insertIndex); - }; + let word = await backend.createWord(wordToAdd); + const wordId = await addAudiosToBackend(word.id, audioURLs); + if (wordId !== word.id) { + word = await backend.getWord(wordId); + } + addToDisplay({ word, senseGuid: word.senses[0].guid }, insertIndex); + }, + [addAudiosToBackend, addDuplicateWord, analysisLang.bcp47] + ); /*** Update the word in the backend and the frontend. */ const updateWordBackAndFront = async ( @@ -747,93 +758,111 @@ export default function DataEntryTable( ///////////////////////////////// /*** Retract a recent entry. */ - const undoRecentEntry = async (eIndex: number): Promise => { - const { word, senseGuid } = state.recentWords[eIndex]; - const sIndex = word.senses.findIndex((s) => s.guid === senseGuid); - if (sIndex === -1) { - throw new Error("Entry does not have specified sense."); - } - defunctWord(word.id); - removeRecentEntry(eIndex); - const senses = [...word.senses]; - const oldSense = senses[sIndex]; - const oldDoms = oldSense.semanticDomains; - if (oldDoms.length > 1) { - // If there is more than one semantic domain in this sense, only remove the domain. - const doms = oldDoms.filter((d) => d.id !== props.semanticDomain.id); - const newSense: Sense = { ...oldSense, semanticDomains: doms }; - senses.splice(sIndex, 1, newSense); - await updateWordInBackend({ ...word, senses }); - } else if (senses.length > 1) { - // If there is more than one sense in this word, only remove this sense. - senses.splice(sIndex, 1); - await updateWordInBackend({ ...word, senses }); - } else { - // Since this is the only sense, delete the word. - await backend.deleteFrontierWord(word.id); - } - }; + const undoRecentEntry = useCallback( + async (eIndex: number): Promise => { + const { word, senseGuid } = state.recentWords[eIndex]; + const sIndex = word.senses.findIndex((s) => s.guid === senseGuid); + if (sIndex === -1) { + throw new Error("Entry does not have specified sense."); + } + defunctWord(word.id); + removeRecentEntry(eIndex); + const senses = [...word.senses]; + const oldSense = senses[sIndex]; + const oldDoms = oldSense.semanticDomains; + if (oldDoms.length > 1) { + // If there is more than one semantic domain in this sense, only remove the domain. + const doms = oldDoms.filter((d) => d.id !== props.semanticDomain.id); + const newSense: Sense = { ...oldSense, semanticDomains: doms }; + senses.splice(sIndex, 1, newSense); + await updateWordInBackend({ ...word, senses }); + } else if (senses.length > 1) { + // If there is more than one sense in this word, only remove this sense. + senses.splice(sIndex, 1); + await updateWordInBackend({ ...word, senses }); + } else { + // Since this is the only sense, delete the word. + await backend.deleteFrontierWord(word.id); + } + }, + [ + defunctWord, + props.semanticDomain.id, + state.recentWords, + updateWordInBackend, + ] + ); /*** Update the vernacular in a recent entry. */ - const updateRecentVern = async ( - index: number, - vernacular: string, - targetWordId?: string - ): Promise => { - if (targetWordId !== undefined) { - throw new Error("VernDialog on RecentEntry is not yet supported."); - } - const oldEntry = state.recentWords[index]; - const oldSenses = oldEntry.word.senses; - const oldSense = oldSenses.find((s) => s.guid === oldEntry.senseGuid); - if (!oldSense) { - throw new Error("Entry does not have specified sense."); - } + const updateRecentVern = useCallback( + async ( + index: number, + vernacular: string, + targetWordId?: string + ): Promise => { + if (targetWordId !== undefined) { + throw new Error("VernDialog on RecentEntry is not yet supported."); + } + const oldEntry = state.recentWords[index]; + const oldSenses = oldEntry.word.senses; + const oldSense = oldSenses.find((s) => s.guid === oldEntry.senseGuid); + if (!oldSense) { + throw new Error("Entry does not have specified sense."); + } - if (oldSenses.length === 1 && oldSense.semanticDomains.length === 1) { - // The word can simply be updated as it stand - await updateWordInBackend({ ...oldEntry.word, vernacular }); - } else { - // Retract and replaced with a new entry. - const word = simpleWord(vernacular, firstGlossText(oldSense)); - word.id = ""; - await undoRecentEntry(index); - await addNewWord(word, [], index); - } - }; + if (oldSenses.length === 1 && oldSense.semanticDomains.length === 1) { + // The word can simply be updated as it stand + await updateWordInBackend({ ...oldEntry.word, vernacular }); + } else { + // Retract and replaced with a new entry. + const word = simpleWord(vernacular, firstGlossText(oldSense)); + word.id = ""; + await undoRecentEntry(index); + await addNewWord(word, [], index); + } + }, + [addNewWord, state.recentWords, undoRecentEntry, updateWordInBackend] + ); /*** Update the gloss def in a recent entry. */ - const updateRecentGloss = async ( - index: number, - def: string - ): Promise => { - const oldEntry = state.recentWords[index]; - defunctWord(oldEntry.word.id); - const newWord = updateEntryGloss(oldEntry, def, props.semanticDomain.id); - await updateWordInBackend(newWord); - - // If a sense with a new guid was added, it needs to replace the old sense in the display. - if (newWord.senses.length > oldEntry.word.senses.length) { - const newSense = newWord.senses.find( - (sense) => !oldEntry.word.senses.find((s) => s.guid === sense.guid) - ); - if (newSense) { - queueSenseSwitch(oldEntry.senseGuid, newSense.guid); + const updateRecentGloss = useCallback( + async (index: number, def: string): Promise => { + const oldEntry = state.recentWords[index]; + defunctWord(oldEntry.word.id); + const newWord = updateEntryGloss(oldEntry, def, props.semanticDomain.id); + await updateWordInBackend(newWord); + + // If a sense with a new guid was added, it needs to replace the old sense in the display. + if (newWord.senses.length > oldEntry.word.senses.length) { + const newSense = newWord.senses.find( + (sense) => !oldEntry.word.senses.find((s) => s.guid === sense.guid) + ); + if (newSense) { + queueSenseSwitch(oldEntry.senseGuid, newSense.guid); + } } - } - }; + }, + [ + defunctWord, + props.semanticDomain.id, + state.recentWords, + updateWordInBackend, + ] + ); + + const handleFocusNewEntry = useCallback(() => focusInput(newVernInput), []); /*** Update the note text in a recent entry. */ - const updateRecentNote = async ( - index: number, - text: string - ): Promise => { - const oldWord = state.recentWords[index].word; - if (text !== oldWord.note.text) { - const note: Note = { ...oldWord.note, text }; - await updateWordInBackend({ ...oldWord, note }); - } - }; + const updateRecentNote = useCallback( + async (index: number, text: string): Promise => { + const oldWord = state.recentWords[index].word; + if (text !== oldWord.note.text) { + const note: Note = { ...oldWord.note, text }; + await updateWordInBackend({ ...oldWord, note }); + } + }, + [state.recentWords, updateWordInBackend] + ); return (
) => e?.preventDefault()}> @@ -874,19 +903,13 @@ export default function DataEntryTable( rowIndex={index} entry={wordAccess.word} senseGuid={wordAccess.senseGuid} - updateGloss={(newDef: string) => updateRecentGloss(index, newDef)} - updateNote={(newText: string) => updateRecentNote(index, newText)} - updateVern={(newVernacular: string, targetWordId?: string) => - updateRecentVern(index, newVernacular, targetWordId) - } - removeEntry={() => undoRecentEntry(index)} - addAudioToWord={(wordId: string, audioFile: File) => - addAudioFileToWord(wordId, audioFile) - } - deleteAudioFromWord={(wordId: string, fileName: string) => - deleteAudioFromWord(wordId, fileName) - } - focusNewEntry={() => focusInput(newVernInput)} + updateGloss={updateRecentGloss} + updateNote={updateRecentNote} + updateVern={updateRecentVern} + removeEntry={undoRecentEntry} + addAudioToWord={addAudioFileToWord} + deleteAudioFromWord={deleteAudioFromWord} + focusNewEntry={handleFocusNewEntry} analysisLang={analysisLang} vernacularLang={vernacularLang} disabled={Object.keys(state.defunctWordIds).includes( diff --git a/src/components/DataEntry/DataEntryTable/tests/RecentEntry.test.tsx b/src/components/DataEntry/DataEntryTable/tests/RecentEntry.test.tsx index fc8525608b..43772b8622 100644 --- a/src/components/DataEntry/DataEntryTable/tests/RecentEntry.test.tsx +++ b/src/components/DataEntry/DataEntryTable/tests/RecentEntry.test.tsx @@ -107,7 +107,7 @@ describe("ExistingEntry", () => { await updateVernAndBlur(mockVern); expect(mockUpdateVern).toBeCalledTimes(0); await updateVernAndBlur(mockText); - expect(mockUpdateVern).toBeCalledWith(mockText); + expect(mockUpdateVern).toBeCalledWith(0, mockText); }); }); @@ -125,7 +125,7 @@ describe("ExistingEntry", () => { await updateGlossAndBlur(mockGloss); expect(mockUpdateGloss).toBeCalledTimes(0); await updateGlossAndBlur(mockText); - expect(mockUpdateGloss).toBeCalledWith(mockText); + expect(mockUpdateGloss).toBeCalledWith(0, mockText); }); }); @@ -136,7 +136,7 @@ describe("ExistingEntry", () => { await act(async () => { testHandle.props.updateText(mockText); }); - expect(mockUpdateNote).toBeCalledWith(mockText); + expect(mockUpdateNote).toBeCalledWith(0, mockText); }); }); }); diff --git a/src/components/DataEntry/DataEntryTable/tests/index.test.tsx b/src/components/DataEntry/DataEntryTable/tests/index.test.tsx index e8e0e70342..5ea0c26f3e 100644 --- a/src/components/DataEntry/DataEntryTable/tests/index.test.tsx +++ b/src/components/DataEntry/DataEntryTable/tests/index.test.tsx @@ -431,7 +431,7 @@ describe("DataEntryTable", () => { await addRecentEntry(); const recentEntry = testRenderer.root.findByType(MockRecentEntry); await act(async () => { - await recentEntry.props.removeEntry(); + await recentEntry.props.removeEntry(recentEntry.props.rowIndex); }); expect(testRenderer.root.findAllByType(MockRecentEntry)).toHaveLength(0); }); @@ -454,7 +454,7 @@ describe("DataEntryTable", () => { // Update the vernacular const newVern = "not the vern generated in addRecentEntry"; await act(async () => { - await recentEntry.props.updateVern(newVern); + await recentEntry.props.updateVern(recentEntry.props.rowIndex, newVern); }); // Confirm the backend update was correctly called From 06f64b8a7052e3f685511c162346ad9d6ead6983 Mon Sep 17 00:00:00 2001 From: "D. Ror" Date: Fri, 3 Nov 2023 11:38:32 -0400 Subject: [PATCH 2/2] Upload Cobertura coverage rather than Clover (#2742) --- .github/workflows/frontend.yml | 4 ++-- package.json | 6 ++++++ 2 files changed, 8 insertions(+), 2 deletions(-) diff --git a/.github/workflows/frontend.yml b/.github/workflows/frontend.yml index 731844e02a..b98e84f796 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 f3296a6e4e..67e5a699bc 100644 --- a/package.json +++ b/package.json @@ -209,6 +209,12 @@ } }, "jest": { + "coverageReporters": [ + "cobertura", + "lcov", + "text", + "text-summary" + ], "transformIgnorePatterns": [ "/node_modules/(!${axios})" ]