import re import time from datetime import datetime, timedelta from time import sleep import dateutil.parser from appium.webdriver.common.touch_action import TouchAction from selenium.common.exceptions import NoSuchElementException, TimeoutException, StaleElementReferenceException, \ InvalidElementStateException from tests import emojis, common_password from views.base_element import Button, EditBox, Text, BaseElement, SilentButton from views.base_view import BaseView from views.home_view import HomeView from views.profile_view import ProfilePictureElement class CommandsButton(Button): def __init__(self, driver): super().__init__(driver, accessibility_id="show-extensions-icon") def click(self): self.click_until_presence_of_element(SendCommand(self.driver)) return self.navigate() class SendCommand(Button): def __init__(self, driver): super().__init__(driver, translation_id="send-transaction") def navigate(self): from views.send_transaction_view import SendTransactionView return SendTransactionView(self.driver) def click(self): self.wait_for_element().click() return self.navigate() class RequestCommand(Button): def __init__(self, driver): super().__init__(driver, translation_id="request-transaction") def navigate(self): from views.send_transaction_view import SendTransactionView return SendTransactionView(self.driver) def click(self): self.wait_for_element().click() return self.navigate() class GroupInfoButton(Button): def __init__(self, driver): super().__init__(driver, translation_id="group-info") def navigate(self): return GroupChatInfoView(self.driver) def click(self): self.wait_for_element().click() return self.navigate() class UnblockContactButton(Button): def __init__(self, driver): super().__init__(driver, accessibility_id="Unblock-item-button") def click(self): self.scroll_to_element() self.wait_for_element().click() class OpenInStatusButton(Button): def __init__(self, driver): super().__init__(driver, translation_id="browsing-open-in-status") def navigate(self): from views.web_views.base_web_view import BaseWebView return BaseWebView(self.driver) class ViewProfileButton(Button): def __init__(self, driver): super().__init__(driver, translation_id="view-profile") def navigate(self): return ChatView(self.driver) class ChatOptionsButton(Button): def __init__(self, driver): super().__init__(driver, accessibility_id="chat-menu-button") def click(self): self.click_until_presence_of_element(HomeView(self.driver).mark_all_messages_as_read_button) def navigate(self): return ChatView(self.driver) class ProfileSendMessageButton(Button): def __init__(self, driver): super().__init__(driver, accessibility_id="Chat-item-button") def navigate(self): return ChatView(self.driver) class ProfileBlockContactButton(Button): def __init__(self, driver): super(ProfileBlockContactButton, self).__init__(driver, accessibility_id="Block-item-button") def click(self): self.scroll_to_element() self.wait_for_element().click() class ChatElementByText(Text): def __init__(self, driver, text): self.message_text = text self.chat_item_locator = "android.view.ViewGroup[@content-desc='chat-item']" if text in ["image", "sticker", "audio"]: self.message_locator = "//android.view.ViewGroup[@content-desc='%s-message']" % text else: self.message_locator = "//*[starts-with(@text,'%s')]" % text super().__init__(driver, prefix=self.message_locator, xpath="/ancestor::%s" % self.chat_item_locator) def find_element(self): for _ in range(2): try: return super(ChatElementByText, self).find_element() except NoSuchElementException: self.wait_for_visibility_of_element(20) @property def image_in_reply(self): class ImageInReply(BaseElement): def __init__(self, driver, parent_locator: str): super().__init__(driver, prefix=parent_locator, xpath="//android.widget.ImageView") try: return ImageInReply(self.driver, self.locator) except NoSuchElementException: return '' @property def timestamp_command_message(self): class TimeStampText(Button): def __init__(self, driver, parent_locator: str): super().__init__(driver, xpath="(%s//android.widget.TextView)[last()]" % parent_locator) return TimeStampText(self.driver, self.locator) @property def timestamp(self): class TimeStampText(Button): def __init__(self, driver, parent_locator: str): super().__init__(driver, xpath="%s//*[@content-desc='message-timestamp']" % parent_locator) return TimeStampText(self.driver, self.locator).text @property def member_photo(self): class MemberPhoto(Button): def __init__(self, driver, parent_locator: str): super().__init__(driver, prefix=parent_locator, xpath="//*[@content-desc='user-avatar']") return MemberPhoto(self.driver, self.locator) @property def username(self): class Username(Text): def __init__(self, driver, parent_locator: str): super().__init__(driver, prefix=parent_locator, xpath="/android.view.ViewGroup/android.widget.TextView[1]") return Username(self.driver, self.locator) @property def message_body(self): return Text( self.driver, xpath="//%s//android.widget.TextView[contains(@text,'%s')]" % (self.chat_item_locator, self.message_text) ) @property def message_body_with_mention(self): return Text(self.driver, xpath=self.message_body.locator + "/../following-sibling::android.widget.TextView") def click_on_link_inside_message_body(self): self.message_body.wait_for_visibility_of_element(30) self.message_body.click_inside_element_by_coordinate(rel_x=0.1, rel_y=0.9) def wait_for_sent_state(self, wait_time=30): return BaseElement(self.driver, prefix=self.locator, xpath="//*[@content-desc='message-sent']").is_element_displayed(wait_time) @property def uncollapse(self) -> bool: class Collapse(Button): def __init__(self, driver, parent_locator: str): super().__init__(driver, prefix=parent_locator, xpath="/../../..//android.widget.ImageView[@content-desc='icon']") return Collapse(self.driver, self.locator).is_element_displayed() @property def status(self) -> str: Text(self.driver, xpath=self.locator).click() status_element = Text(self.driver, prefix=self.locator, xpath="//*[@content-desc='message-status']/android.widget.TextView") status = '' i = 1 while i < 5: i += 1 if Text(self.driver, prefix=self.locator, xpath="//*[@content-desc='message-sending']").is_element_displayed(2): status = "Sending" break else: Text(self.driver, xpath=self.locator).click() try: if status_element.is_element_displayed(2): status = status_element.text break except StaleElementReferenceException: pass time.sleep(2) return status def wait_for_status_to_be(self, expected_status: str, timeout: int = 30): self.driver.info("Waiting for message to be sent for %s sec" % timeout) start_time = time.time() current_status = 'not set' while time.time() - start_time <= timeout: current_status = self.status if current_status == expected_status: return time.sleep(1) raise TimeoutException("Message status was not changed to %s, it's %s" % (expected_status, current_status)) @property def sent_status_checkmark(self) -> object: return Text(self.driver, prefix=self.locator, xpath="//*[@content-desc='sent']") @property def replied_message_text(self): class RepliedMessageText(Text): def __init__(self, driver, parent_locator: str): super().__init__(driver, prefix=parent_locator, xpath="/preceding::android.widget.TextView[@content-desc='quoted-message']") try: return RepliedMessageText(self.driver, self.message_locator).text except NoSuchElementException: return '' @property def replied_to_username_text(self): class RepliedToUsernameText(Text): def __init__(self, driver, parent_locator: str): super().__init__(driver, prefix=parent_locator, xpath="/preceding-sibling::*[1]/android.widget.TextView[1]") try: return RepliedToUsernameText(self.driver, self.message_locator).text except NoSuchElementException: return '' def emojis_below_message(self, emoji: str = 'thumbs-up'): class EmojisNumber(Text): def __init__(self, driver, parent_locator: str): self.emoji = emoji self.emojis_id = 'emoji-reaction-%s' % str(emojis[self.emoji]) super().__init__(driver, prefix=parent_locator, xpath="/../..//*[@content-desc='%s']/android.widget.TextView[2]" % self.emojis_id) @property def text(self): try: text = self.find_element().text self.driver.info("%s is '%s' for '%s'" % (self.name, text, self.emoji)) return int(text.strip()) except NoSuchElementException: return 0 return EmojisNumber(self.driver, self.locator) @property def image_in_message(self): try: self.driver.info("Trying to access image inside message with text '%s'" % self.message_text) ChatElementByText(self.driver, self.message_text).wait_for_sent_state(60) return Button(self.driver, xpath="%s//android.widget.ImageView[@content-desc='image-message']" % self.locator) except NoSuchElementException: self.driver.fail("No image is found in message!") class ImageContainer(Button): def __init__(self, driver, parent_locator): super().__init__(driver, xpath='%s//*[@content-desc="image-container"]' % parent_locator) def image_by_index(self, index: int): return BaseElement(self.driver, xpath="(%s//android.widget.ImageView)[%s]" % (self.locator, index)) @property def image_container_in_message(self): try: self.driver.info( "Trying to access images (image container) inside message with text '%s'" % self.message_text) ChatElementByText(self.driver, self.message_text).wait_for_sent_state(60) return self.ImageContainer(self.driver, self.locator) except NoSuchElementException: self.driver.fail("No image container is found in message!") @property def pinned_by_label(self): class PinnedByLabelText(Text): def __init__(self, driver, parent_locator: str): super().__init__(driver, prefix=parent_locator, xpath="/../..//android.view.ViewGroup[@content-desc='pinned-by']") return PinnedByLabelText(self.driver, self.locator) @property def message_text_content(self): return Text( self.driver, xpath="//%s//*[@content-desc='message-text-content']/android.widget.TextView" % self.chat_item_locator ) class UsernameOptions(Button): def __init__(self, driver, username): super().__init__(driver, xpath="//*[@text='%s']/..//*[@content-desc='menu-option']" % username) def navigate(self): return ChatView(self.driver) def click(self): self.scroll_to_element() self.wait_for_element().click() return self.navigate() class UsernameCheckbox(Button): def __init__(self, driver, username, state_on): self.username = username super().__init__(driver, xpath="//*[@text='%s']/..//*[@content-desc='checkbox-%s']" % ( username, 'on' if state_on else 'off')) def click(self): try: self.scroll_to_element(20).click() except NoSuchElementException: self.scroll_to_element(direction='up', depth=20).click() class GroupChatInfoView(BaseView): def __init__(self, driver): super().__init__(driver) self.add_members = Button(self.driver, translation_id="add-members") def get_username_options(self, username: str): return UsernameOptions(self.driver, username) def user_admin(self, username: str): admin = Button(self.driver, xpath="//*[@text='%s']/..//*[@text='%s']" % ( username, self.get_translation_by_key("group-chat-admin"))) admin.scroll_to_element() return admin def get_user_from_group_info(self, username: str): user = Text(self.driver, xpath="//*[@text='%s']" % username) user.scroll_to_element() return user class CommunityView(HomeView): def __init__(self, driver): super().__init__(driver) #### OLD UI # Main community page (list with channels) self.add_channel_button = HomeView(self.driver).plus_button self.community_create_a_channel_button = Button(self.driver, accessibility_id="community-create-channel") self.channel_name_edit_box = EditBox(self.driver, translation_id="name-your-channel-placeholder") self.channel_descripton = ChatView(self.driver).community_description_edit_box self.community_options_button = Button(self.driver, accessibility_id="community-menu-button") self.view_members_button = Button(self.driver, accessibility_id="view-members") self.community_info_button = Button(self.driver, translation_id="community-info") self.invite_button = Button(self.driver, accessibility_id="community-invite-people") # Community info page self.community_membership_request_value = Text( self.driver, translation_id="members-label", suffix='/following-sibling::android.view.ViewGroup/android.widget.TextView') self.members_button = Button(self.driver, translation_id="members-label") self.community_info_picture = Button(self.driver, accessibility_id="chat-icon") self.leave_community_button = Button(self.driver, translation_id="leave-community") self.edit_community_button = Button(self.driver, translation_id="edit-community") self.share_community_button = Button(self.driver, accessibility_id="share-community") self.share_community_link_button = Button(self.driver, accessibility_id="share-community-link") # Members self.invite_people_button = Button(self.driver, accessibility_id="community-invite-people") self.membership_requests_button = Button(self.driver, translation_id="membership-requests") self.share_invite_button = Button(self.driver, accessibility_id="share-community-link") # Requesting access to community / joining community self.request_access_button = Button(self.driver, translation_id="request-access") self.membership_request_pending_text = Text(self.driver, translation_id="membership-request-pending") self.join_button = Button(self.driver, accessibility_id="show-request-to-join-screen-button") self.join_community_button = Button(self.driver, accessibility_id="join-community-button") self.slide_to_request_to_join_button = Button( self.driver, xpath="(//*[@resource-id='slide-button-track']//*[@content-desc='icon'])[1]") self.follow_button = Button(self.driver, translation_id="follow") self.community_tags = BaseElement( self.driver, xpath="//*[@content-desc='chat-name-text']/../android.widget.HorizontalScrollView") #### NEW UI # Communities initial page self.close_community_view_button = Button( self.driver, xpath="//*[@content-desc='community-options-for-community']/../*[1]//android.widget.ImageView") self.community_title = Text(self.driver, accessibility_id="community-title") self.community_description_text = Text(self.driver, accessibility_id="community-description-text") self.community_status_joined = Text(self.driver, accessibility_id="status-tag-positive") self.community_status_pending = Text(self.driver, accessibility_id="status-tag-pending") def join_community(self, password=common_password, open_community=True): self.driver.info("Joining community") ChatView(self.driver).chat_element_by_text("https://status.app/c/").click_on_link_inside_message_body() self.join_button.wait_and_click(120) self.slide_to_request_to_join_button.swipe_right_on_element(width_percentage=16) self.password_input.send_keys(password) Button(self.driver, xpath="//*[@content-desc='password-input']/../following-sibling::*//*[@text='Join Community']").click() if open_community: self.community_status_joined.wait_for_visibility_of_element(60) def get_channel(self, channel_name: str): self.driver.info("Getting %s channel element in community" % channel_name) chat_element = self.get_chat(username=channel_name, community_channel=True, wait_time=30) return chat_element def add_channel(self, name: str, description="Some new channel"): self.driver.info("Adding channel in community") self.plus_button.click() self.community_create_a_channel_button.wait_and_click() self.channel_name_edit_box.send_keys(name) self.channel_descripton.send_keys(description) chat_view = ChatView(self.driver) chat_view.confirm_create_in_community_button.click() self.get_chat(name).click() return chat_view def leave_community(self, community_name: str): self.driver.info("Leaving %s" % community_name) home = self.get_home_view() home.communities_tab.click() community_element = home.get_chat(community_name, community=True) community_element.long_press_until_element_is_shown(self.leave_community_button) self.leave_community_button.click() self.leave_community_button.click() def get_channel_avatar(self, channel_name='general'): return Button(self.driver, xpath='//*[@text="# %s"]/../*[@content-desc="channel-avatar"]' % channel_name) def copy_community_link(self): self.driver.info("Copy community link") self.community_options_button.click() self.community_info_button.click() self.element_starts_with_text('join.status.im/c/').click() community_link_text = self.element_starts_with_text('join.status.im/c/').text self.home_button.double_click() return 'https://%s' % community_link_text def handle_membership_request(self, username: str, approve=True): self.driver.info("Handling membership request of user '%s', approve='%s'" % (username, str(approve))) self.members_button.click() self.membership_requests_button.click() approve_suffix, decline_suffix = '/following-sibling::android.view.ViewGroup[1]', '/following-sibling::android.view.ViewGroup[2]' if approve: Button(self.driver, xpath="//*[starts-with(@text,'%s')]%s" % (username, approve_suffix)).click() else: Button(self.driver, xpath="//*[starts-with(@text,'%s')]%s" % (username, decline_suffix)).click() self.close_button.click() # Should not be used anymore, outdated flow def send_invite_to_community(self, community_name, user_names_to_invite): if isinstance(user_names_to_invite, str): user_names_to_invite = [user_names_to_invite] self.driver.info("Send %s invite to community" % ', '.join(map(str, user_names_to_invite))) self.jump_to_communities_home() home = self.get_home_view() community_element = home.get_chat(community_name, community=True) community_element.long_press_until_element_is_shown(self.view_members_button) self.view_members_button.click_until_presence_of_element(self.invite_button) self.invite_button.click() for user_name in user_names_to_invite: user_contact = self.element_by_text_part(user_name) user_contact.scroll_and_click() self.share_invite_button.click_until_presence_of_element(self.invite_button) self.back_button.click_until_presence_of_element(self.plus_button) def share_community(self, community_name, user_names_to_share): if isinstance(user_names_to_share, str): user_names_to_share = [user_names_to_share] self.driver.info("Share to %s community" % ', '.join(map(str, user_names_to_share))) self.jump_to_communities_home() home = self.get_home_view() home.communities_tab.click() community_element = home.get_chat(community_name, community=True) community_element.long_press_until_element_is_shown(self.share_community_button) self.share_community_button.click() for user_name in user_names_to_share: user_contact = self.element_by_text_part(user_name) user_contact.scroll_and_click() self.share_community_link_button.click() class PreviewMessage(ChatElementByText): def __init__(self, driver, text: str): super().__init__(driver, text=text) @staticmethod def return_element_or_empty(obj): try: return obj.scroll_to_element() except NoSuchElementException: return '' @property def preview_image(self): class PreviewImage(SilentButton): def __init__(self, driver, parent_locator: str): super().__init__(driver, prefix=parent_locator, xpath="//*[@content-desc='thumbnail']") return PreviewMessage.return_element_or_empty(PreviewImage(self.driver, self.locator)) @property def preview_title(self): class PreviewTitle(SilentButton): def __init__(self, driver, parent_locator: str): super().__init__(driver, prefix=parent_locator, xpath="//*[@content-desc='title']") return PreviewMessage.return_element_or_empty(PreviewTitle(self.driver, self.locator)) @property def preview_subtitle(self): class PreviewSubTitle(SilentButton): def __init__(self, driver, parent_locator: str): super().__init__(driver, prefix=parent_locator, xpath="//*[@content-desc='description']") return PreviewMessage.return_element_or_empty(PreviewSubTitle(self.driver, self.locator)) @property def preview_link(self): class PreviewLink(SilentButton): def __init__(self, driver, parent_locator: str): super().__init__(driver, prefix=parent_locator, xpath="//*[@content-desc='link']") return PreviewMessage.return_element_or_empty(PreviewLink(self.driver, self.locator)) class CommunityLinkPreviewMessage(ChatElementByText): def __init__(self, driver, text: str): super().__init__(driver, text=text) self.locator += "//*[@text='%s']" % self.get_translation_by_key('community') @property def community_name(self) -> str: class CommunityName(SilentButton): def __init__(self, driver, parent_locator: str): super().__init__(driver, prefix=parent_locator, xpath="/following-sibling::android.widget.TextView[1]") return CommunityName(self.driver, self.locator).text @property def community_description(self) -> str: class CommunityDescription(SilentButton): def __init__(self, driver, parent_locator: str): super().__init__(driver, prefix=parent_locator, xpath="/following-sibling::android.widget.TextView[2]") return CommunityDescription(self.driver, self.locator).text @property def community_members_amount(self) -> int: class CommunityMembers(SilentButton): def __init__(self, driver, parent_locator: str): super().__init__(driver, prefix=parent_locator, xpath="/following-sibling::android.widget.TextView[3]") members_string = CommunityMembers(self.driver, self.locator).text return int(re.search(r'\d+', members_string).group()) def view(self) -> object: class CommunityViewButton(SilentButton): def __init__(self, driver, parent_locator: str): super().__init__(driver, prefix=parent_locator, xpath="/..//*[@text='%s']" % self.get_translation_by_key("view")) CommunityViewButton(self.driver, self.locator).click() CommunityView(self.driver).request_access_button.wait_for_element(20) return CommunityView(self.driver) class TransactionMessage(ChatElementByText): def __init__(self, driver, text: str, transaction_value): super().__init__(driver, text=text) if transaction_value: self.xpath = "//*[starts-with(@text,'%s')]/../*[@text='%s']/ancestor::android.view.ViewGroup[@content-desc='chat-item']" % ( text, transaction_value) # Common statuses for incoming and outgoing transactions self.address_requested = self.get_translation_by_key("address-requested") self.confirmed = self.get_translation_by_key("status-confirmed") self.pending = self.get_translation_by_key("pending") self.declined = self.get_translation_by_key("transaction-declined") @property def transaction_status(self): class TransactionStatus(SilentButton): def __init__(self, driver, parent_locator: str): super().__init__(driver, prefix=parent_locator, xpath="/*[1]/*[1]/*[5]/android.widget.TextView") return TransactionStatus(self.driver, self.locator) @property def decline_transaction(self): class DeclineTransaction(Button): def __init__(self, driver, parent_locator): super().__init__(driver, prefix=parent_locator, translation_id="decline") def click(self): self.wait_for_element().click() return self.navigate() return DeclineTransaction(self.driver, self.locator) class OutgoingTransaction(TransactionMessage): def __init__(self, driver, account_name: str, transaction_value): super().__init__(driver, text="↑ Outgoing transaction", transaction_value=transaction_value) self.account_name = account_name self.address_request_accepted = self.get_translation_by_key("address-request-accepted") self.address_received = self.get_translation_by_key("address-received") @property def sign_and_send(self): class SignAndSend(Button): def __init__(self, driver, parent_locator): super().__init__(driver, prefix=parent_locator, translation_id="sign-and-send") def navigate(self): from views.send_transaction_view import SendTransactionView return SendTransactionView(self.driver) def click(self): self.wait_for_element().click() return self.navigate() return SignAndSend(self.driver, self.locator) class IncomingTransaction(TransactionMessage): def __init__(self, driver, account_name: str, transaction_value): super().__init__(driver, text="↓ Incoming transaction", transaction_value=transaction_value) self.account_name = account_name self.shared_account = "Shared '%s'" % account_name @property def accept_and_share_address(self): class AcceptAndShareAddress(Button): def __init__(self, driver, parent_locator): super().__init__(driver, prefix=parent_locator, translation_id="accept-and-share-address") def navigate(self): from views.send_transaction_view import SendTransactionView return SendTransactionView(self.driver) def click(self): self.wait_for_element().click() return self.navigate() return AcceptAndShareAddress(self.driver, self.locator) class PinnedMessagesOnProfileButton(Button): def __init__(self, driver): super().__init__(driver, xpath="//*[@content-desc='pinned-messages-item']") @property def count(self): class PinnedMessageCounter(Text): def __init__(self, driver, parent_locator: str): super().__init__(driver, prefix=parent_locator, xpath="/android.widget.TextView[2]") return PinnedMessageCounter(self.driver, self.locator).text class UnpinMessagePopUp(BaseElement): def __init__(self, driver): # self.message_text = message_text super().__init__(driver, translation_id="pin-limit-reached", suffix='/..') def click_unpin_message_button(self): class UnpinMessageButton(Button): def __init__(self, driver, parent_locator: str): super().__init__(driver, prefix=parent_locator, xpath="//android.widget.TextView[starts-with(@text,'Unpin')]") return UnpinMessageButton(self.driver, self.locator).click() def message_text(self, text): element = Text(self.driver, prefix=self.locator, xpath="//android.widget.TextView[contains(@text,'%s')]" % text) return element class PinnedMessagesList(BaseElement): def __init__(self, driver): super().__init__(driver, xpath="//*[@content-desc='pinned-messages-menu']") def get_pinned_messages_number(self): self.driver.info("Getting number of pinned messages inside pinned messages list element") element = BaseElement(self.driver, prefix=self.locator, xpath="//*[@content-desc='message-sent']") return len(element.find_elements()) def message_element_by_text(self, text): message_element = Button(self.driver, prefix=self.locator, xpath="//*[starts-with(@text,'%s')]" % text) self.driver.info("Looking for a pinned message by text: %s" % message_element.exclude_emoji(text)) return message_element def get_message_pinned_by_text(self, text): xpath = "//*[starts-with(@text,'%s')]/../../*[@content-desc='pinned-by']/android.widget.TextView" % text pinned_by_element = Text(self.driver, prefix=self.locator, xpath=xpath) self.driver.info("Looking for a pinned by message with text: %s" % text) return pinned_by_element class ChatMessageInput(EditBox): def __init__(self, driver): super().__init__(driver, accessibility_id="chat-message-input") def paste_text_from_clipboard(self): action = TouchAction(self.driver) location = self.find_element().location x, y = location['x'], location['y'] action.long_press(x=x + 250, y=y).release().perform() # long press action.tap(x=x + 50, y=y - 50).release().perform() # tap Paste def click_inside(self): action = TouchAction(self.driver) location = self.find_element().location x, y = location['x'], location['y'] action.tap(x=x + 250, y=y).release().perform() class ChatView(BaseView): def __init__(self, driver): super().__init__(driver) # Start new chat self.public_key_edit_box = EditBox(self.driver, accessibility_id="enter-contact-code-input") self.scan_contact_code_button = Button(self.driver, accessibility_id="scan-contact-code-button") self.view_profile_new_contact_button = Button(self.driver, accessibility_id="new-contact-button") # Chat header self.add_to_contacts = Button(self.driver, accessibility_id="add-to-contacts-button") ## Options self.chat_options = ChatOptionsButton(self.driver) self.delete_chat_button = Button(self.driver, translation_id="close-chat") self.clear_history_button = Button(self.driver, translation_id="clear-history") self.reply_message_button = Button(self.driver, translation_id="message-reply") self.share_chat_button = Button(self.driver, accessibility_id="share-chat-button") self.clear_button = Button(self.driver, translation_id="clear-history") self.view_profile_button = ViewProfileButton(self.driver) self.view_profile_by_avatar_button = Button(self.driver, accessibility_id="member-photo") self.user_options = Button(self.driver, accessibility_id="options") self.open_in_status_button = OpenInStatusButton(self.driver) self.close_modal_view_from_chat_button = Button(self.driver, xpath="//androidx.appcompat.widget.LinearLayoutCompat") # Chat input self.chat_message_input = ChatMessageInput(self.driver) self.cancel_reply_button = Button(self.driver, accessibility_id="reply-cancel-button") self.url_preview_composer = Button(self.driver, accessibility_id="url-preview") self.url_preview_composer_text = Text(self.driver, xpath='//*[@content-desc="url-preview"]//*[@content-desc="title"]') self.quote_username_in_message_input = EditBox( self.driver, xpath="//*[@content-desc='reply-cancel-button']/preceding::android.widget.TextView[3]") self.chat_item = Button(self.driver, xpath="(//*[@content-desc='chat-item'])[1]") self.chat_name_editbox = EditBox(self.driver, accessibility_id="chat-name-input") self.commands_button = CommandsButton(self.driver) self.send_command = SendCommand(self.driver) self.request_command = RequestCommand(self.driver) # General chat view self.history_start_icon = Button(self.driver, accessibility_id="history-chat") self.unpin_message_popup = UnpinMessagePopUp(self.driver) self.contact_request_button = Button(self.driver, accessibility_id="contact-request--button") # Stickers self.show_stickers_button = Button(self.driver, accessibility_id="show-stickers-icon") self.get_stickers = Button(self.driver, translation_id="get-stickers") self.sticker_icon = Button(self.driver, accessibility_id="sticker-icon") self.sticker_message = Button(self.driver, accessibility_id="sticker-message") # Images self.show_images_button = Button(self.driver, accessibility_id="open-images-button") self.take_photo_button = Button(self.driver, accessibility_id="camera-button") self.snap_button = Button(self.driver, accessibility_id="snap") self.image_from_gallery_button = Button(self.driver, accessibility_id="open-gallery") self.images_confirm_selection_button = Button(self.driver, accessibility_id="confirm-selection") self.images_area_in_gallery = Button(self.driver, xpath="//*[@content-desc='open-gallery']/following-sibling::android.view.ViewGroup[1]") self.image_message_in_chat = Button(self.driver, accessibility_id="image-message") self.save_image_button = Button(self.driver, translation_id="save") self.recent_image_in_gallery = Button(self.driver, xpath="//*[contains(@resource-id,'thumbnail')]") self.cancel_send_image_button = Button(self.driver, accessibility_id="cancel-send-image") self.share_image_icon_button = Button(self.driver, accessibility_id="share-image") self.view_image_options_button = Button(self.driver, accessibility_id="image-options") self.save_image_icon_button = Button(self.driver, accessibility_id="save-image") self.image_in_android_messenger = Button(self.driver, accessibility_id="Image") # Audio self.audio_message_in_chat = Button(self.driver, accessibility_id="audio-message") self.audio_message_button = Button(self.driver, accessibility_id="show-audio-message-icon") self.record_audio_button = Button(self.driver, accessibility_id="start-stop-audio-recording-button") self.cancel_audio_message_button = Button(self.driver, accessibility_id="cancel-message-button") self.send_audio_message_button = Button(self.driver, accessibility_id="send-message-button") self.play_pause_audio_message_button = Button(self.driver, accessibility_id="play-pause-audio-message-button") self.audio_message_in_chat_timer = Text(self.driver, xpath="//*[@content-desc='play-pause-audio-message-button']/../..//android.widget.TextView[1]") self.audio_message_recorded_time = Text(self.driver, accessibility_id="audio-message-recorded-time") # Group chats self.group_info = GroupInfoButton(self.driver) self.leave_chat_button = Button(self.driver, accessibility_id="leave-chat-button") self.leave_button = Button(self.driver, translation_id="leave", uppercase=True) self.join_chat_button = Button(self.driver, accessibility_id="join-chat-button") self.decline_invitation_button = Button(self.driver, translation_id="group-chat-decline-invitation") self.remove_user_button = Button(self.driver, accessibility_id="remove-from-chat") self.make_admin_button = Button(self.driver, accessibility_id="make-admin") self.edit_group_chat_name_button = Button(self.driver, accessibility_id="edit-button") self.edit_group_chat_name_edit_box = EditBox(self.driver, accessibility_id="new-chat-name") self.done_button = Button(self.driver, accessibility_id="done") self.create_button = Button(self.driver, accessibility_id="create-group-chat-button") ## Group invites self.group_invite_button = Button(self.driver, accessibility_id="invite-chat-button") self.group_invite_link_text = Text(self.driver, xpath="//*[@content-desc='invitation-link']/android.widget.TextView") self.introduce_yourself_edit_box = EditBox(self.driver, accessibility_id="introduce-yourself-input") self.request_membership_button = Button(self.driver, translation_id="request-membership") self.group_membership_request_button = Button(self.driver, accessibility_id="invitation-requests-button") self.accept_group_invitation_button = Button(self.driver, accessibility_id="accept-invitation-button") self.decline_group_invitation_button = Button(self.driver, accessibility_id="decline-invitation-button") self.retry_group_invite_button = Button(self.driver, accessibility_id="retry-button") self.remove_group_invite_button = Button(self.driver, accessibility_id="remove-group-button") # Contact's profile self.contact_profile_picture = ProfilePictureElement(self.driver) self.profile_send_message_button = ProfileSendMessageButton(self.driver) self.profile_block_contact_button = ProfileBlockContactButton(self.driver) self.confirm_block_contact_button = Button(self.driver, accessibility_id="block-contact-confirm") self.unblock_contact_button = UnblockContactButton(self.driver) self.profile_mute_contact = Button(self.driver, accessibility_id="Mute-item-button") self.profile_unmute_contact = Button(self.driver, accessibility_id="Unmute-item-button") self.profile_add_to_contacts_button = Button(self.driver, accessibility_id="Add to contacts-item-button") self.profile_remove_from_contacts = Button(self.driver, accessibility_id="Remove from contacts-item-button") self.profile_details = Button(self.driver, accessibility_id="share-button") self.profile_nickname = Text(self.driver, xpath="//*[@content-desc='profile-nickname-item']/android.widget.TextView[2]") self.profile_nickname_button = Button(self.driver, accessibility_id="profile-nickname-item") self.pinned_messages_button = PinnedMessagesOnProfileButton(self.driver) self.nickname_input_field = EditBox(self.driver, accessibility_id="nickname-input") self.remove_from_contacts = Button(self.driver, accessibility_id="Remove from contacts-item-button") # Communities self.create_community_button = Button(self.driver, translation_id="create-community") self.community_name_edit_box = EditBox(self.driver, translation_id="name-your-community-placeholder") self.community_description_edit_box = EditBox(self.driver, xpath='//android.widget.EditText[@text="%s"]' % self.get_translation_by_key( "give-a-short-description-community")) self.set_community_image_button = Button(self.driver, translation_id='community-thumbnail-image', suffix='/following-sibling::android.view.ViewGroup') self.confirm_create_in_community_button = Button(self.driver, translation_id="create") self.mentions_list = BaseElement(self.driver, accessibility_id="mentions-list") # New UI self.pinned_messages_count = Button(self.driver, xpath="//*[@content-desc='pins-count']//android.widget.TextView") self.pinned_messages_list = PinnedMessagesList(self.driver) self.pin_limit_popover = BaseElement(self.driver, translation_id="pin-limit-reached") self.view_pinned_messages_button = Button(self.driver, accessibility_id="pinned-banner") def get_outgoing_transaction(self, account=None, transaction_value=None) -> object: if account is None: account = self.status_account_name return OutgoingTransaction(self.driver, account, transaction_value) def get_incoming_transaction(self, account=None, transaction_value=None) -> object: if account is None: account = self.status_account_name return IncomingTransaction(self.driver, account, transaction_value) def get_preview_message_by_text(self, text=None) -> object: self.driver.info('Getting preview message for link: %s' % text) return PreviewMessage(self.driver, text) def get_community_link_preview_by_text(self, text=None) -> object: self.driver.info('Getting community preview message for link: %s' % text) return CommunityLinkPreviewMessage(self.driver, text) def delete_chat(self): self.driver.info("Delete chat via options") self.chat_options.click() self.delete_chat_button.click() self.delete_button.click() def leave_chat(self): self.driver.info("Leave chat via options") self.chat_options.click() self.leave_chat_button.click() self.leave_button.click() def clear_history(self): self.driver.info("Clear chat history via options") self.chat_options.click() self.clear_history_button.click() self.clear_button.click() def leave_chat_via_group_info(self): self.driver.info("Leave group chat via group info") self.chat_options.click() self.group_info.click() self.leave_chat_button.click() self.leave_button.click() def rename_chat_via_group_info(self, new_chat_name): self.driver.info("Rename group chat to: %s" % new_chat_name) self.chat_options.click() self.group_info.click() self.edit_group_chat_name_button.click() self.edit_group_chat_name_edit_box.send_keys(new_chat_name) self.done_button.click() def get_group_invite_via_group_info(self): self.driver.info("Copy group invite link") self.chat_options.click() self.group_info.click() self.group_invite_button.click() return self.group_invite_link_text.text def request_membership_for_group_chat(self, intro_message): self.driver.info("Requesting membership to group chat") self.introduce_yourself_edit_box.send_keys(intro_message) self.request_membership_button.click_until_presence_of_element(self.element_by_text('Request pending…')) def get_username_checkbox(self, username: str, state_on=False): self.driver.info("Getting %s checkbox" % username) return UsernameCheckbox(self.driver, username, state_on) def accept_membership_for_group_chat_via_chat_view(self, username, accept=True): info = "%s membership to group chat" % username self.driver.info("Accept %s" % info) if accept else self.driver.info("Decline %s" % info) self.group_membership_request_button.click() self.element_by_text(username).click() self.accept_group_invitation_button.click() if accept else self.decline_group_invitation_button.click() def add_members_to_group_chat(self, user_names_to_add: list): self.driver.info("Add %s to group chat" % ', '.join(map(str, user_names_to_add))) self.chat_options.click() group_info_view = self.group_info.click() group_info_view.add_members.click() for user_name in user_names_to_add: user_contact = self.get_username_checkbox(user_name) user_contact.scroll_to_element() user_contact.click() self.add_button.click() def get_user_options(self, username: str): self.driver.info("Get user options for: '%s'" % username) self.chat_options.click() group_info_view = self.group_info.click() group_info_view.get_username_options(username).click() return self def chat_element_by_text(self, text): chat_element = ChatElementByText(self.driver, text) self.driver.info("Looking for a message by text: %s" % chat_element.exclude_emoji(text)) return chat_element def verify_message_is_under_today_text(self, text, errors, timeout=10): self.driver.info("Verifying that '%s' is under today" % text) message_element = self.chat_element_by_text(text) message_element.wait_for_visibility_of_element(timeout) message_location = message_element.find_element().location['y'] today_text_element = self.element_by_text('Today').find_element() today_location = today_text_element.location['y'] today_height = today_text_element.size['height'] if message_location < today_location + today_height: errors.append("Message '%s' is not under 'Today' text" % text) def send_message(self, message: str = 'test message', wait_chat_input_sec=5): self.driver.info("Sending message '%s'" % BaseElement(self.driver).exclude_emoji(message)) self.chat_message_input.wait_for_element(wait_chat_input_sec) for _ in range(3): try: self.chat_message_input.send_keys(message) break except (StaleElementReferenceException, InvalidElementStateException): time.sleep(1) except Exception as e: raise e else: raise StaleElementReferenceException(msg="Can't send keys to chat message input, loading") try: self.send_message_button.click() except NoSuchElementException: self.chat_message_input.clear() self.chat_message_input.send_keys(message) self.send_message_button.click() def send_contact_request(self, message: str = 'Contact request message', wait_chat_input_sec=5): self.driver.info("Sending contact request message '%s'" % BaseElement(self.driver).exclude_emoji(message)) self.contact_request_button.wait_and_click() self.chat_message_input.wait_for_element(wait_chat_input_sec) self.chat_message_input.send_keys(message) self.send_contact_request_button.click() def pin_message(self, message, action="pin"): self.driver.info("Looking for message '%s' pin" % message) element = self.element_by_translation_id(action) self.chat_element_by_text(message).long_press_until_element_is_shown(element) element.click_until_absense_of_element(element) def edit_message_in_chat(self, message_to_edit, message_to_update): self.driver.info("Looking for message '%s' to edit it" % message_to_edit) element = self.element_by_translation_id("edit-message") self.chat_element_by_text(message_to_edit).message_body.long_press_until_element_is_shown(element) element.click() self.chat_message_input.clear() self.chat_message_input.send_keys(message_to_update) self.send_message_button.click() def delete_message_in_chat(self, message, everyone=True): self.driver.info("Looking for message '%s' to delete it" % message) if everyone: delete_button = self.element_by_translation_id("delete-for-everyone") else: delete_button = self.element_by_translation_id("delete-for-me") # self.chat_element_by_text(message).message_body.long_press_until_element_is_shown(delete_button) self.chat_element_by_text(message).message_body.long_press_element() delete_button.click() def copy_message_text(self, message_text): self.driver.info("Copying '%s' message via long press" % message_text) self.chat_element_by_text(message_text).wait_for_visibility_of_element() self.chat_element_by_text(message_text).long_press_element() self.element_by_translation_id("copy-text").click() def quote_message(self, message: str): self.driver.info("Quoting '%s' message" % message) self.chat_view_element_starts_with_text(message).long_press_until_element_is_shown(self.reply_message_button) self.reply_message_button.click() def set_reaction(self, message: str, emoji: str = 'thumbs-up', emoji_message=False): self.driver.info("Setting '%s' reaction" % emoji) # Audio message is obvious should be tapped not on audio-scroll-line # so we tap on its below element as exception here (not the case for link/tag message!) element = Button(self.driver, accessibility_id='reaction-%s' % emoji) if message == 'audio': self.audio_message_in_chat_timer.long_press_element() else: if not emoji_message: self.chat_element_by_text(message).long_press_until_element_is_shown(element) else: self.element_by_text_part(message).long_press_until_element_is_shown(element) # old UI # element = Button(self.driver, accessibility_id='pick-emoji-%s' % key) element.click() element.wait_for_invisibility_of_element() def add_remove_same_reaction(self, message: str, emoji: str = 'thumbs-up'): self.driver.info("Adding one more '%s' reaction or removing an added one" % emoji) key = emojis[emoji] element = Button(self.driver, accessibility_id='emoji-reaction-%s' % key) element.wait_and_click() def view_profile_long_press(self, message=str): self.chat_element_by_text(message).long_press_element() self.view_profile_by_avatar_button.wait_and_click() self.profile_block_contact_button.wait_for_visibility_of_element(5) def wait_ens_name_resolved_in_chat(self, message=str, username_value=str): self.driver.info("Waiting ENS name '%s' is resolved in chat" % username_value) counter = 0 while True: if counter >= 120: self.driver.fail('Username not updated to %s %s' % (60, username_value)) elif not (self.chat_element_by_text(message).username.text == username_value): counter += 5 time.sleep(5) else: return def move_to_messages_by_time_marker(self, marker='Today'): self.driver.info("Moving to messages by time marker: '%s'" % marker) Button(self.driver, xpath="//*[@text='%s'']" % marker).scroll_to_element(depth=50, direction='up') def install_sticker_pack_by_name(self, pack_name='HCPP20'): self.driver.info("## Installing '%s' stickerpack" % pack_name, device=False) self.chat_message_input.click() self.show_stickers_button.click() self.get_stickers.click() element = Button( self.driver, xpath="//*[@content-desc='sticker-pack-name'][@text='%s']/..//*[@content-desc='sticker-pack-price']" % pack_name) element.scroll_to_element(depth=21) element.click() element.wait_for_invisibility_of_element() self.navigate_up_button.click() time.sleep(2) self.swipe_left() self.driver.info("## Stickerpack is installed successfully!", device=False) def scroll_to_start_of_history(self, depth=20): self.driver.info('Scrolling th the start of chat history') for _ in range(depth): try: return self.history_start_icon.find_element() except NoSuchElementException: size = self.driver.get_window_size() self.driver.swipe(500, size["height"] * 0.25, 500, size["height"] * 0.8) else: raise Exception('Start of chat history is not reached!') def user_profile_image_in_mentions_list(self, username): return Button(self.driver, xpath="//*[@content-desc='suggestions-list']//*[@text='%s']/" "..//*[@content-desc='member-photo']" % username) def search_user_in_mention_suggestion_list(self, username): return Button(self.driver, xpath="//*[@content-desc='suggestions-list']//*[@text='%s']" % username) def select_mention_from_suggestion_list(self, username_in_list, typed_search_pattern=''): self.driver.info("Selecting '%s' from suggestion list by '%s'" % (username_in_list, typed_search_pattern)) self.chat_message_input.send_keys('@' + typed_search_pattern) self.chat_message_input.click() self.search_user_in_mention_suggestion_list(username_in_list).wait_for_visibility_of_element(10).click() def record_audio_message(self, message_length_in_seconds=5): self.driver.info("Recording audiomessage %ss" % message_length_in_seconds) self.audio_message_button.click() self.allow_button.click() self.record_audio_button.click() sleep(message_length_in_seconds) def play_audio_message(self, listen_time=5): self.driver.info("Playing audiomessage during %ss" % listen_time) self.play_pause_audio_message_button.click() sleep(listen_time) self.play_pause_audio_message_button.click() def block_contact(self): self.driver.info("Block contact from other user profile") self.profile_block_contact_button.click() self.confirm_block_contact_button.click() def open_user_profile_from_public_chat(self, message): chat_element = self.chat_element_by_text(message) chat_element.find_element() chat_element.member_photo.click() def open_user_profile_from_1_1_chat(self): self.chat_options.click() self.view_profile_button.click() def set_nickname(self, nickname, close_profile=True): self.driver.info("Setting nickname:%s" % nickname) self.profile_nickname_button.click() self.nickname_input_field.send_keys(nickname) self.element_by_text('Done').click() if close_profile: self.close_button.click() def convert_device_time_to_chat_timestamp(self) -> list: sent_time_object = dateutil.parser.parse(self.driver.device_time) timestamp = datetime.strptime("%s:%s" % (sent_time_object.hour, sent_time_object.minute), '%H:%M').strftime( "%I:%M %p") timestamp_obj = datetime.strptime(timestamp, '%I:%M %p') possible_timestamps_obj = [timestamp_obj + timedelta(0, 0, 0, 0, 1), timestamp_obj, timestamp_obj - timedelta(0, 0, 0, 0, 1), timestamp_obj - timedelta(0, 0, 0, 0, 2)] timestamps = list(map(lambda x: x.strftime("%I:%M %p"), possible_timestamps_obj)) final_timestamps = [t[1:] if t[0] == '0' else t for t in timestamps] return final_timestamps def get_transaction_message_by_asset(self, transaction_value, incoming=True) -> object: if incoming: transaction_message = self.get_incoming_transaction(account=None, transaction_value=transaction_value) else: transaction_message = self.get_outgoing_transaction(account=None, transaction_value=transaction_value) return transaction_message def get_community_by_name(self, community_name: str): community_button = Button( self.driver, xpath="//*[@content-desc='community-name-text'][starts-with(@text,'%s')]/.." % community_name ) community_button.click() return CommunityView(self.driver) def user_list_element_by_name(self, user_name: str): return BaseElement(self.driver, xpath="//*[@content-desc='user-list']//*[@text='%s']" % user_name) def mention_user(self, user_name: str): self.driver.info("Mention user %s in the chat" % user_name) # gboard = self.driver.available_ime_engines[0] # self.driver.activate_ime_engine(gboard) # workaround to get mentions list expanded self.chat_message_input.click_inside() self.chat_message_input.send_keys("@") try: self.mentions_list.wait_for_element() self.user_list_element_by_name(user_name).wait_for_rendering_ended_and_click() except TimeoutException: self.driver.fail("Mentions list is not shown") def get_image_by_index(self, index=0): return Button(self.driver, accessibility_id="image-%s" % index) def send_images_with_description(self, description, indexes=None): if indexes is None: indexes = [0] self.show_images_button.click() self.allow_button.click_if_shown() self.allow_all_button.click_if_shown() confirm_button = self.images_confirm_selection_button for i in indexes: # ToDo: should be changed to just 1 click when https://github.com/status-im/status-mobile/issues/18872 when is fixed self.get_image_by_index(i).click() if not confirm_button.is_element_displayed(sec=3): self.get_image_by_index(i).click() confirm_button.click() self.chat_message_input.send_keys(description) self.send_message_button.click() def send_image_with_camera(self, description=None): self.take_photo_button.click() self.allow_button.click_if_shown() self.snap_button.click() self.element_by_translation_id("use-photo").click() if description: self.chat_message_input.send_keys(description) self.send_message_button.click() @staticmethod def get_resolved_chat_key(username, chat_key): return '%s • %s…%s' % (username, chat_key[:6], chat_key[-4:]) # Group chat system messages @staticmethod def leave_system_message(username): return "%s left the group" % username @staticmethod def has_been_made_admin_system_message(admin, new_admin): return "%s has made %s admin" % (admin, new_admin) @staticmethod def create_system_message(admin, chat_name): return '%s created the group %s' % (admin, chat_name) @staticmethod def invite_system_message(admin, invited_user): return '%s has invited %s' % (admin, invited_user) @staticmethod def has_added_system_message(admin, invited_user): return '%s has added %s' % (admin, invited_user) @staticmethod def invited_to_join_system_message(username, chat_name): return '%s invited you to join the group %s' % (username, chat_name) @staticmethod def join_system_message(username): return '%s joined the group' % username @staticmethod def changed_group_name_system_message(admin, chat_name): return "%s changed the group's name to %s" % (admin, chat_name) ### Push notifications @staticmethod def pn_invited_to_group_chat(admin, chat_name): return '%s invited you to %s' % (admin, chat_name) @staticmethod def pn_wants_you_to_join_to_group_chat(admin, chat_name): return '%s wants you to join group %s' % (admin, chat_name) def authors_for_reaction(self, emoji: str): return Button(self.driver, accessibility_id='authors-for-reaction-%s' % emojis[emoji]) def chat_view_element_starts_with_text(self, text: str): return BaseElement(self.driver, xpath="//*[@content-desc=':chat-floating-screen']//*[starts-with(@text,'%s')]" % text)