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