
最近一直比較有空,所以就在這段空閑時間就打算梳理一下以前做過的一些項目,并且把一些簡單常用的API回顧回顧,今天看了一下指紋解鎖的官方Demo,就想學(xué)習(xí)一下。其實功能實現(xiàn)起來很簡單,畢竟蘋果官方都幫我們封裝好了,我們只需要學(xué)習(xí)幾個方法,實現(xiàn)一下即可讓自己的App有手勢解鎖的功能了。

一、API介紹
iOS的指紋解鎖官方提供了
LocalAuthentication.framework這個系統(tǒng)庫,打開這個庫我們會發(fā)現(xiàn)里面只有以下4個.h文件
- LAContext.h
- LAError.h
- LAPublicDefines.h
- LocalAuthentication.h
①LAContext.h(最核心部分)
一開始我們就看到LAPolicy這個枚舉
typedef NS_ENUM(NSInteger, LAPolicy)
LAPolicyDeviceOwnerAuthenticationWithBiometrics(iOS8以上可用):這種代表的是只用指紋去驗證。第一次指紋失敗,會出現(xiàn)“輸入密碼”按鈕,輸入密碼的標題及功能可以自定義;第三次指紋失敗,彈窗消失;再次啟動驗證,還有兩次機會,如果都失敗了,指紋驗證鎖定,不再彈出驗證窗。直至輸入密碼來解鎖指紋(可以鎖屏重新進來使用輸入密碼的方式解鎖)。
LAPolicyDeviceOwnerAuthentication(iOS9以上可用):這種代表的是可以用指紋或密碼兩種方式去驗證,優(yōu)先用指紋。第一次指紋失敗,會出現(xiàn)“輸入密碼”按鈕,輸入密碼的標題可以自定義,但是功能不能自定義了,而是必須輸入系統(tǒng)密碼(鎖屏密碼);第三次驗證失敗,彈窗消失,彈出輸入系統(tǒng)密碼的界面;如果連續(xù)五次指紋失敗,則指紋鎖定,此時只會彈出輸入密碼界面,直至輸入密碼成功解鎖。
兩種驗證方式的比較:
相同點:都是連續(xù)五次驗證失敗就會鎖定
不同點:前者的輸入密碼功能可以自定義,后者輸入密碼功能是固定輸入系統(tǒng)密碼一般經(jīng)常使用LAPolicyDeviceOwnerAuthenticationWithBiometrics
然后我們可以看到以下面幾個方法
①這個方法是判斷當前設(shè)備是否支持TouchID的
/*
這個方法用來檢查當前設(shè)備是否可用touchID,返回一個BOOL值
policy: 這個就是上面的枚舉的兩個驗證方式,一般用前者
error: 錯誤的類型可參考LAError.h里的類型
*/
- (BOOL)canEvaluatePolicy:(LAPolicy)policy error:(NSError * __autoreleasing *)error __attribute__((swift_error(none)));
@discussion Policies can have certain requirements which, when not satisfied, would always cause
/// the policy evaluation to fail. Examples can be a passcode set or a fingerprint
/// enrolled with Touch ID. This method allows easy checking for such conditions.
///
/// Applications should consume the returned value immediately and avoid relying on it
/// for an extensive period of time. At least, it is guaranteed to stay valid until the
/// application enters background.
@return YES if the policy can be evaluated, NO otherwise.
②這個方法是用來驗證TouchID的,會有彈出框出來
/*
這個方法是開始驗證指紋的方法
policy: 這個就是上面的枚舉的兩個驗證方式,一般用前者
localizedReason: 指紋驗證框上面的提示信息,一般為“通過Home鍵驗證已有手機指紋”(不能為空否則崩潰)
reply: 一個block,返回指紋驗證結(jié)果,成功:success為YES,失敗:success為NO,同時返回錯誤類型的error,同樣參考LAError.h里的類型
*/
- (void)evaluatePolicy:(LAPolicy)policy
localizedReason:(NSString *)localizedReason
reply:(void(^)(BOOL success, NSError * __nullable error))reply;
③這個方法用來廢止該實例對象context
- (void)invalidate NS_AVAILABLE(10_11, 9_0);
然后下面還有幾個方法不是很常用,我們可以忽略
接下來介紹一下比較重要的屬性
// 可以設(shè)置指紋彈框“輸入密碼”按鈕的標題,如果不設(shè)置或設(shè)置為nil,則顯示默認的“輸入密碼”;如果設(shè)置為@"",則彈框不再顯示這個按鈕
@property (nonatomic, nullable, copy) NSString *localizedFallbackTitle;
// 可以設(shè)置指紋彈框“取消”按鈕的標題(iOS10.0以上可用),如果不設(shè)置或設(shè)置為nil或設(shè)置為@"",都顯示默認的“取消”
@property (nonatomic, nullable, copy) NSString *localizedCancelTitle NS_AVAILABLE(10_12, 10_0);
// 最大指紋嘗試錯誤次數(shù)(iOS8.3 - iOS9.0可用)
@property (nonatomic, nullable) NSNumber *maxBiometryFailures NS_DEPRECATED_IOS(8_3, 9_0) __WATCHOS_UNAVAILABLE __TVOS_UNAVAILABLE;
// 這個可以檢測你的指紋數(shù)據(jù)庫的變化,增加或者刪除指紋這個屬性會做出相應(yīng)的反應(yīng)(iOS9.0以上可用)
@property (nonatomic, nullable, readonly) NSData *evaluatedPolicyDomainState NS_AVAILABLE(10_11, 9_0) __WATCHOS_UNAVAILABLE __TVOS_UNAVAILABLE;
// 兩次開啟指紋之間的時間間隔,決定第二次是否需要指紋解鎖
@property (nonatomic) NSTimeInterval touchIDAuthenticationAllowableReuseDuration NS_AVAILABLE(NA, 9_0) __WATCHOS_UNAVAILABLE __TVOS_UNAVAILABLE;。
②LAError.h
這個類主要枚舉了可能出現(xiàn)的錯誤類型,具體中文解釋如下
LAErrorAuthenticationFailed //連續(xù)三次指紋驗證失敗,可能指紋模糊或用錯手指
LAErrorUserCancel //用戶取消驗證,點擊了取消按鈕
LAErrorUserFallback //用戶取消驗證,點擊了輸入密碼按鈕
LAErrorSystemCancel //系統(tǒng)取消授權(quán),如其他APP切入
LAErrorPasscodeNotSet //指紋驗證無法啟動/失敗,因為設(shè)備沒有設(shè)置密碼
LAErrorTouchIDNotAvailable //設(shè)備TouchID不可用,例如未打開
LAErrorTouchIDNotEnrolled //指紋驗證無法啟動,因為沒有錄入指紋(設(shè)置密碼了)
LAErrorTouchIDLockout //設(shè)備TouchID被鎖定,因為失敗的次數(shù)太多了
LAErrorAppCancel //應(yīng)用程序取消了身份驗證,APP調(diào)用了-(void)invalidate方法使LAContext失效
LAErrorInvalidContext //實例化的LAContext對象失效,再次調(diào)用evaluation...方法則會彈出此錯誤信息
③LAPublicDefines.h
只是這個庫的一些宏定義

④LocalAuthentication.h
這個里面就兩行引入頭文件,很顯然這是我們引入指紋庫要調(diào)用的類,在文件中引入#import <LocalAuthentication/LocalAuthentication.h>即可。
二、實現(xiàn)指紋解鎖
*第1步:引入指紋解鎖頭文件
#import <LocalAuthentication/LocalAuthentication.h>
*第2步:實現(xiàn)指紋解鎖的主要方法
- (void)evaluateAuthenticate
{
//iOS 8以上才支持指紋識別接口
if ([[UIDevice currentDevice].systemVersion floatValue] < 8) {
NSLog(@"不支持TouchID (版本必須高于iOS 8.0才能使用)");
return;
}
//創(chuàng)建LAContext
LAContext *context = [[LAContext alloc] init];
context.localizedFallbackTitle = @"輸入密碼吧";
NSError *Error = nil;
//判斷設(shè)備支持狀態(tài)
if ([context canEvaluatePolicy:LAPolicyDeviceOwnerAuthenticationWithBiometrics error:&Error]) {
//支持指紋驗證
[context evaluatePolicy:LAPolicyDeviceOwnerAuthenticationWithBiometrics localizedReason:@"請驗證已有手機指紋" reply:^(BOOL success, NSError *error) {
if (success) {
//驗證成功,主線程處理UI
[[NSOperationQueue mainQueue] addOperationWithBlock:^{
NSLog(@"指紋驗證成功");
}];
} else {
NSLog(@"驗證失敗 == %@", error.localizedDescription);
switch (error.code) {
case LAErrorSystemCancel:{
NSLog(@"系統(tǒng)取消授權(quán),如其他APP切入");
}
break;
case LAErrorUserCancel:{
NSLog(@"用戶取消驗證,點擊了取消按鈕");
}
break;
case LAErrorUserFallback:{
NSLog(@"用戶取消驗證,點擊了輸入密碼按鈕");
[[NSOperationQueue mainQueue] addOperationWithBlock:^{
//用戶選擇輸入密碼,切換主線程處理
}];
}
break;
case LAErrorAuthenticationFailed:{
NSLog(@"連續(xù)三次指紋驗證失敗,可能指紋模糊或用錯手指");
}
break;
case LAErrorTouchIDLockout:{
NSLog(@"設(shè)備TouchID被鎖定,因為失敗的次數(shù)太多了");
}
break;
default:{
NSLog(@"設(shè)備TouchID不可用。。。");
[[NSOperationQueue mainQueue] addOperationWithBlock:^{
//其他情況,切換主線程處理
}];
}
break;
}
}
}];
} else {
//該設(shè)備不支持TouchID
NSLog(@"不支持TouchID == %@", Error.localizedDescription);
switch (Error.code) {
case LAErrorTouchIDNotEnrolled:{
NSLog(@"指紋驗證無法啟動,因為沒有錄入指紋");
}
break;
case LAErrorPasscodeNotSet:{
NSLog(@"指紋驗證無法啟動,因為設(shè)備沒有設(shè)置密碼");
}
break;
case LAErrorTouchIDLockout:{
NSLog(@"設(shè)備TouchID被鎖定,因為失敗的次數(shù)太多了");
}
break;
default:{
NSLog(@"設(shè)備TouchID不可用。。。");
}
break;
}
}
}
最后就附上簡單的Demo原碼,如果文章對你有幫助記得點贊哦!