blob: e5b9a6d2c8e96f0abed35212438649c96d1d9463 [file] [log] [blame]
// Copyright 2015 Google Inc. All Rights Reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
import 'dart:async';
import 'dart:math';
import 'package:webdriver/src/async/common.dart';
import 'package:webdriver/src/async/web_driver.dart';
import 'package:webdriver/src/common/by.dart';
import 'package:webdriver/src/common/request_client.dart';
import 'package:webdriver/src/common/web_element.dart' as common;
import 'package:webdriver/src/common/webdriver_handler.dart';
class WebElement extends common.WebElement implements SearchContext {
@override
final WebDriver driver;
@override
final String id;
/// The context from which this element was found.
final SearchContext? context;
/// How the element was located from the context.
final dynamic /* String | Finder */ locator;
/// The index of this element in the set of element founds. If the method
/// used to find this element always returns one element, then this is null.
final int? index;
final AsyncRequestClient _client;
final WebDriverHandler _handler;
WebElement(this.driver, this._client, this._handler, this.id,
[this.context, this.locator, this.index]);
/// Click on this element.
Future<void> click() => _client.send(_handler.element.buildClickRequest(id),
_handler.element.parseClickResponse);
/// Send [keysToSend] to this element.
Future<void> sendKeys(String keysToSend) => _client.send(
_handler.element.buildSendKeysRequest(id, keysToSend),
_handler.element.parseSendKeysResponse);
/// Clear the content of a text element.
Future<void> clear() => _client.send(_handler.element.buildClearRequest(id),
_handler.element.parseClearResponse);
/// Is this radio button/checkbox selected?
Future<bool> get selected => _client.send(
_handler.element.buildSelectedRequest(id),
_handler.element.parseSelectedResponse);
/// Is this form element enabled?
Future<bool> get enabled => _client.send(
_handler.element.buildEnabledRequest(id),
_handler.element.parseEnabledResponse);
/// Is this element visible in the page?
Future<bool> get displayed => _client.send(
_handler.element.buildDisplayedRequest(id),
_handler.element.parseDisplayedResponse);
/// The location within the document of this element.
Future<Point<int>> get location => _client.send(
_handler.element.buildLocationRequest(id),
_handler.element.parseLocationResponse);
/// The size of this element.
Future<Rectangle<int>> get size => _client.send(
_handler.element.buildSizeRequest(id),
_handler.element.parseSizeResponse);
/// The bounds of this element.
Future<Rectangle<int>> get rect async {
final location = await this.location;
final size = await this.size;
return Rectangle<int>(location.x, location.y, size.width, size.height);
}
/// The tag name for this element.
Future<String> get name => _client.send(_handler.element.buildNameRequest(id),
_handler.element.parseNameResponse);
/// Visible text within this element.
Future<String> get text => _client.send(_handler.element.buildTextRequest(id),
_handler.element.parseTextResponse);
///Find an element nested within this element.
///
/// Throws [NoSuchElementException] if matching element is not found.
@override
Future<WebElement> findElement(By by) => _client.send(
_handler.elementFinder.buildFindElementRequest(by, id),
(response) => driver.getElement(
_handler.elementFinder.parseFindElementResponse(response), this, by));
/// Find multiple elements nested within this element.
@override
Stream<WebElement> findElements(By by) async* {
final ids = await _client.send(
_handler.elementFinder.buildFindElementsRequest(by, id),
_handler.elementFinder.parseFindElementsResponse);
int i = 0;
for (var id in ids) {
yield driver.getElement(id, this, by, i);
i++;
}
}
/// Access to the HTML attributes of this tag.
///
/// TODO(DrMarcII): consider special handling of boolean attributes.
Attributes get attributes => Attributes((name) => _client.send(
_handler.element.buildAttributeRequest(id, name),
_handler.element.parseAttributeResponse));
/// Access to the selenium attributes of this tag.
///
/// This is deprecated, only used to support old pageloader.
@deprecated
Attributes get seleniumAttributes => Attributes((name) => _client.send(
_handler.element.buildSeleniumAttributeRequest(id, name),
_handler.element.parseSeleniumAttributeResponse));
/// Access to the HTML properties of this tag.
Attributes get properties => Attributes((name) => _client.send(
_handler.element.buildPropertyRequest(id, name),
_handler.element.parsePropertyResponse));
/// Access to the cssProperties of this element.
///
/// TODO(DrMarcII): consider special handling of color and possibly other
/// properties.
Attributes get cssProperties => Attributes((name) => _client.send(
_handler.element.buildCssPropertyRequest(id, name),
_handler.element.parseCssPropertyResponse));
Future<bool> equals(WebElement other) async =>
other is WebElement && other.driver == this.driver && other.id == this.id;
@override
int get hashCode => driver.hashCode * 3 + id.hashCode;
@override
bool operator ==(other) =>
other is WebElement && other.driver == this.driver && other.id == this.id;
@override
String toString() {
var out = StringBuffer()..write(context);
if (locator is By) {
if (index == null) {
out..write('.findElement(');
} else {
out..write('.findElements(');
}
out..write(locator)..write(')');
} else {
out..write('.')..write(locator);
}
if (index != null) {
out..write('[')..write(index)..write(']');
}
return out.toString();
}
}