Skip to content

Commit 59eabef

Browse files
MagnusrmlorangSec
andauthoredJun 16, 2022
Bugfix/duplicate validation messages shown (#187)
* only show first validation message in message array * dont run updateFormLayout while unsaved data has been saved * change format of conditional rendering * remove console logging * use flag in layoutsettings to toggle workaround * update component tests with new state * re-resolve merge conflict * change ternary operation for if * formatting * add test for watchUpdateCurrentViewSaga * test: fixed testt + a little refactor * test watchUpdateCurrentViewSaga no unsaved changes * typo * resolve merge conflicts * revert message rendering changes * require regex match to start with whole component-id * update regex and add regressiontests Co-authored-by: EKEBERG Steffen <[email protected]>
1 parent 5abba85 commit 59eabef

File tree

8 files changed

+154
-12
lines changed

8 files changed

+154
-12
lines changed
 

‎src/altinn-app-frontend/src/components/advanced/AddressComponent.test.tsx

+3-3
Original file line numberDiff line numberDiff line change
@@ -60,9 +60,9 @@ const render = (props: Partial<IAddressComponentProps> = {}) => {
6060
const mockStore = createStore({ language: { language: mockLanguage } });
6161

6262
rtlRender(
63-
<Provider store={mockStore}>
64-
<AddressComponent {...allProps} />
65-
</Provider>);
63+
<Provider store={mockStore}>
64+
<AddressComponent {...allProps} />
65+
</Provider>);
6666
};
6767

6868
const getField = ({ method, regex }) =>

‎src/altinn-app-frontend/src/components/base/FileUpload/FileUploadWithTag/EditWindowComponent.tsx

+5-1
Original file line numberDiff line numberDiff line change
@@ -186,7 +186,11 @@ export function EditWindowComponent(props: EditWindowProps): JSX.Element {
186186
whiteSpace: 'pre-wrap',
187187
}}
188188
>
189-
{renderValidationMessages(props.attachmentValidations.filter((i) => i.id === props.attachment.id).map((e) => { return e.message; }), `attachment-error-${props.attachment.id}`, 'error')}
189+
{renderValidationMessages(
190+
props.attachmentValidations.filter((i) => i.id === props.attachment.id).map((e) => { return e.message; }),
191+
`attachment-error-${props.attachment.id}`,
192+
'error',
193+
)}
190194
</div>
191195
: undefined
192196
}

‎src/altinn-app-frontend/src/components/base/LikertComponent.tsx

-1
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,6 @@ import {
1010
import { IRadioButtonsContainerProps } from 'src/components/base/RadioButtons/RadioButtonsContainerComponent';
1111
import { LayoutStyle } from 'src/types';
1212

13-
1413
export const LikertComponent = (props: IRadioButtonsContainerProps) => {
1514
const { layout } = props;
1615
const useRadioProps = useRadioButtons(props);

‎src/altinn-app-frontend/src/features/form/containers/GroupContainer.tsx

+3-1
Original file line numberDiff line numberDiff line change
@@ -83,7 +83,9 @@ export function GroupContainer({
8383
const currentView = useAppSelector(
8484
(state) => state.formLayout.uiConfig.currentView,
8585
);
86-
const language = useAppSelector((state) => state.language.language);
86+
const language = useAppSelector(
87+
(state) => state.language.language
88+
);
8789
const repeatingGroups = useAppSelector(
8890
(state) => state.formLayout.uiConfig.repeatingGroups,
8991
);

‎src/altinn-app-frontend/src/features/form/layout/update/updateFormLayoutSagas.test.ts

+74-2
Original file line numberDiff line numberDiff line change
@@ -1,16 +1,19 @@
11
import { expectSaga, testSaga } from 'redux-saga-test-plan';
2-
import { select } from 'redux-saga/effects';
2+
import { actionChannel, call, select } from 'redux-saga/effects';
33

44
import FormDataActions from 'src/features/form/data/formDataActions';
5-
import { FormLayoutActions } from 'src/features/form/layout/formLayoutSlice';
65
import { getInitialStateMock } from '__mocks__/initialStateMock';
76
import * as sharedUtils from 'altinn-shared/utils';
87
import {
98
calculatePageOrderAndMoveToNextPageSaga,
109
initRepeatingGroupsSaga,
1110
watchInitRepeatingGroupsSaga,
11+
watchUpdateCurrentViewSaga,
12+
updateCurrentViewSaga,
13+
selectUnsavedChanges
1214
} from './updateFormLayoutSagas';
1315
import { IRuntimeState } from 'src/types';
16+
import { FormLayoutActions } from '../formLayoutSlice';
1417

1518
jest.mock('altinn-shared/utils');
1619

@@ -37,6 +40,75 @@ describe('updateLayoutSagas', () => {
3740
});
3841
});
3942

43+
describe('watchUpdateCurrentViewSaga', () => {
44+
it('should save unsaved changes before updating from layout', () => {
45+
const fakeChannel = {
46+
take() { /* Intentionally empty */ },
47+
flush() { /* Intentionally empty */ },
48+
close() { /* Intentionally empty */},
49+
};
50+
51+
const mockAction = FormLayoutActions.updateCurrentView({
52+
newView: 'test'
53+
});
54+
55+
const mockSaga = function*() { /* intentially empty */};
56+
57+
return expectSaga(watchUpdateCurrentViewSaga)
58+
.provide([
59+
[actionChannel(FormLayoutActions.updateCurrentView), fakeChannel],
60+
[select(selectUnsavedChanges), true],
61+
{
62+
take({ channel }, next) {
63+
if (channel === fakeChannel) {
64+
return mockAction;
65+
}
66+
return next();
67+
},
68+
},
69+
[call(updateCurrentViewSaga, mockAction), mockSaga]
70+
])
71+
.dispatch(FormLayoutActions.updateCurrentView)
72+
.dispatch(FormDataActions.submitFormDataFulfilled)
73+
.take(fakeChannel)
74+
.call(updateCurrentViewSaga, mockAction)
75+
.run();
76+
});
77+
it('should not save unsaved changes before updating form layout when no unsaved changes', () => {
78+
const fakeChannel = {
79+
take() { /* Intentionally empty */ },
80+
flush() { /* Intentionally empty */ },
81+
close() { /* Intentionally empty */},
82+
};
83+
84+
const mockAction = FormLayoutActions.updateCurrentView({
85+
newView: 'test'
86+
});
87+
88+
const mockSaga = function*() { /* intentially empty */};
89+
90+
return expectSaga(watchUpdateCurrentViewSaga)
91+
.provide([
92+
[actionChannel(FormLayoutActions.updateCurrentView), fakeChannel],
93+
[select(selectUnsavedChanges), false],
94+
{
95+
take({ channel }, next) {
96+
if (channel === fakeChannel) {
97+
return mockAction;
98+
}
99+
return next();
100+
},
101+
},
102+
[call(updateCurrentViewSaga, mockAction), mockSaga]
103+
])
104+
.dispatch(FormLayoutActions.updateCurrentView)
105+
.not.take(FormDataActions.submitFormDataFulfilled)
106+
.take(fakeChannel)
107+
.call(updateCurrentViewSaga, mockAction)
108+
.run();
109+
});
110+
});
111+
40112
describe('calculatePageOrderAndMoveToNextPageSaga', () => {
41113
const state = getInitialStateMock();
42114
const orderResponse = ['page-1', 'FormLayout', 'page-3'];

‎src/altinn-app-frontend/src/features/form/layout/update/updateFormLayoutSagas.ts

+12-2
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
/* eslint-disable max-len */
22
import { PayloadAction } from '@reduxjs/toolkit';
33
import { SagaIterator } from 'redux-saga';
4-
import { all, call, put, select, take, takeEvery, takeLatest } from 'redux-saga/effects';
4+
import { actionChannel, all, call, put, select, take, takeEvery, takeLatest } from 'redux-saga/effects';
55
import { IFileUploadersWithTag, IFormFileUploaderWithTagComponent, IRepeatingGroups, IRuntimeState, IValidationIssue, IValidations, Triggers } from 'src/types';
66
import { getFileUploadersWithTag, getRepeatingGroups, removeRepeatingGroupFromUIConfig } from 'src/utils/formLayout';
77
import { AxiosRequestConfig } from 'axios';
@@ -26,6 +26,7 @@ const selectFormLayoutState = (state: IRuntimeState): ILayoutState => state.form
2626
const selectFormData = (state: IRuntimeState): IFormDataState => state.formData;
2727
const selectFormLayouts = (state: IRuntimeState): ILayouts => state.formLayout.layouts;
2828
const selectAttachmentState = (state: IRuntimeState): IAttachmentState => state.attachments;
29+
export const selectUnsavedChanges = (state: IRuntimeState): boolean => state.formData.unsavedChanges;
2930

3031
function* updateFocus({ payload: { currentComponentId, step } }: PayloadAction<IUpdateFocus>): SagaIterator {
3132
try {
@@ -292,7 +293,16 @@ export function* watchInitialCalculagePageOrderAndMoveToNextPageSaga(): SagaIter
292293
}
293294

294295
export function* watchUpdateCurrentViewSaga(): SagaIterator {
295-
yield takeEvery(FormLayoutActions.updateCurrentView, updateCurrentViewSaga);
296+
const requestChan = yield actionChannel(FormLayoutActions.updateCurrentView);
297+
while (true) {
298+
yield take(FormLayoutActions.updateCurrentView);
299+
const hasUnsavedChanges = yield select(selectUnsavedChanges);
300+
if (hasUnsavedChanges) {
301+
yield take(FormDataActions.submitFormDataFulfilled);
302+
}
303+
const value = yield take(requestChan);
304+
yield call(updateCurrentViewSaga, value);
305+
}
296306
}
297307

298308
export function* watchUpdateFocusSaga(): SagaIterator {

‎src/altinn-app-frontend/src/utils/layout/index.test.tsx

+56-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
import { LayoutStyle } from 'src/types';
2-
import { shouldUseRowLayout } from './index';
2+
import { matchLayoutComponent, shouldUseRowLayout } from './index';
33

44
describe('shouldUseRowLayout', () => {
55
it('Should be false when layout is column', () => {
@@ -52,3 +52,58 @@ describe('shouldUseRowLayout', () => {
5252
).toBe(false);
5353
});
5454
});
55+
56+
describe('matchLayoutComponent', () => {
57+
it('should not match a component id that partially contains the tested id', () => {
58+
expect(
59+
matchLayoutComponent(
60+
'abc-183d0a1c-5313-45f9-9d02-4d1b0e50b1c8',
61+
'183d0a1c-5313-45f9-9d02-4d1b0e50b1c8'
62+
)
63+
).toBe(null);
64+
65+
expect(
66+
matchLayoutComponent(
67+
'183d0a1c-5313-45f9-9d02-4d1b0e50b1c8-abc',
68+
'183d0a1c-5313-45f9-9d02-4d1b0e50b1c8'
69+
)
70+
).toBe(null);
71+
72+
expect(
73+
matchLayoutComponent(
74+
'123-183d0a1c-5313-45f9-9d02-4d1b0e50b1c8',
75+
'183d0a1c-5313-45f9-9d02-4d1b0e50b1c8'
76+
)
77+
).toBe(null);
78+
79+
expect(
80+
matchLayoutComponent(
81+
'-183d0a1c-5313-45f9-9d02-4d1b0e50b1c8',
82+
'183d0a1c-5313-45f9-9d02-4d1b0e50b1c8'
83+
)
84+
).toBe(null);
85+
86+
expect(
87+
matchLayoutComponent(
88+
'183d0a1c-5313-45f9-9d02-4d1b0e50b1c8-',
89+
'183d0a1c-5313-45f9-9d02-4d1b0e50b1c8'
90+
)
91+
).toBe(null);
92+
});
93+
94+
it('should match a component id that contains a postfix with an index in a repeating group', () => {
95+
expect(
96+
matchLayoutComponent(
97+
'183d0a1c-5313-45f9-9d02-4d1b0e50b1c8-123',
98+
'183d0a1c-5313-45f9-9d02-4d1b0e50b1c8'
99+
)
100+
).toContain('183d0a1c-5313-45f9-9d02-4d1b0e50b1c8');
101+
102+
expect(
103+
matchLayoutComponent(
104+
'183d0a1c-5313-45f9-9d02-4d1b0e50b1c8-0',
105+
'183d0a1c-5313-45f9-9d02-4d1b0e50b1c8'
106+
)
107+
).toContain('183d0a1c-5313-45f9-9d02-4d1b0e50b1c8');
108+
});
109+
});

‎src/altinn-app-frontend/src/utils/layout/index.tsx

+1-1
Original file line numberDiff line numberDiff line change
@@ -54,7 +54,7 @@ export function getLayoutIdForComponent(id: string, layouts: ILayouts): string {
5454
when searching through formLayout for the component definition.
5555
*/
5656
export function matchLayoutComponent(providedId: string, componentId: string) {
57-
return providedId.match(`${componentId}(-[0-9]*)*$`);
57+
return providedId.match(`^(${componentId})(-[0-9]+)*$`);
5858
}
5959

6060
export function renderGenericComponent(

0 commit comments

Comments
 (0)
Please sign in to comment.