iOS動(dòng)態(tài)更換Icon的全過(guò)程記錄
iOS 動(dòng)態(tài)更換Icon
動(dòng)態(tài)切換 App 的 icon 這個(gè)需求,在上一家公司做一款定制 App 時(shí)遇到過(guò)一次,這次領(lǐng)導(dǎo)說(shuō)可能需要做,就又做了一次。雖然不是什么很難的知識(shí)點(diǎn),這里也就記錄一下自己做的過(guò)程吧。
- info.plist 文件編輯
- 更換 Icon
- 靜默切換
info.plist 文件
為了動(dòng)態(tài)更換 icon,我們需要先配置一下我們項(xiàng)目的 info.plist 文件:
1、加入 Icon files(iOS5),其中會(huì)默認(rèn)有兩個(gè) item:
- Newsstand Icon
- Primary Icon
2、我們需要加入我們需要的鍵——CFBundleAlternateIcons,類(lèi)型為 Dictionary。
3、下面再添加一些字典。這里字典的鍵是你希望更換 Icon 的名稱(chēng),在下方的 CFBundleIconFiles 數(shù)組中,寫(xiě)入需要更換的 Icon 的名稱(chēng)。
Primary Icon: 可以設(shè)置 App 的主 Icon,一般都不理會(huì)。一般主 Icon 在 Assets.xcassets 中設(shè)置。
Newsstand Icon: 這個(gè)設(shè)置一般用于在 Newsstand 中顯示使用。我們也不需要理會(huì)。
這里我們就將 info.plist 編輯完成了,下面我們將對(duì)應(yīng)的圖片加入到項(xiàng)目中,這里的圖片需要直接加到項(xiàng)目中,不能放在 Assets.xcassets 中。
更換 Icon
在 iOS 10.3,蘋(píng)果開(kāi)放了這個(gè) API,可以讓我們動(dòng)態(tài)更換我們的 App Icon。
// If false, alternate icons are not supported for the current process. @available(iOS 10.3, *) open var supportsAlternateIcons: Bool { get } // Pass `nil` to use the primary application icon. The completion handler will be invoked asynchronously on an arbitrary background queue; be sure to dispatch back to the main queue before doing any further UI work. @available(iOS 10.3, *) open func setAlternateIconName(_ alternateIconName: String?, completionHandler: ((Error?) -> Void)? = nil) // If `nil`, the primary application icon is being used. @available(iOS 10.3, *) open var alternateIconName: String? { get }
切換到我們需要的 Icon
@IBAction func changeOneClick(_ sender: Any) { if UIApplication.shared.supportsAlternateIcons { UIApplication.shared.setAlternateIconName("lambot") { (error) in if error != nil { print("更換icon錯(cuò)誤") } } } }
這里的 iconName 直接傳入項(xiàng)目中的 icon 名稱(chēng)。這里需要注意的是,項(xiàng)目中的名字、info.plist 中存入的名稱(chēng)以及這里傳入的名稱(chēng)需要一致。
重置為原始的 Icon
@IBAction func resetClick(_ sender: Any) { if UIApplication.shared.supportsAlternateIcons { UIApplication.shared.setAlternateIconName(nil) { (error) in if error != nil { print("更換icon錯(cuò)誤") } } } }
如果需要恢復(fù)為原始的 icon,只需要在傳入 iconName 的地方傳入 nil 即可。
現(xiàn)在,已經(jīng)完成了切換 Icon 的功能了。但是每次切換時(shí),都會(huì)有一個(gè)彈框,下面我們就想辦法去掉這個(gè)彈框。
靜默切換
我們可以利用 Runtime 的方法來(lái)替換掉彈出提示框的方法。
以前 Method Swizzling 的時(shí)候需要在 load 或者 initialize 方法,但是在 Swift 中不能使用了。那就只能自己定義一個(gè)了。
extension UIViewController { public class func initializeMethod() { if self != UIViewController.self { return } // Method Swizzling DispatchQueue.once(token: "ChangeIcon") { let orignal = class_getInstanceMethod(self, #selector(UIViewController.present(_:animated:completion:))) let swizzling = class_getInstanceMethod(self, #selector(UIViewController.jt_present(_:animated:completion:))) if let old = orignal, let new = swizzling { method_exchangeImplementations(old, new) } } } @objc private func jt_present(_ viewControllerToPresent: UIViewController, animated flag: Bool, completion: (() -> Void)? = nil) { // 在這里判斷是否是更換icon時(shí)的彈出框 if viewControllerToPresent is UIAlertController { let alertTitle = (viewControllerToPresent as! UIAlertController).title let alertMessage = (viewControllerToPresent as! UIAlertController).message // 更換icon時(shí)的彈出框,這兩個(gè)string都為nil。 if alertTitle == nil && alertMessage == nil { return } } // 因?yàn)榉椒ㄒ呀?jīng)交換,這個(gè)地方的調(diào)用就相當(dāng)于調(diào)用原先系統(tǒng)的 present self.jt_present(viewControllerToPresent, animated: flag, completion: completion) } }
定義完 UIViewController 的擴(kuò)展方法后,記得在 AppDelegate 中調(diào)用一下。
func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool { UIViewController.initializeMethod() return true }
因?yàn)椋琒wift 中 GCD 之前的 once 函數(shù)沒(méi)有了,這里自己簡(jiǎn)單定義了一個(gè)。
extension DispatchQueue { private static var _onceTracker = [String]() public class func once(token: String, block: () -> ()) { objc_sync_enter(self) defer { objc_sync_exit(self) } if _onceTracker.contains(token) { return } _onceTracker.append(token) block() } }
defer block 里的代碼會(huì)在函數(shù) return 之前執(zhí)行,無(wú)論函數(shù)是從哪個(gè)分支 return 的,還是有 throw,還是自然而然走到最后一行。
現(xiàn)在,我們?cè)俑鼡Q Icon 的時(shí)候,就不會(huì)出現(xiàn)彈出框了。
總結(jié)
簡(jiǎn)單的知識(shí)點(diǎn),時(shí)間長(zhǎng)了不用也有可能忘記。希望自己能堅(jiān)持學(xué)習(xí),堅(jiān)持記錄,不斷成長(zhǎng)。
好了,以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)腳本之家的支持。
參考鏈接:
相關(guān)文章
IOS開(kāi)發(fā)之CocoaPods安裝和使用教程
CocoaPods應(yīng)該是iOS最常用最有名的類(lèi)庫(kù)管理工具了,通過(guò)cocoaPods,只需要一行命令就可以完全解決,當(dāng)然前提是你必須正確設(shè)置它。重要的是,絕大部分有名的開(kāi)源類(lèi)庫(kù),都支持CocoaPods。所以,作為iOS程序員的我們,掌握CocoaPods的使用是必不可少的基本技能了。2014-09-09iOS CoreTelephony 實(shí)現(xiàn)監(jiān)聽(tīng)通話狀態(tài)
這篇文章主要介紹了iOS CoreTelephony 實(shí)現(xiàn)監(jiān)聽(tīng)通話狀態(tài) 的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2016-07-07iOS開(kāi)發(fā)中實(shí)現(xiàn)顯示gif圖片的方法
這篇文章主要介紹了iOS開(kāi)發(fā)中實(shí)現(xiàn)顯示gif圖片的方法,代碼基于傳統(tǒng)的Objective-C,需要的朋友可以參考下2015-09-09iOS開(kāi)發(fā)中實(shí)現(xiàn)新聞圖片的無(wú)限循環(huán)展示的方法
這篇文章主要介紹了iOS開(kāi)發(fā)中實(shí)現(xiàn)新聞圖片的無(wú)限循環(huán)展示的方法,代碼基于傳統(tǒng)的Objective-C,需要的朋友可以參考下2015-12-12Flutter之PageView頁(yè)面緩存與KeepAlive
這篇文章主要為大家介紹了Flutter之PageView頁(yè)面緩存與KeepAlive示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-10-10iOS開(kāi)發(fā)之?dāng)?shù)字每隔3位用逗號(hào)分隔
以前在做電商app時(shí)經(jīng)常會(huì)針對(duì)稍大的金額展示出來(lái),需要每隔千位添加逗號(hào)便于用戶(hù)識(shí)別,下面通過(guò)本文給大家分享ios中數(shù)字每隔3位用逗號(hào)分隔的實(shí)例代碼,需要的朋友參考下吧2017-09-09iOS實(shí)現(xiàn)UITableView左滑刪除復(fù)制即用功能
這篇文章主要介紹了iOS實(shí)現(xiàn)UITableView左滑刪除復(fù)制即用功能,在項(xiàng)目開(kāi)發(fā)中經(jīng)常會(huì)用到這樣的需求,下面小編把實(shí)現(xiàn)代碼分享給大家,需要的朋友可以參考下2017-09-09