@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,502 @@
1
+ import React from 'react';
2
+ import { render, fireEvent } from '@testing-library/react';
3
+ import Konva from 'konva';
4
+ import PolygonComponent from '../polygon';
5
+
6
+ Konva.isBrowser = false;
7
+
8
+ jest.mock('react-konva', () => {
9
+ const React = require('react');
10
+ return {
11
+ Line: ({ onClick, onTap, onMouseEnter, onMouseLeave, ...props }) => {
12
+ const handleClick = (e) => {
13
+ if (onClick) onClick(e);
14
+ if (onTap) onTap(e);
15
+ };
16
+ return React.createElement('div', {
17
+ 'data-testid': 'line',
18
+ onClick: handleClick,
19
+ onMouseEnter,
20
+ onMouseLeave,
21
+ ...props,
22
+ });
23
+ },
24
+ Rect: (props) => React.createElement('div', { 'data-testid': 'rect', ...props }),
25
+ Group: ({ children, ...props }) => React.createElement('div', { 'data-testid': 'group', ...props }, children),
26
+ };
27
+ });
28
+
29
+ jest.mock('../image-konva-tooltip', () => {
30
+ return function ImageComponent({ src, x, y, tooltip }) {
31
+ return <div data-testid="icon-image" data-src={src} data-x={x} data-y={y} data-tooltip={tooltip} />;
32
+ };
33
+ });
34
+
35
+ describe('PolygonComponent', () => {
36
+ let defaultProps;
37
+
38
+ beforeEach(() => {
39
+ defaultProps = {
40
+ id: 'polygon1',
41
+ points: [
42
+ { x: 10, y: 10 },
43
+ { x: 100, y: 10 },
44
+ { x: 100, y: 100 },
45
+ { x: 10, y: 100 },
46
+ ],
47
+ hotspotColor: '#FF0000',
48
+ selectedHotspotColor: '#00FF00',
49
+ outlineColor: '#0000FF',
50
+ hoverOutlineColor: '#FFFF00',
51
+ selected: false,
52
+ isCorrect: false,
53
+ isEvaluateMode: false,
54
+ disabled: false,
55
+ onClick: jest.fn(),
56
+ strokeWidth: 5,
57
+ scale: 1,
58
+ markAsCorrect: false,
59
+ showCorrectEnabled: false,
60
+ };
61
+ });
62
+
63
+ afterEach(() => {
64
+ document.body.style.cursor = 'default';
65
+ });
66
+
67
+ describe('rendering', () => {
68
+ it('should render without crashing', () => {
69
+ const { container } = render(<PolygonComponent {...defaultProps} />);
70
+ expect(container).toBeTruthy();
71
+ });
72
+
73
+ it('should parse points correctly for Konva', () => {
74
+ const { container } = render(<PolygonComponent {...defaultProps} />);
75
+ const line = container.querySelector('[data-testid="line"]');
76
+
77
+ expect(line).toHaveAttribute('points');
78
+ const points = line.getAttribute('points');
79
+ expect(points).toBe('10,10,100,10,100,100,10,100');
80
+ });
81
+
82
+ it('should render with hotspot color when not selected', () => {
83
+ const { container } = render(<PolygonComponent {...defaultProps} />);
84
+ const line = container.querySelector('[data-testid="line"]');
85
+
86
+ expect(line).toHaveAttribute('fill', '#FF0000');
87
+ });
88
+
89
+ it('should render with selected color when selected', () => {
90
+ const { container } = render(<PolygonComponent {...defaultProps} selected={true} />);
91
+ const line = container.querySelector('[data-testid="line"]');
92
+
93
+ expect(line).toHaveAttribute('fill', '#00FF00');
94
+ });
95
+
96
+ it('should apply scale transform', () => {
97
+ const { getByTestId } = render(<PolygonComponent {...defaultProps} scale={1.5} />);
98
+ const group = getByTestId('group');
99
+
100
+ expect(group).toHaveAttribute('scaleX', '1.5');
101
+ expect(group).toHaveAttribute('scaleY', '1.5');
102
+ });
103
+
104
+ it('should render with default scale of 1', () => {
105
+ const { getByTestId } = render(<PolygonComponent {...defaultProps} />);
106
+ const group = getByTestId('group');
107
+
108
+ expect(group).toHaveAttribute('scaleX', '1');
109
+ expect(group).toHaveAttribute('scaleY', '1');
110
+ });
111
+ });
112
+
113
+ describe('interactions', () => {
114
+ it('should call onClick when clicked', () => {
115
+ const onClick = jest.fn();
116
+ const { container } = render(<PolygonComponent {...defaultProps} onClick={onClick} />);
117
+ const line = container.querySelector('[data-testid="line"]');
118
+
119
+ fireEvent.click(line);
120
+
121
+ expect(onClick).toHaveBeenCalledWith({
122
+ id: 'polygon1',
123
+ selected: true,
124
+ selector: 'Mouse',
125
+ });
126
+ });
127
+
128
+ it('should toggle selection state on click', () => {
129
+ const onClick = jest.fn();
130
+ const { container, rerender } = render(<PolygonComponent {...defaultProps} onClick={onClick} selected={false} />);
131
+ const line = container.querySelector('[data-testid="line"]');
132
+
133
+ fireEvent.click(line);
134
+
135
+ expect(onClick).toHaveBeenCalledWith({
136
+ id: 'polygon1',
137
+ selected: true,
138
+ selector: 'Mouse',
139
+ });
140
+
141
+ rerender(<PolygonComponent {...defaultProps} onClick={onClick} selected={true} />);
142
+
143
+ const lineAfter = container.querySelector('[data-testid="line"]');
144
+ fireEvent.click(lineAfter);
145
+
146
+ expect(onClick).toHaveBeenCalledWith({
147
+ id: 'polygon1',
148
+ selected: false,
149
+ selector: 'Mouse',
150
+ });
151
+ });
152
+
153
+ it('should not call onClick when disabled', () => {
154
+ const onClick = jest.fn();
155
+ const { container } = render(<PolygonComponent {...defaultProps} onClick={onClick} disabled={true} />);
156
+ const line = container.querySelector('[data-testid="line"]');
157
+
158
+ fireEvent.click(line);
159
+
160
+ expect(onClick).not.toHaveBeenCalled();
161
+ });
162
+
163
+ it('should change cursor to pointer on mouse enter when not disabled', () => {
164
+ const { container } = render(<PolygonComponent {...defaultProps} />);
165
+ const line = container.querySelector('[data-testid="line"]');
166
+
167
+ fireEvent.mouseEnter(line);
168
+
169
+ expect(document.body.style.cursor).toBe('pointer');
170
+ });
171
+
172
+ it('should not change cursor when disabled', () => {
173
+ const { container } = render(<PolygonComponent {...defaultProps} disabled={true} />);
174
+ const line = container.querySelector('[data-testid="line"]');
175
+
176
+ fireEvent.mouseEnter(line);
177
+
178
+ expect(document.body.style.cursor).toBe('default');
179
+ });
180
+
181
+ it('should reset cursor to default on mouse leave', () => {
182
+ const { container } = render(<PolygonComponent {...defaultProps} />);
183
+ const line = container.querySelector('[data-testid="line"]');
184
+
185
+ fireEvent.mouseEnter(line);
186
+ fireEvent.mouseLeave(line);
187
+
188
+ expect(document.body.style.cursor).toBe('default');
189
+ });
190
+ });
191
+
192
+ describe('center calculation', () => {
193
+ it('should calculate center correctly for square', () => {
194
+ const { getByTestId } = render(
195
+ <PolygonComponent
196
+ {...defaultProps}
197
+ points={[
198
+ { x: 0, y: 0 },
199
+ { x: 100, y: 0 },
200
+ { x: 100, y: 100 },
201
+ { x: 0, y: 100 },
202
+ ]}
203
+ isEvaluateMode={true}
204
+ selected={true}
205
+ isCorrect={true}
206
+ showCorrectEnabled={false}
207
+ />
208
+ );
209
+
210
+ const icon = getByTestId('icon-image');
211
+ expect(icon).toHaveAttribute('data-x', '50');
212
+ expect(icon).toHaveAttribute('data-y', '50');
213
+ });
214
+
215
+ it('should calculate center correctly for triangle', () => {
216
+ const { getByTestId } = render(
217
+ <PolygonComponent
218
+ {...defaultProps}
219
+ points={[
220
+ { x: 0, y: 0 },
221
+ { x: 100, y: 0 },
222
+ { x: 50, y: 100 },
223
+ ]}
224
+ isEvaluateMode={true}
225
+ selected={true}
226
+ isCorrect={true}
227
+ showCorrectEnabled={false}
228
+ />
229
+ );
230
+
231
+ const icon = getByTestId('icon-image');
232
+ expect(icon).toHaveAttribute('data-x', '50');
233
+ expect(icon).toHaveAttribute('data-y', '50');
234
+ });
235
+
236
+ it('should calculate center correctly for irregular polygon', () => {
237
+ const { getByTestId } = render(
238
+ <PolygonComponent
239
+ {...defaultProps}
240
+ points={[
241
+ { x: 10, y: 20 },
242
+ { x: 90, y: 30 },
243
+ { x: 80, y: 70 },
244
+ { x: 20, y: 60 },
245
+ ]}
246
+ isEvaluateMode={true}
247
+ selected={true}
248
+ isCorrect={true}
249
+ showCorrectEnabled={false}
250
+ />
251
+ );
252
+
253
+ const icon = getByTestId('icon-image');
254
+ expect(icon).toHaveAttribute('data-x', '50');
255
+ expect(icon).toHaveAttribute('data-y', '45');
256
+ });
257
+ });
258
+
259
+ describe('hover styling', () => {
260
+ it('should show hover outline when hoverOutlineColor is provided', () => {
261
+ const { container } = render(<PolygonComponent {...defaultProps} hoverOutlineColor="#FFFF00" />);
262
+ const line = container.querySelector('[data-testid="line"]');
263
+
264
+ fireEvent.mouseEnter(line);
265
+
266
+ const rects = container.querySelectorAll('[data-testid="rect"]');
267
+ expect(rects.length).toBeGreaterThan(0);
268
+ });
269
+
270
+ it('should render hover rect with correct dimensions', () => {
271
+ const { container } = render(
272
+ <PolygonComponent
273
+ {...defaultProps}
274
+ points={[
275
+ { x: 10, y: 20 },
276
+ { x: 110, y: 20 },
277
+ { x: 110, y: 120 },
278
+ { x: 10, y: 120 },
279
+ ]}
280
+ hoverOutlineColor="#FFFF00"
281
+ />
282
+ );
283
+
284
+ const line = container.querySelector('[data-testid="line"]');
285
+ fireEvent.mouseEnter(line);
286
+
287
+ const rect = container.querySelector('[data-testid="rect"]');
288
+ if (rect) {
289
+ expect(rect).toHaveAttribute('x', '10');
290
+ expect(rect).toHaveAttribute('y', '20');
291
+ expect(rect).toHaveAttribute('width', '100');
292
+ expect(rect).toHaveAttribute('height', '100');
293
+ }
294
+ });
295
+ });
296
+
297
+ describe('evaluate mode', () => {
298
+ it('should show correct icon when correctly selected in evaluate mode', () => {
299
+ const { getByTestId } = render(
300
+ <PolygonComponent
301
+ {...defaultProps}
302
+ isEvaluateMode={true}
303
+ selected={true}
304
+ isCorrect={true}
305
+ showCorrectEnabled={false}
306
+ />
307
+ );
308
+
309
+ const icon = getByTestId('icon-image');
310
+ expect(icon).toBeInTheDocument();
311
+ });
312
+
313
+ it('should show wrong icon when incorrectly selected in evaluate mode', () => {
314
+ const { getByTestId } = render(
315
+ <PolygonComponent
316
+ {...defaultProps}
317
+ isEvaluateMode={true}
318
+ selected={true}
319
+ isCorrect={false}
320
+ showCorrectEnabled={false}
321
+ />
322
+ );
323
+
324
+ const icon = getByTestId('icon-image');
325
+ expect(icon).toBeInTheDocument();
326
+ });
327
+
328
+ it('should show wrong icon when incorrectly not selected', () => {
329
+ const { getByTestId } = render(
330
+ <PolygonComponent
331
+ {...defaultProps}
332
+ isEvaluateMode={true}
333
+ selected={false}
334
+ isCorrect={false}
335
+ showCorrectEnabled={false}
336
+ />
337
+ );
338
+
339
+ const icon = getByTestId('icon-image');
340
+ expect(icon).toBeInTheDocument();
341
+ });
342
+
343
+ it('should not show icon when correctly not selected in evaluate mode', () => {
344
+ const { queryByTestId } = render(
345
+ <PolygonComponent
346
+ {...defaultProps}
347
+ isEvaluateMode={true}
348
+ selected={false}
349
+ isCorrect={true}
350
+ showCorrectEnabled={false}
351
+ />
352
+ );
353
+
354
+ const icon = queryByTestId('icon-image');
355
+ expect(icon).not.toBeInTheDocument();
356
+ });
357
+
358
+ it('should show correct icon for showCorrect mode when correctly selected', () => {
359
+ const { getByTestId } = render(
360
+ <PolygonComponent
361
+ {...defaultProps}
362
+ isEvaluateMode={true}
363
+ selected={true}
364
+ isCorrect={true}
365
+ showCorrectEnabled={true}
366
+ />
367
+ );
368
+
369
+ const icon = getByTestId('icon-image');
370
+ expect(icon).toBeInTheDocument();
371
+ });
372
+
373
+ it('should show correct icon for showCorrect mode when incorrectly not selected', () => {
374
+ const { getByTestId } = render(
375
+ <PolygonComponent
376
+ {...defaultProps}
377
+ isEvaluateMode={true}
378
+ selected={false}
379
+ isCorrect={false}
380
+ showCorrectEnabled={true}
381
+ />
382
+ );
383
+
384
+ const icon = getByTestId('icon-image');
385
+ expect(icon).toBeInTheDocument();
386
+ });
387
+
388
+ it('should not show icon in showCorrect mode when correctly not selected', () => {
389
+ const { queryByTestId } = render(
390
+ <PolygonComponent
391
+ {...defaultProps}
392
+ isEvaluateMode={true}
393
+ selected={false}
394
+ isCorrect={true}
395
+ showCorrectEnabled={true}
396
+ />
397
+ );
398
+
399
+ const icon = queryByTestId('icon-image');
400
+ expect(icon).not.toBeInTheDocument();
401
+ });
402
+
403
+ it('should not show icon in showCorrect mode when incorrectly selected', () => {
404
+ const { queryByTestId } = render(
405
+ <PolygonComponent
406
+ {...defaultProps}
407
+ isEvaluateMode={true}
408
+ selected={true}
409
+ isCorrect={false}
410
+ showCorrectEnabled={true}
411
+ />
412
+ );
413
+
414
+ const icon = queryByTestId('icon-image');
415
+ expect(icon).not.toBeInTheDocument();
416
+ });
417
+
418
+ it('should show green outline when markAsCorrect is true', () => {
419
+ const { container } = render(
420
+ <PolygonComponent
421
+ {...defaultProps}
422
+ isEvaluateMode={true}
423
+ markAsCorrect={true}
424
+ />
425
+ );
426
+
427
+ const line = container.querySelector('[data-testid="line"]');
428
+ expect(line).toHaveAttribute('stroke', 'green');
429
+ });
430
+
431
+ it('should show red outline when incorrect and not markAsCorrect', () => {
432
+ const { container } = render(
433
+ <PolygonComponent
434
+ {...defaultProps}
435
+ isEvaluateMode={true}
436
+ isCorrect={false}
437
+ markAsCorrect={false}
438
+ />
439
+ );
440
+
441
+ const line = container.querySelector('[data-testid="line"]');
442
+ expect(line).toHaveAttribute('stroke', 'red');
443
+ });
444
+
445
+ it('should display evaluate text in tooltip', () => {
446
+ const { getByTestId } = render(
447
+ <PolygonComponent
448
+ {...defaultProps}
449
+ isEvaluateMode={true}
450
+ selected={true}
451
+ isCorrect={true}
452
+ evaluateText="Well done!"
453
+ showCorrectEnabled={false}
454
+ />
455
+ );
456
+
457
+ const icon = getByTestId('icon-image');
458
+ expect(icon).toHaveAttribute('data-tooltip', 'Well done!');
459
+ });
460
+ });
461
+
462
+ describe('edge cases', () => {
463
+ it('should handle triangular polygon', () => {
464
+ const { container } = render(
465
+ <PolygonComponent
466
+ {...defaultProps}
467
+ points={[
468
+ { x: 50, y: 0 },
469
+ { x: 100, y: 100 },
470
+ { x: 0, y: 100 },
471
+ ]}
472
+ />
473
+ );
474
+
475
+ const line = container.querySelector('[data-testid="line"]');
476
+ expect(line).toHaveAttribute('points', '50,0,100,100,0,100');
477
+ });
478
+
479
+ it('should handle complex polygon with many points', () => {
480
+ const points = [
481
+ { x: 10, y: 10 },
482
+ { x: 50, y: 5 },
483
+ { x: 90, y: 10 },
484
+ { x: 100, y: 50 },
485
+ { x: 90, y: 90 },
486
+ { x: 50, y: 100 },
487
+ { x: 10, y: 90 },
488
+ { x: 0, y: 50 },
489
+ ];
490
+
491
+ const { container } = render(
492
+ <PolygonComponent
493
+ {...defaultProps}
494
+ points={points}
495
+ />
496
+ );
497
+
498
+ const line = container.querySelector('[data-testid="line"]');
499
+ expect(line).toHaveAttribute('points', '10,10,50,5,90,10,100,50,90,90,50,100,10,90,0,50');
500
+ });
501
+ });
502
+ });