view-refactor
$
npx mdskill add openai/plugins/view-refactorRefactor macOS views into stable, explicit scene structures.
- Splits large views into meaningful scene roots and reduces complexity.
- Uses native SwiftUI for layout while limiting AppKit to edge cases.
- Orders code by environment, properties, and body for predictable shapes.
- Restricts single-file usage to small examples under fifty lines.
SKILL.md
.github/skills/view-refactorView on GitHub ↗
--- name: view-refactor description: Refactor macOS SwiftUI views and scenes into stable structure. Use when splitting large views, tightening scene state, or narrowing AppKit escapes. --- # View Refactor ## Overview Refactor macOS views toward small, explicit, stable scene and view types. Default to native SwiftUI for layout, selection, commands, and settings. Reach for AppKit only at the narrow edges where desktop behavior truly requires it. ## Core Guidelines ### 1) Model scenes explicitly - Break the app into meaningful scene roots: main window, settings, utility windows, inspectors, or menu bar extras. - Do not let one giant root view silently own every desktop surface. ### 2) Keep a predictable file shape - Follow this ordering unless the file already has a stronger local convention: - Environment - `private`/`public` `let` - `@State` / other stored properties - computed `var` (non-view) - `init` - `body` - computed view builders / other view helpers - helper / async functions ### 2b) Split files by responsibility - For non-trivial apps, do not keep the full app, all views, models, stores, networking clients, process clients, and helpers in one Swift file. - Accept a single Swift file only for tiny throwaway examples or snippets: roughly under 50 lines, one screen, no persistence, no networking/process client, and no reusable models. - Use `App/<AppName>App.swift` for the `@main` app and `AppDelegate` only. - Keep `Views/ContentView.swift` focused on root layout and composition; move feature UI into files such as `Views/SidebarView.swift`, `Views/DetailView.swift`, and `Views/ComposerView.swift`. - Move value types and selection enums into `Models/*.swift`, stores into `Stores/*.swift`, app-server/network/process clients into `Services/*.swift`, and small formatters/resolvers/extensions into `Support/*.swift`. - Keep files small and named after the primary type they contain. ### 3) Prefer dedicated subview types over many computed `some View` fragments - Extract meaningful desktop sections like sidebar rows, detail panels, inspectors, or toolbar content into focused subviews. - Keep computed `some View` helpers small and rare. - Pass explicit data, bindings, and actions into subviews instead of handing down the whole scene model. ### 4) Keep selection and layout stable - Prefer one stable split or window layout with local conditionals inside it. - Avoid top-level branch swapping between radically different roots when selection changes. - Let the layout be constant; let state drive the content inside it. ### 5) Extract commands, toolbars, and actions out of `body` - Do not bury non-trivial button logic inline. - Do not mix command routing, menu state, and layout in the same block if they can be named clearly. - Keep `body` readable as UI, not as a desktop view controller. ### 6) Use scene and app storage intentionally - Use `@SceneStorage` for per-window ephemeral state when it truly helps restore the scene. - Use `@AppStorage` for durable preferences, not transient UI toggles that only matter in one window. - Keep scene-owned state close to the scene root. ### 7) Keep AppKit escape hatches narrow - If a representable or `NSWindow` bridge exists, isolate it behind a small wrapper or helper. - Do not let AppKit references spread through unrelated SwiftUI views. - If the bridge starts owning the feature, re-evaluate the architecture. ### 8) Observation usage - For `@Observable` reference types on modern macOS targets, store them as `@State` in the owning view. - Pass observables explicitly to children. - On older deployment targets, fall back to `@StateObject` and `@ObservedObject` where needed. ## Workflow 1. Identify the current scene boundary and whether the file is trying to do too much. 2. Reorder the file into a predictable top-to-bottom structure. 3. Extract desktop-specific sections into dedicated subview types. 4. Stabilize the root layout around selection, scenes, and commands rather than top-level branching. 5. Move action logic, command routing, and toolbar behavior into named helpers or separate types. 6. Tighten any AppKit bridge so the imperative edge is small and explicit. 7. Keep behavior intact unless the request explicitly asks for structural and behavioral changes together. ## Refactor Checklist - Split oversized view files before adding more UI. - Move pure models, identifiers, and selection enums out of view files. - Move `Process`, `URLSession`, app-server, and platform client code out of SwiftUI views into `Services/`. - Keep `AppDelegate` and the `@main` app entrypoint minimal. - Build after each major split so compile errors stay local. ## Common Smells - A root view that mixes window scaffolding, settings, toolbar code, command handling, and detail layout. - A single app file that mixes app entrypoint, root layout, feature views, models, stores, service clients, and support extensions. - iOS-style push navigation forced into a Mac sidebar-detail problem. - Several booleans for mutually exclusive inspectors, sheets, or utility windows. - AppKit objects passed through many SwiftUI layers without a clear ownership reason. - Large computed view fragments standing in for real subviews. ## Notes - A good macOS refactor should make scene structure, selection flow, and command ownership obvious. - When the problem is fundamentally a missing desktop pattern, use `swiftui-patterns`. - When the problem is fundamentally a boundary with AppKit, use `appkit-interop`.
More from openai/plugins
- accessibility-and-inclusive-visualizationMake data visualizations accessible and inclusive. Use when the user needs chart or diagram accessibility guidance, text alternatives for complex visuals, color and contrast review, keyboard support, reduced-motion behavior for animation or parallax, or an accessibility QA workflow for exported figures, UML-like diagrams, and dashboards.
- agent-browserBrowser automation CLI for AI agents. Use when the user needs to interact with websites, verify dev server output, test web apps, navigate pages, fill forms, click buttons, take screenshots, extract data, or automate any browser task. Also triggers when a dev server starts so you can verify it visually.
- agent-browser-verifyAutomated browser verification for dev servers. Triggers when a dev server starts to run a visual gut-check with agent-browser — verifies the page loads, checks for console errors, validates key UI elements, and reports pass/fail before continuing.
- agents-sdkBuild AI agents on Cloudflare Workers using the Agents SDK. Load when creating stateful agents, durable workflows, real-time WebSocket apps, scheduled tasks, MCP servers, or chat applications. Covers Agent class, state management, callable RPC, Workflows integration, and React hooks. Biases towards retrieval from Cloudflare docs over pre-trained knowledge.
- ai-elementsAI Elements component library guidance — pre-built React components for AI interfaces built on shadcn/ui. Use when building chat UIs, message displays, tool call rendering, streaming responses, reasoning panels, or any AI-native interface with the AI SDK.
- ai-gatewayVercel AI Gateway expert guidance. Use when configuring model routing, provider failover, cost tracking, or managing multiple AI providers through a unified API.
- ai-generation-persistenceAI generation persistence patterns — unique IDs, addressable URLs, database storage, and cost tracking for every LLM generation
- ai-sdkVercel AI SDK expert guidance. Use when building AI-powered features — chat interfaces, text generation, structured output, tool calling, agents, MCP integration, streaming, embeddings, reranking, image generation, or working with any LLM provider.
- aiq-deploy|
- aiq-research|