@treelocator/runtime 0.1.0

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 (227) hide show
  1. package/.eslintrc +3 -0
  2. package/.turbo/turbo-build.log +30 -0
  3. package/.turbo/turbo-test.log +18 -0
  4. package/.turbo/turbo-ts.log +4 -0
  5. package/LICENSE +22 -0
  6. package/babel.config.js +14 -0
  7. package/dist/_generated_styles.d.ts +2 -0
  8. package/dist/_generated_styles.js +1734 -0
  9. package/dist/_generated_tree_icon.d.ts +2 -0
  10. package/dist/_generated_tree_icon.js +2 -0
  11. package/dist/adapters/HtmlElementTreeNode.d.ts +16 -0
  12. package/dist/adapters/HtmlElementTreeNode.js +43 -0
  13. package/dist/adapters/adapterApi.d.ts +30 -0
  14. package/dist/adapters/adapterApi.js +1 -0
  15. package/dist/adapters/createTreeNode.d.ts +2 -0
  16. package/dist/adapters/createTreeNode.js +17 -0
  17. package/dist/adapters/getElementInfo.d.ts +2 -0
  18. package/dist/adapters/getElementInfo.js +19 -0
  19. package/dist/adapters/getParentsPath.d.ts +2 -0
  20. package/dist/adapters/getParentsPath.js +35 -0
  21. package/dist/adapters/getTree.d.ts +1 -0
  22. package/dist/adapters/getTree.js +35 -0
  23. package/dist/adapters/goUpByTheTree.d.ts +7 -0
  24. package/dist/adapters/goUpByTheTree.js +22 -0
  25. package/dist/adapters/jsx/getExpressionData.d.ts +2 -0
  26. package/dist/adapters/jsx/getExpressionData.js +44 -0
  27. package/dist/adapters/jsx/getJSXComponentBoundingBox.d.ts +5 -0
  28. package/dist/adapters/jsx/getJSXComponentBoundingBox.js +46 -0
  29. package/dist/adapters/jsx/jsxAdapter.d.ts +11 -0
  30. package/dist/adapters/jsx/jsxAdapter.js +208 -0
  31. package/dist/adapters/jsx/runtimeStore.d.ts +10 -0
  32. package/dist/adapters/jsx/runtimeStore.js +87 -0
  33. package/dist/adapters/react/fiberToSimple.d.ts +3 -0
  34. package/dist/adapters/react/fiberToSimple.js +55 -0
  35. package/dist/adapters/react/findDebugSource.d.ts +5 -0
  36. package/dist/adapters/react/findDebugSource.js +13 -0
  37. package/dist/adapters/react/findFiberByHtmlElement.d.ts +2 -0
  38. package/dist/adapters/react/findFiberByHtmlElement.js +22 -0
  39. package/dist/adapters/react/gatherFiberRoots.d.ts +2 -0
  40. package/dist/adapters/react/gatherFiberRoots.js +29 -0
  41. package/dist/adapters/react/getAllFiberChildren.d.ts +2 -0
  42. package/dist/adapters/react/getAllFiberChildren.js +9 -0
  43. package/dist/adapters/react/getAllParentsElementsAndRootComponent.d.ts +8 -0
  44. package/dist/adapters/react/getAllParentsElementsAndRootComponent.js +34 -0
  45. package/dist/adapters/react/getAllWrappingParents.d.ts +2 -0
  46. package/dist/adapters/react/getAllWrappingParents.js +19 -0
  47. package/dist/adapters/react/getFiberComponentBoundingBox.d.ts +3 -0
  48. package/dist/adapters/react/getFiberComponentBoundingBox.js +27 -0
  49. package/dist/adapters/react/getFiberLabel.d.ts +3 -0
  50. package/dist/adapters/react/getFiberLabel.js +14 -0
  51. package/dist/adapters/react/getFiberOwnBoundingBox.d.ts +3 -0
  52. package/dist/adapters/react/getFiberOwnBoundingBox.js +6 -0
  53. package/dist/adapters/react/isStyled.d.ts +2 -0
  54. package/dist/adapters/react/isStyled.js +3 -0
  55. package/dist/adapters/react/makeFiberId.d.ts +2 -0
  56. package/dist/adapters/react/makeFiberId.js +16 -0
  57. package/dist/adapters/react/reactAdapter.d.ts +11 -0
  58. package/dist/adapters/react/reactAdapter.js +114 -0
  59. package/dist/adapters/react/searchDevtoolsRenderersForClosestTarget.d.ts +1 -0
  60. package/dist/adapters/react/searchDevtoolsRenderersForClosestTarget.js +11 -0
  61. package/dist/adapters/svelte/svelteAdapter.d.ts +22 -0
  62. package/dist/adapters/svelte/svelteAdapter.js +88 -0
  63. package/dist/adapters/vue/getVNodeBoundingBox.d.ts +4 -0
  64. package/dist/adapters/vue/getVNodeBoundingBox.js +31 -0
  65. package/dist/adapters/vue/vueAdapter.d.ts +15 -0
  66. package/dist/adapters/vue/vueAdapter.js +113 -0
  67. package/dist/browserApi.d.ts +103 -0
  68. package/dist/browserApi.js +160 -0
  69. package/dist/components/Button.d.ts +4 -0
  70. package/dist/components/Button.js +17 -0
  71. package/dist/components/ComponentOutline.d.ts +7 -0
  72. package/dist/components/ComponentOutline.js +93 -0
  73. package/dist/components/MaybeOutline.d.ts +7 -0
  74. package/dist/components/MaybeOutline.js +43 -0
  75. package/dist/components/Outline.d.ts +25 -0
  76. package/dist/components/Outline.js +135 -0
  77. package/dist/components/RenderBoxes.d.ts +4 -0
  78. package/dist/components/RenderBoxes.js +73 -0
  79. package/dist/components/Runtime.d.ts +3 -0
  80. package/dist/components/Runtime.js +188 -0
  81. package/dist/components/SimpleNodeOutline.d.ts +4 -0
  82. package/dist/components/SimpleNodeOutline.js +47 -0
  83. package/dist/components/Toast.d.ts +4 -0
  84. package/dist/components/Toast.js +68 -0
  85. package/dist/components/Tooltip.d.ts +5 -0
  86. package/dist/components/Tooltip.js +21 -0
  87. package/dist/consts.d.ts +6 -0
  88. package/dist/consts.js +5 -0
  89. package/dist/functions/cropPath.d.ts +1 -0
  90. package/dist/functions/cropPath.js +9 -0
  91. package/dist/functions/cropPath.test.d.ts +1 -0
  92. package/dist/functions/cropPath.test.js +16 -0
  93. package/dist/functions/deduplicateLabels.d.ts +2 -0
  94. package/dist/functions/deduplicateLabels.js +12 -0
  95. package/dist/functions/evalTemplate.d.ts +3 -0
  96. package/dist/functions/evalTemplate.js +7 -0
  97. package/dist/functions/evalTemplate.test.d.ts +1 -0
  98. package/dist/functions/evalTemplate.test.js +11 -0
  99. package/dist/functions/findNames.d.ts +5 -0
  100. package/dist/functions/findNames.js +15 -0
  101. package/dist/functions/formatAncestryChain.d.ts +9 -0
  102. package/dist/functions/formatAncestryChain.js +56 -0
  103. package/dist/functions/getBoundingRect.d.ts +1 -0
  104. package/dist/functions/getBoundingRect.js +11 -0
  105. package/dist/functions/getComposedBoundingBox.d.ts +2 -0
  106. package/dist/functions/getComposedBoundingBox.js +20 -0
  107. package/dist/functions/getIdsOnPathToRoot.d.ts +3 -0
  108. package/dist/functions/getIdsOnPathToRoot.js +15 -0
  109. package/dist/functions/getMultipleElementsBoundingBox.d.ts +2 -0
  110. package/dist/functions/getMultipleElementsBoundingBox.js +20 -0
  111. package/dist/functions/getPathToParent.d.ts +1 -0
  112. package/dist/functions/getPathToParent.js +15 -0
  113. package/dist/functions/getReferenceId.d.ts +1 -0
  114. package/dist/functions/getReferenceId.js +9 -0
  115. package/dist/functions/getUsableFileName.d.ts +1 -0
  116. package/dist/functions/getUsableFileName.js +17 -0
  117. package/dist/functions/getUsableFileName.test.d.ts +1 -0
  118. package/dist/functions/getUsableFileName.test.js +16 -0
  119. package/dist/functions/getUsableName.d.ts +2 -0
  120. package/dist/functions/getUsableName.js +47 -0
  121. package/dist/functions/isCombinationModifiersPressed.d.ts +4 -0
  122. package/dist/functions/isCombinationModifiersPressed.js +16 -0
  123. package/dist/functions/isLocatorsOwnElement.d.ts +1 -0
  124. package/dist/functions/isLocatorsOwnElement.js +3 -0
  125. package/dist/functions/mergeRects.d.ts +2 -0
  126. package/dist/functions/mergeRects.js +10 -0
  127. package/dist/functions/mergeRects.test.d.ts +1 -0
  128. package/dist/functions/mergeRects.test.js +23 -0
  129. package/dist/functions/nonNullable.d.ts +1 -0
  130. package/dist/functions/nonNullable.js +3 -0
  131. package/dist/functions/parseDataId.d.ts +3 -0
  132. package/dist/functions/parseDataId.js +44 -0
  133. package/dist/functions/transformPath.d.ts +1 -0
  134. package/dist/functions/transformPath.js +7 -0
  135. package/dist/functions/transformPath.test.d.ts +1 -0
  136. package/dist/functions/transformPath.test.js +16 -0
  137. package/dist/global.d.js +1 -0
  138. package/dist/index.d.ts +11 -0
  139. package/dist/index.js +13 -0
  140. package/dist/initRuntime.d.ts +8 -0
  141. package/dist/initRuntime.js +80 -0
  142. package/dist/output.css +1733 -0
  143. package/dist/types/LabelData.d.ts +5 -0
  144. package/dist/types/LabelData.js +1 -0
  145. package/dist/types/TreeNode.d.ts +19 -0
  146. package/dist/types/TreeNode.js +1 -0
  147. package/dist/types/types.d.ts +53 -0
  148. package/dist/types/types.js +1 -0
  149. package/jest.config.ts +195 -0
  150. package/package.json +75 -0
  151. package/scripts/wrapCSS.js +26 -0
  152. package/scripts/wrapImage.js +24 -0
  153. package/src/_generated_styles.ts +1734 -0
  154. package/src/_generated_tree_icon.ts +2 -0
  155. package/src/adapters/HtmlElementTreeNode.ts +51 -0
  156. package/src/adapters/adapterApi.ts +35 -0
  157. package/src/adapters/createTreeNode.ts +25 -0
  158. package/src/adapters/getElementInfo.tsx +27 -0
  159. package/src/adapters/getParentsPath.tsx +49 -0
  160. package/src/adapters/getTree.tsx +45 -0
  161. package/src/adapters/goUpByTheTree.ts +20 -0
  162. package/src/adapters/jsx/getExpressionData.ts +47 -0
  163. package/src/adapters/jsx/getJSXComponentBoundingBox.ts +63 -0
  164. package/src/adapters/jsx/jsxAdapter.ts +276 -0
  165. package/src/adapters/jsx/runtimeStore.ts +94 -0
  166. package/src/adapters/react/fiberToSimple.tsx +72 -0
  167. package/src/adapters/react/findDebugSource.ts +15 -0
  168. package/src/adapters/react/findFiberByHtmlElement.ts +27 -0
  169. package/src/adapters/react/gatherFiberRoots.tsx +36 -0
  170. package/src/adapters/react/getAllFiberChildren.tsx +11 -0
  171. package/src/adapters/react/getAllParentsElementsAndRootComponent.ts +52 -0
  172. package/src/adapters/react/getAllWrappingParents.ts +25 -0
  173. package/src/adapters/react/getFiberComponentBoundingBox.ts +30 -0
  174. package/src/adapters/react/getFiberLabel.ts +20 -0
  175. package/src/adapters/react/getFiberOwnBoundingBox.ts +9 -0
  176. package/src/adapters/react/isStyled.ts +5 -0
  177. package/src/adapters/react/makeFiberId.tsx +19 -0
  178. package/src/adapters/react/reactAdapter.ts +148 -0
  179. package/src/adapters/react/searchDevtoolsRenderersForClosestTarget.ts +15 -0
  180. package/src/adapters/svelte/svelteAdapter.ts +111 -0
  181. package/src/adapters/vue/getVNodeBoundingBox.tsx +42 -0
  182. package/src/adapters/vue/vueAdapter.ts +139 -0
  183. package/src/assets/tree-icon.png +0 -0
  184. package/src/browserApi.ts +288 -0
  185. package/src/components/Button.tsx +14 -0
  186. package/src/components/ComponentOutline.tsx +98 -0
  187. package/src/components/MaybeOutline.tsx +49 -0
  188. package/src/components/Outline.tsx +153 -0
  189. package/src/components/RenderBoxes.tsx +57 -0
  190. package/src/components/Runtime.tsx +246 -0
  191. package/src/components/SimpleNodeOutline.tsx +27 -0
  192. package/src/components/Toast.tsx +83 -0
  193. package/src/components/Tooltip.tsx +28 -0
  194. package/src/consts.ts +7 -0
  195. package/src/functions/cropPath.test.ts +18 -0
  196. package/src/functions/cropPath.ts +12 -0
  197. package/src/functions/deduplicateLabels.ts +16 -0
  198. package/src/functions/evalTemplate.test.ts +12 -0
  199. package/src/functions/evalTemplate.ts +8 -0
  200. package/src/functions/findNames.ts +20 -0
  201. package/src/functions/formatAncestryChain.ts +80 -0
  202. package/src/functions/getBoundingRect.tsx +11 -0
  203. package/src/functions/getComposedBoundingBox.tsx +25 -0
  204. package/src/functions/getIdsOnPathToRoot.tsx +21 -0
  205. package/src/functions/getMultipleElementsBoundingBox.tsx +25 -0
  206. package/src/functions/getPathToParent.tsx +17 -0
  207. package/src/functions/getReferenceId.tsx +10 -0
  208. package/src/functions/getUsableFileName.test.tsx +24 -0
  209. package/src/functions/getUsableFileName.tsx +19 -0
  210. package/src/functions/getUsableName.ts +52 -0
  211. package/src/functions/isCombinationModifiersPressed.ts +32 -0
  212. package/src/functions/isLocatorsOwnElement.tsx +9 -0
  213. package/src/functions/mergeRects.test.ts +15 -0
  214. package/src/functions/mergeRects.tsx +12 -0
  215. package/src/functions/nonNullable.ts +3 -0
  216. package/src/functions/parseDataId.ts +62 -0
  217. package/src/functions/transformPath.test.ts +28 -0
  218. package/src/functions/transformPath.ts +7 -0
  219. package/src/global.d.ts +31 -0
  220. package/src/index.ts +18 -0
  221. package/src/initRuntime.ts +83 -0
  222. package/src/main.css +3 -0
  223. package/src/types/LabelData.ts +6 -0
  224. package/src/types/TreeNode.ts +22 -0
  225. package/src/types/types.ts +55 -0
  226. package/tailwind.config.js +9 -0
  227. package/tsconfig.json +20 -0
@@ -0,0 +1,2 @@
1
+ const treeIcon: string = "";
2
+ export default treeIcon;
@@ -0,0 +1,51 @@
1
+ /* eslint-disable @typescript-eslint/ban-ts-comment */
2
+ import { Source } from "@locator/shared";
3
+ import { getReferenceId } from "../functions/getReferenceId";
4
+ import nonNullable from "../functions/nonNullable";
5
+ import { TreeNode, TreeNodeComponent } from "../types/TreeNode";
6
+ import { SimpleDOMRect } from "../types/types";
7
+
8
+ export class HtmlElementTreeNode implements TreeNode {
9
+ type: "element" = "element";
10
+ element: HTMLElement;
11
+ name: string;
12
+ uniqueId: string;
13
+ constructor(element: HTMLElement) {
14
+ this.element = element;
15
+ this.name = element.nodeName.toLowerCase();
16
+ this.uniqueId = String(getReferenceId(element));
17
+ }
18
+ getBox(): SimpleDOMRect | null {
19
+ return this.element.getBoundingClientRect();
20
+ }
21
+ getElement(): Element | Text {
22
+ return this.element;
23
+ }
24
+ getChildren(): TreeNode[] {
25
+ const children = Array.from(this.element.children);
26
+ return children
27
+ .map((child) => {
28
+ if (child instanceof HTMLElement) {
29
+ // @ts-ignore
30
+ return new this.constructor(child);
31
+ } else {
32
+ return null;
33
+ }
34
+ })
35
+ .filter(nonNullable);
36
+ }
37
+ getParent(): TreeNode | null {
38
+ if (this.element.parentElement) {
39
+ // @ts-ignore
40
+ return new this.constructor(this.element.parentElement);
41
+ } else {
42
+ return null;
43
+ }
44
+ }
45
+ getSource(): Source | null {
46
+ throw new Error("Method not implemented.");
47
+ }
48
+ getComponent(): TreeNodeComponent | null {
49
+ throw new Error("Method not implemented.");
50
+ }
51
+ }
@@ -0,0 +1,35 @@
1
+ import { LabelData } from "../types/LabelData";
2
+ import { TreeNode } from "../types/TreeNode";
3
+ import { LinkProps, SimpleDOMRect } from "../types/types";
4
+
5
+ export type ElementInfo = {
6
+ box: SimpleDOMRect;
7
+ label: string;
8
+ link: LinkProps | null;
9
+ };
10
+
11
+ export type FullElementInfo = {
12
+ thisElement: ElementInfo;
13
+ htmlElement: HTMLElement;
14
+ parentElements: ElementInfo[];
15
+ componentBox: SimpleDOMRect;
16
+ componentsLabels: LabelData[];
17
+ };
18
+
19
+ export type TreeState = {
20
+ root: TreeNode;
21
+ originalNode: TreeNode;
22
+ expandedIds: Set<string>;
23
+ highlightedId: string;
24
+ };
25
+
26
+ export type ParentPathItem = {
27
+ title: string;
28
+ link: LinkProps | null;
29
+ };
30
+
31
+ export interface AdapterObject {
32
+ getElementInfo(element: HTMLElement): FullElementInfo | null;
33
+ getTree?(includeElement: HTMLElement): TreeState | null;
34
+ getParentsPaths(element: HTMLElement): ParentPathItem[];
35
+ }
@@ -0,0 +1,25 @@
1
+ import { TreeNode } from "../types/TreeNode";
2
+ import { ReactTreeNodeElement } from "./react/reactAdapter";
3
+ import { JSXTreeNodeElement } from "./jsx/jsxAdapter";
4
+ import {
5
+ detectJSX,
6
+ detectReact,
7
+ } from "@locator/shared";
8
+
9
+ export function createTreeNode(
10
+ element: HTMLElement,
11
+ adapterId?: string
12
+ ): TreeNode | null {
13
+ // Check for React adapter
14
+ if (adapterId === "react" || detectReact()) {
15
+ return new ReactTreeNodeElement(element);
16
+ }
17
+
18
+ // Check for JSX adapter (babel plugin) - check if element has data-locatorjs-id
19
+ if (adapterId === "jsx" || detectJSX() || element.dataset.locatorjsId) {
20
+ return new JSXTreeNodeElement(element);
21
+ }
22
+
23
+ // Fallback - return null for unsupported frameworks
24
+ return null;
25
+ }
@@ -0,0 +1,27 @@
1
+ import reactAdapter from "./react/reactAdapter";
2
+ import jsxAdapter from "./jsx/jsxAdapter";
3
+ import svelteAdapter from "./svelte/svelteAdapter";
4
+ import vueAdapter from "./vue/vueAdapter";
5
+ import { AdapterId } from "../consts";
6
+
7
+ export function getElementInfo(target: HTMLElement, adapterId?: AdapterId) {
8
+ if (adapterId === "react") {
9
+ return reactAdapter.getElementInfo(target);
10
+ }
11
+ if (adapterId === "svelte") {
12
+ return svelteAdapter.getElementInfo(target);
13
+ }
14
+ if (adapterId === "jsx") {
15
+ return jsxAdapter.getElementInfo(target);
16
+ }
17
+ if (adapterId === "vue") {
18
+ return vueAdapter.getElementInfo(target);
19
+ }
20
+
21
+ return (
22
+ jsxAdapter.getElementInfo(target) ||
23
+ reactAdapter.getElementInfo(target) ||
24
+ svelteAdapter.getElementInfo(target) ||
25
+ vueAdapter.getElementInfo(target)
26
+ );
27
+ }
@@ -0,0 +1,49 @@
1
+ import reactAdapter from "./react/reactAdapter";
2
+ import {
3
+ detectJSX,
4
+ detectReact,
5
+ detectSvelte,
6
+ detectVue,
7
+ } from "@locator/shared";
8
+ import { ParentPathItem } from "./adapterApi";
9
+ import svelteAdapter from "./svelte/svelteAdapter";
10
+ import jsxAdapter from "./jsx/jsxAdapter";
11
+ import vueAdapter from "./vue/vueAdapter";
12
+
13
+ export function getParentsPaths(
14
+ target: HTMLElement,
15
+ adapterId?: string
16
+ ): ParentPathItem[] {
17
+ if (adapterId === "react" && reactAdapter.getParentsPaths) {
18
+ return reactAdapter.getParentsPaths(target);
19
+ }
20
+ if (adapterId === "svelte" && svelteAdapter.getParentsPaths) {
21
+ return svelteAdapter.getParentsPaths(target);
22
+ }
23
+ if (adapterId === "vue" && vueAdapter.getParentsPaths) {
24
+ return vueAdapter.getParentsPaths(target);
25
+ }
26
+ if (adapterId === "jsx" && jsxAdapter.getParentsPaths) {
27
+ return jsxAdapter.getParentsPaths(target);
28
+ }
29
+
30
+ if (detectSvelte() && svelteAdapter.getParentsPaths) {
31
+ return svelteAdapter.getParentsPaths(target);
32
+ }
33
+
34
+ if (detectVue() && vueAdapter.getParentsPaths) {
35
+ return vueAdapter.getParentsPaths(target);
36
+ }
37
+
38
+ if (detectReact() && reactAdapter.getParentsPaths) {
39
+ return reactAdapter.getParentsPaths(target);
40
+ }
41
+
42
+ // // Must be last, because its global data leaks from Locator extension.
43
+ // // Because the extension is in SolidJS and it uses JSX plugin in dev mode.
44
+ if (detectJSX() && jsxAdapter.getParentsPaths) {
45
+ return jsxAdapter.getParentsPaths(target);
46
+ }
47
+
48
+ return [];
49
+ }
@@ -0,0 +1,45 @@
1
+ import reactAdapter from "./react/reactAdapter";
2
+ import jsxAdapter from "./jsx/jsxAdapter";
3
+ import svelteAdapter from "./svelte/svelteAdapter";
4
+ import {
5
+ detectJSX,
6
+ detectReact,
7
+ detectSvelte,
8
+ detectVue,
9
+ } from "@locator/shared";
10
+ import vueAdapter from "./vue/vueAdapter";
11
+
12
+ export function getTree(target: HTMLElement, adapterId?: string) {
13
+ if (adapterId === "react" && reactAdapter.getTree) {
14
+ return reactAdapter.getTree(target);
15
+ }
16
+ if (adapterId === "svelte" && svelteAdapter.getTree) {
17
+ return svelteAdapter.getTree(target);
18
+ }
19
+ if (adapterId === "vue" && vueAdapter.getTree) {
20
+ return vueAdapter.getTree(target);
21
+ }
22
+ if (adapterId === "jsx" && jsxAdapter.getTree) {
23
+ return jsxAdapter.getTree(target);
24
+ }
25
+
26
+ if (detectSvelte() && svelteAdapter.getTree) {
27
+ return svelteAdapter.getTree(target);
28
+ }
29
+
30
+ if (detectVue() && vueAdapter.getTree) {
31
+ return vueAdapter.getTree(target);
32
+ }
33
+
34
+ if (detectReact() && reactAdapter.getTree) {
35
+ return reactAdapter.getTree(target);
36
+ }
37
+
38
+ // Must be last, because its global data leaks from Locator extension.
39
+ // Because the extension is in SolidJS and it uses JSX plugin in dev mode.
40
+ if (detectJSX() && jsxAdapter.getTree) {
41
+ return jsxAdapter.getTree(target);
42
+ }
43
+
44
+ return null;
45
+ }
@@ -0,0 +1,20 @@
1
+ import { TreeNode } from "../types/TreeNode";
2
+
3
+ export function goUpByTheTree(originalNode: TreeNode) {
4
+ let root = originalNode;
5
+ const expandedIds = new Set<string>();
6
+ let current: TreeNode | null = root;
7
+
8
+ const highlightedId = root.uniqueId;
9
+ expandedIds.add(current.uniqueId);
10
+ let limit = 2;
11
+ while (current && limit > 0) {
12
+ limit--;
13
+ current = current.getParent();
14
+ if (current) {
15
+ expandedIds.add(current.uniqueId);
16
+ root = current;
17
+ }
18
+ }
19
+ return { expandedIds, highlightedId, root, originalNode };
20
+ }
@@ -0,0 +1,47 @@
1
+ import type { ExpressionInfo, FileStorage } from "@locator/shared";
2
+ import { parseDataId, parseDataPath } from "../../functions/parseDataId";
3
+
4
+ export function getExpressionData(
5
+ target: HTMLElement,
6
+ fileData: FileStorage | null
7
+ ): ExpressionInfo | null {
8
+ // First check for data-locatorjs (path-based, for server components)
9
+ if (target.dataset.locatorjs) {
10
+ const parsed = parseDataPath(target.dataset.locatorjs);
11
+ if (parsed) {
12
+ const [, line, column] = parsed;
13
+
14
+ // Try to find in fileData if available
15
+ if (fileData) {
16
+ const expData = fileData.expressions.find(
17
+ (exp) =>
18
+ exp.loc.start.line === line && exp.loc.start.column === column
19
+ );
20
+ if (expData) {
21
+ return expData;
22
+ }
23
+ }
24
+
25
+ // If no fileData or not found, create minimal ExpressionInfo from path
26
+ return {
27
+ name: target.tagName.toLowerCase(),
28
+ loc: {
29
+ start: { line, column },
30
+ end: { line, column },
31
+ },
32
+ wrappingComponentId: null,
33
+ };
34
+ }
35
+ }
36
+
37
+ // Fall back to data-locatorjs-id (ID-based, traditional approach)
38
+ if (target.dataset.locatorjsId && fileData) {
39
+ const [, id] = parseDataId(target.dataset.locatorjsId);
40
+ const expData = fileData.expressions[Number(id)];
41
+ if (expData) {
42
+ return expData;
43
+ }
44
+ }
45
+
46
+ return null;
47
+ }
@@ -0,0 +1,63 @@
1
+ import type { FileStorage } from "@locator/shared";
2
+ import { mergeRects } from "../../functions/mergeRects";
3
+ import { parseDataId, parseDataPath } from "../../functions/parseDataId";
4
+ import type { SimpleDOMRect } from "../../types/types";
5
+ import { getExpressionData } from "./getExpressionData";
6
+
7
+ export function getJSXComponentBoundingBox(
8
+ found: HTMLElement,
9
+ locatorData: { [filename: string]: FileStorage },
10
+ componentFolder: string,
11
+ componentId: number
12
+ ): SimpleDOMRect {
13
+ let composedBox: SimpleDOMRect = found.getBoundingClientRect();
14
+ // Currently it works well only for components with one root element, but for components with multiple root elements we would need to track instance ids.
15
+ function goParent(current: HTMLElement) {
16
+ const parent = current.parentNode;
17
+ if (!parent) {
18
+ return;
19
+ }
20
+ if (parent instanceof HTMLElement) {
21
+ // Check for either data-locatorjs (path-based) or data-locatorjs-id (ID-based)
22
+ if (parent.dataset.locatorjs || parent.dataset.locatorjsId) {
23
+ let fileFullPath: string;
24
+
25
+ if (parent.dataset.locatorjs) {
26
+ const parsed = parseDataPath(parent.dataset.locatorjs);
27
+ if (!parsed) {
28
+ goParent(parent);
29
+ return;
30
+ }
31
+ [fileFullPath] = parsed;
32
+ } else if (parent.dataset.locatorjsId) {
33
+ [fileFullPath] = parseDataId(parent.dataset.locatorjsId);
34
+ } else {
35
+ goParent(parent);
36
+ return;
37
+ }
38
+
39
+ const fileData: FileStorage | undefined = locatorData[fileFullPath];
40
+ const expData = getExpressionData(parent, fileData || null);
41
+ if (expData) {
42
+ if (
43
+ expData.wrappingComponentId === componentId &&
44
+ componentFolder === fileFullPath
45
+ ) {
46
+ composedBox = mergeRects(
47
+ composedBox,
48
+ parent.getBoundingClientRect()
49
+ );
50
+ goParent(parent);
51
+ }
52
+ expData.wrappingComponentId;
53
+ }
54
+ } else {
55
+ // If there is no locatorjs-id or locatorjs, we should go to the parent, because it can be some library element
56
+ goParent(parent);
57
+ }
58
+ }
59
+ }
60
+ goParent(found);
61
+
62
+ return composedBox;
63
+ }