[iOS]動態修改app 圖標(icon)


動態修改app的圖標,就是在不從新安裝app的狀況下,能夠修改當前的icon圖標;在某些狀況下,是有這個需求的;例如,能夠更換主題的app中,通常都會有一套完整的主題包含相應的icon;還有就是一些節日主題的icon或者促銷的icon,例如淘寶、京東等的節日icon。ios

在iOS 10.3以後,蘋果官方提供了相關的API來實現這個功能,主要是下面這幾個方法:數組

@interface UIApplication (UIAlternateApplicationIcons)
// 若是爲NO,表示當前進程不支持替換圖標
@property (readonly, nonatomic) BOOL supportsAlternateIcons NS_EXTENSION_UNAVAILABLE("Extensions may not have alternate icons") API_AVAILABLE(ios(10.3), tvos(10.2));

// 傳入nil表明使用主圖標. 完成後的操做將會在任意的後臺隊列中異步執行; 若是須要更改UI,請確保在主隊列中執行.
- (void)setAlternateIconName:(nullable NSString *)alternateIconName completionHandler:(nullable void (^)(NSError *_Nullable error))completionHandler NS_EXTENSION_UNAVAILABLE("Extensions may not have alternate icons") API_AVAILABLE(ios(10.3), tvos(10.2));

// 若是alternateIconName爲nil,則表明當前使用的是主圖標.
@property (nullable, readonly, nonatomic) NSString *alternateIconName NS_EXTENSION_UNAVAILABLE("Extensions may not have alternate icons") API_AVAILABLE(ios(10.3), tvos(10.2));
@end複製代碼

方法很簡單,可是使用以前須要進行一些配置:微信

1. 配置icon

添加圖片icon

動態修改的icon不能放在 Assets.xcassets 裏,可是正常的主icon仍是能夠在這裏設置的,也能夠按下面的方法來設置;
首先,把須要修改的icon放在一個文件夾內:app


icons

其文件夾內是這樣的異步


這裏每種icon我只放了一個,若是有多個尺寸的icon,也能夠直接全放進去:學習

多尺寸icon

而後,文件夾會變成這樣:測試

這裏的icon名稱只須要和下面配置一致便可ui

配置info.plist

在info.plist中右鍵 -> Add Row :
輸入Icon... 會有提示,選擇Icon files(iOS 5)atom


這時候,內容是這樣的:spa


這裏的Icon files(iOS 5)是個字典,其中可包含的Key值有CFBundlePrimaryIcon -> Primary Icon
CFBundleAlternateIcons
UINewsstandIcon -> Newsstand Icon

這裏的Primary Icon是設置app的主icon,能夠在這裏的Icon files數組內添加,有多個的話,依次添加,也能夠這裏不用填寫,直接在Assets.xcassets 裏配置;

下面的Newsstand Icon,暫時用不到,不用管,也能夠刪除。
在 Icon files(iOS 5)內添加一個Key: CFBundleAlternateIcons ,類型爲字典,在這個字典裏配置咱們全部須要動態修改的icon:鍵爲icon的名稱,值爲一個字典(這個字典裏包含兩個鍵:CFBundleIconFiles,其值類型爲Array,內容爲icon的名稱;UIPrerenderedIcon,其值類型爲bool,內容爲NO,也能夠不加此key),例如:

把第一步中添加的圖片所有添加進來就是這樣的:


到此,info.plist的配置即完成了;
或者將info.plist文件以 Source code 方式打開,添加如下代碼:

<key>CFBundleIcons</key>
    <dict>
        <key>CFBundleAlternateIcons</key>
        <dict>
            <key>rain</key>
            <dict>
                <key>CFBundleIconFiles</key>
                <array>
                    <string>rain</string>
                </array>
                <key>UIPrerenderedIcon</key>
                <false/>
            </dict>
            <key>snow</key>
            <dict>
                <key>CFBundleIconFiles</key>
                <array>
                    <string>snow</string>
                </array>
                <key>UIPrerenderedIcon</key>
                <false/>
            </dict>
            <key>sunshine</key>
            <dict>
                <key>CFBundleIconFiles</key>
                <array>
                    <string>sunshine</string>
                </array>
                <key>UIPrerenderedIcon</key>
                <false/>
            </dict>
            <key>cloudy</key>
            <dict>
                <key>CFBundleIconFiles</key>
                <array>
                    <string>cloudy</string>
                </array>
                <key>UIPrerenderedIcon</key>
                <false/>
            </dict>
        </dict>
        <key>CFBundlePrimaryIcon</key>
        <dict>
            <key>CFBundleIconFiles</key>
            <array>
                <string></string>
            </array>
            <key>UIPrerenderedIcon</key>
            <false/>
        </dict>
        <key>UINewsstandIcon</key>
        <dict>
            <key>CFBundleIconFiles</key>
            <array>
                <string></string>
            </array>
            <key>UINewsstandBindingType</key>
            <string>UINewsstandBindingTypeMagazine</string>
            <key>UINewsstandBindingEdge</key>
            <string>UINewsstandBindingEdgeLeft</string>
        </dict>
    </dict>複製代碼

若是是添加了多個尺寸icon,也要在這裏分別配置,以上面添加的sunshine圖標爲例:

使用的時候仍是使用sunshine進行賦值便可!

代碼

配置完成後,代碼部分就比較簡單了:

- (void)changeAppIconWithName:(NSString *)iconName {
    if (![[UIApplication sharedApplication] supportsAlternateIcons]) {
        return;
    }
    
    if ([iconName isEqualToString:@""]) {
        iconName = nil;
    }
    [[UIApplication sharedApplication] setAlternateIconName:iconName completionHandler:^(NSError * _Nullable error) {
        if (error) {
            NSLog(@"更換app圖標發生錯誤了 : %@",error);
        }
    }];
}複製代碼

在須要修改icon的地方調用這個方法,並把相應的icon名稱傳進去便可:

- (IBAction)snow:(id)sender {
    [self changeAppIconWithName:@"snow"];
}
- (IBAction)rain:(id)sender {
    [self changeAppIconWithName:@"rain"];
}
- (IBAction)cloudy:(id)sender {
    [self changeAppIconWithName:@"rain"];
}
- (IBAction)sunshine:(id)sender {
    [self changeAppIconWithName:@"sunshine"];
}複製代碼

示意圖:


示意圖

設置iPad動態圖標

iPad的動態圖標設置和上面步驟基本同樣,有的文章說是將 CFBundleIcons 改成 CFBundleIcons~ipad,即:

iPad

可是,在測試中發現,使用上面的key值也是能夠實現動態改變的,即不作任何修改,iPhone和iPad使用相同的配置,即:CFBundleIcons。

去掉更換icon時的彈框

從上面的示意圖能夠發現,在設置icon的時候,會有個系統彈框,這樣有時候會不太友好,咱們可使用Runtime,對UIViewController進行擴展來隱藏這個彈框:

// UIViewController+LQNoPresent.h
#import <UIKit/UIKit.h>

@interface UIViewController (LQNoPresent)

@end


// UIViewController+LQNoPresent.m

#import "UIViewController+LQNoPresent.h"
#import <objc/runtime.h>

@implementation UIViewController (LQNoPresent)

+ (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(lq_presentViewController:animated:completion:));
        
        method_exchangeImplementations(presentM, presentSwizzlingM);
    });
}

- (void)lq_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;
        }
    }
    
    [self lq_presentViewController:viewControllerToPresent animated:flag completion:completion];
}


@end複製代碼

這樣在切換圖標的時候就沒有系統的彈框了:

去掉系統彈框


做者:流火緋瞳
連接:https://www.jianshu.com/p/69313970d0e7

 此文章來源於第三方轉載

小編這呢,給你們推薦一個優秀的iOS交流平臺,平臺裏的夥伴們都是很是優秀的iOS開發人員,咱們專一於技術的分享與技巧的交流,你們能夠在平臺上討論技術,交流學習。歡迎你們的加入(想要進入的可加小編微信)。 13142121176

相關文章
相關標籤/搜索