com.codeborne.selenide.BaseElementsCollection Maven / Gradle / Ivy
Show all versions of selenide-core Show documentation
package com.codeborne.selenide;
import com.codeborne.selenide.ex.ElementNotFound;
import com.codeborne.selenide.ex.UIAssertionError;
import com.codeborne.selenide.impl.BySelectorCollection;
import com.codeborne.selenide.impl.Cleanup;
import com.codeborne.selenide.impl.CollectionElement;
import com.codeborne.selenide.impl.CollectionElementByCondition;
import com.codeborne.selenide.impl.CollectionSnapshot;
import com.codeborne.selenide.impl.CollectionSource;
import com.codeborne.selenide.impl.ElementCommunicator;
import com.codeborne.selenide.impl.FilteringCollection;
import com.codeborne.selenide.impl.HeadOfCollection;
import com.codeborne.selenide.impl.LastCollectionElement;
import com.codeborne.selenide.impl.SelenideElementIterator;
import com.codeborne.selenide.impl.TailOfCollection;
import com.codeborne.selenide.impl.WebElementsCollectionWrapper;
import com.codeborne.selenide.logevents.SelenideLog;
import com.codeborne.selenide.logevents.SelenideLogger;
import com.google.errorprone.annotations.CanIgnoreReturnValue;
import org.jspecify.annotations.Nullable;
import org.openqa.selenium.By;
import org.openqa.selenium.JavascriptException;
import org.openqa.selenium.UnsupportedCommandException;
import org.openqa.selenium.WebDriverException;
import org.openqa.selenium.WebElement;
import java.time.Duration;
import java.util.Collection;
import java.util.Iterator;
import java.util.List;
import java.util.stream.Stream;
import java.util.stream.StreamSupport;
import static com.codeborne.selenide.CheckResult.Verdict.ACCEPT;
import static com.codeborne.selenide.CheckResult.Verdict.REJECT;
import static com.codeborne.selenide.Condition.exist;
import static com.codeborne.selenide.Condition.not;
import static com.codeborne.selenide.commands.Util.classOf;
import static com.codeborne.selenide.impl.Plugins.inject;
import static com.codeborne.selenide.logevents.ErrorsCollector.validateAssertionMode;
import static com.codeborne.selenide.logevents.LogEvent.EventStatus.PASS;
public abstract class BaseElementsCollection>
implements Iterable {
private static final ElementCommunicator communicator = inject(ElementCommunicator.class);
private final Class clazz;
private final CollectionSource collection;
@SafeVarargs
protected BaseElementsCollection(CollectionSource collection, T... clazz) {
this.clazz = classOf(clazz);
this.collection = collection;
}
@SafeVarargs
protected BaseElementsCollection(Driver driver, Collection extends WebElement> elements, T... clazz) {
this(new WebElementsCollectionWrapper(driver, elements), clazz);
}
@SafeVarargs
protected BaseElementsCollection(Driver driver, String cssSelector, T... clazz) {
this(driver, By.cssSelector(cssSelector), clazz);
}
@SafeVarargs
protected BaseElementsCollection(Driver driver, By seleniumSelector, T... clazz) {
this(new BySelectorCollection(driver, seleniumSelector), clazz);
}
/**
* Check if a collection matches given condition(s).
*
* For example:
*
*
* {@code
* $$(".text_list").should(containExactTextsCaseSensitive("text1", "text2"));
* $$(".cat_list").should(allMatch("value==cat", el -> el.getAttribute("value").equals("cat")));
* }
*
*/
@CanIgnoreReturnValue
public SELF should(WebElementsCondition... conditions) {
return should("", Duration.ofMillis(driver().config().timeout()), conditions);
}
/**
* Check if a collection matches a given condition within the given time period.
*
* @param timeout maximum waiting time
*/
@CanIgnoreReturnValue
public SELF should(WebElementsCondition condition, Duration timeout) {
return should("", timeout, toArray(condition));
}
/**
* For example: {@code $$(".error").shouldBe(empty)}
*/
@CanIgnoreReturnValue
public SELF shouldBe(WebElementsCondition... conditions) {
return should("be", Duration.ofMillis(driver().config().timeout()), conditions);
}
@CanIgnoreReturnValue
public SELF shouldBe(WebElementsCondition condition, Duration timeout) {
return should("be", timeout, toArray(condition));
}
/**
* For example:
* {@code $$(".error").shouldHave(size(3))}
* {@code $$(".error").shouldHave(texts("Error1", "Error2"))}
*/
@CanIgnoreReturnValue
public SELF shouldHave(WebElementsCondition... conditions) {
return should("have", Duration.ofMillis(driver().config().timeout()), conditions);
}
/**
* Check if a collection matches given condition within given period
*
* @param timeout maximum waiting time
*/
@CanIgnoreReturnValue
public SELF shouldHave(WebElementsCondition condition, Duration timeout) {
return should("have", timeout, toArray(condition));
}
private WebElementsCondition[] toArray(WebElementsCondition condition) {
return new WebElementsCondition[]{condition};
}
@SuppressWarnings("ErrorNotRethrown")
protected SELF should(String prefix, Duration timeout, WebElementsCondition... conditions) {
validateAssertionMode(driver().config());
SelenideLog log = SelenideLogger.beginStep(collection.shortDescription(), "should " + prefix, (Object[]) conditions);
try {
for (WebElementsCondition condition : conditions) {
waitUntil(condition, timeout);
}
SelenideLogger.commitStep(log, PASS);
return self();
}
catch (AssertionError error) {
Error wrappedError = UIAssertionError.wrap(driver(), error, timeout.toMillis());
SelenideLogger.commitStep(log, wrappedError);
return switch (driver().config().assertionMode()) {
case SOFT -> self();
default -> throw wrappedError;
};
}
catch (RuntimeException e) {
SelenideLogger.commitStep(log, e);
throw e;
}
}
@SuppressWarnings("unchecked")
private SELF self() {
return (SELF) this;
}
@SuppressWarnings("ErrorNotRethrown")
protected void waitUntil(WebElementsCondition condition, Duration timeout) {
Throwable lastError = null;
CheckResult lastCheckResult = new CheckResult(REJECT, null);
Stopwatch stopwatch = new Stopwatch(timeout);
do {
try {
lastCheckResult = condition.check(collection);
if (lastCheckResult.verdict() == ACCEPT) {
return;
}
}
catch (JavascriptException | UnsupportedCommandException e) {
throw e;
}
catch (WebDriverException | IndexOutOfBoundsException | UIAssertionError elementNotFound) {
if (Cleanup.of.isInvalidSelectorError(elementNotFound)) {
throw Cleanup.of.wrapInvalidSelectorException(elementNotFound);
}
if (condition.missingElementsSatisfyCondition()) {
return;
}
lastError = elementNotFound;
}
sleep(stopwatch);
}
while (!stopwatch.isTimeoutReached());
if (lastError instanceof IndexOutOfBoundsException) {
throw new ElementNotFound(collection.getAlias(), collection.description(), exist, lastError);
}
else if (lastError instanceof UIAssertionError uiAssertionError) {
throw uiAssertionError;
}
else {
condition.fail(collection, lastCheckResult, (Exception) lastError, timeout.toMillis());
}
}
void sleep(Stopwatch stopwatch) {
stopwatch.sleep(driver().config().pollingInterval());
}
/**
* Filters collection elements based on the given condition (lazy evaluation)
*
* @param condition condition
* @return ElementsCollection
* @see Lazy loading
*/
public SELF filter(WebElementCondition condition) {
return create(new FilteringCollection(collection, condition));
}
protected abstract SELF create(CollectionSource source);
/**
* Filters collection elements based on the given condition (lazy evaluation)
*
* @param condition condition
* @return ElementsCollection
* @see #filter(WebElementCondition)
* @see Lazy loading
*/
public SELF filterBy(WebElementCondition condition) {
return filter(condition);
}
/**
* Filters elements excluding those which met the given condition (lazy evaluation)
*
* @param condition condition
* @return ElementsCollection
* @see Lazy loading
*/
public SELF exclude(WebElementCondition condition) {
return create(new FilteringCollection(collection, not(condition)));
}
/**
* Filters elements excluding those which met the given condition (lazy evaluation)
*
* @param condition condition
* @return ElementsCollection
* @see #exclude(WebElementCondition)
* @see Lazy loading
*/
public SELF excludeWith(WebElementCondition condition) {
return exclude(condition);
}
/**
* Find the first element which met the given condition (lazy evaluation)
*
* @param condition condition
* @return SelenideElement
* @see Lazy loading
*/
public SelenideElement find(WebElementCondition condition) {
return CollectionElementByCondition.wrap(collection, condition);
}
/**
* Find the first element which met the given condition (lazy evaluation)
*
* @param condition condition
* @return SelenideElement
* @see #find(WebElementCondition)
* @see Lazy loading
*/
public SelenideElement findBy(WebElementCondition condition) {
return find(condition);
}
private List getElements() {
return collection.getElements();
}
/**
* Gets all the texts in elements collection
* @see NOT RECOMMENDED
* Instead of just getting texts, we highly recommend to verify them with {@code $$.shouldHave(texts(...));}.
*/
public List texts() {
return communicator.texts(driver(), getElements());
}
/**
* Gets all the specific attribute values in elements collection
* @see NOT RECOMMENDED
* Instead of just getting attributes, we highly recommend to verify them with {@code $$.shouldHave(attributes(...));}.
*/
public List<@Nullable String> attributes(String attribute) {
return communicator.attributes(driver(), getElements(), attribute);
}
/**
* Gets the n-th element of collection (lazy evaluation)
*
* @param index 0..N
* @return the n-th element of collection
* @see Lazy loading
*/
public T get(int index) {
return CollectionElement.wrap(clazz, collection, index);
}
/**
*
* returns the first element of the collection (lazy evaluation)
*
*
*
* NOTICE: Instead of {@code $$(css).first()}, prefer {@code $(css)} as it's faster and returns the same result
*
*
* @return the first element of the collection
* @see Lazy loading
*/
public SelenideElement first() {
return get(0);
}
/**
* returns the last element of the collection (lazy evaluation)
*
* @return the last element of the collection
* @see Lazy loading
*/
public SelenideElement last() {
return LastCollectionElement.wrap(collection);
}
/**
* returns the first n elements of the collection (lazy evaluation)
*
* @param elements number of elements 1…N
* @see Lazy loading
*/
public SELF first(int elements) {
return create(new HeadOfCollection(collection, elements));
}
/**
* returns the last n elements of the collection (lazy evaluation)
*
* @param elements number of elements 1…N
* @see Lazy loading
*/
public SELF last(int elements) {
return create(new TailOfCollection(collection, elements));
}
/**
*
* return actual size of the collection, doesn't wait until collection is fully loaded.
*
*
* @return actual size of the collection
* @see NOT RECOMMENDED
*/
public int size() {
try {
return getElements().size();
}
catch (IndexOutOfBoundsException outOfCollection) {
return 0;
}
}
/**
*
* return actual state of the collection, doesn't wait until collection is fully loaded.
*
*
* @return actual size of the collection
* @see NOT RECOMMENDED
*/
public boolean isEmpty() {
return size() == 0;
}
/**
* Takes the snapshot of current state of this collection.
* Succeeding calls to this object WILL NOT RELOAD collection element from browser.
*
* Use it to speed up your tests - but only if you know that collection will not be changed during the test.
*
* @return current state of this collection
* @see #asFixedIterable()
*/
public SELF snapshot() {
return create(new CollectionSnapshot(collection));
}
/**
* Does not reload collection elements while iterating it.
*
* Not recommended: As a rule, tests should not iterate collection elements.
* Instead, try to write a {@link CollectionCondition} which verifies the whole collection.
*
* @see NOT RECOMMENDED
*
* To make it explicit, we recommend to use method {@link #asFixedIterable()} or {@link #asDynamicIterable()} instead.
*/
@Override
public Iterator iterator() {
return asFixedIterable().iterator();
}
/**
* Does not reload collection elements while iterating it.
*
* Not recommended: As a rule, tests should not iterate collection elements.
* Instead, try to write a {@link CollectionCondition} which verifies the whole collection.
*
* @see NOT RECOMMENDED
*/
public Stream stream() {
return StreamSupport.stream(spliterator(), false);
}
/**
* Returns a "static" {@link Iterable} which doesn't reload web elements during iteration.
*
* It's faster than {@link #asDynamicIterable()}},
* but can sometimes can cause {@link org.openqa.selenium.StaleElementReferenceException} etc.
* if elements are re-rendered during the iteration.
*
* @see NOT RECOMMENDED
*/
public SelenideElementIterable asFixedIterable() {
return () -> new SelenideElementIterator<>(new CollectionSnapshot(collection), clazz);
}
/**
* Returns a "dynamic" {@link Iterable} which reloads web elements during iteration.
*
* It's slower than {@link #asFixedIterable()}, but helps to avoid {@link org.openqa.selenium.StaleElementReferenceException} etc.
*
* @see NOT RECOMMENDED
*/
public SelenideElementIterable asDynamicIterable() {
return () -> new SelenideElementIterator<>(collection, clazz);
}
/**
* Give this collection a human-readable name
*
* Caution: you probably don't need this method.
* It's always a good idea to have the actual selector instead of "nice" description (which might be misleading or even lying).
*
* @param alias a human-readable name of this collection (null or empty string not allowed)
* @return this collection
*/
@CanIgnoreReturnValue
public SELF as(String alias) {
this.collection.setAlias(alias);
return self();
}
@Override
public String toString() {
return collection.getAlias().getOrElse(collection::toString);
}
/**
* Displays the collection in human-readable format.
* Useful for logging and debugging.
* Not recommended to use for test verifications.
* May work relatively slowly because it fetches actual element information from browser.
*
* @since 7.4.0
* @return e.g. [Order has been confirmed]
* @see com.codeborne.selenide.commands.DescribeElement
* @see NOT RECOMMENDED
*/
public String describe() {
return stream().map(el -> el.describe()).toList().toString();
}
private Driver driver() {
return collection.driver();
}
@FunctionalInterface
public interface SelenideElementIterable extends Iterable {
default Stream stream() {
return StreamSupport.stream(spliterator(), false);
}
}
}