Ich mag wirklich, wie Selen 2 per Konvention Sie dazu treibt, PageObjects als POJOs zu verwenden, und dann einfach die PageFactory zu verwenden, um die Felder in dieser Klasse zu instanziieren.Selenium Page Object Reuse
Was ich einschränkend finde, ist, dass wir viele Elemente auf vielen verschiedenen Seiten wiederverwenden. Das große Problem ist, dass diese wiederverwendeten Komponenten nicht dieselbe ID/denselben Namen haben, wenn sie auf verschiedenen Seiten erscheinen. Die Tests, die wir für jeden von ihnen durchführen würden, sind jedoch gleich.
Als Beispiel sammeln wir Daten an vielen Orten. So ein Beispiel Seitenobjekt für diese sein könnte (Monat, Tag Felder entfernt):
public class DatePageObject {
private WebDriver driver;
DatePageObject(WebDriver driver) {
this.driver = driver;
}
@FindBy(id = "someIdForThisInstance")
private WebElement year;
public void testYearNumeric() {
this.year.sendKeys('aa');
this.year.submit();
//Logic to determine Error message shows up
}
}
Dann konnte ich einfach dies testen mit dem folgenden Code:
public class Test {
public static void main(String[] args) {
WebDriver driver = new FirefoxDriver();
DatePageObject dpo = PageFactory.initElements(driver, DriverPageObject.class);
driver.get("Some URL");
dpo.testYearNumeric();
}
}
Was ich wirklich tun möchte ist ein Setup, wobei mit Spring ich diese ID/Name/XPath, etc ... in die Anwendung injizieren kann.
Gibt es eine Möglichkeit, dies zu tun, ohne die Fähigkeit zu verlieren, die PageFactory zu verwenden?
Edit 1 - Hinzufügen von idealen Basislevel-Klassen, Arbeiten mit Custom Locators und Factories.
public class PageElement {
private WebElement element;
private How how;
private String using;
PageElement(How how, String using) {
this.how = how;
this.using = using;
}
//Getters and Setters
}
public class PageWidget {
private List<PageElement> widgetElements;
}
public class Screen {
private List<PageWidget> fullPage;
private WebDriver driver;
public Screen(WebDriver driver) {
this.driver = driver;
for (PageWidget pw : fullPage) {
CustomPageFactory.initElements(driver, pw.class);
}
}
Edit 2 - Nur als Anmerkung, solange Sie Selen 2.0.a5 oder höher ausgeführt wird, können Sie nun dem Fahrer einen impliziten Timeout-Wert geben.
So können Sie Ihren Code mit ersetzen:
private class CustomElementLocator implements ElementLocator {
private WebDriver driver;
private int timeOutInSeconds;
private final By by;
public CustomElementLocator(WebDriver driver, Field field,
int timeOutInSeconds) {
this.driver = driver;
this.timeOutInSeconds = timeOutInSeconds;
CustomAnnotations annotations = new CustomAnnotations(field);
this.by = annotations.buildBy();
driver.manage().timeouts().implicitlyWait(15, TimeUnit.SECONDS); //Set this value in a more realistic place
}
public WebElement findElement() {
return driver.findElement(by);
}
}
Das bin ich auf jeden Fall auf der richtigen Spur bekommen hat. Ich hatte gehofft, es weiter als nur ID's zu erweitern (du könntest alle notwendigen Mittel injizieren und dann lokalisieren). Mein Problem war, dass ich nicht sah, wie diese Fields erschaffen wurden - was dazu führte, dass ich den ElementLocator nicht richtig verstand. – Scott