Skill v1.0.0
Automated scanversion: "1.0.0" name: accessorysetupkit description: "Discover and configure Bluetooth and Wi-Fi accessories using AccessorySetupKit. Use when presenting a privacy-preserving accessory picker, defining discovery descriptors for BLE or Wi-Fi devices, handling accessory session events, migrating from CoreBluetooth permission-based scanning, or setting up accessories without requiring broad Bluetooth permissions."
AccessorySetupKit
Privacy-preserving accessory discovery and setup for Bluetooth and Wi-Fi devices. Replaces broad Bluetooth/Wi-Fi permission prompts with a system-provided picker that grants per-accessory access with a single tap. Available iOS 18+ / Swift 6.3.
After setup, apps continue using CoreBluetooth and NetworkExtension for communication. AccessorySetupKit handles only the discovery and authorization step.
Contents
- Setup and Entitlements
- Discovery Descriptors
- Presenting the Picker
- Event Handling
- Bluetooth Accessories
- Wi-Fi Accessories
- Migration from CoreBluetooth
- Common Mistakes
- Review Checklist
- References
Setup and Entitlements
Info.plist Configuration
Add these keys to the app's Info.plist:
| Key | Type | Purpose | |
|---|---|---|---|
NSAccessorySetupSupports | [String] | Required. Array containing Bluetooth and/or WiFi | |
NSAccessorySetupBluetoothServices | [String] | Service UUIDs the app discovers (Bluetooth) | |
NSAccessorySetupBluetoothNames | [String] | Bluetooth names or substrings to match | |
NSAccessorySetupBluetoothCompanyIdentifiers | [String] | Two-byte Bluetooth company identifiers |
The Bluetooth-specific keys must match the values used in ASDiscoveryDescriptor. If the app uses identifiers, names, or services not declared in Info.plist, the app crashes during AccessorySetupKit discovery. For Wi-Fi accessories, include WiFi in NSAccessorySetupSupports and match the descriptor's SSID rule.
No Bluetooth Permission Required
When an app declares NSAccessorySetupSupports with Bluetooth, creating a CBCentralManager no longer triggers the system Bluetooth permission dialog. The central manager's state transitions to poweredOn only when the app has at least one paired accessory via AccessorySetupKit.
Discovery Descriptors
ASDiscoveryDescriptor defines the matching criteria for finding accessories. The system matches scanned results against all rules in the descriptor to filter for the target accessory.
Bluetooth Descriptor
import AccessorySetupKitimport CoreBluetoothvar descriptor = ASDiscoveryDescriptor()descriptor.bluetoothServiceUUID = CBUUID(string: "12345678-1234-1234-1234-123456789ABC")descriptor.bluetoothNameSubstring = "MyDevice"descriptor.bluetoothRange = .immediate // Only nearby devices
A Bluetooth descriptor needs at least one of bluetoothCompanyIdentifier or bluetoothServiceUUID. Add narrower matchers as needed:
bluetoothNameSubstringwith a company identifier or service UUIDbluetoothManufacturerDataBlobandbluetoothManufacturerDataMaskwith a
company identifier; blob and mask must have the same length
bluetoothServiceDataBlobandbluetoothServiceDataMaskwith a service UUID;
blob and mask must have the same length
Wi-Fi Descriptor
var descriptor = ASDiscoveryDescriptor()descriptor.ssid = "MyAccessory-Network"// OR use a prefix:// descriptor.ssidPrefix = "MyAccessory-"
Supply either ssid or ssidPrefix, not both. The app crashes if both are set. The ssidPrefix must have a non-zero length.
Bluetooth Range
Control the physical proximity required for discovery:
| Value | Behavior | |
|---|---|---|
.default | Standard Bluetooth range | |
.immediate | Only accessories in close physical proximity |
Support Options
Set supportedOptions on the descriptor to declare the accessory's capabilities:
descriptor.supportedOptions = [.bluetoothPairingLE, .bluetoothTransportBridging]
| Option | Purpose | |
|---|---|---|
.bluetoothPairingLE | BLE pairing support | |
.bluetoothTransportBridging | Bluetooth transport bridging | |
.bluetoothHID | Bluetooth HID device |
Presenting the Picker
Creating the Session
Create and activate an ASAccessorySession to manage discovery lifecycle. Wait for .activated before reading session.accessories or presenting the picker:
import AccessorySetupKitfinal class AccessoryManager {private let session = ASAccessorySession()func start() {session.activate(on: .main) { [weak self] event inself?.handleEvent(event)}}private func handleEvent(_ event: ASAccessoryEvent) {switch event.eventType {case .activated:// Session ready. Check session.accessories for previously paired devices.breakcase .accessoryAdded:guard let accessory = event.accessory else { return }handleAccessoryAdded(accessory)case .accessoryChanged:// Accessory properties changed (e.g., display name updated in Settings)breakcase .accessoryRemoved:// Accessory removed by user or appbreakcase .invalidated:// Session invalidated, cannot be reusedbreak@unknown default:break}}}
Showing the Picker
Create ASPickerDisplayItem instances with a name, product image, and discovery descriptor, then pass them to the activated session:
func showAccessoryPicker() {var descriptor = ASDiscoveryDescriptor()descriptor.bluetoothServiceUUID = CBUUID(string: "ABCD1234-0000-1000-8000-00805F9B34FB")guard let image = UIImage(named: "my-accessory") else { return }let item = ASPickerDisplayItem(name: "My Bluetooth Accessory",productImage: image,descriptor: descriptor)session.showPicker(for: [item]) { error inif let error {print("Picker failed: \(error.localizedDescription)")}}}
The picker runs in a separate system process. It shows each matching device as a separate item. When multiple devices match a given descriptor, the picker creates a horizontal carousel.
Setup Options
Configure picker behavior per display item:
var item = ASPickerDisplayItem(name: "My Accessory",productImage: image,descriptor: descriptor)item.setupOptions = [.rename, .confirmAuthorization]
| Option | Effect | |
|---|---|---|
.rename | Allow renaming the accessory during setup | |
.confirmAuthorization | Show authorization confirmation before setup | |
.finishInApp | Signal that setup continues in the app after pairing |
Product Images
The picker displays images in a 180x120 point container. Best practices:
- Use high-resolution images for all screen scale factors
- Use transparent backgrounds for correct light/dark mode appearance
- Adjust transparent borders as padding to control apparent accessory size
- Test in both light and dark mode
Event Handling
Event Types
The session delivers ASAccessoryEvent objects through the event handler:
| Event | When | |
|---|---|---|
.activated | Session is active, query session.accessories | |
.accessoryAdded | User selected an accessory in the picker | |
.accessoryChanged | Accessory properties updated (e.g., renamed) | |
.accessoryRemoved | Accessory removed from system | |
.invalidated | Session invalidated, create a new one | |
.migrationComplete | Migration of legacy accessories completed | |
.pickerDidPresent | Picker appeared on screen | |
.pickerDidDismiss | Picker dismissed | |
.pickerSetupBridging | Transport bridging setup in progress | |
.pickerSetupPairing | Bluetooth pairing in progress | |
.pickerSetupFailed | Setup failed | |
.pickerSetupRename | User is renaming the accessory | |
.accessoryDiscovered | New accessory found (custom filtering mode) |
Coordinating Picker Dismissal
When the user selects an accessory, .accessoryAdded fires before .pickerDidDismiss. To show custom setup UI after the picker closes, store the accessory on the first event and act on it after dismissal:
private var pendingAccessory: ASAccessory?private func handleEvent(_ event: ASAccessoryEvent) {switch event.eventType {case .accessoryAdded:pendingAccessory = event.accessorycase .pickerDidDismiss:if let accessory = pendingAccessory {pendingAccessory = nilbeginCustomSetup(accessory)}@unknown default:break}}
Bluetooth Accessories
After an accessory is added via the picker, use CoreBluetooth to communicate. The bluetoothIdentifier on the ASAccessory maps to a CBPeripheral.
import CoreBluetoothfunc handleAccessoryAdded(_ accessory: ASAccessory) {guard let btIdentifier = accessory.bluetoothIdentifier else { return }// Create CBCentralManager — no Bluetooth permission prompt appearslet centralManager = CBCentralManager(delegate: self, queue: nil)// After poweredOn, retrieve the peripherallet peripherals = centralManager.retrievePeripherals(withIdentifiers: [btIdentifier])guard let peripheral = peripherals.first else { return }centralManager.connect(peripheral, options: nil)}
Key points:
CBCentralManagerstate reaches.poweredOnonly when the app has paired accessories- Scanning with
scanForPeripherals(withServices:)returns only
accessories paired through AccessorySetupKit
- No
NSBluetoothAlwaysUsageDescriptionis needed when using AccessorySetupKit
exclusively
Wi-Fi Accessories
For Wi-Fi accessories, the ssid on the ASAccessory identifies the network. Use NEHotspotConfiguration from NetworkExtension to join it:
import NetworkExtensionfunc handleWiFiAccessoryAdded(_ accessory: ASAccessory) {guard let ssid = accessory.ssid else { return }let configuration = NEHotspotConfiguration(ssid: ssid)NEHotspotConfigurationManager.shared.apply(configuration) { error inif let error {print("Wi-Fi join failed: \(error.localizedDescription)")}}}
Because the accessory was discovered through AccessorySetupKit, joining the network does not trigger the standard Wi-Fi access prompt.
Migration from CoreBluetooth
Apps with existing CoreBluetooth-authorized accessories can migrate them to AccessorySetupKit using ASMigrationDisplayItem. This is a one-time operation that registers known accessories in the new system.
func migrateExistingAccessories() {guard let image = UIImage(named: "my-accessory") else { return }var descriptor = ASDiscoveryDescriptor()descriptor.bluetoothServiceUUID = CBUUID(string: "ABCD1234-0000-1000-8000-00805F9B34FB")let migrationItem = ASMigrationDisplayItem(name: "My Accessory",productImage: image,descriptor: descriptor)// Set the peripheral identifier from CoreBluetoothmigrationItem.peripheralIdentifier = existingPeripheralUUID// For Wi-Fi accessories:// migrationItem.hotspotSSID = "MyAccessory-WiFi"session.showPicker(for: [migrationItem]) { error inif let error {print("Migration failed: \(error.localizedDescription)")}}}
Migration rules:
- If
showPickercontains only migration items, the system shows an
informational page instead of a discovery picker
- If migration items are mixed with regular display items, migration happens
only when a new accessory is discovered and set up
- Do not initialize
CBCentralManagerbefore migration completes — doing so
causes an error and the picker fails to appear
- The session receives
.migrationCompletewhen migration finishes
Common Mistakes
DON'T: Omit Info.plist keys for Bluetooth discovery
The app crashes if it uses identifiers, names, or services in descriptors that are not declared in Info.plist.
// WRONG — service UUID not in NSAccessorySetupBluetoothServicesvar descriptor = ASDiscoveryDescriptor()descriptor.bluetoothServiceUUID = CBUUID(string: "UNDECLARED-UUID")session.showPicker(for: [item]) { _ in } // Crash// CORRECT — declare all UUIDs in Info.plist first// Info.plist: NSAccessorySetupBluetoothServices = ["ABCD1234-..."]var descriptor = ASDiscoveryDescriptor()descriptor.bluetoothServiceUUID = CBUUID(string: "ABCD1234-...")
DON'T: Set both ssid and ssidPrefix
// WRONG — crashes at runtimevar descriptor = ASDiscoveryDescriptor()descriptor.ssid = "MyNetwork"descriptor.ssidPrefix = "My" // Cannot set both// CORRECT — use one or the othervar descriptor = ASDiscoveryDescriptor()descriptor.ssid = "MyNetwork"
DON'T: Initialize CBCentralManager before migration
// WRONG — migration fails, picker does not appearlet central = CBCentralManager(delegate: self, queue: nil)session.showPicker(for: [migrationItem]) { error in// error is non-nil}// CORRECT — wait for .migrationComplete before using CoreBluetoothsession.activate(on: .main) { event inif event.eventType == .migrationComplete {let central = CBCentralManager(delegate: self, queue: nil)}}
DON'T: Show the picker without user intent
// WRONG — picker appears unexpectedly on app launchoverride func viewDidLoad() {super.viewDidLoad()session.showPicker(for: items) { _ in }}// CORRECT — bind picker to a user action@IBAction func addAccessoryTapped(_ sender: UIButton) {session.showPicker(for: items) { _ in }}
DON'T: Reuse an invalidated session
// WRONG — session is dead after invalidationsession.showPicker(for: items) { _ in } // No effect// CORRECT — create a new sessionlet newSession = ASAccessorySession()newSession.activate(on: .main) { event in// Handle events}
Review Checklist
- [ ]
NSAccessorySetupSupportsadded to Info.plist withBluetoothand/orWiFi - [ ] Bluetooth-specific plist keys (
NSAccessorySetupBluetoothServices,NSAccessorySetupBluetoothNames,NSAccessorySetupBluetoothCompanyIdentifiers) match descriptor values - [ ] Session activated before calling
showPicker - [ ] Event handler uses
[weak self]to avoid retain cycles - [ ] All
ASAccessoryEventTypecases handled, including@unknown default - [ ] Product images use transparent backgrounds and appropriate resolution
- [ ]
ssidandssidPrefixare never set simultaneously on a descriptor - [ ] Picker presentation tied to explicit user action, not automatic
- [ ]
CBCentralManagernot initialized until after migration completes (if migrating) - [ ]
bluetoothIdentifierorssidfromASAccessoryused to connect post-setup - [ ] Invalidated sessions replaced with new instances
- [ ] Accessory removal events handled to clean up app state
References
- Extended patterns (custom filtering, batch setup, removal handling, error recovery): references/accessorysetupkit-patterns.md
- AccessorySetupKit framework
- ASAccessorySession
- ASDiscoveryDescriptor
- ASPickerDisplayItem
- ASAccessory
- ASAccessoryEvent
- ASMigrationDisplayItem
- Discovering and configuring accessories
- Setting up and authorizing a Bluetooth accessory
- Meet AccessorySetupKit — WWDC24