新年上班第二天,呱呱给大家拜个晚年,祝大家工作顺意,生活美满。
相信大家都知道UIAlertView已经被苹果抛弃了,它的使命到iOS7也就结束了,取而代之的是它的哥们UIAlertController,从iOS8开始使用。但尽管如此,UIAlertView现在还是有不少人在使用的,今天我就讲一个这货带来的坑,以及如何填坑。
坑
使用UIAlertView过程中应该会遇到要去掉屏幕所有的alert框或者是只想保留多个弹出框的其中一个这种类似的问题。查阅资料后会发现在iOS7之前还是挺简单的:
AlertVIew 显示的原理是在屏幕上家里一层 Alertwindow , 然后AlertView 的视图加在了这个Window 上,所以可以通过获取window上的视图从而获取UIAlertView 然后令其消失。代码如下:
for (UIWindow* w in [UIApplication sharedApplication].windows)
for (NSObject* o in w.subviews)
if ([o isKindOfClass:[UIAlertView class]])
[(UIAlertView*)o dismissWithClickedButtonIndex:[(UIAlertView*)o cancelButtonIndex] animated:YES];
}
}
}
但是在iOS7之后,就不是这样了:
在iOS 7 中,UIAlertView 的 window 并不是显示在[UIApplication shareApplication].windows 中,而是被加入到了[UIApplication keyWindow] 中,但是并没有留出引用或者说指针,所以我们没有办法获取到
想要狠下心来全部替换成UIAlertController,替换到一半的时候想起来不对,系统自己弹出来的提示框(比如那种权限授权框)没法换啊,这尼玛自己说的要放弃,又是你自己还在用,心中各种草泥马奔腾而过。那就换其他办法好了。
填坑
这里笔者给大家提供一个方法,大家有更好方法的可以在下面留言。
思路是这样的:既然没法通过window获取到显示出来的alert对象,那我们就在alert实例化并且显示的时候将引用保存到内存以供后面使用,而当alert关闭后从内存中删掉对应的引用来确保正常。这样当你需要隐藏屏幕中所有的alert的时候,就可以拿着把内存里面的引用全部删掉。如果是要保留其中某一个特殊的alert的话,那么可以对该alert设置一个特殊的tag,根据tag来进行过滤。#
具体做法如下:
- 在AppDelegate.h文件中申明一个可变数组alertViews,
这个属性用来存放app运行当中所有在屏幕当中显示的alert的引用。当然为确保安全,最好是用懒加载的形式来处理这个属性。
@property (nonatomic,strong) NSMutableArray *alertViews;
- 然后我们需要一个UIAlertView的扩展,用来处理show之前把self添加到alertViews数组中以及当关闭时从alertViews数组中删掉对应的引用,所以你的类别看起来应该像这样:
@implementation UIAlertView (Custom)
+ (void)load
{
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
//替换show方法
NSString *selString = @"show";
NSString *mySelString = @"fm_show";
Method originalMethod = class_getInstanceMethod(self, NSSelectorFromString(selString));
Method myMethod = class_getInstanceMethod(self, NSSelectorFromString(mySelString));
method_exchangeImplementations(originalMethod, myMethod);
//替换关闭方法
selString = @"dismissWithClickedButtonIndex:animated:";
mySelString = @"fm_dismissWithClickedButtonIndex:animated:";
originalMethod = class_getInstanceMethod(self, NSSelectorFromString(selString));
myMethod = class_getInstanceMethod(self, NSSelectorFromString(mySelString));
method_exchangeImplementations(originalMethod, myMethod);
});
}
-(void)fm_show{
[[AppDelegate shareInstance].alertViews addObject:self];
[self fm_show];
}
- (void)fm_dismissWithClickedButtonIndex:(NSInteger)buttonIndex animated:(BOOL)animated{
[[AppDelegate shareInstance].alertViews removeObject:self];
[self fm_dismissWithClickedButtonIndex:buttonIndex animated:animated];
}
@end
- 最后只要把UIAlertView+Custom.h导入就可以正常使用了
这时候如果遇到说不想控制系统弹出来的alert框,可以在UIAlertView的扩展中加一个方法,这个方法来调用保存引用到内存并且show,如下:
-(void)customShow{
[[AppDelegate shareInstance].alertViews addObject:self];
[self show];
}
然后在项目中用到的show方法全都改为新加的这个方法。这样就不会影响到系统自己控制的alert。
我是呱呱,本文为原创作品。欢迎转发给你的朋友或分享到朋友圈,微信公众号转载请通过简信联系作者本人获得授权,谢谢你的支持