По пути в Авалонию

от автора

Здравствуй дорогой наш читатель. Это третья статья из серии переезда нашего ПО на кроссплатформенные рельсы. Она затронет визуальную часть продукта, а именно диалоговое окно.

В рамках первой статьи упоминалось, что данное приложение вызывается через пункт ПКМ, в связи с чем были проблемы при развертывании в разных ОС. Самое интересное заключалось не только в том, как реализовать вызов “визуальщины”, но и в том, как без сильных потерь реализовать диалоговое окно, которое мы сможем спокойно использовать в Linux, а возможно в дальнейшем и в Windows. Для самых нетерпеливых забежим вперёд и скажем, что в данной задаче нам помог framework Avalonia. Ну, а теперь, по порядку.

Мы имеем WinForms приложение, работающее, разумеется, только в Windows. Его задача — выводить диалоговое окно с небольшим количеством контролов, фактически, главными из которых являются листбокс и кнопка.

Как раз только выходит .Net Core 3, с предположением о работе в Linux. Собираем, проверяем и ничего не получаем. В действительности в Linux не добавили визуальную часть. Ну что же, ищем другие варианты.

Самым часто встречающимся фреймворком по данному вопросу является Avalonia. Мы решили опробовать её. Взяли простейший пример, собрали в Linux, запустили и — да, диалоговое оно отобразилось. Теперь дело за малым — перенести WinForms проект на Avalonia.

В качестве проекта был выбран простой шаблон Avalonia Application. При условии, что логика и DTO уже вынесены в отдельные библиотеки, шаблон MVVM здесь был бы избыточен.

Перенос дизайна основного окна с WinForms на XAML прошёл без проблем, несмотря на то, что разработчик в знакомстве с WPF ранее замечен не был.

Логика осталась практически без изменений (BackgroundWorker, файлы, DTO…).

Реализация окон сообщений, на первый взгляд, также проблем не вызвала.

В проекте WinForms в качестве диалогового окна сообщения использовался System.Windows.Forms.MessageBox.

MessageBox.Show("Метка установлена", "Crosstech DSS",                  MessageBoxButtons.OK, MessageBoxIcon.Information);

Для нового проекта можно было использовать окно, производное от Avalonia.Controls.Window, предварительно оформленное как окно сообщения с иконками, текстом, стилями и кнопками, но…

в процессе изучения форумов выяснилось, что за нас уже подумали:). Существует проект MessageBox.Avalonia с иконками и вот этим вот всем. Проект немедленно был установлен.

Сначала это выглядело так:

private void Form_Load(object sender, EventArgs e) {   	...     _backgroundWorker.DoWork += DoLoadForm;     _backgroundWorker.RunWorkerAsync(); 		... }  private void DoLoadForm(object sender, DoWorkEventArgs e) { 		...	 		MessageBoxHelper.ShowErrorBox("Нет разрешённых действий для данного документа. Пожалуйста, обратитесь к администратору."); 		... } 	 internal static class MessageBoxHelper {   	public static async void ShowBox(MessageBoxStandardParams MBSparams)     {     	Action showMSBDialog = () => MessageBox.Avalonia.MessageBoxManager.GetMessageBoxStandardWindow(MBSparams).ShowDialog(new MainWindow());       await Dispatcher.UIThread.InvokeAsync(showMSBDialog);     } 	 		public static void ShowErrorBox(string message)     {     	var MBSparams = new MessageBoxStandardParams       {       	ButtonDefinitions = ButtonEnum.Ok,         ContentTitle = $"Ошибка доступа",         ShowInCenter = true,         ContentMessage = message,         Icon = MessageBox.Avalonia.Enums.Icon.Error,         Style = Style.Windows       };        ShowBox(MBSparams);     } 		... }

Окно сообщения всё-таки выводилось на экран. Для начала неплохо. Проблема была в том, что это окно сообщения существовало независимо от главного окна приложения, они все были доступны и закрывать их можно было в любом порядке, результат диалога не анализировался. Такое положение вещей нас более не устраивало, внесли небольшие изменения, на данном этапе получилось так:

private async void DoLoadForm(object sender, DoWorkEventArgs e) { 	var rezult = await MessageBoxHelper.ShowErrorBox("Нет разрешённых действий для данного документа. Завершить работу с документом?", this);    if (rezult == MessageBox.Avalonia.Enums.ButtonResult.Yes)   {   	Environment.Exit(0);   } 	... } 	 internal static class MessageBoxHelper { 	public static async Task<ButtonResult> ShowBox(MessageBoxStandardParams MBSparams, Window owner = null)   { 		if (owner == null) 		{ 			owner = new MainWindow(); 		}  		Func<Task<ButtonResult>> www = new Func<Task<ButtonResult>>(() => MessageBox.Avalonia.MessageBoxManager.GetMessageBoxStandardWindow(MBSparams).ShowDialog(owner)); 		ButtonResult result = await Dispatcher.UIThread.InvokeAsync(www); 		return result; 	} 		 	public static async Task<ButtonResult> ShowErrorBox(string message, Window owner = null)   {   	var MBSparams = new MessageBoxStandardParams     {     	ButtonDefinitions = ButtonEnum.YesNo,       ContentTitle = $"Ошибка доступа",       ShowInCenter = true,       ContentMessage = message,       Icon = MessageBox.Avalonia.Enums.Icon.Error,       Style = Style.Windows      };       return await ShowBox(MBSparams, owner);    } 	 ... }

Диалоговое окно выводится на переднем плане, не даёт переключить фокус на основное окно, возвращает результат. Уже лучше.

В процессе тестирования столкнулись с багом: на Windows 8 x86 при включенном масштабировании, наше окно отображается с чёрной рамкой. Меняли большинство параметров отображения в XAML, не помогло.

Хотим отдельно отметить прекрасную поддержку проекта Avalonia.

Ссылки на использованные источники:

               Группа в Telegram — (AvaloniaUI (RU)) — https://t.me/AvaloniaRU

               Статьи на Хабре:

                              https://habr.com/ru/post/438920/

                              https://habr.com/ru/company/skbkontur/blog/524518/

                              https://habr.com/ru/post/455896/

                              https://habr.com/ru/post/452818/

ссылка на оригинал статьи https://habr.com/ru/company/crosstech/blog/542172/


Комментарии

Добавить комментарий

Ваш адрес email не будет опубликован. Обязательные поля помечены *