From 9a4a0a6473c4d99fc8e68ef9209f34b2c25881b1 Mon Sep 17 00:00:00 2001 From: Laurent Caouissin Date: Thu, 12 Dec 2024 23:05:48 +0100 Subject: [PATCH 01/16] feat: add option on set Variable in store to handle fined row deletion --- .../behaviours/resizing-behaviour.ts | 45 ++++++++++++++----- .../variables/lunatic-variables-store.ts | 4 +- .../reducer/commons/resize-array-variable.ts | 28 ------------ src/use-lunatic/type.ts | 1 + src/utils/array.ts | 24 ++++++++-- 5 files changed, 60 insertions(+), 42 deletions(-) delete mode 100644 src/use-lunatic/reducer/commons/resize-array-variable.ts diff --git a/src/use-lunatic/commons/variables/behaviours/resizing-behaviour.ts b/src/use-lunatic/commons/variables/behaviours/resizing-behaviour.ts index 941de8324..e5c1ee842 100644 --- a/src/use-lunatic/commons/variables/behaviours/resizing-behaviour.ts +++ b/src/use-lunatic/commons/variables/behaviours/resizing-behaviour.ts @@ -1,9 +1,8 @@ import type { LunaticVariablesStore } from '../lunatic-variables-store'; import type { LunaticSource } from '../../../type'; import { forceInt } from '../../../../utils/number'; -import { resizeArrayVariable } from '../../../reducer/commons'; import { getExpressionAsString } from '../../../../utils/vtl'; -import { resizeArray } from '../../../../utils/array'; +import { resizeArray, resizeDownArrayWithIndex } from '../../../../utils/array'; /** * Resizing behaviour for the store @@ -42,8 +41,20 @@ export function resizingBehaviour( const newSize = forceInt(store.run(resizingInfo.size)); for (const variableName of resizingInfo.variables) { const value = store.get(variableName); - if (!Array.isArray(value) || value.length !== newSize) { - store.set(variableName, resizeArrayVariable(value, newSize, null), { + if (Array.isArray(value) && e.detail.removedIndex) { + store.set( + variableName, + resizeDownArrayWithIndex(value, e.detail.removedIndex), + { + cause: 'resizing', + } + ); + } + if ( + !e.detail.removedIndex && + (!Array.isArray(value) || value.length !== newSize) + ) { + store.set(variableName, resizeArray(value, newSize, null), { cause: 'resizing', }); } @@ -61,6 +72,7 @@ function resizePairwise( }, args: { iteration?: number[]; + removedIndex?: number; } ) { // Handle expression being sent as an array or an object (ensure backward compatibility) @@ -78,12 +90,25 @@ function resizePairwise( }); resizingInfo.linksVariables.forEach((variable) => { const value = store.get(variable, args.iteration); - const resizedValue = resizeArray( - // The value is not an array, force an array - Array.isArray(value) ? value.map((i) => resizeArray(i, ySize, null)) : [], - xSize, - new Array(ySize).fill(null) - ); + let resizedValue; + if (args.removedIndex) { + const removedIndex = args.removedIndex; + resizedValue = resizeDownArrayWithIndex( + Array.isArray(value) + ? value.map((i) => resizeDownArrayWithIndex(i, removedIndex)) + : [], + removedIndex + ); + } else { + resizedValue = resizeArray( + // The value is not an array, force an array + Array.isArray(value) + ? value.map((i) => resizeArray(i, ySize, null)) + : [], + xSize, + new Array(ySize).fill(null) + ); + } store.set(variable, resizedValue); }); } diff --git a/src/use-lunatic/commons/variables/lunatic-variables-store.ts b/src/use-lunatic/commons/variables/lunatic-variables-store.ts index 90d48349b..ccfba6d2e 100644 --- a/src/use-lunatic/commons/variables/lunatic-variables-store.ts +++ b/src/use-lunatic/commons/variables/lunatic-variables-store.ts @@ -33,6 +33,8 @@ export type EventArgs = { value: unknown; /** Iteration changed (for array). */ iteration?: IterationLevel | undefined; + /** removedIndex: when resize an array directly with only one handleChange (remove one line in tableLoop) */ + removedIndex?: number; /** What triggered this change. */ cause?: 'resizing' | 'cleaning'; /** Extra sent when setting the variable. */ @@ -123,7 +125,7 @@ export class LunaticVariablesStore { public set( name: string, value: unknown, - args: Pick = {} + args: Pick = {} ): LunaticVariable { if (!this.dictionary.has(name)) { this.dictionary.set( diff --git a/src/use-lunatic/reducer/commons/resize-array-variable.ts b/src/use-lunatic/reducer/commons/resize-array-variable.ts deleted file mode 100644 index ac7765626..000000000 --- a/src/use-lunatic/reducer/commons/resize-array-variable.ts +++ /dev/null @@ -1,28 +0,0 @@ -/** - * Cast the variable into an array and adjust the length if necessary - */ -function resizeArrayVariable( - array: unknown, - length: number, - defaultValue?: T -): T[] { - if (!Array.isArray(array)) { - // create the array - return new Array(length).fill(defaultValue); - } else if (array.length !== length) { - // renew array end keep previous values - return new Array(length).fill(defaultValue).reduce(function ( - step, - current, - index - ) { - if (index < array.length) { - return [...step, array[index]]; - } - return [...step, current]; - }, []); - } - return array; -} - -export default resizeArrayVariable; diff --git a/src/use-lunatic/type.ts b/src/use-lunatic/type.ts index 13cf8a558..b7cbf119e 100644 --- a/src/use-lunatic/type.ts +++ b/src/use-lunatic/type.ts @@ -349,5 +349,6 @@ export type LunaticChangesHandler = ( name: string; value: any; iteration?: number[]; + removedIndex?: number; }[] ) => void; diff --git a/src/utils/array.ts b/src/utils/array.ts index 4dbc0d12b..2e14b5d08 100644 --- a/src/utils/array.ts +++ b/src/utils/array.ts @@ -51,6 +51,9 @@ export function getAtIndex(arr: unknown, indexes: number[]): unknown { return current; } +/** + * Cast the variable into an array and adjust the length if necessary + */ export function resizeArray( array: unknown, newLength: number, @@ -63,14 +66,29 @@ export function resizeArray( if (array.length === newLength) { return array; } - return new Array(newLength).fill(defaultValue ?? null).map(function ( - value, + return new Array(newLength).fill(defaultValue ?? null).reduce(function ( + step, + current, index ) { - return index < array.length ? array[index] : value; + if (index < array.length) { + return [...step, array[index]]; + } + return [...step, current]; }, []); } +export function resizeDownArrayWithIndex( + array: T[], + removedIndex: number +): T[] { + // the removedIndex is not in array + if (0 > removedIndex || array.length <= removedIndex) { + return array; + } + return [...array].filter((_, i) => i !== removedIndex); +} + /** * Return the first non-null/undefined value of an array */ From d42cd49377a8f732a0cf853fdaa26b15580a6736 Mon Sep 17 00:00:00 2001 From: Laurent Caouissin Date: Thu, 12 Dec 2024 23:07:47 +0100 Subject: [PATCH 02/16] feat: add button in Loop & RosterForLoop --- src/components/Loop/Loop.tsx | 54 ++++++++++++++----- .../RosterForLoop/RosterForLoop.tsx | 44 ++++++++++++++- src/i18n/dictionary.ts | 5 ++ 3 files changed, 89 insertions(+), 14 deletions(-) diff --git a/src/components/Loop/Loop.tsx b/src/components/Loop/Loop.tsx index db5682b7a..9460261da 100644 --- a/src/components/Loop/Loop.tsx +++ b/src/components/Loop/Loop.tsx @@ -51,6 +51,30 @@ export function Loop({ } }, [nbRows, handleChanges, value]); + const removeRowWithIndex = useCallback( + (indexToRemove: number) => { + if (nbRows <= min) { + return; + } + /** + * Case 0: trying to delete + */ + if (indexToRemove >= nbRows || indexToRemove < 0) { + return; + } + const newResponses = Object.entries(value).map(([k, v]) => { + return { + name: k, + value: v?.filter((_, i) => i !== indexToRemove), + removedIndex: indexToRemove, + }; + }); + handleChanges(newResponses); + setNbRows((n) => n - 1); + }, + [nbRows, min, value, handleChanges] + ); + if (nbRows <= 0) { return null; } @@ -64,18 +88,23 @@ export function Loop({ canControlRows={min !== max && Number.isFinite(max)} > {times(nbRows, (n) => ( - ({ - ...props, - ...c, - iteration: n, - id: `${c.id}-${n}`, - errors, - })} - /> + <> + ({ + ...props, + ...c, + iteration: n, + id: `${c.id}-${n}`, + errors, + })} + /> + + ))} ); @@ -124,6 +153,7 @@ export const CustomLoop = slottableComponent('Loop', (props) => { {canControlRows && ( <> +
diff --git a/src/components/RosterForLoop/RosterForLoop.tsx b/src/components/RosterForLoop/RosterForLoop.tsx index 7f05f4c93..2057cb237 100644 --- a/src/components/RosterForLoop/RosterForLoop.tsx +++ b/src/components/RosterForLoop/RosterForLoop.tsx @@ -2,6 +2,7 @@ import { Fragment, useCallback, useState } from 'react'; import type { LunaticComponentProps } from '../type'; import { Table, Tbody, Td, Tr, TableHeader } from '../shared/Table'; import { times } from '../../utils/array'; +import D from '../../i18n'; import { LunaticComponents } from '../LunaticComponents'; import { blockedInLoopComponents } from '../Loop/constant'; import { @@ -9,6 +10,7 @@ import { getComponentErrors, } from '../shared/ComponentErrors/ComponentErrors'; import { CustomLoop } from '../Loop/Loop'; +import { Button } from '../shared/Button/Button'; const DEFAULT_MIN_ROWS = 1; const DEFAULT_MAX_ROWS = 12; @@ -44,6 +46,8 @@ export const RosterForLoop = ( } }, [max, nbRows]); + const canRemove = nbRows === min; + const removeRow = useCallback(() => { if (nbRows <= min) { return; @@ -60,22 +64,50 @@ export const RosterForLoop = ( handleChanges(newResponses); }, [nbRows, min, valueMap, handleChanges]); + const removeRowWithIndex = useCallback( + (indexToRemove: number) => { + if (nbRows <= min) { + return; + } + /** + * Case 0: trying to delete + */ + if (indexToRemove >= nbRows || indexToRemove < 0) { + return; + } + const newResponses = Object.entries(valueMap).map(([k, v]) => { + return { + name: k, + value: v?.filter((_, i) => i !== indexToRemove), + removedIndex: indexToRemove, + }; + }); + handleChanges(newResponses); + setNbRows((n) => n - 1); + }, + [nbRows, min, valueMap, handleChanges] + ); + if (nbRows === 0) { return null; } let cols = 0; + const headerWithActions = header + ? [...header, { label: D.ACTION_HEADER }] + : undefined; + return ( - {header && } + {headerWithActions && } {times(nbRows, (n) => { const components = getComponents(n); @@ -104,6 +136,14 @@ export const RosterForLoop = ( })} wrapper={(props) => {hasLineErrors && ( diff --git a/src/i18n/dictionary.ts b/src/i18n/dictionary.ts index beee52b47..528f88b51 100644 --- a/src/i18n/dictionary.ts +++ b/src/i18n/dictionary.ts @@ -1,6 +1,11 @@ const dictionary = { DEFAULT_BUTTON_ADD: { fr: 'Ajouter une ligne', en: 'Add row' }, DEFAULT_BUTTON_REMOVE: { fr: 'Supprimer une ligne', en: 'Remove row' }, + DEFAULT_BUTTON_REMOVE_THAT_ROW: { + fr: 'Supprimer cette ligne', + en: 'Remove that row', + }, + ACTION_HEADER: { fr: 'Action', en: 'Action' }, MODAL_IGNORE: { fr: 'Poursuivre', en: 'Ignore' }, MODAL_CORRECT: { fr: 'Corriger ma réponse', en: 'Correct' }, DK: { fr: 'Ne sais pas', en: "Don't know" }, From d35362ca96667084a80d97f875f793a462c4dcbd Mon Sep 17 00:00:00 2001 From: Laurent Caouissin Date: Thu, 12 Dec 2024 23:08:39 +0100 Subject: [PATCH 03/16] test: adapt pairwise story to row deletion working --- src/stories/pairwise/data.json | 2 +- src/stories/pairwise/pairwise-links.stories.jsx | 2 +- src/stories/pairwise/source.json | 4 +++- 3 files changed, 5 insertions(+), 3 deletions(-) diff --git a/src/stories/pairwise/data.json b/src/stories/pairwise/data.json index 532a502c9..b54810d0c 100644 --- a/src/stories/pairwise/data.json +++ b/src/stories/pairwise/data.json @@ -1,6 +1,6 @@ { "COLLECTED": { - "PRENOM": { "COLLECTED": ["Dad", "Mom", "Unknow"] }, + "PRENOM": { "COLLECTED": ["Dad", "Mom", "Daughter"] }, "AGE": { "COLLECTED": [30, 29, 5] }, "LINKS": { "COLLECTED": [[null]] diff --git a/src/stories/pairwise/pairwise-links.stories.jsx b/src/stories/pairwise/pairwise-links.stories.jsx index a4bfca16f..3c9b9895a 100644 --- a/src/stories/pairwise/pairwise-links.stories.jsx +++ b/src/stories/pairwise/pairwise-links.stories.jsx @@ -21,7 +21,7 @@ Default.args = { source: source, pagination: true, data, - initialPage: '3', + initialPage: '1', }; export const Filled = Template.bind({}); diff --git a/src/stories/pairwise/source.json b/src/stories/pairwise/source.json index 7741435d6..fa245fda0 100644 --- a/src/stories/pairwise/source.json +++ b/src/stories/pairwise/source.json @@ -341,7 +341,9 @@ "resizing": { "PRENOM": { "sizeForLinksVariables": ["count(PRENOM)", "count(PRENOM)"], - "linksVariables": ["LINKS"] + "linksVariables": ["LINKS"], + "size": "count(PRENOM)", + "variables": ["AGE"] } } } From 942f0efe516719bc65d75c7ea26893637ec03703 Mon Sep 17 00:00:00 2001 From: Laurent Caouissin Date: Thu, 12 Dec 2024 23:09:13 +0100 Subject: [PATCH 04/16] test: add test for new resizing function (pairwise included) --- .../variables/lunatic-variables-store.spec.ts | 69 ++++++++++++++++++- src/utils/array.spec.ts | 19 ++++- 2 files changed, 86 insertions(+), 2 deletions(-) diff --git a/src/use-lunatic/commons/variables/lunatic-variables-store.spec.ts b/src/use-lunatic/commons/variables/lunatic-variables-store.spec.ts index 0cda1fc68..b8f25a0dc 100644 --- a/src/use-lunatic/commons/variables/lunatic-variables-store.spec.ts +++ b/src/use-lunatic/commons/variables/lunatic-variables-store.spec.ts @@ -281,6 +281,28 @@ describe('lunatic-variables-store', () => { cause: 'resizing', }); }); + + it('should resize variables with Index', () => { + variables.set('PRENOM', ['John', 'Jane', 'Marc']); + variables.set('AGE', [20, 30, 40]); + const spy = vi.fn(); + variables.on('change', (e) => spy(e.detail)); + resizingBehaviour(variables, { + PRENOM: { + size: 'count(PRENOM)', + variables: ['AGE'], + }, + }); + variables.set('PRENOM', ['John', 'Marc'], { removedIndex: 1 }); + expect((variables.get('PRENOM') as string[]).length).toEqual(2); + expect((variables.get('AGE') as string[]).length).toEqual(2); + expect(spy).toHaveBeenLastCalledWith({ + name: 'AGE', + value: [20, 40], + cause: 'resizing', + }); + }); + it('should resize pairwise with the array syntax', () => { variables.set('PRENOM', []); variables.set('LINKS', [[]]); @@ -316,7 +338,29 @@ describe('lunatic-variables-store', () => { [null, null, null], ]); }); - it('should handle both pairwise and normal resize', () => { + it('should resize pairwise with the object syntax with index', () => { + variables.set('PRENOM', ['John', 'Jane', 'Marc']); + variables.set('LINKS', [ + [null, 2, 4], + [1, null, 2], + [3, 2, null], + ]); + resizingBehaviour(variables, { + PRENOM: { + sizeForLinksVariables: { + xAxisSize: 'count(PRENOM)', + yAxisSize: 'count(PRENOM)', + }, + linksVariables: ['LINKS'], + }, + }); + variables.set('PRENOM', ['John', 'Marc'], { removedIndex: 1 }); + expect(variables.get('LINKS') as string[][]).toEqual([ + [null, 4], + [3, null], + ]); + }); + it('should handle both: pairwise and normal resize', () => { variables.set('PRENOM', []); variables.set('NOM', []); variables.set('LINKS', [[]]); @@ -336,6 +380,29 @@ describe('lunatic-variables-store', () => { ]); expect(variables.get('NOM') as string[]).toEqual([null, null, null]); }); + it('should handle both: pairwise and normal resize with index', () => { + variables.set('PRENOM', ['John', 'Jane', 'Marc']); + variables.set('AGE', [40, 30, 20]); + variables.set('LINKS', [ + [null, 2, 4], + [1, null, 2], + [3, 2, null], + ]); + resizingBehaviour(variables, { + PRENOM: { + sizeForLinksVariables: ['count(PRENOM)', 'count(PRENOM)'], + linksVariables: ['LINKS'], + size: 'count(PRENOM)', + variables: ['AGE'], + }, + }); + variables.set('PRENOM', ['John', 'Marc'], { removedIndex: 1 }); + expect(variables.get('LINKS') as string[][]).toEqual([ + [null, 4], + [3, null], + ]); + expect(variables.get('AGE') as string[]).toEqual([40, 20]); + }); }); describe('cleaning', () => { diff --git a/src/utils/array.spec.ts b/src/utils/array.spec.ts index aa60a0194..ed8302ec8 100644 --- a/src/utils/array.spec.ts +++ b/src/utils/array.spec.ts @@ -1,5 +1,10 @@ import { describe, it, expect } from 'vitest'; -import { firstValueItem, resizeArray, setAtIndex } from './array'; +import { + firstValueItem, + resizeArray, + resizeDownArrayWithIndex, + setAtIndex, +} from './array'; describe('array', () => { describe('resizeArray()', () => { @@ -45,4 +50,16 @@ describe('array', () => { expect(firstValueItem([null, 1, 2])).toBe(1); expect(firstValueItem([null, undefined, false])).toBe(false); }); + describe('resizeDownArrayWithIndex()', () => { + it('should remove an element of array', () => { + expect(resizeDownArrayWithIndex([1, 2, 3, 4], 2)).toEqual([1, 2, 4]); + expect(resizeDownArrayWithIndex([1, 2, 3, 4], 0)).toEqual([2, 3, 4]); + expect(resizeDownArrayWithIndex([1, 2, 3, 4], 3)).toEqual([1, 2, 3]); + }); + + it('should not remove element (out of index)', () => { + expect(resizeDownArrayWithIndex([1, 2, 3, 4], -1)).toEqual([1, 2, 3, 4]); + expect(resizeDownArrayWithIndex([1, 2, 3, 4], 4)).toEqual([1, 2, 3, 4]); + }); + }); }); From 59476488a89e6673bc54c6776c9e6e7202fcfbef Mon Sep 17 00:00:00 2001 From: Laurent Caouissin Date: Thu, 12 Dec 2024 23:13:15 +0100 Subject: [PATCH 05/16] test: update snapshot --- src/components/Loop/Loop.tsx | 2 +- .../__snapshots__/RosterForLoop.spec.tsx.snap | 18 + .../__snapshots__/use-lunatic.test.ts.snap | 323 ------------------ 3 files changed, 19 insertions(+), 324 deletions(-) diff --git a/src/components/Loop/Loop.tsx b/src/components/Loop/Loop.tsx index 9460261da..704b62323 100644 --- a/src/components/Loop/Loop.tsx +++ b/src/components/Loop/Loop.tsx @@ -106,6 +106,7 @@ export function Loop({ ))} +
); } @@ -153,7 +154,6 @@ export const CustomLoop = slottableComponent('Loop', (props) => { {canControlRows && ( <> -
diff --git a/src/components/RosterForLoop/__snapshots__/RosterForLoop.spec.tsx.snap b/src/components/RosterForLoop/__snapshots__/RosterForLoop.spec.tsx.snap index 67d48fcc5..7bbf8255d 100644 --- a/src/components/RosterForLoop/__snapshots__/RosterForLoop.spec.tsx.snap +++ b/src/components/RosterForLoop/__snapshots__/RosterForLoop.spec.tsx.snap @@ -45,6 +45,15 @@ exports[`RosterForLoop > renders the right number of columns 1`] = ` +
renders the right number of columns 1`] = ` +
} /> + + +
+ +
+ +
diff --git a/src/use-lunatic/__snapshots__/use-lunatic.test.ts.snap b/src/use-lunatic/__snapshots__/use-lunatic.test.ts.snap index f47f7134f..a78032a56 100644 --- a/src/use-lunatic/__snapshots__/use-lunatic.test.ts.snap +++ b/src/use-lunatic/__snapshots__/use-lunatic.test.ts.snap @@ -1855,329 +1855,6 @@ exports[`use-lunatic() > overview > with loop > should handle initialPage 1`] = ] `; -exports[`use-lunatic() > overview > with loop > should handle lastReachedPage 1`] = ` -[ - { - "children": [ - { - "children": [], - "current": false, - "description": undefined, - "id": "lujqeci5", - "label": , - "page": "2", - "reached": true, - "type": "Subsequence", - }, - { - "children": [], - "current": false, - "description": undefined, - "id": "lujqbyzl", - "label": , - "page": "3", - "reached": true, - "type": "Subsequence", - }, - ], - "current": true, - "description": undefined, - "id": "lujqfpva", - "label": , - "page": "1", - "reached": true, - "type": "Sequence", - }, - { - "children": [ - { - "children": [], - "current": false, - "description": undefined, - "id": "lulbmyhr", - "label": , - "page": "8", - "reached": true, - "type": "Subsequence", - }, - ], - "current": false, - "description": undefined, - "id": "lujqrqmp", - "label": , - "page": "5", - "reached": true, - "type": "Sequence", - }, - { - "children": [ - { - "children": [], - "current": false, - "description": undefined, - "id": "lujykwaz", - "label": , - "page": "10.1#1", - "reached": true, - "type": "Subsequence", - }, - { - "children": [], - "current": false, - "description": undefined, - "id": "lujykwaz", - "label": , - "page": "10.1#2", - "reached": true, - "type": "Subsequence", - }, - { - "children": [], - "current": false, - "description": undefined, - "id": "lujykwaz", - "label": , - "page": "10.1#3", - "reached": true, - "type": "Subsequence", - }, - { - "children": [], - "current": false, - "description": undefined, - "id": "lujyik5q", - "label": , - "page": "11.1#2", - "reached": true, - "type": "Subsequence", - }, - { - "children": [], - "current": false, - "description": undefined, - "id": "luk0swcz", - "label": , - "page": "11.2#2", - "reached": true, - "type": "Subsequence", - }, - { - "children": [], - "current": false, - "description": undefined, - "id": "lujyik5q", - "label": , - "page": "11.1#3", - "reached": false, - "type": "Subsequence", - }, - { - "children": [], - "current": false, - "description": undefined, - "id": "luk0swcz", - "label": , - "page": "11.2#3", - "reached": false, - "type": "Subsequence", - }, - ], - "current": false, - "description": undefined, - "id": "lujyi4pe", - "label": , - "page": "9", - "reached": true, - "type": "Sequence", - }, - { - "children": [ - { - "children": [], - "current": false, - "description": undefined, - "id": "lumfc98o", - "label": , - "page": "12.2#1", - "reached": false, - "type": "Subsequence", - }, - { - "children": [], - "current": false, - "description": undefined, - "id": "lumfe3bj", - "label": , - "page": "12.3#1", - "reached": false, - "type": "Subsequence", - }, - ], - "current": false, - "description": undefined, - "id": "luk1ojt5", - "label": , - "page": "12.1#1", - "reached": false, - "type": "Sequence", - }, - { - "children": [], - "current": false, - "description": undefined, - "id": "lulbelgr", - "label": , - "page": "12.4#1", - "reached": false, - "type": "Sequence", - }, - { - "children": [ - { - "children": [], - "current": false, - "description": undefined, - "id": "lumfc98o", - "label": , - "page": "12.2#2", - "reached": false, - "type": "Subsequence", - }, - { - "children": [], - "current": false, - "description": undefined, - "id": "lumfe3bj", - "label": , - "page": "12.3#2", - "reached": false, - "type": "Subsequence", - }, - ], - "current": false, - "description": undefined, - "id": "luk1ojt5", - "label": , - "page": "12.1#2", - "reached": false, - "type": "Sequence", - }, - { - "children": [], - "current": false, - "description": undefined, - "id": "lulbelgr", - "label": , - "page": "12.4#2", - "reached": false, - "type": "Sequence", - }, - { - "children": [ - { - "children": [], - "current": false, - "description": undefined, - "id": "lumfc98o", - "label": , - "page": "12.2#3", - "reached": false, - "type": "Subsequence", - }, - { - "children": [], - "current": false, - "description": undefined, - "id": "lumfe3bj", - "label": , - "page": "12.3#3", - "reached": false, - "type": "Subsequence", - }, - ], - "current": false, - "description": undefined, - "id": "luk1ojt5", - "label": , - "page": "12.1#3", - "reached": false, - "type": "Sequence", - }, - { - "children": [], - "current": false, - "description": undefined, - "id": "lulbelgr", - "label": , - "page": "12.4#3", - "reached": false, - "type": "Sequence", - }, - { - "children": [], - "current": false, - "description": undefined, - "id": "COMMENT-SEQ", - "label": , - "page": "13", - "reached": false, - "type": "Sequence", - }, -] -`; - exports[`use-lunatic() > overview > with loop > should work with loop 1`] = ` [ { From 3bc5168f0473c5a3355ed88dca68e0ede158d553 Mon Sep 17 00:00:00 2001 From: Laurent Caouissin Date: Fri, 13 Dec 2024 09:14:16 +0100 Subject: [PATCH 06/16] bump: 3.4.11-rc.0 --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index ea1247195..913ce7767 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "@inseefr/lunatic", - "version": "3.4.10", + "version": "3.4.11-rc.0", "description": "Library of questionnaire components", "repository": { "type": "git", From 390feba0748e92433d9297f568b2733918ed21b5 Mon Sep 17 00:00:00 2001 From: Laurent Caouissin Date: Fri, 13 Dec 2024 09:26:59 +0100 Subject: [PATCH 07/16] chore: format & lint --- src/components/RosterForLoop/RosterForLoop.tsx | 6 +++--- src/type.source.ts | 1 - src/use-lunatic/reducer/commons/index.ts | 1 - 3 files changed, 3 insertions(+), 5 deletions(-) diff --git a/src/components/RosterForLoop/RosterForLoop.tsx b/src/components/RosterForLoop/RosterForLoop.tsx index 2057cb237..8b00f32cb 100644 --- a/src/components/RosterForLoop/RosterForLoop.tsx +++ b/src/components/RosterForLoop/RosterForLoop.tsx @@ -46,7 +46,7 @@ export const RosterForLoop = ( } }, [max, nbRows]); - const canRemove = nbRows === min; + const cantRemove = nbRows === min; const removeRow = useCallback(() => { if (nbRows <= min) { @@ -103,7 +103,7 @@ export const RosterForLoop = ( {...props} errors={getComponentErrors(errors, props.id)} addRow={nbRows === max ? undefined : addRow} - removeRow={canRemove ? undefined : removeRow} + removeRow={cantRemove ? undefined : removeRow} canControlRows={!!(min && max && min !== max)} > @@ -139,7 +139,7 @@ export const RosterForLoop = (
diff --git a/src/type.source.ts b/src/type.source.ts index 0fff66c92..bd2117e70 100644 --- a/src/type.source.ts +++ b/src/type.source.ts @@ -1,4 +1,3 @@ -/* eslint-disable */ /** * This file was automatically generated by json-schema-to-typescript. * DO NOT MODIFY IT BY HAND. Instead, modify the source JSONSchema file, diff --git a/src/use-lunatic/reducer/commons/index.ts b/src/use-lunatic/reducer/commons/index.ts index 80b2cadf0..5bedc64fa 100644 --- a/src/use-lunatic/reducer/commons/index.ts +++ b/src/use-lunatic/reducer/commons/index.ts @@ -1,2 +1 @@ -export { default as resizeArrayVariable } from './resize-array-variable'; export * from './validate-condition-filter'; From 96999e2e311c10184abc5de54f23f8d97ad6a9c5 Mon Sep 17 00:00:00 2001 From: Laurent Caouissin <38245508+laurentC35@users.noreply.github.com> Date: Fri, 13 Dec 2024 10:13:28 +0100 Subject: [PATCH 08/16] test: fix e2e pairwise --- src/stories/pairwise/pairwise-links.stories.jsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/stories/pairwise/pairwise-links.stories.jsx b/src/stories/pairwise/pairwise-links.stories.jsx index 3c9b9895a..a4bfca16f 100644 --- a/src/stories/pairwise/pairwise-links.stories.jsx +++ b/src/stories/pairwise/pairwise-links.stories.jsx @@ -21,7 +21,7 @@ Default.args = { source: source, pagination: true, data, - initialPage: '1', + initialPage: '3', }; export const Filled = Template.bind({}); From 817c25a53ba9e24d02daca4a92ef2b588130248c Mon Sep 17 00:00:00 2001 From: Laurent Caouissin Date: Fri, 13 Dec 2024 12:16:29 +0100 Subject: [PATCH 09/16] chore: ref (codeReview) --- src/components/Loop/Loop.tsx | 6 ++---- src/components/RosterForLoop/RosterForLoop.tsx | 10 ++++------ .../__snapshots__/RosterForLoop.spec.tsx.snap | 4 ++-- src/i18n/dictionary.ts | 4 ++-- .../commons/variables/behaviours/resizing-behaviour.ts | 3 +-- .../commons/variables/lunatic-variables-store.ts | 2 +- 6 files changed, 12 insertions(+), 17 deletions(-) diff --git a/src/components/Loop/Loop.tsx b/src/components/Loop/Loop.tsx index 704b62323..33428ef90 100644 --- a/src/components/Loop/Loop.tsx +++ b/src/components/Loop/Loop.tsx @@ -56,9 +56,7 @@ export function Loop({ if (nbRows <= min) { return; } - /** - * Case 0: trying to delete - */ + // Case 0: trying to delete with wrong index if (indexToRemove >= nbRows || indexToRemove < 0) { return; } @@ -102,7 +100,7 @@ export function Loop({ })} /> ))} diff --git a/src/components/RosterForLoop/RosterForLoop.tsx b/src/components/RosterForLoop/RosterForLoop.tsx index 8b00f32cb..9a52834de 100644 --- a/src/components/RosterForLoop/RosterForLoop.tsx +++ b/src/components/RosterForLoop/RosterForLoop.tsx @@ -94,10 +94,6 @@ export const RosterForLoop = ( let cols = 0; - const headerWithActions = header - ? [...header, { label: D.ACTION_HEADER }] - : undefined; - return ( - {headerWithActions && } + {header && ( + + )} {times(nbRows, (n) => { const components = getComponents(n); @@ -141,7 +139,7 @@ export const RosterForLoop = ( onClick={() => removeRowWithIndex(n)} disabled={cantRemove} > - {D.DEFAULT_BUTTON_REMOVE_THAT_ROW} + {D.DEFAULT_BUTTON_REMOVE_THIS_ROW} diff --git a/src/components/RosterForLoop/__snapshots__/RosterForLoop.spec.tsx.snap b/src/components/RosterForLoop/__snapshots__/RosterForLoop.spec.tsx.snap index 7bbf8255d..657b7a5b2 100644 --- a/src/components/RosterForLoop/__snapshots__/RosterForLoop.spec.tsx.snap +++ b/src/components/RosterForLoop/__snapshots__/RosterForLoop.spec.tsx.snap @@ -51,7 +51,7 @@ exports[`RosterForLoop > renders the right number of columns 1`] = ` @@ -90,7 +90,7 @@ exports[`RosterForLoop > renders the right number of columns 1`] = ` diff --git a/src/i18n/dictionary.ts b/src/i18n/dictionary.ts index 528f88b51..612fca754 100644 --- a/src/i18n/dictionary.ts +++ b/src/i18n/dictionary.ts @@ -1,9 +1,9 @@ const dictionary = { DEFAULT_BUTTON_ADD: { fr: 'Ajouter une ligne', en: 'Add row' }, DEFAULT_BUTTON_REMOVE: { fr: 'Supprimer une ligne', en: 'Remove row' }, - DEFAULT_BUTTON_REMOVE_THAT_ROW: { + DEFAULT_BUTTON_REMOVE_THIS_ROW: { fr: 'Supprimer cette ligne', - en: 'Remove that row', + en: 'Remove this row', }, ACTION_HEADER: { fr: 'Action', en: 'Action' }, MODAL_IGNORE: { fr: 'Poursuivre', en: 'Ignore' }, diff --git a/src/use-lunatic/commons/variables/behaviours/resizing-behaviour.ts b/src/use-lunatic/commons/variables/behaviours/resizing-behaviour.ts index e5c1ee842..5c02e4da4 100644 --- a/src/use-lunatic/commons/variables/behaviours/resizing-behaviour.ts +++ b/src/use-lunatic/commons/variables/behaviours/resizing-behaviour.ts @@ -49,8 +49,7 @@ export function resizingBehaviour( cause: 'resizing', } ); - } - if ( + } else if ( !e.detail.removedIndex && (!Array.isArray(value) || value.length !== newSize) ) { diff --git a/src/use-lunatic/commons/variables/lunatic-variables-store.ts b/src/use-lunatic/commons/variables/lunatic-variables-store.ts index ccfba6d2e..f4fa43754 100644 --- a/src/use-lunatic/commons/variables/lunatic-variables-store.ts +++ b/src/use-lunatic/commons/variables/lunatic-variables-store.ts @@ -33,7 +33,7 @@ export type EventArgs = { value: unknown; /** Iteration changed (for array). */ iteration?: IterationLevel | undefined; - /** removedIndex: when resize an array directly with only one handleChange (remove one line in tableLoop) */ + /** When resize an array directly with only one handleChange (remove one line in tableLoop) */ removedIndex?: number; /** What triggered this change. */ cause?: 'resizing' | 'cleaning'; From 6f2ad893b87ee7ccfd12385e32e689d5cf0ee1cf Mon Sep 17 00:00:00 2001 From: Laurent Caouissin Date: Mon, 16 Dec 2024 08:51:26 +0100 Subject: [PATCH 10/16] Fix: display button only if min != max for button --- src/components/Loop/Loop.tsx | 15 +++++++++++---- 1 file changed, 11 insertions(+), 4 deletions(-) diff --git a/src/components/Loop/Loop.tsx b/src/components/Loop/Loop.tsx index 33428ef90..feb795e1c 100644 --- a/src/components/Loop/Loop.tsx +++ b/src/components/Loop/Loop.tsx @@ -77,13 +77,15 @@ export function Loop({ return null; } + const canControlRows = min !== max && Number.isFinite(max); + return ( {times(nbRows, (n) => ( <> @@ -99,9 +101,14 @@ export function Loop({ errors, })} /> - + {canControlRows && ( + {hasLineErrors && ( diff --git a/src/components/RosterForLoop/__snapshots__/RosterForLoop.spec.tsx.snap b/src/components/RosterForLoop/__snapshots__/RosterForLoop.spec.tsx.snap index 657b7a5b2..a2b55866b 100644 --- a/src/components/RosterForLoop/__snapshots__/RosterForLoop.spec.tsx.snap +++ b/src/components/RosterForLoop/__snapshots__/RosterForLoop.spec.tsx.snap @@ -50,6 +50,7 @@ exports[`RosterForLoop > renders the right number of columns 1`] = ` > @@ -89,6 +90,7 @@ exports[`RosterForLoop > renders the right number of columns 1`] = ` > From 30285859587ead6996e34e022e9a1040fbf02e0d Mon Sep 17 00:00:00 2001 From: Laurent Caouissin Date: Mon, 16 Dec 2024 13:09:19 +0100 Subject: [PATCH 12/16] fix: condition when removedIndex is 0 --- .../commons/variables/behaviours/resizing-behaviour.ts | 4 ++-- .../commons/variables/lunatic-variables-store.spec.ts | 8 ++++++++ 2 files changed, 10 insertions(+), 2 deletions(-) diff --git a/src/use-lunatic/commons/variables/behaviours/resizing-behaviour.ts b/src/use-lunatic/commons/variables/behaviours/resizing-behaviour.ts index 5c02e4da4..8e9b8a51d 100644 --- a/src/use-lunatic/commons/variables/behaviours/resizing-behaviour.ts +++ b/src/use-lunatic/commons/variables/behaviours/resizing-behaviour.ts @@ -41,7 +41,7 @@ export function resizingBehaviour( const newSize = forceInt(store.run(resizingInfo.size)); for (const variableName of resizingInfo.variables) { const value = store.get(variableName); - if (Array.isArray(value) && e.detail.removedIndex) { + if (Array.isArray(value) && e.detail.removedIndex !== undefined) { store.set( variableName, resizeDownArrayWithIndex(value, e.detail.removedIndex), @@ -50,7 +50,7 @@ export function resizingBehaviour( } ); } else if ( - !e.detail.removedIndex && + e.detail.removedIndex === undefined && (!Array.isArray(value) || value.length !== newSize) ) { store.set(variableName, resizeArray(value, newSize, null), { diff --git a/src/use-lunatic/commons/variables/lunatic-variables-store.spec.ts b/src/use-lunatic/commons/variables/lunatic-variables-store.spec.ts index b8f25a0dc..6d5d5bc8e 100644 --- a/src/use-lunatic/commons/variables/lunatic-variables-store.spec.ts +++ b/src/use-lunatic/commons/variables/lunatic-variables-store.spec.ts @@ -301,6 +301,14 @@ describe('lunatic-variables-store', () => { value: [20, 40], cause: 'resizing', }); + variables.set('PRENOM', ['Marc'], { removedIndex: 0 }); + expect((variables.get('PRENOM') as string[]).length).toEqual(1); + expect((variables.get('AGE') as string[]).length).toEqual(1); + expect(spy).toHaveBeenLastCalledWith({ + name: 'AGE', + value: [40], + cause: 'resizing', + }); }); it('should resize pairwise with the array syntax', () => { From 81758160631935d4acdea226a3be2650060f75eb Mon Sep 17 00:00:00 2001 From: Laurent Caouissin Date: Wed, 8 Jan 2025 15:18:44 +0100 Subject: [PATCH 13/16] chore: code review --- src/components/Loop/Loop.tsx | 9 ++------- src/utils/array.ts | 2 +- 2 files changed, 3 insertions(+), 8 deletions(-) diff --git a/src/components/Loop/Loop.tsx b/src/components/Loop/Loop.tsx index feb795e1c..2c7ab5cbe 100644 --- a/src/components/Loop/Loop.tsx +++ b/src/components/Loop/Loop.tsx @@ -56,10 +56,6 @@ export function Loop({ if (nbRows <= min) { return; } - // Case 0: trying to delete with wrong index - if (indexToRemove >= nbRows || indexToRemove < 0) { - return; - } const newResponses = Object.entries(value).map(([k, v]) => { return { name: k, @@ -111,7 +107,6 @@ export function Loop({ )} ))} -
); } @@ -158,14 +153,14 @@ export const CustomLoop = slottableComponent('Loop', (props) => { {children} {canControlRows && ( - <> +
- +
)} ); diff --git a/src/utils/array.ts b/src/utils/array.ts index 2e14b5d08..1807f43a0 100644 --- a/src/utils/array.ts +++ b/src/utils/array.ts @@ -86,7 +86,7 @@ export function resizeDownArrayWithIndex( if (0 > removedIndex || array.length <= removedIndex) { return array; } - return [...array].filter((_, i) => i !== removedIndex); + return array.filter((_, i) => i !== removedIndex); } /** From 4ae1ff3c723ade2eaf15c51791328378d034a377 Mon Sep 17 00:00:00 2001 From: Laurent Caouissin Date: Wed, 8 Jan 2025 15:20:44 +0100 Subject: [PATCH 14/16] bump: 3.4.12-rc.0 --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index d66e8ffbb..5b1f894ac 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "@inseefr/lunatic", - "version": "3.4.11", + "version": "3.4.12-rc.0", "description": "Library of questionnaire components", "repository": { "type": "git", From e2f002a1948555f1a7d0e32fd18773d387572f88 Mon Sep 17 00:00:00 2001 From: Laurent Caouissin Date: Wed, 8 Jan 2025 15:35:28 +0100 Subject: [PATCH 15/16] test: update snapshot --- .../__snapshots__/RosterForLoop.spec.tsx.snap | 24 +++++++++++-------- 1 file changed, 14 insertions(+), 10 deletions(-) diff --git a/src/components/RosterForLoop/__snapshots__/RosterForLoop.spec.tsx.snap b/src/components/RosterForLoop/__snapshots__/RosterForLoop.spec.tsx.snap index a2b55866b..4215a6647 100644 --- a/src/components/RosterForLoop/__snapshots__/RosterForLoop.spec.tsx.snap +++ b/src/components/RosterForLoop/__snapshots__/RosterForLoop.spec.tsx.snap @@ -98,15 +98,19 @@ exports[`RosterForLoop > renders the right number of columns 1`] = `
+ />
- - +
+ + +
`; From ed1e41df616d4fc2c840f58dc56806abca1d292e Mon Sep 17 00:00:00 2001 From: Laurent Caouissin Date: Thu, 9 Jan 2025 15:10:24 +0100 Subject: [PATCH 16/16] fix: resizing pairwise when removedIndex is 0 --- .../behaviours/resizing-behaviour.ts | 2 +- .../variables/lunatic-variables-store.spec.ts | 21 +++++++++++++++++++ 2 files changed, 22 insertions(+), 1 deletion(-) diff --git a/src/use-lunatic/commons/variables/behaviours/resizing-behaviour.ts b/src/use-lunatic/commons/variables/behaviours/resizing-behaviour.ts index 8e9b8a51d..a718c36ad 100644 --- a/src/use-lunatic/commons/variables/behaviours/resizing-behaviour.ts +++ b/src/use-lunatic/commons/variables/behaviours/resizing-behaviour.ts @@ -90,7 +90,7 @@ function resizePairwise( resizingInfo.linksVariables.forEach((variable) => { const value = store.get(variable, args.iteration); let resizedValue; - if (args.removedIndex) { + if (args.removedIndex !== undefined) { const removedIndex = args.removedIndex; resizedValue = resizeDownArrayWithIndex( Array.isArray(value) diff --git a/src/use-lunatic/commons/variables/lunatic-variables-store.spec.ts b/src/use-lunatic/commons/variables/lunatic-variables-store.spec.ts index 6d5d5bc8e..baec44525 100644 --- a/src/use-lunatic/commons/variables/lunatic-variables-store.spec.ts +++ b/src/use-lunatic/commons/variables/lunatic-variables-store.spec.ts @@ -388,6 +388,27 @@ describe('lunatic-variables-store', () => { ]); expect(variables.get('NOM') as string[]).toEqual([null, null, null]); }); + it('should handle both: pairwise resize with index 0', () => { + variables.set('PRENOM', ['John', 'Jane', 'Marc']); + variables.set('LINKS', [ + [null, 2, 4], + [1, null, 2], + [3, 2, null], + ]); + resizingBehaviour(variables, { + PRENOM: { + sizeForLinksVariables: ['count(PRENOM)', 'count(PRENOM)'], + linksVariables: ['LINKS'], + size: 'count(PRENOM)', + }, + }); + variables.set('PRENOM', ['John', 'Marc'], { removedIndex: 0 }); + expect(variables.get('LINKS') as string[][]).toEqual([ + [null, 2], + [2, null], + ]); + }); + it('should handle both: pairwise and normal resize with index', () => { variables.set('PRENOM', ['John', 'Jane', 'Marc']); variables.set('AGE', [40, 30, 20]);