@pie-lib/render-ui 4.15.10-next.1 → 4.16.0-beta.2

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.
Files changed (58) hide show
  1. package/CHANGELOG.md +27 -43
  2. package/NEXT.CHANGELOG.json +1 -0
  3. package/package.json +6 -4
  4. package/src/__tests__/__snapshots__/html-and-math.test.js.snap +11 -0
  5. package/src/__tests__/__snapshots__/preview-prompt.test.jsx.snap +37 -0
  6. package/src/__tests__/__snapshots__/purpose.test.jsx.snap +42 -0
  7. package/src/__tests__/__snapshots__/readable.test.jsx.snap +64 -0
  8. package/src/__tests__/__snapshots__/response-indicators.test.jsx.snap +95 -0
  9. package/src/__tests__/color.test.js +12 -0
  10. package/src/__tests__/has-media.test.js +20 -0
  11. package/src/__tests__/has-text.test.js +21 -0
  12. package/src/__tests__/html-and-math.test.js +46 -0
  13. package/src/__tests__/preview-prompt.test.jsx +56 -0
  14. package/src/__tests__/purpose.test.jsx +47 -0
  15. package/src/__tests__/readable.test.jsx +64 -0
  16. package/src/__tests__/response-indicators.test.jsx +16 -0
  17. package/src/__tests__/ui-layout.test.jsx +34 -0
  18. package/src/__tests__/withUndoReset.test.jsx +254 -0
  19. package/src/append-css-rules.js +51 -0
  20. package/src/assets/enableAudioAutoplayImage.js +1 -0
  21. package/src/collapsible/__tests__/__snapshots__/index.test.jsx.snap +18 -0
  22. package/src/collapsible/__tests__/index.test.jsx +13 -0
  23. package/src/collapsible/index.jsx +1 -0
  24. package/src/color.js +40 -0
  25. package/src/feedback.jsx +0 -1
  26. package/src/has-media.js +16 -0
  27. package/src/has-text.js +5 -1
  28. package/src/index.js +8 -0
  29. package/src/preview-layout.jsx +14 -3
  30. package/src/preview-prompt.jsx +150 -26
  31. package/src/ui-layout.jsx +66 -0
  32. package/README.md +0 -33
  33. package/lib/collapsible/index.js +0 -134
  34. package/lib/collapsible/index.js.map +0 -1
  35. package/lib/color.js +0 -157
  36. package/lib/color.js.map +0 -1
  37. package/lib/feedback.js +0 -151
  38. package/lib/feedback.js.map +0 -1
  39. package/lib/has-text.js +0 -24
  40. package/lib/has-text.js.map +0 -1
  41. package/lib/html-and-math.js +0 -90
  42. package/lib/html-and-math.js.map +0 -1
  43. package/lib/index.js +0 -104
  44. package/lib/index.js.map +0 -1
  45. package/lib/input-container.js +0 -60
  46. package/lib/input-container.js.map +0 -1
  47. package/lib/preview-layout.js +0 -133
  48. package/lib/preview-layout.js.map +0 -1
  49. package/lib/preview-prompt.js +0 -206
  50. package/lib/preview-prompt.js.map +0 -1
  51. package/lib/purpose.js +0 -28
  52. package/lib/purpose.js.map +0 -1
  53. package/lib/readable.js +0 -28
  54. package/lib/readable.js.map +0 -1
  55. package/lib/response-indicators.js +0 -151
  56. package/lib/response-indicators.js.map +0 -1
  57. package/lib/withUndoReset.js +0 -181
  58. package/lib/withUndoReset.js.map +0 -1
@@ -0,0 +1,64 @@
1
+ import React from 'react';
2
+ import { mount } from 'enzyme';
3
+ import Readable from '../readable';
4
+
5
+ describe('Readable', () => {
6
+ let wrapper;
7
+
8
+ describe('renders fine', () => {
9
+ it('renders child unaltered', () => {
10
+ wrapper = mount(
11
+ <Readable>
12
+ <div>text</div>
13
+ </Readable>,
14
+ );
15
+ expect(wrapper.find('div')).toHaveLength(1);
16
+ expect(wrapper.html().includes('data-pie-readable="true"')).toEqual(true);
17
+ expect(wrapper.html().includes('text')).toEqual(true);
18
+ expect(wrapper).toMatchSnapshot();
19
+ });
20
+ it('renders children unaltered', () => {
21
+ wrapper = mount(
22
+ <Readable>
23
+ <div>
24
+ <div>text1</div>
25
+ <div>text2</div>
26
+ </div>
27
+ </Readable>,
28
+ );
29
+ expect(wrapper.find('div')).toHaveLength(3);
30
+ expect(wrapper.html().includes('data-pie-readable="true"')).toEqual(true);
31
+ expect(wrapper.html().includes('text1')).toEqual(true);
32
+ expect(wrapper.html().includes('text3')).toEqual(false);
33
+ expect(wrapper).toMatchSnapshot();
34
+ });
35
+ it('renders with false tag', () => {
36
+ wrapper = mount(
37
+ <Readable false>
38
+ <div>
39
+ <div>text1</div>
40
+ <div>text2</div>
41
+ </div>
42
+ </Readable>,
43
+ );
44
+ expect(wrapper.find('div')).toHaveLength(3);
45
+ expect(wrapper.html().includes('data-pie-readable="false"')).toEqual(true);
46
+ expect(wrapper.html().includes('text1')).toEqual(true);
47
+ expect(wrapper.html().includes('text3')).toEqual(false);
48
+ expect(wrapper).toMatchSnapshot();
49
+ });
50
+ it('renders even with specific true tag', () => {
51
+ wrapper = mount(
52
+ <Readable false={true}>
53
+ <div>
54
+ <div>text1</div>
55
+ <div>text2</div>
56
+ </div>
57
+ </Readable>,
58
+ );
59
+ expect(wrapper.find('div')).toHaveLength(3);
60
+ expect(wrapper.html().includes('data-pie-readable="false"')).toEqual(true);
61
+ expect(wrapper).toMatchSnapshot();
62
+ });
63
+ });
64
+ });
@@ -0,0 +1,16 @@
1
+ import React from 'react';
2
+ import debug from 'debug';
3
+ import renderer from 'react-test-renderer';
4
+ import { Correct } from '../response-indicators';
5
+
6
+ describe('response-indicators', () => {
7
+ it('snapshot - no feedback ', () => {
8
+ const tree = renderer.create(<Correct />).toJSON();
9
+ expect(tree).toMatchSnapshot();
10
+ });
11
+
12
+ it('snapshot - with feedback ', () => {
13
+ const tree = renderer.create(<Correct feedback="hi" />).toJSON();
14
+ expect(tree).toMatchSnapshot();
15
+ });
16
+ });
@@ -0,0 +1,34 @@
1
+ import React from 'react';
2
+ import { mount, shallow } from 'enzyme';
3
+ import { UiLayout } from '../index';
4
+
5
+ describe('UiLayout', () => {
6
+ let wrapper;
7
+ const mockClasses = { extraCSSRules: 'extra-class' };
8
+ const fontSizeFactor = 1.5;
9
+
10
+ // Mock `getComputedStyle` to return a specific root font size.
11
+ jest.spyOn(window, 'getComputedStyle').mockImplementation(() => ({
12
+ fontSize: '16px', // Default font size for root
13
+ }));
14
+
15
+ wrapper = mount(<UiLayout classes={mockClasses} fontSizeFactor={fontSizeFactor} />);
16
+
17
+ it('renders correctly', () => {
18
+ expect(wrapper.exists()).toBe(true);
19
+ });
20
+
21
+ it('applies the correct classes', () => {
22
+ const div = wrapper.find('.extra-class');
23
+ expect(div.exists()).toBe(true);
24
+ });
25
+
26
+ it('computes style correctly based on fontSizeFactor', () => {
27
+ const div = wrapper.find('.extra-class');
28
+
29
+ // Get the style property of the rendered div
30
+ const computedStyle = div.getDOMNode().style;
31
+ // Assert the computed font size
32
+ expect(computedStyle.fontSize).toBe('24px');
33
+ });
34
+ });
@@ -0,0 +1,254 @@
1
+ import * as React from 'react';
2
+ import withUndoReset from '../withUndoReset';
3
+ import { mount, shallow } from 'enzyme';
4
+ import { shallowChild } from '@pie-lib/test-utils';
5
+
6
+ describe('withUndoReset', () => {
7
+ let wrapper;
8
+ let defaultProps;
9
+ const WrappedClass = class WrappedComponent extends React.Component {
10
+ onSessionChange = (session) => {
11
+ this.props.onSessionChange(session);
12
+ };
13
+
14
+ onAddItem = (item) => {
15
+ this.onSessionChange({
16
+ ...this.props.session,
17
+ items: this.props.session.items.concat(item),
18
+ });
19
+ };
20
+
21
+ onRemoveLastItem = () => {
22
+ const newItems = [...this.props.session.items];
23
+
24
+ newItems.pop();
25
+
26
+ this.onSessionChange({
27
+ ...this.props.session,
28
+ items: newItems,
29
+ });
30
+ };
31
+
32
+ render() {
33
+ const { session } = this.props;
34
+ const items = session.items || {};
35
+
36
+ return (
37
+ <div>
38
+ {items.map((item) => (
39
+ <span key={item.id}>{item.id}</span>
40
+ ))}
41
+ </div>
42
+ );
43
+ }
44
+ };
45
+
46
+ const Component = withUndoReset(WrappedClass);
47
+
48
+ describe('retains internal component functionality', () => {
49
+ beforeEach(() => {
50
+ defaultProps = {
51
+ session: {
52
+ items: [],
53
+ },
54
+ onSessionChange: jest.fn(),
55
+ };
56
+ });
57
+
58
+ it('moves past HoC and returns the rendered component using enzyme', () => {
59
+ wrapper = mount(<Component {...defaultProps} />);
60
+ expect(wrapper.find(WrappedClass).length).toEqual(1);
61
+ expect(wrapper.find(WrappedClass).props()).toEqual(
62
+ expect.objectContaining({
63
+ session: { items: [] },
64
+ }),
65
+ );
66
+ });
67
+
68
+ it('contains the reset logic container', () => {
69
+ wrapper = mount(<Component {...defaultProps} />);
70
+ expect(wrapper.html().includes('Start Over')).toEqual(true);
71
+ expect(wrapper.html().includes('Undo')).toEqual(true);
72
+ });
73
+
74
+ it('can interact with session', () => {
75
+ wrapper = mount(<Component {...defaultProps} />);
76
+
77
+ expect(wrapper.find('span').length).toEqual(2);
78
+
79
+ wrapper
80
+ .find(WrappedClass)
81
+ .instance()
82
+ .onAddItem({ id: 1 });
83
+
84
+ expect(
85
+ wrapper
86
+ .find(WrappedClass)
87
+ .html()
88
+ .includes('span'),
89
+ ).toEqual(true);
90
+
91
+ wrapper
92
+ .find(WrappedClass)
93
+ .instance()
94
+ .onRemoveLastItem();
95
+
96
+ expect(
97
+ wrapper
98
+ .find(WrappedClass)
99
+ .html()
100
+ .includes('span'),
101
+ ).toEqual(false);
102
+ });
103
+
104
+ it('can undo changes in the session', () => {
105
+ wrapper = shallowChild(Component, defaultProps, 1)();
106
+ const instance = wrapper.instance();
107
+
108
+ instance.onSessionChange({
109
+ items: [
110
+ {
111
+ id: 2,
112
+ },
113
+ ],
114
+ });
115
+
116
+ instance.onSessionChange({
117
+ items: [
118
+ {
119
+ id: 2,
120
+ },
121
+ {
122
+ id: 3,
123
+ },
124
+ ],
125
+ });
126
+
127
+ expect(wrapper.state().changes).toEqual([
128
+ {
129
+ items: [
130
+ {
131
+ id: 2,
132
+ },
133
+ ],
134
+ },
135
+ {
136
+ items: [
137
+ {
138
+ id: 2,
139
+ },
140
+ {
141
+ id: 3,
142
+ },
143
+ ],
144
+ },
145
+ ]);
146
+
147
+ expect(wrapper.state().session).toEqual({
148
+ items: [
149
+ {
150
+ id: 2,
151
+ },
152
+ {
153
+ id: 3,
154
+ },
155
+ ],
156
+ });
157
+
158
+ instance.onUndo();
159
+
160
+ expect(wrapper.state().changes).toEqual([
161
+ {
162
+ items: [
163
+ {
164
+ id: 2,
165
+ },
166
+ ],
167
+ },
168
+ ]);
169
+
170
+ expect(wrapper.state().session).toEqual({
171
+ items: [
172
+ {
173
+ id: 2,
174
+ },
175
+ ],
176
+ });
177
+
178
+ instance.onUndo();
179
+
180
+ expect(wrapper.state().changes).toEqual([]);
181
+
182
+ expect(wrapper.state().session).toEqual({
183
+ items: [],
184
+ });
185
+
186
+ expect(wrapper.find(WrappedClass).html()).toEqual('<div></div>');
187
+ });
188
+
189
+ it('can reset changes in the session', () => {
190
+ wrapper = shallowChild(Component, defaultProps, 1)();
191
+ const instance = wrapper.instance();
192
+
193
+ instance.onSessionChange({
194
+ items: [
195
+ {
196
+ id: 2,
197
+ },
198
+ ],
199
+ });
200
+
201
+ instance.onSessionChange({
202
+ items: [
203
+ {
204
+ id: 2,
205
+ },
206
+ {
207
+ id: 3,
208
+ },
209
+ ],
210
+ });
211
+
212
+ expect(wrapper.state().changes).toEqual([
213
+ {
214
+ items: [
215
+ {
216
+ id: 2,
217
+ },
218
+ ],
219
+ },
220
+ {
221
+ items: [
222
+ {
223
+ id: 2,
224
+ },
225
+ {
226
+ id: 3,
227
+ },
228
+ ],
229
+ },
230
+ ]);
231
+
232
+ expect(wrapper.state().session).toEqual({
233
+ items: [
234
+ {
235
+ id: 2,
236
+ },
237
+ {
238
+ id: 3,
239
+ },
240
+ ],
241
+ });
242
+
243
+ instance.onReset();
244
+
245
+ expect(wrapper.state().changes).toEqual([]);
246
+
247
+ expect(wrapper.state().session).toEqual({
248
+ items: [],
249
+ });
250
+
251
+ expect(wrapper.find(WrappedClass).html()).toEqual('<div></div>');
252
+ });
253
+ });
254
+ });
@@ -0,0 +1,51 @@
1
+ import React from 'react';
2
+ import PropTypes from 'prop-types';
3
+ import isEmpty from 'lodash/isEmpty';
4
+
5
+ class ExtraCSSRulesMixin extends React.Component {
6
+ static propTypes = {
7
+ extraCSSRules: PropTypes.shape({
8
+ names: PropTypes.arrayOf(PropTypes.string),
9
+ rules: PropTypes.string,
10
+ }),
11
+ };
12
+
13
+ static defaultProps = {
14
+ extraCSSRules: {},
15
+ };
16
+
17
+ constructor(props) {
18
+ super(props);
19
+ this.classesSheet = document.createElement('style');
20
+ }
21
+
22
+ componentDidMount() {
23
+ this.appendExtraStyles();
24
+ }
25
+
26
+ appendExtraStyles = () => {
27
+ if (isEmpty(this.props.extraCSSRules) || !this.classesSheet) {
28
+ return;
29
+ }
30
+
31
+ const headElement = document.head || document.getElementsByTagName('head')[0];
32
+
33
+ if (!headElement) {
34
+ return;
35
+ }
36
+
37
+ if (!this.classesSheet.parentElement) {
38
+ headElement.appendChild(this.classesSheet);
39
+ }
40
+
41
+ const { extraCSSRules, classes } = this.props;
42
+
43
+ this.classesSheet.innerHTML = `
44
+ .${classes.extraCSSRules} {
45
+ ${extraCSSRules.rules}
46
+ }
47
+ `;
48
+ };
49
+ }
50
+
51
+ export default ExtraCSSRulesMixin;
@@ -0,0 +1 @@
1
+ export default '';
@@ -0,0 +1,18 @@
1
+ // Jest Snapshot v1, https://goo.gl/fbAQLP
2
+
3
+ exports[`collapsible snapshot renders 1`] = `
4
+ <div>
5
+ <div
6
+ onClick={[Function]}
7
+ >
8
+ <span>
9
+ Show
10
+ </span>
11
+ </div>
12
+ <WithStyles(Collapse)
13
+ in={false}
14
+ timeout="auto"
15
+ unmountOnExit={true}
16
+ />
17
+ </div>
18
+ `;
@@ -0,0 +1,13 @@
1
+ import toJson from 'enzyme-to-json';
2
+ import { shallow } from 'enzyme/build';
3
+ import { Collapsible } from '../index';
4
+ import React from 'react';
5
+
6
+ describe('collapsible', () => {
7
+ describe('snapshot', () => {
8
+ it('renders', () => {
9
+ const wrapper = shallow(<Collapsible classes={{}} />);
10
+ expect(toJson(wrapper)).toMatchSnapshot();
11
+ });
12
+ });
13
+ });
@@ -56,6 +56,7 @@ export default withStyles((theme) => ({
56
56
  title: {
57
57
  color: theme.palette.primary.light,
58
58
  borderBottom: `1px dotted ${theme.palette.primary.light}`,
59
+ cursor: 'pointer',
59
60
  },
60
61
  collapsible: {
61
62
  paddingTop: theme.spacing.unit * 2,
package/src/color.js CHANGED
@@ -7,19 +7,38 @@ import red from '@material-ui/core/colors/red';
7
7
  export const defaults = {
8
8
  TEXT: 'black',
9
9
  DISABLED: 'grey',
10
+ DISABLED_SECONDARY: '#ABABAB',
10
11
  CORRECT: green[500],
11
12
  CORRECT_SECONDARY: green[50],
13
+ CORRECT_WITH_ICON: '#087D38',
12
14
  INCORRECT: orange[500],
13
15
  INCORRECT_SECONDARY: red[50],
16
+ INCORRECT_WITH_ICON: '#BF0D00',
14
17
  MISSING: red[700],
18
+ MISSING_WITH_ICON: '#6A78A1',
15
19
  PRIMARY: indigo[500],
16
20
  PRIMARY_LIGHT: indigo[200],
17
21
  PRIMARY_DARK: indigo[800],
18
22
  SECONDARY: pink.A400,
19
23
  SECONDARY_LIGHT: pink[200],
20
24
  SECONDARY_DARK: pink[900],
25
+ TERTIARY: '#146EB3',
26
+ TERTIARY_LIGHT: '#D0E2F0',
21
27
  BACKGROUND: 'rgba(255,255,255,0)',
28
+ BACKGROUND_DARK: '#ECEDF1',
29
+ // this is only used for multi-trait-rubric, we might want to use BACKGROUND_DARK instead
22
30
  SECONDARY_BACKGROUND: 'rgba(241,241,241,1)',
31
+ BORDER: '#9A9A9A',
32
+ BORDER_LIGHT: '#D1D1D1',
33
+ BORDER_DARK: '#646464',
34
+ BLACK: '#000000',
35
+ WHITE: '#ffffff',
36
+ TRANSPARENT: 'transparent',
37
+ // this is used for multiple-choice accessibility
38
+ FOCUS_CHECKED: '#BBDEFB',
39
+ FOCUS_CHECKED_BORDER: '#1565C0',
40
+ FOCUS_UNCHECKED: '#E0E0E0',
41
+ FOCUS_UNCHECKED_BORDER: '#757575',
23
42
  };
24
43
 
25
44
  Object.freeze(defaults);
@@ -35,11 +54,15 @@ const pv = v('pie');
35
54
 
36
55
  export const text = () => pv('text', defaults.TEXT);
37
56
  export const disabled = () => pv('disabled', defaults.DISABLED);
57
+ export const disabledSecondary = () => pv('disabled-secondary', defaults.DISABLED_SECONDARY);
38
58
  export const correct = () => pv('correct', defaults.CORRECT);
39
59
  export const correctSecondary = () => pv('correct-secondary', defaults.CORRECT_SECONDARY);
60
+ export const correctWithIcon = () => pv('correct-icon', defaults.CORRECT_WITH_ICON);
40
61
  export const incorrect = () => pv('incorrect', defaults.INCORRECT);
62
+ export const incorrectWithIcon = () => pv('incorrect-icon', defaults.INCORRECT_WITH_ICON);
41
63
  export const incorrectSecondary = () => pv('incorrect-secondary', defaults.INCORRECT_SECONDARY);
42
64
  export const missing = () => pv('missing', defaults.MISSING);
65
+ export const missingWithIcon = () => pv('missing-icon', defaults.MISSING_WITH_ICON);
43
66
 
44
67
  export const primary = () => pv('primary', defaults.PRIMARY);
45
68
  export const primaryLight = () => pv('primary-light', defaults.PRIMARY_LIGHT);
@@ -52,4 +75,21 @@ export const secondaryDark = () => pv('secondary-dark', defaults.SECONDARY_DARK)
52
75
 
53
76
  export const secondaryText = () => pv('secondary-text', 'text', defaults.TEXT);
54
77
  export const background = () => pv('background', defaults.BACKGROUND);
78
+ export const backgroundDark = () => pv('background-dark', defaults.BACKGROUND_DARK);
55
79
  export const secondaryBackground = () => pv('secondary-background', defaults.SECONDARY_BACKGROUND);
80
+
81
+ export const tertiary = () => pv('tertiary', defaults.TERTIARY);
82
+ export const tertiaryLight = () => pv('tertiary-light', defaults.TERTIARY_LIGHT);
83
+
84
+ export const border = () => pv('border', defaults.BORDER);
85
+ export const borderLight = () => pv('border-light', defaults.BORDER_LIGHT);
86
+ export const borderDark = () => pv('border-dark', defaults.BORDER_DARK);
87
+
88
+ export const black = () => pv('black', defaults.BLACK);
89
+ export const white = () => pv('white', defaults.WHITE);
90
+ export const transparent = () => defaults.TRANSPARENT;
91
+
92
+ export const focusChecked = () => pv('focus-checked', defaults.FOCUS_CHECKED);
93
+ export const focusCheckedBorder = () => pv('focus-checked-border', defaults.FOCUS_CHECKED_BORDER);
94
+ export const focusUnchecked = () => pv('focus-unchecked', defaults.FOCUS_UNCHECKED);
95
+ export const focusUncheckedBorder = () => pv('focus-unchecked-border', defaults.FOCUS_UNCHECKED_BORDER);
package/src/feedback.jsx CHANGED
@@ -24,7 +24,6 @@ const styleSheet = {
24
24
  backgroundColor: `var(--feedback-bg-color, ${color.disabled()})`,
25
25
  borderRadius: '4px',
26
26
  fontFamily: '"Roboto", "Noto", sans-serif',
27
- fontSize: '12px',
28
27
  lineHeight: '25px',
29
28
  margin: '0px',
30
29
  padding: '10px',
@@ -0,0 +1,16 @@
1
+ let parser;
2
+
3
+ if (typeof window !== 'undefined') {
4
+ parser = new DOMParser();
5
+ }
6
+
7
+ /*
8
+ * Check if the string contains at least one media element.
9
+ */
10
+ export const hasMedia = (s) => {
11
+ if (!s) {
12
+ return false;
13
+ }
14
+ const root = dp.parseFromString(s, 'text/html');
15
+ return !!root.body.querySelector('img') || !!root.body.querySelector('video') || !!root.body.querySelector('audio');
16
+ };
package/src/has-text.js CHANGED
@@ -1,4 +1,8 @@
1
- const dp = new DOMParser();
1
+ let parser;
2
+
3
+ if (typeof window !== 'undefined') {
4
+ parser = new DOMParser();
5
+ }
2
6
 
3
7
  const markupToText = (s) => {
4
8
  const root = dp.parseFromString(s, 'text/html');