HomeKit 在那些支持苹果 Home Automation Protocol 和 iOS 设备的附属配件之间实现了无缝集成和融合,从而推进家庭自动化的发展和革新。通过一个通用的家庭自动化设备协议,以及一个可以配置这些设备并与之通信的公开 API,HomeKit 使得 App 用户控制自己的 home 成为可能,而不需要由生产家庭自动化配件的厂商创建。HomeKit 也使得来自多个厂商的家庭自动化配件集成为一体,而无需厂商之间彼此直接协调。该文档旨在帮你编写HomeKit App。HomeKit 库是用来沟通和控制家庭自动化配件的,这些家庭自动化配件都支持苹果的 HomeKit Accessory Protocol。
本文档主要帮助家庭自动化配件的开发者来进行 HomeKit App 的开发,帮助读者了解智能家居平台 HomeKit,并了解智能家居控制的原理。
看起来 HomeKit 非常强大的,灵活性也很高,不过想要获得成功,还得靠硬件。
该文档旨在帮你编写 HomeKit app。HomeKit 库是用来沟通和控制家庭自动化配件的,这些家庭自动化配件都支持苹果的 HomeKit Accessory Protocol。HomeKit 应用程序可让用户发现兼容配件并配置它们。用户可以创建一些 action 来控制智能配件(例如恒温或者光线强弱),对其进行分组,并且可以通过 Siri 触发。HomeKit 对象被存储在用户 iOS 设备的数据库中,并且通过 iCloud 还可以同步到其他 iOS 设备。HomeKit 支持远程访问智能配件,并支持多个用户设备和多个用户。HomeKit 还对用户的安全和隐私做了处理。
注意:如果你是开发设计 HomeKit 硬件的供应商,你可以去 Hardware Developers 下的HomeKit 页面了解 MFi Program 相关信息,也可以阅读 External Accessory Programming Topics。
以下资源提供了更多关于创建 HomeKit 应用程序的信息:
HomeKit 应用服务只提供给通过 App Store 发布的 App 应用程序。在你的 Xcode 工程中,HomeKit 应用程序需要额外的配置,你的 App 必须有开发证书和代码签名才能使用 HomeKit。在 Xcode 的 Capabilities 面板使用 HomeKit,可避免代码签名的问题。你无需直接在 Xcode 或者会员中心编辑授权文件(entitlements)。
为了完成本文档中所有步骤,你需要:
在你开始使用 HomeKit 之前,请确保你已经完成以下任务。创建你团队的配置文件(Provisioning Profile),请参阅:App Distribution Quick Start。
当你成功地完成了之前的任务后,General 面板中 Team 弹出菜单中的错误信息和问题修复按钮将会消失。代码签名配置被成功创建后会展示下方的 General 面板。
解决代码签名和证书配置问题,请参阅App Distribution Guide文档中Troubleshooting这一节。
想要使用 HomeKit,首先要启用它。Xcode 将会添加 HomeKit 权限到你的工程授权文件中和会员中心的 App ID 授权文件中,也会将 HomeKit 框架添加到你的工程中。HomeKit 需要一个明确的 App ID, 这个 App ID 是为了你完成这些步奏而创建的。
启用 HomeKit 的步骤如下:
无需为了开发 Homekit 应用程序而购买硬件产品。你可以使 HomeKit Accessory Simulator 来测试 HomeKit app 和模拟配件设备之间的通信。HomeKit Accessory Simulator 不是和 Xcode 一起发布的。
下载 HomeKit Accessory Simulator 步骤如下:
"Hardware IO Tools for Xcode ".dmg
文件。~/Downloads
中的.dmg
文件。/Application
文件中。之后,你将可以使用 HomeKit Accessory Simulator 测试你的 HomeKit 应用程序,正如 Testing YourHomeKit App(第 30 页)中描述的那样。
HomeKit 允许用户创建一个或者多个 Home 布局。每个 Home(HMHome)代表一个有网络设备的住所。用户拥有Home的数据并可通过自己的任何一台 iOS 设备进行访问。用户也可以和客户共享一个 Home,但是客户的权限会有更多限制。被指定为 primary home 的 home 默认是 Siri 指令的对象,并且不能指定 Home。
每个 Home 一般有多个 room,并且每个 room 一般会有多个智能配件。在 Home(HMHome) 中,每个房间是独立的 room,并具有一个有意义的名字,例如“卧室”或者“厨房”,这些名字可以在 Siri 命令中使用。一个 accessory(HMAccessory)代表实际家庭中的自动化设备,例如车库开门器。一个 sevice(HMService)是 accessory 提供的?种实际服务,例如打开或者关闭车库,或者车库上的灯。
如果你的 App 缓存了 Home 布局的信息,那么当其布局发声改变的时候,App 就需要更新这些信息。使用 HMHomeManager 对象可以从 HomeKit 数据库获取 HMHome 和其他相关的对象。本章描述的 API 获取对象后,你应该通过代理回调函数保持获取对象和 HomeKit 数据库同步,具体描述请参看“Observing HomeKit Database Changes"。
使用 Home Manager—一个 HMHomeManager 对象的访问 home、room、配件、服务以及其他 HomeKit 对象。在创建家庭对象管理器(home manager)之后,直接设置它的代理,以便获取到这些对象之后及时的通知到你。
self.homeManager = [[HMHomeManager alloc] init];self.homeManager.delegate = self;
当你创建一个 home manager 对象时,HomeKit 就开始从 HomeKit 数据库获取这些 homes 和相关对象,例如 room 和 accessory 对象。当 HomeKit 正在获取那些对象时,home manager 的 primaryHome 属性是 nil,并且 homes 属性是个空数组。你的 App 应该处理用户还没有完成创建 home 的情况,但是 App 应该等待直到 HomeKit 完成初始化。当获取对象完成之后,HomeKit 会发送 homeManagerDidUpdateHomes:消息给 home manager 的代理。
注意:当 App 进入前台或者在后台 Home manager 属性发生改变时,这个 homeManagerDidUpdateHomes:方法就会被调用,详情请参阅“Observing Changes to the Collection of Homes”。
通过 home manager 的 primaryHome 属性,可以得到 primary home,代码如下:
HMHome *home = self.homeManager.primaryHome;
使用 home manager 的 homes 属性可以得到用户的所有 home 的集合;例如自家主要居所、度假别墅以及办公室。每个 home 都对应一个独立的 home 对象。
HMHome *home; for(home in self.homeManager.homes ){ …}
在一个 home 中,rooms 属性定义 accessories 的物理位置。用 home 的 rooms 属性可以枚举 home 中的所 room。
HMHome *home = self.homeManager.primaryHome; HMRome *room; for(room in home.rooms){ … }
Accessories 数组属于 home,但是被指定给了 home 中的 room。假如用户没有给一个 accessory 指定 room,那么这个 accessories 被指定一个默认的 room ,这个 room 是 roomForEntireHome 方法的返回值。用 room 的 accessories 属性可以枚举 room 中所有的 accessory。代码如下:
HMAccessory *accessory; for(accessory in room.accessories){ … }
如果你要展示一个个 accessory 的相关信息或者允许用户控制它,可设置 accessory 的代理方法并实现这个代理方法,详情请见“Observing Changes to Accessories”。
一旦你获取到一个 accessory 对象,你就可以访问它的服务和对象,详情请参阅“Accessing Services and Characteristics”。
HomeKit 对象被保存在一个可以共享的 HomeKit 数据库里,它可以通过 HomeKit 框架被多个应英程序访问。所有 HomeKit 调用的方法都是异步写入的,并且这些方法都包含一个完成处理后的参数。如果这个方法处理成功了,你的应用将会在完成处理函数里更新本地对象。应用程序启动时,HomeKit 对象发生改变的并不能收到代理回调?法,只能接受处理完成后的回调函数。
想要观察其他应用程序启动时 HomeKit 对象的变化,请参阅:Observing HomeKit Database Changes。查阅异步消息完成处理后传过来的错误码的信息,请参阅:HomeKit Constants Reference。
HomeKit 对象的名字,例如 home、room 和 zone 对象都可以被 Siri 识别,这一点已经在文档中指出。以下几点是 HomeKit 对象的命名规则:
想了用户可以使用哪些语言与 Siri 进行交互,请参阅HomeKit User Interface Guidelines文档中的"Siri Integration"。
在 HMHomeManager 类中使用 addHomeWithName:completionHandler: 异步方法可以添加一个 home。作为参数传到那个方法中的 home 的名字,必须是唯一独特的,并且是 Siri 可以识别的 home 名字。
[self.homeManager addHomeWithName:@"My Home"completionHandler:^(HMHome *home, NSError *error) {if (error != nil) {// Failed to add a home} else {// Successfully added a home} }];
在 else 语句中,写入代码以更新你应的程序的视图。为了获取 home manager 对象,请参阅Getting the Home Manager Object。
使用 addRoomWithName:completionHandler: 异步方法可以在一个 home 中添加一个 room 对象。作为参数传到那个方法中的 room 的名字,必须是唯一独特的,并且是 Siri 可识别的 room 名字。
NSString *roomName = @"Living Room";[home addRoomWithName:roomName completionHandler:^(HMRoom*room, NSError *error) {if (error != nil) {// Failed to add a room to a home} else {// Successfully added a room to a home} }];
在 else 语句中,写入代码更新应用程序的视图。
Accessories 封装了物理配件的状态,因此它不能被用户创建。想要允许用户给家添加新的配件,我们可以使 HMAccessoryBrowser 对象找到一个与 home 没有关联的配件。HMAccessoryBrower 对象在后台搜寻配件,当它找到配件的时候,使用委托来通知你的应用程序。只有在 startSearchingForNewAccessories 方法调用之后或者 stopSearchingForNewAccessories 方法调用之前,HMAccessoryBrowserDelegate 消息才被发送给代理对象。
1.在你的类接口中添加配件浏览器委托协议,并且添加一个配件浏览器属性。代码如下:
@interface EditHomeViewController ()@property HMAccessoryBrowser *accessoryBrowser;@end
用你自己的类名代替 EditHomeViewController
2.创建配件浏览器对象,并设置它的代理
self.accessoryBrowser = [[HMAccessoryBrowser alloc] init];self.accessoryBrowser.delegate = self;
3.开始搜寻配件
[self.accessoryBrowser startSearchingForNewAccessories];
4.将找到的配件添加到你的收藏里
- (void)accessoryBrowser:(HMAccessoryBrowser *)browserdidFindNewAccessory:(HMAccessory *)accessory {// Update the UI per the new accessory; for example,reload a pickerview.[self.accessoryPicker reloadAllComponents];}
用你自己的代码实现上面的 accessoryBrowser:didFindNewAccessory:方法。 当然也可以实现 accessoryBrowser:didRemoveNewAccessory: 这个方法来移除配件,这个配件对你的视图或者收藏来说不再是新的。
5.停止搜寻配件
如果一个视图控制器正在开始搜寻配件,那么可以通过重写 viewWillDisappear:方法来停止搜寻配件。代码如下:
- (void)viewWillDisappear:(BOOL)animated {[self.accessoryBrowser stopSearchingForNewAccessories];}
注意: 在 WiFi 网络环境下,为了安全地获取新的并且能够被 HomeKit 发现的无线配件,请参阅External Accessory Framework Reference。
配件归属于 home,并且它可以被随意添加到 home 中的任意一个 room 中。使用 addAccessory:completionHandler:这个异步方法可以在 home 中添加配件。这个配件的名字作为一个参数传递到上述异步方法中,并且这个名字在配件所属的 home 中必须是唯一的。使用 assignAccessory:toRoom:completionHandler: 这个异步方法可以给 home 中的 room 添加配件。配件默认的 room 是 roomForEntireHome 这个方法返回值 room。下面的代码演示了如何给 home 和 room 添加配件:
// Add an accesory to a home and a room// 1. Get the home and room objects for the completionhandlers.__block HMHome *home = self.home;__block HMRoom *room = roomInHome;// 2. Add the accessory to the home[home addAccessory:accessory completionHandler:^(NSError*error) {if (error) {// Failed to add accessory to home} else {if (accessory.room != room) {// 3. If successfully, add the accessory tothe room[home assignAccessory:accessory toRoom:roomcompletionHandler:^(NSError *error) {if (error) {// Failed to add accessory to room} }];} }}];
配件可提供一项或者多项服务,这些服务的特性是由制造商定义。想了解配件的服务和特性目的,请参阅 Accessing Services and Characteristics。
使用 updateName:completionHandler: 异步方法可以改变配件的名称,代码如下:
[accessory updateName:@"Kid's Night Light"completionHandler:^(NSError *error) {if (error) {// Failed to change the name} else {// Successfully changed the name}}];
桥接口是配件中的一个特殊对象,它允许你和其他配件交流,但是不允许你直接和 HomeKit 交流。例如一个桥接口可以是控制多个灯的枢纽,它使用的是自己的通信协议,而不是 HomeKit 配件通信协议。想要给home添加多个桥接口 ,你可以按照 Adding Accessories to Homes and Rooms 中所描述的步骤,添加任何类型的配件到 home 中。当你给 home 添加一个桥接口时,在桥接口底层的配件也会被添加到 home 中。正如 Observing HomeKit Database Changes 中所描述的那样,每次更改通知设计模,home 的代理不会接收到桥接口的 home:didAddAccessory: 代理消息,而是接收一个有关于配件的 home:didAddAccessory:代理消息。在 home 中,要把桥接口后的配件和任何类型的配件看成一样的--例如,把它们加入配件列表的配置表中。相反的是,当你给 room 增添一个桥接口时,这个桥接口底层的配件并不会自动地添加到 room 中,原因是桥接口和它的的配件可以位于到不同的 room 中。
分区 (HMZone) 是任意可选的房间(rooms)分组;例如楼上、楼下或者卧室。房间可以被添加到一个或者多个区域。
可使用 addZoneWithName:completionHandler: 异步方法创建分区。所创建的作为参数传递到这个方法中分区的名称,在 home 中必须是唯一的,并且应该能被 Siri 识别。代码如下:
__block HMHome *home = self.home;NSString *zoneName = @"Upstairs";[home addZoneWithName:zoneName completionHandler:^(HMZone*zone, NSError *error){if (error) {// Failed to create zone} else {// Successfully created zone, now add the rooms}}];
可使用 addRoom:completionHandler:异步方法给分区添加一个 room,代码如下:
__block HMRoom *room = roomInHome;[zone addRoom:room completionHandler:^(NSError *error) {if (error) {// Failed to add room to zone} else {// Successfully added room to zone} }];
每个 Home 都有一个 HomeKit 数据库。如下图所示,HomeKit 数据库会安全地和 home 授权的用户的 iOS 设备以及潜在的客人的 iOS 设备进行同步。为了给用户展示当前最新的数据,你的应用需要观察 HomeKit 数据库的变化。
HomKit 使用代理设计模式(delegation design pattern)来通知应用程序 HomeKit 对象的改变。一般来讲,如果你的应用程序调用了一个带有完成处理参数的 HomeKit 方法,并且这个方法被成功调用了,那么相关联的代理消息就会被发送给其他 HomeKit 应用,无论这些应用是安装在同一台 iOS 设备上还是远程 iOS 设备上。这些应用甚至可以运行在客人的 iOS 设备上。如果你的应用发起了数据改变,但是代理消息并没有发送到你的应用,那么添加代码到完成处理方法和相关联的代理方法中来刷新数据和更新视图就成为必须了。如果 home 布局发生了显著变化,那么就重新加载关于这个 home 的所有信息。在完成程序处理的情况下,请在更新应用之前检查那个方法是否成功。Homkit 也会调用代理方法来通知你的应用程序 home 网络状态的改变。
例如,下图演示了使用代理方法的过程:响应用户的操作,你的应用程序调用了 addRoomWithName:completionHandler:方法,并且没有错误发生,完成处理程序应当更新 home 的所有视图。如果成功了,homeKit 将会发送 home:didAddRoom:消息给其他应用中 homes 的代理。因此,你实现的这个 home:didAddRoom:方法也应该更新 home 的所有视图。
应用程序只有在前台运行的时候才能接受代理消息。当你的应用在后台时,HomeKit 数据库的改变并不会成批处理。也就是说,如果你的应用在后台,当其他的应用成功地添加一个 room 到 home 中的时候,你的应用程序并不会接收到 home:didAddRoom: 消息。当你的应用程序到前台运行时,你的应用程序将会接收到 homeManagerDidUpdateHomes:消息,这个消息是表示你的应用程序要重新加载所有的数据。
设置 home manager 的代理并且实现HMHomeManagerDelegate协议,当 primary home 或者 home 集合发生改变时,可以接收代理消息。
所有的应用都需要实现 homeManagerDidUpdateHomes:方法,这个方法在完成最初获取 homes 之后被调用。对新建的 home manager 来说,在这个方法被调用之前,primaryHome 属性的值是 nil,homes 数组是空的数组。当应用程序开始在前台运行时也会调用 homeManagerDidUpdateHomes: 方法,当其在后台运行时数据发生改变。该 homeManagerDidUpdateHomes:方法会重新加载与 homes 相关联的所有数据。
1.在你的类接口中添加 HMHomeManagerDelegate 代理和 homeManager 属性。代码如下:
@interface AppDelegate () @property (strong, nonatomic) HMHomeManager *homeManager;@end
2.创建 home manager 对象并设置其代理
- (BOOL)application:(UIApplication *)applicationdidFinishLaunchingWithOptions:(NSDictionary *)launchOptions{self.homeManager = [[HMHomeManager alloc] init];self.homeManager.delegate = self;return YES;}
3.实现 homes 发生改变时调用的代理方法。例如:如果多个视图控制器展示了 homes 相关信息,你可以发布一个更改通知去更新所有视图。
- (void)homeManagerDidUpdateHomes:(HMHomeManager *)manager {// Send a notification to the other objects[[NSNotificationCenter defaultCenter]postNotificationName:@"UpdateHomesNotification"object:self];}- (void)homeManagerDidUpdatePrimaryHome:(HMHomeManager*)manager {// Send a notification to the other objects[[NSNotificationCenter defaultCenter]postNotificationName:@"UpdatePrimaryHomeNotification"object:self];}
视图控制器注册更改通知并且执行适当的操作。
[[NSNotificationCenter defaultCenter] addObserver:selfselector:@selector(updateHomes:)name:@"UpdateHomesNotification"object:nil];[[NSNotificationCenter defaultCenter] addObserver:selfselector:@selector(updatePrimaryHome:)name:@"UpdatePrimaryHomeNotification" object:nil];
展示 home 信息的视图控制器应该成为 home 对象的代理,并且当 home 发生改变时更新视图控制器的视图。
观察特定 home 对象的改变
1.在类接口中添加 home 代理协议。
@interface HomeViewController () @end
2.设置配件代理
home.delegate = self;
3.实现HMHomeDelegate协议
例如:实现 home:didAddAccessory:和 home:didRemoveAccessory: 方法来更新展示配件的视图。用 HMAccessory 类的 room 属性可以获得配件所属的 room。(对配件来说,默认的 room 是 roomForEntireHome 这个方法的返回值。)
Bridge Note:当你为 home 添加桥接口时,桥接口底层的配件会自动被添加到 home 中。你的代理会接收到桥接口后每个配件的 home:didAddAccessory:消息,但是你的代理不会接收到桥接口的 home:didAddAccessory:消息。
配件的状态可以在任何时间发生变化。配件可能不能被获得,可以被移除,或者被关闭。请更新用户界面以反映配件状态的更改,尤其是如果你的 App 允许用户控制配件时。
这以下步骤中,我们假设你已经从 HomeKit 数据库中检索到了配件对象,正如 Getting the Accessories in a Room中描述的那样。
在类接口中添加配件代理协议。
@interface AccessoryViewController () @end
设置配件的代理
accessory.delegate = self;
比如,执行 accessoryDidUpdateReachability:方法以启用或者禁用配件控制。
- (void)accessoryDidUpdateReachability:(HMAccessory *)accessory { if (accessory.reachable == YES) { // Can communicate with the accessory } else { // The accessory is out of range, turned off, etc }}
如果你展示了配件的服务状态和特性,那么请执行以下代理方法来相应地更新其视图:
accessory:service:didUpdateValueForCharacteristic:
想了解配件的服务,请参阅Accessing Services and Their Characteristics.
服务(HMService)代表了一个配件(accessory)的某个功能和一些具有可读写的特性(HMCharacteristic)。一个配件可以拥有多项服务,一个服务也可以有很多特性。比如一个车库开门器可能拥有一个照明和开关的服务。照明服务可能拥有打开/关闭和调节亮度的特性。用户不能制造智能家电配件和它们的服务-配件制造商会制造配件和它们的服务-但是用户可以改变服务的特性。一些拥有可读写属性的特性代表着某种物理状态,比如,一个恒温器中的当前温度就是一个只可读的值,但是目标温度又是可读写的。苹果预先定义了一些服务和特性的名称,以便让 Siri 能够识别它们。
在依照 Getting the Accessroties in a Room 中描述,你创建了一个配件对象之后,你可以获得配件的服务和特性。当然你也可以直接从 home 中按照类型获得不同的服务。
重要:不要暴露匿名服务-比如固件升级服务-给用户
通过 HMAccessory 类对象的 services 属性,我们可以获得一个配件的服务。
NSArray *services = accessroy.services;
要获得一个 home 当中配件提供的特定服务,使用 HMHome 类对象的 servicesWithTypes:方法。
// Get all lights and thermostats in a homeNSArray *lightServices = [home servicesWithTypes:[HMServicesTypeLightbulb]];NSArray *thermostatServices = [home servicesWithTypes:[HMServicesTypeThermostat]];
使用 HMServices 类对象的 name 属性来获得服务的名称
NSString *name = services.name;
要获得一个服务的特性,请使用 characteristics 属性。
NSArray *characteristics = service.characteristics
使用 servicesType 属性来获得服务的类型
NSString *serviceType = service.serviceType;
苹果定义了一些服务类型,并能被 Siri 识别:
使用 updateName:completionHandler:异步方法来改变服务名称。传入此方法的服务名称参数必须在一个 home 当中是唯一的,并且服务名可被 Siri 识别。
[service updateName:@"Garage 1 Opener" completionHandler:^(NSError *error) { if (error) { // Failed to change the name } else { // Successfully changed the name }}];
特性代表了一个服务的一个参数,它要么是只读、可读写或者只写。它提供了这个参数可能的值的信息,比如,一个布尔或者一个范围值。恒温器中的温度就是只读的,而目标温度又是可读写的。一个执行某个任务的命令且不要求任何返回-比如播放一段声音或者闪烁一下灯光来确认某个配件-可能就是只写的。
苹果定义了一些特性的类型,并能被Siri识别:
比如,对于一个车库开门器来说,目标状态就是打开或者关闭。对于一个锁来说,目标状态又是上锁和未上锁。
在你获得了一个 HMService 对象之后,如 Getting Services and Their Properties 所描述的,你可以获得每个服务的特性的值。因为这些值是从配件中获得的,这些读写的方法都是异步的,并可以传入一个完成回调的 block。
使用 readValueWithCompletionHandler:异步方法来读取一个特性的值。
[characteristic readValueWithCompletionHandler:^(NSError *error) { if (error == nil) { // Successfully read the value id value = characteristic.value; } else { // Unable to read the value} }];
在 if 语句块中,加入你的代码以更新 App 的视图。
使用 writeValue:completionHandler:异步方法来向一个特性写入值。
[self.characteristic writeValue:@42 withCompletionHandler:^(NSError *error) { if (error == nil) { // Successfully wrote the value } else { // Unable to write the value} }];
不要以为函数调用完成就意味着写入成功,实际上只有在当完成回调执行并没有错误产生时才表示写入成功。比如,直到一个开关的特性改变之前都不要改变这个开关的状态。在 if 语句块中,加入你的代码,以更新 App 的视图。
另外,在别的app更新了特性的值时也需要更新视图,在Observing Changes to Accessories中有描述。
一个服务组(HMServiceGroup)提供了控制不同配件的任意数量服务的快捷方式-比如,当用户离开家之后控制家中的某些灯。
在你创建了一个 HMHome 对象之后,如 Getting the Primary Home and Collection of Homes 中描述,你也就在这个家中创建一个服务组。
为了创建一个服务组,我们使用 HMHome 类对象的 addServiceGroupWithName:completionHandler:方法。方法中参数服务组的名称必须在此家中唯一,并可以被 Siri 识别。
[self.home addServiceGroupWithName:@"Away Lights" completionHandler:^(HMServiceGroup *serviceGroup, NSError *error) { if (error == nil) { // Successfully created the service group} else { // Unable to create the service group }];
我们使用 HMServiceGroup 类对象的 addService:completionHandler:方法来向服务组中添加一个服务。服务可以在一个或多个服务组中。
[serviceGroup addService:service completionHandler:^(NSError *error) { if (error == nil) { // Successfully added service to service group } // Unable to add the service to the service group }];
通过 HMHome 类对象的 serviceGroups 属性,来获得这个家的所有服务组。
NSArray *serviceGroups = self.home.serviceGroups;
通过 HMServiceGroup 类对象的 accessory 属性,我们获得服务所对应的智能电器。
HMAccessory *accessory = service.accessory;
和配件类似,代理方法在别的 App 改变服务组时也会被调用。如果你的app使用了服务组,请阅读 HMHomeDelegate Protocol Reference 文档,获悉你应该实现哪些方法以观察这些变化。
如果你没有智能电器(智能配件),你可以使用 HomeKit Accessroy Simulator 来模拟 home 中的智能电器。每个模拟配件都拥有服务和特性,你可以从你的 App 当中控制它。你的 App 在 HomeKit 数据库中创建对象和关系。它可以创建 home 布局,可以添加新的配件到模拟的 home 环境当中,最后向 home 中的每个房间添加智能配件。然后,你的app就能控制这些在 HomeKit Accessory Simulator 展示的模拟智能配件了。为了使用 HomeKit Accessory Simulator,请在iOS模拟器中运行你的应用程序,或者使用Xcode在iOS设备上运行应用程序。
HomeKit Accessory Simulator 是一个附加的开发者工具,不过并没有安装在 Xcode 当中。请按照 Download HomeKit Accessory Simulator 中所述的安装 HomeKit Accessory Simulator。
使用HomeKit Accessory Simulator来添加智能电器到模拟网络中。
向网络中添加智能电器配件,请按照下面的步骤添加:
如果想删除一个智能电器,请选择一个智能电器然后点击键盘上的Delete键。
一个智能电器需要一项服务和特性,你可以从 App 控制它。从预定义了服务列表中选择一项服务,并自定义特性。
按照下面步骤向智能电器中添加服务
1.在 HomeKit Accessory Simulator 中,选择 Accessories 列中的某个配件。
该配件的服务信息会展示在一个详情界面中。
注意:所有智能电器都有一个Accessory Information,显示在所有其他服务的下方。你可以向这个Accessory Information服务添加特性,但你不能删除默认的特性。
2.点击添加服务(Add Service),并从弹出视图中选择一个服务类型。
新添加的服务会在右边详细显示。HomeKit Accessory Simulator 为每种服务创建通用的特性。比如一个灯光服务的默认特性为色彩(Hue),饱和度(Saturation),亮度(Brightness)和开关。(开关特性和电源状态特性是一样的,正如 Accessing Values of Characteristics 中描述的那样。)一些特性是强制性的有一些也是可选择的。比如,开关特性就是强制性的,而色彩,饱和度,亮度这些特性都是可选择的。
你可以向服务中添加预定义的特性,或者自定义的特性。每种特性你都只能添加一个。
按照下面的步骤向服务中添加特性:
点击特性右边的减号来删除一个特性。如果特性右边并没有减号显示,这说明这个特性对这个服务来说是必须的。比如,你可以删除电灯服务中的色彩(Hue),饱和度(Saturation)和亮度(Brightness),但是你不可以删除开关特性。
在你通过 HomeKit Accessory Simulator 创建了一个智能电器后,运行你的App然后添加一个新的智能电器到你的家庭。
如何配对家庭中的智能电器:
在 Xcode 中,点击 Run 并调用 addAccessory:completionHandler:方法(如 Adding Accessories to Homes and Rooms中描述的那样).
在 HomeKit Accessory Simulator,setup code 显示在详情界面智能电器名称下。
关于如何编写代码来添加一个智能电器到家庭和房间请阅读Creating Homes and Adding Accessoris。
在HomeKit Accessory Simulator中,你可以获得智能电器的服务,并在其他HomeKit App中设置服务的特性值来模拟控制这个智能电器,或者手动地模拟控制智能电器。
想要控制一个智能电器你需要:
比如,为了改变一个灯泡的颜色(Hue),饱和度(Saturation)和亮度(Brightness),请滑动这个滑块。为了打开这个灯泡请选择 On 选项。
如果你的 App 展示了一个服务的特性,比如灯泡的开关状态,当你在HomeKit Accessory Simulator中改变这些特性的值时,它应当更新视图。
为了观察HomeKit数据库的变化,请阅读Observing HomeKit Database Changes。如果你想从 App 中通过编写代码来控制一个智能电器,请阅读Accessing Services and Characteristics。
为了模拟那些不支持 HomeKit Accessory Protocol 协议的智能电器,需要添加一个虚拟桥接口,然后将智能电器添加到这个虚拟桥接口。配置虚拟桥接口底层的智能电器和配置其他类型的智能电器差不多。
添加一个虚拟桥接口到网络
添加一个代表这个虚拟桥接口的智能电器。
为了添加一个虚拟桥接口到网络你需要:
可向一个虚拟桥接口添加一个或多个智能电器。
为了向一个虚拟桥接口添加一个智能电器,需要:
想要了解虚拟桥接口中的智能电器的详细信息,请选择虚拟桥接口部分中的智能电器。如果需要的话你可以点击虚拟桥接口旁边的查看详情来查看这个虚拟桥接口的智能电器。在你添加了一个服务和特性到这些智能电器之后,如Adding Services to Accessories和Adding Characteristics to Services中描述。它们会在这个虚拟桥接口被选择之后被展示出来。
将虚拟桥接口和home匹配的过程和将一个智能电器配置到一个home的过程是一样的,如 Adding Accessories to a Home in Your App 描述的。在虚拟桥接口底层的智能电器配件也一样被加入到了 home,如 Adding Bridges to Homes and Rooms 所描述。
如何控制虚拟桥接口底层的智能电器和直接控制智能电器的步骤一致,如 Controlling Accessories in HomeKit Accessory Simulator 中描述,除了你直接选择虚拟桥接口下的智能电器之外。
在iOS模拟器中你不能测试分享 HomeKit 数据库到多个 iOS 设备和用户。你应该安装你的App到多台iOS设备上,在这些设备中输入 iCloud 证书,然后运行你的 App。或者,使用 ad hoc 授权来在多台注册设备中测试你的app,如 Distributing Your App Using Ad Hoc Provisioning in App Distribution Guide 描述。
你的 App 应该应该可以允许一个用户邀请客人到你的家中,如 Managing Users 所述。
一个动作集合 HMActionSet 和触发器 HMTimerTrigger 允许你同时控制多个智能电器。比如,一个动作集合可能会在用户上床休息之前执行一组动作 HMAction。一个写动作向一个特性写入了值。动作集合中的动作是以不确定的顺序执行的。一个触发器会在一个特定的时间出发一个动作集并可以重复执行。每一个动作集合在一个家庭中都有唯一的名称并可被 Siri 识别。
写入动作会向一个服务的特性写入值并被加入到动作集合中去。HMAction 类是 HMCharacteristicWriteAction 具体类的抽象基类。一个动作有一个相关联的特性对象,你可以通过 Accessing Services and Characteristics 中描述的来获取相关的服务和特性,然后创建这个 HMCharacteristicWriteAction。
为了创建一个动作,我们使用 HMCharacteristicWriteAction类中的initWithCharacteristic:targetValue:方法。
HMCharacteristicWriteAction *action = [[HMCharacteristicWriteAction alloc] initWithCharacteristic:characteristic targetValue:value];
在你的代码中,你使用对应的特性的期望来替换 value 参数,并使用对应的 HMCharacteristic 对象来替换 characteristic 参数。
一个动作集就是一个共同执行的动作的集合。比如一个夜间动作集合可能包含关闭电灯,调低恒温水平和锁上房门。
为了创建一个动作集我们使用 addActionSetWithName:completionHandler:异步方法。
[self.home addActionSetWithName:@"NightTime" completionHandler:^(HMActionSet *actionSet, NSError *error) { if (error == nil) { // 成功添加了一个动作集 } else { // 添加一个动作集失败 }}];
为了添加一个动作到动作集,我们使用 addAction:completionHandler:异步方法。
[actionSet addAction:action completionHandler:^(NSError *error) { if (error == nil) { // 成功添加了一个动作到动作集 } else { // 添加一个动作到动作集失败 }}];
想要移除一个动作,可使用 removeAction:completionHandler:方法。
想要执行一个动作集,可使用 HMHome 类的 executeActionSet:completionHandler:方法。比如,用户希望控制所有的节日彩灯。我们就创建一个动作集来打开所有的节日彩灯,另外一个动作集来关闭所有的节日彩灯。为了打开所有的节日彩灯,发送executeActionSet:completionHandler:消息给 home 对象,并传递"打开节日彩灯"动作集。
触发器会执行一个或多个动作集。iOS会在后台管理和运行你的触发器。HMTrigger 类是 HMTimerTrigger 具体类的抽象类。当你创建一个定时触发器时,你需要指定触发时间和触发的周期。创建并开启一个定时触发器需要多个步骤来完成。
遵循下面几步来创建并启动一个定时触发器
创建一个定时触发器
1.创建定时触发器。
self.trigger = [[HMTimerTrigger alloc] initWithName:namefireDate:fireDatetimeZone:niLrecurrence:nilrecurrenceCalendar:nil];
触发时间必须设置在将来的某个时刻,第二个参数必须为 0。如果你设置了一个周期,周期的最小值是 5 分钟,最大值是 5周。关于如何使用NSDateComponents和NSCalendar来设置周期,请阅读Date and Time Programming Guide
2.添加一个动作集到触发器。
使用 HMTrigger 基类方法 addActionSet:completionHandler:,来添加一个动作集到触发器。
3.添加一个触发器到家庭。
使用 HMHome 类中的addTrigger:completionHandler:方法来添加一个触发器到家庭。
4.启动触发器。
新创建的触发器默认是未启动的。需要使用 enable:complationHandler:方法启动触发器。
一个定时触发器被启动后,会周期性的运行它的动作集。
创建 home 的用户是该 home 的管理员,可以执行所有操作,包括添加一个客人用户到 home。任何管理员添加到这个 home 的用户(HMUser)都有一个有限的权限。客人不能更改家庭的布局,但是可以执行下面的动作:
比如,一个家庭的户主可以创建一个 home 布局并向其中添加家庭成员。每个家庭成员必须拥有一个 iOS 设备和 Apple ID 以及相关的 iCloud 账户。iCloud 需要个人输入的 Apple ID 和户主提供的 Apple ID 相吻合,以便让他们访问这个 home。考虑到隐私问题,Apple ID 对你的 App 是不可见的。
管理员需要遵从以下步骤来添加一个客人到 home 中:
管理员调用一个动作将客人添加到home中。
你的App调用addUserWithCompletionHandler:异步方法。
HomeKit展示一个对话框,要求输入客人的Apple ID。
用户输入客人的Apple ID。
在完成回调中返回一个新的用户。
添加一个客人到home,需要在客人的iOS设备上做以下操作:
用户在iCloud偏好设置中输入iCloud凭证(Apple ID和密码)。
用户启动你的App。
你的App通过home manager object获得一个home集合。
客人执行的操作可能会失败。如果一个异步方法中出现 HMErrorCodeInsufficientPrivileges 错误码的话,这就意味着用户没有足够的权限来执行动作-也许这个用户只是客人,而不是管理员。
为了测试你的 App 是否正确处理了客人用户,请阅读Testting Multiple iOS Devices and Users。
为了添加一个客人用户到home,请使用addUserWithCompletionHandler:异步方法。
[self.home addUserWithCompletionHandler:^(HMUser *user, NSError *error) { if (error == nil) { // Successfully added a user } else { // Unable to add a user} }];
想要移除 home 中的用户,请使用 HMHome 类的 removeUser:completionHandler:方法。
通过实现 HMHomeDelegate 协议中的 home:didAddUser:和 home:didRemoveUser:协议方法检查新添加和移除的用户并更新视图。关于如何创建一个delegate,请阅读Observing Changes to Individual Homes。
出于隐私的考虑,你的 App 对用户名只有读得权限,并不能读写用户的 Apple ID。使用 HMHome 对象的 users 属性来获取用户。使用 HMUser 类的 name 属性来获取用户名。