事情的起因是因为使用LaunchScreen.storyboard
适配启动图后经常遇到启动图中的图片变成一团黑色的情况,刚开始是测试机,然后是用户。再然后我在使用今日头条
、微信
这些APP时都遇到了这种情况。
我自己的APP | 微信APP |
大家仔细看微信APP的启动过程,启动图上的那个地球图片和人都不见了。我看了一下别人的手机上同版本的微信启动图,发现显示是正常的,确认这不是微信故意改成这样了。
LLDynamicLaunchScreen
的初衷是为了解决启动图显示异常而开发的,不过后来我又在这个基础上开发了动态修改启动图的功能。想象一下,用户打开相册,选择一张自己喜欢的图片。然后每次打开APP的第一眼看到的都是自己喜欢的图片。这对于用户来说是不是一种很爽的体验呢!在之前,想实现这种功能是不可能的。但是现在你只需要集成一下LLDynamicLaunchScreen
,然后1行代码即可实现你的需求。
在写这篇博客前,我去网上找了一下关于iOS动态修改启动图相关的框架,以下是我找到的框架:
-
DynamicLaunchImage
是百度App技术开发的,但是现在好像已经没有维护了,另外我在测试时发现它有一个很严重的问题,当APP更新版本后就会丢失上次设置的启动图而恢复为默认启动图;它的功能很简单,只实现了一个替换启动图的功能,如果您想使用它来修复启动图显示异常的话,需要您手动生成正确的启动图然后替换指定启动图;另外它没有适配iOS13及以上系统,无法单独修改深色启动图或浅色启动图。 -
LLDynamicLaunchScreen
是我开发的一款可以在自动修复与动态修改启动图的框架,相比较DynamicLaunchImage
来说,它实现了自动修复启动图显示异常,还可以精确的修改某1个启动图(例如单独修改浅色竖屏启动图、深色竖屏启动图、浅色横屏启动图、深色横屏启动图)而不影响其他启动图,它还支持iPad。最最最重要的是我一直在维护它,如果没有意外的话我应该会适配未来的iOS版本,如果您在使用过程中有遇到任何问题也可以向我提问。
接下来我会按照问答的方式简述我在开发过程中遇到的一些问题,这里只描述了一些比较麻烦的问题,具体实现逻辑可以下载源码查看。
如何根据LaunchScreen.storyboard
获取启动图?
关于如何根据storyboard
文件生成UIImage对象这里就不赘述了。修复启动图的逻辑其实就是基于LaunchScreen.storyboard
文件生成一个启动图UIImage对象,然后将这个UIImage源数据写入本地启动图路径下即可。但是我在生成启动图的时候遇到了一些麻烦,具体表现为当前系统是浅色模式时,我想要生成一张深色模式下的启动图时获取到的一直都是浅色状态下的图片内容(只要获取与系统主题模式相反的启动图都会失败)。最初一直以为是代码逻辑的问题,反复检查确认没有问题;后来测试发现在APP启动后通过用户点击的方式触发生成启动图的逻辑可以正常获取到浅色和深色这2种样式的启动图。但是这种方案我并不喜欢,后面测试过在UIWindow显示后执行
、使用延迟的方式执行
,但是都失败了。最后发现把逻辑放到第一个UIViewController的viewDidAppear:
后执行可以达到目的,刚开始是使用runtime
的method_exchangeImplementations
交换系统方法viewDidAppear:
来达到的目的,后来觉得这样不妥,如果使用者在工程里也交换了viewDidAppear:
,那么可能会出现一些难已排查的BUG。最后是通过isa-swizzling
机制实现了同样的目的,具体操作就是运行时创建了一个UIViewController的子类,并且交换了它们的isa指针,然后实现了监听viewDidAppear:
执行的目的,这个思路来自系统KVO的具体实现;具体实现细节可以下载源码查看。
iOS13以下的真机没有系统启动图的读取权限,如何替换启动图?
经过测试发现在iOS13以下的真机上确实无法通过正常的方式获取和写入启动图。最后通过查阅资料加上实践测试,发现通过moveItemAtPath: toPath: error:
可以变相的达到目的,具体的实现细节就是将系统启动图文件夹先暂时移动到一个临时可操作的文件夹路径下,然后进行写入和读取操作,最后在通过同样的方式将临时文件夹移动回原来的文件夹,再把临时文件夹删除。
关于具体的实现细节和流程大家可以下载源码进行查看。下面的视频内容使用了LLDynamicLaunchScreen
来实现动态修改启动图。
LLaunchScreen
是LLDynamicLaunchScreen
的Swift版本,它实现了和LLDynamicLaunchScreen
一样的功能,不过由于Swift语言的限制,您在使用它时需要进行2次初始化,具体步骤可以查看LLaunchScreen
的README.md文件,或者下载源码查看。