skills/generators/spotlight-indexing/SKILL.md
Generates Core Spotlight indexing infrastructure for making app content searchable via system Spotlight with rich attributes and deep link integration. Use when user wants to index content for Spotlight search, Siri suggestions, or system-wide searchability.
npx skillsauth add rshankras/claude-code-apple-skills spotlight-indexingInstall 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.
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.
Use this skill when the user:
Search for existing Spotlight code:
Glob: **/*Spotlight*.swift, **/*Searchable*.swift, **/*CSSearchable*.swift
Grep: "CoreSpotlight" or "CSSearchableIndex" or "CSSearchableItem"
If existing Spotlight code found:
Search for existing deep link or navigation setup:
Grep: "NavigationPath" or "onOpenURL" or "NSUserActivity" or "DeepLink"
If deep linking exists:
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.
Ask user via AskUserQuestion:
Content types to index?
Include thumbnails?
Indexing strategy?
Include Siri suggestions / shortcuts?
Read templates.md for production Swift code.
Read patterns.md for architecture guidance and best practices.
Generate these files:
SpotlightIndexable.swift — Protocol any model can conform toSpotlightIndexManager.swift — Actor wrapping CSSearchableIndex with batchingSpotlightAttributeBuilder.swift — Fluent builder for CSSearchableItemAttributeSetSpotlightSearchHandler.swift — Handles NSUserActivity continuation from Spotlight tapsBased on configuration:
SpotlightSyncModifier.swift — If incremental indexing selected (ViewModifier for auto index/deindex)Check project structure:
Sources/ exists -> Sources/SpotlightIndexing/App/ exists -> App/SpotlightIndexing/SpotlightIndexing/After generation, provide:
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)
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
}
}
@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")
}
Every time a model is created or updated, index it immediately:
await SpotlightIndexManager.shared.index(items: [newItem])
When content is deleted, remove it from the index:
await SpotlightIndexManager.shared.remove(identifiers: [item.spotlightID])
After app update or data migration, reindex all content:
await SpotlightIndexManager.shared.reindexAll(items: allItems, domain: "com.myapp.articles")
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))
}
}
expirationDate on CSSearchableItem to auto-expire stale entriesData (JPEG/PNG) not full PlatformImage objectsCSSearchableIndex.default() returns a singleton but its methods are NOT actor-isolatedindexSearchableItems from multiple threads simultaneously.onContinueUserActivity in SwiftUI, not application(_:continue:) aloneCSSearchableItemActionType (a constant from CoreSpotlight)removeAll(domain:) then re-index on first launch after updatedomainIdentifier on items — it allows bulk removal by domain"com.myapp.articles", "com.myapp.products"generators/deep-linking — Deep link routing from Spotlight tapsdevelopment
Build, install, and launch an iOS app on a physical iPhone or iPad entirely from the command line (no Xcode GUI), using xcodebuild + devicectl. Use when the user wants to run, test, or screenshot their app on a real device without opening Xcode.
development
Comprehensive iOS development guidance including Swift best practices, SwiftUI patterns, UI/UX review against HIG, and app planning. Use for iOS code review, best practices, accessibility audits, or planning new iOS apps.
development
Build, install, launch, and screenshot an iOS app in the Simulator to verify a change visually. Use when the user wants to run the app, see a change live, screenshot the running app, or confirm a UI fix actually works (not just that it compiles).
development
Audits skills in this repo for consistency, API drift, and structural gaps. Produces a prioritized report grouped by severity (Critical/High/Medium/Low). Use when asked to "audit skills", "check the skill repo for drift", or when planning bulk skill cleanup. Read-only — does not apply fixes.