@lego-build/plugins 0.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/auto.d.ts +2 -0
- package/dist/auto.js +13 -0
- package/dist/chunk-2JJEO2HO.js +303 -0
- package/dist/chunk-4ILDPPTG.js +563 -0
- package/dist/chunk-MZJDRENL.js +303 -0
- package/dist/chunk-RM4HZDF3.js +555 -0
- package/dist/index.d.ts +12 -0
- package/dist/index.js +566 -0
- package/package.json +27 -0
|
@@ -0,0 +1,303 @@
|
|
|
1
|
+
// src/index.ts
|
|
2
|
+
function sendToParent(type, payload) {
|
|
3
|
+
window.parent.postMessage({ type, payload }, "*");
|
|
4
|
+
}
|
|
5
|
+
function setupIframeBridge() {
|
|
6
|
+
const pushState = window.history.pushState;
|
|
7
|
+
window.history.pushState = function(...args) {
|
|
8
|
+
const ret = pushState.apply(this, args);
|
|
9
|
+
sendToParent("IFRAME_URL_CHANGED", window.location.href);
|
|
10
|
+
return ret;
|
|
11
|
+
};
|
|
12
|
+
const replaceState = window.history.replaceState;
|
|
13
|
+
window.history.replaceState = function(...args) {
|
|
14
|
+
const ret = replaceState.apply(this, args);
|
|
15
|
+
sendToParent("IFRAME_URL_CHANGED", window.location.href);
|
|
16
|
+
return ret;
|
|
17
|
+
};
|
|
18
|
+
window.addEventListener("popstate", () => {
|
|
19
|
+
sendToParent("IFRAME_URL_CHANGED", window.location.href);
|
|
20
|
+
});
|
|
21
|
+
window.addEventListener("message", (e) => {
|
|
22
|
+
const msg = e.data;
|
|
23
|
+
if (!msg?.type) return;
|
|
24
|
+
switch (msg.type) {
|
|
25
|
+
case "NAVIGATE_BACK":
|
|
26
|
+
window.history.back();
|
|
27
|
+
break;
|
|
28
|
+
case "NAVIGATE_FORWARD":
|
|
29
|
+
window.history.forward();
|
|
30
|
+
break;
|
|
31
|
+
case "NAVIGATE_URL":
|
|
32
|
+
if (msg.payload) {
|
|
33
|
+
window.history.pushState({}, "", msg.payload);
|
|
34
|
+
sendToParent("IFRAME_URL_CHANGED", window.location.href);
|
|
35
|
+
}
|
|
36
|
+
break;
|
|
37
|
+
case "REFRESH":
|
|
38
|
+
window.location.reload();
|
|
39
|
+
break;
|
|
40
|
+
}
|
|
41
|
+
});
|
|
42
|
+
}
|
|
43
|
+
function createElementSelectorState() {
|
|
44
|
+
return {
|
|
45
|
+
isActive: false,
|
|
46
|
+
hoveredElement: null,
|
|
47
|
+
overlay: null,
|
|
48
|
+
tooltip: null,
|
|
49
|
+
depth: 0
|
|
50
|
+
};
|
|
51
|
+
}
|
|
52
|
+
function createOverlay() {
|
|
53
|
+
const overlay = document.createElement("div");
|
|
54
|
+
overlay.id = "lego-element-selector-overlay";
|
|
55
|
+
overlay.style.cssText = `
|
|
56
|
+
position: fixed;
|
|
57
|
+
pointer-events: none;
|
|
58
|
+
z-index: 2147483646;
|
|
59
|
+
background: rgba(59, 130, 246, 0.1);
|
|
60
|
+
border: 2px solid #3b82f6;
|
|
61
|
+
border-radius: 4px;
|
|
62
|
+
transition: all 0.1s ease-out;
|
|
63
|
+
`;
|
|
64
|
+
document.body.appendChild(overlay);
|
|
65
|
+
return overlay;
|
|
66
|
+
}
|
|
67
|
+
function createTooltip() {
|
|
68
|
+
const tooltip = document.createElement("div");
|
|
69
|
+
tooltip.id = "lego-element-selector-tooltip";
|
|
70
|
+
tooltip.style.cssText = `
|
|
71
|
+
position: fixed;
|
|
72
|
+
pointer-events: none;
|
|
73
|
+
z-index: 2147483647;
|
|
74
|
+
background: #3b82f6;
|
|
75
|
+
color: white;
|
|
76
|
+
padding: 4px 8px;
|
|
77
|
+
border-radius: 4px;
|
|
78
|
+
font-family: ui-monospace, monospace;
|
|
79
|
+
font-size: 12px;
|
|
80
|
+
white-space: nowrap;
|
|
81
|
+
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.2);
|
|
82
|
+
transition: all 0.1s ease-out;
|
|
83
|
+
`;
|
|
84
|
+
document.body.appendChild(tooltip);
|
|
85
|
+
return tooltip;
|
|
86
|
+
}
|
|
87
|
+
function getElementDepth(element) {
|
|
88
|
+
let depth = 0;
|
|
89
|
+
let current = element;
|
|
90
|
+
while (current && current !== document.body) {
|
|
91
|
+
depth++;
|
|
92
|
+
current = current.parentElement;
|
|
93
|
+
}
|
|
94
|
+
return depth;
|
|
95
|
+
}
|
|
96
|
+
function updateOverlayPosition(state) {
|
|
97
|
+
if (!state.hoveredElement || !state.overlay || !state.tooltip) return;
|
|
98
|
+
const rect = state.hoveredElement.getBoundingClientRect();
|
|
99
|
+
const viewportWidth = window.innerWidth;
|
|
100
|
+
state.overlay.style.top = `${rect.top}px`;
|
|
101
|
+
state.overlay.style.left = `${rect.left}px`;
|
|
102
|
+
state.overlay.style.width = `${rect.width}px`;
|
|
103
|
+
state.overlay.style.height = `${rect.height}px`;
|
|
104
|
+
let tooltipTop = rect.top - 32;
|
|
105
|
+
let tooltipLeft = rect.left;
|
|
106
|
+
if (tooltipTop < 8) {
|
|
107
|
+
tooltipTop = rect.bottom + 8;
|
|
108
|
+
}
|
|
109
|
+
const estimatedTooltipWidth = state.tooltip.offsetWidth || 150;
|
|
110
|
+
if (tooltipLeft + estimatedTooltipWidth > viewportWidth - 8) {
|
|
111
|
+
tooltipLeft = viewportWidth - estimatedTooltipWidth - 8;
|
|
112
|
+
}
|
|
113
|
+
if (tooltipLeft < 8) {
|
|
114
|
+
tooltipLeft = 8;
|
|
115
|
+
}
|
|
116
|
+
state.tooltip.style.top = `${tooltipTop}px`;
|
|
117
|
+
state.tooltip.style.left = `${tooltipLeft}px`;
|
|
118
|
+
}
|
|
119
|
+
function getElementInfo(element) {
|
|
120
|
+
const filePath = element.getAttribute("data-locator-path") || "";
|
|
121
|
+
const line = element.getAttribute("data-locator-line");
|
|
122
|
+
const depth = getElementDepth(element);
|
|
123
|
+
const id = element.id ? `#${element.id}` : "";
|
|
124
|
+
const classes = element.className && typeof element.className === "string" ? element.className.split(" ").filter((c) => c).map((c) => `.${c}`).join("") : "";
|
|
125
|
+
return {
|
|
126
|
+
tagName: element.tagName.toLowerCase(),
|
|
127
|
+
selector: element.tagName.toLowerCase() + id + classes.slice(0, 50),
|
|
128
|
+
filePath,
|
|
129
|
+
line: line ? parseInt(line) : void 0,
|
|
130
|
+
depth,
|
|
131
|
+
rect: element.getBoundingClientRect()
|
|
132
|
+
};
|
|
133
|
+
}
|
|
134
|
+
function highlightElement(state, element) {
|
|
135
|
+
state.hoveredElement = element;
|
|
136
|
+
if (!state.overlay) {
|
|
137
|
+
state.overlay = createOverlay();
|
|
138
|
+
}
|
|
139
|
+
if (!state.tooltip) {
|
|
140
|
+
state.tooltip = createTooltip();
|
|
141
|
+
}
|
|
142
|
+
state.overlay.style.display = "block";
|
|
143
|
+
state.tooltip.style.display = "block";
|
|
144
|
+
const info = getElementInfo(element);
|
|
145
|
+
state.depth = info.depth;
|
|
146
|
+
state.tooltip.innerHTML = `
|
|
147
|
+
<span style="opacity: 0.8">${info.selector}</span>
|
|
148
|
+
${info.filePath ? `<span style="margin-left: 8px; opacity: 0.6">\u{1F4C4} ${info.filePath.split("/").pop()}</span>` : ""}
|
|
149
|
+
`;
|
|
150
|
+
updateOverlayPosition(state);
|
|
151
|
+
sendToParent("HOVER_ELEMENT", {
|
|
152
|
+
tagName: info.tagName,
|
|
153
|
+
rect: info.rect,
|
|
154
|
+
filePath: info.filePath || void 0,
|
|
155
|
+
line: info.line,
|
|
156
|
+
depth: info.depth
|
|
157
|
+
});
|
|
158
|
+
}
|
|
159
|
+
function navigateDOM(state, direction) {
|
|
160
|
+
if (!state.hoveredElement) return;
|
|
161
|
+
let target = null;
|
|
162
|
+
switch (direction) {
|
|
163
|
+
case "parent":
|
|
164
|
+
target = state.hoveredElement.parentElement;
|
|
165
|
+
if (target && target === document.body) target = null;
|
|
166
|
+
break;
|
|
167
|
+
case "child":
|
|
168
|
+
target = state.hoveredElement.firstElementChild;
|
|
169
|
+
break;
|
|
170
|
+
case "next":
|
|
171
|
+
target = state.hoveredElement.nextElementSibling;
|
|
172
|
+
break;
|
|
173
|
+
case "prev":
|
|
174
|
+
target = state.hoveredElement.previousElementSibling;
|
|
175
|
+
break;
|
|
176
|
+
}
|
|
177
|
+
if (target && target !== document.documentElement && target !== document.body) {
|
|
178
|
+
highlightElement(state, target);
|
|
179
|
+
}
|
|
180
|
+
}
|
|
181
|
+
function selectElement(state) {
|
|
182
|
+
if (!state.hoveredElement) return;
|
|
183
|
+
const info = getElementInfo(state.hoveredElement);
|
|
184
|
+
sendToParent("CLICK_ELEMENT", {
|
|
185
|
+
tagName: info.tagName,
|
|
186
|
+
filePath: info.filePath,
|
|
187
|
+
line: info.line,
|
|
188
|
+
depth: info.depth
|
|
189
|
+
});
|
|
190
|
+
}
|
|
191
|
+
function setupElementSelector() {
|
|
192
|
+
const state = createElementSelectorState();
|
|
193
|
+
const handleMouseMove = (e) => {
|
|
194
|
+
if (!state.isActive) return;
|
|
195
|
+
const target = e.target;
|
|
196
|
+
if (target.id === "lego-element-selector-overlay" || target.id === "lego-element-selector-tooltip") {
|
|
197
|
+
return;
|
|
198
|
+
}
|
|
199
|
+
if (target === state.hoveredElement) return;
|
|
200
|
+
highlightElement(state, target);
|
|
201
|
+
};
|
|
202
|
+
const handleScroll = () => {
|
|
203
|
+
if (state.isActive && state.hoveredElement) {
|
|
204
|
+
updateOverlayPosition(state);
|
|
205
|
+
}
|
|
206
|
+
};
|
|
207
|
+
const handleResize = () => {
|
|
208
|
+
if (state.isActive && state.hoveredElement) {
|
|
209
|
+
updateOverlayPosition(state);
|
|
210
|
+
}
|
|
211
|
+
};
|
|
212
|
+
const handleKeyDown = (e) => {
|
|
213
|
+
if (!state.isActive) return;
|
|
214
|
+
switch (e.key) {
|
|
215
|
+
case "Escape":
|
|
216
|
+
sendToParent("SELECTOR_EXIT", true);
|
|
217
|
+
break;
|
|
218
|
+
case "ArrowUp":
|
|
219
|
+
e.preventDefault();
|
|
220
|
+
navigateDOM(state, "parent");
|
|
221
|
+
break;
|
|
222
|
+
case "ArrowDown":
|
|
223
|
+
e.preventDefault();
|
|
224
|
+
navigateDOM(state, "child");
|
|
225
|
+
break;
|
|
226
|
+
case "ArrowLeft":
|
|
227
|
+
e.preventDefault();
|
|
228
|
+
navigateDOM(state, "prev");
|
|
229
|
+
break;
|
|
230
|
+
case "ArrowRight":
|
|
231
|
+
e.preventDefault();
|
|
232
|
+
navigateDOM(state, "next");
|
|
233
|
+
break;
|
|
234
|
+
case "Enter":
|
|
235
|
+
e.preventDefault();
|
|
236
|
+
selectElement(state);
|
|
237
|
+
break;
|
|
238
|
+
}
|
|
239
|
+
};
|
|
240
|
+
const handleClick = (e) => {
|
|
241
|
+
if (!state.isActive) return;
|
|
242
|
+
e.preventDefault();
|
|
243
|
+
e.stopPropagation();
|
|
244
|
+
e.stopImmediatePropagation();
|
|
245
|
+
selectElement(state);
|
|
246
|
+
};
|
|
247
|
+
const activateSelector = () => {
|
|
248
|
+
state.isActive = true;
|
|
249
|
+
document.addEventListener("mousemove", handleMouseMove, true);
|
|
250
|
+
document.addEventListener("click", handleClick, true);
|
|
251
|
+
document.addEventListener("keydown", handleKeyDown, true);
|
|
252
|
+
window.addEventListener("scroll", handleScroll, true);
|
|
253
|
+
window.addEventListener("resize", handleResize);
|
|
254
|
+
document.body.style.cursor = "crosshair";
|
|
255
|
+
state.overlay = createOverlay();
|
|
256
|
+
state.tooltip = createTooltip();
|
|
257
|
+
};
|
|
258
|
+
const deactivateSelector = () => {
|
|
259
|
+
state.isActive = false;
|
|
260
|
+
document.removeEventListener("mousemove", handleMouseMove, true);
|
|
261
|
+
document.removeEventListener("click", handleClick, true);
|
|
262
|
+
document.removeEventListener("keydown", handleKeyDown, true);
|
|
263
|
+
window.removeEventListener("scroll", handleScroll, true);
|
|
264
|
+
window.removeEventListener("resize", handleResize);
|
|
265
|
+
if (state.overlay) {
|
|
266
|
+
state.overlay.remove();
|
|
267
|
+
state.overlay = null;
|
|
268
|
+
}
|
|
269
|
+
if (state.tooltip) {
|
|
270
|
+
state.tooltip.remove();
|
|
271
|
+
state.tooltip = null;
|
|
272
|
+
}
|
|
273
|
+
state.hoveredElement = null;
|
|
274
|
+
document.body.style.cursor = "";
|
|
275
|
+
state.depth = 0;
|
|
276
|
+
};
|
|
277
|
+
window.addEventListener("message", (e) => {
|
|
278
|
+
const msg = e.data;
|
|
279
|
+
if (msg.type === "TOGGLE_ELEMENT_SELECTOR") {
|
|
280
|
+
if (msg.payload) {
|
|
281
|
+
activateSelector();
|
|
282
|
+
} else {
|
|
283
|
+
deactivateSelector();
|
|
284
|
+
}
|
|
285
|
+
}
|
|
286
|
+
});
|
|
287
|
+
}
|
|
288
|
+
function initIframeBridge() {
|
|
289
|
+
try {
|
|
290
|
+
if (window.parent !== window) {
|
|
291
|
+
setupIframeBridge();
|
|
292
|
+
setupElementSelector();
|
|
293
|
+
}
|
|
294
|
+
} catch {
|
|
295
|
+
}
|
|
296
|
+
}
|
|
297
|
+
|
|
298
|
+
export {
|
|
299
|
+
sendToParent,
|
|
300
|
+
setupIframeBridge,
|
|
301
|
+
setupElementSelector,
|
|
302
|
+
initIframeBridge
|
|
303
|
+
};
|