黑魔法(method swizzling)解决第三方库引发的问题

需求

最近做一个项目中,有个需求,所有网络请求,都不显示 NetworkActvityIndicator(也就是状态栏里旋转的小圈圈).

解决过程1:

全局搜索 NetworkIndicator 关键字, 把所有涉及 NetworkIndicator 的代码去除,比如 [[UIApplication sharedApplication] setNetworkActivityIndicatorVisible:YES];

测试并发现新问题

所有界面都不再显示NetworkActvityIndicator了,唯独一个播放视频的界面依然显示。

猜想: 第三方库引发的问题

无论是哪些第三方库,正常情况都会通过 setNetworkActivityIndicatorVisible 来 显示状态栏小圈圈。

验证过程1

通过继承 UIApplication 来重写了 setNetworkActivityIndicatorVisible 方法。(如何继承UIApplication,请看这里)并把断点打在这个方法体内。

测试了正常调用 [[UIApplication sharedApplication] setNetworkActivityIndicatorVisible:YES]; 是会触发断点的。但是唯独那个视频界面,没有触发该断点的情况下,正常显示小圈圈。

验证过程2

通过 KVO 监听 UIApplication 的 networkActivityIndicatorVisible 属性,结果还是和 验证过程1 的情况一样。
正常调用 [[UIApplication sharedApplication] setNetworkActivityIndicatorVisible:YES]; 是会触发监听, 唯独那个视频界面,没有触发监听的情况下,正常显示小圈圈。

所以, 视频界面里显示的小圈圈,肯定不是通过常规调用 setNetworkActivityIndicatorVisible 方法显示出来的。

更新猜想: 第三方库引发的问题,并且不是通过常规方法调用

验证过程3

显示小圈圈的情况下,分析了该界面的视图层级,发现在 statusBar 上,有 类型为UIActivityIndicatorView的视图存在(并且怪异的存在了两个)。

那正常情况下,通过 [[UIApplication sharedApplication] setNetworkActivityIndicatorVisible:YES]; 显示小圈圈时,视图层级是如何的呢? 通过分析验证, 也是一样的。

层级都是 UIStatusBarView -> UIStatusBarForegroundView -> UIStatusBarActivityItemView -> UIActivityIndicatorView

想到解决方案:

既然小圈圈都是 UIActivityIndicatorView 类型的视图,而 UIActivityIndicatorView 开始动画常规都是调用 startAnimation 方法。
那何不使用黑魔法(method swizzling)来重写它的 startAnimation 方法,
判断它的superView是否为 “UIStatusBarActivityItemView”类型,如果是,则直接跳出。否则,执行原有的 startAnimation方法。

Talk is cheap. Show me the code.

以下是 .m 文件的代码

@implementation UIActivityIndicatorView (HideNetworkActivityIndicator)

+ (void)load {
    
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        Class class = [self class];
        
        SEL originalSelector = @selector(startAnimating);
        SEL swizzledSelector = @selector(xxx_startAnimating);
        
        Method originalMethod = class_getInstanceMethod(class, originalSelector);
        Method swizzledMethod = class_getInstanceMethod(class, swizzledSelector);
        
        // When swizzling a class method, use the following:
        // Class class = object_getClass((id)self);
        // ...
        // Method originalMethod = class_getClassMethod(class, originalSelector);
        // Method swizzledMethod = class_getClassMethod(class, swizzledSelector);
        
        BOOL didAddMethod =
        class_addMethod(class,
                        originalSelector,
                        method_getImplementation(swizzledMethod),
                        method_getTypeEncoding(swizzledMethod));
        
        if (didAddMethod) {
            class_replaceMethod(class,
                                swizzledSelector,
                                method_getImplementation(originalMethod),
                                method_getTypeEncoding(originalMethod));
        } else {
            method_exchangeImplementations(originalMethod, swizzledMethod);
        }
    });
}

#pragma mark - Method Swizzling

- (void)xxx_startAnimating{
    
    if (self.superview != nil && [NSStringFromClass([self.superview class]) isEqualToString: @"UIStatusBarActivityItemView"]) {
        NSLog(@"黑魔法禁止状态栏的loading显示: %@", self);
    } else {
        [self xxx_startAnimating];
    }

}


@end

成功了!!!

(在xxx_startAnimation方法体内打断点,程序进入视频播放界面,触发断点,看调用栈,果然是第三方库引发的问题。)

参考资料:https://nshipster.cn/method-swizzling/

你可能感兴趣的:(黑魔法(method swizzling)解决第三方库引发的问题)