iOS手势密码实现机制

2016-10-25  本文已影响503人  Joshua520

基于手势密码的安全性与交互性比较好,一直以来都是手机APP的标配。在接手公司项目的时候,项目已经实现了手势密码。但是集成在项目中的手势密码功能一直没有达到老大要求和手Q一致的需求。在研究了手Q的弹出机制之后,我发现主要的差距是在app进入后台再次点击图标进入程序:手势密码界面会滞后?(即手势密码弹出是在主界面出现之后再弹出)

我查看了代码实现是写一个基础ViewController,在viewDidAppear监听手势密码监听事件,代码示例如下:

[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(verifyGesturePwd) name:@"VerifyGesturePwd" object:nil];

我们姑且不论代码的书写,此代码在不继承这个基础view controller的情况下,手势密码是弹不出的。并且实现不了直接弹出手势密码的要求。

中间我尝试在这个基础上做修改,即把监听代码放到viewWillAppear中,结果你肯定也猜到了,没有用,因为这还是在界面生成之后弹出的。那怎么解决这个需求呢?我陷入了这个死胡同。难道这个问题就解决不了了吗?答案肯定是可以解决,因为人家已经实现了这个功能。

最近刚好看到订阅专栏里面一篇关于“万能钥匙”的文章,文章的主题就是不要盯着锁头找钥匙,要到其他地方找钥匙。很显然我就是陷入了盯着问题找答案的死胡同,此时我豁然开朗。找到解题的关键:答案在别处。我应该在程序入口的地方来完成项目需求。下面给出我的实现。

1、在每次程序启动时,在didFinishLaunchingWithOptions方法中判断是否设置手势密码,如果有手势密码功能的话,直接把手势密码界面设置为rootViewController,没有的话就判断是否登录账号,登录就设置MainViewController为rootViewController,否则就设置登录界面为rootViewController。并注册一个登录的监听。

if ([DBUtil objectForKey:kDB_GestureValid]) {

VertifyGesViewController *controller = [VertifyGesViewController new];

UINavigationController *nav = [[UINavigationController alloc] initWithRootViewController:controller];

self.window.rootViewController = nav;

}

else{

[self loginStateChange:nil];

[[NSNotificationCenter defaultCenter] addObserver:self

selector:@selector(loginStateChange:)

name:KNOTIFICATION_LOGINCHANGE

object:nil];

其中loginStateChange的处理逻辑如下:

- (void)loginStateChange:(NSNotification *)notification{

if (notification == nil) {

[self userLogined];

}else{

BOOL loginSuccess = [notification.object boolValue];

if (loginSuccess) {

[self userLogined];

}

else{

[self userLoginOut];

}

}

}

- (void)userLogined{

self.window.rootViewController = [MainViewController new];

}

- (void)userLoginOut{

self.window.rootViewController = [[UINavigationController alloc]initWithRootViewController:[LoginViewController new]];

}

2、 为了处理方便,我在手势密码验证后把逻辑放到登录状态监听里面。即在验证后判断如果当前手势界面是设置的rootViewController时,发出通知。监听处需要重新设置rootViewController。

if (self.navigationController) {

[[NSNotificationCenter defaultCenter] postNotificationName:KNOTIFICATION_LOGINCHANGE object:@YES];

}else{

[self dismissViewControllerAnimated:YES completion:^{}];

}

3、在程序进入后台时,如果手势密码已设置,弹出手势密码即可。

if ([DBUtil boolForKey:kDB_GestureValid]) { //手势密码有效

[self.window.rootViewController presentViewController:[VertifyGesViewController new] animated:NO completion:nil];

}

4、在测试的时候发现有个小bug,即在杀死程序再次进入应用时,直接进入后台发现手势密码界面有2个。此处只需要添加如下代码逻辑来解决重复弹出的问题。

UIViewController *rootViewController = [[[UIApplication sharedApplication] delegate] window].rootViewController;

if ([rootViewController isKindOfClass:[UINavigationController class]]) {

return;

}

行文至此,这个功能需求被实现了出来。说一下自己的一点小感悟:1、很多你觉得不可能完成的任务,其实是你还没有找到解决方法。2、不要盯着锁头找钥匙,钥匙一定在其他地方。

今天抽时间把手势密码抽离出来,https://github.com/hohua88/Gesture,欢迎交流。

#未完再续

上述实现在当前页面有Alert或者键盘时,弹出的验证手势密码界面会出现遮挡(既Alert或者键盘)。这是由于Alert/键盘是展示在UIWindow上的,会显示在手势密码界面的上面。

有兴趣的可以去Window官方文档,这里就不展开描述了。这里为了解决遮挡问题,可以通过设置UIWindowLevel为UIWindowLevelAlert来做到手势密码显示在最上层。

- (void)show {

        self.windowLevel = UIWindowLevelAlert;

        [self makeKeyWindow];

        self.hidden = NO;

}

这样做的话,想要再弹出Alert提示框就需要自己自定义添加到window上。想尝试的可以自己实现一下。

参考链接:UIWindow 详解及使用场景

上一篇下一篇

猜你喜欢

热点阅读