@pie-element/hotspot 11.1.2-next.5 → 11.1.3

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/CHANGELOG.json +997 -0
  2. package/CHANGELOG.md +2226 -0
  3. package/LICENSE.md +5 -0
  4. package/README.md +1 -0
  5. package/configure/CHANGELOG.json +682 -0
  6. package/configure/CHANGELOG.md +1963 -0
  7. package/configure/lib/DeleteWidget.js +64 -0
  8. package/configure/lib/DeleteWidget.js.map +1 -0
  9. package/configure/lib/button.js +42 -0
  10. package/configure/lib/button.js.map +1 -0
  11. package/configure/lib/buttons/circle.js +33 -0
  12. package/configure/lib/buttons/circle.js.map +1 -0
  13. package/configure/lib/buttons/polygon.js +39 -0
  14. package/configure/lib/buttons/polygon.js.map +1 -0
  15. package/configure/lib/buttons/rectangle.js +39 -0
  16. package/configure/lib/buttons/rectangle.js.map +1 -0
  17. package/configure/lib/defaults.js +155 -0
  18. package/configure/lib/defaults.js.map +1 -0
  19. package/configure/lib/hotspot-circle.js +192 -0
  20. package/configure/lib/hotspot-circle.js.map +1 -0
  21. package/configure/lib/hotspot-container.js +320 -0
  22. package/configure/lib/hotspot-container.js.map +1 -0
  23. package/configure/lib/hotspot-drawable.js +519 -0
  24. package/configure/lib/hotspot-drawable.js.map +1 -0
  25. package/configure/lib/hotspot-palette.js +107 -0
  26. package/configure/lib/hotspot-palette.js.map +1 -0
  27. package/configure/lib/hotspot-polygon.js +293 -0
  28. package/configure/lib/hotspot-polygon.js.map +1 -0
  29. package/configure/lib/hotspot-rectangle.js +190 -0
  30. package/configure/lib/hotspot-rectangle.js.map +1 -0
  31. package/configure/lib/icons.js +7 -0
  32. package/configure/lib/icons.js.map +1 -0
  33. package/configure/lib/image-konva.js +66 -0
  34. package/configure/lib/image-konva.js.map +1 -0
  35. package/configure/lib/index.js +194 -0
  36. package/configure/lib/index.js.map +1 -0
  37. package/configure/lib/root.js +330 -0
  38. package/configure/lib/root.js.map +1 -0
  39. package/configure/lib/shapes/circle.js +84 -0
  40. package/configure/lib/shapes/circle.js.map +1 -0
  41. package/configure/lib/shapes/index.js +50 -0
  42. package/configure/lib/shapes/index.js.map +1 -0
  43. package/configure/lib/shapes/polygon.js +82 -0
  44. package/configure/lib/shapes/polygon.js.map +1 -0
  45. package/configure/lib/shapes/rectagle.js +84 -0
  46. package/configure/lib/shapes/rectagle.js.map +1 -0
  47. package/configure/lib/shapes/utils.js +21 -0
  48. package/configure/lib/shapes/utils.js.map +1 -0
  49. package/configure/lib/upload-control.js +41 -0
  50. package/configure/lib/upload-control.js.map +1 -0
  51. package/configure/lib/utils.js +185 -0
  52. package/configure/lib/utils.js.map +1 -0
  53. package/configure/package.json +26 -0
  54. package/configure/src/DeleteWidget.jsx +51 -0
  55. package/configure/src/__tests__/DeleteWidget.test.jsx +366 -0
  56. package/configure/src/__tests__/button.test.jsx +198 -0
  57. package/configure/src/__tests__/hotspot-circle.test.jsx +259 -0
  58. package/configure/src/__tests__/hotspot-container.test.js +366 -0
  59. package/configure/src/__tests__/hotspot-drawable.test.js +271 -0
  60. package/configure/src/__tests__/hotspot-palette.test.jsx +71 -0
  61. package/configure/src/__tests__/image-konva.test.jsx +226 -0
  62. package/configure/src/__tests__/index.test.js +329 -0
  63. package/configure/src/__tests__/root.test.js +400 -0
  64. package/configure/src/__tests__/utils.test.js +241 -0
  65. package/configure/src/button.jsx +35 -0
  66. package/configure/src/buttons/circle.jsx +18 -0
  67. package/configure/src/buttons/polygon.jsx +29 -0
  68. package/configure/src/buttons/rectangle.jsx +29 -0
  69. package/configure/src/defaults.js +109 -0
  70. package/configure/src/hotspot-circle.jsx +183 -0
  71. package/configure/src/hotspot-container.jsx +330 -0
  72. package/configure/src/hotspot-drawable.jsx +527 -0
  73. package/configure/src/hotspot-palette.jsx +90 -0
  74. package/configure/src/hotspot-polygon.jsx +294 -0
  75. package/configure/src/hotspot-rectangle.jsx +169 -0
  76. package/configure/src/icons.js +5 -0
  77. package/configure/src/image-konva.jsx +63 -0
  78. package/configure/src/index.js +208 -0
  79. package/configure/src/root.jsx +346 -0
  80. package/configure/src/shapes/circle.js +81 -0
  81. package/configure/src/shapes/index.js +4 -0
  82. package/configure/src/shapes/polygon.js +81 -0
  83. package/configure/src/shapes/rectagle.js +82 -0
  84. package/configure/src/shapes/utils.js +16 -0
  85. package/configure/src/upload-control.jsx +33 -0
  86. package/configure/src/utils.js +210 -0
  87. package/controller/CHANGELOG.json +362 -0
  88. package/controller/CHANGELOG.md +1304 -0
  89. package/controller/lib/defaults.js +33 -0
  90. package/controller/lib/defaults.js.map +1 -0
  91. package/controller/lib/index.js +341 -0
  92. package/controller/lib/index.js.map +1 -0
  93. package/controller/lib/utils.js +32 -0
  94. package/controller/lib/utils.js.map +1 -0
  95. package/controller/package.json +18 -0
  96. package/controller/src/__tests__/index.test.js +419 -0
  97. package/controller/src/__tests__/utils.test.js +5 -0
  98. package/controller/src/defaults.js +19 -0
  99. package/controller/src/index.js +328 -0
  100. package/controller/src/utils.js +29 -0
  101. package/docs/config-schema.json +2023 -0
  102. package/docs/config-schema.json.md +1495 -0
  103. package/docs/demo/config.js +8 -0
  104. package/docs/demo/generate.js +118 -0
  105. package/docs/demo/index.html +1 -0
  106. package/docs/demo/session.js +11 -0
  107. package/docs/pie-schema.json +1204 -0
  108. package/docs/pie-schema.json.md +851 -0
  109. package/lib/hotspot/circle.js +156 -0
  110. package/lib/hotspot/circle.js.map +1 -0
  111. package/lib/hotspot/container.js +206 -0
  112. package/lib/hotspot/container.js.map +1 -0
  113. package/lib/hotspot/icons.js +8 -0
  114. package/lib/hotspot/icons.js.map +1 -0
  115. package/lib/hotspot/image-konva-tooltip.js +86 -0
  116. package/lib/hotspot/image-konva-tooltip.js.map +1 -0
  117. package/lib/hotspot/index.js +163 -0
  118. package/lib/hotspot/index.js.map +1 -0
  119. package/lib/hotspot/polygon.js +203 -0
  120. package/lib/hotspot/polygon.js.map +1 -0
  121. package/lib/hotspot/rectangle.js +175 -0
  122. package/lib/hotspot/rectangle.js.map +1 -0
  123. package/lib/index.js +213 -0
  124. package/lib/index.js.map +1 -0
  125. package/lib/session-updater.js +42 -0
  126. package/lib/session-updater.js.map +1 -0
  127. package/package.json +18 -83
  128. package/src/__tests__/container.test.jsx +58 -0
  129. package/src/__tests__/index.test.js +123 -0
  130. package/src/__tests__/session-updater.test.jsx +69 -0
  131. package/src/hotspot/__tests__/circle.test.jsx +464 -0
  132. package/src/hotspot/__tests__/container.test.jsx +546 -0
  133. package/src/hotspot/__tests__/image-konva-tooltip.test.jsx +510 -0
  134. package/src/hotspot/__tests__/polygon.test.jsx +502 -0
  135. package/src/hotspot/__tests__/rectangle.test.jsx +418 -0
  136. package/src/hotspot/circle.jsx +152 -0
  137. package/src/hotspot/container.jsx +217 -0
  138. package/src/hotspot/icons.js +7 -0
  139. package/src/hotspot/image-konva-tooltip.jsx +76 -0
  140. package/src/hotspot/index.jsx +165 -0
  141. package/src/hotspot/polygon.jsx +195 -0
  142. package/src/hotspot/rectangle.jsx +171 -0
  143. package/src/index.js +226 -0
  144. package/src/session-updater.js +29 -0
  145. package/configure.js +0 -2
  146. package/controller.js +0 -1
  147. package/dist/author/DeleteWidget.d.ts +0 -38
  148. package/dist/author/DeleteWidget.js +0 -46
  149. package/dist/author/button.d.ts +0 -31
  150. package/dist/author/button.js +0 -27
  151. package/dist/author/buttons/circle.d.ts +0 -18
  152. package/dist/author/buttons/circle.js +0 -25
  153. package/dist/author/buttons/polygon.d.ts +0 -18
  154. package/dist/author/buttons/polygon.js +0 -36
  155. package/dist/author/buttons/rectangle.d.ts +0 -18
  156. package/dist/author/buttons/rectangle.js +0 -36
  157. package/dist/author/defaults.d.ts +0 -157
  158. package/dist/author/defaults.js +0 -119
  159. package/dist/author/hotspot-circle.d.ts +0 -21
  160. package/dist/author/hotspot-circle.js +0 -124
  161. package/dist/author/hotspot-container.d.ts +0 -29
  162. package/dist/author/hotspot-container.js +0 -210
  163. package/dist/author/hotspot-drawable.d.ts +0 -31
  164. package/dist/author/hotspot-drawable.js +0 -312
  165. package/dist/author/hotspot-palette.d.ts +0 -14
  166. package/dist/author/hotspot-palette.js +0 -72
  167. package/dist/author/hotspot-polygon.d.ts +0 -38
  168. package/dist/author/hotspot-polygon.js +0 -200
  169. package/dist/author/hotspot-rectangle.d.ts +0 -20
  170. package/dist/author/hotspot-rectangle.js +0 -119
  171. package/dist/author/icons.d.ts +0 -9
  172. package/dist/author/icons.js +0 -4
  173. package/dist/author/image-konva.d.ts +0 -19
  174. package/dist/author/image-konva.js +0 -49
  175. package/dist/author/index.d.ts +0 -52
  176. package/dist/author/index.js +0 -143
  177. package/dist/author/root.d.ts +0 -15
  178. package/dist/author/root.js +0 -215
  179. package/dist/author/shapes/circle.d.ts +0 -18
  180. package/dist/author/shapes/circle.js +0 -47
  181. package/dist/author/shapes/index.d.ts +0 -12
  182. package/dist/author/shapes/polygon.d.ts +0 -19
  183. package/dist/author/shapes/polygon.js +0 -51
  184. package/dist/author/shapes/rectagle.d.ts +0 -18
  185. package/dist/author/shapes/rectagle.js +0 -57
  186. package/dist/author/shapes/utils.d.ts +0 -19
  187. package/dist/author/shapes/utils.js +0 -16
  188. package/dist/author/upload-control.d.ts +0 -29
  189. package/dist/author/upload-control.js +0 -28
  190. package/dist/author/utils.d.ts +0 -24
  191. package/dist/author/utils.js +0 -83
  192. package/dist/browser/ReactKonva-CFo7dxdy.js +0 -19336
  193. package/dist/browser/ReactKonva-CFo7dxdy.js.map +0 -1
  194. package/dist/browser/author/index.js +0 -47549
  195. package/dist/browser/author/index.js.map +0 -1
  196. package/dist/browser/browser-CfnAFove.js +0 -219
  197. package/dist/browser/browser-CfnAFove.js.map +0 -1
  198. package/dist/browser/controller/index.js +0 -198
  199. package/dist/browser/controller/index.js.map +0 -1
  200. package/dist/browser/delivery/index.js +0 -2460
  201. package/dist/browser/delivery/index.js.map +0 -1
  202. package/dist/browser/dist-CeB-1djc.js +0 -100
  203. package/dist/browser/dist-CeB-1djc.js.map +0 -1
  204. package/dist/browser/hotspot.css +0 -2
  205. package/dist/controller/defaults.d.ts +0 -35
  206. package/dist/controller/defaults.js +0 -29
  207. package/dist/controller/index.d.ts +0 -22
  208. package/dist/controller/index.js +0 -154
  209. package/dist/controller/utils.d.ts +0 -10
  210. package/dist/controller/utils.js +0 -12
  211. package/dist/delivery/hotspot/circle.d.ts +0 -19
  212. package/dist/delivery/hotspot/circle.js +0 -100
  213. package/dist/delivery/hotspot/container.d.ts +0 -16
  214. package/dist/delivery/hotspot/container.js +0 -150
  215. package/dist/delivery/hotspot/icons.d.ts +0 -10
  216. package/dist/delivery/hotspot/icons.js +0 -4
  217. package/dist/delivery/hotspot/image-konva-tooltip.d.ts +0 -19
  218. package/dist/delivery/hotspot/image-konva-tooltip.js +0 -66
  219. package/dist/delivery/hotspot/index.d.ts +0 -17
  220. package/dist/delivery/hotspot/index.js +0 -114
  221. package/dist/delivery/hotspot/polygon.d.ts +0 -21
  222. package/dist/delivery/hotspot/polygon.js +0 -108
  223. package/dist/delivery/hotspot/rectangle.d.ts +0 -19
  224. package/dist/delivery/hotspot/rectangle.js +0 -104
  225. package/dist/delivery/index.d.ts +0 -20
  226. package/dist/delivery/index.js +0 -107
  227. package/dist/delivery/session-updater.d.ts +0 -10
  228. package/dist/delivery/session-updater.js +0 -14
  229. package/dist/index.d.ts +0 -1
  230. package/dist/index.iife.d.ts +0 -8
  231. package/dist/index.iife.js +0 -169
  232. package/dist/index.js +0 -2
  233. package/dist/runtime-support.d.ts +0 -12
  234. package/dist/runtime-support.js +0 -12
@@ -0,0 +1,195 @@
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;
@@ -0,0 +1,171 @@
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 ADDED
@@ -0,0 +1,226 @@
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
+ }
@@ -0,0 +1,29 @@
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
+ }
package/configure.js DELETED
@@ -1,2 +0,0 @@
1
- export { default } from './dist/author/index.js';
2
- export * from './dist/author/index.js';
package/controller.js DELETED
@@ -1 +0,0 @@
1
- export * from './dist/controller/index.js';
@@ -1,38 +0,0 @@
1
- /**
2
- * @synced-from pie-elements/packages/hotspot/configure/src/DeleteWidget.jsx
3
- * @auto-generated
4
- *
5
- * This file is automatically synced from pie-elements and converted to TypeScript.
6
- * Manual edits will be overwritten on next sync.
7
- * To make changes, edit the upstream JavaScript file and run sync again.
8
- */
9
- import React from 'react';
10
- import PropTypes from 'prop-types';
11
- declare const DeleteWidget: {
12
- ({ height, id, width, x, y, points, isCircle, radius, handleWidgetClick }: {
13
- height: any;
14
- id: any;
15
- width: any;
16
- x: any;
17
- y: any;
18
- points: any;
19
- isCircle: any;
20
- radius: any;
21
- handleWidgetClick: any;
22
- }): React.JSX.Element;
23
- propTypes: {
24
- id: PropTypes.Validator<string>;
25
- height: PropTypes.Requireable<number>;
26
- width: PropTypes.Requireable<number>;
27
- x: PropTypes.Validator<number>;
28
- y: PropTypes.Validator<number>;
29
- handleWidgetClick: PropTypes.Validator<(...args: any[]) => any>;
30
- radius: PropTypes.Requireable<number>;
31
- isCircle: PropTypes.Requireable<boolean>;
32
- points: PropTypes.Requireable<(PropTypes.InferProps<{
33
- x: PropTypes.Requireable<number>;
34
- y: PropTypes.Requireable<number>;
35
- }> | null | undefined)[]>;
36
- };
37
- };
38
- export default DeleteWidget;