Spotlight Indexing Generator
Generate production Core Spotlight indexing infrastructure — makes app content searchable via Spotlight (and Siri suggestions). Indexes items with rich attributes, handles search continuation into your app, and manages index lifecycle.
When This Skill Activates
Use this skill when the user:
- Asks to "add spotlight search" or "spotlight indexing"
- Mentions "core spotlight" or "CSSearchableItem"
- Wants to make "searchable content" or "index content" for system search
- Asks about "system search" integration or "Siri suggestions"
- Wants content to appear in Spotlight results
- Mentions "NSUserActivity" for search or handoff
Pre-Generation Checks
1. Project Context Detection
- [ ] Check deployment target (iOS 16+ / macOS 13+)
- [ ] Check Swift version (requires Swift 5.9+)
- [ ] Check for @Observable support (iOS 17+ / macOS 14+)
- [ ] Identify source file locations
2. Existing CoreSpotlight Detection
Search for existing Spotlight code:
Glob: **/*Spotlight*.swift, **/*Searchable*.swift, **/*CSSearchable*.swift
Grep: "CoreSpotlight" or "CSSearchableIndex" or "CSSearchableItem"
If existing Spotlight code found:
- Ask if user wants to replace or augment it
- If augmenting, identify what's missing and generate only those pieces
3. Deep Linking Detection
Search for existing deep link or navigation setup:
Grep: "NavigationPath" or "onOpenURL" or "NSUserActivity" or "DeepLink"
If deep linking exists:
- Integrate SpotlightSearchHandler with existing router
- If not, generate standalone handler with guidance on wiring it up
4. Entitlements Check
Verify CoreSpotlight doesn't require special entitlements (it doesn't — it's a standard framework), but check if the app uses App Groups for shared index across extensions.
Configuration Questions
Ask user via AskUserQuestion:
-
Content types to index?
- Articles / blog posts (titles, body text, authors)
- Products (name, description, price, images)
- Contacts / people (name, phone, email)
- Tasks / reminders (title, due date, priority)
- Custom (user describes their model)
-
Include thumbnails?
- Yes — index thumbnail images alongside text attributes
- No — text-only attributes (smaller index, faster)
-
Indexing strategy?
- Batch indexing (index all content at once, e.g., on first launch or sync)
- Incremental indexing (index items as they're created/updated/deleted)
- Both (batch for initial load, incremental for changes)
-
Include Siri suggestions / shortcuts?
- Yes — make content eligible for Siri suggestions and predictions
- No — Spotlight search only
Generation Process
Step 1: Read Templates
Read templates.md for production Swift code.
Read patterns.md for architecture guidance and best practices.
Step 2: Create Core Files
Generate these files:
SpotlightIndexable.swift— Protocol any model can conform toSpotlightIndexManager.swift— Actor wrapping CSSearchableIndex with batchingSpotlightAttributeBuilder.swift— Fluent builder for CSSearchableItemAttributeSet
Step 3: Create Integration Files
SpotlightSearchHandler.swift— Handles NSUserActivity continuation from Spotlight taps
Step 4: Create Optional Files
Based on configuration:
SpotlightSyncModifier.swift— If incremental indexing selected (ViewModifier for auto index/deindex)- Add NSUserActivity + eligibleForSearch if Siri suggestions selected
Step 5: Determine File Location
Check project structure:
- If
Sources/exists ->Sources/SpotlightIndexing/ - If
App/exists ->App/SpotlightIndexing/ - Otherwise ->
SpotlightIndexing/
Output Format
After generation, provide:
Files Created
SpotlightIndexing/
├── SpotlightIndexable.swift # Protocol for indexable models
├── SpotlightIndexManager.swift # Actor-based index management
├── SpotlightAttributeBuilder.swift # Fluent attribute builder
├── SpotlightSearchHandler.swift # Handle Spotlight tap continuation
└── SpotlightSyncModifier.swift # Auto index/deindex ViewModifier (optional)
Integration with Content Creation/Update
Make a model searchable:
// Conform your model to SpotlightIndexable
struct Article: SpotlightIndexable {
let id: UUID
let title: String
let body: String
let author: String
let tags: [String]
var spotlightID: String { id.uuidString }
var spotlightTitle: String { title }
var spotlightDescription: String { String(body.prefix(300)) }
var spotlightKeywords: [String] { tags + [author] }
var spotlightThumbnailData: Data? { nil }
var spotlightDomainIdentifier: String { "com.myapp.articles" }
}
Index on create, remove on delete:
func createArticle(_ article: Article) async throws {
try await repository.save(article)
await SpotlightIndexManager.shared.index(items: [article])
}
func deleteArticle(_ article: Article) async throws {
try await repository.delete(article)
await SpotlightIndexManager.shared.remove(identifiers: [article.spotlightID])
}
Batch reindex (e.g., on first launch):
func reindexAllContent() async {
let articles = await repository.fetchAll()
await SpotlightIndexManager.shared.reindexAll(items: articles, domain: "com.myapp.articles")
}
Handle Spotlight tap (App or Scene delegate):
@main
struct MyApp: App {
var body: some Scene {
WindowGroup {
ContentView()
.onContinueUserActivity(CSSearchableItemActionType) { activity in
SpotlightSearchHandler.shared.handle(activity)
}
}
}
}
Auto index/deindex with ViewModifier:
struct ArticleDetailView: View {
let article: Article
var body: some View {
ScrollView {
Text(article.body)
}
.spotlightIndexed(article) // Indexes on appear, deindexes on disappear
}
}
Testing
@Test
func indexAndRetrieveItem() async throws {
let manager = SpotlightIndexManager(index: MockSearchableIndex())
let article = Article(id: UUID(), title: "Test", body: "Body", author: "Author", tags: ["swift"])
await manager.index(items: [article])
#expect(manager.indexedCount == 1)
}
@Test
func batchIndexChunksCorrectly() async throws {
let mockIndex = MockSearchableIndex()
let manager = SpotlightIndexManager(index: mockIndex, batchSize: 10)
let items = (0..<25).map { makeArticle(index: $0) }
await manager.index(items: items)
#expect(mockIndex.indexCallCount == 3) // 10 + 10 + 5
}
@Test
func handleSpotlightContinuation() async throws {
let handler = SpotlightSearchHandler()
let activity = NSUserActivity(activityType: CSSearchableItemActionType)
activity.userInfo = [CSSearchableItemActivityIdentifier: "article-123"]
let itemID = handler.extractItemID(from: activity)
#expect(itemID == "article-123")
}
Common Patterns
Index Item on Create
Every time a model is created or updated, index it immediately:
await SpotlightIndexManager.shared.index(items: [newItem])
Remove on Delete
When content is deleted, remove it from the index:
await SpotlightIndexManager.shared.remove(identifiers: [item.spotlightID])
Batch Reindex
After app update or data migration, reindex all content:
await SpotlightIndexManager.shared.reindexAll(items: allItems, domain: "com.myapp.articles")
Handle Search Continuation
When user taps a Spotlight result, the app receives an NSUserActivity. Extract the item ID and navigate:
.onContinueUserActivity(CSSearchableItemActionType) { activity in
if let id = activity.userInfo?[CSSearchableItemActivityIdentifier] as? String {
navigationPath.append(Route.detail(id: id))
}
}
Gotchas
Index Size Limits
- CSSearchableIndex has no hard documented limit, but Apple recommends keeping index under ~10,000 items for best performance
- For very large datasets, index only the most relevant/recent items
- Use
expirationDateon CSSearchableItem to auto-expire stale entries
Thumbnail Size for Performance
- Keep thumbnails small: 300x300 pixels max, JPEG compressed
- Large thumbnails slow down indexing and increase on-disk index size
- Use
Data(JPEG/PNG) not full PlatformImage objects
CSSearchableIndex.default() Thread Safety
CSSearchableIndex.default()returns a singleton but its methods are NOT actor-isolated- Wrap all calls in an actor (SpotlightIndexManager) to prevent data races
- Never call
indexSearchableItemsfrom multiple threads simultaneously
Handling App Launch from Spotlight Tap
- The app may be cold-launched — ensure navigation state is ready before handling the activity
- Use
.onContinueUserActivityin SwiftUI, notapplication(_:continue:)alone - The activity type is
CSSearchableItemActionType(a constant from CoreSpotlight)
Index Maintenance on App Update
- After a data model change, old indexed items may have stale attributes
- Call
removeAll(domain:)then re-index on first launch after update - Store indexed schema version in UserDefaults to detect when reindex is needed
Domain Identifiers
- Always set
domainIdentifieron items — it allows bulk removal by domain - Use reverse-DNS style:
"com.myapp.articles","com.myapp.products" - Without domains, you can only remove by individual identifier or remove everything
References
- templates.md — All production Swift templates for Spotlight indexing
- patterns.md — CSSearchableItemAttributeSet content types, batch strategies, Siri suggestions, testing
- Related:
generators/deep-linking— Deep link routing from Spotlight taps