UIAlertView如何在代码中控制隐藏

新年上班第二天,呱呱给大家拜个晚年,祝大家工作顺意,生活美满。

相信大家都知道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。


我是呱呱,本文为原创作品。欢迎转发给你的朋友或分享到朋友圈,微信公众号转载请通过简信联系作者本人获得授权,谢谢你的支持

你可能感兴趣的:(UIAlertView如何在代码中控制隐藏)