2018-05-02 19:01:17 +03:00
|
|
|
import base64
|
|
|
|
from io import BytesIO
|
|
|
|
import os
|
2018-05-16 21:59:36 +02:00
|
|
|
|
|
|
|
import time
|
2018-06-21 18:40:27 +02:00
|
|
|
from timeit import timeit
|
|
|
|
|
2018-05-02 19:01:17 +03:00
|
|
|
from PIL import Image, ImageChops
|
2018-01-14 19:43:36 +02:00
|
|
|
from appium.webdriver.common.mobileby import MobileBy
|
2018-05-16 21:59:36 +02:00
|
|
|
from appium.webdriver.common.touch_action import TouchAction
|
2018-01-26 13:07:09 +02:00
|
|
|
from selenium.common.exceptions import NoSuchElementException
|
|
|
|
from selenium.common.exceptions import TimeoutException
|
2017-08-28 13:02:20 +03:00
|
|
|
from selenium.webdriver.support.wait import WebDriverWait
|
|
|
|
from selenium.webdriver.support import expected_conditions
|
|
|
|
|
|
|
|
|
|
|
|
class BaseElement(object):
|
|
|
|
class Locator(object):
|
|
|
|
|
|
|
|
def __init__(self, by, value):
|
|
|
|
self.by = by
|
|
|
|
self.value = value
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def xpath_selector(locator, value):
|
2017-09-13 17:34:42 +03:00
|
|
|
return locator(MobileBy.XPATH, value)
|
2017-08-28 13:02:20 +03:00
|
|
|
|
2017-09-11 13:43:42 +03:00
|
|
|
@classmethod
|
|
|
|
def accessibility_id(locator, value):
|
|
|
|
return locator(MobileBy.ACCESSIBILITY_ID, value)
|
|
|
|
|
2018-04-26 08:22:11 +02:00
|
|
|
@classmethod
|
|
|
|
def text_selector(locator, text):
|
|
|
|
return BaseElement.Locator.xpath_selector('//*[@text="' + text + '"]')
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def text_part_selector(locator, text):
|
|
|
|
return BaseElement.Locator.xpath_selector('//*[contains(@text, "' + text + '")]')
|
|
|
|
|
2018-08-10 13:09:19 +03:00
|
|
|
@classmethod
|
|
|
|
def id(locator, value):
|
|
|
|
return locator(MobileBy.ID, value)
|
|
|
|
|
2018-10-19 17:16:29 +03:00
|
|
|
@classmethod
|
|
|
|
def webview_selector(cls, value):
|
|
|
|
xpath_expression = '//*[@text="{0}"] | //*[@content-desc="{desc}"]'.format(value, desc=value)
|
|
|
|
return cls(MobileBy.XPATH, xpath_expression)
|
|
|
|
|
2017-08-28 13:02:20 +03:00
|
|
|
def __str__(self, *args, **kwargs):
|
|
|
|
return "%s:%s" % (self.by, self.value)
|
|
|
|
|
|
|
|
def __init__(self, driver):
|
|
|
|
self.driver = driver
|
|
|
|
self.locator = None
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
return self.__class__.__name__
|
|
|
|
|
|
|
|
def navigate(self):
|
|
|
|
return None
|
|
|
|
|
|
|
|
def find_element(self):
|
2018-06-14 12:57:04 +02:00
|
|
|
for _ in range(3):
|
|
|
|
try:
|
|
|
|
return self.driver.find_element(self.locator.by, self.locator.value)
|
2018-08-10 13:09:19 +03:00
|
|
|
except NoSuchElementException:
|
2018-08-15 15:51:52 +03:00
|
|
|
raise NoSuchElementException(
|
|
|
|
"Device %s: '%s' is not found on the screen" % (self.driver.number, self.name)) from None
|
2018-08-10 13:09:19 +03:00
|
|
|
except Exception as exception:
|
|
|
|
if 'Internal Server Error' in str(exception):
|
2018-06-14 12:57:04 +02:00
|
|
|
continue
|
2017-08-28 13:02:20 +03:00
|
|
|
|
2017-10-05 22:41:17 +03:00
|
|
|
def find_elements(self):
|
|
|
|
return self.driver.find_elements(self.locator.by, self.locator.value)
|
|
|
|
|
2018-06-28 20:46:51 +02:00
|
|
|
def click(self):
|
|
|
|
self.find_element().click()
|
2018-08-15 15:51:52 +03:00
|
|
|
self.driver.info('Tap on %s' % self.name)
|
2018-06-28 20:46:51 +02:00
|
|
|
|
2017-08-31 16:39:41 +03:00
|
|
|
def wait_for_element(self, seconds=10):
|
2018-01-26 13:07:09 +02:00
|
|
|
try:
|
2018-05-02 19:01:17 +03:00
|
|
|
return WebDriverWait(self.driver, seconds) \
|
2018-01-26 13:07:09 +02:00
|
|
|
.until(expected_conditions.presence_of_element_located((self.locator.by, self.locator.value)))
|
2018-08-10 13:09:19 +03:00
|
|
|
except TimeoutException:
|
2018-08-15 15:51:52 +03:00
|
|
|
raise TimeoutException(
|
|
|
|
"Device %s: '%s' is not found on the screen" % (self.driver.number, self.name)) from None
|
2018-08-29 17:04:12 +03:00
|
|
|
|
|
|
|
def wait_for_elements(self, seconds=10):
|
|
|
|
try:
|
|
|
|
return WebDriverWait(self.driver, seconds) \
|
|
|
|
.until(expected_conditions.presence_of_all_elements_located((self.locator.by, self.locator.value)))
|
|
|
|
except TimeoutException:
|
|
|
|
raise TimeoutException(
|
|
|
|
"Device %s: '%s' is not found on the screen" % (self.driver.number, self.name)) from None
|
2017-08-28 13:02:20 +03:00
|
|
|
|
2018-07-19 11:57:45 +02:00
|
|
|
def wait_for_visibility_of_element(self, seconds=10, ignored_exceptions=None):
|
2018-03-15 22:01:08 +02:00
|
|
|
try:
|
2018-07-19 11:57:45 +02:00
|
|
|
return WebDriverWait(self.driver, seconds, ignored_exceptions=ignored_exceptions) \
|
2018-03-15 22:01:08 +02:00
|
|
|
.until(expected_conditions.visibility_of_element_located((self.locator.by, self.locator.value)))
|
2018-08-10 13:09:19 +03:00
|
|
|
except TimeoutException:
|
2018-08-15 15:51:52 +03:00
|
|
|
raise TimeoutException(
|
|
|
|
"Device %s: '%s' is not found on the screen" % (self.driver.number, self.name)) from None
|
2018-03-15 22:01:08 +02:00
|
|
|
|
2018-05-25 20:29:07 +03:00
|
|
|
def wait_for_invisibility_of_element(self, seconds=10):
|
|
|
|
try:
|
|
|
|
return WebDriverWait(self.driver, seconds) \
|
|
|
|
.until(expected_conditions.invisibility_of_element_located((self.locator.by, self.locator.value)))
|
2018-08-10 13:09:19 +03:00
|
|
|
except TimeoutException:
|
2018-08-15 15:51:52 +03:00
|
|
|
raise TimeoutException("Device %s: '%s' is still visible on the screen after %s seconds" % (
|
|
|
|
self.driver.number, self.name, seconds)) from None
|
2018-05-25 20:29:07 +03:00
|
|
|
|
2018-11-21 12:44:30 +02:00
|
|
|
def scroll_to_element(self, depth: int = 9, direction='down'):
|
|
|
|
for _ in range(depth):
|
2017-08-31 16:39:41 +03:00
|
|
|
try:
|
2017-10-11 23:10:57 +03:00
|
|
|
return self.find_element()
|
2017-08-31 16:39:41 +03:00
|
|
|
except NoSuchElementException:
|
2018-08-15 15:51:52 +03:00
|
|
|
self.driver.info('Scrolling down to %s' % self.name)
|
2018-11-21 12:44:30 +02:00
|
|
|
if direction == 'down':
|
|
|
|
self.driver.swipe(500, 1000, 500, 500)
|
|
|
|
else:
|
|
|
|
self.driver.swipe(500, 500, 500, 1000)
|
2017-08-31 16:39:41 +03:00
|
|
|
|
2017-09-13 17:34:42 +03:00
|
|
|
def is_element_present(self, sec=5):
|
|
|
|
try:
|
2018-08-15 15:51:52 +03:00
|
|
|
self.driver.info('Wait for %s' % self.name)
|
2018-03-15 22:01:08 +02:00
|
|
|
return self.wait_for_element(sec)
|
|
|
|
except TimeoutException:
|
|
|
|
return False
|
|
|
|
|
2018-07-19 11:57:45 +02:00
|
|
|
def is_element_displayed(self, sec=5, ignored_exceptions=None):
|
2018-03-15 22:01:08 +02:00
|
|
|
try:
|
2018-08-15 15:51:52 +03:00
|
|
|
self.driver.info('Wait for %s' % self.name)
|
2018-07-19 11:57:45 +02:00
|
|
|
return self.wait_for_visibility_of_element(sec, ignored_exceptions=ignored_exceptions)
|
2017-09-13 17:34:42 +03:00
|
|
|
except TimeoutException:
|
|
|
|
return False
|
|
|
|
|
2017-09-21 20:01:04 +03:00
|
|
|
@property
|
|
|
|
def text(self):
|
|
|
|
return self.find_element().text
|
|
|
|
|
2018-05-02 19:01:17 +03:00
|
|
|
@property
|
|
|
|
def template(self):
|
|
|
|
try:
|
|
|
|
return self.__template
|
|
|
|
except FileNotFoundError:
|
|
|
|
raise FileNotFoundError('Please add %s image as template' % self.name)
|
|
|
|
|
|
|
|
@template.setter
|
|
|
|
def template(self, value):
|
|
|
|
self.__template = Image.open(os.sep.join(__file__.split(os.sep)[:-1]) + '/elements_templates/%s' % value)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def image(self):
|
|
|
|
return Image.open(BytesIO(base64.b64decode(self.find_element().screenshot_as_base64)))
|
|
|
|
|
2018-05-16 21:59:36 +02:00
|
|
|
def is_element_image_equals_template(self, file_name: str = ''):
|
|
|
|
if file_name:
|
|
|
|
self.template = file_name
|
2018-05-02 19:01:17 +03:00
|
|
|
return not ImageChops.difference(self.image, self.template).getbbox()
|
|
|
|
|
2018-05-16 21:59:36 +02:00
|
|
|
def swipe_element(self):
|
|
|
|
element = self.find_element()
|
|
|
|
location, size = element.location, element.size
|
|
|
|
x, y = location['x'], location['y']
|
|
|
|
width, height = size['width'], size['height']
|
2018-07-03 20:50:18 +02:00
|
|
|
self.driver.swipe(start_x=x + width * 0.75, start_y=y + height / 2, end_x=x, end_y=y + height / 2)
|
2018-05-16 21:59:36 +02:00
|
|
|
|
|
|
|
def long_press_element(self):
|
|
|
|
element = self.find_element()
|
2018-08-15 15:51:52 +03:00
|
|
|
self.driver.info('Long press %s' % self.name)
|
2018-05-16 21:59:36 +02:00
|
|
|
action = TouchAction(self.driver)
|
|
|
|
action.long_press(element).release().perform()
|
|
|
|
|
2018-06-21 18:40:27 +02:00
|
|
|
def measure_time_before_element_appears(self, max_wait_time=30):
|
|
|
|
def wrapper():
|
|
|
|
return self.wait_for_visibility_of_element(max_wait_time)
|
2018-06-28 20:46:51 +02:00
|
|
|
|
2018-06-21 18:40:27 +02:00
|
|
|
return timeit(wrapper, number=1)
|
|
|
|
|
|
|
|
def measure_time_while_element_is_shown(self, max_wait_time=30):
|
|
|
|
def wrapper():
|
|
|
|
return self.wait_for_invisibility_of_element(max_wait_time)
|
2018-06-28 20:46:51 +02:00
|
|
|
|
2018-06-21 18:40:27 +02:00
|
|
|
return timeit(wrapper, number=1)
|
|
|
|
|
2017-08-28 13:02:20 +03:00
|
|
|
|
|
|
|
class BaseEditBox(BaseElement):
|
|
|
|
|
|
|
|
def __init__(self, driver):
|
|
|
|
super(BaseEditBox, self).__init__(driver)
|
|
|
|
|
|
|
|
def send_keys(self, value):
|
|
|
|
self.find_element().send_keys(value)
|
2018-08-15 15:51:52 +03:00
|
|
|
self.driver.info("Type '%s' to %s" % (value, self.name))
|
2017-08-28 13:02:20 +03:00
|
|
|
|
2017-10-05 22:41:17 +03:00
|
|
|
def set_value(self, value):
|
|
|
|
self.find_element().set_value(value)
|
2018-08-15 15:51:52 +03:00
|
|
|
self.driver.info("Type '%s' to %s" % (value, self.name))
|
2017-10-05 22:41:17 +03:00
|
|
|
|
2017-10-30 13:11:58 +02:00
|
|
|
def clear(self):
|
|
|
|
self.find_element().clear()
|
2018-08-15 15:51:52 +03:00
|
|
|
self.driver.info('Clear text in %s' % self.name)
|
2017-10-30 13:11:58 +02:00
|
|
|
|
2018-05-16 21:59:36 +02:00
|
|
|
def delete_last_symbols(self, number_of_symbols_to_delete: int):
|
2018-08-15 15:51:52 +03:00
|
|
|
self.driver.info('Delete last %s symbols from %s' % (number_of_symbols_to_delete, self.name))
|
2018-05-16 21:59:36 +02:00
|
|
|
self.click()
|
|
|
|
for _ in range(number_of_symbols_to_delete):
|
|
|
|
time.sleep(1)
|
|
|
|
self.driver.press_keycode(67)
|
|
|
|
|
|
|
|
def paste_text_from_clipboard(self):
|
2018-08-15 15:51:52 +03:00
|
|
|
self.driver.info('Paste text from clipboard into %s' % self.name)
|
2018-05-16 21:59:36 +02:00
|
|
|
self.long_press_element()
|
|
|
|
time.sleep(2)
|
|
|
|
action = TouchAction(self.driver)
|
|
|
|
location = self.find_element().location
|
|
|
|
x, y = location['x'], location['y']
|
2018-06-28 20:46:51 +02:00
|
|
|
action.press(x=x + 100, y=y - 50).release().perform()
|
2018-05-16 21:59:36 +02:00
|
|
|
|
|
|
|
def cut_text(self):
|
2018-08-15 15:51:52 +03:00
|
|
|
self.driver.info('Cut text in %s' % self.name)
|
2018-05-16 21:59:36 +02:00
|
|
|
location = self.find_element().location
|
|
|
|
x, y = location['x'], location['y']
|
|
|
|
action = TouchAction(self.driver)
|
|
|
|
action.long_press(x=x, y=y).release().perform()
|
|
|
|
time.sleep(2)
|
2018-06-28 20:46:51 +02:00
|
|
|
action.press(x=x + 50, y=y - 50).release().perform()
|
2018-05-16 21:59:36 +02:00
|
|
|
|
2017-08-28 13:02:20 +03:00
|
|
|
|
|
|
|
class BaseText(BaseElement):
|
|
|
|
|
|
|
|
def __init__(self, driver):
|
|
|
|
super(BaseText, self).__init__(driver)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def text(self):
|
2017-09-21 20:01:04 +03:00
|
|
|
text = self.find_element().text
|
2018-08-15 15:51:52 +03:00
|
|
|
self.driver.info('%s is %s' % (self.name, text))
|
2017-09-21 20:01:04 +03:00
|
|
|
return text
|
2017-08-28 13:02:20 +03:00
|
|
|
|
|
|
|
|
|
|
|
class BaseButton(BaseElement):
|
|
|
|
|
|
|
|
def __init__(self, driver):
|
|
|
|
super(BaseButton, self).__init__(driver)
|
|
|
|
|
|
|
|
def click(self):
|
|
|
|
self.find_element().click()
|
2018-08-15 15:51:52 +03:00
|
|
|
self.driver.info('Tap on %s' % self.name)
|
2017-08-28 13:02:20 +03:00
|
|
|
return self.navigate()
|
2018-02-09 17:16:07 +02:00
|
|
|
|
2018-02-19 13:51:53 +02:00
|
|
|
def click_until_presence_of_element(self, desired_element, attempts=3):
|
2018-02-09 17:16:07 +02:00
|
|
|
counter = 0
|
2018-03-28 13:21:39 +03:00
|
|
|
while not desired_element.is_element_present(1) and counter <= attempts:
|
2018-02-09 17:16:07 +02:00
|
|
|
try:
|
2018-08-15 15:51:52 +03:00
|
|
|
self.driver.info('Tap on %s' % self.name)
|
2018-02-09 17:16:07 +02:00
|
|
|
self.find_element().click()
|
2018-08-15 15:51:52 +03:00
|
|
|
self.driver.info('Wait for %s' % desired_element.name)
|
2018-02-09 17:16:07 +02:00
|
|
|
desired_element.wait_for_element(5)
|
|
|
|
return self.navigate()
|
|
|
|
except (NoSuchElementException, TimeoutException):
|
|
|
|
counter += 1
|