SDK: support more playwright features (#3866)
This commit is contained in:
committed by
GitHub
parent
9f505e74fd
commit
34258dd81a
@@ -1,5 +1,5 @@
|
|||||||
import asyncio
|
import asyncio
|
||||||
from typing import TYPE_CHECKING, Any, overload
|
from typing import TYPE_CHECKING, Any, Pattern, overload
|
||||||
|
|
||||||
from playwright.async_api import Page
|
from playwright.async_api import Page
|
||||||
|
|
||||||
@@ -8,6 +8,7 @@ from skyvern.client.types.workflow_run_response import WorkflowRunResponse
|
|||||||
from skyvern.config import settings
|
from skyvern.config import settings
|
||||||
from skyvern.library.constants import DEFAULT_AGENT_HEARTBEAT_INTERVAL, DEFAULT_AGENT_TIMEOUT
|
from skyvern.library.constants import DEFAULT_AGENT_HEARTBEAT_INTERVAL, DEFAULT_AGENT_TIMEOUT
|
||||||
from skyvern.library.SdkSkyvernPageAi import SdkSkyvernPageAi
|
from skyvern.library.SdkSkyvernPageAi import SdkSkyvernPageAi
|
||||||
|
from skyvern.library.skyvern_locator import SkyvernLocator
|
||||||
from skyvern.webeye.actions import handler_utils
|
from skyvern.webeye.actions import handler_utils
|
||||||
|
|
||||||
if TYPE_CHECKING:
|
if TYPE_CHECKING:
|
||||||
@@ -29,7 +30,7 @@ class SkyvernPageRun:
|
|||||||
self._browser = browser
|
self._browser = browser
|
||||||
self._page = page
|
self._page = page
|
||||||
|
|
||||||
async def run_task(
|
async def task(
|
||||||
self,
|
self,
|
||||||
prompt: str,
|
prompt: str,
|
||||||
engine: RunEngine = RunEngine.skyvern_v2,
|
engine: RunEngine = RunEngine.skyvern_v2,
|
||||||
@@ -144,7 +145,7 @@ class SkyvernPageRun:
|
|||||||
workflow_run = await self._wait_for_run_completion(workflow_run.run_id, timeout)
|
workflow_run = await self._wait_for_run_completion(workflow_run.run_id, timeout)
|
||||||
return WorkflowRunResponse.model_validate(workflow_run.model_dump())
|
return WorkflowRunResponse.model_validate(workflow_run.model_dump())
|
||||||
|
|
||||||
async def run_workflow(
|
async def workflow(
|
||||||
self,
|
self,
|
||||||
workflow_id: str,
|
workflow_id: str,
|
||||||
parameters: dict[str, Any] | None = None,
|
parameters: dict[str, Any] | None = None,
|
||||||
@@ -239,6 +240,7 @@ class SkyvernBrowserPage:
|
|||||||
ai: str | None = "fallback",
|
ai: str | None = "fallback",
|
||||||
data: str | dict[str, Any] | None = None,
|
data: str | dict[str, Any] | None = None,
|
||||||
timeout: float = settings.BROWSER_ACTION_TIMEOUT_MS,
|
timeout: float = settings.BROWSER_ACTION_TIMEOUT_MS,
|
||||||
|
**kwargs: Any,
|
||||||
) -> str | None: ...
|
) -> str | None: ...
|
||||||
|
|
||||||
@overload
|
@overload
|
||||||
@@ -249,6 +251,7 @@ class SkyvernBrowserPage:
|
|||||||
ai: str | None = "fallback",
|
ai: str | None = "fallback",
|
||||||
data: str | dict[str, Any] | None = None,
|
data: str | dict[str, Any] | None = None,
|
||||||
timeout: float = settings.BROWSER_ACTION_TIMEOUT_MS,
|
timeout: float = settings.BROWSER_ACTION_TIMEOUT_MS,
|
||||||
|
**kwargs: Any,
|
||||||
) -> str | None: ...
|
) -> str | None: ...
|
||||||
|
|
||||||
async def click(
|
async def click(
|
||||||
@@ -259,6 +262,7 @@ class SkyvernBrowserPage:
|
|||||||
ai: str | None = "fallback",
|
ai: str | None = "fallback",
|
||||||
data: str | dict[str, Any] | None = None,
|
data: str | dict[str, Any] | None = None,
|
||||||
timeout: float = settings.BROWSER_ACTION_TIMEOUT_MS,
|
timeout: float = settings.BROWSER_ACTION_TIMEOUT_MS,
|
||||||
|
**kwargs: Any,
|
||||||
) -> str | None:
|
) -> str | None:
|
||||||
"""Click an element using a CSS selector, AI-powered prompt matching, or both.
|
"""Click an element using a CSS selector, AI-powered prompt matching, or both.
|
||||||
|
|
||||||
@@ -296,7 +300,7 @@ class SkyvernBrowserPage:
|
|||||||
if selector:
|
if selector:
|
||||||
try:
|
try:
|
||||||
locator = self._page.locator(selector)
|
locator = self._page.locator(selector)
|
||||||
await locator.click(timeout=timeout)
|
await locator.click(timeout=timeout, **kwargs)
|
||||||
return selector
|
return selector
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
error_to_raise = e
|
error_to_raise = e
|
||||||
@@ -323,7 +327,7 @@ class SkyvernBrowserPage:
|
|||||||
)
|
)
|
||||||
|
|
||||||
if selector:
|
if selector:
|
||||||
locator = self._page.locator(selector)
|
locator = self._page.locator(selector, **kwargs)
|
||||||
await locator.click(timeout=timeout)
|
await locator.click(timeout=timeout)
|
||||||
return selector
|
return selector
|
||||||
|
|
||||||
@@ -437,12 +441,13 @@ class SkyvernBrowserPage:
|
|||||||
async def select_option(
|
async def select_option(
|
||||||
self,
|
self,
|
||||||
selector: str,
|
selector: str,
|
||||||
value: str,
|
value: str | None = None,
|
||||||
*,
|
*,
|
||||||
prompt: str | None = None,
|
prompt: str | None = None,
|
||||||
ai: str | None = "fallback",
|
ai: str | None = "fallback",
|
||||||
data: str | dict[str, Any] | None = None,
|
data: str | dict[str, Any] | None = None,
|
||||||
timeout: float = settings.BROWSER_ACTION_TIMEOUT_MS,
|
timeout: float = settings.BROWSER_ACTION_TIMEOUT_MS,
|
||||||
|
**kwargs: Any,
|
||||||
) -> str: ...
|
) -> str: ...
|
||||||
|
|
||||||
@overload
|
@overload
|
||||||
@@ -455,6 +460,7 @@ class SkyvernBrowserPage:
|
|||||||
ai: str | None = "fallback",
|
ai: str | None = "fallback",
|
||||||
data: str | dict[str, Any] | None = None,
|
data: str | dict[str, Any] | None = None,
|
||||||
timeout: float = settings.BROWSER_ACTION_TIMEOUT_MS,
|
timeout: float = settings.BROWSER_ACTION_TIMEOUT_MS,
|
||||||
|
**kwargs: Any,
|
||||||
) -> str: ...
|
) -> str: ...
|
||||||
|
|
||||||
async def select_option(
|
async def select_option(
|
||||||
@@ -466,6 +472,7 @@ class SkyvernBrowserPage:
|
|||||||
ai: str | None = "fallback",
|
ai: str | None = "fallback",
|
||||||
data: str | dict[str, Any] | None = None,
|
data: str | dict[str, Any] | None = None,
|
||||||
timeout: float = settings.BROWSER_ACTION_TIMEOUT_MS,
|
timeout: float = settings.BROWSER_ACTION_TIMEOUT_MS,
|
||||||
|
**kwargs: Any,
|
||||||
) -> str:
|
) -> str:
|
||||||
"""Select an option from a dropdown using a CSS selector, AI-powered prompt matching, or both.
|
"""Select an option from a dropdown using a CSS selector, AI-powered prompt matching, or both.
|
||||||
|
|
||||||
@@ -507,7 +514,7 @@ class SkyvernBrowserPage:
|
|||||||
if selector:
|
if selector:
|
||||||
try:
|
try:
|
||||||
locator = self._page.locator(selector)
|
locator = self._page.locator(selector)
|
||||||
await locator.select_option(value, timeout=timeout)
|
await locator.select_option(value, timeout=timeout, **kwargs)
|
||||||
return value
|
return value
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
error_to_raise = e
|
error_to_raise = e
|
||||||
@@ -533,7 +540,7 @@ class SkyvernBrowserPage:
|
|||||||
)
|
)
|
||||||
if selector:
|
if selector:
|
||||||
locator = self._page.locator(selector)
|
locator = self._page.locator(selector)
|
||||||
await locator.select_option(value, timeout=timeout)
|
await locator.select_option(value, timeout=timeout, **kwargs)
|
||||||
return value
|
return value
|
||||||
|
|
||||||
async def extract(
|
async def extract(
|
||||||
@@ -544,6 +551,35 @@ class SkyvernBrowserPage:
|
|||||||
intention: str | None = None,
|
intention: str | None = None,
|
||||||
data: str | dict[str, Any] | None = None,
|
data: str | dict[str, Any] | None = None,
|
||||||
) -> dict[str, Any] | list | str | None:
|
) -> dict[str, Any] | list | str | None:
|
||||||
|
"""Extract structured data from the page using AI.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
prompt: Natural language description of what data to extract.
|
||||||
|
schema: JSON Schema defining the structure of data to extract.
|
||||||
|
error_code_mapping: Mapping of error codes to custom error messages.
|
||||||
|
intention: Additional context about the extraction intent.
|
||||||
|
data: Additional context data for AI processing.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Extracted data matching the provided schema, or None if extraction fails.
|
||||||
|
|
||||||
|
Examples:
|
||||||
|
```python
|
||||||
|
# Extract structured data with JSON Schema
|
||||||
|
result = await page.extract(
|
||||||
|
prompt="Extract product information",
|
||||||
|
schema={
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"name": {"type": "string", "description": "Product name"},
|
||||||
|
"price": {"type": "number", "description": "Product price"}
|
||||||
|
},
|
||||||
|
"required": ["name", "price"]
|
||||||
|
}
|
||||||
|
)
|
||||||
|
# Returns: {"name": "...", "price": 29.99}
|
||||||
|
```
|
||||||
|
"""
|
||||||
return await self._ai.ai_extract(prompt, schema, error_code_mapping, intention, data)
|
return await self._ai.ai_extract(prompt, schema, error_code_mapping, intention, data)
|
||||||
|
|
||||||
async def reload(self, **kwargs: Any) -> None:
|
async def reload(self, **kwargs: Any) -> None:
|
||||||
@@ -565,6 +601,101 @@ class SkyvernBrowserPage:
|
|||||||
"""
|
"""
|
||||||
return await self._page.screenshot(**kwargs)
|
return await self._page.screenshot(**kwargs)
|
||||||
|
|
||||||
|
def locator(self, selector: str, **kwargs: Any) -> SkyvernLocator:
|
||||||
|
"""Find an element using a CSS selector or other selector syntax.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
selector: CSS selector or other selector syntax (xpath=, text=, etc.).
|
||||||
|
**kwargs: Additional options like has, has_text, has_not, etc.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
SkyvernLocator object that can be used to perform actions or assertions.
|
||||||
|
"""
|
||||||
|
return SkyvernLocator(self._page.locator(selector, **kwargs))
|
||||||
|
|
||||||
|
def get_by_label(self, text: str | Pattern[str], **kwargs: Any) -> SkyvernLocator:
|
||||||
|
"""Find an input element by its associated label text.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
text: Label text to search for (supports substring and regex matching).
|
||||||
|
**kwargs: Additional options like exact.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
SkyvernLocator object for the labeled input element.
|
||||||
|
"""
|
||||||
|
return SkyvernLocator(self._page.get_by_label(text, **kwargs))
|
||||||
|
|
||||||
|
def get_by_text(self, text: str | Pattern[str], **kwargs: Any) -> SkyvernLocator:
|
||||||
|
"""Find an element containing the specified text.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
text: Text content to search for (supports substring and regex matching).
|
||||||
|
**kwargs: Additional options like exact.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
SkyvernLocator object for the element containing the text.
|
||||||
|
"""
|
||||||
|
return SkyvernLocator(self._page.get_by_text(text, **kwargs))
|
||||||
|
|
||||||
|
def get_by_title(self, text: str | Pattern[str], **kwargs: Any) -> SkyvernLocator:
|
||||||
|
"""Find an element by its title attribute.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
text: Title attribute value to search for (supports substring and regex matching).
|
||||||
|
**kwargs: Additional options like exact.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
SkyvernLocator object for the element with matching title.
|
||||||
|
"""
|
||||||
|
return SkyvernLocator(self._page.get_by_title(text, **kwargs))
|
||||||
|
|
||||||
|
def get_by_role(self, role: str, **kwargs: Any) -> SkyvernLocator:
|
||||||
|
"""Find an element by its ARIA role.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
role: ARIA role (e.g., "button", "textbox", "link").
|
||||||
|
**kwargs: Additional options like name, checked, pressed, etc.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
SkyvernLocator object for the element with matching role.
|
||||||
|
"""
|
||||||
|
return SkyvernLocator(self._page.get_by_role(role, **kwargs))
|
||||||
|
|
||||||
|
def get_by_placeholder(self, text: str | Pattern[str], **kwargs: Any) -> SkyvernLocator:
|
||||||
|
"""Find an input element by its placeholder text.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
text: Placeholder text to search for (supports substring and regex matching).
|
||||||
|
**kwargs: Additional options like exact.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
SkyvernLocator object for the input element with matching placeholder.
|
||||||
|
"""
|
||||||
|
return SkyvernLocator(self._page.get_by_placeholder(text, **kwargs))
|
||||||
|
|
||||||
|
def get_by_alt_text(self, text: str | Pattern[str], **kwargs: Any) -> SkyvernLocator:
|
||||||
|
"""Find an element by its alt text (typically images).
|
||||||
|
|
||||||
|
Args:
|
||||||
|
text: Alt text to search for (supports substring and regex matching).
|
||||||
|
**kwargs: Additional options like exact.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
SkyvernLocator object for the element with matching alt text.
|
||||||
|
"""
|
||||||
|
return SkyvernLocator(self._page.get_by_alt_text(text, **kwargs))
|
||||||
|
|
||||||
|
def get_by_test_id(self, test_id: str) -> SkyvernLocator:
|
||||||
|
"""Find an element by its test ID attribute.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
test_id: Test ID value to search for.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
SkyvernLocator object for the element with matching test ID.
|
||||||
|
"""
|
||||||
|
return SkyvernLocator(self._page.get_by_test_id(test_id))
|
||||||
|
|
||||||
async def _input_text(
|
async def _input_text(
|
||||||
self,
|
self,
|
||||||
selector: str,
|
selector: str,
|
||||||
|
|||||||
173
skyvern/library/skyvern_locator.py
Normal file
173
skyvern/library/skyvern_locator.py
Normal file
@@ -0,0 +1,173 @@
|
|||||||
|
from typing import Any, Pattern
|
||||||
|
|
||||||
|
from playwright.async_api import Locator
|
||||||
|
|
||||||
|
|
||||||
|
class SkyvernLocator:
|
||||||
|
"""Locator for finding and interacting with elements on a page.
|
||||||
|
|
||||||
|
Provides methods for performing actions (click, fill, type), querying element state,
|
||||||
|
and chaining locators to find specific elements. Compatible with Playwright's locator API.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, locator: Locator):
|
||||||
|
self._locator = locator
|
||||||
|
|
||||||
|
# Action methods
|
||||||
|
async def click(self, **kwargs: Any) -> None:
|
||||||
|
"""Click the element."""
|
||||||
|
await self._locator.click(**kwargs)
|
||||||
|
|
||||||
|
async def fill(self, value: str, **kwargs: Any) -> None:
|
||||||
|
"""Fill an input element with text."""
|
||||||
|
await self._locator.fill(value, **kwargs)
|
||||||
|
|
||||||
|
async def type(self, text: str, **kwargs: Any) -> None:
|
||||||
|
"""Type text into the element character by character."""
|
||||||
|
await self._locator.type(text, **kwargs)
|
||||||
|
|
||||||
|
async def select_option(
|
||||||
|
self,
|
||||||
|
value: str | list[str] | None = None,
|
||||||
|
**kwargs: Any,
|
||||||
|
) -> list[str]:
|
||||||
|
"""Select an option in a <select> element."""
|
||||||
|
return await self._locator.select_option(value, **kwargs)
|
||||||
|
|
||||||
|
async def check(self, **kwargs: Any) -> None:
|
||||||
|
"""Check a checkbox or radio button."""
|
||||||
|
await self._locator.check(**kwargs)
|
||||||
|
|
||||||
|
async def uncheck(self, **kwargs: Any) -> None:
|
||||||
|
"""Uncheck a checkbox."""
|
||||||
|
await self._locator.uncheck(**kwargs)
|
||||||
|
|
||||||
|
async def clear(self, **kwargs: Any) -> None:
|
||||||
|
"""Clear an input field."""
|
||||||
|
await self._locator.clear(**kwargs)
|
||||||
|
|
||||||
|
async def hover(self, **kwargs: Any) -> None:
|
||||||
|
"""Hover over the element."""
|
||||||
|
await self._locator.hover(**kwargs)
|
||||||
|
|
||||||
|
async def focus(self, **kwargs: Any) -> None:
|
||||||
|
"""Focus the element."""
|
||||||
|
await self._locator.focus(**kwargs)
|
||||||
|
|
||||||
|
async def press(self, key: str, **kwargs: Any) -> None:
|
||||||
|
"""Press a key on the element."""
|
||||||
|
await self._locator.press(key, **kwargs)
|
||||||
|
|
||||||
|
# Query methods
|
||||||
|
async def count(self) -> int:
|
||||||
|
"""Get the number of elements matching the locator."""
|
||||||
|
return await self._locator.count()
|
||||||
|
|
||||||
|
async def text_content(self, **kwargs: Any) -> str | None:
|
||||||
|
"""Get the text content of the element."""
|
||||||
|
return await self._locator.text_content(**kwargs)
|
||||||
|
|
||||||
|
async def inner_text(self, **kwargs: Any) -> str:
|
||||||
|
"""Get the inner text of the element."""
|
||||||
|
return await self._locator.inner_text(**kwargs)
|
||||||
|
|
||||||
|
async def inner_html(self, **kwargs: Any) -> str:
|
||||||
|
"""Get the inner HTML of the element."""
|
||||||
|
return await self._locator.inner_html(**kwargs)
|
||||||
|
|
||||||
|
async def get_attribute(self, name: str, **kwargs: Any) -> str | None:
|
||||||
|
"""Get an attribute value from the element."""
|
||||||
|
return await self._locator.get_attribute(name, **kwargs)
|
||||||
|
|
||||||
|
async def input_value(self, **kwargs: Any) -> str:
|
||||||
|
"""Get the value of an input element."""
|
||||||
|
return await self._locator.input_value(**kwargs)
|
||||||
|
|
||||||
|
# State methods
|
||||||
|
async def is_visible(self, **kwargs: Any) -> bool:
|
||||||
|
"""Check if the element is visible."""
|
||||||
|
return await self._locator.is_visible(**kwargs)
|
||||||
|
|
||||||
|
async def is_hidden(self, **kwargs: Any) -> bool:
|
||||||
|
"""Check if the element is hidden."""
|
||||||
|
return await self._locator.is_hidden(**kwargs)
|
||||||
|
|
||||||
|
async def is_enabled(self, **kwargs: Any) -> bool:
|
||||||
|
"""Check if the element is enabled."""
|
||||||
|
return await self._locator.is_enabled(**kwargs)
|
||||||
|
|
||||||
|
async def is_disabled(self, **kwargs: Any) -> bool:
|
||||||
|
"""Check if the element is disabled."""
|
||||||
|
return await self._locator.is_disabled(**kwargs)
|
||||||
|
|
||||||
|
async def is_editable(self, **kwargs: Any) -> bool:
|
||||||
|
"""Check if the element is editable."""
|
||||||
|
return await self._locator.is_editable(**kwargs)
|
||||||
|
|
||||||
|
async def is_checked(self, **kwargs: Any) -> bool:
|
||||||
|
"""Check if a checkbox or radio button is checked."""
|
||||||
|
return await self._locator.is_checked(**kwargs)
|
||||||
|
|
||||||
|
# Filtering and chaining methods
|
||||||
|
def first(self) -> "SkyvernLocator":
|
||||||
|
"""Get the first matching element."""
|
||||||
|
return SkyvernLocator(self._locator.first)
|
||||||
|
|
||||||
|
def last(self) -> "SkyvernLocator":
|
||||||
|
"""Get the last matching element."""
|
||||||
|
return SkyvernLocator(self._locator.last)
|
||||||
|
|
||||||
|
def nth(self, index: int) -> "SkyvernLocator":
|
||||||
|
"""Get the nth matching element (0-indexed)."""
|
||||||
|
return SkyvernLocator(self._locator.nth(index))
|
||||||
|
|
||||||
|
def filter(self, **kwargs: Any) -> "SkyvernLocator":
|
||||||
|
"""Filter the locator by additional criteria."""
|
||||||
|
return SkyvernLocator(self._locator.filter(**kwargs))
|
||||||
|
|
||||||
|
def locator(self, selector: str, **kwargs: Any) -> "SkyvernLocator":
|
||||||
|
"""Find a descendant element."""
|
||||||
|
return SkyvernLocator(self._locator.locator(selector, **kwargs))
|
||||||
|
|
||||||
|
def get_by_label(self, text: str | Pattern[str], **kwargs: Any) -> "SkyvernLocator":
|
||||||
|
"""Find an input element by its associated label text."""
|
||||||
|
return SkyvernLocator(self._locator.get_by_label(text, **kwargs))
|
||||||
|
|
||||||
|
def get_by_text(self, text: str | Pattern[str], **kwargs: Any) -> "SkyvernLocator":
|
||||||
|
"""Find an element containing the specified text."""
|
||||||
|
return SkyvernLocator(self._locator.get_by_text(text, **kwargs))
|
||||||
|
|
||||||
|
def get_by_title(self, text: str | Pattern[str], **kwargs: Any) -> "SkyvernLocator":
|
||||||
|
"""Find an element by its title attribute."""
|
||||||
|
return SkyvernLocator(self._locator.get_by_title(text, **kwargs))
|
||||||
|
|
||||||
|
def get_by_role(self, role: str, **kwargs: Any) -> "SkyvernLocator":
|
||||||
|
"""Find an element by its ARIA role."""
|
||||||
|
return SkyvernLocator(self._locator.get_by_role(role, **kwargs))
|
||||||
|
|
||||||
|
def get_by_placeholder(self, text: str | Pattern[str], **kwargs: Any) -> "SkyvernLocator":
|
||||||
|
"""Find an input element by its placeholder text."""
|
||||||
|
return SkyvernLocator(self._locator.get_by_placeholder(text, **kwargs))
|
||||||
|
|
||||||
|
def get_by_alt_text(self, text: str | Pattern[str], **kwargs: Any) -> "SkyvernLocator":
|
||||||
|
"""Find an element by its alt text (typically images)."""
|
||||||
|
return SkyvernLocator(self._locator.get_by_alt_text(text, **kwargs))
|
||||||
|
|
||||||
|
def get_by_test_id(self, test_id: str) -> "SkyvernLocator":
|
||||||
|
"""Find an element by its test ID attribute."""
|
||||||
|
return SkyvernLocator(self._locator.get_by_test_id(test_id))
|
||||||
|
|
||||||
|
# Waiting and screenshot
|
||||||
|
async def wait_for(self, **kwargs: Any) -> None:
|
||||||
|
"""Wait for the element to reach a specific state."""
|
||||||
|
await self._locator.wait_for(**kwargs)
|
||||||
|
|
||||||
|
async def screenshot(self, **kwargs: Any) -> bytes:
|
||||||
|
"""Take a screenshot of the element."""
|
||||||
|
return await self._locator.screenshot(**kwargs)
|
||||||
|
|
||||||
|
# Access to underlying Playwright locator
|
||||||
|
@property
|
||||||
|
def playwright_locator(self) -> Locator:
|
||||||
|
"""Get the underlying Playwright Locator object."""
|
||||||
|
return self._locator
|
||||||
Reference in New Issue
Block a user