SeleniumPageComponents 1.0.0

dotnet add package SeleniumPageComponents --version 1.0.0
                    
NuGet\Install-Package SeleniumPageComponents -Version 1.0.0
                    
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="SeleniumPageComponents" Version="1.0.0" />
                    
For projects that support PackageReference, copy this XML node into the project file to reference the package.
<PackageVersion Include="SeleniumPageComponents" Version="1.0.0" />
                    
Directory.Packages.props
<PackageReference Include="SeleniumPageComponents" />
                    
Project file
For projects that support Central Package Management (CPM), copy this XML node into the solution Directory.Packages.props file to version the package.
paket add SeleniumPageComponents --version 1.0.0
                    
#r "nuget: SeleniumPageComponents, 1.0.0"
                    
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
#:package SeleniumPageComponents@1.0.0
                    
#:package directive can be used in C# file-based apps starting in .NET 10 preview 4. Copy this into a .cs file before any lines of code to reference the package.
#addin nuget:?package=SeleniumPageComponents&version=1.0.0
                    
Install as a Cake Addin
#tool nuget:?package=SeleniumPageComponents&version=1.0.0
                    
Install as a Cake Tool

Selenium Page Components

Beyond Page Objects

Page objects are a great way to build robust reusable automated tests. However many times we need a way to build smaller, more modular, and reusable components on a page. This is how page-components was born! Page Components are a hybrid between a page object and an element...a page object that has an element as a root container. All elements in page component use the root element for searches, allowing us to use simple locators.

Overview

The page components library is intended to simplify the process of building reliable, scalable, and complex Selenium-WebDriver tests. Page-Components takes the concepts of the page object model and extends it, allowing users to build page objects as well as small, modular, reusable sub-components on pages (page-components). This helps build page objects which contain components that are reused across multiple pages, and also helps break page objects up into smaller reusable components that can be more easily identified and manipulated. It is primarily composed of a few classes that help make the process of building selenium page objects easier

  • Element - A class that acts as a wrapper around the IWebElement interface. It can be easily instantiated in a class header, or in a method. It handles implicit and explit waiting, automatically caches found IWebElement objects to avoid duplicate lookups, and automatically re-finds new references to IWebElements when they go stale. Essentially it handles the hard parts of Selenium-WebDriver, so you don't have to!

  • Elements - A class that acts as a List of Element objects. This allows you to iterate through the collection, to find a specific item, manipulate each one in turn, or do anything else you might need.

  • PageComponent - An abstract base class that is inherited by all the page objects or subcomponents in our tests. Implementing classes can be used as full page objects, including a URL and a Visit() function to open and instantiate them. Or they can be components, acting as a smaller section of a larger page object. This class inherits from Element, so it can be instantiated with a locator just like an Element, which acts as a root element or container for all further element queries.

  • PageComponentList - A class that acts as a List of PageComponents. Allows the user to instnatiate a list of page objects using one locator, and iterate through the collection. This allows us to build page components that are duplicated multiple times on the page, and iterate through them. It also allows us to quickly query them using LINQ methods to quickly select the appropriate component in the collection.

Usage

To start using, install via nuget : page-components.

Using the built in webdriver manager

You can simply inherit the WebTestBase class from our test classes, and webdriver instantiation will happen automatically before each test, and quit after. An extent html report will be generated including log messages and screenshots on test failure.

Configuration

The WebConfig class contains static values for all the configurable options, such as selecting the appropriate browser, running a remote session, and specifying the default timeout. Default values can be overridden by using the appropriate key in the app.config.

Creating Page Objects and Elements

To create a normal page object, create a new class and inherit the PageComponent class. You can specify a URL that can be used for navigation purposes.

public class LoginPage : PageComponent{
  public Element LoginField = new Element(By.CssSelector(".user"));
  public Element PasswordField = new Element(By.CssSelector(".password"));
  public Element SignInButton = new Element(By.CssSelector(".login"));
  public Elements ErrorMessages = new Elements(By.CssSelector(".error"));
  
  public HomePage Login(string username, string password)
  {
    LoginField.SetText(username);
    PasswordField.SetText(password);
    SignInButton.Click();
    return new HomePage();
  }
  
  public LoginPage AssertNoErrors(){
    ErrorMessages.Verify().Count(0);
  }
}

Creating a Component

Smaller reusable components can be built using the same base PageComponent abstract class, and can be included as part of a larger page object Components act as a page object with a root element. They are instantiated with a locator, and elements use the component as the root node for all searches. This helps to minimize search context when multiple elements are present that match a locator

When instantiating an Element inside a component, the component must be passed into the element as a root element. The component can be passed into the Element constructor using the "this" keyword (new Element(this, By.CssSelector(".class")), or the Element can be instantiated using a function in the component "this.Element(By.CssSelector(".class")"

public class HeaderComponent : PageComponent {
  public Element LogOutLink => new Element(this, By.CssSelector(".logout"));
  public Element ProfileLink => this.Element(By.CssSelector(".profile"));
  
  public LoginPage LogOut() {
    LogOutLink.Click();
    return new LoginPage();
  }
}

public class HomePage : PageComponent {
   public Header = new HeaderComponent(By.CssSelector(".header"));
}

PageComponentLists

Many times components are repeated on a page numerous times. This can be a row of a table, or a panel displayed in a grid.
In this case the PageComponentList class can be used to build a List of components. The PageComponentList is instantiated with a selector that returns the multiple elements on a page. Each iteration of the component uses one of those elements as the root element, and all searches will be inside of that element.

For example, the google results page. When a user searches for some text, the results page is displayed. A GoogleSearchResult is built representing a single google result in the list, containing the search link, title, and text.

The list of google results are represented by building PageComponentList, with a locator that returns the root element of each result in the list. We can now iterate through the collection, or use LINQ to query the collection. This allows me to avoid having to use static locators using an index (such as an xpath expression), or build complicated locators that reference multiple elements on the page. I can now easily click the link of a search result containing specific text.

//The GoogleSearchResult component represents a single google search result item on the google results page
public class GoogleSearchResult : BaseComponent
{
    //Each Element is instantiated using "this" to show that these elements are descendents of the component
    public Element ResultLink => this.Element("a>div");
    public Element ResultText => this.Element(".st");
    public Element ResultTitle => this.Element("a>h3");

}

//The GoogleSearchResultsPage represents the entire google search results page
public class GoogleSearchResultsPage : BasePageObject
{
    //The List of google search results.  Each item in the list will return one GoogleSearchResult found using the selector
    public PageComponentList<GoogleSearchResult> SearchResults = new PageComponentList<GoogleSearchResult>(".g");

    //We can easily query the list of search results using LINQ to find relative elements.  IN this case I want to click on the 
    //search results link for a result that contains specific text.  
    public void ClickLinkWithText(string searchText)
    {
        var result = SearchResults.First(x => x.ResultText.Text.Contains(searchText));
            result.ResultLink.Click();
    }
}
Product Compatible and additional computed target framework versions.
.NET net5.0 was computed.  net5.0-windows was computed.  net6.0 was computed.  net6.0-android was computed.  net6.0-ios was computed.  net6.0-maccatalyst was computed.  net6.0-macos was computed.  net6.0-tvos was computed.  net6.0-windows was computed.  net7.0 was computed.  net7.0-android was computed.  net7.0-ios was computed.  net7.0-maccatalyst was computed.  net7.0-macos was computed.  net7.0-tvos was computed.  net7.0-windows was computed.  net8.0 was computed.  net8.0-android was computed.  net8.0-browser was computed.  net8.0-ios was computed.  net8.0-maccatalyst was computed.  net8.0-macos was computed.  net8.0-tvos was computed.  net8.0-windows was computed.  net9.0 was computed.  net9.0-android was computed.  net9.0-browser was computed.  net9.0-ios was computed.  net9.0-maccatalyst was computed.  net9.0-macos was computed.  net9.0-tvos was computed.  net9.0-windows was computed.  net10.0 was computed.  net10.0-android was computed.  net10.0-browser was computed.  net10.0-ios was computed.  net10.0-maccatalyst was computed.  net10.0-macos was computed.  net10.0-tvos was computed.  net10.0-windows was computed. 
.NET Core netcoreapp2.0 was computed.  netcoreapp2.1 was computed.  netcoreapp2.2 was computed.  netcoreapp3.0 was computed.  netcoreapp3.1 was computed. 
.NET Standard netstandard2.0 is compatible.  netstandard2.1 was computed. 
.NET Framework net461 was computed.  net462 was computed.  net463 was computed.  net47 was computed.  net471 was computed.  net472 was computed.  net48 was computed.  net481 was computed. 
MonoAndroid monoandroid was computed. 
MonoMac monomac was computed. 
MonoTouch monotouch was computed. 
Tizen tizen40 was computed.  tizen60 was computed. 
Xamarin.iOS xamarinios was computed. 
Xamarin.Mac xamarinmac was computed. 
Xamarin.TVOS xamarintvos was computed. 
Xamarin.WatchOS xamarinwatchos was computed. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.

NuGet packages

This package is not used by any NuGet packages.

GitHub repositories

This package is not used by any popular GitHub repositories.

Version Downloads Last Updated
1.0.0 4,954 9/8/2020

Initial Release