最近在做一套登录系统,包括指纹登录、手势登录以及账号密码登录,在此简单记录一下指纹的处理逻辑。

  


指纹处理:
1、处理iOS 9.0之后,指纹被锁不会自动弹出密码解锁的情况;
2、处理iOS 10.*系统不区分未录入指纹和未设置密码的情况;
流程图:此处有图,周末再画
 
 
 周末补上
 
 
 
代码如下:
 
.h:

//  Created by ever on 17/3/7.
// Copyright © 2017年 ever. All rights reserved.
//
#import <Foundation/Foundation.h>
typedef NS_ENUM(NSInteger, TITouchIDVerificationError) {
TITouchIDVerificationErrorLocalizedReason,
TITouchIDVerificationErrorAuthFailMessage,
TITouchIDVerificationErrorNotAvailableMessage,
TITouchIDVerificationErrorNotEnrolledMessage,
TITouchIDVerificationErrorLockOut,
TITouchIDVerificationErrorPasscodeNotSetMessage,
TITouchIDVerificationErrorUserCancel,
TITouchIDVerificationErrorAppCancel,
TITouchIDVerificationErrorUnknown
}; typedef NS_ENUM(NSInteger, TIVerificationStatus) {
TIVerificationStatusSuccess,
TIVerificationStatusPasswordNotSet,
TIVerificationStatusNotEncrolled,
TIVerificationStatusFailed,
TIVerificationStatusFallback,
TIVerificationStatusLockOut,
TIVerificationStatusCancel,
TIVerificationStatusError,
}; typedef void(^VerificationCompletion)(TIVerificationStatus status, NSError *error); @interface LYYTouchID : NSObject
/**
调用TouchID @param completion 验证完成,如果返回YES,验证成功;如果返回NO,验证失败(输入密码和指纹验证失败)
*/
+ (void)touchVerificationWithFallbackTitle:(NSString*)fallbackTitle completion:(VerificationCompletion)completion; @end
 
.m:
//  Created by ever on 17/3/7.
// Copyright © 2017年 ever. All rights reserved. #import "LYYTouchID.h"
#import <UIKit/UIKit.h>
#import <LocalAuthentication/LocalAuthentication.h> #define TI_IS_iOS_LATER(num) ([UIDevice currentDevice].systemVersion.floatValue >= (num))
#define TI_IS_iOS8_LATER TI_IS_iOS_LATER(8.0)
#define TI_IS_iOS9_LATER TI_IS_iOS_LATER(9.0)
#define TI_IS_iOS10_LATER TI_IS_iOS_LATER(10.0)
#define TI_IS_iOS11_LATER TI_IS_iOS_LATER(11.0) NSString * const TouchIDLocalizedReason = @"请您验证已有TouchID";
NSString * const TouchIDNotAvailableMessage = @"您当前设备不支持TouchID";
NSString * const TouchIDNotEnrolledMessage = @"您当前没有录入指纹信息";
NSString * const TouchIDNotCorrectMessage = @"您的指纹信息不匹配";
NSString * const TouchIDPasscodeNotSetMessage = @"您当前没有设置密码";
NSString * const TouchIDCancelMessage = @"您已取消验证TouchID";
NSString * const TouchIDUnknownErrorMessage = @"出现未知错误";
NSString * const TouchIDLockOutMessage = @"TouchID 被锁定"; static BOOL _isTouchIDSecondCheck;//用于存储第一遍canEvaluatePolicy检查到的指纹被锁的状态
static BOOL _touchIDSecondCheck;//用于存储canEvaluatePolicy是否是第二次查询 @interface LYYTouchID ()
@end @implementation LYYTouchID #pragma mark - 验证
+ (void)touchVerificationWithFallbackTitle:(NSString*)fallbackTitle completion:(VerificationCompletion)completion{
LAContext *context = [[LAContext alloc] init];
context.localizedFallbackTitle = fallbackTitle?:@"";
LAPolicy policy = [self setPolicy]; NSError *notSupportError = nil;
if ([context canEvaluatePolicy:policy error:&notSupportError]) { //处理10.*系统纯指纹策略不走passwordNotSet方法
if (_touchIDSecondCheck&&TI_IS_iOS10_LATER&&!TI_IS_iOS11_LATER) {//第二次检测指纹是否可用(指纹+密码策略),且可以
_touchIDSecondCheck = NO;
completion(TIVerificationStatusNotEncrolled,[self returnErrorInfoWithErrorCode:LAErrorTouchIDNotEnrolled]);
return;
} //不是第二次检测指纹是否可用,直接进行验证
[context evaluatePolicy:policy localizedReason:TouchIDLocalizedReason reply:^(BOOL success, NSError * _Nullable error) { dispatch_async(dispatch_get_main_queue(), ^{
if (success) { if (TI_IS_iOS9_LATER&&_isTouchIDSecondCheck) {//TouchID 解锁成功时,更改为纯指纹策略,验证TouchID
_isTouchIDSecondCheck = NO;
[self touchVerificationWithFallbackTitle:fallbackTitle completion:completion];
}else{//ios9.0之前版本不支持密码策略直接返回信息
_isTouchIDSecondCheck = NO;
completion(TIVerificationStatusSuccess,nil);
}
}
else {
int code = (int)error.code;
////设置锁定状态为NO
_isTouchIDSecondCheck = NO;
[self dealErrorWithErrorCode:code fallbackTitle:fallbackTitle complection:completion];
} }); }];
}else { [self dealErrorWithErrorCode:(int)notSupportError.code fallbackTitle:fallbackTitle complection:completion];
}
}
#pragma mark - 设置指纹策略
+ (LAPolicy)setPolicy{
LAPolicy policy; if (_touchIDSecondCheck&&TI_IS_iOS9_LATER){//已经查验过纯指纹策略,且纯指纹策略不支持,改变策略为指纹+密码进行二次验证
policy = LAPolicyDeviceOwnerAuthentication;
return policy;
}
if (_isTouchIDSecondCheck&&TI_IS_iOS9_LATER) {//指纹被锁定且系统大于9.0,使用“指纹+密码”策略
policy = LAPolicyDeviceOwnerAuthentication;
}else{//使用“指纹”策略(默认是指纹策略)
policy = LAPolicyDeviceOwnerAuthenticationWithBiometrics;
}
return policy;
}
#pragma mark - 处理不同错误
+ (void)dealErrorWithErrorCode:(int)code fallbackTitle:(NSString*)fallbackTitle complection:(VerificationCompletion)completion{ switch (code) {
case LAErrorUserFallback:
completion(TIVerificationStatusFallback,[self returnErrorInfoWithErrorCode:code]);
break; case LAErrorAuthenticationFailed:
completion(TIVerificationStatusFailed,[self returnErrorInfoWithErrorCode:code]);
break; case LAErrorSystemCancel:{
_isTouchIDSecondCheck = NO;
_touchIDSecondCheck = NO;
}
break; case LAErrorPasscodeNotSet:
_isTouchIDSecondCheck = NO;
_touchIDSecondCheck = NO;
completion(TIVerificationStatusPasswordNotSet,[self returnErrorInfoWithErrorCode:code]);
break; case LAErrorTouchIDNotAvailable: break; case LAErrorAppCancel:{
_isTouchIDSecondCheck = NO;
_touchIDSecondCheck = NO;
}
break;
case LAErrorUserCancel:
_isTouchIDSecondCheck = NO;
_touchIDSecondCheck = NO;
completion(TIVerificationStatusCancel,[self returnErrorInfoWithErrorCode:code]);
break; default: {
if (TI_IS_iOS9_LATER) {
if (code==LAErrorTouchIDLockout) {
//ios9.0更改为密码策略,解锁TouchID
_isTouchIDSecondCheck = YES;
[self touchVerificationWithFallbackTitle:fallbackTitle completion:completion];
break; }
}
if (TI_IS_iOS8_LATER&&!TI_IS_iOS11_LATER) {
if (code==LAErrorTouchIDNotEnrolled) {
//处理ios 10.*系统不走passwordNotSet
if (TI_IS_iOS10_LATER&&!_touchIDSecondCheck) {//ios 9.0之后,第一次返回指纹不可用的结果(这时是纯指纹策略)
_touchIDSecondCheck = YES;
[self touchVerificationWithFallbackTitle:fallbackTitle completion:completion];
}else{
//ios 9.0 之后是第二次返回不可用结果,此时采用的是指纹+密码策略,此时指纹确实不可用
//ios 9.0 之前,第一次返回指纹不可用的结果,此时直接返回指纹不可用的错误
_touchIDSecondCheck = NO;
completion(TIVerificationStatusNotEncrolled,[self returnErrorInfoWithErrorCode:code]);
} break;
}
}
if (@available(iOS 11.0, *)) {
if (code==LAErrorBiometryNotEnrolled) {
completion(TIVerificationStatusNotEncrolled,[self returnErrorInfoWithErrorCode:code]);
break;
}
} completion(TIVerificationStatusError,[self returnErrorInfoWithErrorCode:code]);
}
break; }
} #pragma mark - 生成error
+ (NSError *)returnErrorInfoWithErrorCode:(int)errorCode { if (TI_IS_iOS9_LATER) {
if (errorCode == LAErrorAppCancel) {
NSError *tempError = [NSError errorWithDomain:@"TouchIDVerfityDomain" code:TITouchIDVerificationErrorAppCancel userInfo:@{@"message":TouchIDCancelMessage}];
return tempError;
}
} NSString *errorMessage = @"";
NSUInteger errCode = ; switch (errorCode) {
case LAErrorTouchIDNotAvailable:
errorMessage = TouchIDNotAvailableMessage;
errCode = TITouchIDVerificationErrorNotAvailableMessage;
break;
case LAErrorAuthenticationFailed:
errorMessage = TouchIDNotCorrectMessage;
errCode = TITouchIDVerificationErrorAuthFailMessage;
break;
case LAErrorPasscodeNotSet:
errorMessage = TouchIDPasscodeNotSetMessage;
errCode = TITouchIDVerificationErrorPasscodeNotSetMessage;
break;
case LAErrorUserCancel:
errorMessage = TouchIDCancelMessage;
errCode = TITouchIDVerificationErrorUserCancel;
break;
default:
if (TI_IS_iOS9_LATER) {
if (errorCode==LAErrorTouchIDLockout) {
errorMessage = TouchIDLockOutMessage;
errCode = TITouchIDVerificationErrorLockOut;
break;
}
}
if (TI_IS_iOS8_LATER&&!TI_IS_iOS11_LATER) {
if (errorCode==LAErrorTouchIDNotEnrolled) {
errorMessage = TouchIDNotEnrolledMessage;
errCode = TITouchIDVerificationErrorNotEnrolledMessage;
break;
}
}
if (@available(iOS 11.0, *)) {
if (errorCode==LAErrorBiometryNotEnrolled) {
errorMessage = TouchIDNotEnrolledMessage;
errCode = TITouchIDVerificationErrorNotEnrolledMessage;
break;
}
}
errorMessage = TouchIDUnknownErrorMessage;
errCode = TITouchIDVerificationErrorUnknown;
break;
} NSError *tempError = [NSError errorWithDomain:@"TouchIDVerfityDomain" code:errCode userInfo:@{@"message":errorMessage}];
//
return tempError;
}
@end
 
 
 

ios --指纹TouchID的更多相关文章

  1. iOS 指纹解锁 验证TouchID

    iOS指纹解锁 1.首先,引入依赖框架 LocalAuthentication.framework #import <LocalAuthentication/LocalAuthenticatio ...

  2. iOS 指纹认证登陆开发(TouchID)

    设计思路 TouchID 关联账号 用户登陆成功 -> 开启TouchID登陆 -> TouchID验证 -> 记录用户信息(验证通过) -> 关联完成 TouchID 登陆 ...

  3. IOS指纹识别调用

    最近正在开发的一个app需要加入指纹识别的功能,先搜索一下找到官方文档,简单易懂: https://developer.apple.com/library/ios/documentation/Loca ...

  4. iOS指纹识别Touch ID的安全性探讨

    苹果公司在 iPhone 5s 的发布会上公布了全新的指纹识别安全技术,也就是 Touch ID,开创了生物安全识别技术在便携设备上使用的新篇章.此后,苹果还将此技术带到了 iPad 上.此前没有任何 ...

  5. iOS指纹识别代码

    1:添加LocalAuthentication.framework框架 2:实现过程 #import "ViewController.h" #import <LocalAut ...

  6. iOS 指纹解锁

    目前常用的App支持指纹解锁的还不是很多,如果在你的项目中用一下是不是显得高大上呢? 废话不说多,干货- 1.在工程中添加LocalAuthentication.framework 2.在需要验证的c ...

  7. iOS指纹识别

    #import "ViewController.h" #import <LocalAuthentication/LocalAuthentication.h> @inte ...

  8. IOS 指纹识别的简单使用

    首先导入LocalAuthentication框架 然后导入头文件 #import <LocalAuthentication/LAPublicDefines.h> - (void)begi ...

  9. ios 指纹识别解锁

    :添加LocalAuthentication.framework框架 :实现过程 #import "ViewController.h" #import <LocalAuthe ...

随机推荐

  1. HDMI SCDC处理过程

    SCDC       State and Control Data Channel 接收端如果有SCDC,应该在E-EDID中包含一个有效的HF-VSDB,并且把SCDC_Present位设置为1.在 ...

  2. Python学习第一章

    1.Python保留字即是关键字,我们不可以把他们当作任何标识符名称. python的标准库提供了一个keyword模板,可以输出当前版本的关键字: >>>import keywor ...

  3. ReactNative学习笔记(二)基础进阶

    一个最简单的HelloWorld页面 先不多解释,直接上代码: import React, { Component } from 'react'; import {AppRegistry, Style ...

  4. MySQL基础--字符函数

    1.UPPER和UCASE返回字符串str,根据当前字符集映射(缺省是ISO-8859-1 Latin1)把所有的字符改变成大写.该函数对多字节是可靠的. 2.LOWER和LCASE返回字符串str, ...

  5. 【BZOJ4025】 二分图(线段树分治)

    传送门 BZOJ Solution 只是为了学习一下线段树分治的啦! 当你学会线段树分治之后,可以跳过下面的一部分: 按照时间搞一颗线段树出来,把包含这段区间的操作用vector压进去. 每一个线段树 ...

  6. HttpRunnerManager接口测试平台部署在服务器上(Centos + python3.6 + Mysql5.7 + uwsgi + nginx)

    整个思路:HttpRunnerManager + python3.6 + Mysql5.7 + uwsgi + nginx 安装依赖环境,将源代码传到服务器,修改settings,增加uwsgi配置, ...

  7. 词向量:part 2 CBoW、Skip-Gram、Negative Sampling、Hierarchical Softmax、GloVe、fastText、doc2vec

  8. 在Windows环境中安装Neo4j

    图形数据库(Graph Database)是NoSQL数据库家族中特殊的存在,用于存储丰富的关系数据,Neo4j 是目前最流行的图形数据库,支持完整的事务,在属性图中,图是由顶点(Vertex),边( ...

  9. 【随笔】nginx add_header指令的使用

    nginx配置文件通过使用add_header指令来设置response header. 具体方法如下: add_header key value add_header Cache-Control n ...

  10. 扩展方法IEnumerable<T>转换为IList<SelectListItem> ,提供@Html.DropDownList使用

    由于在MVC中经常会使用到@Html.DropDownList方法,而该方法接收的是List<SelectListItem> 参数,因此就想着写一个扩展方法,直接把IEnumerable转 ...