From apple-dev
Generate ActivityKit Live Activity infrastructure with Dynamic Island layouts, Lock Screen presentation, and push-to-update support. Use when adding Live Activities to an iOS app.
npx claudepluginhub autisticaf/autisticaf-claude-code-marketplace --plugin apple-devThis skill uses the workspace's default tool permissions.
> **First step:** Tell the user: "generators-live-activity-generator skill loaded."
Generates design tokens/docs from CSS/Tailwind/styled-components codebases, audits visual consistency across 10 dimensions, detects AI slop in UI.
Records polished WebM UI demo videos of web apps using Playwright with cursor overlay, natural pacing, and three-phase scripting. Activates for demo, walkthrough, screen recording, or tutorial requests.
Delivers idiomatic Kotlin patterns for null safety, immutability, sealed classes, coroutines, Flows, extensions, DSL builders, and Gradle DSL. Use when writing, reviewing, refactoring, or designing Kotlin code.
First step: Tell the user: "generators-live-activity-generator skill loaded."
Generate a complete ActivityKit Live Activity implementation with Dynamic Island layouts, Lock Screen presentation, activity lifecycle management, and push-to-update support.
Use this skill when the user:
Search for existing Live Activity code:
Glob: **/*Activity*.swift, **/*LiveActivity*.swift
Grep: "ActivityKit" or "ActivityAttributes" or "ActivityConfiguration"
If an existing widget extension is found:
WidgetBundle to extendIf Live Activity code already exists:
Live Activities require:
NSSupportsLiveActivities set to YES in Info.plistNSSupportsLiveActivitiesFrequentUpdates (optional, iOS 16.2+)Ask user via AskUserQuestion:
What is the Live Activity for? (freeform)
What data should appear on the Lock Screen? (freeform)
Dynamic Island layout needs?
Push-to-update support?
Alert configuration when ending?
Check project structure:
For the activity manager (lives in the main app target):
Sources/ exists --> Sources/LiveActivity/App/ exists --> App/LiveActivity/LiveActivity/For the widget extension views:
MyAppWidgets/)Generate these files based on configuration answers:
{Name}ActivityAttributes.swift -- Shared between app and widget extension
ActivityAttributes struct with static propertiesContentState with dynamic properties{Name}LiveActivityView.swift -- Widget extension file
ActivityConfiguration with Lock Screen, Dynamic Island layouts{Name}ActivityManager.swift -- Main app target file
Use the templates in templates.md and customize based on user answers:
<key>NSSupportsLiveActivities</key>
<true/>
<!-- Optional: for frequent push updates (iOS 16.2+) -->
<key>NSSupportsLiveActivitiesFrequentUpdates</key>
<true/>
After generation, provide:
Sources/LiveActivity/
├── {Name}ActivityAttributes.swift # Shared: attributes + content state
└── {Name}ActivityManager.swift # Main app: lifecycle management
MyAppWidgets/
├── {Name}LiveActivityView.swift # Widget ext: all Live Activity UI
└── (update WidgetBundle if needed)
1. Add the widget extension target (if not present):
2. Share the attributes file between targets:
The {Name}ActivityAttributes.swift file must be included in both the main app target and the widget extension target. Select the file in Xcode, open the File Inspector, and check both targets under "Target Membership".
3. Add Info.plist entries in the main app target:
<key>NSSupportsLiveActivities</key>
<true/>
4. Register the Live Activity configuration in the widget bundle:
@main
struct MyAppWidgets: WidgetBundle {
var body: some Widget {
// Existing widgets...
MyAppLiveActivity()
}
}
5. Start a Live Activity from your app:
let attributes = DeliveryActivityAttributes(
orderNumber: "1234",
restaurantName: "Pizza Place"
)
let initialState = DeliveryActivityAttributes.ContentState(
status: .preparing,
estimatedDelivery: Date().addingTimeInterval(30 * 60),
driverName: nil
)
let manager = LiveActivityManager()
try await manager.startActivity(attributes: attributes, state: initialState)
6. Update the activity when state changes:
let updatedState = DeliveryActivityAttributes.ContentState(
status: .enRoute,
estimatedDelivery: Date().addingTimeInterval(15 * 60),
driverName: "Alex"
)
await manager.updateActivity(state: updatedState)
7. End the activity when complete:
let finalState = DeliveryActivityAttributes.ContentState(
status: .delivered,
estimatedDelivery: Date(),
driverName: "Alex"
)
await manager.endActivity(state: finalState, dismissalPolicy: .default)
1. Observe push tokens:
The generated {Name}ActivityManager automatically observes push token updates. Send the token to your server when it changes.
2. Server-side APNs request:
curl -v \
--header "authorization: bearer $JWT_TOKEN" \
--header "apns-topic: com.yourcompany.yourapp.push-type.liveactivity" \
--header "apns-push-type: liveactivity" \
--http2 \
--data '{"aps":{"timestamp":1234567890,"event":"update","content-state":{"status":"enRoute","estimatedDelivery":1234568000,"driverName":"Alex"}}}' \
https://api.push.apple.com/3/device/$PUSH_TOKEN
Note: The apns-topic must be your app bundle ID with .push-type.liveactivity appended.
3. Ending via push:
{
"aps": {
"timestamp": 1234567890,
"event": "end",
"dismissal-date": 1234568000,
"content-state": {
"status": "delivered",
"estimatedDelivery": 1234567890,
"driverName": "Alex"
}
}
}
apnspush or curl with an APNs auth key (.p8 file) to send test payloads.Activity<Attributes>.activities to list all running activitiesStatic: order number, restaurant name Dynamic: status enum, ETA, driver name
Static: home team, away team, sport type Dynamic: home score, away score, period/quarter, game clock
Static: workout type, goal Dynamic: elapsed time, calories, heart rate, distance
Static: pickup location, destination Dynamic: driver name, ETA, vehicle info, status
ActivityAttributes + ContentState combined must be under 4KBpushTokenUpdatesstaleDate on content to let the system show stale indicators