diff --git a/bellatrix.android/src/main/java/solutions/bellatrix/android/components/AndroidComponent.java b/bellatrix.android/src/main/java/solutions/bellatrix/android/components/AndroidComponent.java index 9a34321d..c1580d81 100644 --- a/bellatrix.android/src/main/java/solutions/bellatrix/android/components/AndroidComponent.java +++ b/bellatrix.android/src/main/java/solutions/bellatrix/android/components/AndroidComponent.java @@ -22,10 +22,7 @@ import org.openqa.selenium.interactions.Actions; import solutions.bellatrix.android.components.contracts.Component; import solutions.bellatrix.android.configuration.AndroidSettings; -import solutions.bellatrix.android.findstrategies.FindStrategy; -import solutions.bellatrix.android.findstrategies.NameFindStrategy; -import solutions.bellatrix.android.findstrategies.TagFindStrategy; -import solutions.bellatrix.android.findstrategies.XPathFindStrategy; +import solutions.bellatrix.android.findstrategies.*; import solutions.bellatrix.android.infrastructure.DriverService; import solutions.bellatrix.android.services.AppService; import solutions.bellatrix.android.services.ComponentCreateService; @@ -36,6 +33,7 @@ import solutions.bellatrix.core.utilities.DebugInformation; import solutions.bellatrix.core.utilities.InstanceFactory; import solutions.bellatrix.core.utilities.Log; +import solutions.bellatrix.plugins.opencv.Base64Encodable; import java.awt.Dimension; import java.util.ArrayList; @@ -221,6 +219,10 @@ public return createAll(componentClass, findStrategy); } + public TComponent createByImage(Class componentClass, Base64Encodable encodedImage) { + return create(componentClass, new ImageBase64FindStrategy(encodedImage)); + } + public TComponent createByXPath(Class componentClass, String xpath) { return create(componentClass, new XPathFindStrategy(xpath)); } @@ -237,6 +239,11 @@ public List createAllByName(Cl return createAll(componentClass, new NameFindStrategy(name)); } + public List createAllByImage(Class componentClass, Base64Encodable encodedImage) { + return createAll(componentClass, new ImageBase64FindStrategy(encodedImage)); + } + + public List createAllByXPath(Class componentClass, String xpath) { return createAll(componentClass, new XPathFindStrategy(xpath)); } diff --git a/bellatrix.android/src/main/java/solutions/bellatrix/android/configuration/AndroidSettings.java b/bellatrix.android/src/main/java/solutions/bellatrix/android/configuration/AndroidSettings.java index de101408..c2f1f156 100644 --- a/bellatrix.android/src/main/java/solutions/bellatrix/android/configuration/AndroidSettings.java +++ b/bellatrix.android/src/main/java/solutions/bellatrix/android/configuration/AndroidSettings.java @@ -41,4 +41,6 @@ public class AndroidSettings { @Getter @Setter private Boolean videosOnFailEnabled; @Getter @Setter private String videosSaveLocation; + + @Getter @Setter private Boolean allowImageFindStrategies; } diff --git a/bellatrix.android/src/main/java/solutions/bellatrix/android/findstrategies/ImageBase64FindStrategy.java b/bellatrix.android/src/main/java/solutions/bellatrix/android/findstrategies/ImageBase64FindStrategy.java new file mode 100644 index 00000000..fd1a21a4 --- /dev/null +++ b/bellatrix.android/src/main/java/solutions/bellatrix/android/findstrategies/ImageBase64FindStrategy.java @@ -0,0 +1,54 @@ +/* + * Copyright 2025 Automate The Planet Ltd. + * Author: Miriyam Kyoseva + * Licensed under the Apache License, Version 2.0 (the "License"); + * You may not use this file except in compliance with the License. + * You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package solutions.bellatrix.android.findstrategies; + +import io.appium.java_client.AppiumBy; +import io.appium.java_client.android.AndroidDriver; +import org.openqa.selenium.WebElement; +import solutions.bellatrix.plugins.opencv.Base64Encodable; + +import java.util.List; + +public class ImageBase64FindStrategy extends FindStrategy { + private final Base64Encodable encodedImage; + public ImageBase64FindStrategy(Base64Encodable encodedImage) { + super(encodedImage.getImageName()); + this.encodedImage = encodedImage; + } + + @Override + public WebElement findElement(AndroidDriver driver) { + return driver.findElement(AppiumBy.image(encodedImage.getBase64Image())); + } + + @Override + public List findAllElements(AndroidDriver driver) { + return driver.findElements(AppiumBy.image(encodedImage.getBase64Image())); + } + + @Override + public WebElement findElement(WebElement element) { + return element.findElement(AppiumBy.image(encodedImage.getBase64Image())); + } + + @Override + public List findAllElements(WebElement element) { + return element.findElements(AppiumBy.image(encodedImage.getBase64Image())); + } + + @Override + public String toString() { + return String.format("image = %s", getValue()); + } +} diff --git a/bellatrix.android/src/main/java/solutions/bellatrix/android/infrastructure/DriverService.java b/bellatrix.android/src/main/java/solutions/bellatrix/android/infrastructure/DriverService.java index 3bbfb124..fcec337b 100644 --- a/bellatrix.android/src/main/java/solutions/bellatrix/android/infrastructure/DriverService.java +++ b/bellatrix.android/src/main/java/solutions/bellatrix/android/infrastructure/DriverService.java @@ -107,6 +107,9 @@ private static AndroidDriver initializeDriverGridMode(GridSettings gridSettings, options.put("name", testName); caps.setCapability(gridSettings.getOptionsName(), options); + if (ConfigurationService.get(AndroidSettings.class).getAllowImageFindStrategies()) + caps.setCapability("use-plugins", "images"); + AndroidDriver driver = null; try { driver = new AndroidDriver(new URL(gridSettings.getUrl()), caps); @@ -134,6 +137,9 @@ private static AndroidDriver initializeDriverRegularMode(String serviceUrl) { caps.setAppActivity(getAppConfiguration().getAppActivity()); } + if (ConfigurationService.get(AndroidSettings.class).getAllowImageFindStrategies()) + caps.setCapability("use-plugins", "images"); + addDriverConfigOptions(caps); addCustomDriverOptions(caps); var driver = new AndroidDriver(new URL(serviceUrl), caps); diff --git a/bellatrix.android/src/main/java/solutions/bellatrix/android/infrastructure/MobileScreenshotPlugin.java b/bellatrix.android/src/main/java/solutions/bellatrix/android/infrastructure/MobileScreenshotPlugin.java index 61250771..f4b6bfa0 100644 --- a/bellatrix.android/src/main/java/solutions/bellatrix/android/infrastructure/MobileScreenshotPlugin.java +++ b/bellatrix.android/src/main/java/solutions/bellatrix/android/infrastructure/MobileScreenshotPlugin.java @@ -13,8 +13,6 @@ package solutions.bellatrix.android.infrastructure; -import lombok.SneakyThrows; -import org.apache.commons.io.FileUtils; import org.openqa.selenium.OutputType; import org.openqa.selenium.TakesScreenshot; import plugins.screenshots.ScreenshotPlugin; @@ -35,7 +33,11 @@ public MobileScreenshotPlugin() { } @Override - @SneakyThrows + public byte[] takeScreenshot() { + return ((TakesScreenshot)DriverService.getWrappedAndroidDriver()).getScreenshotAs(OutputType.BYTES); + } + + @Override public String takeScreenshot(String name) { var screenshotSaveDir = getOutputFolder(); var filename = getUniqueFileName(name); @@ -57,7 +59,6 @@ public String takeScreenshot(String name) { } @Override - @SneakyThrows public String takeScreenshot(String screenshotSaveDir, String filename) { var screenshot = ((TakesScreenshot)DriverService.getWrappedAndroidDriver()).getScreenshotAs(OutputType.BASE64); diff --git a/bellatrix.android/src/main/java/solutions/bellatrix/android/services/ComponentCreateService.java b/bellatrix.android/src/main/java/solutions/bellatrix/android/services/ComponentCreateService.java index e8ebb4f6..c1c49508 100644 --- a/bellatrix.android/src/main/java/solutions/bellatrix/android/services/ComponentCreateService.java +++ b/bellatrix.android/src/main/java/solutions/bellatrix/android/services/ComponentCreateService.java @@ -17,6 +17,7 @@ import solutions.bellatrix.android.findstrategies.*; import solutions.bellatrix.android.infrastructure.DriverService; import solutions.bellatrix.core.utilities.InstanceFactory; +import solutions.bellatrix.plugins.opencv.Base64Encodable; import java.util.ArrayList; import java.util.List; @@ -32,6 +33,10 @@ public return allBy(componentClass, findStrategy); } + public TComponent byImage(Class componentClass, Base64Encodable encodedImage) { + return by(componentClass, new ImageBase64FindStrategy(encodedImage)); + } + public TComponent byId(Class componentClass, String id) { return by(componentClass, new IdFindStrategy(id)); } @@ -76,6 +81,10 @@ public TComponent byIdContaining(Class List allByImage(Class componentClass, Base64Encodable encodedImage) { + return allBy(componentClass, new ImageBase64FindStrategy(encodedImage)); + } + public List allById(Class componentClass, String automationId) { return allBy(componentClass, new IdFindStrategy(automationId)); } diff --git a/bellatrix.desktop/pom.xml b/bellatrix.desktop/pom.xml index 759eff0a..198883f7 100644 --- a/bellatrix.desktop/pom.xml +++ b/bellatrix.desktop/pom.xml @@ -48,5 +48,11 @@ 1.0 compile + + solutions.bellatrix + bellatrix.plugins.opencv + 1.0 + compile + \ No newline at end of file diff --git a/bellatrix.desktop/src/main/java/solutions/bellatrix/desktop/components/DesktopComponent.java b/bellatrix.desktop/src/main/java/solutions/bellatrix/desktop/components/DesktopComponent.java index 928aeb88..7a07911a 100644 --- a/bellatrix.desktop/src/main/java/solutions/bellatrix/desktop/components/DesktopComponent.java +++ b/bellatrix.desktop/src/main/java/solutions/bellatrix/desktop/components/DesktopComponent.java @@ -34,6 +34,7 @@ import solutions.bellatrix.desktop.services.ComponentCreateService; import solutions.bellatrix.desktop.services.ComponentWaitService; import solutions.bellatrix.desktop.waitstrategies.*; +import solutions.bellatrix.plugins.opencv.Base64Encodable; import java.awt.Dimension; import java.util.ArrayList; @@ -247,6 +248,11 @@ public TComponent createByAutomationId(Cla return create(componentClass, new AutomationIdFindStrategy(automationId)); } + public TComponent createByImage(Class componentClass, Base64Encodable encodedImage) { + return create(componentClass, new ImageBase64FindStrategy(encodedImage)); + } + + public List createAllByAccessibilityId(Class componentClass, String accessibilityId) { return createAll(componentClass, new AccessibilityIdFindStrategy(accessibilityId)); } @@ -275,6 +281,10 @@ public List createAllByIdConta return createAll(componentClass, new IdContainingFindStrategy(idContaining)); } + public List createAllByImage(Class componentClass, Base64Encodable encodedImage) { + return createAll(componentClass, new ImageBase64FindStrategy(encodedImage)); + } + protected TComponent create(Class componentClass, TFindStrategy findStrategy) { CREATING_ELEMENT.broadcast(new ComponentActionEventArgs(this)); findElement(); diff --git a/bellatrix.desktop/src/main/java/solutions/bellatrix/desktop/configuration/DesktopSettings.java b/bellatrix.desktop/src/main/java/solutions/bellatrix/desktop/configuration/DesktopSettings.java index 3575b6dd..458cffc8 100644 --- a/bellatrix.desktop/src/main/java/solutions/bellatrix/desktop/configuration/DesktopSettings.java +++ b/bellatrix.desktop/src/main/java/solutions/bellatrix/desktop/configuration/DesktopSettings.java @@ -47,4 +47,6 @@ public class DesktopSettings { @Getter @Setter private Boolean videosOnFailEnabled; @Getter @Setter private String videosSaveLocation; + + @Getter @Setter private Boolean allowImageFindStrategies; } diff --git a/bellatrix.desktop/src/main/java/solutions/bellatrix/desktop/findstrategies/ImageBase64FindStrategy.java b/bellatrix.desktop/src/main/java/solutions/bellatrix/desktop/findstrategies/ImageBase64FindStrategy.java new file mode 100644 index 00000000..b0637dec --- /dev/null +++ b/bellatrix.desktop/src/main/java/solutions/bellatrix/desktop/findstrategies/ImageBase64FindStrategy.java @@ -0,0 +1,54 @@ +/* + * Copyright 2025 Automate The Planet Ltd. + * Author: Miriyam Kyoseva + * Licensed under the Apache License, Version 2.0 (the "License"); + * You may not use this file except in compliance with the License. + * You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package solutions.bellatrix.desktop.findstrategies; + +import io.appium.java_client.AppiumBy; +import io.appium.java_client.windows.WindowsDriver; +import org.openqa.selenium.WebElement; +import solutions.bellatrix.plugins.opencv.Base64Encodable; + +import java.util.List; + +public class ImageBase64FindStrategy extends FindStrategy { + private final Base64Encodable encodedImage; + public ImageBase64FindStrategy(Base64Encodable encodedImage) { + super(encodedImage.getImageName()); + this.encodedImage = encodedImage; + } + + @Override + public WebElement findElement(WindowsDriver driver) { + return driver.findElement(AppiumBy.image(encodedImage.getBase64Image())); + } + + @Override + public List findAllElements(WindowsDriver driver) { + return driver.findElements(AppiumBy.image(encodedImage.getBase64Image())); + } + + @Override + public WebElement findElement(WebElement element) { + return element.findElement(AppiumBy.image(encodedImage.getBase64Image())); + } + + @Override + public List findAllElements(WebElement element) { + return element.findElements(AppiumBy.image(encodedImage.getBase64Image())); + } + + @Override + public String toString() { + return String.format("image = %s", getValue()); + } +} diff --git a/bellatrix.desktop/src/main/java/solutions/bellatrix/desktop/infrastructure/DesktopScreenshotPlugin.java b/bellatrix.desktop/src/main/java/solutions/bellatrix/desktop/infrastructure/DesktopScreenshotPlugin.java index b8c95926..d75b8929 100644 --- a/bellatrix.desktop/src/main/java/solutions/bellatrix/desktop/infrastructure/DesktopScreenshotPlugin.java +++ b/bellatrix.desktop/src/main/java/solutions/bellatrix/desktop/infrastructure/DesktopScreenshotPlugin.java @@ -13,8 +13,6 @@ package solutions.bellatrix.desktop.infrastructure; -import lombok.SneakyThrows; -import org.apache.commons.io.FileUtils; import org.openqa.selenium.OutputType; import org.openqa.selenium.TakesScreenshot; import plugins.screenshots.ScreenshotPlugin; @@ -35,7 +33,11 @@ public DesktopScreenshotPlugin() { } @Override - @SneakyThrows + public byte[] takeScreenshot() { + return ((TakesScreenshot)DriverService.getWrappedDriver()).getScreenshotAs(OutputType.BYTES); + } + + @Override public String takeScreenshot(String name) { var screenshotSaveDir = getOutputFolder(); var filename = getUniqueFileName(name); @@ -56,7 +58,6 @@ public String takeScreenshot(String name) { } @Override - @SneakyThrows public String takeScreenshot(String screenshotSaveDir, String filename) { var screenshot = ((TakesScreenshot)DriverService.getWrappedDriver()).getScreenshotAs(OutputType.BASE64); Path path = Paths.get(screenshotSaveDir, filename); diff --git a/bellatrix.desktop/src/main/java/solutions/bellatrix/desktop/infrastructure/DriverService.java b/bellatrix.desktop/src/main/java/solutions/bellatrix/desktop/infrastructure/DriverService.java index f0945606..68faf44c 100644 --- a/bellatrix.desktop/src/main/java/solutions/bellatrix/desktop/infrastructure/DriverService.java +++ b/bellatrix.desktop/src/main/java/solutions/bellatrix/desktop/infrastructure/DriverService.java @@ -88,6 +88,9 @@ private static WindowsDriver initializeDriverGridMode(GridSettings gridSettings) caps.setApp(getAppConfiguration().getAppPath().replace("\\", "/")); caps.setAppWorkingDir(new File(getAppConfiguration().getAppPath()).getParent()); + if (ConfigurationService.get(DesktopSettings.class).getAllowImageFindStrategies()) + caps.setCapability("use-plugins", "images"); + WindowsDriver driver = null; try { driver = new WindowsDriver(new URL(gridSettings.getUrl()), caps); diff --git a/bellatrix.desktop/src/main/java/solutions/bellatrix/desktop/services/ComponentCreateService.java b/bellatrix.desktop/src/main/java/solutions/bellatrix/desktop/services/ComponentCreateService.java index 56a32a2a..6892051f 100644 --- a/bellatrix.desktop/src/main/java/solutions/bellatrix/desktop/services/ComponentCreateService.java +++ b/bellatrix.desktop/src/main/java/solutions/bellatrix/desktop/services/ComponentCreateService.java @@ -17,6 +17,7 @@ import solutions.bellatrix.desktop.components.DesktopComponent; import solutions.bellatrix.desktop.findstrategies.*; import solutions.bellatrix.desktop.infrastructure.DriverService; +import solutions.bellatrix.plugins.opencv.Base64Encodable; import java.util.ArrayList; import java.util.List; @@ -60,6 +61,10 @@ public TComponent byIdContaining(Class TComponent byImage(Class componentClass, Base64Encodable encodedImage) { + return by(componentClass, new ImageBase64FindStrategy(encodedImage)); + } + public List allByAutomationId(Class componentClass, String automationId) { return allBy(componentClass, new AutomationIdFindStrategy(automationId)); } @@ -88,6 +93,10 @@ public List allByIdContaining( return allBy(componentClass, new IdContainingFindStrategy(idContaining)); } + public List allByImage(Class componentClass, Base64Encodable encodedImage) { + return allBy(componentClass, new ImageBase64FindStrategy(encodedImage)); + } + public TComponent by(Class componentClass, TFindStrategy findStrategy) { var component = InstanceFactory.create(componentClass); component.setFindStrategy(findStrategy); diff --git a/bellatrix.ios/src/main/java/solutions/bellatrix/ios/components/IOSComponent.java b/bellatrix.ios/src/main/java/solutions/bellatrix/ios/components/IOSComponent.java index f4c3bb71..77cf666a 100644 --- a/bellatrix.ios/src/main/java/solutions/bellatrix/ios/components/IOSComponent.java +++ b/bellatrix.ios/src/main/java/solutions/bellatrix/ios/components/IOSComponent.java @@ -33,6 +33,7 @@ import solutions.bellatrix.ios.services.ComponentCreateService; import solutions.bellatrix.ios.services.ComponentWaitService; import solutions.bellatrix.ios.waitstrategies.*; +import solutions.bellatrix.plugins.opencv.Base64Encodable; import java.awt.Dimension; import java.util.ArrayList; @@ -249,6 +250,10 @@ public TComponent createByIOSClassChain(Class< return create(componentClass, new IOSClassChainFindStrategy(iosClassChain)); } + public TComponent createByImage(Class componentClass, Base64Encodable encodedImage) { + return create(componentClass, new ImageBase64FindStrategy(encodedImage)); + } + public List createAllByName(Class componentClass, String name) { return createAll(componentClass, new NameFindStrategy(name)); } @@ -281,6 +286,10 @@ public List createAllByIOSClassCha return createAll(componentClass, new IOSClassChainFindStrategy(iosClassChain)); } + public List createAllByImage(Class componentClass, Base64Encodable encodedImage) { + return createAll(componentClass, new ImageBase64FindStrategy(encodedImage)); + } + protected TComponent create(Class componentClass, TFindStrategy findStrategy) { CREATING_ELEMENT.broadcast(new ComponentActionEventArgs(this)); findElement(); diff --git a/bellatrix.ios/src/main/java/solutions/bellatrix/ios/configuration/IOSSettings.java b/bellatrix.ios/src/main/java/solutions/bellatrix/ios/configuration/IOSSettings.java index 4f9097d3..d2366657 100644 --- a/bellatrix.ios/src/main/java/solutions/bellatrix/ios/configuration/IOSSettings.java +++ b/bellatrix.ios/src/main/java/solutions/bellatrix/ios/configuration/IOSSettings.java @@ -50,4 +50,6 @@ public class IOSSettings { @Getter @Setter private Boolean videosOnFailEnabled; @Getter @Setter private String videosSaveLocation; + + @Getter @Setter private Boolean allowImageFindStrategies; } diff --git a/bellatrix.ios/src/main/java/solutions/bellatrix/ios/findstrategies/ImageBase64FindStrategy.java b/bellatrix.ios/src/main/java/solutions/bellatrix/ios/findstrategies/ImageBase64FindStrategy.java new file mode 100644 index 00000000..6e5b6bc2 --- /dev/null +++ b/bellatrix.ios/src/main/java/solutions/bellatrix/ios/findstrategies/ImageBase64FindStrategy.java @@ -0,0 +1,54 @@ +/* + * Copyright 2025 Automate The Planet Ltd. + * Author: Miriyam Kyoseva + * Licensed under the Apache License, Version 2.0 (the "License"); + * You may not use this file except in compliance with the License. + * You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package solutions.bellatrix.ios.findstrategies; + +import io.appium.java_client.AppiumBy; +import io.appium.java_client.ios.IOSDriver; +import org.openqa.selenium.WebElement; +import solutions.bellatrix.plugins.opencv.Base64Encodable; + +import java.util.List; + +public class ImageBase64FindStrategy extends FindStrategy { + private final Base64Encodable encodedImage; + public ImageBase64FindStrategy(Base64Encodable encodedImage) { + super(encodedImage.getImageName()); + this.encodedImage = encodedImage; + } + + @Override + public WebElement findElement(IOSDriver driver) { + return driver.findElement(AppiumBy.image(encodedImage.getBase64Image())); + } + + @Override + public List findAllElements(IOSDriver driver) { + return driver.findElements(AppiumBy.image(encodedImage.getBase64Image())); + } + + @Override + public WebElement findElement(WebElement element) { + return element.findElement(AppiumBy.image(encodedImage.getBase64Image())); + } + + @Override + public List findAllElements(WebElement element) { + return element.findElements(AppiumBy.image(encodedImage.getBase64Image())); + } + + @Override + public String toString() { + return String.format("image = %s", getValue()); + } +} diff --git a/bellatrix.ios/src/main/java/solutions/bellatrix/ios/infrastructure/DriverService.java b/bellatrix.ios/src/main/java/solutions/bellatrix/ios/infrastructure/DriverService.java index 52e71713..6c77402b 100644 --- a/bellatrix.ios/src/main/java/solutions/bellatrix/ios/infrastructure/DriverService.java +++ b/bellatrix.ios/src/main/java/solutions/bellatrix/ios/infrastructure/DriverService.java @@ -113,6 +113,9 @@ private static IOSDriver initializeDriverRegularMode(String serviceUrl) { caps.setApp(getAppConfiguration().getAppPath().replace("\\", "/")); } + if (ConfigurationService.get(IOSSettings.class).getAllowImageFindStrategies()) + caps.setCapability("use-plugins", "images"); + addDriverOptions(caps); var driver = new IOSDriver(new URL(serviceUrl), caps); solutions.bellatrix.web.infrastructure.DriverService.setWrappedDriver(driver); diff --git a/bellatrix.ios/src/main/java/solutions/bellatrix/ios/infrastructure/MobileScreenshotPlugin.java b/bellatrix.ios/src/main/java/solutions/bellatrix/ios/infrastructure/MobileScreenshotPlugin.java index 6e932d0b..018c3ce4 100644 --- a/bellatrix.ios/src/main/java/solutions/bellatrix/ios/infrastructure/MobileScreenshotPlugin.java +++ b/bellatrix.ios/src/main/java/solutions/bellatrix/ios/infrastructure/MobileScreenshotPlugin.java @@ -13,8 +13,6 @@ package solutions.bellatrix.ios.infrastructure; -import lombok.SneakyThrows; -import org.apache.commons.io.FileUtils; import org.openqa.selenium.OutputType; import org.openqa.selenium.TakesScreenshot; import plugins.screenshots.ScreenshotPlugin; @@ -35,7 +33,11 @@ public MobileScreenshotPlugin() { } @Override - @SneakyThrows + public byte[] takeScreenshot() { + return ((TakesScreenshot)DriverService.getWrappedIOSDriver()).getScreenshotAs(OutputType.BYTES); + } + + @Override public String takeScreenshot(String name) { var screenshotSaveDir = getOutputFolder(); var filename = getUniqueFileName(name); @@ -55,7 +57,6 @@ public String takeScreenshot(String name) { } @Override - @SneakyThrows public String takeScreenshot(String screenshotSaveDir, String filename) { var screenshot = ((TakesScreenshot)DriverService.getWrappedIOSDriver()).getScreenshotAs(OutputType.BASE64); var path = Paths.get(screenshotSaveDir, filename) + ".png"; diff --git a/bellatrix.ios/src/main/java/solutions/bellatrix/ios/services/ComponentCreateService.java b/bellatrix.ios/src/main/java/solutions/bellatrix/ios/services/ComponentCreateService.java index e01e0645..db9ec690 100644 --- a/bellatrix.ios/src/main/java/solutions/bellatrix/ios/services/ComponentCreateService.java +++ b/bellatrix.ios/src/main/java/solutions/bellatrix/ios/services/ComponentCreateService.java @@ -17,6 +17,7 @@ import solutions.bellatrix.ios.components.IOSComponent; import solutions.bellatrix.ios.findstrategies.*; import solutions.bellatrix.ios.infrastructure.DriverService; +import solutions.bellatrix.plugins.opencv.Base64Encodable; import java.util.ArrayList; import java.util.List; @@ -68,6 +69,10 @@ public TComponent byAccessibilityIdContaining( return by(componentClass, new AccessibilityIdFindStrategy(accessibilityId)); } + public TComponent byImage(Class componentClass, Base64Encodable encodedImage) { + return by(componentClass, new ImageBase64FindStrategy(encodedImage)); + } + public List allById(Class componentClass, String id) { return allBy(componentClass, new IdFindStrategy(id)); } @@ -104,6 +109,10 @@ public List allByAccessibilityId(C return allBy(componentClass, new AccessibilityIdFindStrategy(accessibilityId)); } + public List allByImage(Class componentClass, Base64Encodable encodedImage) { + return allBy(componentClass, new ImageBase64FindStrategy(encodedImage)); + } + public TComponent by(Class componentClass, TFindStrategy findStrategy) { var component = InstanceFactory.create(componentClass); component.setFindStrategy(findStrategy); diff --git a/bellatrix.playwright/pom.xml b/bellatrix.playwright/pom.xml index ef684b0d..a8099293 100644 --- a/bellatrix.playwright/pom.xml +++ b/bellatrix.playwright/pom.xml @@ -47,15 +47,30 @@ compile - net.lightbody.bmp - browsermob-core - 2.1.5 + solutions.bellatrix + bellatrix.plugins.opencv + 1.0 compile - solutions.bellatrix - bellatrix.layout - 1.0 + org.projectlombok + lombok + 1.18.30 + + + com.microsoft.playwright + playwright + 1.44.0 + + + org.junit.jupiter + junit-jupiter + 5.10.0 + + + net.lightbody.bmp + browsermob-core + 2.1.5 compile @@ -70,6 +85,12 @@ 5.4.0 compile + + solutions.bellatrix + bellatrix.layout + 1.0 + compile + \ No newline at end of file diff --git a/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/components/ActionImage.java b/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/components/ActionImage.java new file mode 100644 index 00000000..7beb8c1b --- /dev/null +++ b/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/components/ActionImage.java @@ -0,0 +1,63 @@ +package solutions.bellatrix.playwright.components; + +import com.microsoft.playwright.Mouse; +import com.microsoft.playwright.options.MouseButton; +import lombok.Getter; +import lombok.Setter; +import org.openqa.selenium.InvalidArgumentException; +import solutions.bellatrix.core.plugins.EventListener; +import solutions.bellatrix.playwright.components.common.ComponentActionEventArgs; +import solutions.bellatrix.playwright.findstrategies.ImageBase64FindStrategy; +import solutions.bellatrix.playwright.infrastructure.PlaywrightService; +import solutions.bellatrix.plugins.opencv.OpenCvService; + +import java.awt.*; + +@Getter +@Setter +public class ActionImage extends WebComponent { + public final static EventListener CLICKING = new EventListener<>(); + public final static EventListener HOVERING = new EventListener<>(); + + private final Mouse mouse = PlaywrightService.wrappedBrowser().getCurrentPage().mouse(); + + @Override + public Point getLocation() { + ImageBase64FindStrategy findStrategy; + try { + findStrategy = (ImageBase64FindStrategy)getFindStrategy(); + } catch (ClassCastException e) { + throw new InvalidArgumentException("Invalid image base 64 format"); + } + + var encodedImage = findStrategy.getEncodedImage(); + + var location = OpenCvService.getLocation(encodedImage, true); + return new Point((int)location.x + encodedImage.getXOffset(), (int)location.y + encodedImage.getYOffset()); + } + + public void click() { + CLICKING.broadcast(new ComponentActionEventArgs(this, null, "Coordinates: %d, %d".formatted(getLocation().x, getLocation().y))); + + mouse.click(getLocation().x, getLocation().y); + } + + public void rightClick() { + CLICKING.broadcast(new ComponentActionEventArgs(this, null, "Coordinates: %d, %d".formatted(getLocation().x, getLocation().y))); + + mouse.click(getLocation().x, getLocation().y, new Mouse.ClickOptions().setButton(MouseButton.RIGHT)); + } + + public void hover() { + HOVERING.broadcast(new ComponentActionEventArgs(this, null, "Coordinates: %d, %d".formatted(getLocation().x, getLocation().y))); + + mouse.move(getLocation().x, getLocation().y); + } + + public void dragAndDrop(ActionImage image) { + mouse.move(getLocation().x, getLocation().y); + mouse.down(); + mouse.move(image.getLocation().x, image.getLocation().y); + mouse.up(); + } +} \ No newline at end of file diff --git a/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/components/TextArea.java b/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/components/TextArea.java index f2523871..e49ebd26 100644 --- a/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/components/TextArea.java +++ b/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/components/TextArea.java @@ -26,7 +26,7 @@ public String getText() { String text = defaultGetText(); if (text.isEmpty()) { - return defaultGetValue(); + return getWrappedElement().inputValue(); } return text; diff --git a/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/components/TextInput.java b/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/components/TextInput.java index 43568f67..af477f64 100644 --- a/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/components/TextInput.java +++ b/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/components/TextInput.java @@ -26,7 +26,7 @@ public String getText() { String text = defaultGetText(); if (text.isEmpty()) { - return defaultGetValue(); + return getWrappedElement().inputValue(); } return text; diff --git a/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/components/WebComponent.java b/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/components/WebComponent.java index cc9e7418..163003ae 100644 --- a/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/components/WebComponent.java +++ b/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/components/WebComponent.java @@ -32,7 +32,6 @@ import solutions.bellatrix.playwright.components.common.ComponentActionEventArgs; import solutions.bellatrix.playwright.components.common.create.RelativeCreateService; import solutions.bellatrix.playwright.components.common.validate.Validator; -import solutions.bellatrix.playwright.components.common.webelement.FrameElement; import solutions.bellatrix.playwright.components.common.webelement.WebElement; import solutions.bellatrix.playwright.components.contracts.Component; import solutions.bellatrix.playwright.components.contracts.ComponentStyle; diff --git a/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/components/common/webelement/WebElement.java b/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/components/common/webelement/WebElement.java index 34109d6d..ac195208 100644 --- a/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/components/common/webelement/WebElement.java +++ b/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/components/common/webelement/WebElement.java @@ -31,11 +31,18 @@ */ @Getter public class WebElement { + public WebElement(List locators) { + wrappedLocators = locators; + wrappedLocator = wrappedLocators.get(0); + } + public WebElement(Locator locator) { wrappedLocator = locator; + wrappedLocators = wrappedLocator.all(); } private final Locator wrappedLocator; + private final List wrappedLocators; public WebElement getByAltText(String text, GetByAltTextOptions options) { return new WebElement(wrappedLocator.getByAltText(text, options.convertTo(Locator.GetByAltTextOptions.class))); @@ -143,8 +150,7 @@ public FrameElement locateFrame() { public List all() { List elements = new ArrayList<>(); - List locators = wrappedLocator.all(); - for (Locator locator : locators) { + for (Locator locator : wrappedLocators) { elements.add(new WebElement(locator)); } @@ -294,7 +300,7 @@ public WebElement filter() { } public WebElement first() { - return new WebElement(wrappedLocator.first()); + return new WebElement(wrappedLocators.get(0)); } public void focus(Locator.FocusOptions options) { @@ -399,11 +405,11 @@ public boolean isVisible() { } public WebElement last() { - return new WebElement(wrappedLocator.last()); + return new WebElement(wrappedLocators.get(wrappedLocator.count() - 1)); } public WebElement nth(int index) { - return new WebElement(wrappedLocator.nth(index)); + return new WebElement(wrappedLocators.get(index)); } public WebElement or(WebElement webElement) { diff --git a/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/findstrategies/ImageBase64FindStrategy.java b/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/findstrategies/ImageBase64FindStrategy.java new file mode 100644 index 00000000..58d03431 --- /dev/null +++ b/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/findstrategies/ImageBase64FindStrategy.java @@ -0,0 +1,115 @@ +/* + * Copyright 2025 Automate The Planet Ltd. + * Author: Miriam Kyoseva + * Licensed under the Apache License, Version 2.0 (the "License"); + * You may not use this file except in compliance with the License. + * You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package solutions.bellatrix.playwright.findstrategies; + +import com.microsoft.playwright.Locator; +import com.microsoft.playwright.Page; +import lombok.Getter; +import solutions.bellatrix.core.utilities.SingletonFactory; +import solutions.bellatrix.core.utilities.parsing.TypeParser; +import solutions.bellatrix.playwright.components.common.webelement.WebElement; +import solutions.bellatrix.playwright.services.JavaScriptService; +import solutions.bellatrix.plugins.opencv.Base64Encodable; +import solutions.bellatrix.plugins.opencv.OpenCvService; + +import java.awt.*; +import java.util.ArrayList; +import java.util.List; + +@Getter +public class ImageBase64FindStrategy extends FindStrategy { + private final Base64Encodable encodedImage; + + public ImageBase64FindStrategy(Base64Encodable encodedImage) { + this.encodedImage = encodedImage; + } + + @Override + public WebElement convert(Page page) { + var location = OpenCvService.getLocation(encodedImage, false); + + var foundLocators = findElementsOn(location, page); + + this.webElement = new WebElement(foundLocators); + + return webElement; + } + + @Override + public WebElement convert(WebElement webElement) { + var location = OpenCvService.getLocation(encodedImage, false); + + var foundLocators = findElementsOn(location, webElement.page()); + + this.webElement = new WebElement(foundLocators); + + return webElement; + } + + private static List findElementsOn(org.opencv.core.Point coordinates, Page page) { + var elementsXpaths = (List)page.evaluate("(%s)(%s, %s);".formatted(js, TypeParser.parse(coordinates.x, Integer.class), TypeParser.parse(coordinates.y, Integer.class))); + + var locators = new ArrayList(); + + for (var i = 0; i < elementsXpaths.size(); i++) { + if (!elementsXpaths.get(i).isBlank()) + locators.add(page.locator("/" + elementsXpaths.get(i))); + } + + return locators; + } + + private static String js = """ + function getElementsOn(x, y) { + function getAbsoluteXpath(element) { + function indexElement(el) { + let index = 1; + + let previousSibling = el.previousElementSibling; + while (previousSibling) { + if (previousSibling.nodeName.toLowerCase() === el.nodeName.toLowerCase()) { + index++; + } + previousSibling = previousSibling.previousElementSibling; + } + + return "/" + el.tagName.toLowerCase() + "[" + index + "]"; + } + + let xpath = []; + + let currentElement = element; + while (currentElement) { + if (currentElement.tagName.toLowerCase() === 'html' || currentElement.tagName.toLowerCase() === 'body' || currentElement.tagName.startsWith() === '#' || currentElement.tagName.toLowerCase() === "temporary-div") { + break; + } + + xpath.unshift(indexElement(currentElement)); + + currentElement = currentElement.parentElement; + } + return xpath.join(""); + } + + const elements = document.elementsFromPoint(x, y); + const elementsXpaths = []; + + for (let i = 0; i < elements.length; i++) { + elementsXpaths.push(getAbsoluteXpath(elements[i])); + } + + return elementsXpaths; + } + """; +} diff --git a/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/infrastructure/WebScreenshotPlugin.java b/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/infrastructure/WebScreenshotPlugin.java index e8c1ab98..f3cbb6f5 100644 --- a/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/infrastructure/WebScreenshotPlugin.java +++ b/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/infrastructure/WebScreenshotPlugin.java @@ -30,6 +30,14 @@ public WebScreenshotPlugin() { super(Settings.web().getScreenshotsOnFailEnabled()); } + @Override + public byte[] takeScreenshot() { + return PlaywrightService.wrappedBrowser().getCurrentPage() + .screenshot(new Page.ScreenshotOptions() + .setType(ScreenshotType.PNG) + .setFullPage(false)); + } + @Override public String takeScreenshot(String name) { var screenshotSaveDir = getOutputFolder(); diff --git a/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/services/BrowserService.java b/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/services/BrowserService.java index 40cae43b..fcd977d1 100644 --- a/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/services/BrowserService.java +++ b/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/services/BrowserService.java @@ -64,6 +64,10 @@ public String getPageSource() { return page().content(); } + public void setSize(int width, int height) { + page().setViewportSize(width, height); + } + /** * Not possible to implement. */ diff --git a/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/services/ComponentCreateService.java b/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/services/ComponentCreateService.java index 95af6be9..246193bb 100644 --- a/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/services/ComponentCreateService.java +++ b/bellatrix.playwright/src/main/java/solutions/bellatrix/playwright/services/ComponentCreateService.java @@ -22,6 +22,7 @@ import solutions.bellatrix.playwright.findstrategies.options.*; import solutions.bellatrix.playwright.components.common.webelement.FrameElement; import solutions.bellatrix.playwright.components.common.webelement.WebElement; +import solutions.bellatrix.plugins.opencv.Base64Encodable; import java.util.ArrayList; import java.util.List; @@ -166,6 +167,14 @@ public List allById(Class TComponent byImage(Class componentClass, Base64Encodable encodedImage) { + return by(componentClass, new ImageBase64FindStrategy(encodedImage)); + } + + public List allByImage(Class componentClass, Base64Encodable encodedImage) { + return allBy(componentClass, new ImageBase64FindStrategy(encodedImage)); + } + public TComponent byClass(Class componentClass, String value) { return by(componentClass, new ClassFindStrategy(value)); } diff --git a/bellatrix.plugins.opencv/pom.xml b/bellatrix.plugins.opencv/pom.xml new file mode 100644 index 00000000..66dafc6c --- /dev/null +++ b/bellatrix.plugins.opencv/pom.xml @@ -0,0 +1,53 @@ + + + 4.0.0 + + solutions.bellatrix + bellatrix + 1.0-SNAPSHOT + + + solutions.bellatrix + bellatrix.plugins.opencv + 1.0 + + + 19 + 19 + UTF-8 + + + + + solutions.bellatrix + bellatrix.core + 1.0 + + + solutions.bellatrix + bellatrix.plugins.screenshots + 1.0 + + + org.openpnp + opencv + 4.3.0-3 + compile + + + jakarta.xml.bind + jakarta.xml.bind-api + 2.3.2 + test + + + jakarta.xml.bind + jakarta.xml.bind-api + 2.3.2 + compile + + + + \ No newline at end of file diff --git a/bellatrix.plugins.opencv/src/main/java/solutions/bellatrix/plugins/opencv/Base64Encodable.java b/bellatrix.plugins.opencv/src/main/java/solutions/bellatrix/plugins/opencv/Base64Encodable.java new file mode 100644 index 00000000..8589f88c --- /dev/null +++ b/bellatrix.plugins.opencv/src/main/java/solutions/bellatrix/plugins/opencv/Base64Encodable.java @@ -0,0 +1,8 @@ +package solutions.bellatrix.plugins.opencv; + +public interface Base64Encodable { + String getBase64Image(); + String getImageName(); + int getXOffset(); + int getYOffset(); +} \ No newline at end of file diff --git a/bellatrix.plugins.opencv/src/main/java/solutions/bellatrix/plugins/opencv/OpenCvService.java b/bellatrix.plugins.opencv/src/main/java/solutions/bellatrix/plugins/opencv/OpenCvService.java new file mode 100644 index 00000000..705ca8e9 --- /dev/null +++ b/bellatrix.plugins.opencv/src/main/java/solutions/bellatrix/plugins/opencv/OpenCvService.java @@ -0,0 +1,146 @@ +package solutions.bellatrix.plugins.opencv; + +import nu.pattern.OpenCV; +import org.opencv.core.Core; +import org.opencv.core.CvType; +import org.opencv.core.Mat; +import org.opencv.core.Point; +import org.opencv.imgcodecs.Imgcodecs; +import org.opencv.imgproc.Imgproc; +import plugins.screenshots.ScreenshotPlugin; +import solutions.bellatrix.core.utilities.SingletonFactory; + +import javax.imageio.ImageIO; +import java.awt.*; +import java.awt.geom.AffineTransform; +import java.awt.image.BufferedImage; +import java.io.ByteArrayInputStream; +import java.io.IOException; +import java.util.Base64; + +public class OpenCvService { + /** + * @return the current scaling factor (e.g., 1.0 for 100%, 1.25 for 125%) + */ + public static double getJavaMonitorScaling() { + GraphicsConfiguration gc = GraphicsEnvironment + .getLocalGraphicsEnvironment() + .getDefaultScreenDevice() + .getDefaultConfiguration(); + + // Get the default transform + AffineTransform transform = gc.getDefaultTransform(); + + // The scaling factor is typically the same for both X and Y + double scaleX = transform.getScaleX(); + double scaleY = transform.getScaleY(); + + // Return X scale (usually same as Y scale) + return scaleX; + } + + /** + * @return the coordinates of the image found on the screen + */ + public static Point getLocation(Base64Encodable encodedImage, boolean shouldGrayScale) { + var screenshotPlugin = SingletonFactory.getInstance(ScreenshotPlugin.class); + + if (screenshotPlugin == null) { + throw new IllegalArgumentException("It seems that the screenshot plugin isn't registered by the 'ScreenshotPlugin.class' key inside SingletonFactory's map or isn't registered at all!\n" + + "Check the BaseTest class of your project where the plugins are registered. Register the specific screenshot plugin implementation as the base ScreenshotPlugin.class.\n" + + "for example: addPluginAs(ScreenshotPlugin.class, WebScreenshotPlugin.class);"); + } + + var screenshot = screenshotPlugin.takeScreenshot(); + + OpenCV.loadLocally(); + + Mat result = loadImages(encodedImage, screenshot, shouldGrayScale); + + return getMatchLocation(encodedImage, result); + } + + private static Point getMatchLocation(Base64Encodable encodedImage, Mat result) { + BufferedImage bufferedImage; + + Core.MinMaxLocResult mmr = Core.minMaxLoc(result); + Point matchLoc = mmr.maxLoc; + + if (encodedImage.getXOffset() == 0 && encodedImage.getYOffset() == 0) { + try { + bufferedImage = getImageWidthHeight(encodedImage); + } catch (IOException e) { + throw new RuntimeException(e); + } + + double[] imageCenterCoordinates = {matchLoc.x / getJavaMonitorScaling() + (double)(bufferedImage.getWidth() / 2), matchLoc.y / getJavaMonitorScaling() + (double)(bufferedImage.getHeight() / 2)}; + matchLoc.set(imageCenterCoordinates); + } + + return matchLoc; + } + + private static BufferedImage getImageWidthHeight(Base64Encodable encodedImage) throws IOException { + String cleanBase64 = removePrefixFromBase64(encodedImage); + + byte[] decodedBytes = Base64.getDecoder().decode(cleanBase64); + ByteArrayInputStream bis = new ByteArrayInputStream(decodedBytes); + BufferedImage bimg = ImageIO.read(bis); + + return bimg; + } + + private static Mat loadImages(Base64Encodable encodedImage, byte[] screenshot, boolean shouldGrayScale) { + Mat template = getMatrixFromBase64(encodedImage); + if (shouldGrayScale) { + changeToGrayscale(template); + } + + Mat source = getMatrixFromBinaryData(screenshot); + if (shouldGrayScale) { + changeToGrayscale(source); + } + + Mat result = createResultMatrix(source, template); + + Imgproc.matchTemplate(source, template, result, Imgproc.TM_CCOEFF_NORMED); + + return result; + } + + private static Mat changeToGrayscale(Mat template) { + Mat templateGrayscale = new Mat(); + Imgproc.cvtColor(template, templateGrayscale, Imgproc.COLOR_BGR2GRAY); + + template = templateGrayscale; + return template; + } + + private static Mat getMatrixFromBase64(Base64Encodable encodedImage) { + String cleanBase64 = removePrefixFromBase64(encodedImage); + + byte[] decodedBytes = Base64.getDecoder().decode(cleanBase64); + return getMatrixFromBinaryData(decodedBytes); + } + + private static String removePrefixFromBase64(Base64Encodable encodedImage) { + String base64Image = encodedImage.getBase64Image(); + return base64Image.replaceFirst("^data:.+?;base64,", ""); + } + + private static Mat getMatrixFromBinaryData(byte[] decodedBytes) { + Mat mat = new Mat(1, decodedBytes.length, CvType.CV_8U); + mat.put(0, 0, decodedBytes); + + return Imgcodecs.imdecode(mat, Imgcodecs.IMREAD_COLOR); + } + + private static Mat createResultMatrix(Mat source, Mat template) { + Mat result = new Mat(); + int result_cols = source.cols() - template.cols() + 1; + int result_rows = source.rows() - template.rows() + 1; + result.create(result_rows, result_cols, CvType.CV_32FC1); + + return result; + } +} \ No newline at end of file diff --git a/bellatrix.plugins.screenshots/src/main/java/plugins/screenshots/ScreenshotPlugin.java b/bellatrix.plugins.screenshots/src/main/java/plugins/screenshots/ScreenshotPlugin.java index 528d24e7..c6a4e880 100644 --- a/bellatrix.plugins.screenshots/src/main/java/plugins/screenshots/ScreenshotPlugin.java +++ b/bellatrix.plugins.screenshots/src/main/java/plugins/screenshots/ScreenshotPlugin.java @@ -29,6 +29,7 @@ public ScreenshotPlugin(boolean isEnabled) { this.isEnabled = isEnabled; } + public abstract byte[] takeScreenshot(); public abstract String takeScreenshot(String fileName); public abstract String takeScreenshot(String screenshotSaveDir, String filename); protected abstract String getOutputFolder(); diff --git a/bellatrix.plugins.visual-regression-tracker/src/main/java/solutions/bellatrix/plugins/vrt/VisualRegressionService.java b/bellatrix.plugins.visual-regression-tracker/src/main/java/solutions/bellatrix/plugins/vrt/VisualRegressionService.java index eeefd6eb..90ef489f 100644 --- a/bellatrix.plugins.visual-regression-tracker/src/main/java/solutions/bellatrix/plugins/vrt/VisualRegressionService.java +++ b/bellatrix.plugins.visual-regression-tracker/src/main/java/solutions/bellatrix/plugins/vrt/VisualRegressionService.java @@ -78,9 +78,12 @@ public static String takeSnapshot(String name) { var screenshotPlugin = SingletonFactory.getInstance(ScreenshotPlugin.class); if (screenshotPlugin == null) { - throw new IllegalArgumentException("It seems that the screenshot plugin isn't registered by the 'ScreenshotPlugin.class' key inside SingletonFactory's map or isn't registered at all!"); + throw new IllegalArgumentException("It seems that the screenshot plugin isn't registered by the 'ScreenshotPlugin.class' key inside SingletonFactory's map or isn't registered at all!\n" + + "Check the BaseTest class of your project where the plugins are registered. Register the specific screenshot plugin implementation as the base ScreenshotPlugin.class.\n" + + "for example: addPluginAs(ScreenshotPlugin.class, WebScreenshotPlugin.class);"); } + return screenshotPlugin.takeScreenshot(name); } diff --git a/bellatrix.web/pom.xml b/bellatrix.web/pom.xml index c6ffc9a6..b6efa38d 100644 --- a/bellatrix.web/pom.xml +++ b/bellatrix.web/pom.xml @@ -39,6 +39,12 @@ bellatrix.plugins.screenshots 1.0 + + solutions.bellatrix + bellatrix.plugins.opencv + 1.0 + compile + ru.yandex.qatools.ashot ashot diff --git a/bellatrix.web/src/main/java/solutions/bellatrix/web/components/ActionImage.java b/bellatrix.web/src/main/java/solutions/bellatrix/web/components/ActionImage.java new file mode 100644 index 00000000..fee784ee --- /dev/null +++ b/bellatrix.web/src/main/java/solutions/bellatrix/web/components/ActionImage.java @@ -0,0 +1,64 @@ +package solutions.bellatrix.web.components; + +import lombok.Getter; +import lombok.Setter; +import org.openqa.selenium.InvalidArgumentException; +import org.openqa.selenium.interactions.Actions; +import solutions.bellatrix.core.plugins.EventListener; +import solutions.bellatrix.plugins.opencv.OpenCvService; +import solutions.bellatrix.web.findstrategies.ImageBase64FindStrategy; +import solutions.bellatrix.web.services.App; + +import java.awt.*; + +@Getter +@Setter +public class ActionImage extends WebComponent { + public final static EventListener CLICKING = new EventListener<>(); + public final static EventListener HOVERING = new EventListener<>(); + + private Actions actions = new Actions(getWrappedDriver()); + + @Override + public Point getLocation() { + ImageBase64FindStrategy findStrategy; + try { + findStrategy = (ImageBase64FindStrategy)getFindStrategy(); + } catch (ClassCastException e) { + throw new InvalidArgumentException("Invalid image base 64 format"); + } + + var encodedImage = findStrategy.getEncodedImage(); + + var location = OpenCvService.getLocation(encodedImage, true); + return new Point((int)location.x + encodedImage.getXOffset(), (int)location.y + encodedImage.getYOffset()); + } + + public void click() { + CLICKING.broadcast(new ComponentActionEventArgs(this, null, "Coordinates: %d, %d".formatted(getLocation().x, getLocation().y))); + actions.moveByOffset(getLocation().x, getLocation().y) + .click() + .perform(); + } + + public void rightClick() { + CLICKING.broadcast(new ComponentActionEventArgs(this, null, "Coordinates: %d, %d".formatted(getLocation().x, getLocation().y))); + actions.moveByOffset(getLocation().x, getLocation().y) + .contextClick() + .perform(); + } + + public void hover() { + HOVERING.broadcast(new ComponentActionEventArgs(this, null, "Coordinates: %d, %d".formatted(getLocation().x, getLocation().y))); + actions.moveToLocation(getLocation().x, getLocation().y) + .perform(); + } + + public void dragAndDrop(ActionImage image) { + actions.moveByOffset(getLocation().x, getLocation().y) + .clickAndHold() + .moveToLocation(image.getLocation().x, image.getLocation().y) + .release() + .perform(); + } +} \ No newline at end of file diff --git a/bellatrix.web/src/main/java/solutions/bellatrix/web/components/WebComponent.java b/bellatrix.web/src/main/java/solutions/bellatrix/web/components/WebComponent.java index 98cc513a..13b967b1 100644 --- a/bellatrix.web/src/main/java/solutions/bellatrix/web/components/WebComponent.java +++ b/bellatrix.web/src/main/java/solutions/bellatrix/web/components/WebComponent.java @@ -28,6 +28,7 @@ import solutions.bellatrix.core.utilities.DebugInformation; import solutions.bellatrix.core.utilities.InstanceFactory; import solutions.bellatrix.core.utilities.Log; +import solutions.bellatrix.plugins.opencv.Base64Encodable; import solutions.bellatrix.web.components.contracts.Component; import solutions.bellatrix.web.components.contracts.ComponentStyle; import solutions.bellatrix.web.components.contracts.ComponentVisible; @@ -343,6 +344,10 @@ public TComponent createById(Class return create(componentClass, new IdFindStrategy(id)); } + public TComponent createByImage(Class componentClass, Base64Encodable encodedImage) { + return create(componentClass, new ImageBase64FindStrategy(encodedImage)); + } + public TComponent createByAttributeContaining(Class componentClass, String attributeName, String value) { return create(componentClass, new AttributeContainingWithFindStrategy(attributeName, value)); } @@ -403,6 +408,10 @@ public List createAllById(Class List createAllByImage(Class componentClass, Base64Encodable encodedImage) { + return createAll(componentClass, new ImageBase64FindStrategy(encodedImage)); + } + public List createAllByAttributeContaining(Class componentClass, String attributeName, String value) { return createAll(componentClass, new AttributeContainingWithFindStrategy(attributeName, value)); } diff --git a/bellatrix.web/src/main/java/solutions/bellatrix/web/components/listeners/BddConsoleLogging.java b/bellatrix.web/src/main/java/solutions/bellatrix/web/components/listeners/BddConsoleLogging.java index 0543f7d0..d3cc33e1 100644 --- a/bellatrix.web/src/main/java/solutions/bellatrix/web/components/listeners/BddConsoleLogging.java +++ b/bellatrix.web/src/main/java/solutions/bellatrix/web/components/listeners/BddConsoleLogging.java @@ -52,6 +52,8 @@ public void addListener() { // WebComponent.SCROLLING_TO_VISIBLE.addListener((x) -> Log.info("scrolling to %s", x.getComponent().getComponentName())); WebComponent.SETTING_ATTRIBUTE.addListener((x) -> Log.info("setting %s to '%s' in %s", x.getActionValue(), x.getMessage(), x.getComponent().getComponentName())); ComponentValidator.VALIDATING_ATTRIBUTE.addListener((x) -> Log.info(x.getMessage())); + ActionImage.CLICKING.addListener((x) -> Log.info("clicking %s", x.getMessage())); + ActionImage.HOVERED.addListener((x) -> Log.info("hovering %s", x.getMessage())); isBddLoggingTurnedOn = true; } } diff --git a/bellatrix.web/src/main/java/solutions/bellatrix/web/components/listeners/BddToastNotificationsLogging.java b/bellatrix.web/src/main/java/solutions/bellatrix/web/components/listeners/BddToastNotificationsLogging.java index 474b55a0..2701ce2a 100644 --- a/bellatrix.web/src/main/java/solutions/bellatrix/web/components/listeners/BddToastNotificationsLogging.java +++ b/bellatrix.web/src/main/java/solutions/bellatrix/web/components/listeners/BddToastNotificationsLogging.java @@ -25,7 +25,7 @@ public class BddToastNotificationsLogging extends Listener { @Override public void addListener() { - isBddLoggingTurnedOn = ConfigurationService.get(WebSettings.class).getToastNotificationBddLogging(); + isBddLoggingTurnedOn = ConfigurationService.get(WebSettings.class).getToastNotificationBddLogging() == null ? isBddLoggingTurnedOn : ConfigurationService.get(WebSettings.class).getToastNotificationBddLogging(); if (isBddLoggingTurnedOn) { Anchor.CLICKING.addListener((x) -> new BrowserService().injectInfoNotificationToast("clicking %s", x.getComponent().getComponentName())); Button.CLICKING.addListener((x) -> new BrowserService().injectInfoNotificationToast("clicking %s", x.getComponent().getComponentName())); @@ -55,6 +55,8 @@ public void addListener() { // WebComponent.SCROLLING_TO_VISIBLE.addListener((x) -> new BrowserService().injectInfoNotificationToast("scrolling to %s", x.getComponent().getComponentName())); WebComponent.SETTING_ATTRIBUTE.addListener((x) -> new BrowserService().injectInfoNotificationToast("setting %s to '%s' in %s", x.getActionValue(), x.getMessage(), x.getComponent().getComponentName())); ComponentValidator.VALIDATING_ATTRIBUTE.addListener((x) -> new BrowserService().injectInfoNotificationToast(x.getMessage())); + ActionImage.CLICKING.addListener((x) -> new BrowserService().injectInfoNotificationToast("clicking %s", x.getMessage())); + ActionImage.HOVERED.addListener((x) -> new BrowserService().injectInfoNotificationToast("hovering %s", x.getMessage())); isBddLoggingTurnedOn = true; } } diff --git a/bellatrix.web/src/main/java/solutions/bellatrix/web/findstrategies/ImageBase64FindStrategy.java b/bellatrix.web/src/main/java/solutions/bellatrix/web/findstrategies/ImageBase64FindStrategy.java new file mode 100644 index 00000000..6efa1658 --- /dev/null +++ b/bellatrix.web/src/main/java/solutions/bellatrix/web/findstrategies/ImageBase64FindStrategy.java @@ -0,0 +1,58 @@ +package solutions.bellatrix.web.findstrategies; + +import lombok.Getter; +import org.openqa.selenium.By; +import org.openqa.selenium.SearchContext; +import org.openqa.selenium.WebElement; +import solutions.bellatrix.core.utilities.Log; +import solutions.bellatrix.core.utilities.SingletonFactory; +import solutions.bellatrix.plugins.opencv.Base64Encodable; +import solutions.bellatrix.plugins.opencv.OpenCvService; +import solutions.bellatrix.web.services.App; +import solutions.bellatrix.web.services.JavaScriptService; + +import java.util.List; +import java.util.Objects; + +@Getter +public class ImageBase64FindStrategy extends FindStrategy { + private final Base64Encodable encodedImage; + + public ImageBase64FindStrategy(Base64Encodable encodedImage) { + super(encodedImage.getBase64Image()); + this.encodedImage = encodedImage; + } + + @Override + public By convert() { + return new ByImageBase64(encodedImage); + } + + @Override + public String toString() { + return String.format("image base 64: %s", encodedImage.getImageName()); + } + + public static class ByImageBase64 extends By { + private final Base64Encodable base64EncodedImage; + + public ByImageBase64(Base64Encodable base64EncodedImage) { + this.base64EncodedImage = base64EncodedImage; + } + + public static By byImageBase64(Base64Encodable encodedImage) { + return new ByImageBase64(encodedImage); + } + + @Override + public List findElements(SearchContext context) { + var location = OpenCvService.getLocation(base64EncodedImage, false); + Log.info("Coordinates located: %s", location.toString()); + return SingletonFactory.getInstance(JavaScriptService.class).>genericExecute("return document.elementsFromPoint(%s, %s);".formatted(location.x, location.y)); + } + + public String toString() { + return "By.imageBase64: " + this.base64EncodedImage; + } + } +} \ No newline at end of file diff --git a/bellatrix.web/src/main/java/solutions/bellatrix/web/infrastructure/WebScreenshotPlugin.java b/bellatrix.web/src/main/java/solutions/bellatrix/web/infrastructure/WebScreenshotPlugin.java index bc58145e..a3b0283b 100644 --- a/bellatrix.web/src/main/java/solutions/bellatrix/web/infrastructure/WebScreenshotPlugin.java +++ b/bellatrix.web/src/main/java/solutions/bellatrix/web/infrastructure/WebScreenshotPlugin.java @@ -13,6 +13,9 @@ package solutions.bellatrix.web.infrastructure; +import nu.pattern.OpenCV; +import org.openqa.selenium.OutputType; +import org.openqa.selenium.TakesScreenshot; import plugins.screenshots.ScreenshotPlugin; import plugins.screenshots.ScreenshotPluginEventArgs; import ru.yandex.qatools.ashot.AShot; @@ -20,7 +23,10 @@ import solutions.bellatrix.core.configuration.ConfigurationService; import solutions.bellatrix.core.utilities.Log; import solutions.bellatrix.core.utilities.PathNormalizer; +import solutions.bellatrix.core.utilities.SingletonFactory; import solutions.bellatrix.web.configuration.WebSettings; +import solutions.bellatrix.web.services.BrowserService; +import solutions.bellatrix.web.services.WebService; import javax.imageio.ImageIO; import java.awt.image.BufferedImage; @@ -36,6 +42,11 @@ public WebScreenshotPlugin() { super(ConfigurationService.get(WebSettings.class).getScreenshotsOnFailEnabled()); } + @Override + public byte[] takeScreenshot() { + return ((TakesScreenshot)DriverService.getWrappedDriver()).getScreenshotAs(OutputType.BYTES); + } + @Override public String takeScreenshot(String name) { var screenshotSaveDir = getOutputFolder(); diff --git a/bellatrix.web/src/main/java/solutions/bellatrix/web/services/App.java b/bellatrix.web/src/main/java/solutions/bellatrix/web/services/App.java index e66a4920..82f8dd0c 100644 --- a/bellatrix.web/src/main/java/solutions/bellatrix/web/services/App.java +++ b/bellatrix.web/src/main/java/solutions/bellatrix/web/services/App.java @@ -22,31 +22,31 @@ public class App implements BellatrixApp { private boolean disposed = false; public NavigationService navigate() { - return new NavigationService(); + return SingletonFactory.getInstance(NavigationService.class); } public BrowserService browser() { - return new BrowserService(); + return SingletonFactory.getInstance(BrowserService.class); } public CookiesService cookies() { - return new CookiesService(); + return SingletonFactory.getInstance(CookiesService.class); } public DialogService dialogs() { - return new DialogService(); + return SingletonFactory.getInstance(DialogService.class); } public JavaScriptService script() { - return new JavaScriptService(); + return SingletonFactory.getInstance(JavaScriptService.class); } public ComponentCreateService create() { - return new ComponentCreateService(); + return SingletonFactory.getInstance(ComponentCreateService.class); } public ComponentWaitService waitFor() { - return new ComponentWaitService(); + return SingletonFactory.getInstance(ComponentWaitService.class); } @Override diff --git a/bellatrix.web/src/main/java/solutions/bellatrix/web/services/BrowserService.java b/bellatrix.web/src/main/java/solutions/bellatrix/web/services/BrowserService.java index 12e1bd7f..7f4c80a5 100644 --- a/bellatrix.web/src/main/java/solutions/bellatrix/web/services/BrowserService.java +++ b/bellatrix.web/src/main/java/solutions/bellatrix/web/services/BrowserService.java @@ -43,11 +43,12 @@ import java.util.logging.Level; public class BrowserService extends WebService { - private final JavascriptExecutor javascriptExecutor; - public BrowserService() { super(); - javascriptExecutor = (JavascriptExecutor)getWrappedDriver(); + } + + public JavascriptExecutor getJavascriptExecutor() { + return (JavascriptExecutor)getWrappedDriver(); } public String getPageSource() { @@ -66,6 +67,14 @@ public void back() { getWrappedDriver().navigate().back(); } + public void setSize(int width, int height) { + setSize(new Dimension(width, height)); + } + + public void setSize(Dimension dimension) { + getWrappedDriver().manage().window().setSize(dimension); + } + public void maximize() { getWrappedDriver().manage().window().maximize(); } @@ -136,36 +145,36 @@ public void switchToFrame(Frame frame) { } public void clearSessionStorage() { - ((JavascriptExecutor)getWrappedDriver()).executeScript("sessionStorage.clear()"); + getJavascriptExecutor().executeScript("sessionStorage.clear()"); } public void removeItemFromLocalStorage(String item) { - ((JavascriptExecutor)getWrappedDriver()).executeScript(String.format("window.localStorage.removeItem('%s');", item)); + getJavascriptExecutor().executeScript(String.format("window.localStorage.removeItem('%s');", item)); } public void scrollToBottom() { - ((JavascriptExecutor)getWrappedDriver()).executeScript("window.scrollTo(0, document.body.scrollHeight)"); + getJavascriptExecutor().executeScript("window.scrollTo(0, document.body.scrollHeight)"); } public void scrollToTop() { - ((JavascriptExecutor)getWrappedDriver()).executeScript("window.scrollTo(0, 0)"); + getJavascriptExecutor().executeScript("window.scrollTo(0, 0)"); } public boolean isItemPresentInLocalStorage(String item) { - return !(((JavascriptExecutor)getWrappedDriver()).executeScript(String.format("return window.localStorage.getItem('%s');", item)) == null); + return !(getJavascriptExecutor().executeScript(String.format("return window.localStorage.getItem('%s');", item)) == null); } public String getItemFromLocalStorage(String key) { - return (String)((JavascriptExecutor)getWrappedDriver()).executeScript(String.format("return window.localStorage.getItem('%s');", key)); + return (String)getJavascriptExecutor().executeScript(String.format("return window.localStorage.getItem('%s');", key)); } public void setItemInLocalStorage(String item, String value) { - ((JavascriptExecutor)getWrappedDriver()).executeScript(String.format("window.localStorage.setItem('%s','%s');", item, value)); + getJavascriptExecutor().executeScript(String.format("window.localStorage.setItem('%s','%s');", item, value)); } public void clearLocalStorage() { - ((JavascriptExecutor)getWrappedDriver()).executeScript("localStorage.clear()"); + getJavascriptExecutor().executeScript("localStorage.clear()"); } public List getBrowserLogs() { @@ -210,17 +219,16 @@ public List getSevereLogEntries() { } public List getRequestEntries(String partialUrl) { - return (List)((JavascriptExecutor)getWrappedDriver()).executeScript(String.format("return window.performance.getEntriesByType('resource').filter(x => x.name.indexOf('%s') >= 0).map(y => y.name);", partialUrl)); + return (List)getJavascriptExecutor().executeScript(String.format("return window.performance.getEntriesByType('resource').filter(x => x.name.indexOf('%s') >= 0).map(y => y.name);", partialUrl)); } public void waitForAjax() { long ajaxTimeout = ConfigurationService.get(WebSettings.class).getTimeoutSettings().getWaitForAjaxTimeout(); long sleepInterval = ConfigurationService.get(WebSettings.class).getTimeoutSettings().getSleepInterval(); - var javascriptExecutor = (JavascriptExecutor)getWrappedDriver(); AtomicInteger ajaxConnections = new AtomicInteger(); try { Wait.retry(() -> { - var numberOfAjaxConnections = javascriptExecutor.executeScript("return !isNaN(window.$openHTTPs) ? window.$openHTTPs : null"); + var numberOfAjaxConnections = getJavascriptExecutor().executeScript("return !isNaN(window.$openHTTPs) ? window.$openHTTPs : null"); if (Objects.nonNull(numberOfAjaxConnections)) { ajaxConnections.set(Integer.parseInt(numberOfAjaxConnections.toString())); if (ajaxConnections.get() > 0) { @@ -246,7 +254,7 @@ public void waitForAjax() { } private void monkeyPatchXMLHttpRequest() { - var numberOfAjaxConnections = javascriptExecutor.executeScript(("return !isNaN(window.$openHTTPs) ? window.$openHTTPs : null")); + var numberOfAjaxConnections = getJavascriptExecutor().executeScript(("return !isNaN(window.$openHTTPs) ? window.$openHTTPs : null")); if (Objects.isNull(numberOfAjaxConnections)) { var script = "(function() {" + @@ -263,7 +271,7 @@ private void monkeyPatchXMLHttpRequest() { "}" + "})();"; - javascriptExecutor.executeScript(script); + getJavascriptExecutor().executeScript(script); } } @@ -273,7 +281,7 @@ public void waitForAjaxRequest(String requestPartialUrl, int additionalTimeoutIn var webDriverWait = new WebDriverWait(getWrappedDriver(), Duration.ofSeconds(ajaxTimeout + additionalTimeoutInSeconds), Duration.ofSeconds(sleepInterval)); webDriverWait.until(d -> { String script = String.format("return performance.getEntriesByType('resource').filter(item => item.initiatorType == 'xmlhttprequest' && item.name.toLowerCase().includes('%s'))[0] !== undefined;", requestPartialUrl); - boolean result = (boolean)javascriptExecutor.executeScript(script); + boolean result = (boolean)getJavascriptExecutor().executeScript(script); return result; }); } @@ -375,8 +383,8 @@ public void waitForReactPageLoadsCompletely() { long waitUntilReadyTimeout = ConfigurationService.get(WebSettings.class).getTimeoutSettings().getWaitUntilReadyTimeout(); long sleepInterval = ConfigurationService.get(WebSettings.class).getTimeoutSettings().getSleepInterval(); var webDriverWait = new WebDriverWait(getWrappedDriver(), Duration.ofSeconds(waitUntilReadyTimeout), Duration.ofSeconds(sleepInterval)); - webDriverWait.until(d -> javascriptExecutor.executeScript("return document.querySelector('[data-reactroot]') !== null")); - webDriverWait.until(d -> javascriptExecutor.executeScript("return window.performance.timing.loadEventEnd > 0")); + webDriverWait.until(d -> getJavascriptExecutor().executeScript("return document.querySelector('[data-reactroot]') !== null")); + webDriverWait.until(d -> getJavascriptExecutor().executeScript("return window.performance.timing.loadEventEnd > 0")); } // TODO Refactor the other methods to reuse this one @@ -406,7 +414,7 @@ public void waitForJavaScriptAnimations() { long waitForJavaScriptAnimationsTimeout = ConfigurationService.get(WebSettings.class).getTimeoutSettings().getWaitForJavaScriptAnimationsTimeout(); long sleepInterval = ConfigurationService.get(WebSettings.class).getTimeoutSettings().getSleepInterval(); var webDriverWait = new WebDriverWait(getWrappedDriver(), Duration.ofSeconds(waitForJavaScriptAnimationsTimeout), Duration.ofSeconds(sleepInterval)); - webDriverWait.until(d -> (boolean)javascriptExecutor.executeScript("return jQuery && jQuery(':animated').length === 0")); + webDriverWait.until(d -> (boolean)getJavascriptExecutor().executeScript("return jQuery && jQuery(':animated').length === 0")); } public void waitForPartialUrl(String partialUrl) { @@ -424,11 +432,10 @@ public void waitNumberOfWindowsToBe(int numberOfWindows) { } public void waitForRequest(String partialUrl) { - var javascriptExecutor = (JavascriptExecutor)getWrappedDriver(); String script = String.format("return performance.getEntriesByType('resource').filter(item => item.name.toLowerCase().includes('%s'))[0] !== undefined;", partialUrl); try { - waitUntil(e -> (boolean)javascriptExecutor.executeScript(script)); + waitUntil(e -> (boolean)getJavascriptExecutor().executeScript(script)); } catch (TimeoutException exception) { throw new TimeoutException(String.format("The expected request with URL '%s' is not loaded!", partialUrl)); } @@ -439,9 +446,8 @@ public void tryWaitForResponse(String partialUrl, int additionalTimeoutInSeconds if(ProxyServer.get() != null) { ProxyServer.waitForResponse(getWrappedDriver(), partialUrl, HttpMethod.GET, 0); } else { - var javascriptExecutor = (JavascriptExecutor)getWrappedDriver(); String script = "return performance.getEntriesByType('resource').filter(item => item.name.toLowerCase().includes('" + partialUrl.toLowerCase() + "'))[0] !== undefined;"; - waitUntil(e -> (boolean)javascriptExecutor.executeScript(script)); + waitUntil(e -> (boolean)getJavascriptExecutor().executeScript(script)); } } catch (Exception exception) { Log.error("The expected request with URL '%s' is not loaded!", partialUrl); @@ -457,15 +463,15 @@ public void waitForAngular() { long sleepInterval = ConfigurationService.get(WebSettings.class).getTimeoutSettings().getSleepInterval(); var webDriverWait = new WebDriverWait(getWrappedDriver(), Duration.ofSeconds(angularTimeout), Duration.ofSeconds(sleepInterval)); - String isAngular5 = (String)javascriptExecutor.executeScript("return getAllAngularRootElements()[0].attributes['ng-version']"); + String isAngular5 = (String)getJavascriptExecutor().executeScript("return getAllAngularRootElements()[0].attributes['ng-version']"); if (StringUtils.isBlank(isAngular5)) { - webDriverWait.until(d -> (boolean)javascriptExecutor.executeScript("return window.getAllAngularTestabilities().findIndex(x=>!x.isStable()) === -1")); + webDriverWait.until(d -> (boolean)getJavascriptExecutor().executeScript("return window.getAllAngularTestabilities().findIndex(x=>!x.isStable()) === -1")); } else { - boolean isAngularDefined = (boolean)javascriptExecutor.executeScript("return window.angular === undefined"); + boolean isAngularDefined = (boolean)getJavascriptExecutor().executeScript("return window.angular === undefined"); if (!((boolean)isAngularDefined)) { - boolean isAngularInjectorUnDefined = (boolean)javascriptExecutor.executeScript("return angular.element(document).injector() === undefined"); + boolean isAngularInjectorUnDefined = (boolean)getJavascriptExecutor().executeScript("return angular.element(document).injector() === undefined"); if (!isAngularInjectorUnDefined) { - webDriverWait.until(d -> (boolean)javascriptExecutor.executeScript("return angular.element(document).injector().get('$http').pendingRequests.length === 0")); + webDriverWait.until(d -> (boolean)getJavascriptExecutor().executeScript("return angular.element(document).injector().get('$http').pendingRequests.length === 0")); } } } diff --git a/bellatrix.web/src/main/java/solutions/bellatrix/web/services/ComponentCreateService.java b/bellatrix.web/src/main/java/solutions/bellatrix/web/services/ComponentCreateService.java index 214e1ab4..f94dd512 100644 --- a/bellatrix.web/src/main/java/solutions/bellatrix/web/services/ComponentCreateService.java +++ b/bellatrix.web/src/main/java/solutions/bellatrix/web/services/ComponentCreateService.java @@ -14,6 +14,8 @@ package solutions.bellatrix.web.services; import solutions.bellatrix.core.utilities.InstanceFactory; +import solutions.bellatrix.plugins.opencv.Base64Encodable; +import solutions.bellatrix.web.components.ActionImage; import solutions.bellatrix.web.components.WebComponent; import solutions.bellatrix.web.findstrategies.*; import solutions.bellatrix.web.infrastructure.DriverService; @@ -36,6 +38,10 @@ public TComponent byId(Class compo return by(componentClass, new IdFindStrategy(id)); } + public TComponent byImage(Class componentClass, Base64Encodable encodedImage) { + return by(componentClass, new ImageBase64FindStrategy(encodedImage)); + } + public TComponent byAttributeContaining(Class componentClass, String attributeName, String value) { return by(componentClass, new AttributeContainingWithFindStrategy(attributeName, value)); } @@ -96,6 +102,10 @@ public List allById(Class List allByImage(Class componentClass, Base64Encodable encodedImage) { + return allBy(componentClass, new ImageBase64FindStrategy(encodedImage)); + } + public List allByAttributeContaining(Class componentClass, String attributeName, String value) { return allBy(componentClass, new AttributeContainingWithFindStrategy(attributeName, value)); } diff --git a/bellatrix.web/src/main/java/solutions/bellatrix/web/services/JavaScriptService.java b/bellatrix.web/src/main/java/solutions/bellatrix/web/services/JavaScriptService.java index f30b785f..b0ba2e29 100644 --- a/bellatrix.web/src/main/java/solutions/bellatrix/web/services/JavaScriptService.java +++ b/bellatrix.web/src/main/java/solutions/bellatrix/web/services/JavaScriptService.java @@ -18,16 +18,17 @@ import solutions.bellatrix.web.components.WebComponent; public class JavaScriptService extends WebService { - private final JavascriptExecutor javascriptExecutor; - public JavaScriptService() { super(); - javascriptExecutor = (JavascriptExecutor)getWrappedDriver(); + } + + public JavascriptExecutor getJavascriptExecutor() { + return (JavascriptExecutor)getWrappedDriver(); } public T genericExecute(String script, Object... args) { try { - T result = (T)javascriptExecutor.executeScript(script, args); + T result = (T)getJavascriptExecutor().executeScript(script, args); return result; } catch (Exception ex) { DebugInformation.printStackTrace(ex); @@ -37,7 +38,7 @@ public T genericExecute(String script, Object... args) { public Object execute(String script) { try { - var result = javascriptExecutor.executeScript(script); + var result = getJavascriptExecutor().executeScript(script); return result; } catch (Exception ex) { DebugInformation.printStackTrace(ex); @@ -54,7 +55,7 @@ public String execute(String frameName, String script) { public String execute(String script, Object... args) { try { - var result = (String)javascriptExecutor.executeScript(script, args); + var result = (String)getJavascriptExecutor().executeScript(script, args); return result; } catch (Exception ex) { DebugInformation.printStackTrace(ex); @@ -69,7 +70,7 @@ public String execute(String script, TComponen public String execute(String script, WebElement nativeElement) { try { - var result = (String)javascriptExecutor.executeScript(script, nativeElement); + var result = (String)getJavascriptExecutor().executeScript(script, nativeElement); return result; } catch (NoSuchSessionException | NoSuchWindowException ex) { throw ex; diff --git a/framework-tests/bellatrix.android.tests/src/main/resources/testFrameworkSettings.dev.json b/framework-tests/bellatrix.android.tests/src/main/resources/testFrameworkSettings.dev.json index b0e1c1f8..0487588e 100644 --- a/framework-tests/bellatrix.android.tests/src/main/resources/testFrameworkSettings.dev.json +++ b/framework-tests/bellatrix.android.tests/src/main/resources/testFrameworkSettings.dev.json @@ -19,6 +19,7 @@ "screenshotsSaveLocation": "${user.home}/BELLATRIX/videos", "videosOnFailEnabled": "false", "videosSaveLocation": "u${user.home}/BELLATRIX/videos", + "allowImageFindStrategies": "true", "timeoutSettings": { "implicitWaitTimeout": "5", "elementWaitTimeout": "30", diff --git a/framework-tests/bellatrix.android.tests/src/main/resources/testFrameworkSettings.qa.json b/framework-tests/bellatrix.android.tests/src/main/resources/testFrameworkSettings.qa.json index b0e1c1f8..0487588e 100644 --- a/framework-tests/bellatrix.android.tests/src/main/resources/testFrameworkSettings.qa.json +++ b/framework-tests/bellatrix.android.tests/src/main/resources/testFrameworkSettings.qa.json @@ -19,6 +19,7 @@ "screenshotsSaveLocation": "${user.home}/BELLATRIX/videos", "videosOnFailEnabled": "false", "videosSaveLocation": "u${user.home}/BELLATRIX/videos", + "allowImageFindStrategies": "true", "timeoutSettings": { "implicitWaitTimeout": "5", "elementWaitTimeout": "30", diff --git a/framework-tests/bellatrix.desktop.tests/src/main/resources/testFrameworkSettings.dev.json b/framework-tests/bellatrix.desktop.tests/src/main/resources/testFrameworkSettings.dev.json index a35f42aa..6f7aedd3 100644 --- a/framework-tests/bellatrix.desktop.tests/src/main/resources/testFrameworkSettings.dev.json +++ b/framework-tests/bellatrix.desktop.tests/src/main/resources/testFrameworkSettings.dev.json @@ -25,7 +25,8 @@ "elementToNotExistTimeout": "30", "elementToBeClickableTimeout": "30", "elementNotToBeVisibleTimeout": "30", - "elementToHaveContentTimeout": "15" + "elementToHaveContentTimeout": "15", + "allowImageFindStrategies": "true" }, "gridSettings": [ { diff --git a/framework-tests/bellatrix.desktop.tests/src/main/resources/testFrameworkSettings.qa.json b/framework-tests/bellatrix.desktop.tests/src/main/resources/testFrameworkSettings.qa.json index a35f42aa..6f7aedd3 100644 --- a/framework-tests/bellatrix.desktop.tests/src/main/resources/testFrameworkSettings.qa.json +++ b/framework-tests/bellatrix.desktop.tests/src/main/resources/testFrameworkSettings.qa.json @@ -25,7 +25,8 @@ "elementToNotExistTimeout": "30", "elementToBeClickableTimeout": "30", "elementNotToBeVisibleTimeout": "30", - "elementToHaveContentTimeout": "15" + "elementToHaveContentTimeout": "15", + "allowImageFindStrategies": "true" }, "gridSettings": [ { diff --git a/framework-tests/bellatrix.ios.tests/src/main/resources/testFrameworkSettings.dev.json b/framework-tests/bellatrix.ios.tests/src/main/resources/testFrameworkSettings.dev.json index b87c0157..db16647b 100644 --- a/framework-tests/bellatrix.ios.tests/src/main/resources/testFrameworkSettings.dev.json +++ b/framework-tests/bellatrix.ios.tests/src/main/resources/testFrameworkSettings.dev.json @@ -28,7 +28,8 @@ "elementToNotExistTimeout": "30", "elementToBeClickableTimeout": "30", "elementNotToBeVisibleTimeout": "30", - "elementToHaveContentTimeout": "15" + "elementToHaveContentTimeout": "15", + "allowImageFindStrategies": "true" }, "gridSettings": [ { diff --git a/framework-tests/bellatrix.ios.tests/src/main/resources/testFrameworkSettings.qa.json b/framework-tests/bellatrix.ios.tests/src/main/resources/testFrameworkSettings.qa.json index b87c0157..db16647b 100644 --- a/framework-tests/bellatrix.ios.tests/src/main/resources/testFrameworkSettings.qa.json +++ b/framework-tests/bellatrix.ios.tests/src/main/resources/testFrameworkSettings.qa.json @@ -28,7 +28,8 @@ "elementToNotExistTimeout": "30", "elementToBeClickableTimeout": "30", "elementNotToBeVisibleTimeout": "30", - "elementToHaveContentTimeout": "15" + "elementToHaveContentTimeout": "15", + "allowImageFindStrategies": "true" }, "gridSettings": [ { diff --git a/framework-tests/bellatrix.playwright.tests/src/test/java/opencv/OpenCvTests.java b/framework-tests/bellatrix.playwright.tests/src/test/java/opencv/OpenCvTests.java new file mode 100644 index 00000000..00277b65 --- /dev/null +++ b/framework-tests/bellatrix.playwright.tests/src/test/java/opencv/OpenCvTests.java @@ -0,0 +1,89 @@ +package opencv; + +import com.microsoft.playwright.Keyboard; +import com.microsoft.playwright.Locator; +import com.microsoft.playwright.options.ViewportSize; +import opencv.data.EncodedImageDemo; +import opencv.data.EncodedImageNavigationDemo; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.params.ParameterizedTest; +import org.junit.jupiter.params.provider.EnumSource; +import solutions.bellatrix.core.utilities.DebugInformation; +import solutions.bellatrix.playwright.components.*; +import solutions.bellatrix.playwright.infrastructure.BrowserTypes; +import solutions.bellatrix.playwright.infrastructure.ExecutionBrowser; +import solutions.bellatrix.playwright.infrastructure.Lifecycle; +import solutions.bellatrix.playwright.infrastructure.junit.WebTest; + +@ExecutionBrowser(browser = BrowserTypes.CHROME, lifecycle = Lifecycle.REUSE_IF_STARTED) +public class OpenCvTests extends WebTest { + @Override + public void beforeEach() throws Exception { + super.beforeEach(); + app().browser().setSize(1920, 1080); + } + + @Test + public void actionPerformed_when_convertBase64ToImage_and_clickImage() { + app().navigate().to("http://demos.bellatrix.solutions/"); + + var falcon9Image = app().create().byImage(Anchor.class, EncodedImageDemo.FALCON_9); + + falcon9Image.click(); + + app().browser().assertUrlPath("/product/falcon-9/"); + } + + @Test + public void actionPerformed_when_convertBase64ToImage_and_dragAndDropImage() { + app().navigate().to("https://demos.bellatrix.solutions/2018/04/06/proton-rocket-family/"); + + var emailNotes = app().create().byId(Span.class, "email-notes"); + var commentTextArea = app().create().byId(TextArea.class, "comment"); + var falcon9BackButtonImage = app().create().byImage(ActionImage.class, EncodedImageDemo.FALCON_9_BACK_BUTTON); + var commentTextAreaImage = app().create().byImage(ActionImage.class, EncodedImageDemo.COMMENT_TEXT_AREA); + + emailNotes.scrollToVisible(); + falcon9BackButtonImage.dragAndDrop(commentTextAreaImage); + + commentTextArea.validateTextIs("https://demos.bellatrix.solutions/2018/04/06/hello-world/"); + } + + @Test + public void actionPerformed_when_convertBase64ToImage_and_clickButton() { + app().navigate().to("http://demos.bellatrix.solutions/"); + + Button falcon9Image = app().create().byImage(Button.class, EncodedImageDemo.FALCON_9); + + falcon9Image.click(); + + app().browser().assertUrlPath("/product/falcon-9/"); + } + + @Test + public void actionPerformed_when_locateComplexComponents() { + app().navigate().to("http://demos.bellatrix.solutions/"); + + SearchField websiteSearch = app().create().byImage(SearchField.class, EncodedImageDemo.SEARCH_INPUT); + + DebugInformation.debugInfo(websiteSearch.getPlaceholder()); + + websiteSearch.setSearch("Falcon"); + websiteSearch.getWrappedElement().press("Enter", new Locator.PressOptions().setDelay(1500)); + + app().browser().assertUrlPath("?s=Falcon&post_type=product"); + app().create().byImage(ActionImage.class, EncodedImageDemo.FALCON_RESULTS).validateIsVisible(); + } + + @ParameterizedTest + @EnumSource(EncodedImageNavigationDemo.class) + public void actionPerformed_when_locateButtonComponents(EncodedImageNavigationDemo navigationItem) { + app().navigate().to("http://demos.bellatrix.solutions/"); + + Button navgationButton = app().create().byImage(Button.class, navigationItem); + + navgationButton.click(); + + app().browser().validateLandedOnPage(navigationItem.getExpectedUrl()); + } +} \ No newline at end of file diff --git a/framework-tests/bellatrix.playwright.tests/src/test/java/opencv/data/EncodedImageDemo.java b/framework-tests/bellatrix.playwright.tests/src/test/java/opencv/data/EncodedImageDemo.java new file mode 100644 index 00000000..a891aae8 --- /dev/null +++ b/framework-tests/bellatrix.playwright.tests/src/test/java/opencv/data/EncodedImageDemo.java @@ -0,0 +1,63 @@ +package opencv.data; + +import solutions.bellatrix.plugins.opencv.Base64Encodable; + +public enum EncodedImageDemo implements Base64Encodable { + /** + * + */ + FALCON_9("falcon9", ImageInBase64.falcon9, 50, 25), + /** + * + */ + SEARCH_INPUT("searchInput", ImageInBase64.searchInput), + /** + * + */ + FALCON_RESULTS("falconResults", ImageInBase64.falconResults), + /** + * + */ + FALCON_9_BACK_BUTTON("falcon9BackButton", ImageInBase64.falcon9BackButton), + /** + * + */ + COMMENT_TEXT_AREA("commentTextArea", ImageInBase64.commentTextArea); + + private final String imageName; + private final String encodedImage; + private int xOffset = 0; + private int yOffset = 0; + + EncodedImageDemo(String imageName, String encodedImage, int xOffset, int yOffset) { + this.imageName = imageName; + this.encodedImage = encodedImage; + this.xOffset = xOffset; + this.yOffset = yOffset; + } + + EncodedImageDemo(String imageName, String encodedImage) { + this.imageName = imageName; + this.encodedImage = encodedImage; + } + + @Override + public String getBase64Image() { + return encodedImage; + } + + @Override + public String getImageName() { + return imageName; + } + + @Override + public int getXOffset() { + return xOffset; + } + + @Override + public int getYOffset() { + return yOffset; + } +} diff --git a/framework-tests/bellatrix.playwright.tests/src/test/java/opencv/data/EncodedImageNavigationDemo.java b/framework-tests/bellatrix.playwright.tests/src/test/java/opencv/data/EncodedImageNavigationDemo.java new file mode 100644 index 00000000..895cb378 --- /dev/null +++ b/framework-tests/bellatrix.playwright.tests/src/test/java/opencv/data/EncodedImageNavigationDemo.java @@ -0,0 +1,61 @@ +package opencv.data; + +import lombok.Getter; +import solutions.bellatrix.plugins.opencv.Base64Encodable; + +public enum EncodedImageNavigationDemo implements Base64Encodable { + /** + * + */ + HOME_LABEL("homeLabel", NavigationImageInBase64.homeLabel, "https://demos.bellatrix.solutions/"), + /** + * + */ + BLOG_LABEL("blogLabel", NavigationImageInBase64.blogLabel, "/blog/"), + /** + * + */ + CHECKOUT_LABEL("checkoutLabel", NavigationImageInBase64.checkoutLabel, "/cart/"), + /** + * + */ + CONTACT_FORM_LABEL("contactFormLabel", NavigationImageInBase64.contactFormLabel, "/contact-form/"), + /** + * + */ + MY_ACCOUNT_LABEL("myAccountLabel", NavigationImageInBase64.myAccountLabel, "/my-account/"), + /** + * + */ + PROMOTIONS_LABEL("promotionsLabel", NavigationImageInBase64.promotionsLabel, "/welcome/"); + + private final String imageName; + private final String encodedImage; + @Getter private final String expectedUrl; + + EncodedImageNavigationDemo(String imageName, String encodedImage, String expectedUrl) { + this.imageName = imageName; + this.encodedImage = encodedImage; + this.expectedUrl = expectedUrl; + } + + @Override + public String getBase64Image() { + return encodedImage; + } + + @Override + public String getImageName() { + return imageName; + } + + @Override + public int getXOffset() { + return 0; + } + + @Override + public int getYOffset() { + return 0; + } +} diff --git a/framework-tests/bellatrix.playwright.tests/src/test/java/opencv/data/ImageInBase64.java b/framework-tests/bellatrix.playwright.tests/src/test/java/opencv/data/ImageInBase64.java new file mode 100644 index 00000000..2267eabd --- /dev/null +++ b/framework-tests/bellatrix.playwright.tests/src/test/java/opencv/data/ImageInBase64.java @@ -0,0 +1,24 @@ +package opencv.data; + +public class ImageInBase64 { + /** + * + */ + public static final String falcon9BackButton = ""; + /** + * + */ + public static final String commentTextArea = ""; + /** + * + */ + public static final String falcon9 = ""; + /** + * + */ + public static final String searchInput = ""; + /** + * + */ + public static final String falconResults = ""; +} diff --git a/framework-tests/bellatrix.playwright.tests/src/test/java/opencv/data/NavigationImageInBase64.java b/framework-tests/bellatrix.playwright.tests/src/test/java/opencv/data/NavigationImageInBase64.java new file mode 100644 index 00000000..fa7344ff --- /dev/null +++ b/framework-tests/bellatrix.playwright.tests/src/test/java/opencv/data/NavigationImageInBase64.java @@ -0,0 +1,29 @@ +package opencv.data; + +public class NavigationImageInBase64 { + /** + * + */ + public static final String homeLabel = ""; + /** + * + */ + public static final String blogLabel = ""; + /** + * + */ + public static final String checkoutLabel = ""; + /** + * + */ + public static final String contactFormLabel = ""; + /** + * + */ + public static final String myAccountLabel = ""; + + /** + * + */ + public static final String promotionsLabel = ""; +} diff --git a/framework-tests/bellatrix.web.tests/pom.xml b/framework-tests/bellatrix.web.tests/pom.xml index 60718d05..b0e2fdac 100644 --- a/framework-tests/bellatrix.web.tests/pom.xml +++ b/framework-tests/bellatrix.web.tests/pom.xml @@ -34,6 +34,18 @@ javafaker 1.0.2 + + solutions.bellatrix + bellatrix.plugins.opencv + 1.0 + test + + + org.junit.jupiter + junit-jupiter-params + 5.9.2 + test + diff --git a/framework-tests/bellatrix.web.tests/src/test/java/opencv/OpenCvTests.java b/framework-tests/bellatrix.web.tests/src/test/java/opencv/OpenCvTests.java new file mode 100644 index 00000000..04bd7cb0 --- /dev/null +++ b/framework-tests/bellatrix.web.tests/src/test/java/opencv/OpenCvTests.java @@ -0,0 +1,79 @@ +package opencv; + +import opencv.data.EncodedImageDemo; +import opencv.data.EncodedImageNavigationDemo; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.params.ParameterizedTest; +import org.junit.jupiter.params.provider.EnumSource; +import org.openqa.selenium.Keys; +import solutions.bellatrix.web.components.*; +import solutions.bellatrix.web.infrastructure.Browser; +import solutions.bellatrix.web.infrastructure.ExecutionBrowser; +import solutions.bellatrix.web.infrastructure.Lifecycle; +import solutions.bellatrix.web.infrastructure.junit.WebTest; + +@ExecutionBrowser(browser = Browser.CHROME, lifecycle = Lifecycle.REUSE_IF_STARTED) +public class OpenCvTests extends WebTest { + @Override + public void beforeEach() throws Exception { + super.beforeEach(); + app().browser().maximize(); + } + + @Test + public void actionPerformed_when_convertBase64ToImage_and_clickImage() { + var falcon9Image = app().create().byImage(Anchor.class, EncodedImageDemo.FALCON_9); + + app().navigate().to("http://demos.bellatrix.solutions/"); + falcon9Image.click(); + + app().browser().assertLandedOnPage("product/falcon-9"); + } + + @Test + public void actionPerformed_when_convertBase64ToImage_and_dragAndDropImage() { + var emailNotes = app().create().byId(Span.class, "email-notes"); + var commentTextArea = app().create().byId(TextArea.class, "comment"); + var falcon9BackButtonImage = app().create().byImage(ActionImage.class, EncodedImageDemo.FALCON_9_BACK_BUTTON); + var commentTextAreaImage = app().create().byImage(ActionImage.class, EncodedImageDemo.COMMENT_TEXT_AREA); + + app().navigate().to("https://demos.bellatrix.solutions/2018/04/06/proton-rocket-family/"); + emailNotes.scrollToVisible(); + falcon9BackButtonImage.dragAndDrop(commentTextAreaImage); + + commentTextArea.validateTextIs("https://demos.bellatrix.solutions/2018/04/06/hello-world/"); + } + + @Test + public void actionPerformed_when_convertBase64ToImage_and_clickButton() { + Button falcon9Image = app().create().byImage(Button.class, EncodedImageDemo.FALCON_9); + + app().navigate().to("http://demos.bellatrix.solutions/"); + falcon9Image.click(); + + app().browser().assertLandedOnPage("product/falcon-9"); + } + + @Test + public void actionPerformed_when_locateComplexComponents() { + SearchField websiteSearch = app().create().byImage(SearchField.class, EncodedImageDemo.SEARCH_INPUT); + + app().navigate().to("http://demos.bellatrix.solutions/"); + websiteSearch.setSearch("Falcon"); + websiteSearch.getWrappedElement().sendKeys(Keys.ENTER); + + app().browser().assertLandedOnPage("?s=Falcon&post_type=product"); + app().create().byImage(ActionImage.class, EncodedImageDemo.FALCON_RESULTS).validateIsVisible(); + } + + @ParameterizedTest + @EnumSource(EncodedImageNavigationDemo.class) + public void actionPerformed_when_locateButtonComponents(EncodedImageNavigationDemo navigationItem) { + Button navgationButton = app().create().byImage(Button.class, navigationItem); + + app().navigate().to("http://demos.bellatrix.solutions/"); + navgationButton.click(); + + app().browser().assertLandedOnPage(navigationItem.getExpectedUrl()); + } +} \ No newline at end of file diff --git a/framework-tests/bellatrix.web.tests/src/test/java/opencv/data/EncodedImageDemo.java b/framework-tests/bellatrix.web.tests/src/test/java/opencv/data/EncodedImageDemo.java new file mode 100644 index 00000000..a891aae8 --- /dev/null +++ b/framework-tests/bellatrix.web.tests/src/test/java/opencv/data/EncodedImageDemo.java @@ -0,0 +1,63 @@ +package opencv.data; + +import solutions.bellatrix.plugins.opencv.Base64Encodable; + +public enum EncodedImageDemo implements Base64Encodable { + /** + * + */ + FALCON_9("falcon9", ImageInBase64.falcon9, 50, 25), + /** + * + */ + SEARCH_INPUT("searchInput", ImageInBase64.searchInput), + /** + * + */ + FALCON_RESULTS("falconResults", ImageInBase64.falconResults), + /** + * + */ + FALCON_9_BACK_BUTTON("falcon9BackButton", ImageInBase64.falcon9BackButton), + /** + * + */ + COMMENT_TEXT_AREA("commentTextArea", ImageInBase64.commentTextArea); + + private final String imageName; + private final String encodedImage; + private int xOffset = 0; + private int yOffset = 0; + + EncodedImageDemo(String imageName, String encodedImage, int xOffset, int yOffset) { + this.imageName = imageName; + this.encodedImage = encodedImage; + this.xOffset = xOffset; + this.yOffset = yOffset; + } + + EncodedImageDemo(String imageName, String encodedImage) { + this.imageName = imageName; + this.encodedImage = encodedImage; + } + + @Override + public String getBase64Image() { + return encodedImage; + } + + @Override + public String getImageName() { + return imageName; + } + + @Override + public int getXOffset() { + return xOffset; + } + + @Override + public int getYOffset() { + return yOffset; + } +} diff --git a/framework-tests/bellatrix.web.tests/src/test/java/opencv/data/EncodedImageNavigationDemo.java b/framework-tests/bellatrix.web.tests/src/test/java/opencv/data/EncodedImageNavigationDemo.java new file mode 100644 index 00000000..895cb378 --- /dev/null +++ b/framework-tests/bellatrix.web.tests/src/test/java/opencv/data/EncodedImageNavigationDemo.java @@ -0,0 +1,61 @@ +package opencv.data; + +import lombok.Getter; +import solutions.bellatrix.plugins.opencv.Base64Encodable; + +public enum EncodedImageNavigationDemo implements Base64Encodable { + /** + * + */ + HOME_LABEL("homeLabel", NavigationImageInBase64.homeLabel, "https://demos.bellatrix.solutions/"), + /** + * + */ + BLOG_LABEL("blogLabel", NavigationImageInBase64.blogLabel, "/blog/"), + /** + * + */ + CHECKOUT_LABEL("checkoutLabel", NavigationImageInBase64.checkoutLabel, "/cart/"), + /** + * + */ + CONTACT_FORM_LABEL("contactFormLabel", NavigationImageInBase64.contactFormLabel, "/contact-form/"), + /** + * + */ + MY_ACCOUNT_LABEL("myAccountLabel", NavigationImageInBase64.myAccountLabel, "/my-account/"), + /** + * + */ + PROMOTIONS_LABEL("promotionsLabel", NavigationImageInBase64.promotionsLabel, "/welcome/"); + + private final String imageName; + private final String encodedImage; + @Getter private final String expectedUrl; + + EncodedImageNavigationDemo(String imageName, String encodedImage, String expectedUrl) { + this.imageName = imageName; + this.encodedImage = encodedImage; + this.expectedUrl = expectedUrl; + } + + @Override + public String getBase64Image() { + return encodedImage; + } + + @Override + public String getImageName() { + return imageName; + } + + @Override + public int getXOffset() { + return 0; + } + + @Override + public int getYOffset() { + return 0; + } +} diff --git a/framework-tests/bellatrix.web.tests/src/test/java/opencv/data/ImageInBase64.java b/framework-tests/bellatrix.web.tests/src/test/java/opencv/data/ImageInBase64.java new file mode 100644 index 00000000..3b809171 --- /dev/null +++ b/framework-tests/bellatrix.web.tests/src/test/java/opencv/data/ImageInBase64.java @@ -0,0 +1,24 @@ +package opencv.data; + +public class ImageInBase64 { + /** + * + */ + public static final String falcon9BackButton = ""; + /** + * + */ + public static final String commentTextArea = ""; + /** + * + */ + public static final String falcon9 = ""; + /** + * + */ + public static final String searchInput = ""; + /** + * + */ + public static final String falconResults = ""; +} diff --git a/framework-tests/bellatrix.web.tests/src/test/java/opencv/data/NavigationImageInBase64.java b/framework-tests/bellatrix.web.tests/src/test/java/opencv/data/NavigationImageInBase64.java new file mode 100644 index 00000000..fa7344ff --- /dev/null +++ b/framework-tests/bellatrix.web.tests/src/test/java/opencv/data/NavigationImageInBase64.java @@ -0,0 +1,29 @@ +package opencv.data; + +public class NavigationImageInBase64 { + /** + * + */ + public static final String homeLabel = ""; + /** + * + */ + public static final String blogLabel = ""; + /** + * + */ + public static final String checkoutLabel = ""; + /** + * + */ + public static final String contactFormLabel = ""; + /** + * + */ + public static final String myAccountLabel = ""; + + /** + * + */ + public static final String promotionsLabel = ""; +} diff --git a/getting-started/bellatrix.playwright.getting.started/pom.xml b/getting-started/bellatrix.playwright.getting.started/pom.xml index 77ea044f..f11c2982 100644 --- a/getting-started/bellatrix.playwright.getting.started/pom.xml +++ b/getting-started/bellatrix.playwright.getting.started/pom.xml @@ -45,6 +45,11 @@ 1.0 compile + + solutions.bellatrix + bellatrix.plugins.opencv + 1.0 + solutions.bellatrix bellatrix.plugins.visual-regression-tracker diff --git a/getting-started/bellatrix.playwright.getting.started/src/test/java/O26_openCv_image_actions/OpenCvDemoTests.java b/getting-started/bellatrix.playwright.getting.started/src/test/java/O26_openCv_image_actions/OpenCvDemoTests.java new file mode 100644 index 00000000..216c332a --- /dev/null +++ b/getting-started/bellatrix.playwright.getting.started/src/test/java/O26_openCv_image_actions/OpenCvDemoTests.java @@ -0,0 +1,40 @@ +package O26_openCv_image_actions; + +import O26_openCv_image_actions.data.enums.EncodedImageDemo; +import org.junit.jupiter.api.Test; +import solutions.bellatrix.playwright.components.ActionImage; +import solutions.bellatrix.playwright.components.Span; +import solutions.bellatrix.playwright.components.TextArea; +import solutions.bellatrix.playwright.infrastructure.junit.WebTest; + +public class OpenCvDemoTests extends WebTest { + @Override + public void beforeEach() throws Exception { + super.beforeEach(); + app().browser().setSize(1920, 1080); + } + + @Test + public void actionPerformed_when_convertBase64ToImage_and_clickImage() { + var falcon9Image = app().create().byImage(ActionImage.class, EncodedImageDemo.FALCON_9); + + app().navigate().to("http://demos.bellatrix.solutions/"); + falcon9Image.click(); + + app().browser().validateLandedOnPage("product/falcon-9"); + } + + @Test + public void actionPerformed_when_convertBase64ToImage_and_dragAndDropImage() { + var emailNotes = app().create().byId(Span.class, "email-notes"); + var commentTextArea = app().create().byId(TextArea.class, "comment"); + var falcon9BackButtonImage = app().create().byImage(ActionImage.class, EncodedImageDemo.FALCON_9_BACK_BUTTON); + var commentTextAreaImage = app().create().byImage(ActionImage.class, EncodedImageDemo.COMMENT_TEXT_AREA); + + app().navigate().to("https://demos.bellatrix.solutions/2018/04/06/proton-rocket-family/"); + emailNotes.scrollToVisible(); + falcon9BackButtonImage.dragAndDrop(commentTextAreaImage); + + commentTextArea.validateTextIs("https://demos.bellatrix.solutions/2018/04/06/hello-world/"); + } +} \ No newline at end of file diff --git a/getting-started/bellatrix.playwright.getting.started/src/test/java/O26_openCv_image_actions/data/ImageInBase64.java b/getting-started/bellatrix.playwright.getting.started/src/test/java/O26_openCv_image_actions/data/ImageInBase64.java new file mode 100644 index 00000000..69137665 --- /dev/null +++ b/getting-started/bellatrix.playwright.getting.started/src/test/java/O26_openCv_image_actions/data/ImageInBase64.java @@ -0,0 +1,16 @@ +package O26_openCv_image_actions.data; + +public class ImageInBase64 { + /** + * + */ + public static final String falcon9BackButton = ""; + /** + * + */ + public static final String commentTextArea = ""; + /** + * + */ + public static final String falcon9 = ""; +} diff --git a/getting-started/bellatrix.playwright.getting.started/src/test/java/O26_openCv_image_actions/data/enums/EncodedImageDemo.java b/getting-started/bellatrix.playwright.getting.started/src/test/java/O26_openCv_image_actions/data/enums/EncodedImageDemo.java new file mode 100644 index 00000000..2801310f --- /dev/null +++ b/getting-started/bellatrix.playwright.getting.started/src/test/java/O26_openCv_image_actions/data/enums/EncodedImageDemo.java @@ -0,0 +1,56 @@ +package O26_openCv_image_actions.data.enums; + +import O26_openCv_image_actions.data.ImageInBase64; +import solutions.bellatrix.plugins.opencv.Base64Encodable; + +public enum EncodedImageDemo implements Base64Encodable { + /** + * + */ + FALCON_9("falcon9", ImageInBase64.falcon9, 50, 25), + /** + * + */ + FALCON_9_BACK_BUTTON("falcon9BackButton", ImageInBase64.falcon9BackButton), + /** + * + */ + COMMENT_TEXT_AREA("commentTextArea", ImageInBase64.commentTextArea); + + private final String imageName; + private final String encodedImage; + private int xOffset = 0; + private int yOffset = 0; + + EncodedImageDemo(String imageName, String encodedImage, int xOffset, int yOffset) { + this.imageName = imageName; + this.encodedImage = encodedImage; + this.xOffset = xOffset; + this.yOffset = yOffset; + } + + EncodedImageDemo(String imageName, String encodedImage) { + this.imageName = imageName; + this.encodedImage = encodedImage; + } + + @Override + public String getBase64Image() { + return encodedImage; + } + + @Override + public String getImageName() { + return imageName; + } + + @Override + public int getXOffset() { + return xOffset; + } + + @Override + public int getYOffset() { + return yOffset; + } +} diff --git a/getting-started/bellatrix.playwright.getting.started/src/test/java/O26_openCv_image_actions/todo.txt b/getting-started/bellatrix.playwright.getting.started/src/test/java/O26_openCv_image_actions/todo.txt new file mode 100644 index 00000000..688c96b8 --- /dev/null +++ b/getting-started/bellatrix.playwright.getting.started/src/test/java/O26_openCv_image_actions/todo.txt @@ -0,0 +1,6 @@ + 1. Take a screenshot of the place on the screen that you want to perform actions on + 2. Convert the image to a base 64 format with an online tool + 3. Create an enum implementing the Base64Encodable interface + 4. Add the base 64 string, the name of the image and the offsets to the enum - check example enum: EncodedImageDemo + (Note) If the x and y offsets are set to 0 -> it will take the coordinates of the center of the picture + 5. Create ActionImage component using the ImageBase64FindStrategy and passing the enum as an argument \ No newline at end of file diff --git a/getting-started/bellatrix.web.getting.started/pom.xml b/getting-started/bellatrix.web.getting.started/pom.xml index b5dc88f2..140e874f 100644 --- a/getting-started/bellatrix.web.getting.started/pom.xml +++ b/getting-started/bellatrix.web.getting.started/pom.xml @@ -50,6 +50,18 @@ 5.9.2 test + + jakarta.xml.bind + jakarta.xml.bind-api + 2.3.2 + test + + + jakarta.xml.bind + jakarta.xml.bind-api + 2.3.2 + compile + solutions.bellatrix bellatrix.plugins.visual-regression-tracker diff --git a/getting-started/bellatrix.web.getting.started/src/test/java/O26_openCv_image_actions/OpenCvDemoTests.java b/getting-started/bellatrix.web.getting.started/src/test/java/O26_openCv_image_actions/OpenCvDemoTests.java new file mode 100644 index 00000000..3fe3b9d3 --- /dev/null +++ b/getting-started/bellatrix.web.getting.started/src/test/java/O26_openCv_image_actions/OpenCvDemoTests.java @@ -0,0 +1,70 @@ +package O26_openCv_image_actions; + +import O26_openCv_image_actions.data.enums.EncodedImageDemo; +import org.junit.jupiter.api.Test; +import plugins.screenshots.ScreenshotPlugin; +import solutions.bellatrix.web.components.ActionImage; +import solutions.bellatrix.web.components.Button; +import solutions.bellatrix.web.components.Span; +import solutions.bellatrix.web.components.TextArea; +import solutions.bellatrix.web.infrastructure.*; +import solutions.bellatrix.web.infrastructure.junit.WebTest; + +/** + * Image locators are generated from real images using the tool: OpenCV Image Match Finder + * Where you can supply either: + * - Image from disc + * - Paste image from clipboard + * - Drag and drop image + * In addition it helps in troubleshooting where you want to check the confidence level of a match in a context image. + */ +@ExecutionBrowser(browser = Browser.CHROME, lifecycle = Lifecycle.RESTART_EVERY_TIME) +public class OpenCvDemoTests extends WebTest { + + @Override + protected void configure() { + addPlugin(BrowserLifecyclePlugin.class); + // Adding the screenshot plugin as a WebScreenshot plugin is required for the feature to work correctly + addPluginAs(ScreenshotPlugin.class, WebScreenshotPlugin.class); + } + + @Override + public void beforeEach() throws Exception { + super.beforeEach(); + app().browser().setSize(1920, 1080); + } + + @Test + public void actionPerformed_when_convertBase64ToImage_and_clickImage() { + var falcon9Image = app().create().byImage(ActionImage.class, EncodedImageDemo.FALCON_9); + + app().navigate().to("http://demos.bellatrix.solutions/"); + falcon9Image.click(); + + app().browser().assertLandedOnPage("product/falcon-9"); + } + + @Test + public void actionPerformed_when_convertBase64ToImage_and_clickButton() { + var falcon9Image = app().create().byImage(Button.class, EncodedImageDemo.FALCON_9); + + app().navigate().to("http://demos.bellatrix.solutions/"); + falcon9Image.click(); + + app().browser().assertLandedOnPage("product/falcon-9"); + } + + @Test + public void actionPerformed_when_convertBase64ToImage_and_dragAndDropImage() { + var emailNotes = app().create().byId(Span.class, "email-notes"); + var commentTextArea = app().create().byId(TextArea.class, "comment"); + var falcon9BackButtonImage = app().create().byImage(ActionImage.class, EncodedImageDemo.FALCON_9_BACK_BUTTON); + var commentTextAreaImage = app().create().byImage(ActionImage.class, EncodedImageDemo.COMMENT_TEXT_AREA); + + app().navigate().to("https://demos.bellatrix.solutions/2018/04/06/proton-rocket-family/"); + emailNotes.scrollToVisible(); + falcon9BackButtonImage.dragAndDrop(commentTextAreaImage); + + commentTextArea.validateTextIs("https://demos.bellatrix.solutions/2018/04/06/hello-world/"); + } +} \ No newline at end of file diff --git a/getting-started/bellatrix.web.getting.started/src/test/java/O26_openCv_image_actions/data/ImageInBase64.java b/getting-started/bellatrix.web.getting.started/src/test/java/O26_openCv_image_actions/data/ImageInBase64.java new file mode 100644 index 00000000..9af74c49 --- /dev/null +++ b/getting-started/bellatrix.web.getting.started/src/test/java/O26_openCv_image_actions/data/ImageInBase64.java @@ -0,0 +1,24 @@ +package O26_openCv_image_actions.data; + +/** + * Base64 data is generated from real images using the tool: OpenCV Image Match Finder + * Where you can supply either: + * - Image from disc + * - Paste image from clipboard + * - Drag and drop image + * In addition it helps in troubleshooting where you want to check the confidence level of a match in a context image. + */ +public class ImageInBase64 { + /** + * + */ + public static final String falcon9BackButton = ""; + /** + * + */ + public static final String commentTextArea = ""; + /** + * + */ + public static final String falcon9 = ""; +} diff --git a/getting-started/bellatrix.web.getting.started/src/test/java/O26_openCv_image_actions/data/enums/EncodedImageDemo.java b/getting-started/bellatrix.web.getting.started/src/test/java/O26_openCv_image_actions/data/enums/EncodedImageDemo.java new file mode 100644 index 00000000..2801310f --- /dev/null +++ b/getting-started/bellatrix.web.getting.started/src/test/java/O26_openCv_image_actions/data/enums/EncodedImageDemo.java @@ -0,0 +1,56 @@ +package O26_openCv_image_actions.data.enums; + +import O26_openCv_image_actions.data.ImageInBase64; +import solutions.bellatrix.plugins.opencv.Base64Encodable; + +public enum EncodedImageDemo implements Base64Encodable { + /** + * + */ + FALCON_9("falcon9", ImageInBase64.falcon9, 50, 25), + /** + * + */ + FALCON_9_BACK_BUTTON("falcon9BackButton", ImageInBase64.falcon9BackButton), + /** + * + */ + COMMENT_TEXT_AREA("commentTextArea", ImageInBase64.commentTextArea); + + private final String imageName; + private final String encodedImage; + private int xOffset = 0; + private int yOffset = 0; + + EncodedImageDemo(String imageName, String encodedImage, int xOffset, int yOffset) { + this.imageName = imageName; + this.encodedImage = encodedImage; + this.xOffset = xOffset; + this.yOffset = yOffset; + } + + EncodedImageDemo(String imageName, String encodedImage) { + this.imageName = imageName; + this.encodedImage = encodedImage; + } + + @Override + public String getBase64Image() { + return encodedImage; + } + + @Override + public String getImageName() { + return imageName; + } + + @Override + public int getXOffset() { + return xOffset; + } + + @Override + public int getYOffset() { + return yOffset; + } +} diff --git a/getting-started/bellatrix.web.getting.started/src/test/java/O26_openCv_image_actions/todo.txt b/getting-started/bellatrix.web.getting.started/src/test/java/O26_openCv_image_actions/todo.txt new file mode 100644 index 00000000..11644461 --- /dev/null +++ b/getting-started/bellatrix.web.getting.started/src/test/java/O26_openCv_image_actions/todo.txt @@ -0,0 +1,7 @@ + 1. Take a screenshot of the place on the screen that you want to perform actions on + 2. Convert the image to a base 64 format with the tool created for this purpose: https://automatetheplanet.github.io/OpenCV-Image-Match-Finder/ + 3. Create an enum implementing the Base64Encodable interface + 4. Add the base 64 string, the name of the image and the offsets to the enum - check example enum: EncodedImageDemo + (Note) If the x and y offsets are set to 0 -> it will take the coordinates of the center of the picture + 5. Create ActionImage component using the ImageBase64FindStrategy and passing the enum as an argument + 6. If an image match is not found - use the Image Match Finder tool to troubleshoot: https://automatetheplanet.github.io/OpenCV-Image-Match-Finder/ \ No newline at end of file diff --git a/pom.xml b/pom.xml index bc875e58..b40c83bd 100644 --- a/pom.xml +++ b/pom.xml @@ -15,6 +15,8 @@ bellatrix.plugins.video bellatrix.plugins.jira.zephyr bellatrix.plugins.visual-regression-tracker + bellatrix.plugins.opencv + bellatrix.sms bellatrix.web bellatrix.playwright bellatrix.desktop @@ -31,7 +33,8 @@ getting-started/bellatrix.web.getting.started getting-started/bellatrix.playwright.getting.started - bellatrix.sms + +