iOS动态更换App图标(二):无弹框更换App图标

  • 戴奕
  • 6 Minutes
  • May 1, 2017

上篇文章我们详细查看了更换App图标的使用方法,并做了个小Demo。尽管当前我们可以实现动态更换App图标了,但是每次更换图标时,苹果官方给加的“友好提示”对用户以及开发者都不是那么“友好“。官方并没有给出可以不弹框的方法,毕竟App图标对于苹果来说是一个很重要的审核部分,如果任由开发者在上架后不提示用户而随意修改图标,会造成不好的用户体验,所以苹果会在使用此API时弹框告知用户该App图标已修改(个人猜想)。

不过今天我们想谈谈如何突破这个弹框限制(毕竟开发者也不是傻,不会胡乱更换图标的是不是🤣)。

本系列文章

  1. iOS动态更换App图标(一):基础使用
  2. iOS动态更换App图标(二):无弹框更换App图标
  3. iOS动态更换App图标(三):动态下载App图标进行更换

Demo演示

DynamicAppIconDemo2

Demo地址:https://github.com/maybeisyi/ChangeAppIconDemo

本篇文章对应工程为:DynamicAppIcon(二)

Demo中可以看到,现在我们可以做到不弹框直接修改App图标。实现该功能后,某些有意思的小功能就能有良好的用户体验了:白天/夜间模式切换,在切换App主色调同时切换App图标。

下面将详细讲解如何”突破”苹果的限制。

什么是弹框

让我们查看弹框的本质

Alert

查看原Demo中的弹框,此弹框与UIAlertController长的倒是挺像的。让我们来剖析下这个弹框:

更换图标的弹框

可以看到弹框就是私有类_UIAlertControllerView,让我们再对比下系统的UIAlertController:

UIAlertController弹框

所以更换App时的弹框就是UIAlertController,只不过上面的控件不太一样罢了。(其实我们也能做到在UIAlertController上添加任意控件)

拦截弹框

既然知道了弹框是UIAlertController,那么我们自然而然想到,该弹框是由ViewController通过presentViewController:animated:completion:方法弹出。那么我们就可以通过Method swizzling hook该弹框,不让其进行弹出即可:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
#import "UIViewController+Present.h"
#import <objc/runtime.h>
@implementation UIViewController (Present)
+ (void)load {
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
Method presentM = class_getInstanceMethod(self.class, @selector(presentViewController:animated:completion:));
Method presentSwizzlingM = class_getInstanceMethod(self.class, @selector(dy_presentViewController:animated:completion:));
// 交换方法实现
method_exchangeImplementations(presentM, presentSwizzlingM);
});
}
- (void)dy_presentViewController:(UIViewController *)viewControllerToPresent animated:(BOOL)flag completion:(void (^)(void))completion {
if ([viewControllerToPresent isKindOfClass:[UIAlertController class]]) {
NSLog(@"title : %@",((UIAlertController *)viewControllerToPresent).title);
NSLog(@"message : %@",((UIAlertController *)viewControllerToPresent).message);
UIAlertController *alertController = (UIAlertController *)viewControllerToPresent;
if (alertController.title == nil && alertController.message == nil) {
return;
} else {
[self dy_presentViewController:viewControllerToPresent animated:flag completion:completion];
return;
}
}
[self dy_presentViewController:viewControllerToPresent animated:flag completion:completion];
}
@end

这段代码交换了UIViewControllerpresentViewController:animated:completion:方法。通过打印UIAlertController的特征,我们可以发现,更换App图标时的弹框是没有title与message的,但是我们一般使用的UIAlertController都是带title、message的,毕竟不会弹个空白的框给用户玩。

所以该方法中通过判断title与message来捕捉更换App图标时的弹框,并直接return即可。

总结

其实关于界面上的东西,利用动态特性没有什么是不能做的,苹果既然公开了动态API,我们就可以通过动态方法去了解甚至改造我们想要的东西,如系统控件如何实现等。苹果的”规范“在应用层面其实是无法阻挡开发者步伐的,当然动态特性也不能够滥用(如私有方法),毕竟审核人员才是爸爸。

尽管目前实现了在用户无感的情况下替换App图标,但是可替换的图标还是必须预先放入工程中,并且要在Info.plist内指定。这很大程度上限制了更换图标的动态性:比如我们某天想要推出一款新主题以及对应的App图标,但是新的App图标并没有预先放入工程的main bundle中,也没有在Info.plist中进行指定,所以我们在不上架新版本的情况下,无法推出该新App图标,因此有这了第三篇文章。

第三篇文章:《iOS动态更换App图标(三):动态下载App图标进行更换》短期内应该无法实现,具体原因会在文章中说明。