Skip to content

Commit d20e87c

Browse files
authored
[Signing] Signee list summary override (#3088)
* adds signee list summary override for title * fixes and adds new tests
1 parent 29ea0bf commit d20e87c

File tree

8 files changed

+150
-33
lines changed

8 files changed

+150
-33
lines changed

src/language/texts/en.ts

-1
Original file line numberDiff line numberDiff line change
@@ -410,7 +410,6 @@ export function en() {
410410
header_status: 'Status',
411411
},
412412
signee_list_summary: {
413-
header: 'The following have signed',
414413
name_placeholder: 'Unknown name',
415414
on_behalf_of: 'on behalf of',
416415
loading: 'Loading signature data...',

src/language/texts/nb.ts

-1
Original file line numberDiff line numberDiff line change
@@ -411,7 +411,6 @@ export function nb(): FixedLanguageList {
411411
header_status: 'Status',
412412
},
413413
signee_list_summary: {
414-
header: 'Disse har signert',
415414
name_placeholder: 'Ukjent navn',
416415
on_behalf_of: 'på vegne av',
417416
loading: 'Laster signaturdata...',

src/language/texts/nn.ts

-1
Original file line numberDiff line numberDiff line change
@@ -410,7 +410,6 @@ export function nn(): FixedLanguageList {
410410
header_status: 'Status',
411411
},
412412
signee_list_summary: {
413-
header: 'Desse har signert',
414413
name_placeholder: 'Ukjent namn',
415414
on_behalf_of: 'på vegne av',
416415
loading: 'Lastar signaturdata...',

src/layout/SigneeList/SigneeListSummary.test.tsx

+99-7
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,7 @@ jest.mock('@tanstack/react-query');
2020

2121
describe('SigneeListSummary', () => {
2222
const mockedUseQuery = jest.mocked(useQuery);
23+
const mockedUseNodeItem = jest.mocked(useNodeItem);
2324

2425
beforeEach(() => {
2526
jest.resetAllMocks();
@@ -29,7 +30,7 @@ describe('SigneeListSummary', () => {
2930
instanceGuid: 'instanceGuid',
3031
taskId: 'taskId',
3132
});
32-
jest.mocked(useNodeItem).mockReturnValue('summary_title' as unknown as ReturnType<typeof useNodeItem>);
33+
mockedUseNodeItem.mockReturnValue('title' as unknown as ReturnType<typeof useNodeItem>);
3334
jest.mocked(Lang).mockImplementation(({ id }: { id: string }) => id);
3435
});
3536

@@ -41,7 +42,12 @@ describe('SigneeListSummary', () => {
4142
} as unknown as UseQueryResult);
4243

4344
// Test case
44-
render(<SigneeListSummary componentNode={{} as LayoutNode<'SigneeList'>} />);
45+
render(
46+
<SigneeListSummary
47+
titleOverride={null}
48+
componentNode={{} as LayoutNode<'SigneeList'>}
49+
/>,
50+
);
4551

4652
// Assertion
4753
expect(mockedUseQuery).toHaveBeenCalledTimes(1);
@@ -56,7 +62,12 @@ describe('SigneeListSummary', () => {
5662
} as unknown as UseQueryResult);
5763

5864
// Test case
59-
render(<SigneeListSummary componentNode={{} as LayoutNode<'SigneeList'>} />);
65+
render(
66+
<SigneeListSummary
67+
titleOverride={null}
68+
componentNode={{} as LayoutNode<'SigneeList'>}
69+
/>,
70+
);
6071

6172
// Assertion
6273
expect(mockedUseQuery).toHaveBeenCalledTimes(1);
@@ -71,7 +82,12 @@ describe('SigneeListSummary', () => {
7182
} as unknown as UseQueryResult);
7283

7384
// Test case
74-
render(<SigneeListSummary componentNode={{} as LayoutNode<'SigneeList'>} />);
85+
render(
86+
<SigneeListSummary
87+
titleOverride={null}
88+
componentNode={{} as LayoutNode<'SigneeList'>}
89+
/>,
90+
);
7591

7692
// Assertion
7793
expect(mockedUseQuery).toHaveBeenCalledTimes(1);
@@ -86,7 +102,12 @@ describe('SigneeListSummary', () => {
86102
} as unknown as UseQueryResult);
87103

88104
// Test case
89-
render(<SigneeListSummary componentNode={{} as LayoutNode<'SigneeList'>} />);
105+
render(
106+
<SigneeListSummary
107+
titleOverride={null}
108+
componentNode={{} as LayoutNode<'SigneeList'>}
109+
/>,
110+
);
90111

91112
// Assertion
92113
expect(mockedUseQuery).toHaveBeenCalledTimes(1);
@@ -132,13 +153,84 @@ describe('SigneeListSummary', () => {
132153
} as unknown as UseQueryResult);
133154

134155
// Test case
135-
render(<SigneeListSummary componentNode={{} as LayoutNode<'SigneeList'>} />);
156+
render(
157+
<SigneeListSummary
158+
titleOverride={undefined}
159+
componentNode={{} as LayoutNode<'SigneeList'>}
160+
/>,
161+
);
136162

137163
// Assertion
138164
expect(mockedUseQuery).toHaveBeenCalledTimes(1);
139-
screen.getByText('summary_title');
165+
screen.getByText('title');
140166
screen.getByText('Signee 1');
141167
screen.getByText("Signee 2, signee_list_summary.on_behalf_of Signee 2's organisation");
142168
expect(screen.queryByText(/Signee 3/i)).not.toBeInTheDocument();
143169
});
170+
171+
it('should render original title if summary override title is undefined', () => {
172+
mockedUseQuery.mockReturnValue({
173+
data: [] satisfies SigneeState[],
174+
isLoading: false,
175+
error: null,
176+
} as unknown as UseQueryResult);
177+
178+
mockedUseNodeItem.mockReturnValue('originalTitle' as unknown as ReturnType<typeof useNodeItem>);
179+
180+
// Test case
181+
render(
182+
<SigneeListSummary
183+
titleOverride={undefined}
184+
componentNode={{} as LayoutNode<'SigneeList'>}
185+
/>,
186+
);
187+
188+
// Assertion
189+
expect(mockedUseQuery).toHaveBeenCalledTimes(1);
190+
screen.getByText('originalTitle');
191+
});
192+
193+
it('should not render title if originalTitle and overrideTitle are not set', () => {
194+
mockedUseQuery.mockReturnValue({
195+
data: [] satisfies SigneeState[],
196+
isLoading: false,
197+
error: null,
198+
} as unknown as UseQueryResult);
199+
200+
mockedUseNodeItem.mockReturnValue(undefined);
201+
202+
// Test case
203+
render(
204+
<SigneeListSummary
205+
titleOverride={undefined}
206+
componentNode={{} as LayoutNode<'SigneeList'>}
207+
/>,
208+
);
209+
210+
// Assertion
211+
expect(mockedUseQuery).toHaveBeenCalledTimes(1);
212+
screen.getByText('signee_list_summary.no_signatures');
213+
expect(screen.queryByText('title')).not.toBeInTheDocument();
214+
});
215+
216+
it.each([null, ''])('should not render title if summary title override is null or empty string', (titleOverride) => {
217+
mockedUseQuery.mockReturnValue({
218+
data: [] satisfies SigneeState[],
219+
isLoading: false,
220+
error: null,
221+
} as unknown as UseQueryResult);
222+
223+
// Test case
224+
render(
225+
<SigneeListSummary
226+
titleOverride={titleOverride}
227+
componentNode={{} as LayoutNode<'SigneeList'>}
228+
/>,
229+
);
230+
231+
// Assertion
232+
expect(mockedUseQuery).toHaveBeenCalledTimes(1);
233+
screen.getByText('signee_list_summary.no_signatures');
234+
expect(screen.queryByText('title')).not.toBeInTheDocument();
235+
});
144236
});

src/layout/SigneeList/SigneeListSummary.tsx

+12-8
Original file line numberDiff line numberDiff line change
@@ -17,16 +17,18 @@ import type { LayoutNode } from 'src/utils/layout/LayoutNode';
1717

1818
interface SigneeListSummaryProps {
1919
componentNode: LayoutNode<'SigneeList'>;
20+
titleOverride: string | null | undefined;
2021
}
2122

22-
export function SigneeListSummary({ componentNode }: SigneeListSummaryProps) {
23+
export function SigneeListSummary({ componentNode, titleOverride }: SigneeListSummaryProps) {
2324
const { instanceOwnerPartyId, instanceGuid, taskId } = useParams();
2425
const { data, isLoading, error } = useQuery(signeeListQuery(instanceOwnerPartyId, instanceGuid, taskId));
2526

26-
const summaryTitle = useNodeItem(componentNode, (i) => i.textResourceBindings?.summary_title);
27+
const originalTitle = useNodeItem(componentNode, (i) => i.textResourceBindings?.title);
28+
const title = titleOverride === undefined ? originalTitle : titleOverride;
29+
const heading = title ? <Lang id={title} /> : undefined;
2730

2831
const signatures = data?.filter((signee) => isSignedSignee(signee)) ?? [];
29-
const heading = <Lang id={summaryTitle ?? 'signee_list_summary.header'} />;
3032

3133
if (isLoading) {
3234
return (
@@ -89,13 +91,15 @@ export function SigneeListSummary({ componentNode }: SigneeListSummaryProps) {
8991
);
9092
}
9193

92-
function SigneeListSummaryContainer({ heading, children }: PropsWithChildren<{ heading: ReactElement }>) {
94+
function SigneeListSummaryContainer({ heading, children }: PropsWithChildren<{ heading: ReactElement | undefined }>) {
9395
return (
9496
<div>
95-
<Label
96-
label={heading}
97-
size='lg'
98-
/>
97+
{heading && (
98+
<Label
99+
label={heading}
100+
size='lg'
101+
/>
102+
)}
99103
{children}
100104
</div>
101105
);

src/layout/SigneeList/config.ts

+16
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,22 @@
11
import { CG } from 'src/codegen/CG';
22
import { CompCategory } from 'src/layout/common';
33

4+
export const SIGNEE_LIST_OVERRIDE_PROPS = new CG.obj(
5+
new CG.prop(
6+
'title',
7+
new CG.union(new CG.str(), CG.null)
8+
.setUnionType('discriminated')
9+
.optional()
10+
.setTitle('Summary title')
11+
.setDescription('Title of the summary'),
12+
),
13+
)
14+
.extends(CG.common('ISummaryOverridesCommon'))
15+
.optional()
16+
.setTitle('Summary properties')
17+
.setDescription('Properties for how to display the summary of the component')
18+
.exportAs('SigneeListSummaryOverrideProps');
19+
420
export const Config = new CG.component({
521
category: CompCategory.Presentation,
622
capabilities: {

src/layout/SigneeList/index.tsx

+8-2
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@ import { ProcessTaskType } from 'src/types';
1010
import { NodesInternal } from 'src/utils/layout/NodesContext';
1111
import type { PropsFromGenericComponent } from 'src/layout';
1212
import type { NodeValidationProps } from 'src/layout/layout';
13+
import type { SigneeListSummaryOverrideProps } from 'src/layout/Summary2/config.generated';
1314
import type { Summary2Props } from 'src/layout/Summary2/SummaryComponent2/types';
1415

1516
export class SigneeList extends SigneeListDef {
@@ -33,7 +34,12 @@ export class SigneeList extends SigneeListDef {
3334
return null;
3435
}
3536

36-
renderSummary2(props: Summary2Props<'SigneeList'>): JSX.Element | null {
37-
return <SigneeListSummary componentNode={props.target} />;
37+
renderSummary2({ target, override }: Summary2Props<'SigneeList'>): JSX.Element | null {
38+
return (
39+
<SigneeListSummary
40+
componentNode={target}
41+
titleOverride={(override as SigneeListSummaryOverrideProps)?.title}
42+
/>
43+
);
3844
}
3945
}

src/layout/Summary2/config.tsx

+15-13
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@ import { PARAGRAPH_SUMMARY_OVERRIDE_PROPS } from 'src/layout/Paragraph/config';
1616
import { PAYMENT_SUMMARY_OVERRIDE_PROPS } from 'src/layout/Payment/config';
1717
import { RADIO_SUMMARY_OVERRIDE_PROPS } from 'src/layout/RadioButtons/config';
1818
import { REPEATING_GROUP_SUMMARY_OVERRIDE_PROPS } from 'src/layout/RepeatingGroup/config';
19+
import { SIGNEE_LIST_OVERRIDE_PROPS } from 'src/layout/SigneeList/config';
1920
import { SUBFORM_SUMMARY_OVERRIDE_PROPS } from 'src/layout/Subform/config';
2021
import { TABS_SUMMARY_PROPS } from 'src/layout/Tabs/config';
2122
import { TEXT_SUMMARY_OVERRIDE_PROPS } from 'src/layout/Text/config';
@@ -73,26 +74,27 @@ export const Config = new CG.component({
7374
'overrides',
7475
new CG.arr(
7576
new CG.union(
76-
INPUT_SUMMARY_OVERRIDE_PROPS,
7777
CHECKBOX_SUMMARY_OVERRIDE_PROPS,
78-
RADIO_SUMMARY_OVERRIDE_PROPS,
78+
DATEPICKER_SUMMARY_OVERRIDE_PROPS,
79+
DATE_SUMMARY_OVERRIDE_PROPS,
7980
DROPDOWN_SUMMARY_OVERRIDE_PROPS,
80-
MULTIPLE_SELECT_SUMMARY_OVERRIDE_PROPS,
8181
GROUP_SUMMARY_PROPS,
82-
TEXTAREA_SUMMARY_PROPS,
83-
REPEATING_GROUP_SUMMARY_OVERRIDE_PROPS,
84-
DATEPICKER_SUMMARY_OVERRIDE_PROPS,
82+
HEADER_SUMMARY_OVERRIDE_PROPS,
83+
INPUT_SUMMARY_OVERRIDE_PROPS,
84+
LIKERT_SUMMARY_OVERRIDE_PROPS,
8585
LIST_SUMMARY_OVERRIDE_PROPS,
86-
TABS_SUMMARY_PROPS,
8786
MAP_SUMMARY_OVERRIDE_PROPS,
88-
SUBFORM_SUMMARY_OVERRIDE_PROPS,
89-
LIKERT_SUMMARY_OVERRIDE_PROPS,
90-
PAYMENT_SUMMARY_OVERRIDE_PROPS,
91-
HEADER_SUMMARY_OVERRIDE_PROPS,
87+
MULTIPLE_SELECT_SUMMARY_OVERRIDE_PROPS,
88+
NUMBER_SUMMARY_OVERRIDE_PROPS,
9289
PARAGRAPH_SUMMARY_OVERRIDE_PROPS,
90+
PAYMENT_SUMMARY_OVERRIDE_PROPS,
91+
RADIO_SUMMARY_OVERRIDE_PROPS,
92+
REPEATING_GROUP_SUMMARY_OVERRIDE_PROPS,
93+
SIGNEE_LIST_OVERRIDE_PROPS,
94+
SUBFORM_SUMMARY_OVERRIDE_PROPS,
95+
TABS_SUMMARY_PROPS,
96+
TEXTAREA_SUMMARY_PROPS,
9397
TEXT_SUMMARY_OVERRIDE_PROPS,
94-
NUMBER_SUMMARY_OVERRIDE_PROPS,
95-
DATE_SUMMARY_OVERRIDE_PROPS,
9698
).exportAs('AnySummaryOverrideProps'),
9799
).optional(),
98100
),

0 commit comments

Comments
 (0)