scordi-extension 1.14.4 → 1.14.5

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 (75) hide show
  1. package/dist/sdk/index.js +605 -587
  2. package/package.json +2 -2
  3. package/dist/.vite/manifest.json +0 -67
  4. package/dist/assets/DataExtractBlock-B7AuVonx.js +0 -6
  5. package/dist/assets/index-B9PKMhAi.js +0 -6
  6. package/dist/assets/index-BAWpTgrx.js +0 -4
  7. package/dist/assets/index-CMlx_xZE.css +0 -1
  8. package/dist/assets/index-CZjAbH8z.js +0 -1
  9. package/dist/assets/index-DkFIvNAm.css +0 -1
  10. package/dist/assets/index-r5QRyJ4m.js +0 -1
  11. package/dist/assets/index.html-BIbPySIb.js +0 -74
  12. package/dist/assets/index.html-cr-_Qdxq.js +0 -49
  13. package/dist/assets/index.ts-BqxJYCmp.js +0 -160
  14. package/dist/assets/index.ts-Yg98L0Rk.js +0 -158
  15. package/dist/assets/internal-messages-Drf0Gb9U.js +0 -1
  16. package/dist/assets/loading-page-1924caaa.js +0 -51
  17. package/dist/assets/main.tsx-BEyI95mZ.js +0 -2
  18. package/dist/assets/main.tsx-BKzdlur-.js +0 -2
  19. package/dist/assets/main.tsx-CiPjkJNz.js +0 -2
  20. package/dist/assets/main.tsx-loader-BYP9L25W.js +0 -13
  21. package/dist/assets/main.tsx-loader-CRGmwXNh.js +0 -13
  22. package/dist/assets/main.tsx-loader-FYb0LhDu.js +0 -13
  23. package/dist/assets/schemas-C8r0jYHO.js +0 -21
  24. package/dist/assets/types-oHwqLVs0.js +0 -1
  25. package/dist/logo.png +0 -0
  26. package/dist/manifest.json +0 -58
  27. package/dist/public/logo.png +0 -0
  28. package/dist/service-worker-loader.js +0 -1
  29. package/dist/src/blocks/AiParseDataBlock.ts.js +0 -101
  30. package/dist/src/blocks/ClearValueFormBlock.ts.js +0 -55
  31. package/dist/src/blocks/DataExtractBlock.ts.js +0 -28
  32. package/dist/src/blocks/ElementExistsBlock.ts.js +0 -26
  33. package/dist/src/blocks/EventClickBlock.ts.js +0 -143
  34. package/dist/src/blocks/FetchApiBlock.ts.js +0 -50
  35. package/dist/src/blocks/GetAttributeValueBlock.ts.js +0 -33
  36. package/dist/src/blocks/GetElementDataBlock.ts.js +0 -114
  37. package/dist/src/blocks/GetTextBlock.ts.js +0 -152
  38. package/dist/src/blocks/GetValueFormBlock.ts.js +0 -52
  39. package/dist/src/blocks/KeypressBlock.ts.js +0 -89
  40. package/dist/src/blocks/SaveAssetsBlock.ts.js +0 -35
  41. package/dist/src/blocks/ScrollBlock.ts.js +0 -111
  42. package/dist/src/blocks/SetValueFormBlock.ts.js +0 -56
  43. package/dist/src/blocks/WaitBlock.ts.js +0 -24
  44. package/dist/src/blocks/index.ts.js +0 -163
  45. package/dist/src/blocks/types.ts.js +0 -11
  46. package/dist/src/content/elements/finders/CssSelector.ts.js +0 -51
  47. package/dist/src/content/elements/finders/ElementSelector.ts.js +0 -20
  48. package/dist/src/content/elements/finders/IframeSelector.ts.js +0 -32
  49. package/dist/src/content/elements/finders/ShadowDOMSelector.ts.js +0 -38
  50. package/dist/src/content/elements/finders/XPathFinder.ts.js +0 -32
  51. package/dist/src/content/elements/index.ts.js +0 -26
  52. package/dist/src/content/elements/utils/CSSSelectorGenerator.ts.js +0 -72
  53. package/dist/src/content/elements/utils/XPathGenerator.ts.js +0 -62
  54. package/dist/src/content/handler/ExternalMessageHandler.ts.js +0 -78
  55. package/dist/src/content/handler/InternalMessageHandler.ts.js +0 -18
  56. package/dist/src/content/kernel/MessageKernel.ts.js +0 -83
  57. package/dist/src/content/main.tsx-loader.js +0 -22
  58. package/dist/src/content/main.tsx.js +0 -13
  59. package/dist/src/content/utils/index.ts.js +0 -1
  60. package/dist/src/content/utils/synchronizedLock.ts.js +0 -35
  61. package/dist/src/popup/index.html +0 -11
  62. package/dist/src/types/internal-messages.ts.js +0 -15
  63. package/dist/vendor/.vite-deps-chunk-PR4QN5HX.js__v--57e88532.js +0 -42
  64. package/dist/vendor/.vite-deps-chunk-PR4QN5HX.js__v--ba9031c0.js +0 -42
  65. package/dist/vendor/.vite-deps-chunk-VBBIEGWT.js__v--57e88532.js +0 -4156
  66. package/dist/vendor/.vite-deps-jsonata.js__v--5da6ca24.js +0 -5761
  67. package/dist/vendor/.vite-deps-jsonata.js__v--ba9031c0.js +0 -5761
  68. package/dist/vendor/.vite-deps-zod.js__v--13965f26.js +0 -219
  69. package/dist/vendor/.vite-deps-zod.js__v--ba9031c0.js +0 -12651
  70. package/dist/vendor/crx-client-port.js +0 -66
  71. package/dist/vendor/crx-client-preamble.js +0 -4
  72. package/dist/vendor/react-refresh.js +0 -670
  73. package/dist/vendor/vite-client.js +0 -1098
  74. package/dist/vendor/vite-dist-client-env.mjs.js +0 -19
  75. package/dist/vendor/webcomponents-custom-elements.js +0 -47
@@ -1,101 +0,0 @@
1
- import z from "/vendor/.vite-deps-zod.js__v--13965f26.js";
2
- const ObjectSchemaDefinitionSchema = z.object({
3
- type: z.literal("object"),
4
- shape: z.record(z.string(), z.any())
5
- });
6
- const ArraySchemaDefinitionSchema = z.object({
7
- type: z.literal("array"),
8
- items: z.any()
9
- });
10
- const SchemaDefinitionSchema = z.discriminatedUnion("type", [
11
- ObjectSchemaDefinitionSchema,
12
- ArraySchemaDefinitionSchema
13
- ]);
14
- export const AiParseDataBlockSchema = z.object({
15
- name: z.literal("ai-parse-data"),
16
- sourceData: z.any().optional(),
17
- schemaDefinition: SchemaDefinitionSchema,
18
- prompt: z.string().optional(),
19
- model: z.string().optional(),
20
- apiKey: z.string().min(1, "OpenAI API key is required")
21
- // 필수
22
- });
23
- export function validateAiParseDataBlock(data) {
24
- return AiParseDataBlockSchema.parse(data);
25
- }
26
- export async function handlerAiParseData(data) {
27
- try {
28
- const {
29
- sourceData,
30
- schemaDefinition,
31
- prompt,
32
- model = "gpt-4o-mini",
33
- apiKey
34
- } = data;
35
- if (sourceData === void 0 || sourceData === null) {
36
- throw new Error("sourceData is required for ai-parse-data block");
37
- }
38
- if (!apiKey || apiKey.trim() === "") {
39
- throw new Error("apiKey is required for ai-parse-data block");
40
- }
41
- const response = await chrome.runtime.sendMessage({
42
- type: "AI_PARSE_DATA",
43
- data: {
44
- sourceData,
45
- schemaDefinition,
46
- prompt,
47
- model,
48
- apiKey
49
- // API 키 포함
50
- }
51
- });
52
- if (response.$isError) {
53
- throw new Error(response.message || "AI parsing failed");
54
- }
55
- return {
56
- data: response.data
57
- };
58
- } catch (error) {
59
- return {
60
- hasError: true,
61
- message: error instanceof Error ? error.message : "Unknown error in ai-parse-data handler",
62
- data: null
63
- };
64
- }
65
- }
66
- export function createSchema(shape) {
67
- return {
68
- type: "object",
69
- shape
70
- };
71
- }
72
- export function createArraySchema(items) {
73
- return {
74
- type: "array",
75
- items
76
- };
77
- }
78
- export const Schema = {
79
- string: (options) => ({
80
- type: "string",
81
- ...options
82
- }),
83
- number: (options) => ({
84
- type: "number",
85
- ...options
86
- }),
87
- boolean: (options) => ({
88
- type: "boolean",
89
- ...options
90
- }),
91
- array: (items, options) => ({
92
- type: "array",
93
- items,
94
- ...options
95
- }),
96
- object: (shape, options) => ({
97
- type: "object",
98
- shape,
99
- ...options
100
- })
101
- };
@@ -1,55 +0,0 @@
1
- import { findElement } from "/src/content/elements/index.ts.js";
2
- import { BaseBlockSchema } from "/src/blocks/types.ts.js";
3
- import z from "/vendor/.vite-deps-zod.js__v--13965f26.js";
4
- export const ClearValueFormsBlockSchema = BaseBlockSchema.extend({
5
- name: z.literal("clear-value-form"),
6
- type: z.enum(["text-field", "select", "checkbox"]).optional()
7
- });
8
- export function validateClearValueFormsBlock(data) {
9
- return ClearValueFormsBlockSchema.parse(data);
10
- }
11
- export async function handlerClearValueForm(data) {
12
- try {
13
- const { selector = "", type = "text-field", findBy = "cssSelector" } = data;
14
- if (!selector) {
15
- throw new Error("Selector is required for clear-value-form block");
16
- }
17
- const element = await findElement({ selector, findBy, option: data.option });
18
- if (!element) {
19
- throw new Error("Form element not found");
20
- }
21
- clearFormValue(element, type);
22
- return { data: "Form element cleared successfully" };
23
- } catch (error) {
24
- return {
25
- hasError: true,
26
- message: error instanceof Error ? error.message : "Unknown error in clear-value-form handler",
27
- data: null
28
- };
29
- }
30
- }
31
- function clearFormValue(element, type) {
32
- switch (type) {
33
- case "text-field":
34
- if (element instanceof HTMLInputElement || element instanceof HTMLTextAreaElement) {
35
- element.value = "";
36
- element.dispatchEvent(new Event("input", { bubbles: true }));
37
- element.dispatchEvent(new Event("change", { bubbles: true }));
38
- }
39
- break;
40
- case "select":
41
- if (element instanceof HTMLSelectElement) {
42
- element.selectedIndex = 0;
43
- element.dispatchEvent(new Event("change", { bubbles: true }));
44
- }
45
- break;
46
- case "checkbox":
47
- if (element instanceof HTMLInputElement && element.type === "checkbox") {
48
- element.checked = false;
49
- element.dispatchEvent(new Event("change", { bubbles: true }));
50
- }
51
- break;
52
- default:
53
- throw new Error(`Unsupported form element type: ${type}`);
54
- }
55
- }
@@ -1,28 +0,0 @@
1
- import z from "/vendor/.vite-deps-zod.js__v--13965f26.js";
2
- import __vite__cjsImport1_jsonata from "/vendor/.vite-deps-jsonata.js__v--5da6ca24.js"; const jsonata = __vite__cjsImport1_jsonata.__esModule ? __vite__cjsImport1_jsonata.default : __vite__cjsImport1_jsonata;
3
- export const DataExtractBlockSchema = z.object({
4
- name: z.literal("data-extract"),
5
- code: z.string(),
6
- inputData: z.any().optional()
7
- });
8
- export function validateDataExtractBlock(data) {
9
- return DataExtractBlockSchema.parse(data);
10
- }
11
- export async function handlerDataExtract(data) {
12
- try {
13
- console.log("[DataExtractBlock] Executing JSONata query:", data.code);
14
- const expression = jsonata(data.code);
15
- const result = await expression.evaluate(data.inputData);
16
- console.log("[DataExtractBlock] Data extraction successful");
17
- return {
18
- data: result
19
- };
20
- } catch (error) {
21
- console.error("[DataExtractBlock] Data extraction error:", error);
22
- return {
23
- hasError: true,
24
- message: error instanceof Error ? error.message : "Unknown error in data extraction",
25
- data: void 0
26
- };
27
- }
28
- }
@@ -1,26 +0,0 @@
1
- import { findElement } from "/src/content/elements/index.ts.js";
2
- import { BaseBlockSchema } from "/src/blocks/types.ts.js";
3
- import z from "/vendor/.vite-deps-zod.js__v--13965f26.js";
4
- export const ElementExistsBlockSchema = BaseBlockSchema.extend({
5
- name: z.literal("element-exists")
6
- });
7
- export function validateElementExistsBlock(data) {
8
- return ElementExistsBlockSchema.parse(data);
9
- }
10
- export async function handlerElementExists(data) {
11
- try {
12
- const { selector = "", findBy = "cssSelector", option } = data;
13
- if (!selector) {
14
- throw new Error("Selector is required for element-exists block");
15
- }
16
- const element = await findElement({ selector, findBy, option });
17
- const exists = element !== null;
18
- return { data: exists };
19
- } catch (error) {
20
- return {
21
- hasError: true,
22
- message: error instanceof Error ? error.message : "Unknown error in element-exists handler",
23
- data: null
24
- };
25
- }
26
- }
@@ -1,143 +0,0 @@
1
- import z from "/vendor/.vite-deps-zod.js__v--13965f26.js";
2
- import { BaseBlockSchema } from "/src/blocks/types.ts.js";
3
- import { findElement } from "/src/content/elements/index.ts.js";
4
- export const EventClickBlockSchema = BaseBlockSchema.extend({
5
- name: z.literal("event-click"),
6
- textFilter: z.object({
7
- text: z.union([z.string(), z.array(z.string())]),
8
- mode: z.enum(["exact", "contains", "startsWith", "endsWith", "regex"])
9
- }).optional()
10
- });
11
- export function validateEventClickBlock(data) {
12
- return EventClickBlockSchema.parse(data);
13
- }
14
- export async function handlerEventClick(data) {
15
- try {
16
- const { selector = "", findBy = "cssSelector", option, textFilter } = data;
17
- if (!selector) {
18
- throw new Error("Selector is required for event-click block");
19
- }
20
- const elements = await findElement({ selector, findBy, option });
21
- if (!elements) {
22
- throw new Error("Element not found for clicking");
23
- }
24
- let targetElement;
25
- if (Array.isArray(elements)) {
26
- if (textFilter) {
27
- const filteredElement = selectElementByText(elements, textFilter.text, textFilter.mode);
28
- if (!filteredElement) {
29
- const textDisplay = Array.isArray(textFilter.text) ? textFilter.text.join(", ") : textFilter.text;
30
- throw new Error(`No element found with text filter: "${textDisplay}" (mode: ${textFilter.mode})`);
31
- }
32
- targetElement = filteredElement;
33
- } else {
34
- targetElement = elements[0];
35
- }
36
- } else {
37
- targetElement = elements;
38
- }
39
- await simulateClickElement(targetElement);
40
- return { data: true };
41
- } catch (error) {
42
- console.log(error);
43
- return {
44
- hasError: true,
45
- message: error instanceof Error ? error.message : "Unknown error in event-click handler",
46
- data: false
47
- };
48
- }
49
- }
50
- function selectElementByText(elements, textFilter, mode) {
51
- for (const element of elements) {
52
- const text = getElementText(element);
53
- const textsToCheck = Array.isArray(textFilter) ? textFilter : [textFilter];
54
- for (const filterText of textsToCheck) {
55
- let matches = false;
56
- switch (mode) {
57
- case "exact":
58
- matches = text === filterText;
59
- break;
60
- case "contains":
61
- matches = text.includes(filterText);
62
- break;
63
- case "startsWith":
64
- matches = text.startsWith(filterText);
65
- break;
66
- case "endsWith":
67
- matches = text.endsWith(filterText);
68
- break;
69
- case "regex":
70
- try {
71
- const regex = new RegExp(filterText);
72
- matches = regex.test(text);
73
- } catch (error) {
74
- console.warn("Invalid regex pattern:", filterText);
75
- continue;
76
- }
77
- break;
78
- }
79
- if (matches) {
80
- return element;
81
- }
82
- }
83
- }
84
- return null;
85
- }
86
- function getElementText(element) {
87
- if (element.innerText) {
88
- return element.innerText.trim();
89
- }
90
- if (element.textContent) {
91
- return element.textContent.trim();
92
- }
93
- if (element instanceof HTMLInputElement && element.value) {
94
- return element.value.trim();
95
- }
96
- const placeholder = element.getAttribute("placeholder");
97
- if (placeholder) {
98
- return placeholder.trim();
99
- }
100
- const title = element.getAttribute("title");
101
- if (title) {
102
- return title.trim();
103
- }
104
- const ariaLabel = element.getAttribute("aria-label");
105
- if (ariaLabel) {
106
- return ariaLabel.trim();
107
- }
108
- return "";
109
- }
110
- async function simulateClickElement(element) {
111
- element.scrollIntoView({
112
- behavior: "instant",
113
- block: "center",
114
- inline: "center"
115
- });
116
- await new Promise((resolve) => setTimeout(resolve, 50));
117
- if (element.focus) {
118
- element.focus();
119
- }
120
- const rect = element.getBoundingClientRect();
121
- const centerX = rect.left + rect.width / 2;
122
- const centerY = rect.top + rect.height / 2;
123
- try {
124
- const response = await chrome.runtime.sendMessage({
125
- type: "CDP_CLICK",
126
- data: {
127
- x: centerX,
128
- y: centerY
129
- }
130
- });
131
- if (response && !response.$isError) {
132
- console.log("[EventClick] CDP click successful:", response);
133
- } else {
134
- throw new Error(response?.message || "CDP click failed");
135
- }
136
- } catch (error) {
137
- console.error("[EventClick] CDP click failed, falling back to native click:", error);
138
- if (element.click) {
139
- element.click();
140
- }
141
- }
142
- await new Promise((resolve) => setTimeout(resolve, 50));
143
- }
@@ -1,50 +0,0 @@
1
- import z from "/vendor/.vite-deps-zod.js__v--13965f26.js";
2
- export const FetchApiBlockSchema = z.object({
3
- name: z.literal("fetch-api"),
4
- url: z.string(),
5
- method: z.enum(["GET", "POST", "PUT", "DELETE", "PATCH", "HEAD", "OPTIONS"]).optional(),
6
- headers: z.record(z.string(), z.string()).optional(),
7
- body: z.record(z.string(), z.any()).optional(),
8
- timeout: z.number().min(0).optional(),
9
- parseJson: z.boolean().optional(),
10
- returnHeaders: z.boolean().optional()
11
- });
12
- export function validateFetchApiBlock(data) {
13
- return FetchApiBlockSchema.parse(data);
14
- }
15
- export async function handlerFetchApi(data) {
16
- try {
17
- console.log("[FetchApiBlock] Sending API request to background:", data.url);
18
- const response = await chrome.runtime.sendMessage({
19
- type: "FETCH_API",
20
- data: {
21
- url: data.url,
22
- method: data.method || "GET",
23
- headers: data.headers || {},
24
- body: data.body,
25
- timeout: data.timeout || 3e4,
26
- parseJson: data.parseJson ?? true,
27
- // 기본값 true
28
- returnHeaders: data.returnHeaders ?? false
29
- }
30
- });
31
- if (response.$isError) {
32
- return {
33
- hasError: true,
34
- message: response.message || "API request failed",
35
- data: void 0
36
- };
37
- }
38
- console.log("[FetchApiBlock] API request successful");
39
- return {
40
- data: response.data
41
- };
42
- } catch (error) {
43
- console.error("[FetchApiBlock] API request error:", error);
44
- return {
45
- hasError: true,
46
- message: error instanceof Error ? error.message : "Unknown error in API request",
47
- data: void 0
48
- };
49
- }
50
- }
@@ -1,33 +0,0 @@
1
- import z from "/vendor/.vite-deps-zod.js__v--13965f26.js";
2
- import { BaseBlockSchema } from "/src/blocks/types.ts.js";
3
- import { findElement } from "/src/content/elements/index.ts.js";
4
- export const GetAttributeValueBlockSchema = BaseBlockSchema.extend({
5
- name: z.literal("attribute-value"),
6
- attributeName: z.string()
7
- });
8
- export function validateGetAttributeValueBlock(data) {
9
- return GetAttributeValueBlockSchema.parse(data);
10
- }
11
- export async function handlerGetAttributeValue(data) {
12
- try {
13
- const { selector = "", attributeName, findBy = "cssSelector", option } = data;
14
- if (!selector) {
15
- throw new Error("Selector is required for attribute-value block");
16
- }
17
- if (!attributeName) {
18
- throw new Error("Attribute name is required for attribute-value block");
19
- }
20
- const elements = await findElement({ selector, findBy, option });
21
- if (!elements) {
22
- return { data: null };
23
- }
24
- const result = Array.isArray(elements) ? elements.map((element) => element.getAttribute(attributeName)).filter((value) => value !== null) : elements.getAttribute(attributeName);
25
- return { data: result };
26
- } catch (error) {
27
- return {
28
- hasError: true,
29
- message: error instanceof Error ? error.message : "Unknown error in attribute-value handler",
30
- data: null
31
- };
32
- }
33
- }
@@ -1,114 +0,0 @@
1
- import z from "/vendor/.vite-deps-zod.js__v--13965f26.js";
2
- import { BaseBlockSchema } from "/src/blocks/types.ts.js";
3
- import { findElement } from "/src/content/elements/index.ts.js";
4
- import { CSSSelectorGenerator } from "/src/content/elements/utils/CSSSelectorGenerator.ts.js";
5
- import { XPathGenerator } from "/src/content/elements/utils/XPathGenerator.ts.js";
6
- export const GetElementDataBlockSchema = BaseBlockSchema.extend({
7
- name: z.literal("get-element-data"),
8
- includeText: z.boolean().optional(),
9
- includeTags: z.boolean().optional(),
10
- useTextContent: z.boolean().optional(),
11
- regex: z.string().optional(),
12
- prefixText: z.string().optional(),
13
- suffixText: z.string().optional(),
14
- attributes: z.array(z.string()).optional(),
15
- includeSelector: z.boolean().optional(),
16
- includeXPath: z.boolean().optional()
17
- });
18
- export function validateGetElementDataBlock(data) {
19
- return GetElementDataBlockSchema.parse(data);
20
- }
21
- export async function handlerGetElementData(data) {
22
- try {
23
- const {
24
- selector = "",
25
- includeText = true,
26
- includeTags = false,
27
- useTextContent = false,
28
- regex,
29
- prefixText = "",
30
- suffixText = "",
31
- attributes = [],
32
- includeSelector = false,
33
- includeXPath = false,
34
- findBy = "cssSelector",
35
- option
36
- } = data;
37
- if (!selector) {
38
- throw new Error("Selector is required for get-element-data block");
39
- }
40
- if (!includeText && attributes.length === 0) {
41
- throw new Error("Either includeText must be true or attributes must be provided");
42
- }
43
- const elements = await findElement({ selector, findBy, option });
44
- if (!elements) {
45
- return { data: Array.isArray(elements) ? [] : {} };
46
- }
47
- const extractElementData = createElementDataExtractor(
48
- includeText,
49
- includeTags,
50
- useTextContent,
51
- regex,
52
- prefixText,
53
- suffixText,
54
- attributes,
55
- includeSelector,
56
- includeXPath
57
- );
58
- if (Array.isArray(elements)) {
59
- const elementsData = elements.map(extractElementData);
60
- return { data: elementsData };
61
- } else {
62
- const elementData = extractElementData(elements);
63
- return { data: elementData };
64
- }
65
- } catch (error) {
66
- return {
67
- hasError: true,
68
- message: error instanceof Error ? error.message : "Unknown error in get-element-data handler",
69
- data: {}
70
- };
71
- }
72
- }
73
- function createElementDataExtractor(includeText, includeTags, useTextContent, regex, prefixText = "", suffixText = "", attributes = [], includeSelector = false, includeXPath = false) {
74
- return (element) => {
75
- const result = {};
76
- if (includeText) {
77
- let text = "";
78
- if (includeTags) {
79
- text = element.innerHTML;
80
- } else {
81
- text = useTextContent ? element.textContent || "" : element.innerText || "";
82
- }
83
- if (regex) {
84
- const regexPattern = new RegExp(regex, "g");
85
- const matches = text.match(regexPattern);
86
- text = matches ? matches.join(" ") : "";
87
- }
88
- if (text.trim()) {
89
- text = prefixText + text.trim() + suffixText;
90
- }
91
- result.text = text;
92
- }
93
- if (attributes.length > 0) {
94
- result.attributes = {};
95
- attributes.forEach((attrName) => {
96
- let attrValue = element.getAttribute(attrName);
97
- if (attrValue === null) {
98
- const childWithAttr = element.querySelector(`[${attrName}]`);
99
- if (childWithAttr) {
100
- attrValue = childWithAttr.getAttribute(attrName);
101
- }
102
- }
103
- result.attributes[attrName] = attrValue;
104
- });
105
- }
106
- if (includeSelector) {
107
- result.selector = CSSSelectorGenerator.generate(element);
108
- }
109
- if (includeXPath) {
110
- result.xpath = XPathGenerator.generate(element);
111
- }
112
- return result;
113
- };
114
- }