Кроссплатформенная разработка мобильного приложения на .NET MAUI
.NET MAUI (Multi-platform App UI) — прямой наследник Xamarin.Forms, переписанный под .NET 6+. Главное архитектурное изменение: один проект вместо отдельных iOS, Android, WinUI, macOS targets — всё через единый csproj с условными зависимостями по платформе. Это означает один appsettings.json, один DI-контейнер (Microsoft.Extensions.DependencyInjection), одни модели.
Если команда уже работает с C# и ASP.NET Core, переход на MAUI технически минимален — паттерны те же, инструментарий тот же (Visual Studio 2022 / Rider), NuGet тот же. Для предприятий, где мобильное приложение является фронтендом к .NET-бэкенду, это часто единственный разумный выбор с точки зрения унификации команды.
Где возникают реальные проблемы
MAUI рендерит нативные контролы платформы через Handlers — в отличие от Xamarin.Forms с Renderers. На бумаге это чище. На практике: если стандартный Handler не поддерживает нужное свойство, пишем partial class с платформозависимым кодом и AppendToMapping. Это нормальный паттерн, но требует понимания нативных API каждой платформы.
Hot Reload работает с оговорками. XAML Hot Reload в Visual Studio 2022 17.6+ нормально работает для простых изменений разметки. Но изменения в C# code-behind или view model требуют перезапуска — .NET Hot Reload в MAUI ненадёжен при сложных иерархиях. На практике итерация по UI занимает больше времени, чем в Flutter или React Native.
Размер бинарника. Базовый APK .NET MAUI приложения без trimming — 30–50 МБ. С PublishTrimmed=true и RuntimeIdentifier=android-arm64 удаётся сжать до 15–25 МБ, но trimming ломает отражение (Reflection), на котором строится часть NuGet-пакетов. Каждый добавленный пакет нужно проверять на совместимость с AOT/trimming.
NativeAOT на iOS — с .NET 8 доступен. Даёт лучший startup time и размер, но требует полного отказа от runtime Reflection. Если проект использует System.Text.Json с source generators — окей. Если Newtonsoft.Json — миграция обязательна.
Стек и архитектура
MVVM — стандарт для MAUI. CommunityToolkit.Mvvm покрывает 90% потребностей: [ObservableProperty], [RelayCommand], [NotifyCanExecuteChangedFor] через source generators. Никакого бойлерплейта INotifyPropertyChanged вручную.
Shell-навигация (Shell) для стандартных случаев; для сложной навигации с передачей объектов — NavigationPage или ReactiveUI.Maui. Prism.Maui — если команда с ним знакома по WPF/UWP.
Сетевой слой — HttpClientFactory через DI, Refit для типизированных REST-клиентов, Polly для retry/circuit breaker. gRPC-клиент через Grpc.Net.Client работает на iOS и Android без дополнительных настроек начиная с .NET 7.
Локальная БД — SQLite-net-pcl для простых случаев, LiteDB для документ-ориентированного хранилища. Entity Framework Core с SQLite провайдером работает, но в trim-сценарии требует EFCore.TrimmingSupport.
Push-уведомления — Plugin.Firebase.CloudMessaging для FCM; APNs на iOS настраивается через UNUserNotificationCenter в MauiProgram.cs.
Кейс из практики. Корпоративное приложение для полевых инспекторов: оффлайн-работа с формами и фотографиями, синхронизация при появлении сети. Стек: MAUI + EF Core SQLite + Polly для retry очереди + Xamarin.Essentials (теперь Microsoft.Maui.Essentials) для доступа к камере и GPS. Платформозависимая часть — только работа с файловой системой (пути разные на iOS Documents vs Android GetExternalFilesDir). CI — Azure DevOps Pipelines с задачами dotnet publish под каждую платформу, подпись через KeyVault для Android keystore.
Интеграция с платформой
Доступ к нативным API — через Microsoft.Maui.Essentials: геолокация, акселерометр, Bluetooth, камера, биометрия (SecureStorage + Fingerprint). Если нужен доступ за пределами Essentials — DependencyService с платформозависимыми реализациями или прямой вызов нативного кода через #if ANDROID / #if IOS в partial классах.
Биндинги к нативным SDK: на Android — Java/Kotlin Binding Library через Binding Project; на iOS — ObjectiveC/Swift Binding через ApiDefinitions.cs. Процедура трудоёмкая, но отработанная.
Сроки
| Масштаб | Сроки |
|---|---|
| Корпоративный MVP, 8–12 экранов | 8–14 недель |
| Продукт с оффлайном и синхронизацией | 4–7 месяцев |
| Платформа с интеграцией корп. систем (SAP, 1C) | 6–12 месяцев |
Стоимость рассчитывается индивидуально после анализа требований и существующей .NET-инфраструктуры клиента.
На что обращаем внимание при аудите
Три вещи, которые видим чаще всего в унаследованных MAUI-проектах: отсутствие trimming-совместимости (приложение весит 60+ МБ), использование устаревших Renderer из Xamarin.Forms вместо перехода на Handler, и синхронные операции с БД на UI thread через EF Core без async/await. Последнее — гарантированный ANR на Android.







