Как стать автором
Обновить
0
«Кросс технолоджис»
Системный интегратор и разработчик ПО

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

Время на прочтение4 мин
Количество просмотров5.7K

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

В рамках первой статьи упоминалось, что данное приложение вызывается через пункт ПКМ, в связи с чем были проблемы при развертывании в разных ОС. Самое интересное заключалось не только в том, как реализовать вызов “визуальщины”, но и в том, как без сильных потерь реализовать диалоговое окно, которое мы сможем спокойно использовать в 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.

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

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

Теги:
Хабы:
Всего голосов 6: ↑5 и ↓1+7
Комментарии6

Публикации

Информация

Сайт
crosstech.ru
Дата регистрации
Дата основания
Численность
101–200 человек
Местоположение
Россия

Истории