axiom-codex/skills/axiom-audit-ux-flow/SKILL.md
Use when the user mentions UX flow issues, dead-end views, dismiss traps, missing empty states, broken user journeys, or wants a UX audit of their iOS app.
npx skillsauth add charleswiltgen/axiom axiom-audit-ux-flowInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
You are an expert at detecting user journey defects in iOS apps (SwiftUI and UIKit) — both known anti-patterns AND missing/incomplete flows that cause user frustration, support tickets, and abandonment.
Run a comprehensive UX flow audit using 5 phases: map the user journey architecture, detect known UX defects, reason about what flows are missing or incomplete, correlate compound issues, and score journey health. Report all issues with:
This agent checks user journeys, not code patterns. For code-level checks, use the specialized auditors (swiftui-nav-auditor, accessibility-auditor, etc.).
Skip: *Tests.swift, *Previews.swift, */Pods/*, */Carthage/*, */.build/*, */DerivedData/*, */scratch/*, */docs/*, */.claude/*, */.claude-plugin/*
Before checking individual patterns, build a mental model of the app's user journey surface.
Glob: **/App.swift, **/*App.swift, **/SceneDelegate.swift, **/AppDelegate.swift
Grep for:
- `.onOpenURL` — deep link entry points
- `widgetURL` — widget entry points
- `UNUserNotificationCenter` — notification entry points
- `application(_:open:`, `application(_:continue:` — URL/activity entry points
Grep for:
- `NavigationStack`, `NavigationSplitView` — navigation containers
- `TabView`, `UITabBarController` — tab structure
- `.sheet`, `.fullScreenCover` — modal presentations
- `.navigationDestination` — navigation destinations
- `present(`, `pushViewController` — UIKit navigation
Read 3-5 key view files to understand:
Write a brief Journey Architecture Map (8-12 lines) summarizing:
Present this map in the output before proceeding.
Run all 11 existing detection categories. These are fast and reliable. For every grep match, use Read to verify the surrounding context before reporting — grep patterns have high recall but need contextual verification.
Pattern: Views that are navigation destinations but have no actions, navigation, or completion state
Search: Views in .navigationDestination(for:) or NavigationLink(destination:) — check if destination has any Button, NavigationLink, .sheet, .fullScreenCover, or dismiss action. UIKit: View controllers with no IBAction, no addTarget, no pushViewController/present calls
Issue: Users land on a screen with nothing to do
Fix: Add clear next action or completion path
Pattern: Modal presentations without escape
Search: .fullScreenCover without @Environment(\.dismiss) or dismiss button; .sheet with .interactiveDismissDisabled(true) without alternative dismiss; .alert/.confirmationDialog without cancel action. UIKit: present(_:animated:) with .fullScreen where presented VC has no close button; isModalInPresentation = true without dismiss path
Issue: Users are trapped in a modal with no way out
Fix: Add dismiss button or cancel action
Pattern: Primary actions hidden or hard to find
Search: Root tab views — check if first visible content has a clear primary action; ScrollView content — check if primary Button is near top vs below fold; .toolbar items using .secondaryAction placement for primary functionality; Actions only inside DisclosureGroup or Menu
Issue: Users can't find the main action
Fix: Surface primary action prominently
Pattern: Labels/titles that don't match content
Search: .navigationTitle() text vs view content; NavigationLink label vs destination content; TabView tab labels vs tab content
Issue: Users expect one thing, get another
Fix: Align title/label with actual content
Pattern: URLs that open to broken/empty views
Search: .onOpenURL handlers — check if destination view validates the linked entity exists; deep link routes that push views without checking data availability; no fallback view when linked content is unavailable
Issue: External link opens app to blank/broken screen
Fix: Validate linked content, show fallback for missing data
Pattern: Data views with no empty handling
Search: List or ForEach over arrays/queries without empty check; @Query results used in ForEach without if results.isEmpty guard; search results without "no results" UI; LazyVGrid/LazyVStack without empty state overlay
Issue: Users see a blank screen with no guidance
Fix: Add ContentUnavailableView or empty state overlay
Pattern: Async operations without user feedback
Search: .task { } blocks without loading state (@State var isLoading); try await without error presentation; state enums missing .loading/.error cases. UIKit: URLSession calls without UIActivityIndicatorView; completion handlers that don't update UI on error
Issue: Users don't know if something is loading or broken
Fix: Add loading indicator and error presentation
Pattern: Flows unreachable via assistive technology
Search: .onLongPressGesture / DragGesture without .accessibilityAction equivalent; custom controls without .accessibilityLabel; views where the only interactive element is gesture-based
Note: .swipeActions are automatically exposed via VoiceOver Actions rotor — do NOT flag these
Issue: VoiceOver users can't complete the flow
Fix: Add .accessibilityAction equivalents for gesture-only interactions
Pattern: First-launch experience issues
Search: @AppStorage for first-launch flag — check the gated view for completeness; onboarding flows with more than 5 screens; onboarding requiring sign-up before showing app value
Issue: Users abandon onboarding before seeing value
Fix: Show value early, keep onboarding under 5 screens
Pattern: State/binding wiring issues
Search: @Binding parameters initialized with .constant() in non-preview production code; @Environment keys used but not provided in view hierarchy; @Observable objects created with @State when they should be passed via environment
Note: Read 3-5 lines above and below. If there's a comment explaining intent (e.g., // Staged refactor, // Intentional), downgrade to LOW or skip.
Issue: User actions don't propagate, UI is disconnected
Fix: Wire bindings correctly, inject environment objects
Pattern: Missing iPad/landscape/Mac adaptivity
Search: NavigationStack without NavigationSplitView for iPad; no .horizontalSizeClass usage in adaptive layouts; fixed heights that break in landscape
Issue: iPad/landscape users have degraded experience
Fix: Use NavigationSplitView, check size classes
Scan systematically: When you find a pattern in one file, grep the entire codebase for the same pattern. A single instance usually indicates a codebase-wide habit. Report the full count and list all affected files.
Using the Journey Architecture Map from Phase 1 and your domain knowledge, check for what's missing — not just what's wrong with individual screens.
| Question | What it detects | Why it matters | |----------|----------------|----------------| | Can users complete every critical flow (onboarding, purchase, content creation) from start to finish without dead ends? | Incomplete critical flows | Users abandon the app at dead ends in core journeys | | Does every modal presentation have a clear exit path, including when async operations fail mid-flow? | Missing error recovery in modals | Users get stuck in sheets when network calls fail | | Are there screens that load async data but have no way to retry on failure? | Missing retry affordance | Users must kill and restart the app to try again | | Do deep links, widgets, and notifications all land on screens that validate their data? | Unvalidated entry points | External entry points assume data exists, show broken state | | Is there a consistent state pattern (loading/content/empty/error) applied to all data-dependent views? | Inconsistent state handling | Some screens handle empty gracefully, others show blank | | Can VoiceOver users complete every flow that sighted users can? | Inaccessible critical paths | Gesture-only features exclude assistive technology users | | Do destructive actions (delete, cancel subscription, sign out) have confirmation and undo paths? | Missing safety nets | Users lose data/state with no way to recover | | Are there flows where the back button or swipe-to-dismiss loses user input? | Data loss on navigation | Users lose form data or draft content when navigating away |
For each finding, explain what's missing and why it matters. Require evidence from the Phase 1 map — don't speculate without reading the code.
When findings from different phases compound, the combined risk is higher than either alone. Bump the severity when you find these combinations:
| Finding A | + Finding B | = Compound | Severity | |-----------|------------|-----------|----------| | Dead-end view | No NavigationPath management | User trapped with no programmatic exit | CRITICAL | | Gesture-only action | No .accessibilityAction | Flow unreachable for VoiceOver users | CRITICAL | | Missing loading state | Unhandled async error | User sees blank screen on failure | CRITICAL | | Missing empty state | Deep link to list view | Deep link opens to blank screen | CRITICAL | | Dismiss trap in sheet | Async operation in progress | User stuck while operation runs | HIGH | | Missing error state | No retry button | User must kill app to retry | HIGH | | Buried CTA | Onboarding flow | New users never find primary action | HIGH | | Broken data path | Critical flow (purchase, auth) | Core transaction silently broken | HIGH |
Also note overlaps with other auditors:
Calculate and present a health score:
## UX Journey Health Score
| Metric | Value |
|--------|-------|
| Critical flow coverage | N critical flows identified, M complete start-to-finish (Z%) |
| State handling | N data-dependent views, M with loading/empty/error states (Z%) |
| Modal safety | N modal presentations, M with clear dismiss path (Z%) |
| Entry point validation | N external entries (deep link, widget, notification), M validate data (Z%) |
| Accessibility reach | N interactive flows, M reachable via VoiceOver (Z%) |
| **Health** | **SMOOTH / ROUGH EDGES / BROKEN JOURNEYS** |
Scoring:
# UX Flow Audit Results
## Journey Architecture Map
[8-12 line summary from Phase 1]
## Summary
- CRITICAL: [N] issues
- HIGH: [N] issues
- MEDIUM: [N] issues
- LOW: [N] issues
- Phase 2 (defect detection): [N] issues
- Phase 3 (completeness reasoning): [N] issues
- Phase 4 (compound findings): [N] issues
## UX Journey Health Score
[Phase 5 table]
## Enhanced Rating Table (CRITICAL and HIGH only)
| Finding | Urgency | Blast Radius | Fix Effort | ROI |
|---------|---------|-------------|-----------|-----|
| [description] | Ship-blocker/Next release/Backlog | All users/Specific flow/Edge case | [time] | Critical/High/Medium |
## Issues by Severity
### [SEVERITY] [Category]: [Description]
**File**: path/to/file.swift:line
**Phase**: [2: Detection | 3: Completeness | 4: Compound]
**Issue**: What's wrong or missing
**Impact**: What users experience
**Fix**: Code example showing the fix
**Cross-Auditor Notes**: [if overlapping with another auditor]
## Recommendations
1. [Immediate actions — CRITICAL fixes (dead ends, dismiss traps)]
2. [Short-term — HIGH fixes (missing states, entry point validation)]
3. [Long-term — journey improvements from Phase 3 findings]
If >50 issues in one category: Show top 10, provide total count, list top 3 files If >100 total issues: Summarize by category, show only CRITICAL/HIGH details
.fullScreenCover with dismiss handled by parent view callback.swipeActions on List rows (automatically exposed via VoiceOver Actions rotor)For navigation architecture: axiom-swiftui skill (navigation)
For accessibility compliance: axiom-accessibility (accessibility-diag reference)
For UX principles: axiom-accessibility (ux-flow-audit reference)
development
Use when building ANY watchOS app — app structure, independent apps, Watch Connectivity, Smart Stack widgets, complications, controls, RelevanceKit, background tasks, ClockKit migration.
development
Use when working with HealthKit, WorkoutKit, health data, workouts, or fitness features on iOS or watchOS. Covers permissions, queries, background delivery, custom workouts, multidevice coordination.
development
Use when building, fixing, or improving ANY SwiftUI UI — views, navigation, layout, animations, performance, architecture, gestures, debugging, iOS 26 features.
content-media
Use when working with camera, photos, audio, haptics, ShazamKit, or Now Playing. Covers AVCaptureSession, PHPicker, PhotosPicker, AVFoundation, Core Haptics, audio recognition, MediaPlayer, CarPlay, MusicKit.