@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.
- package/CHANGELOG.json +997 -0
- package/CHANGELOG.md +2226 -0
- package/LICENSE.md +5 -0
- package/README.md +1 -0
- package/configure/CHANGELOG.json +682 -0
- package/configure/CHANGELOG.md +1963 -0
- package/configure/lib/DeleteWidget.js +64 -0
- package/configure/lib/DeleteWidget.js.map +1 -0
- package/configure/lib/button.js +42 -0
- package/configure/lib/button.js.map +1 -0
- package/configure/lib/buttons/circle.js +33 -0
- package/configure/lib/buttons/circle.js.map +1 -0
- package/configure/lib/buttons/polygon.js +39 -0
- package/configure/lib/buttons/polygon.js.map +1 -0
- package/configure/lib/buttons/rectangle.js +39 -0
- package/configure/lib/buttons/rectangle.js.map +1 -0
- package/configure/lib/defaults.js +155 -0
- package/configure/lib/defaults.js.map +1 -0
- package/configure/lib/hotspot-circle.js +192 -0
- package/configure/lib/hotspot-circle.js.map +1 -0
- package/configure/lib/hotspot-container.js +320 -0
- package/configure/lib/hotspot-container.js.map +1 -0
- package/configure/lib/hotspot-drawable.js +519 -0
- package/configure/lib/hotspot-drawable.js.map +1 -0
- package/configure/lib/hotspot-palette.js +107 -0
- package/configure/lib/hotspot-palette.js.map +1 -0
- package/configure/lib/hotspot-polygon.js +293 -0
- package/configure/lib/hotspot-polygon.js.map +1 -0
- package/configure/lib/hotspot-rectangle.js +190 -0
- package/configure/lib/hotspot-rectangle.js.map +1 -0
- package/configure/lib/icons.js +7 -0
- package/configure/lib/icons.js.map +1 -0
- package/configure/lib/image-konva.js +66 -0
- package/configure/lib/image-konva.js.map +1 -0
- package/configure/lib/index.js +194 -0
- package/configure/lib/index.js.map +1 -0
- package/configure/lib/root.js +330 -0
- package/configure/lib/root.js.map +1 -0
- package/configure/lib/shapes/circle.js +84 -0
- package/configure/lib/shapes/circle.js.map +1 -0
- package/configure/lib/shapes/index.js +50 -0
- package/configure/lib/shapes/index.js.map +1 -0
- package/configure/lib/shapes/polygon.js +82 -0
- package/configure/lib/shapes/polygon.js.map +1 -0
- package/configure/lib/shapes/rectagle.js +84 -0
- package/configure/lib/shapes/rectagle.js.map +1 -0
- package/configure/lib/shapes/utils.js +21 -0
- package/configure/lib/shapes/utils.js.map +1 -0
- package/configure/lib/upload-control.js +41 -0
- package/configure/lib/upload-control.js.map +1 -0
- package/configure/lib/utils.js +185 -0
- package/configure/lib/utils.js.map +1 -0
- package/configure/package.json +26 -0
- package/configure/src/DeleteWidget.jsx +51 -0
- package/configure/src/__tests__/DeleteWidget.test.jsx +366 -0
- package/configure/src/__tests__/button.test.jsx +198 -0
- package/configure/src/__tests__/hotspot-circle.test.jsx +259 -0
- package/configure/src/__tests__/hotspot-container.test.js +366 -0
- package/configure/src/__tests__/hotspot-drawable.test.js +271 -0
- package/configure/src/__tests__/hotspot-palette.test.jsx +71 -0
- package/configure/src/__tests__/image-konva.test.jsx +226 -0
- package/configure/src/__tests__/index.test.js +329 -0
- package/configure/src/__tests__/root.test.js +400 -0
- package/configure/src/__tests__/utils.test.js +241 -0
- package/configure/src/button.jsx +35 -0
- package/configure/src/buttons/circle.jsx +18 -0
- package/configure/src/buttons/polygon.jsx +29 -0
- package/configure/src/buttons/rectangle.jsx +29 -0
- package/configure/src/defaults.js +109 -0
- package/configure/src/hotspot-circle.jsx +183 -0
- package/configure/src/hotspot-container.jsx +330 -0
- package/configure/src/hotspot-drawable.jsx +527 -0
- package/configure/src/hotspot-palette.jsx +90 -0
- package/configure/src/hotspot-polygon.jsx +294 -0
- package/configure/src/hotspot-rectangle.jsx +169 -0
- package/configure/src/icons.js +5 -0
- package/configure/src/image-konva.jsx +63 -0
- package/configure/src/index.js +208 -0
- package/configure/src/root.jsx +346 -0
- package/configure/src/shapes/circle.js +81 -0
- package/configure/src/shapes/index.js +4 -0
- package/configure/src/shapes/polygon.js +81 -0
- package/configure/src/shapes/rectagle.js +82 -0
- package/configure/src/shapes/utils.js +16 -0
- package/configure/src/upload-control.jsx +33 -0
- package/configure/src/utils.js +210 -0
- package/controller/CHANGELOG.json +362 -0
- package/controller/CHANGELOG.md +1304 -0
- package/controller/lib/defaults.js +33 -0
- package/controller/lib/defaults.js.map +1 -0
- package/controller/lib/index.js +341 -0
- package/controller/lib/index.js.map +1 -0
- package/controller/lib/utils.js +32 -0
- package/controller/lib/utils.js.map +1 -0
- package/controller/package.json +18 -0
- package/controller/src/__tests__/index.test.js +419 -0
- package/controller/src/__tests__/utils.test.js +5 -0
- package/controller/src/defaults.js +19 -0
- package/controller/src/index.js +328 -0
- package/controller/src/utils.js +29 -0
- package/docs/config-schema.json +2023 -0
- package/docs/config-schema.json.md +1495 -0
- package/docs/demo/config.js +8 -0
- package/docs/demo/generate.js +118 -0
- package/docs/demo/index.html +1 -0
- package/docs/demo/session.js +11 -0
- package/docs/pie-schema.json +1204 -0
- package/docs/pie-schema.json.md +851 -0
- package/lib/hotspot/circle.js +156 -0
- package/lib/hotspot/circle.js.map +1 -0
- package/lib/hotspot/container.js +206 -0
- package/lib/hotspot/container.js.map +1 -0
- package/lib/hotspot/icons.js +8 -0
- package/lib/hotspot/icons.js.map +1 -0
- package/lib/hotspot/image-konva-tooltip.js +86 -0
- package/lib/hotspot/image-konva-tooltip.js.map +1 -0
- package/lib/hotspot/index.js +163 -0
- package/lib/hotspot/index.js.map +1 -0
- package/lib/hotspot/polygon.js +203 -0
- package/lib/hotspot/polygon.js.map +1 -0
- package/lib/hotspot/rectangle.js +175 -0
- package/lib/hotspot/rectangle.js.map +1 -0
- package/lib/index.js +213 -0
- package/lib/index.js.map +1 -0
- package/lib/session-updater.js +42 -0
- package/lib/session-updater.js.map +1 -0
- package/package.json +18 -83
- package/src/__tests__/container.test.jsx +58 -0
- package/src/__tests__/index.test.js +123 -0
- package/src/__tests__/session-updater.test.jsx +69 -0
- package/src/hotspot/__tests__/circle.test.jsx +464 -0
- package/src/hotspot/__tests__/container.test.jsx +546 -0
- package/src/hotspot/__tests__/image-konva-tooltip.test.jsx +510 -0
- package/src/hotspot/__tests__/polygon.test.jsx +502 -0
- package/src/hotspot/__tests__/rectangle.test.jsx +418 -0
- package/src/hotspot/circle.jsx +152 -0
- package/src/hotspot/container.jsx +217 -0
- package/src/hotspot/icons.js +7 -0
- package/src/hotspot/image-konva-tooltip.jsx +76 -0
- package/src/hotspot/index.jsx +165 -0
- package/src/hotspot/polygon.jsx +195 -0
- package/src/hotspot/rectangle.jsx +171 -0
- package/src/index.js +226 -0
- package/src/session-updater.js +29 -0
- package/configure.js +0 -2
- package/controller.js +0 -1
- package/dist/author/DeleteWidget.d.ts +0 -38
- package/dist/author/DeleteWidget.js +0 -46
- package/dist/author/button.d.ts +0 -31
- package/dist/author/button.js +0 -27
- package/dist/author/buttons/circle.d.ts +0 -18
- package/dist/author/buttons/circle.js +0 -25
- package/dist/author/buttons/polygon.d.ts +0 -18
- package/dist/author/buttons/polygon.js +0 -36
- package/dist/author/buttons/rectangle.d.ts +0 -18
- package/dist/author/buttons/rectangle.js +0 -36
- package/dist/author/defaults.d.ts +0 -157
- package/dist/author/defaults.js +0 -119
- package/dist/author/hotspot-circle.d.ts +0 -21
- package/dist/author/hotspot-circle.js +0 -124
- package/dist/author/hotspot-container.d.ts +0 -29
- package/dist/author/hotspot-container.js +0 -210
- package/dist/author/hotspot-drawable.d.ts +0 -31
- package/dist/author/hotspot-drawable.js +0 -312
- package/dist/author/hotspot-palette.d.ts +0 -14
- package/dist/author/hotspot-palette.js +0 -72
- package/dist/author/hotspot-polygon.d.ts +0 -38
- package/dist/author/hotspot-polygon.js +0 -200
- package/dist/author/hotspot-rectangle.d.ts +0 -20
- package/dist/author/hotspot-rectangle.js +0 -119
- package/dist/author/icons.d.ts +0 -9
- package/dist/author/icons.js +0 -4
- package/dist/author/image-konva.d.ts +0 -19
- package/dist/author/image-konva.js +0 -49
- package/dist/author/index.d.ts +0 -52
- package/dist/author/index.js +0 -143
- package/dist/author/root.d.ts +0 -15
- package/dist/author/root.js +0 -215
- package/dist/author/shapes/circle.d.ts +0 -18
- package/dist/author/shapes/circle.js +0 -47
- package/dist/author/shapes/index.d.ts +0 -12
- package/dist/author/shapes/polygon.d.ts +0 -19
- package/dist/author/shapes/polygon.js +0 -51
- package/dist/author/shapes/rectagle.d.ts +0 -18
- package/dist/author/shapes/rectagle.js +0 -57
- package/dist/author/shapes/utils.d.ts +0 -19
- package/dist/author/shapes/utils.js +0 -16
- package/dist/author/upload-control.d.ts +0 -29
- package/dist/author/upload-control.js +0 -28
- package/dist/author/utils.d.ts +0 -24
- package/dist/author/utils.js +0 -83
- package/dist/browser/ReactKonva-CFo7dxdy.js +0 -19336
- package/dist/browser/ReactKonva-CFo7dxdy.js.map +0 -1
- package/dist/browser/author/index.js +0 -47549
- package/dist/browser/author/index.js.map +0 -1
- package/dist/browser/browser-CfnAFove.js +0 -219
- package/dist/browser/browser-CfnAFove.js.map +0 -1
- package/dist/browser/controller/index.js +0 -198
- package/dist/browser/controller/index.js.map +0 -1
- package/dist/browser/delivery/index.js +0 -2460
- package/dist/browser/delivery/index.js.map +0 -1
- package/dist/browser/dist-CeB-1djc.js +0 -100
- package/dist/browser/dist-CeB-1djc.js.map +0 -1
- package/dist/browser/hotspot.css +0 -2
- package/dist/controller/defaults.d.ts +0 -35
- package/dist/controller/defaults.js +0 -29
- package/dist/controller/index.d.ts +0 -22
- package/dist/controller/index.js +0 -154
- package/dist/controller/utils.d.ts +0 -10
- package/dist/controller/utils.js +0 -12
- package/dist/delivery/hotspot/circle.d.ts +0 -19
- package/dist/delivery/hotspot/circle.js +0 -100
- package/dist/delivery/hotspot/container.d.ts +0 -16
- package/dist/delivery/hotspot/container.js +0 -150
- package/dist/delivery/hotspot/icons.d.ts +0 -10
- package/dist/delivery/hotspot/icons.js +0 -4
- package/dist/delivery/hotspot/image-konva-tooltip.d.ts +0 -19
- package/dist/delivery/hotspot/image-konva-tooltip.js +0 -66
- package/dist/delivery/hotspot/index.d.ts +0 -17
- package/dist/delivery/hotspot/index.js +0 -114
- package/dist/delivery/hotspot/polygon.d.ts +0 -21
- package/dist/delivery/hotspot/polygon.js +0 -108
- package/dist/delivery/hotspot/rectangle.d.ts +0 -19
- package/dist/delivery/hotspot/rectangle.js +0 -104
- package/dist/delivery/index.d.ts +0 -20
- package/dist/delivery/index.js +0 -107
- package/dist/delivery/session-updater.d.ts +0 -10
- package/dist/delivery/session-updater.js +0 -14
- package/dist/index.d.ts +0 -1
- package/dist/index.iife.d.ts +0 -8
- package/dist/index.iife.js +0 -169
- package/dist/index.js +0 -2
- package/dist/runtime-support.d.ts +0 -12
- package/dist/runtime-support.js +0 -12
|
@@ -0,0 +1,510 @@
|
|
|
1
|
+
import React from 'react';
|
|
2
|
+
import { render, waitFor, fireEvent } from '@testing-library/react';
|
|
3
|
+
import Konva from 'konva';
|
|
4
|
+
import ImageComponent from '../image-konva-tooltip';
|
|
5
|
+
|
|
6
|
+
Konva.isBrowser = false;
|
|
7
|
+
|
|
8
|
+
jest.mock('react-konva', () => {
|
|
9
|
+
const React = require('react');
|
|
10
|
+
return {
|
|
11
|
+
Group: ({ children, ...props }) => React.createElement('div', { 'data-testid': 'group', ...props }, children),
|
|
12
|
+
Image: ({ onMouseEnter, onMouseLeave, ...props }) =>
|
|
13
|
+
React.createElement('div', {
|
|
14
|
+
'data-testid': 'image',
|
|
15
|
+
onMouseEnter,
|
|
16
|
+
onMouseLeave,
|
|
17
|
+
...props
|
|
18
|
+
}),
|
|
19
|
+
Text: (props) => React.createElement('div', { 'data-testid': 'text', ...props }),
|
|
20
|
+
Tag: (props) => React.createElement('div', { 'data-testid': 'tag', ...props }),
|
|
21
|
+
Label: ({ children, ...props }) => React.createElement('div', { 'data-testid': 'label', ...props }, children),
|
|
22
|
+
};
|
|
23
|
+
});
|
|
24
|
+
|
|
25
|
+
class MockImage {
|
|
26
|
+
constructor() {
|
|
27
|
+
this.src = '';
|
|
28
|
+
this._onload = null;
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
addEventListener(event, callback) {
|
|
32
|
+
if (event === 'load') {
|
|
33
|
+
this._onload = callback;
|
|
34
|
+
setTimeout(() => {
|
|
35
|
+
if (this._onload) {
|
|
36
|
+
this._onload();
|
|
37
|
+
}
|
|
38
|
+
}, 10);
|
|
39
|
+
}
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
removeEventListener() {
|
|
43
|
+
this._onload = null;
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
global.Image = MockImage;
|
|
48
|
+
|
|
49
|
+
describe('ImageComponent', () => {
|
|
50
|
+
let defaultProps;
|
|
51
|
+
|
|
52
|
+
beforeEach(() => {
|
|
53
|
+
defaultProps = {
|
|
54
|
+
src: 'test-image.png',
|
|
55
|
+
x: 100,
|
|
56
|
+
y: 150,
|
|
57
|
+
tooltip: 'Test tooltip',
|
|
58
|
+
};
|
|
59
|
+
});
|
|
60
|
+
|
|
61
|
+
describe('rendering', () => {
|
|
62
|
+
it('should render without crashing', () => {
|
|
63
|
+
const { container } = render(<ImageComponent {...defaultProps} />);
|
|
64
|
+
expect(container).toBeTruthy();
|
|
65
|
+
});
|
|
66
|
+
|
|
67
|
+
it('should render Group component', () => {
|
|
68
|
+
const { getByTestId } = render(<ImageComponent {...defaultProps} />);
|
|
69
|
+
const group = getByTestId('group');
|
|
70
|
+
expect(group).toBeInTheDocument();
|
|
71
|
+
});
|
|
72
|
+
|
|
73
|
+
it('should render Image component', () => {
|
|
74
|
+
const { getByTestId } = render(<ImageComponent {...defaultProps} />);
|
|
75
|
+
const image = getByTestId('image');
|
|
76
|
+
expect(image).toBeInTheDocument();
|
|
77
|
+
});
|
|
78
|
+
|
|
79
|
+
it('should have correct image dimensions', () => {
|
|
80
|
+
const { getByTestId } = render(<ImageComponent {...defaultProps} />);
|
|
81
|
+
const image = getByTestId('image');
|
|
82
|
+
expect(image).toHaveAttribute('width', '20');
|
|
83
|
+
expect(image).toHaveAttribute('height', '20');
|
|
84
|
+
});
|
|
85
|
+
|
|
86
|
+
it('should have correct position', () => {
|
|
87
|
+
const { getByTestId } = render(<ImageComponent {...defaultProps} />);
|
|
88
|
+
const image = getByTestId('image');
|
|
89
|
+
expect(image).toHaveAttribute('x', '100');
|
|
90
|
+
expect(image).toHaveAttribute('y', '150');
|
|
91
|
+
});
|
|
92
|
+
});
|
|
93
|
+
|
|
94
|
+
describe('image loading', () => {
|
|
95
|
+
it('should load image on mount', async () => {
|
|
96
|
+
const { getByTestId } = render(<ImageComponent {...defaultProps} />);
|
|
97
|
+
const image = getByTestId('image');
|
|
98
|
+
|
|
99
|
+
await waitFor(() => {
|
|
100
|
+
expect(image).toHaveAttribute('image');
|
|
101
|
+
}, { timeout: 100 });
|
|
102
|
+
});
|
|
103
|
+
|
|
104
|
+
it('should reload image when src prop changes', async () => {
|
|
105
|
+
const { getByTestId, rerender } = render(<ImageComponent {...defaultProps} />);
|
|
106
|
+
|
|
107
|
+
await waitFor(() => {
|
|
108
|
+
const image = getByTestId('image');
|
|
109
|
+
expect(image).toBeInTheDocument();
|
|
110
|
+
});
|
|
111
|
+
|
|
112
|
+
rerender(<ImageComponent {...defaultProps} src="new-image.png" />);
|
|
113
|
+
|
|
114
|
+
await waitFor(() => {
|
|
115
|
+
const image = getByTestId('image');
|
|
116
|
+
expect(image).toBeInTheDocument();
|
|
117
|
+
});
|
|
118
|
+
});
|
|
119
|
+
|
|
120
|
+
it('should set image state after load', async () => {
|
|
121
|
+
const { getByTestId } = render(<ImageComponent {...defaultProps} />);
|
|
122
|
+
|
|
123
|
+
await waitFor(() => {
|
|
124
|
+
const image = getByTestId('image');
|
|
125
|
+
expect(image).toBeInTheDocument();
|
|
126
|
+
}, { timeout: 100 });
|
|
127
|
+
});
|
|
128
|
+
});
|
|
129
|
+
|
|
130
|
+
describe('tooltip functionality', () => {
|
|
131
|
+
it('should not show tooltip initially', () => {
|
|
132
|
+
const { queryByTestId } = render(<ImageComponent {...defaultProps} />);
|
|
133
|
+
const label = queryByTestId('label');
|
|
134
|
+
expect(label).not.toBeInTheDocument();
|
|
135
|
+
});
|
|
136
|
+
|
|
137
|
+
it('should show tooltip on mouse enter', async () => {
|
|
138
|
+
const { getByTestId, queryByTestId } = render(<ImageComponent {...defaultProps} />);
|
|
139
|
+
const image = getByTestId('image');
|
|
140
|
+
|
|
141
|
+
// Initially no tooltip
|
|
142
|
+
expect(queryByTestId('label')).not.toBeInTheDocument();
|
|
143
|
+
|
|
144
|
+
// Simulate mouse enter
|
|
145
|
+
fireEvent.mouseEnter(image);
|
|
146
|
+
|
|
147
|
+
// Wait for state update
|
|
148
|
+
await waitFor(() => {
|
|
149
|
+
const label = queryByTestId('label');
|
|
150
|
+
expect(label).toBeInTheDocument();
|
|
151
|
+
});
|
|
152
|
+
});
|
|
153
|
+
|
|
154
|
+
it('should hide tooltip on mouse leave', async () => {
|
|
155
|
+
const { getByTestId, queryByTestId } = render(<ImageComponent {...defaultProps} />);
|
|
156
|
+
const image = getByTestId('image');
|
|
157
|
+
|
|
158
|
+
// Show tooltip
|
|
159
|
+
fireEvent.mouseEnter(image);
|
|
160
|
+
|
|
161
|
+
await waitFor(() => {
|
|
162
|
+
expect(queryByTestId('label')).toBeInTheDocument();
|
|
163
|
+
});
|
|
164
|
+
|
|
165
|
+
// Hide tooltip
|
|
166
|
+
fireEvent.mouseLeave(image);
|
|
167
|
+
|
|
168
|
+
await waitFor(() => {
|
|
169
|
+
expect(queryByTestId('label')).not.toBeInTheDocument();
|
|
170
|
+
});
|
|
171
|
+
});
|
|
172
|
+
|
|
173
|
+
it('should render tooltip with correct text', async () => {
|
|
174
|
+
const { getByTestId } = render(
|
|
175
|
+
<ImageComponent {...defaultProps} tooltip="Custom tooltip" />
|
|
176
|
+
);
|
|
177
|
+
const image = getByTestId('image');
|
|
178
|
+
|
|
179
|
+
fireEvent.mouseEnter(image);
|
|
180
|
+
|
|
181
|
+
await waitFor(() => {
|
|
182
|
+
const text = getByTestId('text');
|
|
183
|
+
expect(text).toHaveAttribute('text', 'Custom tooltip');
|
|
184
|
+
});
|
|
185
|
+
});
|
|
186
|
+
|
|
187
|
+
it('should not show tooltip when tooltip prop is empty', async () => {
|
|
188
|
+
const { getByTestId, queryByTestId } = render(
|
|
189
|
+
<ImageComponent {...defaultProps} tooltip="" />
|
|
190
|
+
);
|
|
191
|
+
const image = getByTestId('image');
|
|
192
|
+
|
|
193
|
+
fireEvent.mouseEnter(image);
|
|
194
|
+
|
|
195
|
+
// Even after mouse enter, tooltip should not show if text is empty
|
|
196
|
+
await new Promise(resolve => setTimeout(resolve, 50));
|
|
197
|
+
expect(queryByTestId('label')).not.toBeInTheDocument();
|
|
198
|
+
});
|
|
199
|
+
|
|
200
|
+
it('should handle long tooltip text', async () => {
|
|
201
|
+
const longTooltip = 'This is a very long tooltip text that should still render correctly without breaking the component layout';
|
|
202
|
+
const { getByTestId } = render(
|
|
203
|
+
<ImageComponent {...defaultProps} tooltip={longTooltip} />
|
|
204
|
+
);
|
|
205
|
+
const image = getByTestId('image');
|
|
206
|
+
|
|
207
|
+
fireEvent.mouseEnter(image);
|
|
208
|
+
|
|
209
|
+
await waitFor(() => {
|
|
210
|
+
const text = getByTestId('text');
|
|
211
|
+
expect(text).toHaveAttribute('text', longTooltip);
|
|
212
|
+
});
|
|
213
|
+
});
|
|
214
|
+
});
|
|
215
|
+
|
|
216
|
+
describe('positioning', () => {
|
|
217
|
+
it('should render at correct x position', () => {
|
|
218
|
+
const { getByTestId } = render(
|
|
219
|
+
<ImageComponent {...defaultProps} x={200} />
|
|
220
|
+
);
|
|
221
|
+
const image = getByTestId('image');
|
|
222
|
+
expect(image).toHaveAttribute('x', '200');
|
|
223
|
+
});
|
|
224
|
+
|
|
225
|
+
it('should render at correct y position', () => {
|
|
226
|
+
const { getByTestId } = render(
|
|
227
|
+
<ImageComponent {...defaultProps} y={300} />
|
|
228
|
+
);
|
|
229
|
+
const image = getByTestId('image');
|
|
230
|
+
expect(image).toHaveAttribute('y', '300');
|
|
231
|
+
});
|
|
232
|
+
|
|
233
|
+
it('should handle position at origin (0, 0)', () => {
|
|
234
|
+
const { getByTestId } = render(
|
|
235
|
+
<ImageComponent {...defaultProps} x={0} y={0} />
|
|
236
|
+
);
|
|
237
|
+
const image = getByTestId('image');
|
|
238
|
+
expect(image).toHaveAttribute('x', '0');
|
|
239
|
+
expect(image).toHaveAttribute('y', '0');
|
|
240
|
+
});
|
|
241
|
+
|
|
242
|
+
it('should handle negative positions', () => {
|
|
243
|
+
const { getByTestId } = render(
|
|
244
|
+
<ImageComponent {...defaultProps} x={-10} y={-20} />
|
|
245
|
+
);
|
|
246
|
+
const image = getByTestId('image');
|
|
247
|
+
expect(image).toHaveAttribute('x', '-10');
|
|
248
|
+
expect(image).toHaveAttribute('y', '-20');
|
|
249
|
+
});
|
|
250
|
+
|
|
251
|
+
it('should handle large position values', () => {
|
|
252
|
+
const { getByTestId } = render(
|
|
253
|
+
<ImageComponent {...defaultProps} x={1000} y={2000} />
|
|
254
|
+
);
|
|
255
|
+
const image = getByTestId('image');
|
|
256
|
+
expect(image).toHaveAttribute('x', '1000');
|
|
257
|
+
expect(image).toHaveAttribute('y', '2000');
|
|
258
|
+
});
|
|
259
|
+
|
|
260
|
+
it('should position tooltip relative to image', async () => {
|
|
261
|
+
const { getByTestId } = render(<ImageComponent {...defaultProps} x={100} y={150} />);
|
|
262
|
+
const image = getByTestId('image');
|
|
263
|
+
|
|
264
|
+
fireEvent.mouseEnter(image);
|
|
265
|
+
|
|
266
|
+
await waitFor(() => {
|
|
267
|
+
const label = getByTestId('label');
|
|
268
|
+
// Tooltip should be positioned at x - 30, y + 25
|
|
269
|
+
expect(label).toHaveAttribute('x', '70');
|
|
270
|
+
expect(label).toHaveAttribute('y', '175');
|
|
271
|
+
});
|
|
272
|
+
});
|
|
273
|
+
});
|
|
274
|
+
|
|
275
|
+
describe('image sources', () => {
|
|
276
|
+
it('should handle data URI image source', async () => {
|
|
277
|
+
const dataUri = 'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAYAAAAfFcSJAAAADUlEQVR42mNk+M9QDwADhgGAWjR9awAAAABJRU5ErkJggg==';
|
|
278
|
+
const { getByTestId } = render(
|
|
279
|
+
<ImageComponent {...defaultProps} src={dataUri} />
|
|
280
|
+
);
|
|
281
|
+
const image = getByTestId('image');
|
|
282
|
+
expect(image).toBeInTheDocument();
|
|
283
|
+
});
|
|
284
|
+
|
|
285
|
+
it('should handle relative path image source', () => {
|
|
286
|
+
const { getByTestId } = render(
|
|
287
|
+
<ImageComponent {...defaultProps} src="./images/icon.png" />
|
|
288
|
+
);
|
|
289
|
+
const image = getByTestId('image');
|
|
290
|
+
expect(image).toBeInTheDocument();
|
|
291
|
+
});
|
|
292
|
+
|
|
293
|
+
it('should handle absolute URL image source', () => {
|
|
294
|
+
const { getByTestId } = render(
|
|
295
|
+
<ImageComponent {...defaultProps} src="https://example.com/image.png" />
|
|
296
|
+
);
|
|
297
|
+
const image = getByTestId('image');
|
|
298
|
+
expect(image).toBeInTheDocument();
|
|
299
|
+
});
|
|
300
|
+
|
|
301
|
+
it('should handle SVG image source', () => {
|
|
302
|
+
const { getByTestId } = render(
|
|
303
|
+
<ImageComponent {...defaultProps} src="icon.svg" />
|
|
304
|
+
);
|
|
305
|
+
const image = getByTestId('image');
|
|
306
|
+
expect(image).toBeInTheDocument();
|
|
307
|
+
});
|
|
308
|
+
});
|
|
309
|
+
|
|
310
|
+
describe('component lifecycle', () => {
|
|
311
|
+
it('should clean up event listener on unmount', () => {
|
|
312
|
+
const { unmount } = render(<ImageComponent {...defaultProps} />);
|
|
313
|
+
|
|
314
|
+
expect(() => {
|
|
315
|
+
unmount();
|
|
316
|
+
}).not.toThrow();
|
|
317
|
+
});
|
|
318
|
+
|
|
319
|
+
it('should handle multiple mount/unmount cycles', () => {
|
|
320
|
+
const { unmount } = render(<ImageComponent {...defaultProps} />);
|
|
321
|
+
|
|
322
|
+
unmount();
|
|
323
|
+
|
|
324
|
+
expect(() => {
|
|
325
|
+
render(<ImageComponent {...defaultProps} />);
|
|
326
|
+
}).not.toThrow();
|
|
327
|
+
});
|
|
328
|
+
|
|
329
|
+
it('should handle rapid prop changes', async () => {
|
|
330
|
+
const { rerender, container } = render(<ImageComponent {...defaultProps} />);
|
|
331
|
+
|
|
332
|
+
rerender(<ImageComponent {...defaultProps} x={150} y={200} />);
|
|
333
|
+
|
|
334
|
+
await new Promise(resolve => setTimeout(resolve, 20));
|
|
335
|
+
|
|
336
|
+
rerender(<ImageComponent {...defaultProps} x={200} y={250} />);
|
|
337
|
+
|
|
338
|
+
const image = container.querySelector('[data-testid="image"]');
|
|
339
|
+
expect(image).toHaveAttribute('x', '200');
|
|
340
|
+
expect(image).toHaveAttribute('y', '250');
|
|
341
|
+
});
|
|
342
|
+
|
|
343
|
+
it('should handle src changes during image load', async () => {
|
|
344
|
+
const { rerender, container } = render(<ImageComponent {...defaultProps} src="image1.png" />);
|
|
345
|
+
|
|
346
|
+
// Change src before image loads
|
|
347
|
+
rerender(<ImageComponent {...defaultProps} src="image2.png" />);
|
|
348
|
+
|
|
349
|
+
await waitFor(() => {
|
|
350
|
+
const image = container.querySelector('[data-testid="image"]');
|
|
351
|
+
expect(image).toBeInTheDocument();
|
|
352
|
+
});
|
|
353
|
+
});
|
|
354
|
+
});
|
|
355
|
+
|
|
356
|
+
describe('edge cases', () => {
|
|
357
|
+
it('should handle zero dimensions gracefully', () => {
|
|
358
|
+
const { getByTestId } = render(
|
|
359
|
+
<ImageComponent {...defaultProps} x={0} y={0} />
|
|
360
|
+
);
|
|
361
|
+
const image = getByTestId('image');
|
|
362
|
+
expect(image).toHaveAttribute('x', '0');
|
|
363
|
+
expect(image).toHaveAttribute('y', '0');
|
|
364
|
+
});
|
|
365
|
+
|
|
366
|
+
it('should render with special characters in tooltip', async () => {
|
|
367
|
+
const specialTooltip = 'Test <>&"\'tooltip';
|
|
368
|
+
const { getByTestId } = render(
|
|
369
|
+
<ImageComponent {...defaultProps} tooltip={specialTooltip} />
|
|
370
|
+
);
|
|
371
|
+
const image = getByTestId('image');
|
|
372
|
+
|
|
373
|
+
fireEvent.mouseEnter(image);
|
|
374
|
+
|
|
375
|
+
await waitFor(() => {
|
|
376
|
+
const text = getByTestId('text');
|
|
377
|
+
expect(text).toHaveAttribute('text', specialTooltip);
|
|
378
|
+
});
|
|
379
|
+
});
|
|
380
|
+
|
|
381
|
+
it('should handle Unicode characters in tooltip', async () => {
|
|
382
|
+
const unicodeTooltip = 'Test 你好 🎉 tooltip';
|
|
383
|
+
const { getByTestId } = render(
|
|
384
|
+
<ImageComponent {...defaultProps} tooltip={unicodeTooltip} />
|
|
385
|
+
);
|
|
386
|
+
const image = getByTestId('image');
|
|
387
|
+
|
|
388
|
+
fireEvent.mouseEnter(image);
|
|
389
|
+
|
|
390
|
+
await waitFor(() => {
|
|
391
|
+
const text = getByTestId('text');
|
|
392
|
+
expect(text).toHaveAttribute('text', unicodeTooltip);
|
|
393
|
+
});
|
|
394
|
+
});
|
|
395
|
+
|
|
396
|
+
it('should handle missing tooltip gracefully', () => {
|
|
397
|
+
const { getByTestId } = render(
|
|
398
|
+
<ImageComponent src="test.png" x={100} y={150} />
|
|
399
|
+
);
|
|
400
|
+
const image = getByTestId('image');
|
|
401
|
+
expect(image).toBeInTheDocument();
|
|
402
|
+
});
|
|
403
|
+
});
|
|
404
|
+
|
|
405
|
+
describe('tooltip styling', () => {
|
|
406
|
+
it('should render tooltip with Tag component', async () => {
|
|
407
|
+
const { getByTestId } = render(<ImageComponent {...defaultProps} />);
|
|
408
|
+
const image = getByTestId('image');
|
|
409
|
+
|
|
410
|
+
fireEvent.mouseEnter(image);
|
|
411
|
+
|
|
412
|
+
await waitFor(() => {
|
|
413
|
+
const tag = getByTestId('tag');
|
|
414
|
+
expect(tag).toBeInTheDocument();
|
|
415
|
+
});
|
|
416
|
+
});
|
|
417
|
+
|
|
418
|
+
it('should render tooltip with correct styling attributes', async () => {
|
|
419
|
+
const { getByTestId } = render(<ImageComponent {...defaultProps} />);
|
|
420
|
+
const image = getByTestId('image');
|
|
421
|
+
|
|
422
|
+
fireEvent.mouseEnter(image);
|
|
423
|
+
|
|
424
|
+
await waitFor(() => {
|
|
425
|
+
const tag = getByTestId('tag');
|
|
426
|
+
expect(tag).toHaveAttribute('fill', 'white');
|
|
427
|
+
expect(tag).toHaveAttribute('cornerRadius', '5');
|
|
428
|
+
expect(tag).toHaveAttribute('opacity', '0.9');
|
|
429
|
+
});
|
|
430
|
+
});
|
|
431
|
+
|
|
432
|
+
it('should render tooltip text with padding', async () => {
|
|
433
|
+
const { getByTestId } = render(<ImageComponent {...defaultProps} />);
|
|
434
|
+
const image = getByTestId('image');
|
|
435
|
+
|
|
436
|
+
fireEvent.mouseEnter(image);
|
|
437
|
+
|
|
438
|
+
await waitFor(() => {
|
|
439
|
+
const text = getByTestId('text');
|
|
440
|
+
expect(text).toHaveAttribute('padding', '5');
|
|
441
|
+
});
|
|
442
|
+
});
|
|
443
|
+
});
|
|
444
|
+
|
|
445
|
+
describe('state management', () => {
|
|
446
|
+
it('should update showTooltip state on mouse events', async () => {
|
|
447
|
+
const { getByTestId, queryByTestId } = render(<ImageComponent {...defaultProps} />);
|
|
448
|
+
const image = getByTestId('image');
|
|
449
|
+
|
|
450
|
+
// Initially no tooltip
|
|
451
|
+
expect(queryByTestId('label')).not.toBeInTheDocument();
|
|
452
|
+
|
|
453
|
+
// Show tooltip
|
|
454
|
+
fireEvent.mouseEnter(image);
|
|
455
|
+
|
|
456
|
+
await waitFor(() => {
|
|
457
|
+
expect(queryByTestId('label')).toBeInTheDocument();
|
|
458
|
+
});
|
|
459
|
+
|
|
460
|
+
// Hide tooltip
|
|
461
|
+
fireEvent.mouseLeave(image);
|
|
462
|
+
|
|
463
|
+
await waitFor(() => {
|
|
464
|
+
expect(queryByTestId('label')).not.toBeInTheDocument();
|
|
465
|
+
});
|
|
466
|
+
});
|
|
467
|
+
|
|
468
|
+
it('should update image state after load', async () => {
|
|
469
|
+
const { getByTestId } = render(<ImageComponent {...defaultProps} />);
|
|
470
|
+
|
|
471
|
+
await waitFor(() => {
|
|
472
|
+
const image = getByTestId('image');
|
|
473
|
+
expect(image).toHaveAttribute('image');
|
|
474
|
+
}, { timeout: 100 });
|
|
475
|
+
});
|
|
476
|
+
});
|
|
477
|
+
|
|
478
|
+
describe('PropTypes validation', () => {
|
|
479
|
+
it('should accept all required props', () => {
|
|
480
|
+
const { getByTestId } = render(<ImageComponent {...defaultProps} />);
|
|
481
|
+
const image = getByTestId('image');
|
|
482
|
+
expect(image).toBeInTheDocument();
|
|
483
|
+
});
|
|
484
|
+
|
|
485
|
+
it('should render with string src prop', () => {
|
|
486
|
+
const { getByTestId } = render(
|
|
487
|
+
<ImageComponent {...defaultProps} src="test.png" />
|
|
488
|
+
);
|
|
489
|
+
const image = getByTestId('image');
|
|
490
|
+
expect(image).toBeInTheDocument();
|
|
491
|
+
});
|
|
492
|
+
|
|
493
|
+
it('should render with number x and y props', () => {
|
|
494
|
+
const { getByTestId } = render(
|
|
495
|
+
<ImageComponent {...defaultProps} x={100} y={200} />
|
|
496
|
+
);
|
|
497
|
+
const image = getByTestId('image');
|
|
498
|
+
expect(image).toHaveAttribute('x', '100');
|
|
499
|
+
expect(image).toHaveAttribute('y', '200');
|
|
500
|
+
});
|
|
501
|
+
|
|
502
|
+
it('should render with string tooltip prop', () => {
|
|
503
|
+
const { getByTestId } = render(
|
|
504
|
+
<ImageComponent {...defaultProps} tooltip="Tooltip text" />
|
|
505
|
+
);
|
|
506
|
+
const image = getByTestId('image');
|
|
507
|
+
expect(image).toBeInTheDocument();
|
|
508
|
+
});
|
|
509
|
+
});
|
|
510
|
+
});
|