@pie-lib/config-ui 11.30.3-next.2 → 11.30.4-next.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +7 -78
- package/lib/__tests__/alert-dialog.test.js +262 -0
- package/lib/__tests__/checkbox.test.js +227 -0
- package/lib/__tests__/choice-utils.test.js +14 -0
- package/lib/__tests__/form-section.test.js +252 -0
- package/lib/__tests__/help.test.js +270 -0
- package/lib/__tests__/input.test.js +268 -0
- package/lib/__tests__/langs.test.js +541 -0
- package/lib/__tests__/number-text-field-custom.test.js +362 -0
- package/lib/__tests__/number-text-field.test.js +421 -0
- package/lib/__tests__/radio-with-label.test.js +233 -0
- package/lib/__tests__/settings-panel.test.js +184 -0
- package/lib/__tests__/settings.test.js +653 -0
- package/lib/__tests__/tabs.test.js +211 -0
- package/lib/__tests__/two-choice.test.js +124 -0
- package/lib/__tests__/with-stateful-model.test.js +221 -0
- package/lib/alert-dialog.js +36 -43
- package/lib/alert-dialog.js.map +1 -1
- package/lib/checkbox.js +57 -72
- package/lib/checkbox.js.map +1 -1
- package/lib/choice-configuration/__tests__/feedback-menu.test.js +287 -0
- package/lib/choice-configuration/__tests__/index.test.js +253 -0
- package/lib/choice-configuration/feedback-menu.js +30 -65
- package/lib/choice-configuration/feedback-menu.js.map +1 -1
- package/lib/choice-configuration/index.js +203 -263
- package/lib/choice-configuration/index.js.map +1 -1
- package/lib/choice-utils.js +7 -19
- package/lib/choice-utils.js.map +1 -1
- package/lib/feedback-config/__tests__/feedback-config.test.js +201 -0
- package/lib/feedback-config/__tests__/feedback-selector.test.js +177 -0
- package/lib/feedback-config/feedback-selector.js +80 -116
- package/lib/feedback-config/feedback-selector.js.map +1 -1
- package/lib/feedback-config/group.js +27 -41
- package/lib/feedback-config/group.js.map +1 -1
- package/lib/feedback-config/index.js +48 -91
- package/lib/feedback-config/index.js.map +1 -1
- package/lib/form-section.js +32 -34
- package/lib/form-section.js.map +1 -1
- package/lib/help.js +40 -81
- package/lib/help.js.map +1 -1
- package/lib/index.js +2 -32
- package/lib/index.js.map +1 -1
- package/lib/input.js +22 -55
- package/lib/input.js.map +1 -1
- package/lib/inputs.js +62 -96
- package/lib/inputs.js.map +1 -1
- package/lib/langs.js +59 -102
- package/lib/langs.js.map +1 -1
- package/lib/layout/__tests__/config.layout.test.js +70 -0
- package/lib/layout/__tests__/layout-content.test.js +6 -0
- package/lib/layout/config-layout.js +41 -71
- package/lib/layout/config-layout.js.map +1 -1
- package/lib/layout/index.js +1 -4
- package/lib/layout/index.js.map +1 -1
- package/lib/layout/layout-contents.js +73 -104
- package/lib/layout/layout-contents.js.map +1 -1
- package/lib/layout/settings-box.js +28 -57
- package/lib/layout/settings-box.js.map +1 -1
- package/lib/mui-box/index.js +42 -58
- package/lib/mui-box/index.js.map +1 -1
- package/lib/number-text-field-custom.js +80 -162
- package/lib/number-text-field-custom.js.map +1 -1
- package/lib/number-text-field.js +81 -115
- package/lib/number-text-field.js.map +1 -1
- package/lib/radio-with-label.js +31 -32
- package/lib/radio-with-label.js.map +1 -1
- package/lib/settings/display-size.js +17 -33
- package/lib/settings/display-size.js.map +1 -1
- package/lib/settings/index.js +15 -48
- package/lib/settings/index.js.map +1 -1
- package/lib/settings/panel.js +160 -230
- package/lib/settings/panel.js.map +1 -1
- package/lib/settings/settings-radio-label.js +29 -31
- package/lib/settings/settings-radio-label.js.map +1 -1
- package/lib/settings/toggle.js +36 -47
- package/lib/settings/toggle.js.map +1 -1
- package/lib/tabs/index.js +23 -58
- package/lib/tabs/index.js.map +1 -1
- package/lib/tags-input/__tests__/index.test.js +183 -0
- package/lib/tags-input/index.js +51 -100
- package/lib/tags-input/index.js.map +1 -1
- package/lib/two-choice.js +47 -91
- package/lib/two-choice.js.map +1 -1
- package/lib/with-stateful-model.js +9 -32
- package/lib/with-stateful-model.js.map +1 -1
- package/package.json +12 -20
- package/src/__tests__/alert-dialog.test.jsx +283 -0
- package/src/__tests__/checkbox.test.jsx +249 -0
- package/src/__tests__/form-section.test.jsx +334 -0
- package/src/__tests__/help.test.jsx +184 -0
- package/src/__tests__/input.test.jsx +192 -0
- package/src/__tests__/langs.test.jsx +435 -15
- package/src/__tests__/number-text-field-custom.test.jsx +438 -0
- package/src/__tests__/number-text-field.test.jsx +295 -102
- package/src/__tests__/radio-with-label.test.jsx +259 -0
- package/src/__tests__/settings-panel.test.js +66 -83
- package/src/__tests__/settings.test.jsx +515 -0
- package/src/__tests__/tabs.test.jsx +193 -0
- package/src/__tests__/two-choice.test.js +104 -18
- package/src/__tests__/with-stateful-model.test.jsx +145 -0
- package/src/alert-dialog.jsx +21 -19
- package/src/checkbox.jsx +42 -46
- package/src/choice-configuration/__tests__/feedback-menu.test.jsx +157 -4
- package/src/choice-configuration/__tests__/index.test.jsx +198 -56
- package/src/choice-configuration/feedback-menu.jsx +6 -6
- package/src/choice-configuration/index.jsx +201 -196
- package/src/feedback-config/__tests__/feedback-config.test.jsx +130 -60
- package/src/feedback-config/__tests__/feedback-selector.test.jsx +87 -40
- package/src/feedback-config/feedback-selector.jsx +52 -53
- package/src/feedback-config/group.jsx +21 -22
- package/src/feedback-config/index.jsx +27 -29
- package/src/form-section.jsx +26 -18
- package/src/help.jsx +20 -28
- package/src/input.jsx +1 -1
- package/src/inputs.jsx +34 -50
- package/src/langs.jsx +41 -46
- package/src/layout/__tests__/config.layout.test.jsx +55 -38
- package/src/layout/config-layout.jsx +38 -32
- package/src/layout/layout-contents.jsx +38 -39
- package/src/layout/settings-box.jsx +16 -19
- package/src/mui-box/index.jsx +35 -43
- package/src/number-text-field-custom.jsx +30 -36
- package/src/number-text-field.jsx +45 -29
- package/src/radio-with-label.jsx +25 -13
- package/src/settings/display-size.jsx +12 -11
- package/src/settings/panel.jsx +97 -91
- package/src/settings/settings-radio-label.jsx +25 -13
- package/src/settings/toggle.jsx +30 -29
- package/src/tabs/index.jsx +8 -8
- package/src/tags-input/__tests__/index.test.jsx +88 -37
- package/src/tags-input/index.jsx +35 -38
- package/src/two-choice.jsx +15 -19
- package/esm/index.css +0 -847
- package/esm/index.js +0 -213950
- package/esm/index.js.map +0 -1
- package/esm/package.json +0 -3
- package/src/__tests__/__snapshots__/langs.test.jsx.snap +0 -32
- package/src/__tests__/__snapshots__/settings-panel.test.js.snap +0 -115
- package/src/__tests__/__snapshots__/two-choice.test.js.snap +0 -171
- package/src/choice-configuration/__tests__/__snapshots__/feedback-menu.test.jsx.snap +0 -51
- package/src/choice-configuration/__tests__/__snapshots__/index.test.jsx.snap +0 -519
- package/src/feedback-config/__tests__/__snapshots__/feedback-config.test.jsx.snap +0 -27
- package/src/feedback-config/__tests__/__snapshots__/feedback-selector.test.jsx.snap +0 -38
- package/src/layout/__tests__/__snapshots__/config.layout.test.jsx.snap +0 -59
- package/src/tags-input/__tests__/__snapshots__/index.test.jsx.snap +0 -170
|
@@ -1,24 +1,110 @@
|
|
|
1
1
|
import React from 'react';
|
|
2
|
-
import
|
|
2
|
+
import { render, screen } from '@testing-library/react';
|
|
3
|
+
import userEvent from '@testing-library/user-event';
|
|
3
4
|
import { NChoice } from '../two-choice';
|
|
4
5
|
|
|
5
6
|
describe('NChoice', () => {
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
7
|
+
const onChange = jest.fn();
|
|
8
|
+
|
|
9
|
+
beforeEach(() => {
|
|
10
|
+
onChange.mockClear();
|
|
11
|
+
});
|
|
12
|
+
|
|
13
|
+
it('renders radio buttons with vertical direction', () => {
|
|
14
|
+
render(
|
|
15
|
+
<NChoice
|
|
16
|
+
direction="vertical"
|
|
17
|
+
header="n-choice-vertical"
|
|
18
|
+
value={'left'}
|
|
19
|
+
onChange={onChange}
|
|
20
|
+
opts={[
|
|
21
|
+
{ label: 'left', value: 'left' },
|
|
22
|
+
{ label: 'center', value: 'center' },
|
|
23
|
+
{ label: 'right', value: 'right' },
|
|
24
|
+
]}
|
|
25
|
+
/>,
|
|
26
|
+
);
|
|
27
|
+
|
|
28
|
+
expect(screen.getByText('n-choice-vertical')).toBeInTheDocument();
|
|
29
|
+
expect(screen.getByLabelText('left')).toBeInTheDocument();
|
|
30
|
+
expect(screen.getByLabelText('center')).toBeInTheDocument();
|
|
31
|
+
expect(screen.getByLabelText('right')).toBeInTheDocument();
|
|
32
|
+
});
|
|
33
|
+
|
|
34
|
+
it('shows selected value as checked', () => {
|
|
35
|
+
render(
|
|
36
|
+
<NChoice
|
|
37
|
+
direction="vertical"
|
|
38
|
+
header="Options"
|
|
39
|
+
value={'center'}
|
|
40
|
+
onChange={onChange}
|
|
41
|
+
opts={[
|
|
42
|
+
{ label: 'left', value: 'left' },
|
|
43
|
+
{ label: 'center', value: 'center' },
|
|
44
|
+
{ label: 'right', value: 'right' },
|
|
45
|
+
]}
|
|
46
|
+
/>,
|
|
47
|
+
);
|
|
48
|
+
|
|
49
|
+
const centerRadio = screen.getByLabelText('center');
|
|
50
|
+
expect(centerRadio).toBeChecked();
|
|
51
|
+
expect(screen.getByLabelText('left')).not.toBeChecked();
|
|
52
|
+
expect(screen.getByLabelText('right')).not.toBeChecked();
|
|
53
|
+
});
|
|
54
|
+
|
|
55
|
+
it('calls onChange when user selects a different option', async () => {
|
|
56
|
+
const user = userEvent.setup();
|
|
57
|
+
render(
|
|
58
|
+
<NChoice
|
|
59
|
+
direction="vertical"
|
|
60
|
+
header="Options"
|
|
61
|
+
value={'left'}
|
|
62
|
+
onChange={onChange}
|
|
63
|
+
opts={[
|
|
64
|
+
{ label: 'left', value: 'left' },
|
|
65
|
+
{ label: 'center', value: 'center' },
|
|
66
|
+
{ label: 'right', value: 'right' },
|
|
67
|
+
]}
|
|
68
|
+
/>,
|
|
69
|
+
);
|
|
70
|
+
|
|
71
|
+
await user.click(screen.getByLabelText('center'));
|
|
72
|
+
|
|
73
|
+
expect(onChange).toHaveBeenCalledWith('center');
|
|
74
|
+
});
|
|
75
|
+
|
|
76
|
+
it('renders with horizontal direction', () => {
|
|
77
|
+
const { container } = render(
|
|
78
|
+
<NChoice
|
|
79
|
+
direction="horizontal"
|
|
80
|
+
header="Options"
|
|
81
|
+
value={'left'}
|
|
82
|
+
onChange={onChange}
|
|
83
|
+
opts={[
|
|
84
|
+
{ label: 'left', value: 'left' },
|
|
85
|
+
{ label: 'right', value: 'right' },
|
|
86
|
+
]}
|
|
87
|
+
/>,
|
|
88
|
+
);
|
|
89
|
+
|
|
90
|
+
// Check that all options are rendered
|
|
91
|
+
expect(screen.getByLabelText('left')).toBeInTheDocument();
|
|
92
|
+
expect(screen.getByLabelText('right')).toBeInTheDocument();
|
|
93
|
+
});
|
|
94
|
+
|
|
95
|
+
it('handles string options by converting them', () => {
|
|
96
|
+
render(
|
|
97
|
+
<NChoice
|
|
98
|
+
direction="vertical"
|
|
99
|
+
header="Options"
|
|
100
|
+
value={'option1'}
|
|
101
|
+
onChange={onChange}
|
|
102
|
+
opts={['option1', 'option2', 'option3']}
|
|
103
|
+
/>,
|
|
104
|
+
);
|
|
105
|
+
|
|
106
|
+
expect(screen.getByLabelText('option1')).toBeInTheDocument();
|
|
107
|
+
expect(screen.getByLabelText('option2')).toBeInTheDocument();
|
|
108
|
+
expect(screen.getByLabelText('option3')).toBeInTheDocument();
|
|
23
109
|
});
|
|
24
110
|
});
|
|
@@ -0,0 +1,145 @@
|
|
|
1
|
+
import React from 'react';
|
|
2
|
+
import { render, screen } from '@testing-library/react';
|
|
3
|
+
import userEvent from '@testing-library/user-event';
|
|
4
|
+
import withStatefulModel from '../with-stateful-model';
|
|
5
|
+
|
|
6
|
+
const MockComponent = ({ model, onChange }) => (
|
|
7
|
+
<div>
|
|
8
|
+
<div data-testid="model-value">{JSON.stringify(model)}</div>
|
|
9
|
+
<button onClick={() => onChange({ ...model, updated: true })}>Update Model</button>
|
|
10
|
+
</div>
|
|
11
|
+
);
|
|
12
|
+
|
|
13
|
+
describe('withStatefulModel', () => {
|
|
14
|
+
it('wraps component and passes model and onChange', () => {
|
|
15
|
+
const WrappedComponent = withStatefulModel(MockComponent);
|
|
16
|
+
const model = { name: 'test' };
|
|
17
|
+
const onChange = jest.fn();
|
|
18
|
+
|
|
19
|
+
render(<WrappedComponent model={model} onChange={onChange} />);
|
|
20
|
+
|
|
21
|
+
expect(screen.getByTestId('model-value')).toBeInTheDocument();
|
|
22
|
+
expect(screen.getByText(JSON.stringify(model))).toBeInTheDocument();
|
|
23
|
+
});
|
|
24
|
+
|
|
25
|
+
it('manages local state with initial model prop', () => {
|
|
26
|
+
const WrappedComponent = withStatefulModel(MockComponent);
|
|
27
|
+
const model = { id: 1, name: 'test' };
|
|
28
|
+
|
|
29
|
+
render(<WrappedComponent model={model} onChange={jest.fn()} />);
|
|
30
|
+
|
|
31
|
+
expect(screen.getByText(JSON.stringify(model))).toBeInTheDocument();
|
|
32
|
+
});
|
|
33
|
+
|
|
34
|
+
it('calls onChange with updated model when component calls onChange', async () => {
|
|
35
|
+
const user = userEvent.setup();
|
|
36
|
+
const WrappedComponent = withStatefulModel(MockComponent);
|
|
37
|
+
const model = { name: 'test' };
|
|
38
|
+
const onChange = jest.fn();
|
|
39
|
+
|
|
40
|
+
render(<WrappedComponent model={model} onChange={onChange} />);
|
|
41
|
+
|
|
42
|
+
const updateButton = screen.getByRole('button', { name: /Update Model/i });
|
|
43
|
+
await user.click(updateButton);
|
|
44
|
+
|
|
45
|
+
expect(onChange).toHaveBeenCalled();
|
|
46
|
+
expect(onChange).toHaveBeenCalledWith(expect.objectContaining({ updated: true }));
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
it('updates internal state when model prop changes', () => {
|
|
50
|
+
const WrappedComponent = withStatefulModel(MockComponent);
|
|
51
|
+
const initialModel = { name: 'initial' };
|
|
52
|
+
const updatedModel = { name: 'updated' };
|
|
53
|
+
const onChange = jest.fn();
|
|
54
|
+
|
|
55
|
+
const { rerender } = render(
|
|
56
|
+
<WrappedComponent model={initialModel} onChange={onChange} />
|
|
57
|
+
);
|
|
58
|
+
|
|
59
|
+
expect(screen.getByText(JSON.stringify(initialModel))).toBeInTheDocument();
|
|
60
|
+
|
|
61
|
+
rerender(<WrappedComponent model={updatedModel} onChange={onChange} />);
|
|
62
|
+
|
|
63
|
+
expect(screen.getByText(JSON.stringify(updatedModel))).toBeInTheDocument();
|
|
64
|
+
});
|
|
65
|
+
|
|
66
|
+
it('maintains state across multiple onChange calls', async () => {
|
|
67
|
+
const user = userEvent.setup();
|
|
68
|
+
const WrappedComponent = withStatefulModel(MockComponent);
|
|
69
|
+
const model = { count: 0 };
|
|
70
|
+
const onChange = jest.fn();
|
|
71
|
+
|
|
72
|
+
render(<WrappedComponent model={model} onChange={onChange} />);
|
|
73
|
+
|
|
74
|
+
const updateButton = screen.getByRole('button', { name: /Update Model/i });
|
|
75
|
+
|
|
76
|
+
await user.click(updateButton);
|
|
77
|
+
expect(onChange).toHaveBeenCalledTimes(1);
|
|
78
|
+
|
|
79
|
+
await user.click(updateButton);
|
|
80
|
+
expect(onChange).toHaveBeenCalledTimes(2);
|
|
81
|
+
});
|
|
82
|
+
|
|
83
|
+
it('passes updated model state to onChange callback', async () => {
|
|
84
|
+
const user = userEvent.setup();
|
|
85
|
+
const WrappedComponent = withStatefulModel(MockComponent);
|
|
86
|
+
const model = { value: 'original' };
|
|
87
|
+
const onChange = jest.fn();
|
|
88
|
+
|
|
89
|
+
render(<WrappedComponent model={model} onChange={onChange} />);
|
|
90
|
+
|
|
91
|
+
const updateButton = screen.getByRole('button', { name: /Update Model/i });
|
|
92
|
+
await user.click(updateButton);
|
|
93
|
+
|
|
94
|
+
const lastCall = onChange.mock.calls[onChange.mock.calls.length - 1];
|
|
95
|
+
expect(lastCall[0]).toHaveProperty('updated', true);
|
|
96
|
+
expect(lastCall[0]).toHaveProperty('value', 'original');
|
|
97
|
+
});
|
|
98
|
+
|
|
99
|
+
it('handles empty model object', () => {
|
|
100
|
+
const WrappedComponent = withStatefulModel(MockComponent);
|
|
101
|
+
const model = {};
|
|
102
|
+
const onChange = jest.fn();
|
|
103
|
+
|
|
104
|
+
render(<WrappedComponent model={model} onChange={onChange} />);
|
|
105
|
+
|
|
106
|
+
expect(screen.getByText(JSON.stringify(model))).toBeInTheDocument();
|
|
107
|
+
});
|
|
108
|
+
|
|
109
|
+
it('handles model with nested properties', () => {
|
|
110
|
+
const WrappedComponent = withStatefulModel(MockComponent);
|
|
111
|
+
const model = { user: { name: 'John', age: 30 }, settings: { theme: 'dark' } };
|
|
112
|
+
const onChange = jest.fn();
|
|
113
|
+
|
|
114
|
+
render(<WrappedComponent model={model} onChange={onChange} />);
|
|
115
|
+
|
|
116
|
+
expect(screen.getByText(JSON.stringify(model))).toBeInTheDocument();
|
|
117
|
+
});
|
|
118
|
+
|
|
119
|
+
it('requires model and onChange props', () => {
|
|
120
|
+
const WrappedComponent = withStatefulModel(MockComponent);
|
|
121
|
+
|
|
122
|
+
const consoleSpy = jest.spyOn(console, 'error').mockImplementation();
|
|
123
|
+
|
|
124
|
+
render(
|
|
125
|
+
<WrappedComponent model={{ test: 'data' }} onChange={jest.fn()} />
|
|
126
|
+
);
|
|
127
|
+
|
|
128
|
+
expect(screen.getByTestId('model-value')).toBeInTheDocument();
|
|
129
|
+
|
|
130
|
+
consoleSpy.mockRestore();
|
|
131
|
+
});
|
|
132
|
+
|
|
133
|
+
it('renders wrapped component with correct props structure', () => {
|
|
134
|
+
const WrappedComponent = withStatefulModel(MockComponent);
|
|
135
|
+
const model = { id: 1 };
|
|
136
|
+
const onChange = jest.fn();
|
|
137
|
+
|
|
138
|
+
const { container } = render(
|
|
139
|
+
<WrappedComponent model={model} onChange={onChange} />
|
|
140
|
+
);
|
|
141
|
+
|
|
142
|
+
expect(container.querySelector('[data-testid="model-value"]')).toBeInTheDocument();
|
|
143
|
+
expect(container.querySelector('button')).toBeInTheDocument();
|
|
144
|
+
});
|
|
145
|
+
});
|
package/src/alert-dialog.jsx
CHANGED
|
@@ -1,14 +1,22 @@
|
|
|
1
1
|
import React from 'react';
|
|
2
2
|
import PropTypes from 'prop-types';
|
|
3
|
-
import { Button, Dialog, DialogActions, DialogContent, DialogContentText, DialogTitle } from '@material
|
|
4
|
-
import {
|
|
3
|
+
import { Button, Dialog, DialogActions, DialogContent, DialogContentText, DialogTitle } from '@mui/material';
|
|
4
|
+
import { styled } from '@mui/material/styles';
|
|
5
5
|
|
|
6
|
-
const
|
|
7
|
-
|
|
8
|
-
|
|
6
|
+
const StyledDialogTitle = styled(DialogTitle)(() => ({
|
|
7
|
+
fontSize: 'max(1.25rem, 18px)',
|
|
8
|
+
}));
|
|
9
|
+
|
|
10
|
+
const StyledDialogContentText = styled(DialogContentText)(() => ({
|
|
11
|
+
fontSize: 'max(1rem, 14px)',
|
|
12
|
+
}));
|
|
13
|
+
|
|
14
|
+
const AlertDialog = ({ text, title, onClose, onConfirm, open, onCloseText, onConfirmText, disableAutoFocus, disableEnforceFocus, disableRestoreFocus }) => (
|
|
15
|
+
<Dialog open={open} disableAutoFocus={disableAutoFocus} disableEnforceFocus={disableEnforceFocus} disableRestoreFocus={disableRestoreFocus} onClose={onClose}>
|
|
16
|
+
{title && <StyledDialogTitle>{title}</StyledDialogTitle>}
|
|
9
17
|
{text && (
|
|
10
18
|
<DialogContent>
|
|
11
|
-
<
|
|
19
|
+
<StyledDialogContentText>{text}</StyledDialogContentText>
|
|
12
20
|
</DialogContent>
|
|
13
21
|
)}
|
|
14
22
|
<DialogActions>
|
|
@@ -29,6 +37,9 @@ const AlertDialog = ({ text, title, onClose, onConfirm, open, onCloseText, onCon
|
|
|
29
37
|
AlertDialog.defaultProps = {
|
|
30
38
|
onCloseText: 'CANCEL',
|
|
31
39
|
onConfirmText: 'OK',
|
|
40
|
+
disableAutoFocus: false,
|
|
41
|
+
disableEnforceFocus: false,
|
|
42
|
+
disableRestoreFocus: false,
|
|
32
43
|
};
|
|
33
44
|
|
|
34
45
|
AlertDialog.propTypes = {
|
|
@@ -39,18 +50,9 @@ AlertDialog.propTypes = {
|
|
|
39
50
|
open: PropTypes.bool,
|
|
40
51
|
onConfirmText: PropTypes.string,
|
|
41
52
|
onCloseText: PropTypes.string,
|
|
42
|
-
|
|
53
|
+
disableAutoFocus: PropTypes.bool,
|
|
54
|
+
disableEnforceFocus: PropTypes.bool,
|
|
55
|
+
disableRestoreFocus: PropTypes.bool,
|
|
43
56
|
};
|
|
44
57
|
|
|
45
|
-
|
|
46
|
-
heading: {
|
|
47
|
-
'& h2': {
|
|
48
|
-
fontSize: 'max(1.25rem, 18px)',
|
|
49
|
-
},
|
|
50
|
-
},
|
|
51
|
-
subheading: {
|
|
52
|
-
fontSize: 'max(1rem, 14px)',
|
|
53
|
-
},
|
|
54
|
-
});
|
|
55
|
-
|
|
56
|
-
export default withStyles(styles)(AlertDialog);
|
|
58
|
+
export default AlertDialog;
|
package/src/checkbox.jsx
CHANGED
|
@@ -1,27 +1,50 @@
|
|
|
1
|
-
import FormControlLabel from '@material
|
|
2
|
-
import MuiCheckbox from '@material
|
|
1
|
+
import FormControlLabel from '@mui/material/FormControlLabel';
|
|
2
|
+
import MuiCheckbox from '@mui/material/Checkbox';
|
|
3
3
|
import PropTypes from 'prop-types';
|
|
4
4
|
import React from 'react';
|
|
5
|
-
import {
|
|
6
|
-
import classNames from 'classnames';
|
|
7
|
-
import grey from '@material-ui/core/colors/grey';
|
|
5
|
+
import { styled } from '@mui/material/styles';
|
|
8
6
|
import { color } from '@pie-lib/render-ui';
|
|
7
|
+
import { grey } from '@mui/material/colors';
|
|
9
8
|
|
|
10
|
-
const
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
9
|
+
const StyledFormControlLabel = styled(FormControlLabel)(({ theme, mini }) => ({
|
|
10
|
+
margin: 0,
|
|
11
|
+
marginLeft: 0,
|
|
12
|
+
padding: 0,
|
|
13
|
+
'& .MuiFormControlLabel-label': {
|
|
14
|
+
fontSize: theme.typography.fontSize - 1,
|
|
15
|
+
transform: 'translate(-4%, 2%)',
|
|
16
|
+
color: 'rgba(0,0,0,1.0)',
|
|
17
|
+
...(mini && {
|
|
18
|
+
marginLeft: theme.spacing(1),
|
|
19
|
+
color: grey[700],
|
|
20
|
+
fontSize: theme.typography.fontSize - 3,
|
|
21
|
+
}),
|
|
22
|
+
},
|
|
23
|
+
}));
|
|
24
|
+
|
|
25
|
+
const StyledCheckbox = styled(MuiCheckbox)(({ theme, mini, error }) => ({
|
|
26
|
+
color: `${color.tertiary()} !important`,
|
|
27
|
+
...(mini && {
|
|
28
|
+
margin: 0,
|
|
29
|
+
padding: 0,
|
|
30
|
+
width: theme.spacing(3),
|
|
31
|
+
height: theme.spacing(3),
|
|
32
|
+
}),
|
|
33
|
+
...(error && {
|
|
34
|
+
color: `${theme.palette.error.main} !important`,
|
|
35
|
+
}),
|
|
36
|
+
}));
|
|
37
|
+
|
|
38
|
+
const Checkbox = ({ mini, checked, onChange, value, label, error }) => (
|
|
39
|
+
<StyledFormControlLabel
|
|
40
|
+
mini={mini}
|
|
16
41
|
control={
|
|
17
|
-
<
|
|
42
|
+
<StyledCheckbox
|
|
18
43
|
checked={checked}
|
|
19
44
|
onChange={onChange}
|
|
20
45
|
value={value}
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
[classes.error]: error,
|
|
24
|
-
})}
|
|
46
|
+
mini={mini}
|
|
47
|
+
error={error}
|
|
25
48
|
/>
|
|
26
49
|
}
|
|
27
50
|
label={label}
|
|
@@ -30,44 +53,17 @@ const Checkbox = ({ mini, checked, onChange, value, label, classes, error }) =>
|
|
|
30
53
|
|
|
31
54
|
Checkbox.propTypes = {
|
|
32
55
|
mini: PropTypes.bool,
|
|
33
|
-
classes: PropTypes.object.isRequired,
|
|
34
56
|
checked: PropTypes.bool.isRequired,
|
|
35
57
|
onChange: PropTypes.func.isRequired,
|
|
36
58
|
value: PropTypes.string,
|
|
37
59
|
label: PropTypes.string.isRequired,
|
|
60
|
+
error: PropTypes.bool,
|
|
38
61
|
};
|
|
39
62
|
|
|
40
63
|
Checkbox.defaultProps = {
|
|
41
64
|
value: '',
|
|
42
65
|
mini: false,
|
|
66
|
+
error: false,
|
|
43
67
|
};
|
|
44
68
|
|
|
45
|
-
export default
|
|
46
|
-
label: {
|
|
47
|
-
fontSize: theme.typography.fontSize - 1,
|
|
48
|
-
transform: 'translate(-4%, 2%)',
|
|
49
|
-
color: 'rgba(0,0,0,1.0)',
|
|
50
|
-
},
|
|
51
|
-
miniCheckbox: {
|
|
52
|
-
margin: 0,
|
|
53
|
-
padding: 0,
|
|
54
|
-
width: theme.spacing.unit * 3,
|
|
55
|
-
height: theme.spacing.unit * 3,
|
|
56
|
-
},
|
|
57
|
-
miniLabel: {
|
|
58
|
-
marginLeft: theme.spacing.unit,
|
|
59
|
-
color: grey[700],
|
|
60
|
-
fontSize: theme.typography.fontSize - 3,
|
|
61
|
-
},
|
|
62
|
-
mini: {
|
|
63
|
-
margin: 0,
|
|
64
|
-
marginLeft: 0,
|
|
65
|
-
padding: 0,
|
|
66
|
-
},
|
|
67
|
-
error: {
|
|
68
|
-
color: theme.palette.error.main,
|
|
69
|
-
},
|
|
70
|
-
customColor: {
|
|
71
|
-
color: `${color.tertiary()} !important`,
|
|
72
|
-
},
|
|
73
|
-
}))(Checkbox);
|
|
69
|
+
export default Checkbox;
|
|
@@ -1,10 +1,163 @@
|
|
|
1
1
|
import React from 'react';
|
|
2
|
+
import { render, screen } from '@testing-library/react';
|
|
3
|
+
import userEvent from '@testing-library/user-event';
|
|
2
4
|
import FeedbackMenu from '../feedback-menu';
|
|
3
|
-
import renderer from 'react-test-renderer';
|
|
4
5
|
|
|
5
6
|
describe('feedback-menu', () => {
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
7
|
+
const onChange = jest.fn();
|
|
8
|
+
|
|
9
|
+
beforeEach(() => {
|
|
10
|
+
onChange.mockClear();
|
|
11
|
+
});
|
|
12
|
+
|
|
13
|
+
describe('rendering', () => {
|
|
14
|
+
it('renders feedback menu with default options', () => {
|
|
15
|
+
const { container } = render(<FeedbackMenu onChange={onChange} classes={{}} />);
|
|
16
|
+
expect(container.firstChild).toBeInTheDocument();
|
|
17
|
+
});
|
|
18
|
+
|
|
19
|
+
it('renders menu with feedback type selector', () => {
|
|
20
|
+
render(<FeedbackMenu onChange={onChange} value={{ type: 'default' }} classes={{}} />);
|
|
21
|
+
// Menu button should be in the document
|
|
22
|
+
expect(screen.getByRole('button')).toBeInTheDocument();
|
|
23
|
+
});
|
|
24
|
+
|
|
25
|
+
it('renders with no feedback type', () => {
|
|
26
|
+
render(<FeedbackMenu onChange={onChange} value={{ type: 'none' }} classes={{}} />);
|
|
27
|
+
expect(screen.getByRole('button')).toBeInTheDocument();
|
|
28
|
+
});
|
|
29
|
+
|
|
30
|
+
it('renders with custom feedback type', () => {
|
|
31
|
+
render(<FeedbackMenu onChange={onChange} value={{ type: 'custom' }} classes={{}} />);
|
|
32
|
+
expect(screen.getByRole('button')).toBeInTheDocument();
|
|
33
|
+
});
|
|
34
|
+
|
|
35
|
+
it('renders feedback icon button', () => {
|
|
36
|
+
render(<FeedbackMenu onChange={onChange} classes={{}} />);
|
|
37
|
+
const button = screen.getByRole('button');
|
|
38
|
+
expect(button).toBeInTheDocument();
|
|
39
|
+
});
|
|
40
|
+
});
|
|
41
|
+
|
|
42
|
+
describe('user interactions', () => {
|
|
43
|
+
it('opens menu when button is clicked', async () => {
|
|
44
|
+
const user = userEvent.setup();
|
|
45
|
+
render(<FeedbackMenu onChange={onChange} value={{ type: 'default' }} classes={{}} />);
|
|
46
|
+
|
|
47
|
+
const button = screen.getByRole('button');
|
|
48
|
+
await user.click(button);
|
|
49
|
+
|
|
50
|
+
// Menu items should appear
|
|
51
|
+
expect(screen.getByText('No Feedback')).toBeInTheDocument();
|
|
52
|
+
expect(screen.getByText('Default')).toBeInTheDocument();
|
|
53
|
+
expect(screen.getByText('Custom')).toBeInTheDocument();
|
|
54
|
+
});
|
|
55
|
+
|
|
56
|
+
it('calls onChange with "none" when No Feedback is selected', async () => {
|
|
57
|
+
const user = userEvent.setup();
|
|
58
|
+
render(<FeedbackMenu onChange={onChange} value={{ type: 'default' }} classes={{}} />);
|
|
59
|
+
|
|
60
|
+
const button = screen.getByRole('button');
|
|
61
|
+
await user.click(button);
|
|
62
|
+
|
|
63
|
+
const noFeedbackOption = screen.getByText('No Feedback');
|
|
64
|
+
await user.click(noFeedbackOption);
|
|
65
|
+
|
|
66
|
+
expect(onChange).toHaveBeenCalledWith('none');
|
|
67
|
+
});
|
|
68
|
+
|
|
69
|
+
it('calls onChange with "default" when Default is selected', async () => {
|
|
70
|
+
const user = userEvent.setup();
|
|
71
|
+
render(<FeedbackMenu onChange={onChange} value={{ type: 'none' }} classes={{}} />);
|
|
72
|
+
|
|
73
|
+
const button = screen.getByRole('button');
|
|
74
|
+
await user.click(button);
|
|
75
|
+
|
|
76
|
+
const defaultOption = screen.getByText('Default');
|
|
77
|
+
await user.click(defaultOption);
|
|
78
|
+
|
|
79
|
+
expect(onChange).toHaveBeenCalledWith('default');
|
|
80
|
+
});
|
|
81
|
+
|
|
82
|
+
it('calls onChange with "custom" when Custom is selected', async () => {
|
|
83
|
+
const user = userEvent.setup();
|
|
84
|
+
render(<FeedbackMenu onChange={onChange} value={{ type: 'none' }} classes={{}} />);
|
|
85
|
+
|
|
86
|
+
const button = screen.getByRole('button');
|
|
87
|
+
await user.click(button);
|
|
88
|
+
|
|
89
|
+
const customOption = screen.getByText('Custom');
|
|
90
|
+
await user.click(customOption);
|
|
91
|
+
|
|
92
|
+
expect(onChange).toHaveBeenCalledWith('custom');
|
|
93
|
+
});
|
|
94
|
+
|
|
95
|
+
it('closes menu after selection', async () => {
|
|
96
|
+
const user = userEvent.setup();
|
|
97
|
+
render(<FeedbackMenu onChange={onChange} value={{ type: 'default' }} classes={{}} />);
|
|
98
|
+
|
|
99
|
+
const button = screen.getByRole('button');
|
|
100
|
+
await user.click(button);
|
|
101
|
+
|
|
102
|
+
expect(screen.getByText('No Feedback')).toBeInTheDocument();
|
|
103
|
+
|
|
104
|
+
const noFeedbackOption = screen.getByText('No Feedback');
|
|
105
|
+
await user.click(noFeedbackOption);
|
|
106
|
+
|
|
107
|
+
// Menu should close after selection
|
|
108
|
+
expect(onChange).toHaveBeenCalledWith('none');
|
|
109
|
+
});
|
|
110
|
+
});
|
|
111
|
+
|
|
112
|
+
describe('feedback type indicators', () => {
|
|
113
|
+
it('shows correct icon color for default feedback', () => {
|
|
114
|
+
const { container } = render(
|
|
115
|
+
<FeedbackMenu onChange={onChange} value={{ type: 'default' }} classes={{}} />
|
|
116
|
+
);
|
|
117
|
+
// Icon should be present with primary color
|
|
118
|
+
expect(container.querySelector('svg')).toBeInTheDocument();
|
|
119
|
+
});
|
|
120
|
+
|
|
121
|
+
it('shows correct icon color for custom feedback', () => {
|
|
122
|
+
const { container } = render(
|
|
123
|
+
<FeedbackMenu onChange={onChange} value={{ type: 'custom' }} classes={{}} />
|
|
124
|
+
);
|
|
125
|
+
// Icon should be present with primary color
|
|
126
|
+
expect(container.querySelector('svg')).toBeInTheDocument();
|
|
127
|
+
});
|
|
128
|
+
|
|
129
|
+
it('shows correct icon color for disabled feedback', () => {
|
|
130
|
+
const { container } = render(
|
|
131
|
+
<FeedbackMenu onChange={onChange} value={{ type: 'none' }} classes={{}} />
|
|
132
|
+
);
|
|
133
|
+
// Icon should be present with disabled color
|
|
134
|
+
expect(container.querySelector('svg')).toBeInTheDocument();
|
|
135
|
+
});
|
|
136
|
+
});
|
|
137
|
+
|
|
138
|
+
describe('aria labels', () => {
|
|
139
|
+
it('has aria-label on button for default feedback', () => {
|
|
140
|
+
render(
|
|
141
|
+
<FeedbackMenu onChange={onChange} value={{ type: 'default' }} classes={{}} />
|
|
142
|
+
);
|
|
143
|
+
const button = screen.getByRole('button', { name: /Default Feedback/i });
|
|
144
|
+
expect(button).toBeInTheDocument();
|
|
145
|
+
});
|
|
146
|
+
|
|
147
|
+
it('has aria-label on button for custom feedback', () => {
|
|
148
|
+
render(
|
|
149
|
+
<FeedbackMenu onChange={onChange} value={{ type: 'custom' }} classes={{}} />
|
|
150
|
+
);
|
|
151
|
+
const button = screen.getByRole('button', { name: /Custom Feedback/i });
|
|
152
|
+
expect(button).toBeInTheDocument();
|
|
153
|
+
});
|
|
154
|
+
|
|
155
|
+
it('has aria-label on button for disabled feedback', () => {
|
|
156
|
+
render(
|
|
157
|
+
<FeedbackMenu onChange={onChange} value={{ type: 'none' }} classes={{}} />
|
|
158
|
+
);
|
|
159
|
+
const button = screen.getByRole('button', { name: /Feedback disabled/i });
|
|
160
|
+
expect(button).toBeInTheDocument();
|
|
161
|
+
});
|
|
9
162
|
});
|
|
10
163
|
});
|