Вот моя попытка полностью универсального решения на Python:
Во-первых, общая функция ожидания (используйте WebDriverWait, если хотите, я считаю их уродливыми):
def wait_for(condition_function):
start_time = time.time()
while time.time() < start_time + 3:
if condition_function():
return True
else:
time.sleep(0.1)
raise Exception('Timeout waiting for {}'.format(condition_function.__name__))
Далее, решение основывается на том факте, что селен записывает (внутренний) id-номер для всех элементов на странице, включая <html>
элемент верхнего уровня . Когда страница обновляется или загружается, она получает новый элемент html с новым идентификатором.
Итак, предположим, что вы хотите щелкнуть ссылку с текстом «моя ссылка», например:
old_page = browser.find_element_by_tag_name('html')
browser.find_element_by_link_text('my link').click()
def page_has_loaded():
new_page = browser.find_element_by_tag_name('html')
return new_page.id != old_page.id
wait_for(page_has_loaded)
Чтобы получить больше универсального помощника многократного использования на Python, вы можете создать диспетчер контекста:
from contextlib import contextmanager
@contextmanager
def wait_for_page_load(browser):
old_page = browser.find_element_by_tag_name('html')
yield
def page_has_loaded():
new_page = browser.find_element_by_tag_name('html')
return new_page.id != old_page.id
wait_for(page_has_loaded)
И затем вы можете использовать его практически для любого взаимодействия с селеном:
with wait_for_page_load(browser):
browser.find_element_by_link_text('my link').click()
Я считаю, что это пуленепробиваемое! Что вы думаете?
Больше информации в блоге об этом здесь