name: ios-dev-guidelines description: Context-aware routing to Swift/iOS development patterns, architecture, and best practices. Use when working with .swift files, ViewModels, Coordinators, refactoring, or discussing Swift/SwiftUI patterns.
iOS Development Guidelines (Smart Router)
Purpose
Context-aware routing to iOS development patterns, code style, and architecture guidelines. This skill provides critical rules and points you to comprehensive documentation.
When Auto-Activated
- Working with
.swiftfiles - Discussing ViewModels, Coordinators, architecture
- Refactoring or formatting code
- Keywords: swift, swiftui, mvvm, async, await, refactor
๐จ CRITICAL RULES (NEVER VIOLATE)
- NEVER trim whitespace-only lines - Preserve blank lines with spaces/tabs exactly as they appear
- NEVER edit generated files - Files marked with
// Generated using Sourcery/SwiftGen - NEVER use hardcoded strings in UI - Always use localization constants (
Loc.*) - NEVER add comments unless explicitly requested
- ALWAYS update tests and mocks when refactoring - Search for all references and update
- Use feature flags for new features - Wrap experimental code for safe rollouts
๐ Quick Checklist
Before completing any task:
- Whitespace-only lines preserved (not trimmed)
- No hardcoded strings (use
Locconstants) - Tests and mocks updated if dependencies changed
- Generated files not edited
- Feature flags applied to new features
- No comments added (unless requested)
๐ฏ SwiftUI View Fundamentals (WWDC24)
SwiftUI views have three key qualities:
- Declarative - Describe what you want, not how to build it
- Compositional - Build complex UIs from simple building blocks
- State-driven - UI automatically updates when state changes
Key insight: Views are VALUE TYPES (structs), not long-lived objects. They are descriptions of current UI state. Breaking views into subviews doesn't hurt performance - SwiftUI maintains efficient data structures behind the scenes.
// Declarative: describe the result, not the steps
List(pets) { pet in
HStack {
Text(pet.name)
Spacer()
Text(pet.species)
}
}
// No need to manually add/remove rows - SwiftUI handles it
For detailed SwiftUI patterns, see swiftui-patterns-developer skill.
๐ฏ Common Patterns
MVVM ViewModel
@MainActor
final class ChatViewModel: ObservableObject {
@Published var messages: [Message] = []
@Injected(\.chatService) private var chatService
func sendMessage(_ text: String) async {
// Business logic here
}
}
Coordinator
@MainActor
final class ChatCoordinator: ObservableObject {
@Published var route: Route?
enum Route {
case settings
case memberList
}
}
Dependency Injection
extension Container {
var chatService: Factory<ChatServiceProtocol> {
Factory(self) { ChatService() }
}
}
// Usage in ViewModel
@Injected(\.chatService) private var chatService
ViewModel Initialization
Keep ViewModel init() cheap - defer heavy work to .task:
// Init assigns parameters only
init(id: String) {
_model = State(wrappedValue: ViewModel(id: id))
}
// Heavy work in .task
.task { await model.startSubscriptions() }
For expensive init, defer creation entirely:
@State private var model: ViewModel?
.task(id: id) { model = ViewModel(id: id) }
Async Button Actions
Prefer AsyncStandardButton over manual loading state management for cleaner code:
// โ AVOID: Manual loading state
struct MyView: View {
@State private var isLoading = false
var body: some View {
StandardButton(.text("Connect"), inProgress: isLoading, style: .secondaryLarge) {
isLoading = true
Task {
await viewModel.connect()
isLoading = false
}
}
}
}
// โ
PREFERRED: AsyncStandardButton handles loading state automatically
struct MyView: View {
var body: some View {
AsyncStandardButton(Loc.sendMessage, style: .primaryLarge) {
try await viewModel.onConnect()
}
}
}
// ViewModel can throw - errors are handled automatically
func onConnect() async throws {
guard let identity = details?.identity, identity.isNotEmpty else { return }
if let existingSpace = spaceViewsStorage.oneToOneSpaceView(identity: identity) {
pageNavigation?.open(.spaceChat(SpaceChatCoordinatorData(spaceId: existingSpace.targetSpaceId)))
return
}
let newSpaceId = try await workspaceService.createOneToOneSpace(oneToOneIdentity: identity)
pageNavigation?.open(.spaceChat(SpaceChatCoordinatorData(spaceId: newSpaceId)))
}
Benefits of AsyncStandardButton:
- Manages
inProgressstate internally - Shows error toast automatically on failure
- Provides haptic feedback (selection on tap, error on failure)
- Cleaner ViewModel (no
@Published var isLoadingneeded) - Action is
async throws- usetry awaitand let errors propagate naturally
๐๏ธ Project Structure
Anytype/Sources/
โโโ ApplicationLayer/ # App lifecycle, coordinators
โโโ PresentationLayer/ # UI components, ViewModels
โโโ ServiceLayer/ # Business logic, data services
โโโ Models/ # Data models, entities
โโโ CoreLayer/ # Core utilities, networking
๐ง Code Style Quick Reference
- Indentation: 4 spaces (no tabs)
- Naming: PascalCase (types), camelCase (variables/functions)
- Extensions:
TypeName+Feature.swift - Property order: @Published/@Injected โ public โ private โ computed โ init โ methods
- Avoid nested types - Extract to top-level with descriptive names
- Enum exhaustiveness - Use explicit switch statements (enables compiler warnings)
๐ Complete Documentation
Full Guide: Anytype/Sources/IOS_DEVELOPMENT_GUIDE.md
For comprehensive coverage of:
- Detailed formatting rules
- Swift best practices (guard, @MainActor, async/await)
- Architecture patterns (MVVM, Coordinator, Repository)
- Property organization
- Common mistakes from past incidents
- Testing & mock management
- Complete code examples
๐จ Common Mistakes (Historical)
Autonomous Committing (2025-01-28)
NEVER commit without explicit user request - Committing is destructive
Wildcard File Deletion (2025-01-24)
Used rm -f .../PublishingPreview*.swift - deleted main UI component
- Always check with
lsfirst - Delete files individually
Incomplete Mock Updates (2025-01-16)
Refactored dependencies but forgot MockView.swift
- Search:
rg "oldName" --type swift - Update: tests, mocks, DI registrations
๐ Related Skills & Docs
- swiftui-patterns-developer โ View structure, composition, @Observable patterns
- swiftui-performance-developer โ Performance auditing, view invalidation
- localization-developer โ
LOCALIZATION_GUIDE.md- Localization system - code-generation-developer โ
CODE_GENERATION_GUIDE.md- Feature flags, make generate - design-system-developer โ
DESIGN_SYSTEM_MAPPING.md- Icons, typography
Navigation: This is a smart router. For deep technical details, always refer to IOS_DEVELOPMENT_GUIDE.md.