@pie-element/hotspot 11.1.1 → 11.1.2-next.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.
Files changed (234) hide show
  1. package/configure.js +2 -0
  2. package/controller.js +1 -0
  3. package/dist/author/DeleteWidget.d.ts +38 -0
  4. package/dist/author/DeleteWidget.js +46 -0
  5. package/dist/author/button.d.ts +31 -0
  6. package/dist/author/button.js +27 -0
  7. package/dist/author/buttons/circle.d.ts +18 -0
  8. package/dist/author/buttons/circle.js +25 -0
  9. package/dist/author/buttons/polygon.d.ts +18 -0
  10. package/dist/author/buttons/polygon.js +36 -0
  11. package/dist/author/buttons/rectangle.d.ts +18 -0
  12. package/dist/author/buttons/rectangle.js +36 -0
  13. package/dist/author/defaults.d.ts +157 -0
  14. package/dist/author/defaults.js +119 -0
  15. package/dist/author/hotspot-circle.d.ts +21 -0
  16. package/dist/author/hotspot-circle.js +124 -0
  17. package/dist/author/hotspot-container.d.ts +29 -0
  18. package/dist/author/hotspot-container.js +210 -0
  19. package/dist/author/hotspot-drawable.d.ts +31 -0
  20. package/dist/author/hotspot-drawable.js +312 -0
  21. package/dist/author/hotspot-palette.d.ts +14 -0
  22. package/dist/author/hotspot-palette.js +72 -0
  23. package/dist/author/hotspot-polygon.d.ts +38 -0
  24. package/dist/author/hotspot-polygon.js +200 -0
  25. package/dist/author/hotspot-rectangle.d.ts +20 -0
  26. package/dist/author/hotspot-rectangle.js +119 -0
  27. package/dist/author/icons.d.ts +9 -0
  28. package/dist/author/icons.js +4 -0
  29. package/dist/author/image-konva.d.ts +19 -0
  30. package/dist/author/image-konva.js +49 -0
  31. package/dist/author/index.d.ts +52 -0
  32. package/dist/author/index.js +143 -0
  33. package/dist/author/root.d.ts +15 -0
  34. package/dist/author/root.js +215 -0
  35. package/dist/author/shapes/circle.d.ts +18 -0
  36. package/dist/author/shapes/circle.js +47 -0
  37. package/dist/author/shapes/index.d.ts +12 -0
  38. package/dist/author/shapes/polygon.d.ts +19 -0
  39. package/dist/author/shapes/polygon.js +51 -0
  40. package/dist/author/shapes/rectagle.d.ts +18 -0
  41. package/dist/author/shapes/rectagle.js +57 -0
  42. package/dist/author/shapes/utils.d.ts +19 -0
  43. package/dist/author/shapes/utils.js +16 -0
  44. package/dist/author/upload-control.d.ts +29 -0
  45. package/dist/author/upload-control.js +28 -0
  46. package/dist/author/utils.d.ts +24 -0
  47. package/dist/author/utils.js +83 -0
  48. package/dist/browser/ReactKonva-Z9-1H-UZ.js +19329 -0
  49. package/dist/browser/ReactKonva-Z9-1H-UZ.js.map +1 -0
  50. package/dist/browser/author/index.js +41646 -0
  51. package/dist/browser/author/index.js.map +1 -0
  52. package/dist/browser/browser-CfnAFove.js +219 -0
  53. package/dist/browser/browser-CfnAFove.js.map +1 -0
  54. package/dist/browser/controller/index.js +198 -0
  55. package/dist/browser/controller/index.js.map +1 -0
  56. package/dist/browser/delivery/index.js +2460 -0
  57. package/dist/browser/delivery/index.js.map +1 -0
  58. package/dist/browser/dist-C78LDz6R.js +96 -0
  59. package/dist/browser/dist-C78LDz6R.js.map +1 -0
  60. package/dist/browser/hotspot.css +2 -0
  61. package/dist/controller/defaults.d.ts +35 -0
  62. package/dist/controller/defaults.js +29 -0
  63. package/dist/controller/index.d.ts +22 -0
  64. package/dist/controller/index.js +154 -0
  65. package/dist/controller/utils.d.ts +10 -0
  66. package/dist/controller/utils.js +12 -0
  67. package/dist/delivery/hotspot/circle.d.ts +19 -0
  68. package/dist/delivery/hotspot/circle.js +100 -0
  69. package/dist/delivery/hotspot/container.d.ts +16 -0
  70. package/dist/delivery/hotspot/container.js +150 -0
  71. package/dist/delivery/hotspot/icons.d.ts +10 -0
  72. package/dist/delivery/hotspot/icons.js +4 -0
  73. package/dist/delivery/hotspot/image-konva-tooltip.d.ts +19 -0
  74. package/dist/delivery/hotspot/image-konva-tooltip.js +66 -0
  75. package/dist/delivery/hotspot/index.d.ts +17 -0
  76. package/dist/delivery/hotspot/index.js +114 -0
  77. package/dist/delivery/hotspot/polygon.d.ts +21 -0
  78. package/dist/delivery/hotspot/polygon.js +108 -0
  79. package/dist/delivery/hotspot/rectangle.d.ts +19 -0
  80. package/dist/delivery/hotspot/rectangle.js +104 -0
  81. package/dist/delivery/index.d.ts +20 -0
  82. package/dist/delivery/index.js +107 -0
  83. package/dist/delivery/session-updater.d.ts +10 -0
  84. package/dist/delivery/session-updater.js +14 -0
  85. package/dist/index.d.ts +1 -0
  86. package/dist/index.iife.d.ts +8 -0
  87. package/dist/index.iife.js +169 -0
  88. package/dist/index.js +2 -0
  89. package/dist/runtime-support.d.ts +12 -0
  90. package/dist/runtime-support.js +12 -0
  91. package/package.json +84 -18
  92. package/CHANGELOG.json +0 -997
  93. package/CHANGELOG.md +0 -2214
  94. package/LICENSE.md +0 -5
  95. package/README.md +0 -1
  96. package/configure/CHANGELOG.json +0 -682
  97. package/configure/CHANGELOG.md +0 -1951
  98. package/configure/lib/DeleteWidget.js +0 -64
  99. package/configure/lib/DeleteWidget.js.map +0 -1
  100. package/configure/lib/button.js +0 -42
  101. package/configure/lib/button.js.map +0 -1
  102. package/configure/lib/buttons/circle.js +0 -33
  103. package/configure/lib/buttons/circle.js.map +0 -1
  104. package/configure/lib/buttons/polygon.js +0 -39
  105. package/configure/lib/buttons/polygon.js.map +0 -1
  106. package/configure/lib/buttons/rectangle.js +0 -39
  107. package/configure/lib/buttons/rectangle.js.map +0 -1
  108. package/configure/lib/defaults.js +0 -155
  109. package/configure/lib/defaults.js.map +0 -1
  110. package/configure/lib/hotspot-circle.js +0 -192
  111. package/configure/lib/hotspot-circle.js.map +0 -1
  112. package/configure/lib/hotspot-container.js +0 -320
  113. package/configure/lib/hotspot-container.js.map +0 -1
  114. package/configure/lib/hotspot-drawable.js +0 -519
  115. package/configure/lib/hotspot-drawable.js.map +0 -1
  116. package/configure/lib/hotspot-palette.js +0 -107
  117. package/configure/lib/hotspot-palette.js.map +0 -1
  118. package/configure/lib/hotspot-polygon.js +0 -293
  119. package/configure/lib/hotspot-polygon.js.map +0 -1
  120. package/configure/lib/hotspot-rectangle.js +0 -190
  121. package/configure/lib/hotspot-rectangle.js.map +0 -1
  122. package/configure/lib/icons.js +0 -7
  123. package/configure/lib/icons.js.map +0 -1
  124. package/configure/lib/image-konva.js +0 -66
  125. package/configure/lib/image-konva.js.map +0 -1
  126. package/configure/lib/index.js +0 -194
  127. package/configure/lib/index.js.map +0 -1
  128. package/configure/lib/root.js +0 -330
  129. package/configure/lib/root.js.map +0 -1
  130. package/configure/lib/shapes/circle.js +0 -84
  131. package/configure/lib/shapes/circle.js.map +0 -1
  132. package/configure/lib/shapes/index.js +0 -50
  133. package/configure/lib/shapes/index.js.map +0 -1
  134. package/configure/lib/shapes/polygon.js +0 -82
  135. package/configure/lib/shapes/polygon.js.map +0 -1
  136. package/configure/lib/shapes/rectagle.js +0 -84
  137. package/configure/lib/shapes/rectagle.js.map +0 -1
  138. package/configure/lib/shapes/utils.js +0 -21
  139. package/configure/lib/shapes/utils.js.map +0 -1
  140. package/configure/lib/upload-control.js +0 -41
  141. package/configure/lib/upload-control.js.map +0 -1
  142. package/configure/lib/utils.js +0 -185
  143. package/configure/lib/utils.js.map +0 -1
  144. package/configure/package.json +0 -26
  145. package/configure/src/DeleteWidget.jsx +0 -51
  146. package/configure/src/__tests__/DeleteWidget.test.jsx +0 -366
  147. package/configure/src/__tests__/button.test.jsx +0 -198
  148. package/configure/src/__tests__/hotspot-circle.test.jsx +0 -259
  149. package/configure/src/__tests__/hotspot-container.test.js +0 -366
  150. package/configure/src/__tests__/hotspot-drawable.test.js +0 -271
  151. package/configure/src/__tests__/hotspot-palette.test.jsx +0 -71
  152. package/configure/src/__tests__/image-konva.test.jsx +0 -226
  153. package/configure/src/__tests__/index.test.js +0 -329
  154. package/configure/src/__tests__/root.test.js +0 -400
  155. package/configure/src/__tests__/utils.test.js +0 -241
  156. package/configure/src/button.jsx +0 -35
  157. package/configure/src/buttons/circle.jsx +0 -18
  158. package/configure/src/buttons/polygon.jsx +0 -29
  159. package/configure/src/buttons/rectangle.jsx +0 -29
  160. package/configure/src/defaults.js +0 -109
  161. package/configure/src/hotspot-circle.jsx +0 -183
  162. package/configure/src/hotspot-container.jsx +0 -330
  163. package/configure/src/hotspot-drawable.jsx +0 -527
  164. package/configure/src/hotspot-palette.jsx +0 -90
  165. package/configure/src/hotspot-polygon.jsx +0 -294
  166. package/configure/src/hotspot-rectangle.jsx +0 -169
  167. package/configure/src/icons.js +0 -5
  168. package/configure/src/image-konva.jsx +0 -63
  169. package/configure/src/index.js +0 -208
  170. package/configure/src/root.jsx +0 -346
  171. package/configure/src/shapes/circle.js +0 -81
  172. package/configure/src/shapes/index.js +0 -4
  173. package/configure/src/shapes/polygon.js +0 -81
  174. package/configure/src/shapes/rectagle.js +0 -82
  175. package/configure/src/shapes/utils.js +0 -16
  176. package/configure/src/upload-control.jsx +0 -33
  177. package/configure/src/utils.js +0 -210
  178. package/controller/CHANGELOG.json +0 -362
  179. package/controller/CHANGELOG.md +0 -1304
  180. package/controller/lib/defaults.js +0 -33
  181. package/controller/lib/defaults.js.map +0 -1
  182. package/controller/lib/index.js +0 -341
  183. package/controller/lib/index.js.map +0 -1
  184. package/controller/lib/utils.js +0 -32
  185. package/controller/lib/utils.js.map +0 -1
  186. package/controller/package.json +0 -18
  187. package/controller/src/__tests__/index.test.js +0 -419
  188. package/controller/src/__tests__/utils.test.js +0 -5
  189. package/controller/src/defaults.js +0 -19
  190. package/controller/src/index.js +0 -328
  191. package/controller/src/utils.js +0 -29
  192. package/docs/config-schema.json +0 -2023
  193. package/docs/config-schema.json.md +0 -1495
  194. package/docs/demo/config.js +0 -8
  195. package/docs/demo/generate.js +0 -118
  196. package/docs/demo/index.html +0 -1
  197. package/docs/demo/session.js +0 -11
  198. package/docs/pie-schema.json +0 -1204
  199. package/docs/pie-schema.json.md +0 -851
  200. package/lib/hotspot/circle.js +0 -156
  201. package/lib/hotspot/circle.js.map +0 -1
  202. package/lib/hotspot/container.js +0 -206
  203. package/lib/hotspot/container.js.map +0 -1
  204. package/lib/hotspot/icons.js +0 -8
  205. package/lib/hotspot/icons.js.map +0 -1
  206. package/lib/hotspot/image-konva-tooltip.js +0 -86
  207. package/lib/hotspot/image-konva-tooltip.js.map +0 -1
  208. package/lib/hotspot/index.js +0 -163
  209. package/lib/hotspot/index.js.map +0 -1
  210. package/lib/hotspot/polygon.js +0 -203
  211. package/lib/hotspot/polygon.js.map +0 -1
  212. package/lib/hotspot/rectangle.js +0 -175
  213. package/lib/hotspot/rectangle.js.map +0 -1
  214. package/lib/index.js +0 -213
  215. package/lib/index.js.map +0 -1
  216. package/lib/session-updater.js +0 -42
  217. package/lib/session-updater.js.map +0 -1
  218. package/src/__tests__/container.test.jsx +0 -58
  219. package/src/__tests__/index.test.js +0 -123
  220. package/src/__tests__/session-updater.test.jsx +0 -69
  221. package/src/hotspot/__tests__/circle.test.jsx +0 -464
  222. package/src/hotspot/__tests__/container.test.jsx +0 -546
  223. package/src/hotspot/__tests__/image-konva-tooltip.test.jsx +0 -510
  224. package/src/hotspot/__tests__/polygon.test.jsx +0 -502
  225. package/src/hotspot/__tests__/rectangle.test.jsx +0 -418
  226. package/src/hotspot/circle.jsx +0 -152
  227. package/src/hotspot/container.jsx +0 -217
  228. package/src/hotspot/icons.js +0 -7
  229. package/src/hotspot/image-konva-tooltip.jsx +0 -76
  230. package/src/hotspot/index.jsx +0 -165
  231. package/src/hotspot/polygon.jsx +0 -195
  232. package/src/hotspot/rectangle.jsx +0 -171
  233. package/src/index.js +0 -226
  234. package/src/session-updater.js +0 -29
@@ -1,195 +0,0 @@
1
- import React from 'react';
2
- import PropTypes from 'prop-types';
3
- import { Line, Group, Rect } from 'react-konva';
4
- import ImageComponent from './image-konva-tooltip';
5
- import { faCorrect, faWrong } from './icons';
6
-
7
- class PolygonComponent extends React.Component {
8
- constructor(props) {
9
- super(props);
10
- this.state = {
11
- hovered: false,
12
- };
13
- }
14
-
15
- getPolygonCenter = (points) => {
16
- const x = points.map(({ x }) => x);
17
- const y = points.map(({ y }) => y);
18
- const minX = Math.min.apply(null, x);
19
- const maxX = Math.max.apply(null, x);
20
- const minY = Math.min.apply(null, y);
21
- const maxY = Math.max.apply(null, y);
22
- return [(minX + maxX) / 2, (minY + maxY) / 2];
23
- };
24
-
25
- parsePointsForKonva = (points) => {
26
- const parsedPoints = [];
27
- points.forEach(({ x, y }) => {
28
- parsedPoints.push(x);
29
- parsedPoints.push(y);
30
- });
31
- return parsedPoints;
32
- };
33
-
34
- handleClick = (e) => {
35
- const { onClick, id, selected, disabled } = this.props;
36
-
37
- if (!disabled) {
38
- e.cancelBubble = true;
39
- onClick({ id, selected: !selected, selector: 'Mouse' });
40
- }
41
- };
42
-
43
- handleMouseEnter = () => {
44
- const { disabled } = this.props;
45
-
46
- if (!disabled) {
47
- document.body.style.cursor = 'pointer';
48
- }
49
- this.setState({ hovered: true });
50
- };
51
-
52
- handleMouseLeave = () => {
53
- document.body.style.cursor = 'default';
54
- this.setState({ hovered: false });
55
- };
56
-
57
- getEvaluateOutlineColor = (isCorrect, markAsCorrect, outlineColor) =>
58
- markAsCorrect ? 'green' : isCorrect ? outlineColor : 'red';
59
-
60
- getOutlineWidth = (showCorrectEnabled, selected, markAsCorrect, strokeWidth) =>
61
- markAsCorrect || (!markAsCorrect && !showCorrectEnabled && selected) ? strokeWidth : 0;
62
-
63
- render() {
64
- const {
65
- hotspotColor,
66
- isCorrect,
67
- isEvaluateMode,
68
- hoverOutlineColor,
69
- outlineColor,
70
- selected,
71
- points,
72
- evaluateText,
73
- strokeWidth,
74
- scale,
75
- markAsCorrect,
76
- selectedHotspotColor,
77
- showCorrectEnabled,
78
- } = this.props;
79
-
80
- const { hovered } = this.state;
81
-
82
- const outlineColorParsed = isEvaluateMode
83
- ? this.getEvaluateOutlineColor(isCorrect, markAsCorrect, outlineColor)
84
- : outlineColor;
85
- const outlineWidth = this.getOutlineWidth(showCorrectEnabled, selected, markAsCorrect, strokeWidth);
86
-
87
- const pointsParsed = this.parsePointsForKonva(points);
88
- const center = this.getPolygonCenter(points);
89
- const iconX = center[0];
90
- const iconY = center[1];
91
-
92
- // "Show Correct Answer" Enabled:
93
- // - Correctly Selected: white checkmark in green circle
94
- // - Correctly Not Selected: none
95
- // - Incorrectly Selected: none
96
- // - Incorrectly Not Selected: white checkmark in green circle
97
- // "Show Correct Answer" Disabled:
98
- // - Correctly Selected:
99
- // - white checkmark in green circle
100
- // - heavy outline, as on “Gather”
101
- // - Correctly Not Selected: none
102
- // - Incorrectly Selected:
103
- // - white "X" in red circle
104
- // - heavy outline around the selection should appear in red
105
- // - Incorrectly Not Selected: white "X" in red circle
106
- let iconSrc;
107
-
108
- if (showCorrectEnabled) {
109
- if ((selected && isCorrect) || (!selected && !isCorrect)) {
110
- iconSrc = faCorrect;
111
- }
112
- } else {
113
- if (selected) {
114
- if (isCorrect) {
115
- iconSrc = faCorrect;
116
- } else {
117
- iconSrc = faWrong;
118
- }
119
- } else if (!isCorrect) {
120
- iconSrc = faWrong;
121
- }
122
- }
123
- const useHoveredStyle = hovered && hoverOutlineColor;
124
-
125
- const xValues = pointsParsed.filter((_, index) => index % 2 === 0); // Even indices are x-coordinates
126
- const yValues = pointsParsed.filter((_, index) => index % 2 !== 0); // Odd indices are y-coordinates
127
-
128
- const minX = Math.min(...xValues);
129
- const maxX = Math.max(...xValues);
130
- const minY = Math.min(...yValues);
131
- const maxY = Math.max(...yValues);
132
-
133
- const rectX = minX;
134
- const rectY = minY;
135
- const rectWidth = maxX - minX;
136
- const rectHeight = maxY - minY;
137
-
138
- return (
139
- <Group scaleX={scale} scaleY={scale}>
140
- {useHoveredStyle && (
141
- <Rect
142
- x={rectX}
143
- y={rectY}
144
- width={rectWidth}
145
- height={rectHeight}
146
- stroke={selected ? 'transparent' : hoverOutlineColor}
147
- strokeWidth={strokeWidth}
148
- />
149
- )}
150
- <Line
151
- points={pointsParsed}
152
- closed={true}
153
- fill={selected && selectedHotspotColor? selectedHotspotColor : hotspotColor}
154
- onClick={this.handleClick}
155
- onTap={this.handleClick}
156
- draggable={false}
157
- stroke={useHoveredStyle && !selected ? 'transparent' : outlineColorParsed}
158
- strokeWidth={useHoveredStyle && !selected ? 0 : outlineWidth}
159
- onMouseLeave={this.handleMouseLeave}
160
- onMouseEnter={this.handleMouseEnter}
161
- cursor='pointer'
162
- position='relative'
163
- />
164
- {isEvaluateMode && iconSrc ? <ImageComponent src={iconSrc} x={iconX} y={iconY} tooltip={evaluateText} /> : null}
165
- </Group>
166
- );
167
- }
168
- }
169
-
170
- PolygonComponent.propTypes = {
171
- hotspotColor: PropTypes.string.isRequired,
172
- id: PropTypes.string.isRequired,
173
- isCorrect: PropTypes.bool.isRequired,
174
- isEvaluateMode: PropTypes.bool.isRequired,
175
- hoverOutlineColor: PropTypes.string,
176
- disabled: PropTypes.bool.isRequired,
177
- onClick: PropTypes.func.isRequired,
178
- outlineColor: PropTypes.string.isRequired,
179
- points: PropTypes.array.isRequired,
180
- selected: PropTypes.bool.isRequired,
181
- evaluateText: PropTypes.string,
182
- selectedHotspotColor: PropTypes.string,
183
- strokeWidth: PropTypes.number,
184
- scale: PropTypes.number,
185
- markAsCorrect: PropTypes.bool.isRequired,
186
- showCorrectEnabled: PropTypes.bool.isRequired,
187
- };
188
-
189
- PolygonComponent.defaultProps = {
190
- evaluateText: null,
191
- strokeWidth: 5,
192
- scale: 1,
193
- };
194
-
195
- export default PolygonComponent;
@@ -1,171 +0,0 @@
1
- import React from 'react';
2
- import PropTypes from 'prop-types';
3
- import { Rect, Group } from 'react-konva';
4
- import ImageComponent from './image-konva-tooltip';
5
- import { faCorrect, faWrong } from './icons';
6
-
7
- class RectComponent extends React.Component {
8
- constructor(props) {
9
- super(props);
10
- this.state = {
11
- hovered: false,
12
- };
13
- }
14
-
15
- handleClick = (e) => {
16
- const { onClick, id, selected, disabled } = this.props;
17
-
18
- if (!disabled) {
19
- e.cancelBubble = true;
20
- onClick({ id, selected: !selected, selector: 'Mouse' });
21
- }
22
- };
23
-
24
- handleMouseEnter = () => {
25
- const { disabled } = this.props;
26
-
27
- if (!disabled) {
28
- document.body.style.cursor = 'pointer';
29
- }
30
- this.setState({ hovered: true });
31
- };
32
-
33
- handleMouseLeave = () => {
34
- document.body.style.cursor = 'default';
35
- this.setState({ hovered: false });
36
- };
37
-
38
- getEvaluateOutlineColor = (isCorrect, markAsCorrect, outlineColor) =>
39
- markAsCorrect ? 'green' : isCorrect ? outlineColor : 'red';
40
-
41
- getOutlineWidth = (showCorrectEnabled, selected, markAsCorrect, strokeWidth) =>
42
- markAsCorrect || (!markAsCorrect && !showCorrectEnabled && selected) ? strokeWidth : 0;
43
-
44
- render() {
45
- const {
46
- height,
47
- hotspotColor,
48
- hoverOutlineColor,
49
- selectedHotspotColor,
50
- isCorrect,
51
- isEvaluateMode,
52
- outlineColor,
53
- selected,
54
- width,
55
- x,
56
- y,
57
- evaluateText,
58
- strokeWidth,
59
- scale,
60
- markAsCorrect,
61
- showCorrectEnabled,
62
- } = this.props;
63
-
64
- const outlineColorParsed = isEvaluateMode
65
- ? this.getEvaluateOutlineColor(isCorrect, markAsCorrect, outlineColor)
66
- : outlineColor;
67
-
68
- const outlineWidth = this.getOutlineWidth(showCorrectEnabled, selected, markAsCorrect, strokeWidth);
69
-
70
- const iconX = x + width / 2 - 10;
71
- const iconY = y + height / 2 - 10;
72
-
73
- // "Show Correct Answer" Enabled:
74
- // - Correctly Selected: white checkmark in green circle
75
- // - Correctly Not Selected: none
76
- // - Incorrectly Selected: none
77
- // - Incorrectly Not Selected: white checkmark in green circle
78
- // "Show Correct Answer" Disabled:
79
- // - Correctly Selected:
80
- // - white checkmark in green circle
81
- // - heavy outline, as on “Gather”
82
- // - Correctly Not Selected: none
83
- // - Incorrectly Selected:
84
- // - white "X" in red circle
85
- // - heavy outline around the selection should appear in red
86
- // - Incorrectly Not Selected: white "X" in red circle
87
- let iconSrc;
88
-
89
- if (showCorrectEnabled) {
90
- if ((selected && isCorrect) || (!selected && !isCorrect)) {
91
- iconSrc = faCorrect;
92
- }
93
- } else {
94
- if (selected) {
95
- if (isCorrect) {
96
- iconSrc = faCorrect;
97
- } else {
98
- iconSrc = faWrong;
99
- }
100
- } else if (!isCorrect) {
101
- iconSrc = faWrong;
102
- }
103
- }
104
-
105
- const { hovered } = this.state;
106
- const useHoveredStyle = hovered && hoverOutlineColor;
107
-
108
- return (
109
- <Group scaleX={scale} scaleY={scale}>
110
- {useHoveredStyle && (
111
- <Rect
112
- x={x}
113
- y={y}
114
- width={width}
115
- height={height}
116
- stroke={selected ? 'transparent' : hoverOutlineColor}
117
- strokeWidth={strokeWidth}
118
- listening={false}
119
- />
120
- )}
121
- <Rect
122
- x={x}
123
- y={y}
124
- width={width}
125
- height={height}
126
- fill={selected && selectedHotspotColor ? selectedHotspotColor : hotspotColor}
127
- onClick={this.handleClick}
128
- onTap={this.handleClick}
129
- draggable={false}
130
- stroke={useHoveredStyle && !selected ? 'transparent' : outlineColorParsed}
131
- strokeWidth={useHoveredStyle && !selected ? 0 : outlineWidth}
132
- onMouseLeave={this.handleMouseLeave}
133
- onMouseEnter={this.handleMouseEnter}
134
- cursor="pointer"
135
- />
136
- {isEvaluateMode && iconSrc ? <ImageComponent src={iconSrc} x={iconX} y={iconY} tooltip={evaluateText} /> : null}
137
- </Group>
138
- );
139
- }
140
- }
141
-
142
- RectComponent.propTypes = {
143
- height: PropTypes.number.isRequired,
144
- hotspotColor: PropTypes.string.isRequired,
145
- id: PropTypes.string.isRequired,
146
- isCorrect: PropTypes.oneOfType([PropTypes.bool, PropTypes.string]),
147
- isEvaluateMode: PropTypes.bool.isRequired,
148
- hoverOutlineColor: PropTypes.string,
149
- disabled: PropTypes.bool.isRequired,
150
- onClick: PropTypes.func.isRequired,
151
- outlineColor: PropTypes.string.isRequired,
152
- selected: PropTypes.bool.isRequired,
153
- width: PropTypes.number.isRequired,
154
- x: PropTypes.number.isRequired,
155
- y: PropTypes.number.isRequired,
156
- evaluateText: PropTypes.string,
157
- strokeWidth: PropTypes.number,
158
- scale: PropTypes.number,
159
- selectedHotspotColor: PropTypes.string,
160
- markAsCorrect: PropTypes.bool.isRequired,
161
- showCorrectEnabled: PropTypes.bool.isRequired,
162
- };
163
-
164
- RectComponent.defaultProps = {
165
- isCorrect: false,
166
- evaluateText: null,
167
- strokeWidth: 5,
168
- scale: 1,
169
- };
170
-
171
- export default RectComponent;
package/src/index.js DELETED
@@ -1,226 +0,0 @@
1
- import React from 'react';
2
- import { createRoot } from 'react-dom/client';
3
- import { renderMath } from '@pie-lib/math-rendering';
4
- import { EnableAudioAutoplayImage } from '@pie-lib/render-ui';
5
- import { SessionChangedEvent, ModelSetEvent } from '@pie-framework/pie-player-events';
6
-
7
- import HotspotComponent from './hotspot';
8
- import { updateSessionValue, updateSessionMetadata } from './session-updater';
9
-
10
- export default class Hotspot extends HTMLElement {
11
- constructor() {
12
- super();
13
- this._model = null;
14
- this._session = null;
15
- this._audioInitialized = false;
16
- this.audioComplete = false;
17
- this._root = null;
18
- }
19
-
20
- set model(m) {
21
- this._model = m;
22
-
23
- this.dispatchEvent(new ModelSetEvent(this.tagName.toLowerCase(), this.isComplete(), !!this._model));
24
- this._audioInitialized = false;
25
- this._render();
26
- }
27
-
28
- isComplete() {
29
- if (!this._session || !this._session.answers) {
30
- return false;
31
- }
32
-
33
- const { autoplayAudioEnabled, completeAudioEnabled } = this._model || {};
34
- const elementContext = this;
35
-
36
- // check audio completion if audio settings are enabled and audio actually exists
37
- if (autoplayAudioEnabled && completeAudioEnabled && !this.audioComplete) {
38
- if (elementContext) {
39
- const audio = elementContext.querySelector('audio');
40
- const isInsidePrompt = audio && audio.closest('#preview-prompt');
41
-
42
- // only require audio completion if audio exists and is inside the prompt
43
- if (audio && isInsidePrompt) {
44
- return false;
45
- }
46
- }
47
- }
48
-
49
- if (!Array.isArray(this._session.answers)) {
50
- return false;
51
- }
52
-
53
- return this._session.answers.length > 0;
54
- }
55
-
56
- set session(s) {
57
- if (s && !s.answers) {
58
- s.answers = [];
59
- }
60
-
61
- this._session = s;
62
- this._render();
63
- }
64
-
65
- get session() {
66
- return this._session;
67
- }
68
-
69
- onSelectChoice(data) {
70
- updateSessionValue(this._session, this._model, data);
71
-
72
- this.dispatchEvent(new SessionChangedEvent(this.tagName.toLowerCase(), this.isComplete()));
73
-
74
- this._render();
75
- }
76
-
77
- _createAudioInfoToast() {
78
- const info = document.createElement('div');
79
- info.id = 'play-audio-info';
80
-
81
- Object.assign(info.style, {
82
- position: 'absolute',
83
- top: 0,
84
- width: '100%',
85
- height: '100%',
86
- display: 'flex',
87
- justifyContent: 'center',
88
- alignItems: 'center',
89
- background: 'white',
90
- zIndex: '1000',
91
- cursor: 'pointer',
92
- });
93
-
94
- const img = document.createElement('img');
95
- img.src = EnableAudioAutoplayImage;
96
- img.alt = 'Click anywhere to enable audio autoplay';
97
- img.width = 500;
98
- img.height = 300;
99
-
100
- info.appendChild(img);
101
- return info;
102
- }
103
-
104
- connectedCallback() {
105
- this._render();
106
-
107
- // Observation: audio in Chrome will have the autoplay attribute,
108
- // while other browsers will not have the autoplay attribute and will need a user interaction to play the audio
109
- // This workaround fixes the issue of audio being cached and played on any user interaction in Safari and Firefox
110
- const observer = new MutationObserver((mutationsList, observer) => {
111
- mutationsList.forEach((mutation) => {
112
- if (mutation.type === 'childList') {
113
- if (this._audioInitialized) return;
114
- const audio = this.querySelector('audio');
115
- const isInsidePrompt = audio && audio.closest('#preview-prompt');
116
-
117
- if (!this._model) return;
118
- if (!this._model.autoplayAudioEnabled) return;
119
- if (audio && !isInsidePrompt) return;
120
- if (!audio) return;
121
-
122
- const info = this._createAudioInfoToast();
123
- const container = this.querySelector('#main-container');
124
- const enableAudio = () => {
125
- if (this.querySelector('#play-audio-info')) {
126
- audio.play();
127
- container.removeChild(info);
128
- }
129
-
130
- document.removeEventListener('click', enableAudio);
131
- };
132
-
133
- // if the audio is paused, it means the user has not interacted with the page yet and the audio will not play
134
- // FIX FOR SAFARI: play with a slight delay to check if autoplay was blocked
135
- setTimeout(() => {
136
- if (audio.paused && !this.querySelector('#play-audio-info')) {
137
- // add info message as a toast to enable audio playback
138
- container.appendChild(info);
139
- document.addEventListener('click', enableAudio);
140
- } else {
141
- document.removeEventListener('click', enableAudio);
142
- }
143
- }, 500);
144
-
145
- // we need to listen for the playing event to remove the toast in case the audio plays because of re-rendering
146
- const handlePlaying = () => {
147
- //timestamp when auto-played audio started playing
148
- updateSessionMetadata(this._session, { audioStartTime: new Date().getTime() });
149
-
150
- const info = this.querySelector('#play-audio-info');
151
- if (info) {
152
- container.removeChild(info);
153
- }
154
-
155
- audio.removeEventListener('playing', handlePlaying);
156
- };
157
-
158
- audio.addEventListener('playing', handlePlaying);
159
-
160
- // we need to listen for the ended event to update the isComplete state
161
- const handleEnded = () => {
162
- //timestamp when auto-played audio completed playing
163
- updateSessionMetadata(this._session, { audioEndTime: new Date().getTime() });
164
-
165
- let { audioStartTime, audioEndTime, waitTime } = this._session;
166
- if (!waitTime && audioStartTime && audioEndTime) {
167
- // waitTime is elapsed time the user waited for auto-played audio to finish
168
- this._session.waitTime = audioEndTime - audioStartTime;
169
- }
170
-
171
- this.audioComplete = true;
172
- this.dispatchEvent(new SessionChangedEvent(this.tagName.toLowerCase(), this.isComplete()));
173
-
174
- audio.removeEventListener('ended', handleEnded);
175
- };
176
-
177
- audio.addEventListener('ended', handleEnded);
178
-
179
- // store references to remove later
180
- this._audio = audio;
181
- this._handlePlaying = handlePlaying;
182
- this._handleEnded = handleEnded;
183
- this._enableAudio = enableAudio;
184
- // set to true to prevent multiple initializations
185
- this._audioInitialized = true;
186
-
187
- observer.disconnect();
188
- }
189
- });
190
- });
191
-
192
- observer.observe(this, { childList: true, subtree: true });
193
- }
194
-
195
- _render() {
196
- if (this._model && this._session) {
197
- const el = React.createElement(HotspotComponent, {
198
- model: this._model,
199
- session: this._session,
200
- onSelectChoice: this.onSelectChoice.bind(this),
201
- });
202
-
203
- if (!this._root) {
204
- this._root = createRoot(this);
205
- }
206
- this._root.render(el);
207
- queueMicrotask(() => {
208
- renderMath(this);
209
- });
210
- }
211
- }
212
-
213
- disconnectedCallback() {
214
- document.removeEventListener('click', this._enableAudio);
215
-
216
- if (this._audio) {
217
- this._audio.removeEventListener('playing', this._handlePlaying);
218
- this._audio.removeEventListener('ended', this._handleEnded);
219
- this._audio = null;
220
- }
221
-
222
- if (this._root) {
223
- this._root.unmount();
224
- }
225
- }
226
- }
@@ -1,29 +0,0 @@
1
- export function updateSessionValue(session, model, data) {
2
- const { id, selected } = data;
3
- const { multipleCorrect } = model || {};
4
- session.answers = session.answers || [];
5
-
6
- if (!selected) {
7
- session.answers = session.answers.filter((answer) => answer.id !== id);
8
- } else {
9
- const item = { id };
10
- if (multipleCorrect) {
11
- session.answers.push(item);
12
- } else {
13
- session.answers = [item];
14
- }
15
-
16
- //update session metadata
17
- session.selector = data.selector;
18
- }
19
- }
20
-
21
- export function updateSessionMetadata(session, metadata) {
22
- session.audioStartTime = session.audioStartTime || metadata.audioStartTime; //timestamp when auto-played audio started playing
23
- session.audioEndTime = session.audioEndTime || metadata.audioEndTime; //timestamp when auto-played audio completed playing
24
-
25
- if (!session.waitTime && session.audioStartTime && session.audioEndTime) {
26
- // waitTime is elapsed time the user waited for auto-played audio to finish
27
- session.waitTime = session.audioEndTime - session.audioStartTime;
28
- }
29
- }