@pie-lib/config-ui 13.0.4-next.31 → 13.0.4-next.36
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.json +32 -0
- package/CHANGELOG.md +2419 -0
- package/LICENSE.md +5 -0
- package/lib/alert-dialog.js +68 -0
- package/lib/alert-dialog.js.map +1 -0
- package/lib/checkbox.js +84 -0
- package/lib/checkbox.js.map +1 -0
- package/lib/choice-configuration/feedback-menu.js +129 -0
- package/lib/choice-configuration/feedback-menu.js.map +1 -0
- package/lib/choice-configuration/index.js +381 -0
- package/lib/choice-configuration/index.js.map +1 -0
- package/lib/choice-utils.js +42 -0
- package/lib/choice-utils.js.map +1 -0
- package/lib/feedback-config/feedback-selector.js +155 -0
- package/lib/feedback-config/feedback-selector.js.map +1 -0
- package/lib/feedback-config/group.js +61 -0
- package/lib/feedback-config/group.js.map +1 -0
- package/lib/feedback-config/index.js +146 -0
- package/lib/feedback-config/index.js.map +1 -0
- package/lib/form-section.js +44 -0
- package/lib/form-section.js.map +1 -0
- package/lib/help.js +106 -0
- package/lib/help.js.map +1 -0
- package/lib/index.js +186 -0
- package/lib/index.js.map +1 -0
- package/lib/input.js +106 -0
- package/lib/input.js.map +1 -0
- package/lib/inputs.js +105 -0
- package/lib/inputs.js.map +1 -0
- package/lib/langs.js +136 -0
- package/lib/langs.js.map +1 -0
- package/lib/layout/config-layout.js +137 -0
- package/lib/layout/config-layout.js.map +1 -0
- package/lib/layout/index.js +21 -0
- package/lib/layout/index.js.map +1 -0
- package/lib/layout/layout-contents.js +160 -0
- package/lib/layout/layout-contents.js.map +1 -0
- package/lib/layout/settings-box.js +57 -0
- package/lib/layout/settings-box.js.map +1 -0
- package/lib/mui-box/index.js +63 -0
- package/lib/mui-box/index.js.map +1 -0
- package/lib/number-text-field-custom.js +376 -0
- package/lib/number-text-field-custom.js.map +1 -0
- package/lib/number-text-field.js +229 -0
- package/lib/number-text-field.js.map +1 -0
- package/lib/radio-with-label.js +48 -0
- package/lib/radio-with-label.js.map +1 -0
- package/lib/settings/display-size.js +61 -0
- package/lib/settings/display-size.js.map +1 -0
- package/lib/settings/index.js +110 -0
- package/lib/settings/index.js.map +1 -0
- package/lib/settings/panel.js +392 -0
- package/lib/settings/panel.js.map +1 -0
- package/lib/settings/settings-radio-label.js +51 -0
- package/lib/settings/settings-radio-label.js.map +1 -0
- package/lib/settings/toggle.js +63 -0
- package/lib/settings/toggle.js.map +1 -0
- package/lib/tabs/index.js +75 -0
- package/lib/tabs/index.js.map +1 -0
- package/lib/tags-input/index.js +149 -0
- package/lib/tags-input/index.js.map +1 -0
- package/lib/two-choice.js +136 -0
- package/lib/two-choice.js.map +1 -0
- package/lib/with-stateful-model.js +61 -0
- package/lib/with-stateful-model.js.map +1 -0
- package/package.json +19 -33
- package/src/__tests__/alert-dialog.test.jsx +183 -0
- package/src/__tests__/checkbox.test.jsx +152 -0
- package/src/__tests__/choice-utils.test.js +12 -0
- package/src/__tests__/form-section.test.jsx +328 -0
- package/src/__tests__/help.test.jsx +184 -0
- package/src/__tests__/input.test.jsx +156 -0
- package/src/__tests__/langs.test.jsx +376 -0
- package/src/__tests__/number-text-field-custom.test.jsx +255 -0
- package/src/__tests__/number-text-field.test.jsx +263 -0
- package/src/__tests__/radio-with-label.test.jsx +155 -0
- package/src/__tests__/settings-panel.test.js +187 -0
- package/src/__tests__/settings.test.jsx +452 -0
- package/src/__tests__/tabs.test.jsx +188 -0
- package/src/__tests__/two-choice.test.js +110 -0
- package/src/__tests__/with-stateful-model.test.jsx +139 -0
- package/src/alert-dialog.jsx +75 -0
- package/src/checkbox.jsx +61 -0
- package/src/choice-configuration/__tests__/feedback-menu.test.jsx +151 -0
- package/src/choice-configuration/__tests__/index.test.jsx +234 -0
- package/src/choice-configuration/feedback-menu.jsx +96 -0
- package/src/choice-configuration/index.jsx +357 -0
- package/src/choice-utils.js +30 -0
- package/src/feedback-config/__tests__/feedback-config.test.jsx +141 -0
- package/src/feedback-config/__tests__/feedback-selector.test.jsx +97 -0
- package/src/feedback-config/feedback-selector.jsx +112 -0
- package/src/feedback-config/group.jsx +51 -0
- package/src/feedback-config/index.jsx +111 -0
- package/src/form-section.jsx +31 -0
- package/src/help.jsx +79 -0
- package/src/index.js +55 -0
- package/src/input.jsx +72 -0
- package/src/inputs.jsx +69 -0
- package/src/langs.jsx +111 -0
- package/src/layout/__tests__/config.layout.test.jsx +59 -0
- package/src/layout/__tests__/layout-content.test.jsx +3 -0
- package/src/layout/config-layout.jsx +103 -0
- package/src/layout/index.js +4 -0
- package/src/layout/layout-contents.jsx +117 -0
- package/src/layout/settings-box.jsx +32 -0
- package/src/mui-box/index.jsx +56 -0
- package/src/number-text-field-custom.jsx +333 -0
- package/src/number-text-field.jsx +215 -0
- package/src/radio-with-label.jsx +30 -0
- package/src/settings/display-size.jsx +53 -0
- package/src/settings/index.js +83 -0
- package/src/settings/panel.jsx +333 -0
- package/src/settings/settings-radio-label.jsx +32 -0
- package/src/settings/toggle.jsx +46 -0
- package/src/tabs/index.jsx +47 -0
- package/src/tags-input/__tests__/index.test.jsx +113 -0
- package/src/tags-input/index.jsx +116 -0
- package/src/two-choice.jsx +90 -0
- package/src/with-stateful-model.jsx +36 -0
- package/dist/_virtual/_rolldown/runtime.js +0 -11
- package/dist/alert-dialog.d.ts +0 -44
- package/dist/alert-dialog.js +0 -47
- package/dist/checkbox.d.ts +0 -34
- package/dist/checkbox.js +0 -57
- package/dist/choice-configuration/feedback-menu.d.ts +0 -32
- package/dist/choice-configuration/feedback-menu.js +0 -85
- package/dist/choice-configuration/index.d.ts +0 -62
- package/dist/choice-configuration/index.js +0 -240
- package/dist/choice-utils.d.ts +0 -21
- package/dist/choice-utils.js +0 -15
- package/dist/feedback-config/feedback-selector.d.ts +0 -33
- package/dist/feedback-config/feedback-selector.js +0 -92
- package/dist/feedback-config/group.d.ts +0 -21
- package/dist/feedback-config/group.js +0 -33
- package/dist/feedback-config/index.d.ts +0 -48
- package/dist/feedback-config/index.js +0 -96
- package/dist/form-section.d.ts +0 -25
- package/dist/form-section.js +0 -25
- package/dist/fraction-to-number.d.ts +0 -7
- package/dist/fraction-to-number.js +0 -9
- package/dist/help.d.ts +0 -41
- package/dist/help.js +0 -61
- package/dist/index.d.ts +0 -31
- package/dist/index.js +0 -34
- package/dist/input.d.ts +0 -29
- package/dist/input.js +0 -65
- package/dist/inputs.d.ts +0 -63
- package/dist/inputs.js +0 -70
- package/dist/langs.d.ts +0 -41
- package/dist/langs.js +0 -76
- package/dist/layout/config-layout.d.ts +0 -10
- package/dist/layout/config-layout.js +0 -75
- package/dist/layout/index.d.ts +0 -11
- package/dist/layout/index.js +0 -10
- package/dist/layout/layout-contents.d.ts +0 -21
- package/dist/layout/layout-contents.js +0 -70
- package/dist/layout/settings-box.d.ts +0 -19
- package/dist/layout/settings-box.js +0 -31
- package/dist/mui-box/index.d.ts +0 -21
- package/dist/mui-box/index.js +0 -47
- package/dist/node_modules/.bun/@babel_runtime@7.28.6/node_modules/@babel/runtime/helpers/esm/extends.js +0 -12
- package/dist/node_modules/.bun/@babel_runtime@7.28.6/node_modules/@babel/runtime/helpers/esm/inheritsLoose.js +0 -7
- package/dist/node_modules/.bun/@babel_runtime@7.28.6/node_modules/@babel/runtime/helpers/esm/objectWithoutPropertiesLoose.js +0 -12
- package/dist/node_modules/.bun/@babel_runtime@7.28.6/node_modules/@babel/runtime/helpers/esm/setPrototypeOf.js +0 -8
- package/dist/node_modules/.bun/react-measure@2.5.2_6dbf9a050bc9aadb/node_modules/react-measure/dist/index.esm.js +0 -122
- package/dist/node_modules/.bun/resize-observer-polyfill@1.5.1/node_modules/resize-observer-polyfill/dist/ResizeObserver.es.js +0 -276
- package/dist/number-text-field-custom.d.ts +0 -51
- package/dist/number-text-field-custom.js +0 -192
- package/dist/number-text-field.d.ts +0 -47
- package/dist/number-text-field.js +0 -122
- package/dist/radio-with-label.d.ts +0 -25
- package/dist/radio-with-label.js +0 -27
- package/dist/settings/display-size.d.ts +0 -26
- package/dist/settings/display-size.js +0 -45
- package/dist/settings/index.d.ts +0 -45
- package/dist/settings/index.js +0 -63
- package/dist/settings/panel.d.ts +0 -27
- package/dist/settings/panel.js +0 -201
- package/dist/settings/settings-radio-label.d.ts +0 -25
- package/dist/settings/settings-radio-label.js +0 -29
- package/dist/settings/toggle.d.ts +0 -25
- package/dist/settings/toggle.js +0 -33
- package/dist/tabs/index.d.ts +0 -22
- package/dist/tabs/index.js +0 -39
- package/dist/tags-input/index.d.ts +0 -21
- package/dist/tags-input/index.js +0 -83
- package/dist/two-choice.d.ts +0 -43
- package/dist/two-choice.js +0 -79
- package/dist/with-stateful-model.d.ts +0 -42
- package/dist/with-stateful-model.js +0 -32
|
@@ -0,0 +1,141 @@
|
|
|
1
|
+
import React from 'react';
|
|
2
|
+
import { render, screen } from '@testing-library/react';
|
|
3
|
+
import userEvent from '@testing-library/user-event';
|
|
4
|
+
import { buildDefaults, FeedbackConfig } from '../index';
|
|
5
|
+
|
|
6
|
+
// Mock FeedbackSelector to simplify testing
|
|
7
|
+
jest.mock('../feedback-selector', () => {
|
|
8
|
+
return function FeedbackSelector({ label, feedback, onChange }) {
|
|
9
|
+
return (
|
|
10
|
+
<div data-testid="feedback-selector">
|
|
11
|
+
<label>{label}</label>
|
|
12
|
+
<select
|
|
13
|
+
aria-label={label}
|
|
14
|
+
value={feedback.type}
|
|
15
|
+
onChange={(e) => onChange({ ...feedback, type: e.target.value })}
|
|
16
|
+
>
|
|
17
|
+
<option value="default">Default</option>
|
|
18
|
+
<option value="custom">Custom</option>
|
|
19
|
+
<option value="none">None</option>
|
|
20
|
+
</select>
|
|
21
|
+
</div>
|
|
22
|
+
);
|
|
23
|
+
};
|
|
24
|
+
});
|
|
25
|
+
|
|
26
|
+
describe('FeedbackConfig', () => {
|
|
27
|
+
const onChange = jest.fn();
|
|
28
|
+
|
|
29
|
+
beforeEach(() => {
|
|
30
|
+
onChange.mockClear();
|
|
31
|
+
});
|
|
32
|
+
|
|
33
|
+
describe('rendering', () => {
|
|
34
|
+
it('renders with default feedback types', () => {
|
|
35
|
+
const feedback = buildDefaults();
|
|
36
|
+
|
|
37
|
+
render(<FeedbackConfig feedback={feedback} onChange={onChange} />);
|
|
38
|
+
|
|
39
|
+
expect(screen.getByText('Feedback')).toBeInTheDocument();
|
|
40
|
+
expect(screen.getByText('If correct, show')).toBeInTheDocument();
|
|
41
|
+
expect(screen.getByText('If partially correct, show')).toBeInTheDocument();
|
|
42
|
+
expect(screen.getByText('If incorrect, show')).toBeInTheDocument();
|
|
43
|
+
});
|
|
44
|
+
|
|
45
|
+
it('renders all three feedback selectors by default', () => {
|
|
46
|
+
const feedback = buildDefaults();
|
|
47
|
+
|
|
48
|
+
render(<FeedbackConfig feedback={feedback} onChange={onChange} />);
|
|
49
|
+
|
|
50
|
+
const selectors = screen.getAllByTestId('feedback-selector');
|
|
51
|
+
expect(selectors).toHaveLength(3);
|
|
52
|
+
});
|
|
53
|
+
|
|
54
|
+
it('does not render partial feedback selector when allowPartial is false', () => {
|
|
55
|
+
const feedback = buildDefaults();
|
|
56
|
+
|
|
57
|
+
render(<FeedbackConfig allowPartial={false} feedback={feedback} onChange={onChange} />);
|
|
58
|
+
|
|
59
|
+
expect(screen.getByText('If correct, show')).toBeInTheDocument();
|
|
60
|
+
expect(screen.queryByText('If partially correct, show')).not.toBeInTheDocument();
|
|
61
|
+
expect(screen.getByText('If incorrect, show')).toBeInTheDocument();
|
|
62
|
+
|
|
63
|
+
const selectors = screen.getAllByTestId('feedback-selector');
|
|
64
|
+
expect(selectors).toHaveLength(2);
|
|
65
|
+
});
|
|
66
|
+
});
|
|
67
|
+
|
|
68
|
+
describe('user interactions', () => {
|
|
69
|
+
it('calls onChange when correct feedback type changes', async () => {
|
|
70
|
+
const user = userEvent.setup();
|
|
71
|
+
const feedback = buildDefaults();
|
|
72
|
+
|
|
73
|
+
render(<FeedbackConfig feedback={feedback} onChange={onChange} />);
|
|
74
|
+
|
|
75
|
+
const correctSelect = screen.getByLabelText('If correct, show');
|
|
76
|
+
await user.selectOptions(correctSelect, 'custom');
|
|
77
|
+
|
|
78
|
+
expect(onChange).toHaveBeenCalledWith(
|
|
79
|
+
expect.objectContaining({
|
|
80
|
+
correct: expect.objectContaining({ type: 'custom' }),
|
|
81
|
+
}),
|
|
82
|
+
);
|
|
83
|
+
});
|
|
84
|
+
|
|
85
|
+
it('calls onChange when incorrect feedback type changes', async () => {
|
|
86
|
+
const user = userEvent.setup();
|
|
87
|
+
const feedback = buildDefaults();
|
|
88
|
+
|
|
89
|
+
render(<FeedbackConfig feedback={feedback} onChange={onChange} />);
|
|
90
|
+
|
|
91
|
+
const incorrectSelect = screen.getByLabelText('If incorrect, show');
|
|
92
|
+
await user.selectOptions(incorrectSelect, 'none');
|
|
93
|
+
|
|
94
|
+
expect(onChange).toHaveBeenCalledWith(
|
|
95
|
+
expect.objectContaining({
|
|
96
|
+
incorrect: expect.objectContaining({ type: 'none' }),
|
|
97
|
+
}),
|
|
98
|
+
);
|
|
99
|
+
});
|
|
100
|
+
|
|
101
|
+
it('calls onChange when partial feedback type changes', async () => {
|
|
102
|
+
const user = userEvent.setup();
|
|
103
|
+
const feedback = buildDefaults();
|
|
104
|
+
|
|
105
|
+
render(<FeedbackConfig feedback={feedback} onChange={onChange} />);
|
|
106
|
+
|
|
107
|
+
const partialSelect = screen.getByLabelText('If partially correct, show');
|
|
108
|
+
await user.selectOptions(partialSelect, 'custom');
|
|
109
|
+
|
|
110
|
+
expect(onChange).toHaveBeenCalledWith(
|
|
111
|
+
expect.objectContaining({
|
|
112
|
+
partial: expect.objectContaining({ type: 'custom' }),
|
|
113
|
+
}),
|
|
114
|
+
);
|
|
115
|
+
});
|
|
116
|
+
});
|
|
117
|
+
|
|
118
|
+
describe('buildDefaults helper', () => {
|
|
119
|
+
it('returns default feedback configuration', () => {
|
|
120
|
+
const defaults = buildDefaults();
|
|
121
|
+
|
|
122
|
+
expect(defaults).toEqual({
|
|
123
|
+
correct: { type: 'default', default: 'Correct' },
|
|
124
|
+
incorrect: { type: 'default', default: 'Incorrect' },
|
|
125
|
+
partial: { type: 'default', default: 'Nearly' },
|
|
126
|
+
});
|
|
127
|
+
});
|
|
128
|
+
|
|
129
|
+
it('merges custom values with defaults', () => {
|
|
130
|
+
const defaults = buildDefaults({
|
|
131
|
+
correct: { type: 'custom', custom: 'Great job!' },
|
|
132
|
+
});
|
|
133
|
+
|
|
134
|
+
expect(defaults).toEqual({
|
|
135
|
+
correct: { type: 'custom', default: 'Correct', custom: 'Great job!' },
|
|
136
|
+
incorrect: { type: 'default', default: 'Incorrect' },
|
|
137
|
+
partial: { type: 'default', default: 'Nearly' },
|
|
138
|
+
});
|
|
139
|
+
});
|
|
140
|
+
});
|
|
141
|
+
});
|
|
@@ -0,0 +1,97 @@
|
|
|
1
|
+
import { render, screen } from '@testing-library/react';
|
|
2
|
+
import userEvent from '@testing-library/user-event';
|
|
3
|
+
import React from 'react';
|
|
4
|
+
import { FeedbackSelector } from '../feedback-selector';
|
|
5
|
+
|
|
6
|
+
// Mock the editable-html component to avoid window dependencies in tests
|
|
7
|
+
jest.mock('@pie-lib/editable-html-tip-tap', () => {
|
|
8
|
+
return {
|
|
9
|
+
__esModule: true,
|
|
10
|
+
default: ({ markup, onChange }) => (
|
|
11
|
+
<textarea data-testid="editable-html" defaultValue={markup || ''} onChange={(e) => onChange(e.target.value)} />
|
|
12
|
+
),
|
|
13
|
+
};
|
|
14
|
+
});
|
|
15
|
+
|
|
16
|
+
describe('feedback-selector', () => {
|
|
17
|
+
let onChange;
|
|
18
|
+
|
|
19
|
+
const renderComponent = (feedback = { type: 'default', default: 'hi' }) => {
|
|
20
|
+
return render(<FeedbackSelector label={'Feedback'} onChange={onChange} feedback={feedback} />);
|
|
21
|
+
};
|
|
22
|
+
|
|
23
|
+
beforeEach(() => {
|
|
24
|
+
onChange = jest.fn();
|
|
25
|
+
});
|
|
26
|
+
|
|
27
|
+
describe('rendering', () => {
|
|
28
|
+
it('renders feedback type selector', () => {
|
|
29
|
+
renderComponent();
|
|
30
|
+
expect(screen.getByText('Simple Feedback')).toBeInTheDocument();
|
|
31
|
+
expect(screen.getByText('No Feedback')).toBeInTheDocument();
|
|
32
|
+
expect(screen.getByText('Customized Feedback')).toBeInTheDocument();
|
|
33
|
+
});
|
|
34
|
+
|
|
35
|
+
it('shows default feedback text when type is default', () => {
|
|
36
|
+
renderComponent({ type: 'default', default: 'hi' });
|
|
37
|
+
expect(screen.getByText('hi')).toBeInTheDocument();
|
|
38
|
+
});
|
|
39
|
+
|
|
40
|
+
it('shows editable html when type is custom', () => {
|
|
41
|
+
renderComponent({ type: 'custom', default: 'hi', custom: 'custom text' });
|
|
42
|
+
expect(screen.getByTestId('editable-html')).toBeInTheDocument();
|
|
43
|
+
expect(screen.getByTestId('editable-html')).toHaveValue('custom text');
|
|
44
|
+
});
|
|
45
|
+
|
|
46
|
+
it('does not show default feedback when type is none', () => {
|
|
47
|
+
renderComponent({ type: 'none', default: 'hi' });
|
|
48
|
+
expect(screen.queryByText('hi')).not.toBeInTheDocument();
|
|
49
|
+
});
|
|
50
|
+
});
|
|
51
|
+
|
|
52
|
+
describe('user interactions', () => {
|
|
53
|
+
it('calls onChange when switching to default feedback', async () => {
|
|
54
|
+
const user = userEvent.setup();
|
|
55
|
+
renderComponent({ type: 'custom', default: 'hi' });
|
|
56
|
+
|
|
57
|
+
await user.click(screen.getByLabelText('Simple Feedback'));
|
|
58
|
+
|
|
59
|
+
expect(onChange).toHaveBeenCalledWith({ type: 'default', default: 'hi' });
|
|
60
|
+
});
|
|
61
|
+
|
|
62
|
+
it('calls onChange when switching to custom feedback', async () => {
|
|
63
|
+
const user = userEvent.setup();
|
|
64
|
+
renderComponent({ type: 'default', default: 'hi' });
|
|
65
|
+
|
|
66
|
+
await user.click(screen.getByLabelText('Customized Feedback'));
|
|
67
|
+
|
|
68
|
+
expect(onChange).toHaveBeenCalledWith({ type: 'custom', default: 'hi' });
|
|
69
|
+
});
|
|
70
|
+
|
|
71
|
+
it('calls onChange when switching to no feedback', async () => {
|
|
72
|
+
const user = userEvent.setup();
|
|
73
|
+
renderComponent({ type: 'default', default: 'hi' });
|
|
74
|
+
|
|
75
|
+
await user.click(screen.getByLabelText('No Feedback'));
|
|
76
|
+
|
|
77
|
+
expect(onChange).toHaveBeenCalledWith({ type: 'none', default: 'hi' });
|
|
78
|
+
});
|
|
79
|
+
|
|
80
|
+
it('calls onChange with custom text when editing custom feedback', async () => {
|
|
81
|
+
const user = userEvent.setup();
|
|
82
|
+
renderComponent({ type: 'custom', default: 'hi', custom: '' });
|
|
83
|
+
|
|
84
|
+
const editor = screen.getByTestId('editable-html');
|
|
85
|
+
await user.clear(editor);
|
|
86
|
+
await user.type(editor, 'text');
|
|
87
|
+
|
|
88
|
+
// user.type triggers onChange for each character, so check that onChange was called
|
|
89
|
+
// and that the last call has 'text' in custom field
|
|
90
|
+
expect(onChange).toHaveBeenCalled();
|
|
91
|
+
const lastCall = onChange.mock.calls[onChange.mock.calls.length - 1][0];
|
|
92
|
+
expect(lastCall.type).toBe('custom');
|
|
93
|
+
expect(lastCall.custom).toBe('text');
|
|
94
|
+
expect(lastCall.default).toBe('hi');
|
|
95
|
+
});
|
|
96
|
+
});
|
|
97
|
+
});
|
|
@@ -0,0 +1,112 @@
|
|
|
1
|
+
import EditableHtml from '@pie-lib/editable-html-tip-tap';
|
|
2
|
+
import { InputContainer } from '@pie-lib/render-ui';
|
|
3
|
+
import PropTypes from 'prop-types';
|
|
4
|
+
import React from 'react';
|
|
5
|
+
import { styled } from '@mui/material/styles';
|
|
6
|
+
import Group from './group';
|
|
7
|
+
|
|
8
|
+
// - mathquill error window not defined
|
|
9
|
+
const StyledEditableHTML = styled(EditableHtml)(({ theme }) => ({
|
|
10
|
+
fontFamily: theme.typography.fontFamily,
|
|
11
|
+
}));
|
|
12
|
+
|
|
13
|
+
const feedbackLabels = {
|
|
14
|
+
default: 'Simple Feedback',
|
|
15
|
+
none: 'No Feedback',
|
|
16
|
+
custom: 'Customized Feedback',
|
|
17
|
+
};
|
|
18
|
+
|
|
19
|
+
const StyledFeedbackSelector = styled('div')(({ theme }) => ({
|
|
20
|
+
marginBottom: theme.spacing(1),
|
|
21
|
+
}));
|
|
22
|
+
|
|
23
|
+
const StyledInputContainer = styled(InputContainer)(() => ({
|
|
24
|
+
paddingBottom: 0,
|
|
25
|
+
}));
|
|
26
|
+
|
|
27
|
+
const StyledCustomHolder = styled('div')(({ theme }) => ({
|
|
28
|
+
marginTop: '0px',
|
|
29
|
+
background: theme.palette.grey[300],
|
|
30
|
+
padding: 0,
|
|
31
|
+
marginBottom: theme.spacing(2),
|
|
32
|
+
borderRadius: '4px',
|
|
33
|
+
}));
|
|
34
|
+
|
|
35
|
+
const StyledDefaultHolder = styled('div')(({ theme }) => ({
|
|
36
|
+
marginTop: '0px',
|
|
37
|
+
background: theme.palette.grey[300],
|
|
38
|
+
padding: theme.spacing(2),
|
|
39
|
+
marginBottom: theme.spacing(2),
|
|
40
|
+
borderRadius: '4px',
|
|
41
|
+
fontFamily: theme.typography.fontFamily,
|
|
42
|
+
cursor: 'default',
|
|
43
|
+
}));
|
|
44
|
+
|
|
45
|
+
const StyledGroup = styled(Group)(({ theme }) => ({
|
|
46
|
+
paddingTop: theme.spacing(1),
|
|
47
|
+
}));
|
|
48
|
+
|
|
49
|
+
export const FeedbackType = {
|
|
50
|
+
type: PropTypes.oneOf(['default', 'custom', 'none']),
|
|
51
|
+
default: PropTypes.string,
|
|
52
|
+
custom: PropTypes.string,
|
|
53
|
+
};
|
|
54
|
+
|
|
55
|
+
export class FeedbackSelector extends React.Component {
|
|
56
|
+
static propTypes = {
|
|
57
|
+
keys: PropTypes.arrayOf(PropTypes.string),
|
|
58
|
+
label: PropTypes.string.isRequired,
|
|
59
|
+
feedback: PropTypes.shape(FeedbackType).isRequired,
|
|
60
|
+
onChange: PropTypes.func.isRequired,
|
|
61
|
+
toolbarOpts: PropTypes.object,
|
|
62
|
+
mathMlOptions: PropTypes.object,
|
|
63
|
+
};
|
|
64
|
+
|
|
65
|
+
changeType = (type) => {
|
|
66
|
+
const { onChange, feedback } = this.props;
|
|
67
|
+
|
|
68
|
+
onChange({ ...feedback, type });
|
|
69
|
+
};
|
|
70
|
+
|
|
71
|
+
changeCustom = (custom) => {
|
|
72
|
+
const { onChange, feedback } = this.props;
|
|
73
|
+
|
|
74
|
+
onChange({ ...feedback, type: 'custom', custom });
|
|
75
|
+
};
|
|
76
|
+
|
|
77
|
+
render() {
|
|
78
|
+
const { keys, label, feedback, toolbarOpts, mathMlOptions = {} } = this.props;
|
|
79
|
+
|
|
80
|
+
const feedbackKeys = keys || Object.keys(feedbackLabels);
|
|
81
|
+
|
|
82
|
+
return (
|
|
83
|
+
<StyledFeedbackSelector>
|
|
84
|
+
<StyledInputContainer label={label} extraClasses={{ label: { transform: 'translateY(-20%)' } }}>
|
|
85
|
+
<StyledGroup
|
|
86
|
+
keys={feedbackKeys}
|
|
87
|
+
label={label}
|
|
88
|
+
value={feedback.type}
|
|
89
|
+
onChange={this.changeType}
|
|
90
|
+
feedbackLabels={feedbackLabels}
|
|
91
|
+
/>
|
|
92
|
+
</StyledInputContainer>
|
|
93
|
+
|
|
94
|
+
{feedback.type === 'custom' && (
|
|
95
|
+
<StyledCustomHolder>
|
|
96
|
+
<StyledEditableHTML
|
|
97
|
+
onChange={this.changeCustom}
|
|
98
|
+
markup={feedback.custom || ''}
|
|
99
|
+
toolbarOpts={toolbarOpts}
|
|
100
|
+
languageCharactersProps={[{ language: 'spanish' }, { language: 'special' }]}
|
|
101
|
+
mathMlOptions={mathMlOptions}
|
|
102
|
+
/>
|
|
103
|
+
</StyledCustomHolder>
|
|
104
|
+
)}
|
|
105
|
+
|
|
106
|
+
{feedback.type === 'default' && <StyledDefaultHolder> {feedback.default}</StyledDefaultHolder>}
|
|
107
|
+
</StyledFeedbackSelector>
|
|
108
|
+
);
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
export default FeedbackSelector;
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
import PropTypes from 'prop-types';
|
|
2
|
+
import RadioWithLabel from '../radio-with-label';
|
|
3
|
+
import React from 'react';
|
|
4
|
+
import { styled } from '@mui/material/styles';
|
|
5
|
+
|
|
6
|
+
const StyledChoiceHolder = styled('div')(() => ({
|
|
7
|
+
display: 'flex',
|
|
8
|
+
alignItems: 'center',
|
|
9
|
+
}));
|
|
10
|
+
|
|
11
|
+
const StyledChoice = styled('div')(() => ({
|
|
12
|
+
display: 'flex',
|
|
13
|
+
alignItems: 'center',
|
|
14
|
+
}));
|
|
15
|
+
|
|
16
|
+
const StyledRadioWithLabel = styled(RadioWithLabel)(({ theme }) => ({
|
|
17
|
+
'& .MuiFormControlLabel-label': {
|
|
18
|
+
fontSize: theme.typography.fontSize - 2,
|
|
19
|
+
},
|
|
20
|
+
}));
|
|
21
|
+
|
|
22
|
+
const Group = (props) => {
|
|
23
|
+
const { feedbackLabels, value, className, onChange, keys } = props;
|
|
24
|
+
|
|
25
|
+
return (
|
|
26
|
+
<StyledChoiceHolder className={className}>
|
|
27
|
+
{keys.map((key) => {
|
|
28
|
+
return (
|
|
29
|
+
<StyledChoice key={key}>
|
|
30
|
+
<StyledRadioWithLabel
|
|
31
|
+
value={key}
|
|
32
|
+
checked={value === key}
|
|
33
|
+
onChange={(e) => onChange(e.currentTarget.value)}
|
|
34
|
+
label={feedbackLabels[key]}
|
|
35
|
+
/>
|
|
36
|
+
</StyledChoice>
|
|
37
|
+
);
|
|
38
|
+
})}
|
|
39
|
+
</StyledChoiceHolder>
|
|
40
|
+
);
|
|
41
|
+
};
|
|
42
|
+
|
|
43
|
+
Group.propTypes = {
|
|
44
|
+
className: PropTypes.string,
|
|
45
|
+
feedbackLabels: PropTypes.object.isRequired,
|
|
46
|
+
value: PropTypes.string.isRequired,
|
|
47
|
+
keys: PropTypes.arrayOf(PropTypes.string),
|
|
48
|
+
onChange: PropTypes.func,
|
|
49
|
+
};
|
|
50
|
+
|
|
51
|
+
export default Group;
|
|
@@ -0,0 +1,111 @@
|
|
|
1
|
+
import FeedbackSelector, { FeedbackType } from './feedback-selector';
|
|
2
|
+
import PropTypes from 'prop-types';
|
|
3
|
+
import React from 'react';
|
|
4
|
+
import { merge } from 'lodash-es';
|
|
5
|
+
|
|
6
|
+
import { styled } from '@mui/material/styles';
|
|
7
|
+
import Accordion from '@mui/material/Accordion';
|
|
8
|
+
import AccordionSummary from '@mui/material/AccordionSummary';
|
|
9
|
+
import AccordionDetails from '@mui/material/AccordionDetails';
|
|
10
|
+
import Typography from '@mui/material/Typography';
|
|
11
|
+
import ExpandMoreIcon from '@mui/icons-material/ExpandMore';
|
|
12
|
+
|
|
13
|
+
export { FeedbackSelector };
|
|
14
|
+
|
|
15
|
+
const StyledFeedbackContainer = styled('div')(() => ({
|
|
16
|
+
display: 'flex',
|
|
17
|
+
flex: 1,
|
|
18
|
+
flexDirection: 'column',
|
|
19
|
+
}));
|
|
20
|
+
|
|
21
|
+
const StyledAccordionDetails = styled(AccordionDetails)(() => ({
|
|
22
|
+
paddingTop: 0,
|
|
23
|
+
paddingBottom: 0,
|
|
24
|
+
}));
|
|
25
|
+
|
|
26
|
+
export const buildDefaults = (input) => {
|
|
27
|
+
return merge(
|
|
28
|
+
{},
|
|
29
|
+
{
|
|
30
|
+
correct: { type: 'default', default: 'Correct' },
|
|
31
|
+
incorrect: { type: 'default', default: 'Incorrect' },
|
|
32
|
+
partial: { type: 'default', default: 'Nearly' },
|
|
33
|
+
},
|
|
34
|
+
input,
|
|
35
|
+
);
|
|
36
|
+
};
|
|
37
|
+
|
|
38
|
+
export class FeedbackConfig extends React.Component {
|
|
39
|
+
static propTypes = {
|
|
40
|
+
allowPartial: PropTypes.bool,
|
|
41
|
+
className: PropTypes.string,
|
|
42
|
+
feedback: PropTypes.shape({
|
|
43
|
+
correct: PropTypes.shape(FeedbackType),
|
|
44
|
+
incorrect: PropTypes.shape(FeedbackType),
|
|
45
|
+
partial: PropTypes.shape(FeedbackType),
|
|
46
|
+
}),
|
|
47
|
+
onChange: PropTypes.func.isRequired,
|
|
48
|
+
toolbarOpts: PropTypes.object,
|
|
49
|
+
};
|
|
50
|
+
|
|
51
|
+
static defaultProps = {
|
|
52
|
+
allowPartial: true,
|
|
53
|
+
feedback: buildDefaults(),
|
|
54
|
+
};
|
|
55
|
+
|
|
56
|
+
onChange(key, config) {
|
|
57
|
+
const { feedback, onChange } = this.props;
|
|
58
|
+
const update = { ...feedback, [key]: config };
|
|
59
|
+
|
|
60
|
+
onChange(update);
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
onCorrectChange = this.onChange.bind(this, 'correct');
|
|
64
|
+
|
|
65
|
+
onIncorrectChange = this.onChange.bind(this, 'incorrect');
|
|
66
|
+
|
|
67
|
+
onPartialChange = this.onChange.bind(this, 'partial');
|
|
68
|
+
|
|
69
|
+
render() {
|
|
70
|
+
const { className, allowPartial, feedback, toolbarOpts } = this.props;
|
|
71
|
+
|
|
72
|
+
return (
|
|
73
|
+
<div className={className}>
|
|
74
|
+
<Accordion slotProps={{ transition: { timeout: { enter: 225, exit: 195 } } }}>
|
|
75
|
+
<AccordionSummary expandIcon={<ExpandMoreIcon />}>
|
|
76
|
+
<Typography>Feedback</Typography>
|
|
77
|
+
</AccordionSummary>
|
|
78
|
+
|
|
79
|
+
<StyledAccordionDetails>
|
|
80
|
+
<StyledFeedbackContainer>
|
|
81
|
+
<FeedbackSelector
|
|
82
|
+
label="If correct, show"
|
|
83
|
+
feedback={feedback.correct}
|
|
84
|
+
onChange={this.onCorrectChange}
|
|
85
|
+
toolbarOpts={toolbarOpts}
|
|
86
|
+
/>
|
|
87
|
+
|
|
88
|
+
{allowPartial && (
|
|
89
|
+
<FeedbackSelector
|
|
90
|
+
label="If partially correct, show"
|
|
91
|
+
feedback={feedback.partial}
|
|
92
|
+
onChange={this.onPartialChange}
|
|
93
|
+
toolbarOpts={toolbarOpts}
|
|
94
|
+
/>
|
|
95
|
+
)}
|
|
96
|
+
|
|
97
|
+
<FeedbackSelector
|
|
98
|
+
label="If incorrect, show"
|
|
99
|
+
feedback={feedback.incorrect}
|
|
100
|
+
onChange={this.onIncorrectChange}
|
|
101
|
+
toolbarOpts={toolbarOpts}
|
|
102
|
+
/>
|
|
103
|
+
</StyledFeedbackContainer>
|
|
104
|
+
</StyledAccordionDetails>
|
|
105
|
+
</Accordion>
|
|
106
|
+
</div>
|
|
107
|
+
);
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
export default FeedbackConfig;
|
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
import React from 'react';
|
|
2
|
+
import PropTypes from 'prop-types';
|
|
3
|
+
import Typography from '@mui/material/Typography';
|
|
4
|
+
import { styled } from '@mui/material/styles';
|
|
5
|
+
|
|
6
|
+
const StyledFormSection = styled('div')(({ theme }) => ({
|
|
7
|
+
marginTop: theme.spacing(2),
|
|
8
|
+
marginBottom: theme.spacing(2),
|
|
9
|
+
}));
|
|
10
|
+
|
|
11
|
+
const StyledTypography = styled(Typography)(({ theme }) => ({
|
|
12
|
+
marginBottom: theme.spacing(1),
|
|
13
|
+
}));
|
|
14
|
+
|
|
15
|
+
const FormSection = ({ className, label, children, labelExtraStyle }) => (
|
|
16
|
+
<StyledFormSection className={className}>
|
|
17
|
+
<StyledTypography variant="subtitle1" style={labelExtraStyle}>
|
|
18
|
+
{label}
|
|
19
|
+
</StyledTypography>
|
|
20
|
+
{children}
|
|
21
|
+
</StyledFormSection>
|
|
22
|
+
);
|
|
23
|
+
|
|
24
|
+
FormSection.propTypes = {
|
|
25
|
+
className: PropTypes.string,
|
|
26
|
+
label: PropTypes.string,
|
|
27
|
+
children: PropTypes.node,
|
|
28
|
+
labelExtraStyle: PropTypes.object,
|
|
29
|
+
};
|
|
30
|
+
|
|
31
|
+
export default FormSection;
|
package/src/help.jsx
ADDED
|
@@ -0,0 +1,79 @@
|
|
|
1
|
+
import Dialog from '@mui/material/Dialog';
|
|
2
|
+
import DialogTitle from '@mui/material/DialogTitle';
|
|
3
|
+
import DialogContentText from '@mui/material/DialogContentText';
|
|
4
|
+
import DialogContent from '@mui/material/DialogContent';
|
|
5
|
+
import DialogActions from '@mui/material/DialogActions';
|
|
6
|
+
import PropTypes from 'prop-types';
|
|
7
|
+
|
|
8
|
+
import Button from '@mui/material/Button';
|
|
9
|
+
import HelpIcon from '@mui/icons-material/Help';
|
|
10
|
+
import IconButton from '@mui/material/IconButton';
|
|
11
|
+
import React from 'react';
|
|
12
|
+
import { styled } from '@mui/material/styles';
|
|
13
|
+
|
|
14
|
+
const StyledIconButton = styled(IconButton)(({ theme }) => ({
|
|
15
|
+
'&:hover': {
|
|
16
|
+
color: theme.palette.grey[300],
|
|
17
|
+
},
|
|
18
|
+
}));
|
|
19
|
+
|
|
20
|
+
export const HelpButton = ({ onClick }) => (
|
|
21
|
+
<StyledIconButton onClick={onClick} size="large">
|
|
22
|
+
<HelpIcon />
|
|
23
|
+
</StyledIconButton>
|
|
24
|
+
);
|
|
25
|
+
|
|
26
|
+
HelpButton.propTypes = {
|
|
27
|
+
onClick: PropTypes.func,
|
|
28
|
+
};
|
|
29
|
+
|
|
30
|
+
export const HelpDialog = ({ open, onClose, children, title }) => (
|
|
31
|
+
<Dialog open={open} onClose={onClose}>
|
|
32
|
+
<DialogTitle>{title}</DialogTitle>
|
|
33
|
+
<DialogContent>
|
|
34
|
+
<DialogContentText>{children}</DialogContentText>
|
|
35
|
+
</DialogContent>
|
|
36
|
+
<DialogActions>
|
|
37
|
+
<Button onClick={onClose} color="primary">
|
|
38
|
+
OK
|
|
39
|
+
</Button>
|
|
40
|
+
</DialogActions>
|
|
41
|
+
</Dialog>
|
|
42
|
+
);
|
|
43
|
+
|
|
44
|
+
HelpDialog.propTypes = {
|
|
45
|
+
open: PropTypes.bool,
|
|
46
|
+
onClose: PropTypes.func,
|
|
47
|
+
children: PropTypes.oneOfType([PropTypes.arrayOf(PropTypes.node), PropTypes.node]).isRequired,
|
|
48
|
+
title: PropTypes.string.isRequired,
|
|
49
|
+
};
|
|
50
|
+
|
|
51
|
+
class Help extends React.Component {
|
|
52
|
+
static propTypes = {
|
|
53
|
+
children: PropTypes.oneOfType([PropTypes.arrayOf(PropTypes.node), PropTypes.node]).isRequired,
|
|
54
|
+
title: PropTypes.string,
|
|
55
|
+
};
|
|
56
|
+
|
|
57
|
+
constructor(props) {
|
|
58
|
+
super(props);
|
|
59
|
+
this.state = {
|
|
60
|
+
open: false,
|
|
61
|
+
};
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
render() {
|
|
65
|
+
const { children, title } = this.props;
|
|
66
|
+
|
|
67
|
+
return (
|
|
68
|
+
<div>
|
|
69
|
+
<HelpButton color="accent" onClick={() => this.setState({ open: true })} />
|
|
70
|
+
|
|
71
|
+
<HelpDialog open={this.state.open} title={title} onClose={() => this.setState({ open: false })}>
|
|
72
|
+
{children}
|
|
73
|
+
</HelpDialog>
|
|
74
|
+
</div>
|
|
75
|
+
);
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
export default Help;
|
package/src/index.js
ADDED
|
@@ -0,0 +1,55 @@
|
|
|
1
|
+
import AlertDialog from './alert-dialog';
|
|
2
|
+
import FeedbackConfig, { buildDefaults as feedbackConfigDefaults, FeedbackSelector } from './feedback-config';
|
|
3
|
+
import { InputCheckbox, InputRadio, InputSwitch } from './inputs';
|
|
4
|
+
import Langs, { LanguageControls } from './langs';
|
|
5
|
+
import Tabs from './tabs';
|
|
6
|
+
import Checkbox from './checkbox';
|
|
7
|
+
import FormSection from './form-section';
|
|
8
|
+
import Help from './help';
|
|
9
|
+
import Input from './input';
|
|
10
|
+
import { InputContainer } from '@pie-lib/render-ui';
|
|
11
|
+
import NumberTextField from './number-text-field';
|
|
12
|
+
import NumberTextFieldCustom from './number-text-field-custom';
|
|
13
|
+
import TwoChoice, { NChoice } from './two-choice';
|
|
14
|
+
import TagsInput from './tags-input';
|
|
15
|
+
import MuiBox from './mui-box';
|
|
16
|
+
import ChoiceConfiguration from './choice-configuration';
|
|
17
|
+
import * as layout from './layout';
|
|
18
|
+
|
|
19
|
+
import * as choiceUtils from './choice-utils';
|
|
20
|
+
import withStatefulModel from './with-stateful-model';
|
|
21
|
+
import Toggle from './settings/toggle';
|
|
22
|
+
import DisplaySize from './settings/display-size';
|
|
23
|
+
|
|
24
|
+
import * as settings from './settings';
|
|
25
|
+
|
|
26
|
+
export {
|
|
27
|
+
AlertDialog,
|
|
28
|
+
FeedbackConfig,
|
|
29
|
+
FeedbackSelector,
|
|
30
|
+
feedbackConfigDefaults,
|
|
31
|
+
InputCheckbox,
|
|
32
|
+
InputSwitch,
|
|
33
|
+
InputRadio,
|
|
34
|
+
Langs,
|
|
35
|
+
LanguageControls,
|
|
36
|
+
Tabs,
|
|
37
|
+
Checkbox,
|
|
38
|
+
FormSection,
|
|
39
|
+
Help,
|
|
40
|
+
Input,
|
|
41
|
+
InputContainer,
|
|
42
|
+
NumberTextField,
|
|
43
|
+
NumberTextFieldCustom,
|
|
44
|
+
TwoChoice,
|
|
45
|
+
NChoice,
|
|
46
|
+
TagsInput,
|
|
47
|
+
MuiBox,
|
|
48
|
+
ChoiceConfiguration,
|
|
49
|
+
layout,
|
|
50
|
+
choiceUtils,
|
|
51
|
+
withStatefulModel,
|
|
52
|
+
Toggle,
|
|
53
|
+
DisplaySize,
|
|
54
|
+
settings,
|
|
55
|
+
};
|