Задачи:
— Поиск окна по части заголовка
— Поиск фрагмента изображения похожего на сохраненный в файле кусок.
— Возможность позиционировать курсор, нажимать кнопки мышки.
— Возможность посылать нажатия клавиш
Использовать SikuliX, Selenium не хотелось, слишком громоздко, нужно переучиваться.
В сети, нашел пример поиска части изображения в другом изображении.
Решил добавить функциональность в ВК созданную для проекта SmartConfigurator (WinExt) и опубликовать.
Сначала небольшой пример возможностей ВК.
- Активировать окно по части заголовка
- Через меню Операции открыть список документов
- Создать документ
- Создать контрагента с наименованием: Контрагент_<ТекущаяДата()>
- Выбрать контрагента
- Заполнить ТЧ документа
- Провести документ
- Сформировать отчет по остаткам
#Использовать WinExt
Перем РаботаСОкнами;
Перем Экран;
Перем Мышь;
Перем Клавиши;
Процедура ПриСозданииОбъекта()
РаботаСОкнами = Новый РаботаСОкнами();
Мышь = Новый Мышь();
Экран = Новый Экран();
Клавиши = Новый МСПослатьКлавиши();
// Активируем окно по части заголовка (окно не должно быть свернуто)
РаботаСОкнами.АктивироватьОкноПоЗаголовку("Win ext example ordinary");
СоздатьПоступлениеТМЦ();
СформироватьОтчетОстаткиТМЦ();
КонецПроцедуры
Процедура СоздатьПоступлениеТМЦ()
// меню Операции
ВыполнитьШаг("img/menu_operations.png");
Приостановить(1000);
// Меню Документы
ВыполнитьШаг("img/menu_documents.png");
Приостановить(1000);
// Откроем список Поступлений ТМЦ
// Для разнообразия нажмем Enter
Клавиши.ПослатьКлавиши("{ENTER}");
Приостановить(1000);
// Нажнем кнопку добавить в списке ПоступлениеТМЦ
ВыполнитьШаг("img/btn_add.png");
Приостановить(1000);
// Найдем подпись контрагенты и кликнем правей, в поле ввода
ВыполнитьШаг("img/form_title_contragent.png",,100);
// Откроем окно выбора из списка, создадим контрагента, выберем его
Клавиши.ПослатьКлавиши("{F4}{INSERT}{TAB}Контрагент_"+ТекущаяДата()+"#k8SjZc9Dxk({ENTER}){ENTER}");
// Добавим товар в табличную часть и введем количество, цену, сумму
Клавиши.ПослатьКлавиши("{TAB}{INSERT}товар1{ENTER}10{ENTER}1{ENTER}10{ENTER}");
Приостановить(1000);
// На панели документа нажмем Ок (скрин всей панели, снова делаем сдвиг курсора)
ВыполнитьШаг("img/panel_doc_buttons.png",,-50);
КонецПроцедуры
Процедура СформироватьОтчетОстаткиТМЦ()
// меню Операции
ВыполнитьШаг("img/menu_operations.png");
Приостановить(1000);
ВыполнитьШаг("img/menu_reports.png");
Приостановить(1000);
Мышь.УстановитьПозициюКурсора(0, 0);
Приостановить(300);
ВыполнитьШаг("img/btn_big_ok.png");
Приостановить(1000);
ВыполнитьШаг("img/btn_report_run.png");
Приостановить(1000);
Мышь.УстановитьПозициюКурсора(0, 0);
Приостановить(1000);
ПутьКФрагменту = Новый Файл("img/report_after_receipt.png").ПолноеИмя;
ФрагментРезультатОтчета = Экран.НайтиФрагмент(ПутьКФрагменту);
Если ФрагментРезультатОтчета = Неопределено Тогда
ВызватьИсключение "Ожидаемый фрагмент с результатом отчета не найден.";
КонецЕсли;
КонецПроцедуры
Процедура ВыполнитьШаг(ИмяФайлаФрагмента, ДвойнойКлик = Ложь, СмещениеЛево = 0, СмещениеВерх = 0)
ПутьКФрагменту = Новый Файл(ИмяФайлаФрагмента).ПолноеИмя;
Фрагмент = Экран.НайтиФрагмент(ПутьКФрагменту);
Если Фрагмент = Неопределено Тогда
ВызватьИсключение "Не найден фрагмент: " + ИмяФайлаФрагмента;
КонецЕсли;
Координаты = КоординатыЦентра(Фрагмент);
Если ДвойнойКлик Тогда
Мышь.ЛеваяКнопкаКлик(Координаты.Лево + СмещениеЛево, Координаты.Верх + СмещениеВерх);
Приостановить(50);
Мышь.ЛеваяКнопкаКлик(Координаты.Лево + СмещениеЛево, Координаты.Верх + СмещениеВерх);
Иначе
Мышь.ЛеваяКнопкаКлик(Координаты.Лево + СмещениеЛево, Координаты.Верх + СмещениеВерх);
КонецЕсли;
КонецПроцедуры
Функция КоординатыЦентра(Фрагмент)
Лево = Фрагмент.Лево + (Фрагмент.Ширина / 2);
Верх = Фрагмент.Верх + (Фрагмент.Высота / 2);
Координаты = Новый Структура();
Координаты.Вставить("Лево", Лево);
Координаты.Вставить("Верх", Верх);
Возврат Координаты;
КонецФункции
Установка
Скачать архив здесь: https://github.com/ret-Phoenix/WinExt/releases
Выполнить команду: `opm install WinExt.ospx`
* Укажите свой путь до файла WinExt.ospx
Позже будет доступна установка через OneScript OPM без предварительной скачки.
Возможности ВК:
Экран / Sreen
Методы
РазрешениеЭкрана / ScreenResolution()
Получить разрешение экрана
Параметры
- SreenNumber: Число — Номер экрана, если ничего не задано — берет основной экран
Возвращаемое значение
ФиксированнаяСтруктура (Ширина, Высота)
НайтиФрагмент / FindFragment()
Найти на экране фрагмент из файла
Параметры
- fragmentFileName: Путь к файлу с искомым фрагментом
Возвращаемое значение
ФрагментЭкрана
ФрагментЭкрана / ScreenFragment
Свойства
Верх / Top
Доступ: Чтение
Лево / Left
Доступ: Чтение
Высота / Height
Доступ: Чтение
Ширина / Width
Доступ: Чтение
РаботаСОкнами / WorkWithWindows
Методы
ЗапомнитьТекущееОкно / GetLinkToCurWindow()
АктивироватьЗапомненноеОкно / WndActivate()
АктивироватьОкноПоЗаголовку / SwitchToWinByTitle()
Мышь / Mouse
Управление мышкой. Установить/Считать позицию. Кликнуть кнопкой.
Методы
ЛеваяКнопкаКлик / LeftMouseClick()
ПраваяКнопкаКлик / RightMouseClick()
СредняяКнопкаКлик / MiddleMouseClick()
УстановитьПозициюКурсора / SetCursorPosition()
Установить позицию курсора
Параметры
-
posX: Позиция X
-
posY: Позиция Y
Возвращаемое значение
Булево — Удалось установить позицию курсора
ПолучитьПозициюКурсора / GetCursorPosition()
Получить позицию курсора
Возвращаемое значение
Структура — Ключи: Верх, Лево
МСПослатьКлавиши / MSSendKeys
Класс для отправки нажатий клавиш, основан на родном .net SendKeys.
Методы
ПослатьКлавиши / SendKeys()
Послать нажатия клавиш. Правила: https://docs.microsoft.com/ru-ru/dotnet/api/system.windows.forms.sendkeys?view=netframework-4.7
Параметры
- keys: Строка — набор клавиш
Ого, теперь приложения на OneScript, написанные с применением c применением библиотекиGUI можно будет покрывать сценарными тестами ))
А если серьезно, то два вопроса:
https://github.com/ret-Phoenix/oscript-simple-gui , чтобы не приходилось в исходники смотреть для применения ))
— Предполагается ли кросплатформенность с применением mono или хотя бы wine?
— Будет ли на github документация? Хотелось бы более полную документацию чем по
(1)
1. Изначально ВК тестировалась как раз на OneScript GUI (лежит в тестах)
2. Кроссплатформенность — с этим проблемы, поэтому и называется WinExt, т.е. упор на Windows.
Например работа с мышкой и хоткеи — работа через WinAPI, как себя поведет на Wine я не знаю.
Работу под Mono не проверял, по идее класс Экран должен работать, что-то из управления мышкой может получится сделать без WinApi.
3. Документация, на уровне API есть. остальное уже скорее howto.
По тому же GUI громадный фронт работ, по документации так же. Если есть что добавить/подправить — присылайте.
В дальнейшем планируется добавить еще алгоритмы анализа слепков, с возможностью указания какой использовать.
Кстати, есть возможность использовать плагин из 1С. Например SQL уже так используется. Не помню точно кто это делал, поищу. Как найду — будет 2 статья, но уже по работе из 1С с DLL.
Ништяаааак!
круто.
а откуда брать координаты для метода ФрагментЭкрана () ?
для тестирования 1С обычные формы еще нельзя использовать?
(5)
фрагмент экрана — это что ищем.
см. gif и см код. как раз тестирование обычных форм сделано.
Было бы неплохо добавить функцию для снятия скриншотов…
Крут крут! И на шарпе наконец то и под оскрипт в исходниках… ещё в ВК обернуть заодно :).
Слушай, а можешь FindSubimage на что-нить другое заменить?
Может заюзаешьhttps://www.nuget.org/packages/AForge.Vision/ или https://www.nuget.org/packages/OpenCV/ ?
Уже лучше чем сикули работать будет… Точное соответствие это несколько неюзабельно.
Я себе конечно записал в личный бэклог к этой теме вернуться, но чувствую ты быстрее доберёшься…
(8) AForge, насколько помню мертв. OpenCV — что-то мелькало по нему, не разбирался.
Точность соответствия пока не регулируется, но это есть в планах.
В будущем хотел разные движки-анализаторы добавлять.
(9) Да, Aforge умер, прочитал :(. Раньше помогал он, в любом случае применение то простое.
https://www.nuget.org/packages/Accord.Imaging
Сейчас nuget по CV самый популярный:
Если юзать хотя бы эти движки — с целью скриптов получим лучшее что сейчас есть…
(10) еще Magic.Net есть, более известный как ImageMagic
(6)то есть координаты фрагмента экрана нужно писать наугад, пока не попадешь? если например у мена нету линейки вертикальной и горизонтальной?
Например в сикули выбираешь область экрана рамкой и она сама координаты фрагмента определяет
(12) не понимаю о чем вы вообще пишите.
Читайте API и код.
ФрагментЭкрана — это искомая область, загружаемая из файла. что видно из примеров, документации.
Экран.НайтиФрагмент — ищет на основном экране изображение из ФрагментЭкрана. Если находит вернет структуру с координтами, иначе неопределено. Это видно из примера и документации.
Я ищу не по координатам, а по содержимому. Сикули кстати так же.
(7) добавил в задачи, это просто, сделаю в ближайшее время.
(1) и часто приходилось лезть в исходник?
Из текущих проблем, знаю только о поиске подчиненного элемента. В остальном, как мне кажется, документация с примерами решают все.
Но я разработчик, мне многое понятно т.к. сам делал, поэтому могу что-то не замечать…
(13) Закроем эту тему. Видимо мне не удастся донести свой вопрос. Закину в кладовку может найду время разобраться с этим.
пока буду по-старому Сикули использовать.
(16) Ваше право, только ФрагментЭкрана лишь возвращает координаты найденного изображения, и принимает только путь с файлом картинки которую нужно найти. Поэтому и не понимаю как писать координаты экрана фрагмента, когда они впринципе не указываются, а возвращаются. В документации и примере это четко видно:
Показать
(17) Я, наверное невнимательно смотрел. У вас ФрагментЭкрана — это некий объект со свойствами, который возвращается методом НайтиФрагмент(). А я было подумал, что это метод ФрагментЭкрана(), который аналогично Region() в Сикули, указывает область поиска фрагмента. От этого и возник вопрос.
Ну ОК.
А есть ли у вас метод, который указывает область поиска ? или поиск фрагмента всегда идет по всему экрану?
И как устроен процесс подготовки фрагментов для сценария? Нужно заранее составить библиотеку картинок-скринов элементов интерфейса, например, в Paint ?
(18) Метода ограничивающего область поиска нет.
github
Подготовка фрагментов — нарезка картинок-скринов интерфейса.
Пример из исходников на
(19) Ок. спасибо.
Про AutoHotKey было уже?)
Сергей, зайди плз на github, посмотри форк
Добрый день. Подскажите, пожалуйста, по каким причинам может быть не найден фрагмент на экране? Есть какие-то условия качества/формата картинки, которую нужно использовать для поиска? Проверила все функции, все работает кроме фрагментов. Может у вас есть какие-то предположения или вы с чем-то подобным сталкивались?
Сталкивался. Когда много шума. Скрины делал в png, программа joxi.
Для начала попробуйте запустить мои тесты. И код точно верный?
Да, код верен. Так же делала скрины с помощью joxi. Буду тестировать на других картинках, посмотрю, что получится. Спасибо!
(25) Напомню, приложение не должно быть свернутым.
(23) образец лучше всего делать в формате bmp с той же разрядностью цвета как у экрана, т.к. если делать в jpg, то картинка может и не совпадать из-за сжатия…
(27) PNG — вполне неплохой вариант.
У меня в тестах он используется.
(28) да, PNG и GIF тоже подходят, они используют сжатие без потерь, но надо следить, что бы палитра в них правильная была выбрана… особенно в GIF…
(0)
А в управляемых формах тоже можно дочернее окно по части заголовка найти?
ActiveX контейнер для управляемого интерфейса системы 1С:Предприятие столкнулся с этой проблемой.
Я, насколько помню, их там просто нет. Заголовков в смысле. Когда писал компоненту