com.codeborne.selenide.Condition Maven / Gradle / Ivy
Show all versions of selenide-core Show documentation
package com.codeborne.selenide;
import com.codeborne.selenide.conditions.And;
import com.codeborne.selenide.conditions.Animated;
import com.codeborne.selenide.conditions.Attribute;
import com.codeborne.selenide.conditions.AttributeWithValue;
import com.codeborne.selenide.conditions.CaseSensitiveText;
import com.codeborne.selenide.conditions.Checked;
import com.codeborne.selenide.conditions.CssClass;
import com.codeborne.selenide.conditions.CssValue;
import com.codeborne.selenide.conditions.CustomMatch;
import com.codeborne.selenide.conditions.Disabled;
import com.codeborne.selenide.conditions.DomAttribute;
import com.codeborne.selenide.conditions.DomAttributeValue;
import com.codeborne.selenide.conditions.DomProperty;
import com.codeborne.selenide.conditions.DomPropertyValue;
import com.codeborne.selenide.conditions.Editable;
import com.codeborne.selenide.conditions.Enabled;
import com.codeborne.selenide.conditions.ExactOwnText;
import com.codeborne.selenide.conditions.ExactOwnTextCaseSensitive;
import com.codeborne.selenide.conditions.ExactText;
import com.codeborne.selenide.conditions.ExactTextCaseSensitive;
import com.codeborne.selenide.conditions.Exist;
import com.codeborne.selenide.conditions.Focused;
import com.codeborne.selenide.conditions.Hidden;
import com.codeborne.selenide.conditions.Href;
import com.codeborne.selenide.conditions.InnerText;
import com.codeborne.selenide.conditions.Interactable;
import com.codeborne.selenide.conditions.IsImageLoaded;
import com.codeborne.selenide.conditions.MatchAttributeWithValue;
import com.codeborne.selenide.conditions.MatchText;
import com.codeborne.selenide.conditions.NamedCondition;
import com.codeborne.selenide.conditions.OneOfExactTexts;
import com.codeborne.selenide.conditions.OneOfExactTextsCaseSensitive;
import com.codeborne.selenide.conditions.OneOfTexts;
import com.codeborne.selenide.conditions.OneOfTextsCaseSensitive;
import com.codeborne.selenide.conditions.Or;
import com.codeborne.selenide.conditions.OwnText;
import com.codeborne.selenide.conditions.OwnTextCaseSensitive;
import com.codeborne.selenide.conditions.PartialText;
import com.codeborne.selenide.conditions.PartialTextCaseSensitive;
import com.codeborne.selenide.conditions.PartialValue;
import com.codeborne.selenide.conditions.PseudoElementPropertyWithValue;
import com.codeborne.selenide.conditions.Readonly;
import com.codeborne.selenide.conditions.Selected;
import com.codeborne.selenide.conditions.SelectedText;
import com.codeborne.selenide.conditions.TagName;
import com.codeborne.selenide.conditions.Text;
import com.codeborne.selenide.conditions.Value;
import com.codeborne.selenide.conditions.Visible;
import org.jspecify.annotations.Nullable;
import org.openqa.selenium.WebElement;
import java.util.function.Predicate;
import static com.codeborne.selenide.conditions.ConditionHelpers.merge;
import static java.util.Arrays.asList;
/**
* Conditions to match web elements: checks for visibility, text etc.
*/
public final class Condition {
/**
* Checks if element is visible
*
* Sample: {@code $("input").shouldBe(visible);}
*/
public static final WebElementCondition visible = new Visible();
/**
* Check if element exist. It can be visible or hidden.
*
* Sample: {@code $("input").should(exist);}
*/
public static final WebElementCondition exist = new Exist();
/**
* Checks that element is not visible or does not exist.
*
* Opposite to {@link #appear}
*
*
Sample: {@code $("input").shouldBe(hidden);}
*/
public static final WebElementCondition hidden = new Hidden();
/**
* Synonym for {@link #visible} - may be used for better readability
*
* Sample: {@code $("#logoutLink").should(appear);}
*/
public static final WebElementCondition appear = be(visible);
/**
* Synonym for {@link #hidden} - may be used for better readability:
*
* {@code $("#loginLink").should(disappear);}
*/
public static final WebElementCondition disappear = be(hidden);
/**
* Check if element is interactable:
*
* - either is visible, or
* - has css property "opacity: 0"
*
*
* Elements which are transparent (opacity:0) are considered to be invisible, but interactable.
* User can click, doubleClick etc., and enter text etc. to transparent elements
* (for all major browsers).
*
*
* Example:
* {@code $("input[type=file]").shouldBe(interactable);}
*
*/
public static final WebElementCondition interactable = new Interactable();
/**
*
* Check if element has "readonly" attribute (with any value)
*
*
* Sample:
* {@code $("input").shouldBe(readonly);}
*
*/
public static final WebElementCondition readonly = new Readonly();
/**
* Check if element is "editable":
*
* - is {@link #interactable}, and
* - is {@link #enabled}, and
* - is not {@link #readonly}
*
*
* Sample: {@code $("input").shouldBe(editable);}
*
*/
public static final WebElementCondition editable = new Editable();
/**
*
* Check that the element is animated. An animated element changes its position or size over time.
* Implemented for web browser context only.
*
*
* Sample:
* {@code $("popup").shouldBe(animated);}
*
*
* @since v7.0.7
*/
public static final WebElementCondition animated = new Animated();
/**
* Check if element has given attribute (with any value)
*
* Sample: {@code $("#mydiv").shouldHave(attribute("fileId"));}
*
* @param attributeName name of attribute, not null
*/
public static WebElementCondition attribute(String attributeName) {
return new Attribute(attributeName);
}
/**
* Sample: {@code $("#mydiv").shouldHave(attribute("fileId", "12345"));}
*
* @param attributeName name of attribute
* @param expectedAttributeValue expected value of attribute
*/
public static WebElementCondition attribute(String attributeName, String expectedAttributeValue) {
return new AttributeWithValue(attributeName, expectedAttributeValue);
}
/**
* Assert that given element's attribute matches given regular expression
*
* Sample: {@code $("h1").shouldHave(attributeMatching("fileId", ".*12345.*"))}
*
* @param attributeName name of attribute
* @param attributeRegex regex to match attribute value
*/
public static WebElementCondition attributeMatching(String attributeName, String attributeRegex) {
return new MatchAttributeWithValue(attributeName, attributeRegex);
}
/**
* Sample: {@code $("#mydiv").shouldHave(href("/one/two/three.pdf"));}
*
* It looks similar to `$.shouldHave(attribute("href", href))`, but
* it overcomes the fact that Selenium returns full url (even if "href" attribute in html contains relative url).
*
* @param href expected value of "href" attribute
*/
public static WebElementCondition href(String href) {
return new Href(href);
}
/**
* Assert that element contains given "value" attribute as substring
* NB! Ignores difference in non-visible characters like spaces, non-breakable spaces, tabs, newlines etc.
*
*
Sample: {@code $("input").shouldHave(value("12345 666 77"));}
*
* @param expectedValue expected value of "value" attribute
*/
public static WebElementCondition value(String expectedValue) {
return new Value(expectedValue);
}
/**
* Assert that element contains given "value" attribute as substring
* NB! Ignores difference in non-visible characters like spaces, non-breakable spaces, tabs, newlines etc.
*
* Sample: {@code $("input").shouldHave(partialValue("12345 666 77"));}
*
* @param expectedValue expected value of "value" attribute
*/
public static WebElementCondition partialValue(String expectedValue) {
return new PartialValue(expectedValue);
}
/**
* Check that element has given the property value of the pseudo-element
* Sample: {@code $("input").shouldHave(pseudo(":first-letter", "color", "#ff0000"));}
*
* @param pseudoElementName pseudo-element name of the element,
* ":before", ":after", ":first-letter", ":first-line", ":selection"
* @param propertyName property name of the pseudo-element
* @param expectedValue expected value of the property
*/
public static WebElementCondition pseudo(String pseudoElementName, String propertyName, String expectedValue) {
return new PseudoElementPropertyWithValue(pseudoElementName, propertyName, expectedValue);
}
/**
* Check that element has given the "content" property of the pseudo-element
* Sample: {@code $("input").shouldHave(pseudo(":before", "Hello"));}
*
* @param pseudoElementName pseudo-element name of the element, ":before", ":after"
* @param expectedValue expected content of the pseudo-element
*/
public static WebElementCondition pseudo(String pseudoElementName, String expectedValue) {
return new PseudoElementPropertyWithValue(pseudoElementName, "content", expectedValue);
}
/**
* Sample: {@code $("#input").shouldHave(exactValue("John"));}
*
* @param value expected value of input field
*/
public static WebElementCondition exactValue(String value) {
return attribute("value", value);
}
/**
* Asserts the name attribute of the element to be exact string
* Sample: {@code $("#input").shouldHave(name("username"))}
*
* @param name expected name of input field
*/
public static WebElementCondition name(String name) {
return attribute("name", name);
}
/**
* Asserts the type attribute of the element to be exact string
* Sample: {@code $("#input").shouldHave(type("checkbox"))}
*
* @param type expected type of input field
*/
public static WebElementCondition type(String type) {
return attribute("type", type);
}
/**
* Sample: {@code $("#input").shouldHave(id("myForm"))}
*
* @param id expected id of input field
*/
public static WebElementCondition id(String id) {
return attribute("id", id);
}
/**
* 1) For input element, check that value is missing or empty
* Sample: {@code $("#input").shouldBe(empty)}
*
* 2) For other elements, check that text is empty
*
Sample: {@code $("h2").shouldBe(empty)}
*/
public static final WebElementCondition empty = and("empty", exactValue(""), exactText(""));
/**
* Assert that given element's text matches given regular expression
*
* Sample: {@code $("h1").should(matchText("Hello\s*John"))}
*
* @param regex e.g. Kicked.*Chuck Norris - in this case ".*" can contain any characters including spaces, tabs, CR etc.
*/
public static WebElementCondition matchText(String regex) {
return new MatchText(regex);
}
/**
* Assert that given element's TEXT case-insensitively CONTAINS at least
* one of the given {@code texts}. Assertion fails if specified collection is empty.
*
* NB! Ignores multiple whitespaces between words.
* NB! Nulls and blank strings are not allowed in the specified collection
* (because any element does contain an empty text).
*
* @throws IllegalArgumentException If specified collection contains {@code null}s or blank strings.
* @since 7.0.3
*/
public static WebElementCondition oneOfTexts(String... texts) {
return new OneOfTexts(asList(texts));
}
/**
* Assert that given element's TEXT case-sensitively CONTAINS at least
* one of the given {@code texts}. Assertion fails if specified collection is empty.
*
* NB! Ignores multiple whitespaces between words.
* NB! Nulls and blank strings are not allowed in the specified collection
* (because any element does contain an empty text).
*
* @throws IllegalArgumentException If specified collection contains {@code null}s or blank strings.
* @since 7.0.3
*/
public static WebElementCondition oneOfTextsCaseSensitive(String... texts) {
return new OneOfTextsCaseSensitive(asList(texts));
}
/**
* Assert that given element's TEXT case-insensitively EQUALS to
* one of the given {@code texts}. Assertion fails if specified collection is empty.
*
* NB! Ignores multiple whitespaces between words.
*
* @throws IllegalArgumentException If specified collection contains {@code null} elements.
* @since 7.0.3
*/
public static WebElementCondition oneOfExactTexts(String... texts) {
return new OneOfExactTexts(asList(texts));
}
/**
* Assert that given element's TEXT case-sensitively EQUALS to
* one of the given {@code texts}. Assertion fails if specified collection is empty.
*
* NB! Ignores multiple whitespaces between words.
*
* @throws IllegalArgumentException If specified collection contains {@code null} elements.
* @since 7.0.3
*/
public static WebElementCondition oneOfExactTextsCaseSensitive(String... texts) {
return new OneOfExactTextsCaseSensitive(asList(texts));
}
/**
* Assert that given element's text CONTAINS given text
*
* Sample: {@code $("h1").shouldHave(partialText("ello Joh"))}
*/
public static WebElementCondition partialText(String expectedText) {
return new PartialText(expectedText);
}
/**
* Assert that given element's text CONTAINS given text (case-sensitive)
*
* Sample: {@code $("h1").should(partialTextCaseSensitive("ELLO jOH"))}
*/
public static WebElementCondition partialTextCaseSensitive(String expectedText) {
return new PartialTextCaseSensitive(expectedText);
}
/**
*
* Assert that element contains given text as a substring.
*
*
* Sample: {@code $("h1").shouldHave(text("Hello\s*John"))}
*
* NB! Case insensitive
* NB! Ignores multiple whitespaces between words
*
* @param text expected text of HTML element.
* NB! Empty string is not allowed (because any element does contain an empty text).
* @throws IllegalArgumentException if given text is null or empty
*/
public static WebElementCondition text(String text) {
return new Text(text);
}
/**
* Checks on {@code } element that exactly given text is selected (=marked with mouse/keyboard)
*
* Sample: {@code $("input").shouldHave(selectedText("Text"))}
*
* NB! Case sensitive
*
* @param expectedText expected selected text of the element
*/
public static WebElementCondition selectedText(String expectedText) {
return new SelectedText(expectedText);
}
/**
* Assert that element contains given text as a case-sensitive substring
*
* Sample: {@code $("h1").shouldHave(textCaseSensitive("Hello\s*John"))}
*
* NB! Ignores multiple whitespaces between words
*
* @param text expected text of HTML element
*/
public static WebElementCondition textCaseSensitive(String text) {
return new CaseSensitiveText(text);
}
/**
* Assert that element has exactly (case-insensitive) given text
* Sample: {@code $("h1").shouldHave(exactText("Hello"))}
*
* Case insensitive
* NB! Ignores multiple whitespaces between words
*
* @param text expected text of HTML element
*/
public static WebElementCondition exactText(String text) {
return new ExactText(text);
}
/**
* Assert that element contains given inner text.
* Sample: {@code $("h1").shouldHave(innerText("Hello"))}
*
* It can be used to check the text of a hidden element.
*
*
Case insensitive
* NB! Ignores multiple whitespaces between words
*/
public static WebElementCondition innerText(String text) {
return new InnerText(text);
}
/**
* Assert that element contains given text (without checking child elements).
* Sample: {@code $("h1").shouldHave(ownText("Hello"))}
*
* Case insensitive
* NB! Ignores multiple whitespaces between words
*
* @param text expected text of HTML element without its children
*/
public static WebElementCondition ownText(String text) {
return new OwnText(text);
}
/**
* Assert that element contains given text (without checking child elements).
* Sample: {@code $("h1").shouldHave(ownTextCaseSensitive("Hello"))}
*
* Case sensitive
* NB! Ignores multiple whitespaces between words
*
* @param text expected text of HTML element without its children
*/
public static WebElementCondition ownTextCaseSensitive(String text) {
return new OwnTextCaseSensitive(text);
}
/**
* Assert that element has given text (without checking child elements).
* Sample: {@code $("h1").shouldHave(ownText("Hello"))}
*
* Case insensitive
* NB! Ignores multiple whitespaces between words
*
* @param text expected text of HTML element without its children
*/
public static WebElementCondition exactOwnText(String text) {
return new ExactOwnText(text);
}
/**
* Assert that element has given text (without checking child elements).
* Sample: {@code $("h1").shouldHave(exactOwnTextCaseSensitive("Hello"))}
*
* Case sensitive
* NB! Ignores multiple whitespaces between words
*
* @param text expected text of HTML element without its children
*/
public static WebElementCondition exactOwnTextCaseSensitive(String text) {
return new ExactOwnTextCaseSensitive(text);
}
/**
* Assert that element has exactly the given text
* Sample: {@code $("h1").shouldHave(exactTextCaseSensitive("Hello"))}
*
* NB! Ignores multiple whitespaces between words
*
* @param text expected text of HTML element
*/
public static WebElementCondition exactTextCaseSensitive(String text) {
return new ExactTextCaseSensitive(text);
}
/**
* Asserts that element has the given tag name.
* Sample: {@code $(".btn-primary").shouldHave(tagName("button"));}
*/
public static WebElementCondition tagName(String cssClass) {
return new TagName(cssClass);
}
/**
* Asserts that element has the given class. Element may have other classes as well.
* Sample: {@code $("input").shouldHave(cssClass("active"));}
*/
public static WebElementCondition cssClass(String cssClass) {
return new CssClass(cssClass);
}
/**
* Checks if css property (style) applies for the element.
* Both explicit and computed properties are supported.
*
* Note that if css property is missing {@link WebElement#getCssValue} return empty string.
* In this case you should assert against empty string.
*
* Sample:
*
* {@code }
*
* {@code $("input").shouldHave(cssValue("font-size", "12"));}
*
* {@code $("input").shouldHave(cssValue("display", "block"));}
*
* @param propertyName the css property (style) name of the element
* @param expectedValue expected value of css property
* @see WebElement#getCssValue
*/
public static WebElementCondition cssValue(String propertyName, @Nullable String expectedValue) {
return new CssValue(propertyName, expectedValue);
}
/**
* Check if element has given dom attribute (with any value)
*
*
Sample: {@code $("#mydiv").shouldHave(domAttribute("hidden"));}
*
* @param domAttributeName name of dom attribute, not null
* @since 7.4.0
*/
public static WebElementCondition domAttribute(String domAttributeName) {
return new DomAttribute(domAttributeName);
}
/**
* Sample: {@code $("#mydiv").shouldHave(domAttributeValue("hidden", "hidden"));}
*
* @param domAttributeName name of dom attribute
* @param expectedDomAttributeValue expected value of dom attribute
* @since 7.4.0
*/
public static WebElementCondition domAttribute(String domAttributeName, @Nullable String expectedDomAttributeValue) {
return new DomAttributeValue(domAttributeName, expectedDomAttributeValue);
}
/**
* Check if element has given dom property (with any value)
*
* Sample: {@code $("#mydiv").shouldHave(domProperty("id"));}
*
* @param domPropertyName name of dom property, not null
* @since 7.4.0
*/
public static WebElementCondition domProperty(String domPropertyName) {
return new DomProperty(domPropertyName);
}
/**
* Sample: {@code $("#my-div").shouldHave(domPropertyValue("id", "my-div"));}
*
* @param domPropertyName name of dom property
* @param expectedDomPropertyValue expected value of dom property
* @since 7.4.0
*/
public static WebElementCondition domProperty(String domPropertyName, @Nullable String expectedDomPropertyValue) {
return new DomPropertyValue(domPropertyName, expectedDomPropertyValue);
}
/**
* Checks if element matches the given predicate.
*
* Sample: {@code $("input").should(match("empty value attribute", el -> el.getAttribute("value").isEmpty()));}
*
* @param description the description of the predicate
* @param predicate the {@link Predicate} to match
*/
public static WebElementCondition match(String description, Predicate predicate) {
return new CustomMatch(description, predicate);
}
/**
* Check if image is loaded.
*/
public static final WebElementCondition image = new IsImageLoaded();
/**
* Check if browser focus is currently in given element.
*/
public static final WebElementCondition focused = new Focused();
/**
* Checks that element is not disabled
*
* @see WebElement#isEnabled()
*/
public static final WebElementCondition enabled = new Enabled();
/**
* Checks that element is disabled
*
* @see WebElement#isEnabled()
*/
public static final WebElementCondition disabled = new Disabled();
/**
* Checks that element is selected (inputs like drop-downs etc.)
*
* @see WebElement#isSelected()
*/
public static final WebElementCondition selected = new Selected();
/**
* Checks that checkbox is checked
*
* @see WebElement#isSelected()
*/
public static final WebElementCondition checked = new Checked();
/**
* Negate given condition.
*
* Used for methods like $.shouldNot(exist), $.shouldNotBe(visible)
*
* Typically, you don't need to use it.
*/
public static WebElementCondition not(WebElementCondition condition) {
return condition.negate();
}
/**
* Check if element matches ALL given conditions.
* The method signature makes you to pass at least 2 conditions, otherwise it would be nonsense.
*
* @param name Name of this condition, like "empty" (meaning e.g. empty text AND empty value).
* @param condition1 first condition to match
* @param condition2 second condition to match
* @param conditions Other conditions to match
* @return logical AND for given conditions.
*/
public static WebElementCondition and(String name, WebElementCondition condition1, WebElementCondition condition2,
WebElementCondition... conditions) {
return new And(name, merge(condition1, condition2, conditions));
}
/**
* Synonym for {@link #and(String, WebElementCondition, WebElementCondition, WebElementCondition...)}.
* Useful for better readability.
*/
public static WebElementCondition allOf(String name, WebElementCondition condition1, WebElementCondition condition2,
WebElementCondition... conditions) {
return and(name, condition1, condition2, conditions);
}
/**
* Synonym for {@link #and(String, WebElementCondition, WebElementCondition, WebElementCondition...)}
* with "all of" name. Useful for better readability.
*/
public static WebElementCondition allOf(WebElementCondition condition1, WebElementCondition condition2,
WebElementCondition... conditions) {
return and("all of", condition1, condition2, conditions);
}
/**
* Check if element matches ANY of given conditions.
* The method signature makes you to pass at least 2 conditions, otherwise it would be nonsense.
*
* Using "or" checks in tests is probably a flag of bad test design.
* Consider splitting this "or" check into two different methods or tests.
*
* @param name Name of this condition, like "error" (meaning e.g. "error" OR "failed").
* @param condition1 first condition to match
* @param condition2 second condition to match
* @param conditions Other conditions to match
* @return logical OR for given conditions.
* @see NOT RECOMMENDED
*/
public static WebElementCondition or(String name, WebElementCondition condition1, WebElementCondition condition2,
WebElementCondition... conditions) {
return new Or(name, merge(condition1, condition2, conditions));
}
/**
* Synonym for {@link #or(String, WebElementCondition, WebElementCondition, WebElementCondition...)}.
* Useful for better readability.
*/
public static WebElementCondition anyOf(String name, WebElementCondition condition1, WebElementCondition condition2,
WebElementCondition... conditions) {
return or(name, condition1, condition2, conditions);
}
/**
* Synonym for {@link #or(String, WebElementCondition, WebElementCondition, WebElementCondition...)}
* with "any of" name. Useful for better readability.
*/
public static WebElementCondition anyOf(WebElementCondition condition1, WebElementCondition condition2,
WebElementCondition... conditions) {
return or("any of", condition1, condition2, conditions);
}
/**
* Check if element is clickable: {@link #interactable} AND {@link #enabled}.
*
* Usually you don't need to use this condition.
* When you just call {@code $("button").click()}, Selenide automatically checks that the element is clickable.
*
*
*
Example:
* {@code $("input[type=button]").shouldBe(clickable);}
*
*
* @since 7.2.0
*/
public static final WebElementCondition clickable = and("clickable", interactable, enabled);
/**
* Used to form human-readable condition expression
* Example element.should(be(visible),have(text("abc"))
*
* @param delegate next condition to wrap
* @return WebElementCondition
*/
public static WebElementCondition be(WebElementCondition delegate) {
return wrap("be", delegate);
}
/**
* Used to form human-readable condition expression
* Example element.should(be(visible),have(text("abc"))
*
* @param delegate next condition to wrap
* @return WebElementCondition
*/
public static WebElementCondition have(WebElementCondition delegate) {
return wrap("have", delegate);
}
private static WebElementCondition wrap(String prefix, WebElementCondition delegate) {
return new NamedCondition(prefix, delegate);
}
}