Bases: ModalWindowComponent
Модальное окно шаблона.
Наследует ModalWindowComponent и добавляет функционал для:
1. Инициализации модального окна с конкретным шаблоном
2. Закрытия модального окна
3. Получения конфигурационных данных шаблона
4. Проверки содержимого модального окна
Source code in components_derived\modal_view_template.py
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160 | class ViewTemplateModalWindow(ModalWindowComponent):
"""Модальное окно шаблона.
Наследует ModalWindowComponent и добавляет функционал для:
1. Инициализации модального окна с конкретным шаблоном
2. Закрытия модального окна
3. Получения конфигурационных данных шаблона
4. Проверки содержимого модального окна
"""
def __init__(self, page: Page, template_name: str):
"""Инициализирует элементы формы модального окна шаблона."""
super().__init__(page)
# Настройка заголовка и кнопки закрытия
self.window_title = template_name
locator_button_toolbar_close = self.page.get_by_role("navigation").filter(
has_text=re.compile(self.window_title)
).get_by_role("button")
self.add_toolbar_title(self.window_title)
self.add_toolbar_button(locator_button_toolbar_close, "close")
def close_window(self) -> None:
"""Закрывает окно через кнопку 'Закрыть'."""
close_button = self.get_button_by_name("close")
close_button.click()
def check_content(self) -> None:
"""Проверяет наличие и корректность элементов окна.
Проверяет:
1. Наличие заголовка окна с именем шаблона
2. Видимость кнопки закрытия
3. Подсказку кнопки закрытия
"""
self.check_by_window_title()
self.check_toolbar_button_visibility("close")
self.check_toolbar_button_tooltip("close", "Закрыть")
def get_modal_window_data(self) -> dict:
"""Извлекает данные из модального окна шаблона и структурирует по кодам и значениям.
Returns:
dict: Данные в формате {'код': 'значение'} как в API
"""
modal_data = {}
# Получаем все значения из input полей
input_locator = self.get_locator(ModalWindowLocators.MODAL_WINDOW_TEXT_FIELD_INPUT)
# Проверка наличия элементов
input_count = input_locator.count()
if input_count == 0:
logger.warning("Поля ввода не найдены в модальном окне")
return modal_data
all_values = []
# Обрабатываем каждое поле с обработкой возможных ошибок
for i in range(input_count):
input_field = input_locator.nth(i)
# Проверяем, что элемент видим и доступен
if not input_field.is_visible():
logger.debug("Поле %s не видимо, пропускаем", i)
continue
# Получаем значение с обработкой возможных ошибок состояния элемента
if input_field.is_visible():
value = input_field.input_value().strip()
if value: # Игнорируем пустые значения
all_values.append(value)
else:
logger.debug("Поле %s стало невидимым после проверки, пропускаем", i)
logger.info("Все значения из полей: %s", all_values)
# Анализируем пары код-значение
i = 0
while i < len(all_values) - 1:
current_value = all_values[i]
next_value = all_values[i + 1]
# Определяем, является ли текущее значение кодом (число)
if current_value.isdigit():
# Текущее значение - код, следующее - значение
modal_data[current_value] = next_value
i += 2 # Перескакиваем через пару
else:
# Если текущее значение не число, ищем следующую пару
i += 1
# Добавляем имя шаблона с ключом 'Шаблон' вместо 'template'
if all_values:
modal_data['Шаблон'] = all_values[-1]
logger.info("Структурированные данные из модального окна: %s", modal_data)
return modal_data
def compare_modal_with_api_data(self, modal_data: dict, api_data: dict,
template_name: str) -> None:
"""Сравнивает данные из модального окна с данными из API."""
errors = []
# Создаем копию API данных с заменой 'template' на 'Шаблон'
api_data_adapted = api_data.copy()
if 'template' in api_data_adapted:
api_data_adapted['Шаблон'] = api_data_adapted.pop('template')
# Сравниваем все поля
for code, expected_value in api_data_adapted.items():
if code in modal_data:
actual_value = modal_data[code]
if actual_value != expected_value:
error_msg = (
f"Расхождение для кода {code}: "
f"модальное окно='{actual_value}', API='{expected_value}'"
)
logger.error(error_msg)
errors.append(error_msg)
else:
error_msg = f"Код {code} не найден в модальном окне"
logger.error(error_msg)
errors.append(error_msg)
# Дополнительная проверка имени шаблона
modal_template = modal_data.get('Шаблон', '')
if modal_template != template_name:
error_msg = (
f"Расхождение в имени шаблона: "
f"модальное окно='{modal_template}', ожидается='{template_name}'"
)
logger.error(error_msg)
errors.append(error_msg)
# Если есть расхождения, выбрасываем ошибку
if errors:
error_details = "\n".join(errors)
assert False, (
f"Обнаружены расхождения для шаблона '{template_name}':\n{error_details}"
)
logger.info("Данные модального окна соответствуют API для шаблона '%s'", template_name)
|
__init__(page, template_name)
Инициализирует элементы формы модального окна шаблона.
Source code in components_derived\modal_view_template.py
27
28
29
30
31
32
33
34
35
36
37
38 | def __init__(self, page: Page, template_name: str):
"""Инициализирует элементы формы модального окна шаблона."""
super().__init__(page)
# Настройка заголовка и кнопки закрытия
self.window_title = template_name
locator_button_toolbar_close = self.page.get_by_role("navigation").filter(
has_text=re.compile(self.window_title)
).get_by_role("button")
self.add_toolbar_title(self.window_title)
self.add_toolbar_button(locator_button_toolbar_close, "close")
|
check_content()
Проверяет наличие и корректность элементов окна.
Проверяет:
1. Наличие заголовка окна с именем шаблона
2. Видимость кнопки закрытия
3. Подсказку кнопки закрытия
Source code in components_derived\modal_view_template.py
45
46
47
48
49
50
51
52
53
54
55 | def check_content(self) -> None:
"""Проверяет наличие и корректность элементов окна.
Проверяет:
1. Наличие заголовка окна с именем шаблона
2. Видимость кнопки закрытия
3. Подсказку кнопки закрытия
"""
self.check_by_window_title()
self.check_toolbar_button_visibility("close")
self.check_toolbar_button_tooltip("close", "Закрыть")
|
close_window()
Закрывает окно через кнопку 'Закрыть'.
Source code in components_derived\modal_view_template.py
| def close_window(self) -> None:
"""Закрывает окно через кнопку 'Закрыть'."""
close_button = self.get_button_by_name("close")
close_button.click()
|
compare_modal_with_api_data(modal_data, api_data, template_name)
Сравнивает данные из модального окна с данными из API.
Source code in components_derived\modal_view_template.py
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160 | def compare_modal_with_api_data(self, modal_data: dict, api_data: dict,
template_name: str) -> None:
"""Сравнивает данные из модального окна с данными из API."""
errors = []
# Создаем копию API данных с заменой 'template' на 'Шаблон'
api_data_adapted = api_data.copy()
if 'template' in api_data_adapted:
api_data_adapted['Шаблон'] = api_data_adapted.pop('template')
# Сравниваем все поля
for code, expected_value in api_data_adapted.items():
if code in modal_data:
actual_value = modal_data[code]
if actual_value != expected_value:
error_msg = (
f"Расхождение для кода {code}: "
f"модальное окно='{actual_value}', API='{expected_value}'"
)
logger.error(error_msg)
errors.append(error_msg)
else:
error_msg = f"Код {code} не найден в модальном окне"
logger.error(error_msg)
errors.append(error_msg)
# Дополнительная проверка имени шаблона
modal_template = modal_data.get('Шаблон', '')
if modal_template != template_name:
error_msg = (
f"Расхождение в имени шаблона: "
f"модальное окно='{modal_template}', ожидается='{template_name}'"
)
logger.error(error_msg)
errors.append(error_msg)
# Если есть расхождения, выбрасываем ошибку
if errors:
error_details = "\n".join(errors)
assert False, (
f"Обнаружены расхождения для шаблона '{template_name}':\n{error_details}"
)
logger.info("Данные модального окна соответствуют API для шаблона '%s'", template_name)
|
get_modal_window_data()
Извлекает данные из модального окна шаблона и структурирует по кодам и значениям.
Returns:
| Name | Type |
Description |
dict |
dict
|
Данные в формате {'код': 'значение'} как в API
|
Source code in components_derived\modal_view_template.py
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115 | def get_modal_window_data(self) -> dict:
"""Извлекает данные из модального окна шаблона и структурирует по кодам и значениям.
Returns:
dict: Данные в формате {'код': 'значение'} как в API
"""
modal_data = {}
# Получаем все значения из input полей
input_locator = self.get_locator(ModalWindowLocators.MODAL_WINDOW_TEXT_FIELD_INPUT)
# Проверка наличия элементов
input_count = input_locator.count()
if input_count == 0:
logger.warning("Поля ввода не найдены в модальном окне")
return modal_data
all_values = []
# Обрабатываем каждое поле с обработкой возможных ошибок
for i in range(input_count):
input_field = input_locator.nth(i)
# Проверяем, что элемент видим и доступен
if not input_field.is_visible():
logger.debug("Поле %s не видимо, пропускаем", i)
continue
# Получаем значение с обработкой возможных ошибок состояния элемента
if input_field.is_visible():
value = input_field.input_value().strip()
if value: # Игнорируем пустые значения
all_values.append(value)
else:
logger.debug("Поле %s стало невидимым после проверки, пропускаем", i)
logger.info("Все значения из полей: %s", all_values)
# Анализируем пары код-значение
i = 0
while i < len(all_values) - 1:
current_value = all_values[i]
next_value = all_values[i + 1]
# Определяем, является ли текущее значение кодом (число)
if current_value.isdigit():
# Текущее значение - код, следующее - значение
modal_data[current_value] = next_value
i += 2 # Перескакиваем через пару
else:
# Если текущее значение не число, ищем следующую пару
i += 1
# Добавляем имя шаблона с ключом 'Шаблон' вместо 'template'
if all_values:
modal_data['Шаблон'] = all_values[-1]
logger.info("Структурированные данные из модального окна: %s", modal_data)
return modal_data
|