testdriverai 6.2.2 → 7.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.
- package/.github/workflows/acceptance-linux.yml +75 -0
- package/.github/workflows/acceptance-sdk-tests.yml +133 -0
- package/.vscode/settings.json +5 -1
- package/AGENTS.md +550 -0
- package/CODEOWNERS +0 -1
- package/README.md +126 -0
- package/{testdriver → _testdriver}/acceptance/drag-and-drop.yaml +2 -2
- package/{testdriver → _testdriver}/acceptance/snippets/login.yaml +1 -1
- package/_testdriver/examples/desktop/lifecycle/prerun.yaml +0 -0
- package/{testdriver → _testdriver}/examples/web/lifecycle/prerun.yaml +6 -1
- package/{testdriver → _testdriver}/lifecycle/postrun.yaml +3 -2
- package/_testdriver/lifecycle/prerun.yaml +15 -0
- package/{testdriver → _testdriver}/lifecycle/provision.yaml +7 -2
- package/agent/index.js +300 -85
- package/agent/interface.js +15 -0
- package/agent/lib/cache.js +142 -0
- package/agent/lib/commander.js +1 -39
- package/agent/lib/commands.js +910 -296
- package/agent/lib/redraw.js +129 -41
- package/agent/lib/sandbox.js +29 -6
- package/agent/lib/sdk.js +22 -0
- package/agent/lib/system.js +0 -3
- package/agent/lib/validation.js +1 -7
- package/debug-locate-response.js +82 -0
- package/debugger/index.html +15 -4
- package/docs/ARCHITECTURE.md +424 -0
- package/docs/AWESOME_LOGS_QUICK_REF.md +100 -0
- package/docs/MIGRATION.md +425 -0
- package/docs/PRESETS.md +210 -0
- package/docs/QUICK_START_TEST_RECORDING.md +215 -0
- package/docs/SDK_AWESOME_LOGS.md +468 -0
- package/docs/TEST_RECORDING.md +388 -0
- package/docs/docs.json +286 -152
- package/docs/guide/best-practices-polling.mdx +154 -0
- package/docs/sdk-browser-rendering.md +167 -0
- package/docs/v6/getting-started/self-hosting.mdx +407 -0
- package/docs/{guide → v6/guide}/dashcam.mdx +1 -1
- package/docs/{guide → v6/guide}/environment-variables.mdx +4 -5
- package/docs/{guide → v6/guide}/lifecycle.mdx +1 -1
- package/docs/v6/overview/comparison.mdx +101 -0
- package/docs/v7/README.md +135 -0
- package/docs/v7/api/ai.mdx +205 -0
- package/docs/v7/api/assert.mdx +285 -0
- package/docs/v7/api/assertions.mdx +403 -0
- package/docs/v7/api/click.mdx +287 -0
- package/docs/v7/api/client.mdx +322 -0
- package/docs/v7/api/dashcam.mdx +497 -0
- package/docs/v7/api/doubleClick.mdx +102 -0
- package/docs/v7/api/elements.mdx +479 -0
- package/docs/v7/api/exec.mdx +346 -0
- package/docs/v7/api/find.mdx +316 -0
- package/docs/v7/api/focusApplication.mdx +294 -0
- package/docs/v7/api/hover.mdx +279 -0
- package/docs/v7/api/mouseDown.mdx +161 -0
- package/docs/v7/api/mouseUp.mdx +164 -0
- package/docs/v7/api/pressKeys.mdx +349 -0
- package/docs/v7/api/rightClick.mdx +123 -0
- package/docs/v7/api/sandbox.mdx +404 -0
- package/docs/v7/api/scroll.mdx +300 -0
- package/docs/v7/api/type.mdx +314 -0
- package/docs/v7/commands/assert.mdx +45 -0
- package/docs/v7/commands/exec.mdx +282 -0
- package/docs/v7/commands/focus-application.mdx +44 -0
- package/docs/v7/commands/hover-image.mdx +69 -0
- package/docs/v7/commands/hover-text.mdx +47 -0
- package/docs/v7/commands/if.mdx +53 -0
- package/docs/v7/commands/match-image.mdx +67 -0
- package/docs/v7/commands/press-keys.mdx +87 -0
- package/docs/v7/commands/remember.mdx +49 -0
- package/docs/v7/commands/run.mdx +44 -0
- package/docs/v7/commands/scroll-until-image.mdx +66 -0
- package/docs/v7/commands/scroll-until-text.mdx +60 -0
- package/docs/v7/commands/scroll.mdx +69 -0
- package/docs/v7/commands/type.mdx +45 -0
- package/docs/v7/commands/wait-for-image.mdx +54 -0
- package/docs/v7/commands/wait-for-text.mdx +48 -0
- package/docs/v7/commands/wait.mdx +45 -0
- package/docs/v7/getting-started/configuration.mdx +380 -0
- package/docs/v7/getting-started/quickstart.mdx +332 -0
- package/docs/v7/guides/best-practices.mdx +486 -0
- package/docs/v7/guides/caching-ai.mdx +215 -0
- package/docs/v7/guides/caching-selectors.mdx +292 -0
- package/docs/v7/guides/caching.mdx +366 -0
- package/docs/v7/guides/ci-cd/azure.mdx +587 -0
- package/docs/v7/guides/ci-cd/circleci.mdx +523 -0
- package/docs/v7/guides/ci-cd/github-actions.mdx +457 -0
- package/docs/v7/guides/ci-cd/gitlab.mdx +498 -0
- package/docs/v7/guides/ci-cd/jenkins.mdx +664 -0
- package/docs/v7/guides/ci-cd/travis.mdx +438 -0
- package/docs/v7/guides/debugging.mdx +349 -0
- package/docs/v7/guides/faq.mdx +393 -0
- package/docs/v7/guides/migration.mdx +562 -0
- package/docs/v7/guides/performance.mdx +517 -0
- package/docs/{getting-started → v7/guides}/self-hosting.mdx +11 -12
- package/docs/v7/guides/troubleshooting.mdx +526 -0
- package/docs/v7/guides/vitest-plugin.mdx +477 -0
- package/docs/v7/guides/vitest.mdx +535 -0
- package/docs/v7/platforms/linux.mdx +308 -0
- package/docs/v7/platforms/macos.mdx +433 -0
- package/docs/v7/platforms/windows.mdx +430 -0
- package/docs/v7/playwright.mdx +342 -0
- package/docs/v7/presets/chrome-extension.mdx +223 -0
- package/docs/v7/presets/chrome.mdx +287 -0
- package/docs/v7/presets/electron.mdx +435 -0
- package/docs/v7/presets/vscode.mdx +398 -0
- package/docs/v7/presets/webapp.mdx +396 -0
- package/docs/v7/progressive-apis/CORE.md +459 -0
- package/docs/v7/progressive-apis/HOOKS.md +360 -0
- package/docs/v7/progressive-apis/PROGRESSIVE_DISCLOSURE.md +230 -0
- package/docs/v7/progressive-apis/PROVISION.md +266 -0
- package/eslint.config.js +19 -1
- package/interfaces/cli/lib/base.js +10 -4
- package/interfaces/logger.js +2 -1
- package/interfaces/shared-test-state.mjs +69 -0
- package/interfaces/vitest-plugin.mjs +830 -0
- package/package.json +29 -5
- package/schema.json +8 -29
- package/scripts/view-test-results.mjs +96 -0
- package/sdk-log-formatter.js +714 -0
- package/sdk.d.ts +1028 -0
- package/sdk.js +2567 -0
- package/{.github/workflows/self-hosted.yml → self-hosted.yml} +13 -4
- package/setup/aws/cloudformation.yaml +9 -2
- package/src/core/Dashcam.js +469 -0
- package/src/core/index.d.ts +150 -0
- package/src/core/index.js +12 -0
- package/src/presets/index.mjs +331 -0
- package/src/vitest/extended.mjs +108 -0
- package/src/vitest/hooks.d.ts +119 -0
- package/src/vitest/hooks.mjs +298 -0
- package/src/vitest/index.mjs +64 -0
- package/src/vitest/lifecycle.mjs +277 -0
- package/src/vitest/utils.mjs +150 -0
- package/test/dashcam.test.js +137 -0
- package/test/mcp-example-test.yaml +27 -0
- package/testdriver/acceptance-sdk/QUICK_REFERENCE.md +61 -0
- package/testdriver/acceptance-sdk/README.md +128 -0
- package/testdriver/acceptance-sdk/TEST_REPORTING.md +245 -0
- package/testdriver/acceptance-sdk/assert.test.mjs +26 -0
- package/testdriver/acceptance-sdk/auto-cache-key-demo.test.mjs +56 -0
- package/testdriver/acceptance-sdk/chrome-extension.test.mjs +89 -0
- package/testdriver/acceptance-sdk/drag-and-drop.test.mjs +58 -0
- package/testdriver/acceptance-sdk/element-not-found.test.mjs +25 -0
- package/testdriver/acceptance-sdk/exec-js.test.mjs +43 -0
- package/testdriver/acceptance-sdk/exec-output.test.mjs +59 -0
- package/testdriver/acceptance-sdk/exec-pwsh.test.mjs +57 -0
- package/testdriver/acceptance-sdk/focus-window.test.mjs +36 -0
- package/testdriver/acceptance-sdk/formatted-logging.test.mjs +26 -0
- package/testdriver/acceptance-sdk/hooks-example.test.mjs +38 -0
- package/testdriver/acceptance-sdk/hover-image.test.mjs +34 -0
- package/testdriver/acceptance-sdk/hover-text-with-description.test.mjs +38 -0
- package/testdriver/acceptance-sdk/hover-text.test.mjs +27 -0
- package/testdriver/acceptance-sdk/match-image.test.mjs +36 -0
- package/testdriver/acceptance-sdk/presets-example.test.mjs +87 -0
- package/testdriver/acceptance-sdk/press-keys.test.mjs +50 -0
- package/testdriver/acceptance-sdk/prompt.test.mjs +33 -0
- package/testdriver/acceptance-sdk/scroll-keyboard.test.mjs +38 -0
- package/testdriver/acceptance-sdk/scroll-until-image.test.mjs +39 -0
- package/testdriver/acceptance-sdk/scroll-until-text.test.mjs +28 -0
- package/testdriver/acceptance-sdk/scroll.test.mjs +41 -0
- package/testdriver/acceptance-sdk/setup/globalTeardown.mjs +11 -0
- package/testdriver/acceptance-sdk/setup/testHelpers.mjs +420 -0
- package/testdriver/acceptance-sdk/setup/vitestSetup.mjs +40 -0
- package/testdriver/acceptance-sdk/sully-ai.test.mjs +234 -0
- package/testdriver/acceptance-sdk/test-console-logs.test.mjs +42 -0
- package/testdriver/acceptance-sdk/type-checking-demo.js +49 -0
- package/testdriver/acceptance-sdk/type.test.mjs +45 -0
- package/verify-element-api.js +89 -0
- package/verify-types.js +0 -0
- package/vitest.config.example.js +19 -0
- package/vitest.config.mjs +66 -0
- package/vitest.config.mjs.bak +44 -0
- package/.github/workflows/acceptance-v6.yml +0 -169
- package/.vscode/mcp.json +0 -9
- package/docs/overview/comparison.mdx +0 -82
- package/testdriver/lifecycle/prerun.yaml +0 -17
- /package/{testdriver/examples/desktop/lifecycle/prerun.yaml → .env.example} +0 -0
- /package/{testdriver → _testdriver}/acceptance/assert.yaml +0 -0
- /package/{testdriver → _testdriver}/acceptance/dashcam.yaml +0 -0
- /package/{testdriver → _testdriver}/acceptance/embed.yaml +0 -0
- /package/{testdriver → _testdriver}/acceptance/exec-js.yaml +0 -0
- /package/{testdriver → _testdriver}/acceptance/exec-output.yaml +0 -0
- /package/{testdriver → _testdriver}/acceptance/exec-shell.yaml +0 -0
- /package/{testdriver → _testdriver}/acceptance/focus-window.yaml +0 -0
- /package/{testdriver → _testdriver}/acceptance/hover-image.yaml +0 -0
- /package/{testdriver → _testdriver}/acceptance/hover-text-with-description.yaml +0 -0
- /package/{testdriver → _testdriver}/acceptance/hover-text.yaml +0 -0
- /package/{testdriver → _testdriver}/acceptance/if-else.yaml +0 -0
- /package/{testdriver → _testdriver}/acceptance/match-image.yaml +0 -0
- /package/{testdriver → _testdriver}/acceptance/press-keys.yaml +0 -0
- /package/{testdriver → _testdriver}/acceptance/prompt.yaml +0 -0
- /package/{testdriver → _testdriver}/acceptance/remember.yaml +0 -0
- /package/{testdriver → _testdriver}/acceptance/screenshots/cart.png +0 -0
- /package/{testdriver → _testdriver}/acceptance/scroll-keyboard.yaml +0 -0
- /package/{testdriver → _testdriver}/acceptance/scroll-until-image.yaml +0 -0
- /package/{testdriver → _testdriver}/acceptance/scroll-until-text.yaml +0 -0
- /package/{testdriver → _testdriver}/acceptance/scroll.yaml +0 -0
- /package/{testdriver → _testdriver}/acceptance/snippets/match-cart.yaml +0 -0
- /package/{testdriver → _testdriver}/acceptance/type.yaml +0 -0
- /package/{testdriver → _testdriver}/behavior/failure.yaml +0 -0
- /package/{testdriver → _testdriver}/behavior/hover-text.yaml +0 -0
- /package/{testdriver → _testdriver}/behavior/lifecycle/postrun.yaml +0 -0
- /package/{testdriver → _testdriver}/behavior/lifecycle/prerun.yaml +0 -0
- /package/{testdriver → _testdriver}/behavior/lifecycle/provision.yaml +0 -0
- /package/{testdriver → _testdriver}/behavior/secrets.yaml +0 -0
- /package/{testdriver → _testdriver}/edge-cases/dashcam-chrome.yaml +0 -0
- /package/{testdriver → _testdriver}/edge-cases/exec-pwsh-multiline.yaml +0 -0
- /package/{testdriver → _testdriver}/edge-cases/js-exception.yaml +0 -0
- /package/{testdriver → _testdriver}/edge-cases/js-promise.yaml +0 -0
- /package/{testdriver → _testdriver}/edge-cases/lifecycle/postrun.yaml +0 -0
- /package/{testdriver → _testdriver}/edge-cases/prompt-in-middle.yaml +0 -0
- /package/{testdriver → _testdriver}/edge-cases/prompt-nested.yaml +0 -0
- /package/{testdriver → _testdriver}/edge-cases/success-test.yaml +0 -0
- /package/{testdriver → _testdriver}/examples/android/example.yaml +0 -0
- /package/{testdriver → _testdriver}/examples/android/lifecycle/postrun.yaml +0 -0
- /package/{testdriver → _testdriver}/examples/android/lifecycle/provision.yaml +0 -0
- /package/{testdriver → _testdriver}/examples/android/readme.md +0 -0
- /package/{testdriver → _testdriver}/examples/chrome-extension/lifecycle/provision.yaml +0 -0
- /package/{testdriver → _testdriver}/examples/desktop/lifecycle/provision.yaml +0 -0
- /package/{testdriver → _testdriver}/examples/vscode-extension/lifecycle/provision.yaml +0 -0
- /package/{testdriver → _testdriver}/examples/web/lifecycle/postrun.yaml +0 -0
- /package/docs/{account → v6/account}/dashboard.mdx +0 -0
- /package/docs/{account → v6/account}/enterprise.mdx +0 -0
- /package/docs/{account → v6/account}/pricing.mdx +0 -0
- /package/docs/{account → v6/account}/projects.mdx +0 -0
- /package/docs/{account → v6/account}/team.mdx +0 -0
- /package/docs/{action → v6/action}/ami.mdx +0 -0
- /package/docs/{action → v6/action}/performance.mdx +0 -0
- /package/docs/{action → v6/action}/secrets.mdx +0 -0
- /package/docs/{apps → v6/apps}/chrome-extensions.mdx +0 -0
- /package/docs/{apps → v6/apps}/desktop-apps.mdx +0 -0
- /package/docs/{apps → v6/apps}/mobile-apps.mdx +0 -0
- /package/docs/{apps → v6/apps}/static-websites.mdx +0 -0
- /package/docs/{apps → v6/apps}/tauri-apps.mdx +0 -0
- /package/docs/{bugs → v6/bugs}/jira.mdx +0 -0
- /package/docs/{cli → v6/cli}/overview.mdx +0 -0
- /package/docs/{commands → v6/commands}/assert.mdx +0 -0
- /package/docs/{commands → v6/commands}/exec.mdx +0 -0
- /package/docs/{commands → v6/commands}/focus-application.mdx +0 -0
- /package/docs/{commands → v6/commands}/hover-image.mdx +0 -0
- /package/docs/{commands → v6/commands}/hover-text.mdx +0 -0
- /package/docs/{commands → v6/commands}/if.mdx +0 -0
- /package/docs/{commands → v6/commands}/match-image.mdx +0 -0
- /package/docs/{commands → v6/commands}/press-keys.mdx +0 -0
- /package/docs/{commands → v6/commands}/remember.mdx +0 -0
- /package/docs/{commands → v6/commands}/run.mdx +0 -0
- /package/docs/{commands → v6/commands}/scroll-until-image.mdx +0 -0
- /package/docs/{commands → v6/commands}/scroll-until-text.mdx +0 -0
- /package/docs/{commands → v6/commands}/scroll.mdx +0 -0
- /package/docs/{commands → v6/commands}/type.mdx +0 -0
- /package/docs/{commands → v6/commands}/wait-for-image.mdx +0 -0
- /package/docs/{commands → v6/commands}/wait-for-text.mdx +0 -0
- /package/docs/{commands → v6/commands}/wait.mdx +0 -0
- /package/docs/{exporting → v6/exporting}/junit.mdx +0 -0
- /package/docs/{exporting → v6/exporting}/playwright.mdx +0 -0
- /package/docs/{features → v6/features}/auto-healing.mdx +0 -0
- /package/docs/{features → v6/features}/generation.mdx +0 -0
- /package/docs/{features → v6/features}/parallel-testing.mdx +0 -0
- /package/docs/{features → v6/features}/reusable-snippets.mdx +0 -0
- /package/docs/{features → v6/features}/selectorless.mdx +0 -0
- /package/docs/{features → v6/features}/visual-assertions.mdx +0 -0
- /package/docs/{getting-started → v6/getting-started}/ci.mdx +0 -0
- /package/docs/{getting-started → v6/getting-started}/cli.mdx +0 -0
- /package/docs/{getting-started → v6/getting-started}/editing.mdx +0 -0
- /package/docs/{getting-started → v6/getting-started}/playwright.mdx +0 -0
- /package/docs/{getting-started → v6/getting-started}/running.mdx +0 -0
- /package/docs/{getting-started → v6/getting-started}/vscode.mdx +0 -0
- /package/docs/{guide → v6/guide}/assertions.mdx +0 -0
- /package/docs/{guide → v6/guide}/authentication.mdx +0 -0
- /package/docs/{guide → v6/guide}/code.mdx +0 -0
- /package/docs/{guide → v6/guide}/locating.mdx +0 -0
- /package/docs/{guide → v6/guide}/protips.mdx +0 -0
- /package/docs/{guide → v6/guide}/variables.mdx +0 -0
- /package/docs/{guide → v6/guide}/waiting.mdx +0 -0
- /package/docs/{importing → v6/importing}/csv.mdx +0 -0
- /package/docs/{importing → v6/importing}/gherkin.mdx +0 -0
- /package/docs/{importing → v6/importing}/jira.mdx +0 -0
- /package/docs/{importing → v6/importing}/testrail.mdx +0 -0
- /package/docs/{integrations → v6/integrations}/electron.mdx +0 -0
- /package/docs/{integrations → v6/integrations}/netlify.mdx +0 -0
- /package/docs/{integrations → v6/integrations}/vercel.mdx +0 -0
- /package/docs/{interactive → v6/interactive}/explore.mdx +0 -0
- /package/docs/{interactive → v6/interactive}/run.mdx +0 -0
- /package/docs/{interactive → v6/interactive}/save.mdx +0 -0
- /package/docs/{overview → v6/overview}/faq.mdx +0 -0
- /package/docs/{overview → v6/overview}/performance.mdx +0 -0
- /package/docs/{overview → v6/overview}/quickstart.mdx +0 -0
- /package/docs/{overview → v6/overview}/what-is-testdriver.mdx +0 -0
- /package/docs/{scenarios → v6/scenarios}/ai-chatbot.mdx +0 -0
- /package/docs/{scenarios → v6/scenarios}/cookie-banner.mdx +0 -0
- /package/docs/{scenarios → v6/scenarios}/file-upload.mdx +0 -0
- /package/docs/{scenarios → v6/scenarios}/form-filling.mdx +0 -0
- /package/docs/{scenarios → v6/scenarios}/log-in.mdx +0 -0
- /package/docs/{scenarios → v6/scenarios}/pdf-generation.mdx +0 -0
- /package/docs/{scenarios → v6/scenarios}/spell-check.mdx +0 -0
- /package/docs/{security → v6/security}/action.mdx +0 -0
- /package/docs/{security → v6/security}/agent.mdx +0 -0
- /package/docs/{security → v6/security}/platform.mdx +0 -0
- /package/docs/{tutorials → v6/tutorials}/advanced-test.mdx +0 -0
- /package/docs/{tutorials → v6/tutorials}/basic-test.mdx +0 -0
|
@@ -0,0 +1,479 @@
|
|
|
1
|
+
---
|
|
2
|
+
title: "Element Finding"
|
|
3
|
+
sidebarTitle: "Elements"
|
|
4
|
+
description: "Locate and interact with UI elements using AI"
|
|
5
|
+
icon: "mouse-pointer"
|
|
6
|
+
---
|
|
7
|
+
|
|
8
|
+
## Overview
|
|
9
|
+
|
|
10
|
+
TestDriver's element finding system uses AI to locate elements on screen using natural language descriptions. The `find()` method returns an `Element` object that you can interact with.
|
|
11
|
+
|
|
12
|
+
## Finding Elements
|
|
13
|
+
|
|
14
|
+
### find()
|
|
15
|
+
|
|
16
|
+
Locate an element on screen using a natural language description.
|
|
17
|
+
|
|
18
|
+
```javascript
|
|
19
|
+
const element = await testdriver.find(description)
|
|
20
|
+
```
|
|
21
|
+
|
|
22
|
+
**Parameters:**
|
|
23
|
+
- `description` (string) - Natural language description of the element to find
|
|
24
|
+
|
|
25
|
+
**Returns:** `Promise<Element>` - Element instance that has been located
|
|
26
|
+
|
|
27
|
+
**Example:**
|
|
28
|
+
```javascript
|
|
29
|
+
// Find a button
|
|
30
|
+
const submitButton = await testdriver.find('the submit button');
|
|
31
|
+
|
|
32
|
+
// Find an input field with context
|
|
33
|
+
const emailField = await testdriver.find('email input field in the login form');
|
|
34
|
+
|
|
35
|
+
// Find an element by visual characteristics
|
|
36
|
+
const redButton = await testdriver.find('red button in the top right corner');
|
|
37
|
+
```
|
|
38
|
+
|
|
39
|
+
<Tip>
|
|
40
|
+
Be specific in your descriptions. Include visual details, location context, or nearby text to improve accuracy.
|
|
41
|
+
</Tip>
|
|
42
|
+
|
|
43
|
+
## Element Class
|
|
44
|
+
|
|
45
|
+
The `Element` class represents a located (or to-be-located) UI element. It provides methods for interaction and properties for element information.
|
|
46
|
+
|
|
47
|
+
### Methods
|
|
48
|
+
|
|
49
|
+
#### found()
|
|
50
|
+
|
|
51
|
+
Check if the element was successfully located.
|
|
52
|
+
|
|
53
|
+
```javascript
|
|
54
|
+
element.found()
|
|
55
|
+
```
|
|
56
|
+
|
|
57
|
+
**Returns:** `boolean` - True if element coordinates were found
|
|
58
|
+
|
|
59
|
+
**Example:**
|
|
60
|
+
```javascript
|
|
61
|
+
const element = await testdriver.find('login button');
|
|
62
|
+
if (element.found()) {
|
|
63
|
+
await element.click();
|
|
64
|
+
} else {
|
|
65
|
+
console.log('Element not found');
|
|
66
|
+
}
|
|
67
|
+
```
|
|
68
|
+
|
|
69
|
+
#### find()
|
|
70
|
+
|
|
71
|
+
Re-locate the element, optionally with a new description.
|
|
72
|
+
|
|
73
|
+
```javascript
|
|
74
|
+
await element.find(newDescription)
|
|
75
|
+
```
|
|
76
|
+
|
|
77
|
+
**Parameters:**
|
|
78
|
+
- `newDescription` (string, optional) - New description to search for
|
|
79
|
+
|
|
80
|
+
**Returns:** `Promise<Element>` - This element instance
|
|
81
|
+
|
|
82
|
+
**Example:**
|
|
83
|
+
```javascript
|
|
84
|
+
// Re-locate if the UI changed
|
|
85
|
+
const element = await testdriver.find('submit button');
|
|
86
|
+
// ... page updates ...
|
|
87
|
+
await element.find(); // Re-locate with same description
|
|
88
|
+
|
|
89
|
+
// Or update the description
|
|
90
|
+
await element.find('blue submit button'); // Now looking for blue button
|
|
91
|
+
```
|
|
92
|
+
|
|
93
|
+
#### click()
|
|
94
|
+
|
|
95
|
+
Click on the element.
|
|
96
|
+
|
|
97
|
+
```javascript
|
|
98
|
+
await element.click(action)
|
|
99
|
+
```
|
|
100
|
+
|
|
101
|
+
**Parameters:**
|
|
102
|
+
- `action` (string, optional) - Type of click: `'click'` (default), `'double-click'`, `'right-click'`, `'hover'`, `'mouseDown'`, `'mouseUp'`
|
|
103
|
+
|
|
104
|
+
**Returns:** `Promise<void>`
|
|
105
|
+
|
|
106
|
+
**Example:**
|
|
107
|
+
```javascript
|
|
108
|
+
const button = await testdriver.find('submit button');
|
|
109
|
+
await button.click(); // Regular click
|
|
110
|
+
|
|
111
|
+
const file = await testdriver.find('document.txt');
|
|
112
|
+
await file.click('double-click'); // Double-click
|
|
113
|
+
|
|
114
|
+
const menu = await testdriver.find('settings icon');
|
|
115
|
+
await menu.click('right-click'); // Right-click
|
|
116
|
+
```
|
|
117
|
+
|
|
118
|
+
<Note>
|
|
119
|
+
The element must be found before clicking. The `find()` method automatically locates the element.
|
|
120
|
+
</Note>
|
|
121
|
+
|
|
122
|
+
#### hover()
|
|
123
|
+
|
|
124
|
+
Hover over the element without clicking.
|
|
125
|
+
|
|
126
|
+
```javascript
|
|
127
|
+
await element.hover()
|
|
128
|
+
```
|
|
129
|
+
|
|
130
|
+
**Returns:** `Promise<void>`
|
|
131
|
+
|
|
132
|
+
**Example:**
|
|
133
|
+
```javascript
|
|
134
|
+
const tooltip = await testdriver.find('info icon');
|
|
135
|
+
await tooltip.hover();
|
|
136
|
+
// Wait to see tooltip
|
|
137
|
+
await new Promise(resolve => setTimeout(resolve, 1000));
|
|
138
|
+
```
|
|
139
|
+
|
|
140
|
+
#### doubleClick()
|
|
141
|
+
|
|
142
|
+
Double-click on the element.
|
|
143
|
+
|
|
144
|
+
```javascript
|
|
145
|
+
await element.doubleClick()
|
|
146
|
+
```
|
|
147
|
+
|
|
148
|
+
**Returns:** `Promise<void>`
|
|
149
|
+
|
|
150
|
+
**Example:**
|
|
151
|
+
```javascript
|
|
152
|
+
const file = await testdriver.find('README.txt file icon');
|
|
153
|
+
await file.doubleClick();
|
|
154
|
+
```
|
|
155
|
+
|
|
156
|
+
#### rightClick()
|
|
157
|
+
|
|
158
|
+
Right-click on the element to open context menu.
|
|
159
|
+
|
|
160
|
+
```javascript
|
|
161
|
+
await element.rightClick()
|
|
162
|
+
```
|
|
163
|
+
|
|
164
|
+
**Returns:** `Promise<void>`
|
|
165
|
+
|
|
166
|
+
**Example:**
|
|
167
|
+
```javascript
|
|
168
|
+
const folder = await testdriver.find('Documents folder');
|
|
169
|
+
await folder.rightClick();
|
|
170
|
+
```
|
|
171
|
+
|
|
172
|
+
#### mouseDown() / mouseUp()
|
|
173
|
+
|
|
174
|
+
Press or release mouse button on the element (for drag operations).
|
|
175
|
+
|
|
176
|
+
```javascript
|
|
177
|
+
await element.mouseDown()
|
|
178
|
+
await element.mouseUp()
|
|
179
|
+
```
|
|
180
|
+
|
|
181
|
+
**Returns:** `Promise<void>`
|
|
182
|
+
|
|
183
|
+
**Example:**
|
|
184
|
+
```javascript
|
|
185
|
+
// Drag and drop
|
|
186
|
+
const item = await testdriver.find('draggable item');
|
|
187
|
+
await item.mouseDown();
|
|
188
|
+
|
|
189
|
+
// Move to drop target (using coordinates or another element)
|
|
190
|
+
const target = await testdriver.find('drop zone');
|
|
191
|
+
await target.hover();
|
|
192
|
+
await target.mouseUp();
|
|
193
|
+
```
|
|
194
|
+
|
|
195
|
+
### Properties
|
|
196
|
+
|
|
197
|
+
Element properties provide additional information about located elements:
|
|
198
|
+
|
|
199
|
+
#### coordinates
|
|
200
|
+
|
|
201
|
+
Get the element's coordinates object.
|
|
202
|
+
|
|
203
|
+
```javascript
|
|
204
|
+
const coords = element.getCoordinates()
|
|
205
|
+
// or access directly
|
|
206
|
+
element.coordinates
|
|
207
|
+
```
|
|
208
|
+
|
|
209
|
+
**Returns:** Object with `{ x, y, centerX, centerY }` or `null`
|
|
210
|
+
|
|
211
|
+
**Example:**
|
|
212
|
+
```javascript
|
|
213
|
+
const button = await testdriver.find('submit button');
|
|
214
|
+
const coords = button.coordinates;
|
|
215
|
+
console.log(`Top-left: (${coords.x}, ${coords.y})`);
|
|
216
|
+
console.log(`Center: (${coords.centerX}, ${coords.centerY})`);
|
|
217
|
+
```
|
|
218
|
+
|
|
219
|
+
#### x, y, centerX, centerY
|
|
220
|
+
|
|
221
|
+
Direct access to coordinate values.
|
|
222
|
+
|
|
223
|
+
```javascript
|
|
224
|
+
element.x // Top-left X coordinate
|
|
225
|
+
element.y // Top-left Y coordinate
|
|
226
|
+
element.centerX // Center X coordinate
|
|
227
|
+
element.centerY // Center Y coordinate
|
|
228
|
+
```
|
|
229
|
+
|
|
230
|
+
**Example:**
|
|
231
|
+
```javascript
|
|
232
|
+
const button = await testdriver.find('submit button');
|
|
233
|
+
console.log(`Button center: (${button.centerX}, ${button.centerY})`);
|
|
234
|
+
```
|
|
235
|
+
|
|
236
|
+
#### screenshot
|
|
237
|
+
|
|
238
|
+
Get a base64-encoded screenshot of the element (if available).
|
|
239
|
+
|
|
240
|
+
```javascript
|
|
241
|
+
element.screenshot
|
|
242
|
+
```
|
|
243
|
+
|
|
244
|
+
**Returns:** `string | null` - Base64-encoded image
|
|
245
|
+
|
|
246
|
+
**Example:**
|
|
247
|
+
```javascript
|
|
248
|
+
const element = await testdriver.find('error message');
|
|
249
|
+
if (element.screenshot) {
|
|
250
|
+
console.log('Element screenshot:', element.screenshot);
|
|
251
|
+
}
|
|
252
|
+
```
|
|
253
|
+
|
|
254
|
+
#### text
|
|
255
|
+
|
|
256
|
+
Get text content from the element (if available).
|
|
257
|
+
|
|
258
|
+
```javascript
|
|
259
|
+
element.text
|
|
260
|
+
```
|
|
261
|
+
|
|
262
|
+
**Returns:** `string | null`
|
|
263
|
+
|
|
264
|
+
**Example:**
|
|
265
|
+
```javascript
|
|
266
|
+
const message = await testdriver.find('notification message');
|
|
267
|
+
console.log('Message text:', message.text);
|
|
268
|
+
```
|
|
269
|
+
|
|
270
|
+
#### confidence
|
|
271
|
+
|
|
272
|
+
Get the AI's confidence score for the element match.
|
|
273
|
+
|
|
274
|
+
```javascript
|
|
275
|
+
element.confidence
|
|
276
|
+
```
|
|
277
|
+
|
|
278
|
+
**Returns:** `number | null` - Confidence score (0-1)
|
|
279
|
+
|
|
280
|
+
**Example:**
|
|
281
|
+
```javascript
|
|
282
|
+
const element = await testdriver.find('submit button');
|
|
283
|
+
if (element.confidence < 0.8) {
|
|
284
|
+
console.warn('Low confidence match:', element.confidence);
|
|
285
|
+
}
|
|
286
|
+
```
|
|
287
|
+
|
|
288
|
+
#### width, height
|
|
289
|
+
|
|
290
|
+
Get element dimensions (if available).
|
|
291
|
+
|
|
292
|
+
```javascript
|
|
293
|
+
element.width
|
|
294
|
+
element.height
|
|
295
|
+
```
|
|
296
|
+
|
|
297
|
+
**Returns:** `number | null`
|
|
298
|
+
|
|
299
|
+
#### boundingBox
|
|
300
|
+
|
|
301
|
+
Get the complete bounding box information.
|
|
302
|
+
|
|
303
|
+
```javascript
|
|
304
|
+
element.boundingBox
|
|
305
|
+
```
|
|
306
|
+
|
|
307
|
+
**Returns:** `Object | null` - Bounding box with coordinates and dimensions
|
|
308
|
+
|
|
309
|
+
#### label
|
|
310
|
+
|
|
311
|
+
Get the element's label or accessible name (if available).
|
|
312
|
+
|
|
313
|
+
```javascript
|
|
314
|
+
element.label
|
|
315
|
+
```
|
|
316
|
+
|
|
317
|
+
**Returns:** `string | null`
|
|
318
|
+
|
|
319
|
+
## Polling for Elements
|
|
320
|
+
|
|
321
|
+
Use polling to wait for elements that may not be immediately visible:
|
|
322
|
+
|
|
323
|
+
```javascript
|
|
324
|
+
// Poll until element appears
|
|
325
|
+
let loginButton;
|
|
326
|
+
const maxAttempts = 30;
|
|
327
|
+
|
|
328
|
+
for (let i = 0; i < maxAttempts; i++) {
|
|
329
|
+
loginButton = await testdriver.find('login button');
|
|
330
|
+
if (loginButton.found()) break;
|
|
331
|
+
await new Promise(resolve => setTimeout(resolve, 1000));
|
|
332
|
+
}
|
|
333
|
+
|
|
334
|
+
if (loginButton.found()) {
|
|
335
|
+
await loginButton.click();
|
|
336
|
+
} else {
|
|
337
|
+
throw new Error('Login button never appeared');
|
|
338
|
+
}
|
|
339
|
+
```
|
|
340
|
+
|
|
341
|
+
**Helper function for polling:**
|
|
342
|
+
```javascript
|
|
343
|
+
async function waitForElement(testdriver, description, timeout = 30000) {
|
|
344
|
+
const startTime = Date.now();
|
|
345
|
+
|
|
346
|
+
while (Date.now() - startTime < timeout) {
|
|
347
|
+
const element = await testdriver.find(description);
|
|
348
|
+
if (element.found()) return element;
|
|
349
|
+
await new Promise(resolve => setTimeout(resolve, 1000));
|
|
350
|
+
}
|
|
351
|
+
|
|
352
|
+
throw new Error(`Element "${description}" not found after ${timeout}ms`);
|
|
353
|
+
}
|
|
354
|
+
|
|
355
|
+
// Usage
|
|
356
|
+
const button = await waitForElement(testdriver, 'submit button', 10000);
|
|
357
|
+
await button.click();
|
|
358
|
+
```
|
|
359
|
+
|
|
360
|
+
## Examples
|
|
361
|
+
|
|
362
|
+
### Basic Element Interaction
|
|
363
|
+
|
|
364
|
+
```javascript
|
|
365
|
+
// Find and click
|
|
366
|
+
const submitButton = await testdriver.find('submit button');
|
|
367
|
+
await submitButton.click();
|
|
368
|
+
|
|
369
|
+
// Find, verify, then interact
|
|
370
|
+
const emailInput = await testdriver.find('email input field');
|
|
371
|
+
if (emailInput.found()) {
|
|
372
|
+
await emailInput.click();
|
|
373
|
+
await testdriver.type('user@example.com');
|
|
374
|
+
}
|
|
375
|
+
```
|
|
376
|
+
|
|
377
|
+
### Working with Forms
|
|
378
|
+
|
|
379
|
+
```javascript
|
|
380
|
+
// Fill out a multi-field form
|
|
381
|
+
const nameField = await testdriver.find('name input field');
|
|
382
|
+
await nameField.click();
|
|
383
|
+
await testdriver.type('John Doe');
|
|
384
|
+
|
|
385
|
+
const emailField = await testdriver.find('email input field');
|
|
386
|
+
await emailField.click();
|
|
387
|
+
await testdriver.type('john@example.com');
|
|
388
|
+
|
|
389
|
+
const submitButton = await testdriver.find('submit button');
|
|
390
|
+
await submitButton.click();
|
|
391
|
+
```
|
|
392
|
+
|
|
393
|
+
### Conditional Interactions
|
|
394
|
+
|
|
395
|
+
```javascript
|
|
396
|
+
// Check if element exists before interacting
|
|
397
|
+
const closeButton = await testdriver.find('close popup button');
|
|
398
|
+
|
|
399
|
+
if (closeButton.found()) {
|
|
400
|
+
await closeButton.click();
|
|
401
|
+
console.log('Popup closed');
|
|
402
|
+
} else {
|
|
403
|
+
console.log('No popup to close');
|
|
404
|
+
}
|
|
405
|
+
```
|
|
406
|
+
|
|
407
|
+
### Re-locating Dynamic Elements
|
|
408
|
+
|
|
409
|
+
```javascript
|
|
410
|
+
// Element that moves or changes
|
|
411
|
+
const notification = await testdriver.find('success notification');
|
|
412
|
+
|
|
413
|
+
// Do something that might cause it to move
|
|
414
|
+
await testdriver.scroll('down', 300);
|
|
415
|
+
|
|
416
|
+
// Re-locate the element
|
|
417
|
+
await notification.find();
|
|
418
|
+
|
|
419
|
+
if (notification.found()) {
|
|
420
|
+
await notification.click();
|
|
421
|
+
}
|
|
422
|
+
```
|
|
423
|
+
|
|
424
|
+
## Best Practices
|
|
425
|
+
|
|
426
|
+
<AccordionGroup>
|
|
427
|
+
<Accordion title="Be specific with descriptions">
|
|
428
|
+
Include visual details, position context, and nearby text:
|
|
429
|
+
|
|
430
|
+
```javascript
|
|
431
|
+
// ❌ Too vague
|
|
432
|
+
await testdriver.find('button');
|
|
433
|
+
|
|
434
|
+
// ✅ Specific
|
|
435
|
+
await testdriver.find('blue submit button below the email field');
|
|
436
|
+
```
|
|
437
|
+
</Accordion>
|
|
438
|
+
|
|
439
|
+
<Accordion title="Check if element was found">
|
|
440
|
+
Always verify elements were located before interacting:
|
|
441
|
+
|
|
442
|
+
```javascript
|
|
443
|
+
const element = await testdriver.find('submit button');
|
|
444
|
+
if (!element.found()) {
|
|
445
|
+
throw new Error('Submit button not found');
|
|
446
|
+
}
|
|
447
|
+
await element.click();
|
|
448
|
+
```
|
|
449
|
+
</Accordion>
|
|
450
|
+
|
|
451
|
+
<Accordion title="Use polling for dynamic content">
|
|
452
|
+
Don't assume elements are immediately visible:
|
|
453
|
+
|
|
454
|
+
```javascript
|
|
455
|
+
let element;
|
|
456
|
+
for (let i = 0; i < 30; i++) {
|
|
457
|
+
element = await testdriver.find('loading complete message');
|
|
458
|
+
if (element.found()) break;
|
|
459
|
+
await new Promise(resolve => setTimeout(resolve, 1000));
|
|
460
|
+
}
|
|
461
|
+
```
|
|
462
|
+
</Accordion>
|
|
463
|
+
|
|
464
|
+
<Accordion title="Reuse element references when possible">
|
|
465
|
+
If you need to interact with the same element multiple times, reuse the reference:
|
|
466
|
+
|
|
467
|
+
```javascript
|
|
468
|
+
const input = await testdriver.find('search input');
|
|
469
|
+
await input.click();
|
|
470
|
+
await testdriver.type('first search');
|
|
471
|
+
await testdriver.pressKeys(['enter']);
|
|
472
|
+
|
|
473
|
+
// Re-use the same element reference
|
|
474
|
+
await input.click();
|
|
475
|
+
await testdriver.pressKeys(['ctrl', 'a']); // Select all
|
|
476
|
+
await testdriver.type('second search');
|
|
477
|
+
```
|
|
478
|
+
</Accordion>
|
|
479
|
+
</AccordionGroup>
|