Mi piace molto il modo in cui il selenio 2 per convenzione ti spinge ad usare PageObjects come POJOs e quindi semplicemente usando PageFactory per istanziare i campi di questa classe.Oggetto selenio Riutilizzo degli oggetti
Quello che sto trovando limitante è che riutilizziamo molti elementi su molte pagine diverse. Il grosso problema è che questi componenti riutilizzati non hanno lo stesso id/nome quando appaiono su pagine diverse; tuttavia i test che eseguiremo per ciascuno di essi sono gli stessi.
Ad esempio, raccogliamo le date in molti posti. Quindi un oggetto ad esempio la pagina di questo potrebbe essere (mese, giorno campi rimossi):
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
}
}
Poi ho potuto semplicemente testare questo con il codice qui sotto:
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();
}
}
Quello che mi piacerebbe davvero fare è una configurazione in cui con Spring posso iniettare quell'id/name/xpath, ecc ... nell'applicazione.
C'è un modo per farlo, senza perdere la possibilità di utilizzare PageFactory?
Modifica 1 - Aggiunta di classi di livello base ideali, lavorando su localizzatori personalizzati e fabbriche.
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 - Proprio come una nota, a patto che si esegue selenio 2.0.a5 o superiore, è ora possibile dare al conducente un valore di timeout implicita.
Quindi è possibile sostituire il codice con:
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);
}
}
Questo ha sicuramente mi ha ottenuto sulla pista corretta. Speravo di estenderlo ulteriormente oltre ai soli ID (potevi iniettare ogni mezzo necessario e quindi individuare). Il mio problema era che non stavo vedendo come venivano creati quei Field, il che mi stava impedendo di afferrare correttamente l'ElementLocator. – Scott