From dotnet-wpf
Guia completo para migrar projetos WinForms para WPF com MVVM usando CommunityToolkit.Mvvm e WPF-UI. Cria ViewModels, configura DataBinding, Commands, navegacao e DI. Use quando o usuario quiser: migrar WinForms para WPF; adicionar MVVM a projeto WPF existente; criar ViewModel para uma tela; configurar navegacao WPF-UI com MVVM; substituir code-behind por bindings e commands; configurar DI com Microsoft.Extensions.Hosting em WPF; usar CommunityToolkit.Mvvm (ObservableProperty, RelayCommand); criar ObservableCollection para listas; implementar Messenger para comunicacao entre ViewModels. Tambem use quando o usuario mencionar "MVVM", "ViewModel", "data binding WPF", "migrar WinForms", "code-behind para commands", ou "navegacao WPF". NAO use para: setup inicial de projeto .NET (use dotnet-desktop-setup), testes unitarios, deploy, CI/CD, projetos web/API/mobile, ou configuracao de .editorconfig/CLAUDE.md.
How this skill is triggered — by the user, by Claude, or both
Slash command
/dotnet-wpf:dotnet-wpf-mvvmThe summary Claude sees in its skill listing — used to decide when to auto-load this skill
Skill para migrar projetos WinForms para WPF com MVVM e para construir novas telas WPF
Skill para migrar projetos WinForms para WPF com MVVM e para construir novas telas WPF seguindo o padrao MVVM moderno com CommunityToolkit.Mvvm + WPF-UI.
Usa progressive disclosure — este arquivo contem o workflow e decisoes. Templates,
exemplos de codigo e guias detalhados ficam em references/ e sao lidos sob demanda.
| Componente | Pacote NuGet | Funcao |
|---|---|---|
| MVVM Framework | CommunityToolkit.Mvvm | ObservableObject, source generators |
| DI + Lifecycle | Microsoft.Extensions.Hosting | IHost, IServiceProvider |
| UI Framework | WPF-UI (Wpf.Ui) | Fluent Design, NavigationView, Theming |
| Navegacao | Wpf.Ui.INavigationService | MVVM-friendly page navigation |
| Dialogs | Wpf.Ui.IContentDialogService | Substitui MessageBox |
Antes de aplicar MVVM, o projeto deve ter:
*Service.cs, nao em Forms/code-behindResult<T> ou lancam excecoesSe o projeto nao atende esses requisitos, use a skill dotnet-desktop-setup primeiro para
desacoplar e configurar. O MVVM funciona melhor quando os services ja existem — o ViewModel
simplesmente orquestra chamadas aos services e expoe dados para a View.
Execute os passos em ordem. Cada passo verifica o estado atual antes de agir.
Avalie o projeto para entender o ponto de partida:
# Verificar framework UI
grep -r "UseWPF\|UseWindowsForms" *.csproj
# Verificar se CommunityToolkit.Mvvm ja esta instalado
grep -r "CommunityToolkit.Mvvm" *.csproj
# Contar event handlers no code-behind (quanto trabalho tem pela frente)
grep -rn "_Click\|_Changed\|_Loaded\|_SelectionChanged" *.xaml.cs *.cs
# Verificar services existentes
find . -name "*Service.cs" -type f
# Verificar se tem MessageBox em services (anti-padrao)
grep -rn "MessageBox" --include="*Service.cs"
Apresente o relatorio ao usuario:
Adicione os pacotes necessarios via dotnet add:
dotnet add <projeto>.csproj package CommunityToolkit.Mvvm
dotnet add <projeto>.csproj package Microsoft.Extensions.Hosting
Se WPF-UI nao estiver instalado:
dotnet add <projeto>.csproj package WPF-UI
Verifique que o .csproj tem:
<UseWPF>true</UseWPF>
Leia references/wpfui-integration.md para o template completo de App.xaml.cs.
O App.xaml.cs deve:
IHost com Host.CreateDefaultBuilder()OnStartup, parar em OnExitPadrao de registro:
// Services de negocio
services.AddSingleton<ILicenseService, LicenseService>();
// ViewModels — Singleton para evitar memory leak quando assinam PropertyChanged
// de servicos Singleton (ver Detalhe #27)
services.AddSingleton<MainWindowViewModel>();
// Windows/Pages
services.AddTransient<MainWindow>();
Para apps simples (1 janela, sem navegacao entre paginas), o registro minimo e:
Leia references/communitytoolkit-patterns.md para patterns detalhados.
Para cada tela, crie um ViewModel seguindo este template:
using CommunityToolkit.Mvvm.ComponentModel;
using CommunityToolkit.Mvvm.Input;
namespace MeuProjeto.ViewModels;
public partial class MainWindowViewModel : ObservableObject
{
private readonly IMyService _service;
// Propriedades observaveis — o source generator cria a propriedade publica
[ObservableProperty]
private string _titulo;
[ObservableProperty]
private bool _isProcessando;
// Injecao de dependencia via construtor
public MainWindowViewModel(IMyService service)
{
_service = service;
}
// Commands — o source generator cria TituloCommand (IRelayCommand)
[RelayCommand]
private async Task CarregarDadosAsync()
{
IsProcessando = true;
try
{
var dados = await _service.ObterDadosAsync();
Titulo = dados.Nome;
}
finally
{
IsProcessando = false;
}
}
}
Regras criticas:
partial — source generators precisam disso[ObservableProperty] DEVEM ser private — _name gera propriedade Name[RelayCommand] geram propriedade com sufixo Command — Salvar() gera SalvarCommandIAsyncRelayCommand com cancelamento automaticoSubstitua event handlers por bindings e commands:
Antes (code-behind):
<Button Content="Carregar" Click="BtnCarregar_Click" />
<TextBox x:Name="txtNome" />
private void BtnCarregar_Click(object sender, RoutedEventArgs e)
{
txtNome.Text = _service.Carregar();
}
Depois (MVVM):
<Button Content="Carregar" Command="{Binding CarregarDadosCommand}" />
<TextBox Text="{Binding Titulo, UpdateSourceTrigger=PropertyChanged}" />
// Code-behind fica so com DI wiring
public MainWindow(MainWindowViewModel viewModel)
{
InitializeComponent();
DataContext = viewModel;
}
Mapeamento rapido de controles:
| WinForms / Code-behind | WPF MVVM |
|---|---|
button.Click += handler | Command="{Binding XCommand}" |
textBox.Text = valor | Text="{Binding Propriedade}" |
listBox.Items.Add(x) | ItemsSource="{Binding Lista}" + ObservableCollection<T> |
checkBox.Checked += handler | IsChecked="{Binding Flag}" |
comboBox.SelectedItem | SelectedItem="{Binding ItemSelecionado}" |
label.Content = texto | Content="{Binding Texto}" |
progressBar.Value | Value="{Binding Progresso}" |
control.Enabled = false | IsEnabled="{Binding PodeExecutar}" ou CanExecute no Command |
element.Visibility = Visible/Collapsed | Visibility="{Binding IsXxx, Converter={StaticResource BoolToVis}}" |
comboBox.SelectedItem (ComboBoxItem) | SelectionChanged handler ou SelectedValue="{Binding Prop}" com SelectedValuePath |
scrollViewer.ScrollToTop() | PropertyChanged handler no code-behind (excecao MVVM documentada) |
Mouse.OverrideCursor = Wait | [ObservableProperty] bool IsLoading + trigger ou converter no XAML |
Dialogs MVVM-friendly:
// Em vez de: MessageBox.Show("Erro")
// Use Microsoft.Win32 para file dialogs:
var dialog = new Microsoft.Win32.OpenFileDialog { Filter = "HID files|*.hid" };
if (dialog.ShowDialog() == true)
{
CaminhoArquivo = dialog.FileName;
}
Para dialogs mais complexos, use IContentDialogService do WPF-UI
(veja references/wpfui-integration.md).
Apos aplicar MVVM, verifique:
dotnet build deve compilar sem erros nem warnings de source generatorsdotnet test — todos os testes existentes devem passar (MVVM nao muda services).xaml.cs deve ter apenas:
InitializeComponent()DataContext = viewModel (ou atribuicao via DI)Cada migracao MVVM deve incluir testes de ViewModel. Eles sao a melhor forma de blindar o projeto contra regressoes durante refatoracoes — testam toda a logica de apresentacao sem abrir janelas, sao rapidos e confiaveis no CI.
| Aspecto | Exemplo |
|---|---|
| Estado inicial | Propriedades iniciam com valores default corretos |
| Commands executam | CarregarCommand.Execute() popula propriedades |
| CanExecute | Botao desabilitado quando pre-condicao nao e atendida |
| Validacao | Dados invalidos mostram erro, nao executam acao |
Commands que abrem OpenFileDialog nao sao testaveis unitariamente. Extraia a logica
para um metodo publico testavel:
// No ViewModel — o Command chama o dialog e depois o metodo testavel
[RelayCommand]
private void CarregarHardwareId()
{
var dialog = new Microsoft.Win32.OpenFileDialog { Filter = "*.hid" };
if (dialog.ShowDialog() == true)
PopularCampos(_service.RecuperarDeArquivo(dialog.FileName));
}
// Metodo publico testavel (sem dialog)
public void PopularCampos(HardwareInfo hwInfo)
{
CompanyName = hwInfo.CompanyName;
ProcessorId = hwInfo.ProcessorID;
// ...
IsSaveEnabled = true;
}
// No teste
[Fact]
public void PopularCampos_AtualizaPropriedadesEHabilitaSave()
{
var vm = new MainWindowViewModel(service);
vm.PopularCampos(new HardwareInfo { CompanyName = "JRC" });
Assert.Equal("JRC", vm.CompanyName);
Assert.True(vm.IsSaveEnabled);
}
Testes que acessam metodos privados via typeof(Page).GetMethod("NomeMetodo", BindingFlags.NonPublic)
quebrarao quando o metodo for movido do code-behind para o ViewModel. O typeof precisa ser
atualizado de typeof(MinhaPage) para typeof(MinhaPageViewModel). Identifique esses testes
ANTES de mover codigo — consulte a checklist pre-migracao.
Para smoke tests visuais em projetos com muitas telas, considere FlaUI
(framework de automacao UI para WPF). Veja TODO_SPECS/SPEC-Automated-UI-Testing.md
para o plano completo.
Este e o cenario mais comum — o projeto ja e WPF mas usa event handlers diretamente. Execute todos os 6 passos. O Passo 4 e o mais trabalhoso: extrair logica dos event handlers para ViewModels.
Leia references/migration-winforms-to-wpf.md antes de comecar.
A migracao acontece em duas fases:
Migre form-a-form usando Strangler Fig pattern. Nao migre tudo de uma vez.
Leia references/project-structure.md para a estrutura de pastas recomendada.
Crie a estrutura Models/Views/ViewModels/Services antes de comecar a codar.
Comece pelo Passo 2 (pacotes), pule para Passo 3 (DI), depois crie ViewModels e Views.
Leia references/wpfui-integration.md secao sobre NavigationView.
Use INavigationService + IPageService do WPF-UI para navegacao DI-friendly.
Classes DEVEM ser partial — source generators do CommunityToolkit exigem partial class.
Sem partial, [ObservableProperty] e [RelayCommand] nao geram codigo e o build falha.
Campos [ObservableProperty] devem ser private — o generator cria a propriedade publica
a partir do nome do campo: _nomeDoNavio gera NomeDoNavio. Se o campo for publico, conflita.
Nao misturar dialogs WinForms e WPF — em projetos WPF, usar Microsoft.Win32.OpenFileDialog
e Microsoft.Win32.SaveFileDialog, nao os equivalentes de System.Windows.Forms.
ObservableCollection<T> nao precisa de [ObservableProperty] — declare como propriedade
publica simples: public ObservableCollection<Item> Items { get; } = new();. A collection ja
implementa INotifyCollectionChanged internamente.
Atualizar CLAUDE.md apos migrar — referencias a Form*.cs ficam desatualizadas apos migracao. Atualizar descricao do stack, nomes de arquivos UI, e tabela de projetos.
CanExecute com [RelayCommand] — para habilitar/desabilitar botoes automaticamente,
use [RelayCommand(CanExecute = nameof(PodeSalvar))] e chame
SalvarCommand.NotifyCanExecuteChanged() quando a condicao mudar.
Async commands cancelam automaticamente — se o metodo retorna Task, o [RelayCommand]
gera IAsyncRelayCommand que desabilita o botao durante execucao e suporta cancelamento.
Inicializar campos string com = string.Empty — campos [ObservableProperty] do tipo
string devem ser inicializados: private string _nome = string.Empty;. Sem isso, bindings
podem receber null e causar warnings ou comportamento inesperado.
StatusMessage como alternativa a MessageBox — para apps simples (1-2 telas), substituir
MessageBox.Show() por atualizar uma propriedade StatusMessage no ViewModel e exibi-la
na barra de status e mais simples e testavel que criar IDialogService. Reservar
IContentDialogService do WPF-UI para apps com multiplas telas ou dialogs complexos.
Icone da aplicacao em FluentWindow — nao usar Icon= no XAML nem BitmapImage no
code-behind (ambos carregam o menor frame do .ico e ficam pixelados). Usar BitmapDecoder
para selecionar o frame de maior resolucao. Declarar o .ico como <ApplicationIcon> E
<Resource> no .csproj. Veja references/wpfui-integration.md secao "Icone da Aplicacao".
ui:Page NAO existe no WPF-UI 4.2.0 — usar <Page> padrao do WPF
(namespace System.Windows.Controls). Code-behind herda Page, NAO INavigableView<T>.
INavigationViewPageProvider esta em Wpf.Ui.Abstractions — NAO em Wpf.Ui nem
Wpf.Ui.Controls. Metodo: GetPage(Type pageType) retorna object?.
MessageBoxButton conflita com WPF-UI — quando ambos namespaces sao usados, adicionar
alias: using MessageBoxButton = System.Windows.MessageBoxButton; e
using MessageBoxImage = System.Windows.MessageBoxImage;.
NAO importar Wpf.Ui.Controls globalmente — causa conflitos com MessageBoxButton,
Page, etc. Qualificar tipos WPF-UI individualmente:
public partial class MainWindow : Wpf.Ui.Controls.FluentWindow.
PageService deve ser criado manualmente — WPF-UI nao fornece implementacao built-in de
INavigationViewPageProvider. Criar classe PageService(IServiceProvider sp) com
GetPage(Type) => sp.GetService(pageType). Setup:
RootNavigation.SetPageProviderService(pageProvider) (NAO SetPageService()).
NavigationView action items: usar PreviewMouseLeftButtonUp — no WPF-UI 4.2.0,
nem ItemInvoked nem SelectionChanged disparam para NavigationViewItems sem
TargetPageType (items de acao como Browse/Upload). Usar PreviewMouseLeftButtonUp
diretamente no NavigationViewItem com e.Handled = true.
DataGrid: usar AutoGenerateColumns="False" — definir colunas explicitamente em XAML
para controlar visibilidade, headers e formatacao. Colunas que nao devem aparecer simplesmente
nao sao declaradas (mais limpo que Visibility="Collapsed" em cada coluna).
NavigationView quebra virtualizacao — o NavigationView do WPF-UI internamente usa layout
que da altura infinita as paginas. Qualquer ListBox/DataGrid/ListView dentro de uma Page
recebe ActualHeight infinito e renderiza TODOS os items (virtualizacao desabilitada).
Fix obrigatorio: usar MaxHeight fixo + Page_SizeChanged para ajustar dinamicamente:
private void Page_SizeChanged(object sender, SizeChangedEventArgs e)
{
if (dgvLog != null && e.NewSize.Height > 100)
dgvLog.MaxHeight = e.NewSize.Height - 120;
}
Singleton para paginas pesadas — Pages registradas como Transient sao recriadas a cada
navegacao (visual tree, bindings, tudo reconstruido). Para paginas com dados grandes, registrar
como Singleton no DI evita reconstrucao e mantem estado de scroll/filtro. Adicionar metodo
ReloadData() para resetar quando novos dados sao carregados.
WindowBackdropType="None" com WindowsFormsHost — Mica habilita transparencia
internamente, o que torna controles WinForms (via WindowsFormsHost) invisiveis. Bug
documentado pela Microsoft. Usar WindowBackdropType="None" se WindowsFormsHost for necessario.
Page.Resources ANTES do conteudo — declarar <Page.Resources> com Styles/converters
ANTES do conteudo XAML (DockPanel, Grid, etc). Se declarado depois, StaticResource falha
com erro "StaticResourceExtension" em runtime.
SolidColorBrush.Freeze() — brushes estaticos devem ser frozen para thread-safety:
private static SolidColorBrush CreateFrozenBrush(byte r, byte g, byte b)
{
var brush = new SolidColorBrush(Color.FromRgb(r, g, b));
brush.Freeze();
return brush;
}
LINQ filter em POCOs em vez de DataView.RowFilter — para listas grandes (100K+),
converter DataTable para List<T> tipado em background e filtrar com LINQ e mais rapido
e thread-safe que DataView.RowFilter (que usa reflexao e nao e thread-safe).
Debounce para filtros — em TextBoxes de filtro, usar debounce de 300ms com
CancellationTokenSource para filtrar enquanto o usuario digita sem travar a UI:
_filterCts?.Cancel();
_filterCts?.Dispose();
_filterCts = new CancellationTokenSource();
_ = Task.Delay(300, _filterCts.Token).ContinueWith(t => {
if (!t.IsCanceled) Dispatcher.Invoke(ApplyFilter);
});
Lazy property caching com ??= — para propriedades formatadas chamadas repetidamente pelo binding (ex: DateTimeFormatted), usar lazy initialization para evitar ToString() em cada frame de renderizacao:
private string? _formatted;
public string Formatted => _formatted ??= DateTime.ToString("dd/MM/yyyy HH:mm:ss");
IReadOnlyList para caches — expor caches estaticos como IReadOnlyList<T> em vez de
List<T> para prevenir modificacao acidental por consumidores.
Lifecycle mismatch: Transient VM + Singleton Service = memory leak — se um ViewModel
registrado como Transient assina PropertyChanged de um servico Singleton (ex: IAppStateService),
cada navegacao cria uma nova instancia que nunca e dessubscrita. O Singleton mantem delegate
references para instancias mortas, impedindo o GC. Em apps com NavigationView, onde paginas sao
recriadas a cada navegacao, isso causa leak cumulativo. Fix preferido: registrar ViewModels
como Singleton (consistente com Detalhe #19 sobre paginas pesadas). Alternativas: implementar
IDisposable com unsubscribe, ou usar WeakEventManager (mas este requer System.Windows
que viola a separacao ViewModel/UI).
Visibility bindings esquecidos ao migrar handlers — ao converter Click handlers que
alternavam Visibility de paineis para Commands no ViewModel, e comum criar as propriedades
IsXxxVisible no VM mas esquecer de adicionar Visibility="{Binding IsXxxVisible, Converter={StaticResource BoolToVis}}" no XAML. O resultado e que os Commands executam mas
nada muda visualmente. Sempre auditar o XAML apos converter handlers de visibilidade.
System.Windows ou acessar
controles da View. Use bindings e Messenger para tudo.new ViewModel() no XAML — funciona, mas impede DI. Prefira injetar via construtor.TextBox default e LostFocus. Use
UpdateSourceTrigger=PropertyChanged para validacao em tempo real.List<T> em vez de ObservableCollection<T> — List nao notifica a View quando
itens sao adicionados/removidos. Sempre use ObservableCollection para listas bindadas.List<T> tipado ou copiar o DataTable antes de filtrar. DefaultView compartilhado entre
consumidores causa race conditions.SignalStrength24, PlugConnected24. Testar
em runtime antes de commitar.async void em metodos que nao sao event handlers — metodos async void fora de
handlers UI (Click, Loaded) causam excecoes nao-observadas que podem crashar a aplicacao.
Sempre usar async Task e await no chamador. [RelayCommand] gera IAsyncRelayCommand
que ja usa async Task internamente — nunca converter para async void.using Wpf.Ui.Controls; global — conflita com System.Windows.Controls (TextBox,
ComboBox, Page, Button). Usar type aliases: using ControlAppearance = Wpf.Ui.Controls.ControlAppearance;<ui:ContentDialogHost>, nao
<ui:ContentPresenter> ou <ui:ContentDialogPresenter>. Erro comum que causa crash.ShowSimpleDialogAsync e extension method em
Wpf.Ui.Extensions. Requer using Wpf.Ui.Extensions; no arquivo.new Service() dentro do ViewModel — ViewModel NAO deve instanciar servicos diretamente.
Use injecao de construtor. Se o service e thin wrapper (ex: new UsuariosServicos(repo)), injete
a interface subjacente diretamente (IUsuariosRepositorio) e chame _repo.Salvar(). Instanciar
services no VM impede mocking nos testes e viola o principio de inversao de dependencias.try/catch
com MessageBox.Show() no catch. Ao migrar para [RelayCommand], e facil esquecer o error path.
O resultado e que falhas sao engolidas silenciosamente (o usuario nao recebe feedback). Sempre
preservar error handling: use StatusMessage property ou IContentDialogService no catch.Antes de migrar cada Page para MVVM, audite o code-behind e verifique:
element.Visibility = Visible/Collapsed → precisara de
binding com BooleanToVisibilityConverter. Facil de esquecer (ver Detalhe #28)SelectionChanged handler que atualiza o ViewModeltry/catch com MessageBox → preservar no ViewModel com
StatusMessage ou IContentDialogService (nao remover silenciosamente)GetValue()/SetValue()/SetDate() sem
DependencyProperties → nao suportam binding (ver secao Custom Controls abaixo)ScrollToTop(), Focus(), Mouse.OverrideCursor → manter em
code-behind como excecao documentada (SC-002 exception)typeof(Page).GetMethod() para metodos privados
quebrarao quando o metodo for movido para o ViewModel. Atualizar typeof apos moverPara apps com multiplas paginas que compartilham estado (ex: dados carregados, modo de operacao,
filtros ativos), um servico Singleton com INotifyPropertyChanged e mais simples e direto que
IMessenger (WeakReferenceMessenger):
public interface IAppStateService : INotifyPropertyChanged
{
VDR? Vdr { get; }
bool IsVdrLoaded { get; }
bool ModoCoCAtivado { get; }
string SelectedPath { get; }
void CarregarVdr(VDR vdr, string path, bool modoCoc);
}
Quando usar IAppStateService vs IMessenger:
| Cenario | Padrao |
|---|---|
| Estado central que multiplos VMs leem | IAppStateService (Singleton + INotifyPropertyChanged) |
| Evento pontual entre VMs sem estado | IMessenger (WeakReferenceMessenger) |
| Notificacao de navegacao | IMessenger |
| Dados de sessao (usuario logado, modo) | IAppStateService |
Regra critica: se ViewModels assinam PropertyChanged de um servico Singleton, registrar
os VMs tambem como Singleton para evitar memory leak (ver Detalhe #27).
Testabilidade: IAppStateService e facilmente mockavel com NSubstitute:
var appState = Substitute.For<IAppStateService>();
appState.IsVdrLoaded.Returns(true);
appState.Vdr.Returns(new VDR1800());
var vm = new ChannelsPageViewModel(appState);
Se o projeto usa UserControls custom (ex: controles de formulario especializados como APTCheckBoxWPF, AptDateWPF), audite ANTES de planejar a migracao:
Verificar DependencyProperties — o controle expoe DP para seu valor principal?
grep -r "DependencyProperty" VDAControls/WPF/
Se retorna vazio, o controle nao suporta data binding.
API imperativa = sem binding — se o controle usa GetValue()/SetValue()/SetDate()/GetDay()
em vez de DependencyProperties, data binding bidirecional e impossivel.
Abordagem pragmatica para migracao:
Adicionar DependencyProperties (spec separada) — cada controle precisa de pelo menos uma DP para seu valor principal. Exemplo para um checkbox custom:
public static readonly DependencyProperty ValueProperty =
DependencyProperty.Register(nameof(Value), typeof(string), typeof(APTCheckBoxWPF),
new FrameworkPropertyMetadata(string.Empty, FrameworkPropertyMetadataOptions.BindsTwoWayByDefault,
OnValueChanged));
Leia estes arquivos somente quando necessario no passo correspondente:
| Arquivo | Leia quando... |
|---|---|
references/mvvm-fundamentals.md | Usuario e novo em MVVM ou quer entender conceitos |
references/communitytoolkit-patterns.md | Passo 4 — criando ViewModels com source generators |
references/wpfui-integration.md | Passo 3 — configurando DI, navegacao e theming com WPF-UI |
references/migration-winforms-to-wpf.md | Projeto e WinForms e precisa migrar para WPF |
references/project-structure.md | Criando projeto do zero ou reorganizando pastas |
npx claudepluginhub j0ruge/skills_commands_manager --plugin dotnet-wpfProvides guidelines for MVVM in WPF using CommunityToolkit.Mvvm 8.4+ with ObservableProperty attributes, source generators, ViewModels, commands, and project structure.
Migrate WPF apps to WinUI 3: namespace replacement, control mapping, threading, imaging, MVVM conversion. Use for converting WPF code or fixing migration build errors.
Building WPF on .NET 8+. Host builder, MVVM Toolkit, Fluent theme, performance, modern C# patterns.