@pie-lib/render-ui 4.31.1-next.0 → 4.32.1
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/esm/index.js +1159 -0
- package/esm/index.js.map +1 -0
- package/package.json +12 -5
package/esm/index.js
ADDED
|
@@ -0,0 +1,1159 @@
|
|
|
1
|
+
import * as React from 'react';
|
|
2
|
+
import React__default, { Component } from 'react';
|
|
3
|
+
import PropTypes from 'prop-types';
|
|
4
|
+
import * as icons from '@pie-lib/icons';
|
|
5
|
+
import Popover from '@material-ui/core/Popover';
|
|
6
|
+
import { withStyles, createMuiTheme, MuiThemeProvider } from '@material-ui/core/styles';
|
|
7
|
+
import { TransitionGroup, CSSTransition } from 'react-transition-group';
|
|
8
|
+
import classNames from 'classnames';
|
|
9
|
+
import green from '@material-ui/core/colors/green';
|
|
10
|
+
import orange from '@material-ui/core/colors/orange';
|
|
11
|
+
import pink from '@material-ui/core/colors/pink';
|
|
12
|
+
import indigo from '@material-ui/core/colors/indigo';
|
|
13
|
+
import red from '@material-ui/core/colors/red';
|
|
14
|
+
import debug from 'debug';
|
|
15
|
+
import { withStyles as withStyles$1 } from '@material-ui/core/styles/index';
|
|
16
|
+
import Collapse from '@material-ui/core/Collapse/index';
|
|
17
|
+
import { renderMath } from '@pie-lib/math-rendering';
|
|
18
|
+
import Button from '@material-ui/core/Button';
|
|
19
|
+
import Restore from '@material-ui/icons/Restore';
|
|
20
|
+
import Undo from '@material-ui/icons/Undo';
|
|
21
|
+
import isEmpty from 'lodash/isEmpty';
|
|
22
|
+
import InputLabel from '@material-ui/core/InputLabel';
|
|
23
|
+
import FormControl from '@material-ui/core/FormControl';
|
|
24
|
+
|
|
25
|
+
const defaults = {
|
|
26
|
+
TEXT: 'black',
|
|
27
|
+
DISABLED: 'grey',
|
|
28
|
+
DISABLED_SECONDARY: '#ABABAB',
|
|
29
|
+
CORRECT: green[500],
|
|
30
|
+
CORRECT_SECONDARY: green[50],
|
|
31
|
+
CORRECT_TERTIARY: '#0EA449',
|
|
32
|
+
CORRECT_WITH_ICON: '#087D38',
|
|
33
|
+
INCORRECT: orange[500],
|
|
34
|
+
INCORRECT_SECONDARY: red[50],
|
|
35
|
+
INCORRECT_WITH_ICON: '#BF0D00',
|
|
36
|
+
MISSING: red[700],
|
|
37
|
+
MISSING_WITH_ICON: '#6A78A1',
|
|
38
|
+
PRIMARY: indigo[500],
|
|
39
|
+
PRIMARY_LIGHT: indigo[200],
|
|
40
|
+
PRIMARY_DARK: indigo[800],
|
|
41
|
+
SECONDARY: pink.A400,
|
|
42
|
+
SECONDARY_LIGHT: pink[200],
|
|
43
|
+
SECONDARY_DARK: pink[900],
|
|
44
|
+
TERTIARY: '#146EB3',
|
|
45
|
+
TERTIARY_LIGHT: '#D0E2F0',
|
|
46
|
+
BACKGROUND: 'rgba(255,255,255,0)',
|
|
47
|
+
BACKGROUND_DARK: '#ECEDF1',
|
|
48
|
+
DROPDOWN_BACKGROUND: '#E0E1E6',
|
|
49
|
+
// this is used for inline-dropdown
|
|
50
|
+
// this is only used for multi-trait-rubric, we might want to use BACKGROUND_DARK instead
|
|
51
|
+
SECONDARY_BACKGROUND: 'rgba(241,241,241,1)',
|
|
52
|
+
BORDER: '#9A9A9A',
|
|
53
|
+
BORDER_LIGHT: '#D1D1D1',
|
|
54
|
+
BORDER_DARK: '#646464',
|
|
55
|
+
BORDER_GRAY: '#7E8494',
|
|
56
|
+
BLACK: '#000000',
|
|
57
|
+
WHITE: '#ffffff',
|
|
58
|
+
TRANSPARENT: 'transparent',
|
|
59
|
+
// this is used for multiple-choice accessibility
|
|
60
|
+
FOCUS_CHECKED: '#BBDEFB',
|
|
61
|
+
FOCUS_CHECKED_BORDER: '#1565C0',
|
|
62
|
+
FOCUS_UNCHECKED: '#E0E0E0',
|
|
63
|
+
FOCUS_UNCHECKED_BORDER: '#757575',
|
|
64
|
+
// this is used for select text tokens
|
|
65
|
+
BLUE_GREY100: '#F3F5F7',
|
|
66
|
+
BLUE_GREY300: '#C0C3CF',
|
|
67
|
+
BLUE_GREY600: '#7E8494',
|
|
68
|
+
BLUE_GREY900: '#152452',
|
|
69
|
+
// this is used for charting
|
|
70
|
+
FADED_PRIMARY: '#DCDAFB'
|
|
71
|
+
};
|
|
72
|
+
Object.freeze(defaults);
|
|
73
|
+
const v = prefix => (...args) => {
|
|
74
|
+
const fallback = args.pop();
|
|
75
|
+
return args.reduceRight((acc, v) => {
|
|
76
|
+
return `var(--${prefix}-${v}, ${acc})`;
|
|
77
|
+
}, fallback);
|
|
78
|
+
};
|
|
79
|
+
const pv = v('pie');
|
|
80
|
+
const text = () => pv('text', defaults.TEXT);
|
|
81
|
+
const disabled = () => pv('disabled', defaults.DISABLED);
|
|
82
|
+
const disabledSecondary = () => pv('disabled-secondary', defaults.DISABLED_SECONDARY);
|
|
83
|
+
const correct = () => pv('correct', defaults.CORRECT);
|
|
84
|
+
const correctSecondary = () => pv('correct-secondary', defaults.CORRECT_SECONDARY);
|
|
85
|
+
const correctTertiary = () => pv('correct-tertiary', defaults.CORRECT_TERTIARY);
|
|
86
|
+
const correctWithIcon = () => pv('correct-icon', defaults.CORRECT_WITH_ICON);
|
|
87
|
+
const incorrect = () => pv('incorrect', defaults.INCORRECT);
|
|
88
|
+
const incorrectWithIcon = () => pv('incorrect-icon', defaults.INCORRECT_WITH_ICON);
|
|
89
|
+
const incorrectSecondary = () => pv('incorrect-secondary', defaults.INCORRECT_SECONDARY);
|
|
90
|
+
const missing = () => pv('missing', defaults.MISSING);
|
|
91
|
+
const missingWithIcon = () => pv('missing-icon', defaults.MISSING_WITH_ICON);
|
|
92
|
+
const primary = () => pv('primary', defaults.PRIMARY);
|
|
93
|
+
const primaryLight = () => pv('primary-light', defaults.PRIMARY_LIGHT);
|
|
94
|
+
const primaryDark = () => pv('primary-dark', defaults.PRIMARY_DARK);
|
|
95
|
+
const primaryText = () => pv('primary-text', 'text', defaults.TEXT);
|
|
96
|
+
const fadedPrimary = () => pv('faded-primary', defaults.FADED_PRIMARY);
|
|
97
|
+
const secondary = () => pv('secondary', defaults.SECONDARY);
|
|
98
|
+
const secondaryLight = () => pv('secondary-light', defaults.SECONDARY_LIGHT);
|
|
99
|
+
const secondaryDark = () => pv('secondary-dark', defaults.SECONDARY_DARK);
|
|
100
|
+
const secondaryText = () => pv('secondary-text', 'text', defaults.TEXT);
|
|
101
|
+
const background = () => pv('background', defaults.BACKGROUND);
|
|
102
|
+
const backgroundDark = () => pv('background-dark', defaults.BACKGROUND_DARK);
|
|
103
|
+
const secondaryBackground = () => pv('secondary-background', defaults.SECONDARY_BACKGROUND);
|
|
104
|
+
const dropdownBackground = () => pv('dropdown-background', defaults.DROPDOWN_BACKGROUND);
|
|
105
|
+
const tertiary = () => pv('tertiary', defaults.TERTIARY);
|
|
106
|
+
const tertiaryLight = () => pv('tertiary-light', defaults.TERTIARY_LIGHT);
|
|
107
|
+
const border = () => pv('border', defaults.BORDER);
|
|
108
|
+
const borderLight = () => pv('border-light', defaults.BORDER_LIGHT);
|
|
109
|
+
const borderDark = () => pv('border-dark', defaults.BORDER_DARK);
|
|
110
|
+
const borderGray = () => pv('border-gray', defaults.BORDER_GRAY);
|
|
111
|
+
const black = () => pv('black', defaults.BLACK);
|
|
112
|
+
const white = () => pv('white', defaults.WHITE);
|
|
113
|
+
const transparent = () => defaults.TRANSPARENT;
|
|
114
|
+
const focusChecked = () => pv('focus-checked', defaults.FOCUS_CHECKED);
|
|
115
|
+
const focusCheckedBorder = () => pv('focus-checked-border', defaults.FOCUS_CHECKED_BORDER);
|
|
116
|
+
const focusUnchecked = () => pv('focus-unchecked', defaults.FOCUS_UNCHECKED);
|
|
117
|
+
const focusUncheckedBorder = () => pv('focus-unchecked-border', defaults.FOCUS_UNCHECKED_BORDER);
|
|
118
|
+
const blueGrey100 = () => pv('blue-grey-100', defaults.BLUE_GREY100);
|
|
119
|
+
const blueGrey300 = () => pv('blue-grey-300', defaults.BLUE_GREY300);
|
|
120
|
+
const blueGrey600 = () => pv('blue-grey-600', defaults.BLUE_GREY600);
|
|
121
|
+
const blueGrey900 = () => pv('blue-grey-900', defaults.BLUE_GREY900);
|
|
122
|
+
const visualElementsColors = {
|
|
123
|
+
AXIS_LINE_COLOR: '#5A53C9',
|
|
124
|
+
ROLLOVER_FILL_BAR_COLOR: '#050F2D',
|
|
125
|
+
GRIDLINES_COLOR: '#8E88EA',
|
|
126
|
+
PLOT_FILL_COLOR: '#1463B3'
|
|
127
|
+
};
|
|
128
|
+
|
|
129
|
+
var color = /*#__PURE__*/Object.freeze({
|
|
130
|
+
__proto__: null,
|
|
131
|
+
background: background,
|
|
132
|
+
backgroundDark: backgroundDark,
|
|
133
|
+
black: black,
|
|
134
|
+
blueGrey100: blueGrey100,
|
|
135
|
+
blueGrey300: blueGrey300,
|
|
136
|
+
blueGrey600: blueGrey600,
|
|
137
|
+
blueGrey900: blueGrey900,
|
|
138
|
+
border: border,
|
|
139
|
+
borderDark: borderDark,
|
|
140
|
+
borderGray: borderGray,
|
|
141
|
+
borderLight: borderLight,
|
|
142
|
+
correct: correct,
|
|
143
|
+
correctSecondary: correctSecondary,
|
|
144
|
+
correctTertiary: correctTertiary,
|
|
145
|
+
correctWithIcon: correctWithIcon,
|
|
146
|
+
defaults: defaults,
|
|
147
|
+
disabled: disabled,
|
|
148
|
+
disabledSecondary: disabledSecondary,
|
|
149
|
+
dropdownBackground: dropdownBackground,
|
|
150
|
+
fadedPrimary: fadedPrimary,
|
|
151
|
+
focusChecked: focusChecked,
|
|
152
|
+
focusCheckedBorder: focusCheckedBorder,
|
|
153
|
+
focusUnchecked: focusUnchecked,
|
|
154
|
+
focusUncheckedBorder: focusUncheckedBorder,
|
|
155
|
+
incorrect: incorrect,
|
|
156
|
+
incorrectSecondary: incorrectSecondary,
|
|
157
|
+
incorrectWithIcon: incorrectWithIcon,
|
|
158
|
+
missing: missing,
|
|
159
|
+
missingWithIcon: missingWithIcon,
|
|
160
|
+
primary: primary,
|
|
161
|
+
primaryDark: primaryDark,
|
|
162
|
+
primaryLight: primaryLight,
|
|
163
|
+
primaryText: primaryText,
|
|
164
|
+
secondary: secondary,
|
|
165
|
+
secondaryBackground: secondaryBackground,
|
|
166
|
+
secondaryDark: secondaryDark,
|
|
167
|
+
secondaryLight: secondaryLight,
|
|
168
|
+
secondaryText: secondaryText,
|
|
169
|
+
tertiary: tertiary,
|
|
170
|
+
tertiaryLight: tertiaryLight,
|
|
171
|
+
text: text,
|
|
172
|
+
transparent: transparent,
|
|
173
|
+
v: v,
|
|
174
|
+
visualElementsColors: visualElementsColors,
|
|
175
|
+
white: white
|
|
176
|
+
});
|
|
177
|
+
|
|
178
|
+
/**
|
|
179
|
+
* Lifted from multiple-choice - TODO: create a shared package for it.
|
|
180
|
+
*/
|
|
181
|
+
const styleSheet = {
|
|
182
|
+
corespringFeedback: {
|
|
183
|
+
transformOrigin: '0% 0px 0px',
|
|
184
|
+
width: '100%',
|
|
185
|
+
display: 'block',
|
|
186
|
+
overflow: 'hidden',
|
|
187
|
+
'&:.incorrect': {
|
|
188
|
+
color: '#946202'
|
|
189
|
+
}
|
|
190
|
+
},
|
|
191
|
+
content: {
|
|
192
|
+
'-webkit-font-smoothing': 'antialiased',
|
|
193
|
+
backgroundColor: `var(--feedback-bg-color, ${disabled()})`,
|
|
194
|
+
borderRadius: '4px',
|
|
195
|
+
fontFamily: '"Roboto", "Noto", sans-serif',
|
|
196
|
+
lineHeight: '25px',
|
|
197
|
+
margin: '0px',
|
|
198
|
+
padding: '10px',
|
|
199
|
+
verticalAlign: 'middle',
|
|
200
|
+
color: 'var(--feedback-color, white)'
|
|
201
|
+
},
|
|
202
|
+
correct: {
|
|
203
|
+
backgroundColor: `var(--feedback-correct-bg-color, ${correct()})`
|
|
204
|
+
},
|
|
205
|
+
incorrect: {
|
|
206
|
+
backgroundColor: `var(--feedback-incorrect-bg-color, ${incorrect()})`
|
|
207
|
+
},
|
|
208
|
+
feedbackEnter: {
|
|
209
|
+
height: '1px'
|
|
210
|
+
},
|
|
211
|
+
feedbackEnterActive: {
|
|
212
|
+
height: '45px',
|
|
213
|
+
transition: 'height 500ms'
|
|
214
|
+
},
|
|
215
|
+
feedbackLeave: {
|
|
216
|
+
height: '45px'
|
|
217
|
+
},
|
|
218
|
+
feedbackLeaveActive: {
|
|
219
|
+
height: '1px',
|
|
220
|
+
transition: 'height 200ms'
|
|
221
|
+
}
|
|
222
|
+
};
|
|
223
|
+
class Feedback extends React__default.Component {
|
|
224
|
+
render() {
|
|
225
|
+
const {
|
|
226
|
+
correctness,
|
|
227
|
+
feedback,
|
|
228
|
+
classes
|
|
229
|
+
} = this.props;
|
|
230
|
+
|
|
231
|
+
function chooseFeedback(correctness) {
|
|
232
|
+
if (correctness && feedback) {
|
|
233
|
+
return /*#__PURE__*/React__default.createElement(CSSTransition, {
|
|
234
|
+
classNames: {
|
|
235
|
+
enter: classes.feedbackEnter,
|
|
236
|
+
enterActive: classes.feedbackEnterActive,
|
|
237
|
+
leave: classes.feedbackLeave,
|
|
238
|
+
leaveActive: classes.feedbackLeaveActive
|
|
239
|
+
},
|
|
240
|
+
key: "hasFeedback",
|
|
241
|
+
timeout: {
|
|
242
|
+
enter: 500,
|
|
243
|
+
exit: 300
|
|
244
|
+
}
|
|
245
|
+
}, /*#__PURE__*/React__default.createElement("div", {
|
|
246
|
+
className: classes.corespringFeedback
|
|
247
|
+
}, /*#__PURE__*/React__default.createElement("div", {
|
|
248
|
+
className: classNames(classes.content, classes[correctness]),
|
|
249
|
+
dangerouslySetInnerHTML: {
|
|
250
|
+
__html: feedback
|
|
251
|
+
}
|
|
252
|
+
})));
|
|
253
|
+
} else {
|
|
254
|
+
return null;
|
|
255
|
+
}
|
|
256
|
+
}
|
|
257
|
+
|
|
258
|
+
return /*#__PURE__*/React__default.createElement("div", null, /*#__PURE__*/React__default.createElement(TransitionGroup, null, chooseFeedback(correctness)));
|
|
259
|
+
}
|
|
260
|
+
|
|
261
|
+
}
|
|
262
|
+
Feedback.propTypes = {
|
|
263
|
+
correctness: PropTypes.string,
|
|
264
|
+
feedback: PropTypes.string,
|
|
265
|
+
classes: PropTypes.object.isRequired
|
|
266
|
+
};
|
|
267
|
+
var Feedback$1 = withStyles(styleSheet, {
|
|
268
|
+
name: 'Feedback'
|
|
269
|
+
})(Feedback);
|
|
270
|
+
|
|
271
|
+
const log = debug('pie-libs:render-ui:response-indicators');
|
|
272
|
+
|
|
273
|
+
const styles$4 = () => ({
|
|
274
|
+
responseIndicator: {
|
|
275
|
+
cursor: 'pointer'
|
|
276
|
+
},
|
|
277
|
+
paper: {
|
|
278
|
+
padding: '0',
|
|
279
|
+
borderRadius: '4px'
|
|
280
|
+
},
|
|
281
|
+
popover: {
|
|
282
|
+
cursor: 'pointer'
|
|
283
|
+
},
|
|
284
|
+
popperClose: {
|
|
285
|
+
cursor: 'pointer'
|
|
286
|
+
}
|
|
287
|
+
});
|
|
288
|
+
|
|
289
|
+
const BuildIndicator = (Icon, correctness) => {
|
|
290
|
+
class RawIndicator extends React__default.Component {
|
|
291
|
+
constructor(props) {
|
|
292
|
+
super(props);
|
|
293
|
+
|
|
294
|
+
this.handlePopoverOpen = event => {
|
|
295
|
+
log('[handlePopoverOpen]', event.target);
|
|
296
|
+
this.setState({
|
|
297
|
+
anchorEl: event.target
|
|
298
|
+
});
|
|
299
|
+
};
|
|
300
|
+
|
|
301
|
+
this.handlePopoverClose = () => {
|
|
302
|
+
this.setState({
|
|
303
|
+
anchorEl: null
|
|
304
|
+
});
|
|
305
|
+
};
|
|
306
|
+
|
|
307
|
+
this.state = {};
|
|
308
|
+
}
|
|
309
|
+
|
|
310
|
+
render() {
|
|
311
|
+
const {
|
|
312
|
+
feedback,
|
|
313
|
+
classes
|
|
314
|
+
} = this.props;
|
|
315
|
+
const {
|
|
316
|
+
anchorEl
|
|
317
|
+
} = this.state;
|
|
318
|
+
return /*#__PURE__*/React__default.createElement("div", {
|
|
319
|
+
className: feedback && classes.responseIndicator
|
|
320
|
+
}, /*#__PURE__*/React__default.createElement("span", {
|
|
321
|
+
ref: r => this.icon = r,
|
|
322
|
+
onClick: this.handlePopoverOpen
|
|
323
|
+
}, /*#__PURE__*/React__default.createElement(Icon, null)), feedback && /*#__PURE__*/React__default.createElement(Popover, {
|
|
324
|
+
className: classes.popover,
|
|
325
|
+
classes: {
|
|
326
|
+
paper: classes.paper
|
|
327
|
+
},
|
|
328
|
+
open: !!anchorEl,
|
|
329
|
+
anchorEl: anchorEl,
|
|
330
|
+
anchorOrigin: {
|
|
331
|
+
vertical: 'bottom',
|
|
332
|
+
horizontal: 'left'
|
|
333
|
+
},
|
|
334
|
+
transformOrigin: {
|
|
335
|
+
vertical: 'top',
|
|
336
|
+
horizontal: 'left'
|
|
337
|
+
},
|
|
338
|
+
onClose: this.handlePopoverClose
|
|
339
|
+
}, /*#__PURE__*/React__default.createElement(Feedback$1, {
|
|
340
|
+
feedback: feedback,
|
|
341
|
+
correctness: correctness
|
|
342
|
+
})));
|
|
343
|
+
}
|
|
344
|
+
|
|
345
|
+
}
|
|
346
|
+
|
|
347
|
+
RawIndicator.propTypes = {
|
|
348
|
+
feedback: PropTypes.string,
|
|
349
|
+
classes: PropTypes.object.isRequired
|
|
350
|
+
};
|
|
351
|
+
return withStyles(styles$4)(RawIndicator);
|
|
352
|
+
};
|
|
353
|
+
|
|
354
|
+
const Correct = BuildIndicator(icons.Correct, 'correct');
|
|
355
|
+
const Incorrect = BuildIndicator(icons.Incorrect, 'incorrect');
|
|
356
|
+
const PartiallyCorrect = BuildIndicator(icons.PartiallyCorrect, 'partially-correct');
|
|
357
|
+
const NothingSubmitted = BuildIndicator(icons.NothingSubmitted, 'nothing-submitted');
|
|
358
|
+
|
|
359
|
+
var responseIndicators = /*#__PURE__*/Object.freeze({
|
|
360
|
+
__proto__: null,
|
|
361
|
+
Correct: Correct,
|
|
362
|
+
Incorrect: Incorrect,
|
|
363
|
+
NothingSubmitted: NothingSubmitted,
|
|
364
|
+
PartiallyCorrect: PartiallyCorrect
|
|
365
|
+
});
|
|
366
|
+
|
|
367
|
+
class Collapsible extends React__default.Component {
|
|
368
|
+
constructor(...args) {
|
|
369
|
+
super(...args);
|
|
370
|
+
this.state = {
|
|
371
|
+
expanded: false
|
|
372
|
+
};
|
|
373
|
+
|
|
374
|
+
this.toggleExpanded = () => {
|
|
375
|
+
this.setState(state => ({
|
|
376
|
+
expanded: !state.expanded
|
|
377
|
+
}));
|
|
378
|
+
};
|
|
379
|
+
}
|
|
380
|
+
|
|
381
|
+
componentDidMount() {
|
|
382
|
+
renderMath(this.root);
|
|
383
|
+
}
|
|
384
|
+
|
|
385
|
+
componentDidUpdate() {
|
|
386
|
+
renderMath(this.root);
|
|
387
|
+
}
|
|
388
|
+
|
|
389
|
+
render() {
|
|
390
|
+
const {
|
|
391
|
+
classes,
|
|
392
|
+
labels,
|
|
393
|
+
children,
|
|
394
|
+
className
|
|
395
|
+
} = this.props;
|
|
396
|
+
const title = this.state.expanded ? labels.visible || 'Hide' : labels.hidden || 'Show';
|
|
397
|
+
return /*#__PURE__*/React__default.createElement("div", {
|
|
398
|
+
className: className,
|
|
399
|
+
ref: r => this.root = r
|
|
400
|
+
}, /*#__PURE__*/React__default.createElement("div", {
|
|
401
|
+
onClick: this.toggleExpanded
|
|
402
|
+
}, /*#__PURE__*/React__default.createElement("span", {
|
|
403
|
+
className: classes.title
|
|
404
|
+
}, title)), /*#__PURE__*/React__default.createElement(Collapse, {
|
|
405
|
+
in: this.state.expanded,
|
|
406
|
+
timeout: "auto",
|
|
407
|
+
unmountOnExit: true,
|
|
408
|
+
className: classes.collapsible
|
|
409
|
+
}, children));
|
|
410
|
+
}
|
|
411
|
+
|
|
412
|
+
}
|
|
413
|
+
Collapsible.propTypes = {
|
|
414
|
+
classes: PropTypes.object.isRequired,
|
|
415
|
+
className: PropTypes.string,
|
|
416
|
+
children: PropTypes.object,
|
|
417
|
+
labels: PropTypes.shape({
|
|
418
|
+
visible: PropTypes.string,
|
|
419
|
+
hidden: PropTypes.string
|
|
420
|
+
})
|
|
421
|
+
};
|
|
422
|
+
Collapsible.defaultProps = {
|
|
423
|
+
labels: {}
|
|
424
|
+
};
|
|
425
|
+
var index = withStyles$1(theme => ({
|
|
426
|
+
title: {
|
|
427
|
+
color: theme.palette.primary.light,
|
|
428
|
+
borderBottom: `1px dotted ${theme.palette.primary.light}`,
|
|
429
|
+
cursor: 'pointer'
|
|
430
|
+
},
|
|
431
|
+
collapsible: {
|
|
432
|
+
paddingTop: theme.spacing.unit * 2
|
|
433
|
+
}
|
|
434
|
+
}))(Collapsible);
|
|
435
|
+
|
|
436
|
+
function _extends() {
|
|
437
|
+
_extends = Object.assign || function (target) {
|
|
438
|
+
for (var i = 1; i < arguments.length; i++) {
|
|
439
|
+
var source = arguments[i];
|
|
440
|
+
|
|
441
|
+
for (var key in source) {
|
|
442
|
+
if (Object.prototype.hasOwnProperty.call(source, key)) {
|
|
443
|
+
target[key] = source[key];
|
|
444
|
+
}
|
|
445
|
+
}
|
|
446
|
+
}
|
|
447
|
+
|
|
448
|
+
return target;
|
|
449
|
+
};
|
|
450
|
+
|
|
451
|
+
return _extends.apply(this, arguments);
|
|
452
|
+
}
|
|
453
|
+
|
|
454
|
+
function _objectWithoutPropertiesLoose(source, excluded) {
|
|
455
|
+
if (source == null) return {};
|
|
456
|
+
var target = {};
|
|
457
|
+
var sourceKeys = Object.keys(source);
|
|
458
|
+
var key, i;
|
|
459
|
+
|
|
460
|
+
for (i = 0; i < sourceKeys.length; i++) {
|
|
461
|
+
key = sourceKeys[i];
|
|
462
|
+
if (excluded.indexOf(key) >= 0) continue;
|
|
463
|
+
target[key] = source[key];
|
|
464
|
+
}
|
|
465
|
+
|
|
466
|
+
return target;
|
|
467
|
+
}
|
|
468
|
+
|
|
469
|
+
const _excluded$1 = ["classes"];
|
|
470
|
+
|
|
471
|
+
const styles$3 = theme => ({
|
|
472
|
+
wrapper: {
|
|
473
|
+
display: 'flex',
|
|
474
|
+
flexDirection: 'column'
|
|
475
|
+
},
|
|
476
|
+
resetUndoContainer: {
|
|
477
|
+
display: 'flex',
|
|
478
|
+
alignItems: 'center',
|
|
479
|
+
justifyContent: 'center'
|
|
480
|
+
},
|
|
481
|
+
icon: {
|
|
482
|
+
width: '24px',
|
|
483
|
+
height: '24px',
|
|
484
|
+
color: 'gray',
|
|
485
|
+
marginRight: theme.spacing.unit
|
|
486
|
+
},
|
|
487
|
+
buttonContainer: {
|
|
488
|
+
display: 'flex',
|
|
489
|
+
alignItems: 'center',
|
|
490
|
+
marginLeft: theme.spacing.unit * 3,
|
|
491
|
+
marginRight: theme.spacing.unit * 3
|
|
492
|
+
}
|
|
493
|
+
});
|
|
494
|
+
/**
|
|
495
|
+
* HOC that adds undo and reset functionality for session values
|
|
496
|
+
*/
|
|
497
|
+
|
|
498
|
+
|
|
499
|
+
const withUndoReset = WrappedComponent => {
|
|
500
|
+
class WithUndoReset extends React.Component {
|
|
501
|
+
constructor(props) {
|
|
502
|
+
super(props);
|
|
503
|
+
|
|
504
|
+
this.onSessionChange = session => {
|
|
505
|
+
this.setState(state => ({
|
|
506
|
+
session,
|
|
507
|
+
changes: [...state.changes, session]
|
|
508
|
+
}), () => this.props.onSessionChange(session));
|
|
509
|
+
};
|
|
510
|
+
|
|
511
|
+
this.onUndo = () => {
|
|
512
|
+
this.setState(state => {
|
|
513
|
+
const newChanges = [...state.changes];
|
|
514
|
+
newChanges.pop();
|
|
515
|
+
return {
|
|
516
|
+
changes: newChanges,
|
|
517
|
+
session: newChanges.length ? newChanges[newChanges.length - 1] : state.sessionInitialValues
|
|
518
|
+
};
|
|
519
|
+
}, () => this.props.onSessionChange(this.state.session));
|
|
520
|
+
};
|
|
521
|
+
|
|
522
|
+
this.onReset = () => {
|
|
523
|
+
this.setState(state => ({
|
|
524
|
+
session: state.sessionInitialValues,
|
|
525
|
+
changes: []
|
|
526
|
+
}), () => this.props.onSessionChange(this.state.sessionInitialValues));
|
|
527
|
+
};
|
|
528
|
+
|
|
529
|
+
this.state = {
|
|
530
|
+
sessionInitialValues: JSON.parse(JSON.stringify(props.session)),
|
|
531
|
+
session: props.session,
|
|
532
|
+
changes: []
|
|
533
|
+
};
|
|
534
|
+
}
|
|
535
|
+
|
|
536
|
+
render() {
|
|
537
|
+
const _this$props = this.props,
|
|
538
|
+
{
|
|
539
|
+
classes
|
|
540
|
+
} = _this$props,
|
|
541
|
+
rest = _objectWithoutPropertiesLoose(_this$props, _excluded$1);
|
|
542
|
+
|
|
543
|
+
const {
|
|
544
|
+
changes,
|
|
545
|
+
session
|
|
546
|
+
} = this.state;
|
|
547
|
+
return /*#__PURE__*/React.createElement("div", {
|
|
548
|
+
className: classes.wrapper
|
|
549
|
+
}, /*#__PURE__*/React.createElement("div", {
|
|
550
|
+
className: classes.resetUndoContainer
|
|
551
|
+
}, /*#__PURE__*/React.createElement(Button, {
|
|
552
|
+
className: classes.buttonContainer,
|
|
553
|
+
color: "primary",
|
|
554
|
+
disabled: changes.length === 0,
|
|
555
|
+
onClick: this.onUndo
|
|
556
|
+
}, /*#__PURE__*/React.createElement(Undo, {
|
|
557
|
+
className: classes.icon
|
|
558
|
+
}), " Undo"), /*#__PURE__*/React.createElement(Button, {
|
|
559
|
+
className: classes.buttonContainer,
|
|
560
|
+
color: "primary",
|
|
561
|
+
disabled: changes.length === 0,
|
|
562
|
+
onClick: this.onReset
|
|
563
|
+
}, /*#__PURE__*/React.createElement(Restore, {
|
|
564
|
+
className: classes.icon
|
|
565
|
+
}), " Start Over")), /*#__PURE__*/React.createElement(WrappedComponent, _extends({}, rest, {
|
|
566
|
+
session: session,
|
|
567
|
+
onSessionChange: this.onSessionChange
|
|
568
|
+
})));
|
|
569
|
+
}
|
|
570
|
+
|
|
571
|
+
}
|
|
572
|
+
|
|
573
|
+
WithUndoReset.propTypes = {
|
|
574
|
+
classes: PropTypes.object,
|
|
575
|
+
session: PropTypes.object,
|
|
576
|
+
onSessionChange: PropTypes.func
|
|
577
|
+
};
|
|
578
|
+
return withStyles(styles$3)(WithUndoReset);
|
|
579
|
+
};
|
|
580
|
+
|
|
581
|
+
class ExtraCSSRulesMixin extends React__default.Component {
|
|
582
|
+
constructor(props) {
|
|
583
|
+
super(props);
|
|
584
|
+
|
|
585
|
+
this.appendExtraStyles = () => {
|
|
586
|
+
if (isEmpty(this.props.extraCSSRules) || !this.classesSheet) {
|
|
587
|
+
return;
|
|
588
|
+
}
|
|
589
|
+
|
|
590
|
+
const headElement = document.head || document.getElementsByTagName('head')[0];
|
|
591
|
+
|
|
592
|
+
if (!headElement) {
|
|
593
|
+
return;
|
|
594
|
+
}
|
|
595
|
+
|
|
596
|
+
if (!this.classesSheet.parentElement) {
|
|
597
|
+
headElement.appendChild(this.classesSheet);
|
|
598
|
+
}
|
|
599
|
+
|
|
600
|
+
const {
|
|
601
|
+
extraCSSRules,
|
|
602
|
+
classes
|
|
603
|
+
} = this.props;
|
|
604
|
+
this.classesSheet.innerHTML = `
|
|
605
|
+
.${classes.extraCSSRules} {
|
|
606
|
+
${extraCSSRules.rules}
|
|
607
|
+
}
|
|
608
|
+
`;
|
|
609
|
+
};
|
|
610
|
+
|
|
611
|
+
this.classesSheet = document.createElement('style');
|
|
612
|
+
}
|
|
613
|
+
|
|
614
|
+
componentDidMount() {
|
|
615
|
+
this.appendExtraStyles();
|
|
616
|
+
}
|
|
617
|
+
|
|
618
|
+
}
|
|
619
|
+
|
|
620
|
+
ExtraCSSRulesMixin.propTypes = {
|
|
621
|
+
extraCSSRules: PropTypes.shape({
|
|
622
|
+
names: PropTypes.arrayOf(PropTypes.string),
|
|
623
|
+
rules: PropTypes.string
|
|
624
|
+
})
|
|
625
|
+
};
|
|
626
|
+
ExtraCSSRulesMixin.defaultProps = {
|
|
627
|
+
extraCSSRules: {}
|
|
628
|
+
};
|
|
629
|
+
|
|
630
|
+
const _excluded = ["children", "className", "classes", "fontSizeFactor"],
|
|
631
|
+
_excluded2 = ["extraCSSRules"];
|
|
632
|
+
|
|
633
|
+
class UiLayout extends ExtraCSSRulesMixin {
|
|
634
|
+
constructor(props) {
|
|
635
|
+
super(props);
|
|
636
|
+
this.classesSheet = document.createElement('style');
|
|
637
|
+
}
|
|
638
|
+
|
|
639
|
+
computeStyle(fontSizeFactor) {
|
|
640
|
+
const getFontSize = element => parseFloat(getComputedStyle(element).fontSize);
|
|
641
|
+
|
|
642
|
+
const rootFontSize = getFontSize(document.documentElement);
|
|
643
|
+
const bodyFontSize = getFontSize(document.body);
|
|
644
|
+
const effectiveFontSize = Math.max(rootFontSize, bodyFontSize);
|
|
645
|
+
return fontSizeFactor !== 1 ? {
|
|
646
|
+
fontSize: `${effectiveFontSize * fontSizeFactor}px`
|
|
647
|
+
} : null;
|
|
648
|
+
}
|
|
649
|
+
|
|
650
|
+
render() {
|
|
651
|
+
const _this$props = this.props,
|
|
652
|
+
{
|
|
653
|
+
children,
|
|
654
|
+
className,
|
|
655
|
+
classes,
|
|
656
|
+
fontSizeFactor
|
|
657
|
+
} = _this$props,
|
|
658
|
+
rest = _objectWithoutPropertiesLoose(_this$props, _excluded);
|
|
659
|
+
|
|
660
|
+
const finalClass = classNames(className, classes.extraCSSRules, classes.uiLayoutContainer);
|
|
661
|
+
|
|
662
|
+
const restProps = _objectWithoutPropertiesLoose(rest, _excluded2);
|
|
663
|
+
|
|
664
|
+
const style = this.computeStyle(fontSizeFactor);
|
|
665
|
+
return /*#__PURE__*/React__default.createElement("div", _extends({
|
|
666
|
+
className: finalClass
|
|
667
|
+
}, restProps, style && {
|
|
668
|
+
style
|
|
669
|
+
}), children);
|
|
670
|
+
}
|
|
671
|
+
|
|
672
|
+
}
|
|
673
|
+
|
|
674
|
+
UiLayout.propTypes = {
|
|
675
|
+
classes: PropTypes.object,
|
|
676
|
+
className: PropTypes.string,
|
|
677
|
+
children: PropTypes.array,
|
|
678
|
+
extraCSSRules: PropTypes.shape({
|
|
679
|
+
names: PropTypes.arrayOf(PropTypes.string),
|
|
680
|
+
rules: PropTypes.string
|
|
681
|
+
}),
|
|
682
|
+
fontSizeFactor: PropTypes.number
|
|
683
|
+
};
|
|
684
|
+
UiLayout.defaultProps = {
|
|
685
|
+
extraCSSRules: {},
|
|
686
|
+
fontSizeFactor: 1
|
|
687
|
+
};
|
|
688
|
+
const styles$2 = {
|
|
689
|
+
extraCSSRules: {},
|
|
690
|
+
// need this because some browsers set their own style on table
|
|
691
|
+
uiLayoutContainer: {
|
|
692
|
+
'& table, th, td': {
|
|
693
|
+
fontSize: 'inherit'
|
|
694
|
+
/* Ensure table elements inherit font size */
|
|
695
|
+
|
|
696
|
+
}
|
|
697
|
+
}
|
|
698
|
+
};
|
|
699
|
+
const Styled$1 = withStyles(styles$2)(UiLayout);
|
|
700
|
+
|
|
701
|
+
class PreviewLayout extends React__default.Component {
|
|
702
|
+
render() {
|
|
703
|
+
const {
|
|
704
|
+
children,
|
|
705
|
+
classes,
|
|
706
|
+
ariaLabel,
|
|
707
|
+
role,
|
|
708
|
+
extraCSSRules,
|
|
709
|
+
fontSizeFactor
|
|
710
|
+
} = this.props;
|
|
711
|
+
const accessibility = ariaLabel ? {
|
|
712
|
+
'aria-label': ariaLabel,
|
|
713
|
+
role
|
|
714
|
+
} : {};
|
|
715
|
+
return /*#__PURE__*/React__default.createElement(Styled$1, _extends({
|
|
716
|
+
className: classes.container
|
|
717
|
+
}, accessibility, {
|
|
718
|
+
extraCSSRules: extraCSSRules,
|
|
719
|
+
fontSizeFactor: fontSizeFactor
|
|
720
|
+
}), children);
|
|
721
|
+
}
|
|
722
|
+
|
|
723
|
+
}
|
|
724
|
+
|
|
725
|
+
PreviewLayout.propTypes = {
|
|
726
|
+
ariaLabel: PropTypes.string,
|
|
727
|
+
children: PropTypes.oneOfType([PropTypes.arrayOf(PropTypes.node), PropTypes.node]).isRequired,
|
|
728
|
+
classes: PropTypes.object,
|
|
729
|
+
role: PropTypes.string,
|
|
730
|
+
extraCSSRules: PropTypes.shape({
|
|
731
|
+
names: PropTypes.arrayOf(PropTypes.string),
|
|
732
|
+
rules: PropTypes.string
|
|
733
|
+
}),
|
|
734
|
+
fontSizeFactor: PropTypes.number
|
|
735
|
+
};
|
|
736
|
+
|
|
737
|
+
const styles$1 = () => ({
|
|
738
|
+
container: {
|
|
739
|
+
display: 'flex',
|
|
740
|
+
flexDirection: 'column',
|
|
741
|
+
position: 'relative'
|
|
742
|
+
}
|
|
743
|
+
});
|
|
744
|
+
|
|
745
|
+
const theme = createMuiTheme({
|
|
746
|
+
typography: {
|
|
747
|
+
useNextVariants: true
|
|
748
|
+
},
|
|
749
|
+
palette: {
|
|
750
|
+
action: {
|
|
751
|
+
disabled: 'rgba(0, 0, 0, 0.54);'
|
|
752
|
+
}
|
|
753
|
+
},
|
|
754
|
+
overrides: {
|
|
755
|
+
MuiRadio: {
|
|
756
|
+
root: {
|
|
757
|
+
'&$checked': {
|
|
758
|
+
color: '#3f51b5 !important'
|
|
759
|
+
}
|
|
760
|
+
}
|
|
761
|
+
},
|
|
762
|
+
MuiCheckbox: {
|
|
763
|
+
root: {
|
|
764
|
+
'&$checked': {
|
|
765
|
+
color: '#3f51b5 !important'
|
|
766
|
+
}
|
|
767
|
+
}
|
|
768
|
+
},
|
|
769
|
+
MuiTabs: {
|
|
770
|
+
root: {
|
|
771
|
+
borderBottom: '1px solid #eee'
|
|
772
|
+
}
|
|
773
|
+
},
|
|
774
|
+
MuiSwitch: {
|
|
775
|
+
root: {
|
|
776
|
+
'&$checked': {
|
|
777
|
+
color: '#3f51b5 !important',
|
|
778
|
+
'& + $bar': {
|
|
779
|
+
backgroundColor: '#3f51b5 !important',
|
|
780
|
+
opacity: 0.5
|
|
781
|
+
}
|
|
782
|
+
}
|
|
783
|
+
}
|
|
784
|
+
}
|
|
785
|
+
}
|
|
786
|
+
});
|
|
787
|
+
const Styled = withStyles(styles$1)(PreviewLayout);
|
|
788
|
+
|
|
789
|
+
const RootElem = props => /*#__PURE__*/React__default.createElement(MuiThemeProvider, {
|
|
790
|
+
theme: theme
|
|
791
|
+
}, /*#__PURE__*/React__default.createElement(Styled, props));
|
|
792
|
+
|
|
793
|
+
class HtmlAndMath extends React__default.Component {
|
|
794
|
+
render() {
|
|
795
|
+
const {
|
|
796
|
+
tag,
|
|
797
|
+
className,
|
|
798
|
+
html
|
|
799
|
+
} = this.props;
|
|
800
|
+
const Tag = tag || 'div';
|
|
801
|
+
return /*#__PURE__*/React__default.createElement(Tag, {
|
|
802
|
+
ref: r => this.node = r,
|
|
803
|
+
className: className,
|
|
804
|
+
dangerouslySetInnerHTML: {
|
|
805
|
+
__html: html
|
|
806
|
+
}
|
|
807
|
+
});
|
|
808
|
+
}
|
|
809
|
+
|
|
810
|
+
}
|
|
811
|
+
HtmlAndMath.propTypes = {
|
|
812
|
+
tag: PropTypes.string,
|
|
813
|
+
className: PropTypes.string,
|
|
814
|
+
html: PropTypes.string
|
|
815
|
+
};
|
|
816
|
+
HtmlAndMath.defaultProps = {
|
|
817
|
+
tag: 'div',
|
|
818
|
+
html: ''
|
|
819
|
+
};
|
|
820
|
+
|
|
821
|
+
const RawInputContainer = props => {
|
|
822
|
+
const {
|
|
823
|
+
label,
|
|
824
|
+
className,
|
|
825
|
+
children,
|
|
826
|
+
classes
|
|
827
|
+
} = props;
|
|
828
|
+
const names = classNames(classes.formControl, className);
|
|
829
|
+
return /*#__PURE__*/React__default.createElement(FormControl, {
|
|
830
|
+
className: names
|
|
831
|
+
}, /*#__PURE__*/React__default.createElement(InputLabel, {
|
|
832
|
+
className: classes.label,
|
|
833
|
+
shrink: true
|
|
834
|
+
}, label), children);
|
|
835
|
+
};
|
|
836
|
+
|
|
837
|
+
RawInputContainer.propTypes = {
|
|
838
|
+
label: PropTypes.oneOfType([PropTypes.string, PropTypes.object]).isRequired,
|
|
839
|
+
className: PropTypes.string,
|
|
840
|
+
children: PropTypes.oneOfType([PropTypes.arrayOf(PropTypes.node), PropTypes.node]).isRequired,
|
|
841
|
+
classes: PropTypes.object.isRequired
|
|
842
|
+
};
|
|
843
|
+
var inputContainer = withStyles(theme => ({
|
|
844
|
+
formControl: {
|
|
845
|
+
marginLeft: 0,
|
|
846
|
+
marginRight: 0,
|
|
847
|
+
paddingBottom: theme.spacing.unit,
|
|
848
|
+
flex: '1 0 auto',
|
|
849
|
+
minWidth: theme.spacing.unit * 4
|
|
850
|
+
},
|
|
851
|
+
label: {
|
|
852
|
+
fontSize: 'inherit',
|
|
853
|
+
whiteSpace: 'nowrap'
|
|
854
|
+
}
|
|
855
|
+
}))(RawInputContainer);
|
|
856
|
+
|
|
857
|
+
const NEWLINE_BLOCK_REGEX = /\\embed\{newLine\}\[\]/g;
|
|
858
|
+
const NEWLINE_LATEX = '\\newline ';
|
|
859
|
+
class PreviewPrompt extends Component {
|
|
860
|
+
constructor(...args) {
|
|
861
|
+
super(...args);
|
|
862
|
+
|
|
863
|
+
this.parsedText = text => {
|
|
864
|
+
const {
|
|
865
|
+
customAudioButton
|
|
866
|
+
} = this.props;
|
|
867
|
+
const div = document.createElement('div');
|
|
868
|
+
div.innerHTML = text;
|
|
869
|
+
const audio = div.querySelector('audio');
|
|
870
|
+
|
|
871
|
+
if (audio) {
|
|
872
|
+
const source = document.createElement('source');
|
|
873
|
+
source.setAttribute('type', 'audio/mp3');
|
|
874
|
+
source.setAttribute('src', audio.getAttribute('src'));
|
|
875
|
+
audio.removeAttribute('src');
|
|
876
|
+
audio.setAttribute('id', 'pie-prompt-audio-player');
|
|
877
|
+
audio.appendChild(source);
|
|
878
|
+
|
|
879
|
+
if (customAudioButton) {
|
|
880
|
+
audio.style.display = 'none';
|
|
881
|
+
const playButton = document.createElement('div');
|
|
882
|
+
playButton.id = 'play-audio-button';
|
|
883
|
+
Object.assign(playButton.style, {
|
|
884
|
+
cursor: 'pointer',
|
|
885
|
+
display: 'block',
|
|
886
|
+
width: '128px',
|
|
887
|
+
height: '128px',
|
|
888
|
+
backgroundImage: `url(${customAudioButton.pauseImage})`,
|
|
889
|
+
backgroundSize: 'cover',
|
|
890
|
+
borderRadius: '50%',
|
|
891
|
+
border: '1px solid #326295'
|
|
892
|
+
});
|
|
893
|
+
audio.parentNode.insertBefore(playButton, audio);
|
|
894
|
+
}
|
|
895
|
+
}
|
|
896
|
+
|
|
897
|
+
return div.innerHTML;
|
|
898
|
+
};
|
|
899
|
+
}
|
|
900
|
+
|
|
901
|
+
addCustomAudioButtonControls() {
|
|
902
|
+
const {
|
|
903
|
+
autoplayAudioEnabled,
|
|
904
|
+
customAudioButton
|
|
905
|
+
} = this.props;
|
|
906
|
+
const playButton = document.getElementById('play-audio-button');
|
|
907
|
+
const audio = document.getElementById('pie-prompt-audio-player');
|
|
908
|
+
|
|
909
|
+
if (autoplayAudioEnabled && audio) {
|
|
910
|
+
audio.play().then(() => {
|
|
911
|
+
if (playButton && customAudioButton) {
|
|
912
|
+
audio.addEventListener('ended', handleAudioEnded);
|
|
913
|
+
}
|
|
914
|
+
}).catch(error => {
|
|
915
|
+
console.error('Error playing audio', error);
|
|
916
|
+
});
|
|
917
|
+
}
|
|
918
|
+
|
|
919
|
+
if (!playButton || !audio || !customAudioButton) return;
|
|
920
|
+
|
|
921
|
+
const handlePlayClick = () => {
|
|
922
|
+
// if already playing, don't play again
|
|
923
|
+
if (!audio.paused) return;
|
|
924
|
+
if (playButton.style.backgroundImage.includes(customAudioButton.pauseImage)) return;
|
|
925
|
+
audio.play();
|
|
926
|
+
};
|
|
927
|
+
|
|
928
|
+
const handleAudioEnded = () => {
|
|
929
|
+
playButton.style.backgroundImage = `url(${customAudioButton.playImage})`;
|
|
930
|
+
};
|
|
931
|
+
|
|
932
|
+
const handleAudioPlay = () => {
|
|
933
|
+
Object.assign(playButton.style, {
|
|
934
|
+
backgroundImage: `url(${customAudioButton.pauseImage})`,
|
|
935
|
+
border: '1px solid #ccc'
|
|
936
|
+
});
|
|
937
|
+
};
|
|
938
|
+
|
|
939
|
+
const handleAudioPause = () => {
|
|
940
|
+
Object.assign(playButton.style, {
|
|
941
|
+
backgroundImage: `url(${customAudioButton.playImage})`,
|
|
942
|
+
border: '1px solid #326295'
|
|
943
|
+
});
|
|
944
|
+
};
|
|
945
|
+
|
|
946
|
+
playButton.addEventListener('click', handlePlayClick);
|
|
947
|
+
audio.addEventListener('play', handleAudioPlay);
|
|
948
|
+
audio.addEventListener('pause', handleAudioPause);
|
|
949
|
+
audio.addEventListener('ended', handleAudioEnded); // store event handler references so they can be removed later
|
|
950
|
+
|
|
951
|
+
this._handlePlayClick = handlePlayClick;
|
|
952
|
+
this._handleAudioPlay = handleAudioPlay;
|
|
953
|
+
this._handleAudioPause = handleAudioPause;
|
|
954
|
+
this._handleAudioEnded = handleAudioEnded;
|
|
955
|
+
}
|
|
956
|
+
|
|
957
|
+
removeCustomAudioButtonListeners() {
|
|
958
|
+
const playButton = document.getElementById('play-audio-button');
|
|
959
|
+
const audio = document.querySelector('audio');
|
|
960
|
+
if (!playButton || !audio) return; // remove event listeners using stored references
|
|
961
|
+
|
|
962
|
+
playButton.removeEventListener('click', this._handlePlayClick);
|
|
963
|
+
audio.removeEventListener('play', this._handleAudioPlay);
|
|
964
|
+
audio.removeEventListener('pause', this._handleAudioPause);
|
|
965
|
+
audio.removeEventListener('ended', this._handleAudioEnded);
|
|
966
|
+
}
|
|
967
|
+
|
|
968
|
+
componentDidMount() {
|
|
969
|
+
this.alignImages();
|
|
970
|
+
this.addCustomAudioButtonControls();
|
|
971
|
+
}
|
|
972
|
+
|
|
973
|
+
componentDidUpdate() {
|
|
974
|
+
this.alignImages();
|
|
975
|
+
}
|
|
976
|
+
|
|
977
|
+
componentWillUnmount() {
|
|
978
|
+
this.removeCustomAudioButtonListeners();
|
|
979
|
+
}
|
|
980
|
+
|
|
981
|
+
alignImages() {
|
|
982
|
+
const previewPrompts = document.querySelectorAll('#preview-prompt');
|
|
983
|
+
previewPrompts.forEach(previewPrompt => {
|
|
984
|
+
const images = previewPrompt.getElementsByTagName('img');
|
|
985
|
+
|
|
986
|
+
if (images && images.length) {
|
|
987
|
+
for (let image of images) {
|
|
988
|
+
// check if alignment property was set
|
|
989
|
+
if (image.attributes && image.attributes.alignment && image.attributes.alignment.value) {
|
|
990
|
+
const parentNode = image.parentElement; // check if div is not already added to dom and replace current image with wrapped image
|
|
991
|
+
|
|
992
|
+
if (!(parentNode.tagName === 'DIV' && parentNode.style.display === 'flex' && parentNode.style.width === '100%')) {
|
|
993
|
+
const div = document.createElement('div');
|
|
994
|
+
div.style.display = 'flex';
|
|
995
|
+
div.style.width = '100%';
|
|
996
|
+
const copyImage = image.cloneNode(true);
|
|
997
|
+
div.appendChild(copyImage);
|
|
998
|
+
parentNode.replaceChild(div, image);
|
|
999
|
+
}
|
|
1000
|
+
}
|
|
1001
|
+
}
|
|
1002
|
+
}
|
|
1003
|
+
});
|
|
1004
|
+
}
|
|
1005
|
+
|
|
1006
|
+
render() {
|
|
1007
|
+
const {
|
|
1008
|
+
prompt,
|
|
1009
|
+
classes,
|
|
1010
|
+
tagName,
|
|
1011
|
+
className,
|
|
1012
|
+
onClick,
|
|
1013
|
+
defaultClassName
|
|
1014
|
+
} = this.props;
|
|
1015
|
+
const CustomTag = tagName || 'div'; // legend tag was added once with accessibility tasks, wee need extra style to make it work with images alignment
|
|
1016
|
+
|
|
1017
|
+
const legendClass = tagName === 'legend' ? 'legend' : '';
|
|
1018
|
+
const customClasses = `${classes.promptTable} ${classes[className] || ''} ${defaultClassName || ''} ${classes[legendClass] || ''}`;
|
|
1019
|
+
return /*#__PURE__*/React__default.createElement(CustomTag, {
|
|
1020
|
+
id: 'preview-prompt',
|
|
1021
|
+
onClick: onClick,
|
|
1022
|
+
className: customClasses,
|
|
1023
|
+
dangerouslySetInnerHTML: {
|
|
1024
|
+
__html: this.parsedText(prompt || '').replace(NEWLINE_BLOCK_REGEX, NEWLINE_LATEX)
|
|
1025
|
+
}
|
|
1026
|
+
});
|
|
1027
|
+
}
|
|
1028
|
+
|
|
1029
|
+
}
|
|
1030
|
+
PreviewPrompt.propTypes = {
|
|
1031
|
+
classes: PropTypes.object,
|
|
1032
|
+
prompt: PropTypes.string,
|
|
1033
|
+
tagName: PropTypes.string,
|
|
1034
|
+
className: PropTypes.string,
|
|
1035
|
+
onClick: PropTypes.func,
|
|
1036
|
+
defaultClassName: PropTypes.string,
|
|
1037
|
+
autoplayAudioEnabled: PropTypes.bool,
|
|
1038
|
+
customAudioButton: {
|
|
1039
|
+
playImage: PropTypes.string,
|
|
1040
|
+
pauseImage: PropTypes.string
|
|
1041
|
+
}
|
|
1042
|
+
};
|
|
1043
|
+
PreviewPrompt.defaultProps = {
|
|
1044
|
+
onClick: () => {}
|
|
1045
|
+
};
|
|
1046
|
+
|
|
1047
|
+
const styles = theme => ({
|
|
1048
|
+
prompt: {
|
|
1049
|
+
verticalAlign: 'middle',
|
|
1050
|
+
color: text()
|
|
1051
|
+
},
|
|
1052
|
+
legend: {
|
|
1053
|
+
width: '100%',
|
|
1054
|
+
fontSize: 'inherit !important'
|
|
1055
|
+
},
|
|
1056
|
+
rationale: {
|
|
1057
|
+
paddingLeft: theme.spacing.unit * 4,
|
|
1058
|
+
paddingBottom: theme.spacing.unit
|
|
1059
|
+
},
|
|
1060
|
+
label: {
|
|
1061
|
+
color: `${text()} !important`,
|
|
1062
|
+
//'var(--choice-input-color, black)',
|
|
1063
|
+
display: 'flex',
|
|
1064
|
+
flexDirection: 'column',
|
|
1065
|
+
verticalAlign: 'middle',
|
|
1066
|
+
cursor: 'pointer',
|
|
1067
|
+
'& > p': {
|
|
1068
|
+
margin: '0 0 0 0 !important'
|
|
1069
|
+
}
|
|
1070
|
+
},
|
|
1071
|
+
promptTable: {
|
|
1072
|
+
'&:not(.MathJax) > table': {
|
|
1073
|
+
borderCollapse: 'collapse'
|
|
1074
|
+
},
|
|
1075
|
+
'&:not(.MathJax) > table tr': {
|
|
1076
|
+
'&:nth-child(2n)': {
|
|
1077
|
+
backgroundColor: '#f6f8fa',
|
|
1078
|
+
color: theme.palette.common.black
|
|
1079
|
+
}
|
|
1080
|
+
},
|
|
1081
|
+
// align table content to left as per STAR requirement PD-3687
|
|
1082
|
+
'&:not(.MathJax) table td, &:not(.MathJax) table th': {
|
|
1083
|
+
padding: '.6em 1em',
|
|
1084
|
+
textAlign: 'left'
|
|
1085
|
+
},
|
|
1086
|
+
// added this to fix alignment of text in prompt imported from studio (PD-3423)
|
|
1087
|
+
'&:not(.MathJax) > table td > p.kds-indent': {
|
|
1088
|
+
textAlign: 'initial'
|
|
1089
|
+
}
|
|
1090
|
+
}
|
|
1091
|
+
});
|
|
1092
|
+
|
|
1093
|
+
var previewPrompt = withStyles(styles)(PreviewPrompt);
|
|
1094
|
+
|
|
1095
|
+
const Readable = props => {
|
|
1096
|
+
return /*#__PURE__*/React__default.createElement(React__default.Fragment, null, React__default.Children.map(props.children, child => /*#__PURE__*/React__default.cloneElement(child, {
|
|
1097
|
+
'data-pie-readable': props.false === undefined
|
|
1098
|
+
})));
|
|
1099
|
+
};
|
|
1100
|
+
|
|
1101
|
+
Readable.propTypes = {
|
|
1102
|
+
children: PropTypes.node,
|
|
1103
|
+
false: PropTypes.bool
|
|
1104
|
+
};
|
|
1105
|
+
|
|
1106
|
+
const Purpose = props => {
|
|
1107
|
+
return /*#__PURE__*/React__default.createElement(React__default.Fragment, null, React__default.Children.map(props.children, child => /*#__PURE__*/React__default.cloneElement(child, {
|
|
1108
|
+
'data-pie-purpose': props.purpose
|
|
1109
|
+
})));
|
|
1110
|
+
};
|
|
1111
|
+
|
|
1112
|
+
Purpose.propTypes = {
|
|
1113
|
+
children: PropTypes.node,
|
|
1114
|
+
purpose: PropTypes.string
|
|
1115
|
+
};
|
|
1116
|
+
|
|
1117
|
+
let parser$1;
|
|
1118
|
+
|
|
1119
|
+
if (typeof window !== 'undefined') {
|
|
1120
|
+
parser$1 = new DOMParser();
|
|
1121
|
+
}
|
|
1122
|
+
|
|
1123
|
+
const markupToText = s => {
|
|
1124
|
+
const root = parser$1.parseFromString(s, 'text/html');
|
|
1125
|
+
return root.body.textContent;
|
|
1126
|
+
};
|
|
1127
|
+
|
|
1128
|
+
const hasText = s => {
|
|
1129
|
+
if (!s) {
|
|
1130
|
+
return false;
|
|
1131
|
+
}
|
|
1132
|
+
|
|
1133
|
+
const tc = markupToText(s);
|
|
1134
|
+
return !!(tc && tc.trim());
|
|
1135
|
+
};
|
|
1136
|
+
|
|
1137
|
+
let parser;
|
|
1138
|
+
|
|
1139
|
+
if (typeof window !== 'undefined') {
|
|
1140
|
+
parser = new DOMParser();
|
|
1141
|
+
}
|
|
1142
|
+
/*
|
|
1143
|
+
* Check if the string contains at least one media element.
|
|
1144
|
+
*/
|
|
1145
|
+
|
|
1146
|
+
|
|
1147
|
+
const hasMedia = s => {
|
|
1148
|
+
if (!s) {
|
|
1149
|
+
return false;
|
|
1150
|
+
}
|
|
1151
|
+
|
|
1152
|
+
const root = parser.parseFromString(s, 'text/html');
|
|
1153
|
+
return !!root.body.querySelector('img') || !!root.body.querySelector('video') || !!root.body.querySelector('audio');
|
|
1154
|
+
};
|
|
1155
|
+
|
|
1156
|
+
var enableAudioAutoplayImage = '';
|
|
1157
|
+
|
|
1158
|
+
export { ExtraCSSRulesMixin as AppendCSSRules, index as Collapsible, enableAudioAutoplayImage as EnableAudioAutoplayImage, Feedback$1 as Feedback, HtmlAndMath, inputContainer as InputContainer, RootElem as PreviewLayout, previewPrompt as PreviewPrompt, Purpose, Readable, Styled$1 as UiLayout, color, hasMedia, hasText, responseIndicators as indicators, withUndoReset };
|
|
1159
|
+
//# sourceMappingURL=index.js.map
|