From maui-skills
Guides .NET MAUI Shell navigation including AppShell setup, tab bars, flyout menus, GoToAsync URI navigation, route registration, query parameters, back navigation, and events.
How this skill is triggered — by the user, by Claude, or both
Slash command
/maui-skills:maui-shell-navigationThe summary Claude sees in its skill listing — used to decide when to auto-load this skill
Always use `ContentTemplate` with `DataTemplate` so pages are created on demand.
Always use ContentTemplate with DataTemplate so pages are created on demand.
Using Content directly creates all pages during Shell init, hurting startup time.
<!-- ✅ Lazy — page created on first navigation -->
<ShellContent ContentTemplate="{DataTemplate views:HomePage}" />
<!-- ❌ Eager — page created at Shell startup -->
<ShellContent>
<views:HomePage />
</ShellContent>
IQueryAttributable gives you all parameters in one call and works on ViewModels:
public class AnimalDetailsViewModel : ObservableObject, IQueryAttributable
{
public void ApplyQueryAttributes(IDictionary<string, object> query)
{
if (query.TryGetValue("id", out var id))
AnimalId = id.ToString();
}
}
For complex objects, use ShellNavigationQueryParameters to avoid serializing:
var parameters = new ShellNavigationQueryParameters
{
{ "animal", selectedAnimal }
};
await Shell.Current.GoToAsync("animaldetails", parameters);
Use GetDeferral() for async checks (e.g., "save unsaved changes?"):
protected override async void OnNavigating(ShellNavigatingEventArgs args)
{
base.OnNavigating(args);
if (hasUnsavedChanges && args.Source == ShellNavigationSource.Pop)
{
var deferral = args.GetDeferral();
bool discard = await ShowConfirmationDialog();
if (!discard)
args.Cancel();
deferral.Complete();
}
}
Duplicate route names — Routing.RegisterRoute throws ArgumentException
if a route name is already registered or matches a visual hierarchy route.
Every route must be unique across the entire app.
Relative routes require registration — you cannot GoToAsync("somepage")
unless somepage was registered with Routing.RegisterRoute. Visual hierarchy
pages use absolute // routes instead.
Pages are created on demand — when using ContentTemplate, the page
constructor runs only on first navigation. Don't assume pages exist at startup.
Tab.Stack is read-only — you cannot manipulate the navigation stack directly;
use GoToAsync for all navigation changes.
GoToAsync is async — always await it — fire-and-forget navigation causes race conditions and can silently fail:
// ❌ Fire-and-forget — race conditions
Shell.Current.GoToAsync("details");
// ✅ Always await
await Shell.Current.GoToAsync("details");
Route hierarchy matters — absolute routes must match the full path through
the visual hierarchy (//FlyoutItem/Tab/ShellContent). Getting the path
wrong produces silent no-ops, not exceptions.
npx claudepluginhub davidortinau/maui-skills --plugin maui-skillsCreates .NET MAUI Shell pages, ViewModels, navigation, and source-generated routes with Shiny MAUI Shell. Useful for setting up navigation, tab badges, dialogs, and AI-driven route discovery.
Building .NET MAUI apps. Project structure, XAML/MVVM, platform services, current caveats.
Guides dependency injection in .NET MAUI apps: service registration, lifetimes (Singleton/Transient/Scoped), constructor injection, Shell navigation resolution, gotchas like XAML timing and unregistered pages.