This will remove imports from __future__ that are no longer needed. https://docs.python.org/3.5/library/2to3.html#2to3fixer-future
743 lines
29 KiB
Python
743 lines
29 KiB
Python
"""
|
|
LMS discussion page
|
|
"""
|
|
|
|
|
|
from contextlib import contextmanager
|
|
|
|
from bok_choy.javascript import wait_for_js
|
|
from bok_choy.page_object import PageObject
|
|
from bok_choy.promise import EmptyPromise
|
|
|
|
from common.test.acceptance.pages.common.utils import hover
|
|
from common.test.acceptance.pages.lms.course_page import CoursePage
|
|
from common.test.acceptance.tests.helpers import is_focused_on_element
|
|
|
|
|
|
class DiscussionPageMixin(object):
|
|
|
|
def is_ajax_finished(self):
|
|
return self.browser.execute_script("return jQuery.active") == 0
|
|
|
|
def find_visible_element(self, selector):
|
|
"""
|
|
Finds a single visible element with the specified selector.
|
|
"""
|
|
full_selector = selector
|
|
if self.root_selector:
|
|
full_selector = self.root_selector + " " + full_selector
|
|
elements = self.q(css=full_selector)
|
|
return next((element for element in elements if element.is_displayed()), None)
|
|
|
|
@property
|
|
def new_post_button(self):
|
|
"""
|
|
Returns the new post button if visible, else it returns None.
|
|
"""
|
|
return self.find_visible_element(".new-post-btn")
|
|
|
|
@property
|
|
def new_post_form(self):
|
|
"""
|
|
Returns the new post form if visible, else it returns None.
|
|
"""
|
|
return self.find_visible_element(".forum-new-post-form")
|
|
|
|
def click_new_post_button(self):
|
|
"""
|
|
Clicks the 'New Post' button.
|
|
"""
|
|
self.wait_for(
|
|
lambda: self.new_post_button,
|
|
description="Waiting for new post button"
|
|
)
|
|
self.new_post_button.click()
|
|
self.wait_for(
|
|
lambda: self.new_post_form,
|
|
description="Waiting for new post form"
|
|
)
|
|
|
|
def click_cancel_new_post(self):
|
|
"""
|
|
Clicks the 'Cancel' button from the new post form.
|
|
"""
|
|
self.click_element(".cancel")
|
|
self.wait_for(
|
|
lambda: not self.new_post_form,
|
|
"Waiting for new post form to close"
|
|
)
|
|
|
|
|
|
class DiscussionThreadPage(PageObject, DiscussionPageMixin):
|
|
url = None
|
|
|
|
def __init__(self, browser, thread_selector):
|
|
super(DiscussionThreadPage, self).__init__(browser)
|
|
self.thread_selector = thread_selector
|
|
|
|
def _find_within(self, selector):
|
|
"""
|
|
Returns a query corresponding to the given CSS selector within the scope
|
|
of this thread page
|
|
"""
|
|
return self.q(css=self.thread_selector + " " + selector)
|
|
|
|
def is_browser_on_page(self):
|
|
return self.q(css=self.thread_selector).visible
|
|
|
|
def _get_element_text(self, selector):
|
|
"""
|
|
Returns the text of the first element matching the given selector, or
|
|
None if no such element exists
|
|
"""
|
|
text_list = self._find_within(selector).text
|
|
return text_list[0] if text_list else None
|
|
|
|
def is_element_visible(self, selector):
|
|
"""
|
|
Returns true if the element matching the specified selector is visible.
|
|
|
|
Args:
|
|
selector (str): The CSS selector that matches the desired element.
|
|
|
|
Returns:
|
|
bool: True if the element is visible.
|
|
|
|
"""
|
|
query = self._find_within(selector)
|
|
return query.present and query.visible
|
|
|
|
@contextmanager
|
|
def secondary_action_menu_open(self, ancestor_selector):
|
|
"""
|
|
Given the selector for an ancestor of a secondary menu, return a context
|
|
manager that will open and close the menu
|
|
"""
|
|
self.wait_for_ajax()
|
|
self._find_within(ancestor_selector + " .action-more").click()
|
|
EmptyPromise(
|
|
lambda: self.is_element_visible(ancestor_selector + " .actions-dropdown"),
|
|
"Secondary action menu opened"
|
|
).fulfill()
|
|
yield
|
|
if self.is_element_visible(ancestor_selector + " .actions-dropdown"):
|
|
self._find_within(ancestor_selector + " .action-more").click()
|
|
EmptyPromise(
|
|
lambda: not self.is_element_visible(ancestor_selector + " .actions-dropdown"),
|
|
"Secondary action menu closed"
|
|
).fulfill()
|
|
|
|
def get_group_visibility_label(self):
|
|
"""
|
|
Returns the group visibility label shown for the thread.
|
|
"""
|
|
return self._get_element_text(".group-visibility-label")
|
|
|
|
def get_response_total_text(self):
|
|
"""Returns the response count text, or None if not present"""
|
|
self.wait_for_ajax()
|
|
return self._get_element_text(".response-count")
|
|
|
|
def get_num_displayed_responses(self):
|
|
"""Returns the number of responses actually rendered"""
|
|
return len(self._find_within(".discussion-response"))
|
|
|
|
def get_shown_responses_text(self):
|
|
"""Returns the shown response count text, or None if not present"""
|
|
return self._get_element_text(".response-display-count")
|
|
|
|
def get_load_responses_button_text(self):
|
|
"""Returns the load more responses button text, or None if not present"""
|
|
return self._get_element_text(".load-response-button")
|
|
|
|
def load_more_responses(self):
|
|
"""Clicks the load more responses button and waits for responses to load"""
|
|
self._find_within(".load-response-button").click()
|
|
|
|
EmptyPromise(
|
|
self.is_ajax_finished,
|
|
"Loading more Responses"
|
|
).fulfill()
|
|
|
|
def has_add_response_button(self):
|
|
"""Returns true if the add response button is visible, false otherwise"""
|
|
return self.is_element_visible(".add-response-btn")
|
|
|
|
def has_discussion_reply_editor(self):
|
|
"""
|
|
Returns true if the discussion reply editor is is visible
|
|
"""
|
|
return self.is_element_visible(".discussion-reply-new")
|
|
|
|
def click_add_response_button(self):
|
|
"""
|
|
Clicks the add response button and ensures that the response text
|
|
field receives focus
|
|
"""
|
|
self._find_within(".add-response-btn").first.click()
|
|
EmptyPromise(
|
|
lambda: self._find_within(".discussion-reply-new textarea:focus").present,
|
|
"Response field received focus"
|
|
).fulfill()
|
|
|
|
@wait_for_js
|
|
def is_response_editor_visible(self, response_id):
|
|
"""Returns true if the response editor is present, false otherwise"""
|
|
return self.is_element_visible(u".response_{} .edit-post-body".format(response_id))
|
|
|
|
@wait_for_js
|
|
def is_discussion_body_visible(self):
|
|
return self.is_element_visible(".post-body")
|
|
|
|
def verify_mathjax_preview_available(self):
|
|
""" Checks that MathJax Preview css class is present """
|
|
self.wait_for(
|
|
lambda: len(self.q(css=".MathJax_Preview").text) > 0 and self.q(css=".MathJax_Preview").text[0] == "",
|
|
description="MathJax Preview is rendered"
|
|
)
|
|
|
|
def verify_mathjax_rendered(self):
|
|
""" Checks that MathJax css class is present """
|
|
self.wait_for(
|
|
lambda: self.is_element_visible(".MathJax_SVG"),
|
|
description="MathJax Preview is rendered"
|
|
)
|
|
|
|
def is_response_visible(self, comment_id):
|
|
"""Returns true if the response is viewable onscreen"""
|
|
self.wait_for_ajax()
|
|
return self.is_element_visible(u".response_{} .response-body".format(comment_id))
|
|
|
|
def is_response_editable(self, response_id):
|
|
"""Returns true if the edit response button is present, false otherwise"""
|
|
with self.secondary_action_menu_open(u".response_{} .discussion-response".format(response_id)):
|
|
return self.is_element_visible(u".response_{} .discussion-response .action-edit".format(response_id))
|
|
|
|
def is_response_deletable(self, response_id):
|
|
"""
|
|
Returns true if the delete response button is present, false otherwise
|
|
"""
|
|
with self.secondary_action_menu_open(u".response_{} .discussion-response".format(response_id)):
|
|
return self.is_element_visible(u".response_{} .discussion-response .action-delete".format(response_id))
|
|
|
|
def get_response_body(self, response_id):
|
|
return self._get_element_text(".response_{} .response-body".format(response_id))
|
|
|
|
def start_response_edit(self, response_id):
|
|
"""Click the edit button for the response, loading the editing view"""
|
|
with self.secondary_action_menu_open(u".response_{} .discussion-response".format(response_id)):
|
|
self._find_within(u".response_{} .discussion-response .action-edit".format(response_id)).first.click()
|
|
EmptyPromise(
|
|
lambda: self.is_response_editor_visible(response_id),
|
|
"Response edit started"
|
|
).fulfill()
|
|
|
|
def get_link_href(self):
|
|
"""Extracts href attribute of the referenced link"""
|
|
link_href = self._find_within(".post-body p a").attrs('href')
|
|
return link_href[0] if link_href else None
|
|
|
|
def get_response_vote_count(self, response_id):
|
|
vote_count_css = '.response_{} .discussion-response .action-vote'.format(response_id)
|
|
vote_count_element = self.browser.find_element_by_css_selector(vote_count_css)
|
|
# To get the vote count, one must hover over the element first.
|
|
hover(self.browser, vote_count_element)
|
|
return self._get_element_text(u".response_{} .discussion-response .action-vote .vote-count".format(response_id))
|
|
|
|
def vote_response(self, response_id):
|
|
current_count = self.get_response_vote_count(response_id)
|
|
self._find_within(u".response_{} .discussion-response .action-vote".format(response_id)).first.click()
|
|
self.wait_for(
|
|
lambda: current_count != self.get_response_vote_count(response_id),
|
|
description=u"Vote updated for {response_id}".format(response_id=response_id)
|
|
)
|
|
|
|
def cannot_vote_response(self, response_id):
|
|
"""Assert that the voting button is not visible on this response"""
|
|
return not self.is_element_visible(u".response_{} .discussion-response .action-vote".format(response_id))
|
|
|
|
def is_response_reported(self, response_id):
|
|
return self.is_element_visible(".response_{} .discussion-response .post-label-reported".format(response_id))
|
|
|
|
def report_response(self, response_id):
|
|
with self.secondary_action_menu_open(".response_{} .discussion-response".format(response_id)):
|
|
self._find_within(u".response_{} .discussion-response .action-report".format(response_id)).first.click()
|
|
self.wait_for_ajax()
|
|
EmptyPromise(
|
|
lambda: self.is_response_reported(response_id),
|
|
"Response is reported"
|
|
).fulfill()
|
|
|
|
def cannot_report_response(self, response_id):
|
|
"""Assert that the reporting button is not visible on this response"""
|
|
return not self.is_element_visible(u".response_{} .discussion-response .action-report".format(response_id))
|
|
|
|
def is_response_endorsed(self, response_id):
|
|
return "endorsed" in self._get_element_text(".response_{} .discussion-response .posted-details".format(response_id))
|
|
|
|
def endorse_response(self, response_id):
|
|
self._find_within(".response_{} .discussion-response .action-endorse".format(response_id)).first.click()
|
|
self.wait_for_ajax()
|
|
EmptyPromise(
|
|
lambda: self.is_response_endorsed(response_id),
|
|
"Response edit started"
|
|
).fulfill()
|
|
|
|
def set_response_editor_value(self, response_id, new_body):
|
|
"""Replace the contents of the response editor"""
|
|
self._find_within(u".response_{} .discussion-response .wmd-input".format(response_id)).fill(new_body)
|
|
|
|
def verify_link_editor_error_messages_shown(self):
|
|
"""
|
|
Confirm that the error messages are displayed in the editor.
|
|
"""
|
|
def errors_visible():
|
|
"""
|
|
Returns True if both errors are visible, False otherwise.
|
|
"""
|
|
return (
|
|
self.q(css="#new-url-input-field-message.has-error").visible and
|
|
self.q(css="#new-url-desc-input-field-message.has-error").visible
|
|
)
|
|
|
|
self.wait_for(errors_visible, "Form errors should be visible.")
|
|
|
|
def add_content_via_editor_button(self, content_type, response_id, url, description, is_decorative=False):
|
|
"""Replace the contents of the response editor"""
|
|
self._find_within(
|
|
"#wmd-{}-button-edit-post-body-{}".format(
|
|
content_type,
|
|
response_id,
|
|
)
|
|
).click()
|
|
self.q(css='#new-url-input').fill(url)
|
|
self.q(css='#new-url-desc-input').fill(description)
|
|
|
|
if is_decorative:
|
|
self.q(css='#img-is-decorative').click()
|
|
|
|
self.q(css='input[value="OK"]').click()
|
|
|
|
def submit_response_edit(self, response_id, new_response_body):
|
|
"""Click the submit button on the response editor"""
|
|
|
|
def submit_response_check_func():
|
|
"""
|
|
Tries to click "Update post" and returns True if the post
|
|
was successfully updated, False otherwise.
|
|
"""
|
|
self._find_within(
|
|
u".response_{} .discussion-response .post-update".format(
|
|
response_id
|
|
)
|
|
).first.click()
|
|
|
|
return (
|
|
not self.is_response_editor_visible(response_id) and
|
|
self.is_response_visible(response_id) and
|
|
self.get_response_body(response_id) == new_response_body
|
|
)
|
|
|
|
self.wait_for(submit_response_check_func, "Comment edit succeeded")
|
|
|
|
def is_show_comments_visible(self, response_id):
|
|
"""Returns true if the "show comments" link is visible for a response"""
|
|
return self.is_element_visible(u".response_{} .action-show-comments".format(response_id))
|
|
|
|
def show_comments(self, response_id):
|
|
"""Click the "show comments" link for a response"""
|
|
self._find_within(u".response_{} .action-show-comments".format(response_id)).first.click()
|
|
EmptyPromise(
|
|
lambda: self.is_element_visible(u".response_{} .comments".format(response_id)),
|
|
"Comments shown"
|
|
).fulfill()
|
|
|
|
def is_add_comment_visible(self, response_id):
|
|
"""Returns true if the "add comment" form is visible for a response"""
|
|
return self.is_element_visible("#wmd-input-comment-body-{}".format(response_id))
|
|
|
|
def is_comment_visible(self, comment_id):
|
|
"""Returns true if the comment is viewable onscreen"""
|
|
return self.is_element_visible(u"#comment_{} .response-body".format(comment_id))
|
|
|
|
def get_comment_body(self, comment_id):
|
|
return self._get_element_text("#comment_{} .response-body".format(comment_id))
|
|
|
|
def is_comment_deletable(self, comment_id):
|
|
"""Returns true if the delete comment button is present, false otherwise"""
|
|
with self.secondary_action_menu_open("#comment_{}".format(comment_id)):
|
|
return self.is_element_visible(u"#comment_{} .action-delete".format(comment_id))
|
|
|
|
def delete_comment(self, comment_id):
|
|
with self.handle_alert():
|
|
with self.secondary_action_menu_open("#comment_{}".format(comment_id)):
|
|
self._find_within(u"#comment_{} .action-delete".format(comment_id)).first.click()
|
|
EmptyPromise(
|
|
lambda: not self.is_comment_visible(comment_id),
|
|
"Deleted comment was removed"
|
|
).fulfill()
|
|
|
|
def is_comment_editable(self, comment_id):
|
|
"""Returns true if the edit comment button is present, false otherwise"""
|
|
with self.secondary_action_menu_open("#comment_{}".format(comment_id)):
|
|
return self.is_element_visible(u"#comment_{} .action-edit".format(comment_id))
|
|
|
|
def is_comment_editor_visible(self, comment_id):
|
|
"""Returns true if the comment editor is present, false otherwise"""
|
|
return self.is_element_visible(".edit-comment-body[data-id='{}']".format(comment_id))
|
|
|
|
def _get_comment_editor_value(self, comment_id):
|
|
return self._find_within("#wmd-input-edit-comment-body-{}".format(comment_id)).text[0]
|
|
|
|
def start_comment_edit(self, comment_id):
|
|
"""Click the edit button for the comment, loading the editing view"""
|
|
old_body = self.get_comment_body(comment_id)
|
|
with self.secondary_action_menu_open("#comment_{}".format(comment_id)):
|
|
self._find_within(u"#comment_{} .action-edit".format(comment_id)).first.click()
|
|
EmptyPromise(
|
|
lambda: (
|
|
self.is_comment_editor_visible(comment_id) and
|
|
not self.is_comment_visible(comment_id) and
|
|
self._get_comment_editor_value(comment_id) == old_body
|
|
),
|
|
"Comment edit started"
|
|
).fulfill()
|
|
|
|
def set_comment_editor_value(self, comment_id, new_body):
|
|
"""Replace the contents of the comment editor"""
|
|
self._find_within(u"#comment_{} .wmd-input".format(comment_id)).fill(new_body)
|
|
|
|
def submit_comment_edit(self, comment_id, new_comment_body):
|
|
"""Click the submit button on the comment editor"""
|
|
self._find_within(u"#comment_{} .post-update".format(comment_id)).first.click()
|
|
self.wait_for_ajax()
|
|
EmptyPromise(
|
|
lambda: (
|
|
not self.is_comment_editor_visible(comment_id) and
|
|
self.is_comment_visible(comment_id) and
|
|
self.get_comment_body(comment_id) == new_comment_body
|
|
),
|
|
"Comment edit succeeded"
|
|
).fulfill()
|
|
|
|
def cancel_comment_edit(self, comment_id, original_body):
|
|
"""Click the cancel button on the comment editor"""
|
|
self._find_within(u"#comment_{} .post-cancel".format(comment_id)).first.click()
|
|
EmptyPromise(
|
|
lambda: (
|
|
not self.is_comment_editor_visible(comment_id) and
|
|
self.is_comment_visible(comment_id) and
|
|
self.get_comment_body(comment_id) == original_body
|
|
),
|
|
"Comment edit was canceled"
|
|
).fulfill()
|
|
|
|
|
|
class DiscussionSortPreferencePage(CoursePage):
|
|
"""
|
|
Page that contain the discussion board with sorting options
|
|
"""
|
|
def __init__(self, browser, course_id):
|
|
super(DiscussionSortPreferencePage, self).__init__(browser, course_id)
|
|
self.url_path = "discussion/forum"
|
|
|
|
def is_browser_on_page(self):
|
|
"""
|
|
Return true if the browser is on the right page else false.
|
|
"""
|
|
return self.q(css="body.discussion .forum-nav-sort-control").present
|
|
|
|
def show_all_discussions(self):
|
|
""" Show the list of all discussions. """
|
|
self.q(css=".all-topics").click()
|
|
|
|
def get_selected_sort_preference(self):
|
|
"""
|
|
Return the text of option that is selected for sorting.
|
|
"""
|
|
# Using this workaround (execute script) to make this test work with Chrome browser
|
|
selected_value = self.browser.execute_script(
|
|
'var selected_value = $(".forum-nav-sort-control").val(); return selected_value')
|
|
return selected_value
|
|
|
|
def change_sort_preference(self, sort_by):
|
|
"""
|
|
Change the option of sorting by clicking on new option.
|
|
"""
|
|
self.q(css=u".forum-nav-sort-control option[value='{0}']".format(sort_by)).click()
|
|
# Click initiates an ajax call, waiting for it to complete
|
|
self.wait_for_ajax()
|
|
|
|
def refresh_page(self):
|
|
"""
|
|
Reload the page.
|
|
"""
|
|
self.browser.refresh()
|
|
|
|
|
|
class DiscussionTabSingleThreadPage(CoursePage):
|
|
def __init__(self, browser, course_id, discussion_id, thread_id):
|
|
super(DiscussionTabSingleThreadPage, self).__init__(browser, course_id)
|
|
self.thread_page = DiscussionThreadPage(
|
|
browser,
|
|
u"body.discussion .discussion-article[data-id='{thread_id}']".format(thread_id=thread_id)
|
|
)
|
|
self.url_path = "discussion/forum/{discussion_id}/threads/{thread_id}".format(
|
|
discussion_id=discussion_id, thread_id=thread_id
|
|
)
|
|
|
|
def is_browser_on_page(self):
|
|
return self.thread_page.is_browser_on_page()
|
|
|
|
def __getattr__(self, name):
|
|
return getattr(self.thread_page, name)
|
|
|
|
def show_all_discussions(self):
|
|
""" Show the list of all discussions. """
|
|
self.q(css=".all-topics").click()
|
|
|
|
def close_open_thread(self):
|
|
with self.thread_page.secondary_action_menu_open(".thread-main-wrapper"):
|
|
self._find_within(".thread-main-wrapper .action-close").first.click()
|
|
|
|
def _thread_is_rendered_successfully(self, thread_id):
|
|
return self.q(css=".discussion-article[data-id='{}']".format(thread_id)).visible
|
|
|
|
def click_and_open_thread(self, thread_id):
|
|
"""
|
|
Click specific thread on the list.
|
|
"""
|
|
thread_selector = "li[data-id='{}']".format(thread_id)
|
|
self.show_all_discussions()
|
|
self.q(css=thread_selector).first.click()
|
|
EmptyPromise(
|
|
lambda: self._thread_is_rendered_successfully(thread_id),
|
|
"Thread has been rendered"
|
|
).fulfill()
|
|
|
|
def check_threads_rendered_successfully(self, thread_count):
|
|
"""
|
|
Count the number of threads available on page.
|
|
"""
|
|
return len(self.q(css=".forum-nav-thread").results) == thread_count
|
|
|
|
|
|
class InlineDiscussionPage(PageObject, DiscussionPageMixin):
|
|
"""
|
|
Acceptance tests for inline discussions.
|
|
"""
|
|
url = None
|
|
|
|
def __init__(self, browser, discussion_id):
|
|
super(InlineDiscussionPage, self).__init__(browser)
|
|
self.root_selector = (
|
|
u".discussion-module[data-discussion-id='{discussion_id}'] ".format(
|
|
discussion_id=discussion_id
|
|
)
|
|
)
|
|
|
|
def _find_within(self, selector):
|
|
"""
|
|
Returns a query corresponding to the given CSS selector within the scope
|
|
of this discussion page
|
|
"""
|
|
return self.q(css=self.root_selector + " " + selector)
|
|
|
|
def is_browser_on_page(self):
|
|
self.wait_for_ajax()
|
|
return self.q(css=self.root_selector).present
|
|
|
|
def is_discussion_expanded(self):
|
|
return self._find_within(".discussion").present
|
|
|
|
def expand_discussion(self):
|
|
"""Click the link to expand the discussion"""
|
|
self._find_within(".discussion-show").first.click()
|
|
EmptyPromise(
|
|
self.is_discussion_expanded,
|
|
"Discussion expanded"
|
|
).fulfill()
|
|
|
|
def get_num_displayed_threads(self):
|
|
return len(self._find_within(".forum-nav-thread"))
|
|
|
|
def element_exists(self, selector):
|
|
return self.q(css=self.root_selector + " " + selector).present
|
|
|
|
def click_element(self, selector):
|
|
self.wait_for_element_presence(
|
|
u"{discussion} {selector}".format(discussion=self.root_selector, selector=selector),
|
|
u"{selector} is visible".format(selector=selector)
|
|
)
|
|
self._find_within(selector).click()
|
|
|
|
def is_new_post_button_visible(self):
|
|
"""
|
|
Check if new post button present and visible
|
|
"""
|
|
return self._is_element_visible('.new-post-btn')
|
|
|
|
@wait_for_js
|
|
def _is_element_visible(self, selector):
|
|
query = self._find_within(selector)
|
|
return query.present and query.visible
|
|
|
|
def show_thread(self, thread_id):
|
|
"""
|
|
Clicks the link for the specified thread to show the detailed view.
|
|
"""
|
|
self.wait_for_element_presence('.forum-nav-thread-link', 'Thread list has loaded')
|
|
thread_selector = u".forum-nav-thread[data-id='{thread_id}'] .forum-nav-thread-link".format(thread_id=thread_id)
|
|
self._find_within(thread_selector).first.click()
|
|
self.thread_page = InlineDiscussionThreadPage(self.browser, thread_id) # pylint: disable=attribute-defined-outside-init
|
|
self.thread_page.wait_for_page()
|
|
|
|
|
|
class InlineDiscussionThreadPage(DiscussionThreadPage):
|
|
"""
|
|
Page object to manipulate an individual thread view in an inline discussion.
|
|
"""
|
|
def __init__(self, browser, thread_id):
|
|
super(InlineDiscussionThreadPage, self).__init__(
|
|
browser,
|
|
u".discussion-module .discussion-article[data-id='{thread_id}']".format(thread_id=thread_id)
|
|
)
|
|
|
|
def is_thread_anonymous(self):
|
|
return not self.q(css=".posted-details > .username").present
|
|
|
|
@wait_for_js
|
|
def check_if_selector_is_focused(self, selector):
|
|
"""
|
|
Check if selector is focused
|
|
"""
|
|
return is_focused_on_element(self.browser, selector)
|
|
|
|
|
|
class DiscussionUserProfilePage(CoursePage):
|
|
|
|
TEXT_NEXT = u'Next >'
|
|
TEXT_PREV = u'< Previous'
|
|
PAGING_SELECTOR = ".discussion-pagination[data-page-number]"
|
|
|
|
def __init__(self, browser, course_id, user_id, username, page=1):
|
|
super(DiscussionUserProfilePage, self).__init__(browser, course_id)
|
|
self.url_path = "discussion/forum/dummy/users/{}?page={}".format(user_id, page)
|
|
self.username = username
|
|
|
|
def is_browser_on_page(self):
|
|
return (
|
|
self.q(css='.discussion-user-threads[data-course-id="{}"]'.format(self.course_id)).present
|
|
and
|
|
self.q(css='.user-name').present
|
|
and
|
|
self.q(css='.user-name').text[0] == self.username
|
|
)
|
|
|
|
@wait_for_js
|
|
def is_window_on_top(self):
|
|
return self.browser.execute_script("return $('html, body').offset().top") == 0
|
|
|
|
def get_shown_thread_ids(self):
|
|
elems = self.q(css="li.forum-nav-thread")
|
|
return [elem.get_attribute("data-id") for elem in elems]
|
|
|
|
def click_on_sidebar_username(self):
|
|
self.wait_for_page()
|
|
self.q(css='.user-name').first.click()
|
|
|
|
def get_user_roles(self):
|
|
"""Get user roles"""
|
|
return self.q(css='.user-roles').text[0]
|
|
|
|
|
|
class DiscussionTabHomePage(CoursePage, DiscussionPageMixin):
|
|
|
|
ALERT_SELECTOR = ".discussion-body .forum-nav .search-alert"
|
|
|
|
def __init__(self, browser, course_id):
|
|
super(DiscussionTabHomePage, self).__init__(browser, course_id)
|
|
self.url_path = "discussion/forum/"
|
|
self.root_selector = None
|
|
|
|
def is_browser_on_page(self):
|
|
return self.q(css=".discussion-body section.home-header").present
|
|
|
|
def perform_search(self, text="dummy"):
|
|
self.q(css=".search-input").fill(text + chr(10))
|
|
EmptyPromise(
|
|
self.is_ajax_finished,
|
|
"waiting for server to return result"
|
|
).fulfill()
|
|
|
|
def is_element_visible(self, selector):
|
|
"""
|
|
Returns true if the element matching the specified selector is visible.
|
|
"""
|
|
query = self.q(css=selector)
|
|
return query.present and query.visible
|
|
|
|
def is_checkbox_selected(self, selector):
|
|
"""
|
|
Returns true or false depending upon the matching checkbox is checked.
|
|
"""
|
|
return self.q(css=selector).selected
|
|
|
|
def refresh_and_wait_for_load(self):
|
|
"""
|
|
Refresh the page and wait for all resources to load.
|
|
"""
|
|
self.browser.refresh()
|
|
self.wait_for_page()
|
|
|
|
def get_search_alert_messages(self):
|
|
return self.q(css=self.ALERT_SELECTOR + " .message").text
|
|
|
|
def get_search_alert_links(self):
|
|
return self.q(css=self.ALERT_SELECTOR + " .link-jump")
|
|
|
|
def dismiss_alert_message(self, text):
|
|
"""
|
|
dismiss any search alert message containing the specified text.
|
|
"""
|
|
def _match_messages(text):
|
|
return self.q(css=".search-alert").filter(lambda elem: text in elem.text)
|
|
|
|
for alert_id in _match_messages(text).attrs("id"):
|
|
self.q(css=u"{}#{} .dismiss".format(self.ALERT_SELECTOR, alert_id)).click()
|
|
EmptyPromise(
|
|
lambda: _match_messages(text).results == [],
|
|
"waiting for dismissed alerts to disappear"
|
|
).fulfill()
|
|
|
|
def click_element(self, selector):
|
|
"""
|
|
Clicks the element specified by selector
|
|
"""
|
|
element = self.q(css=selector)
|
|
return element.click()
|
|
|
|
def set_new_post_editor_value(self, new_body):
|
|
"""
|
|
Set the Discussions new post editor (wmd) with the content in new_body
|
|
"""
|
|
self.q(css=".wmd-input").fill(new_body)
|
|
|
|
def get_new_post_preview_value(self, selector=".wmd-preview > *"):
|
|
"""
|
|
Get the rendered preview of the contents of the Discussions new post editor
|
|
Waits for content to appear, as the preview is triggered on debounced/delayed onchange
|
|
"""
|
|
self.scroll_to_element(selector)
|
|
self.wait_for_element_visibility(selector, "WMD preview pane has contents", timeout=10)
|
|
return self.q(css=".wmd-preview").html[0]
|
|
|
|
def get_new_post_preview_text(self, selector=".wmd-preview > div"):
|
|
"""
|
|
Get the rendered preview of the contents of the Discussions new post editor
|
|
Waits for content to appear, as the preview is triggered on debounced/delayed onchange
|
|
"""
|
|
self.scroll_to_element(selector)
|
|
self.wait_for_element_visibility(selector, "WMD preview pane has contents", timeout=10)
|
|
return self.q(css=".wmd-preview").text[0]
|