乐趣区

黑魔法(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-s…

退出移动版