@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 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