iOS开发:Swift主题色顶级解决方案

一、常规主题色使用点

应用在发布前都会对其主题色进行设置,以统一应用的风格(可能有多套主题)。在主题色设置上有几个方面,如下:

1.TabBar部分,设置图片高亮、文本高度颜色2.NavigationBar部分,设置导航栏颜色及字体颜色3.应用标签等,设置字体的颜色4.应用图片主题色

主题色的设置点,大体从上面四个方面着手,图片的主题色我们可通过图片更换的方式进行处理。而通过代码来处理的 1-3 条,有着不同的处理方法。大家常规处理方法如下:

步骤一:变化分离

1.利用Swift扩展语法扩展UIColor,将应用主题色在扩展中统一处理(适合单一主题色)2.将主题色的配置写入文件中,由相应逻辑进行解析。此方法将主题色逻辑封装成主题色管理类(适合多套主题)

步骤二:离散使用上步封装的类

1.在任何使用主题色的地方,使用扩展中的UIColor方法来设置,一般包括背景色,文字颜色等

这里给出UIColor的扩展

 
extension UIColor {    //主题色    class func applicationMainColor() -> UIColor {        return UIColor(red: 238/255, green: 64/255, blue: 86/255, alpha:1)    }    //第二主题色    class func applicationSecondColor() -> UIColor {        return UIColor.lightGrayColor()    }    //警告颜色    class func applicationWarningColor() -> UIColor {        return UIColor(red: 0.1, green: 1, blue: 0, alpha: 1)    }    //链接颜色    class func applicationLinkColor() -> UIColor {        return UIColor(red: 59/255, green: 89/255, blue: 152/255, alpha:1)    }}

二、TabBar主题色设置

很多应用中,默认情况下都使用了TabBar控件,但是TabBar主题色等设置根据使用情况的不同,设置起来也不一样。代码创建比较灵活,更改主题色比较容易。而使用了Xib/Storyboard也是有办法做统一处理的,如下,迭代更改TabBar默认字体颜色

 
 func configTabBar() {        let items = self.tabBar.items        for item in items as [UITabBarItem] {            let dic = NSDictionary(object: UIColor.applicationMainColor(),             forKey:   NSForegroundColorAttributeName)            item.setTitleTextAttributes(dic,             forState: UIControlState.Selected)        }    }

设置TabBar图片及文字默认选中颜色

 
    self.tabBar.selectedImageTintColor = UIColor.applicationMainColor()

Tips注意事项

Changing this property’s value provides visual feedback in the user interface, including the running of any associated animations. The selected item displays the tab bar item’s selectedImage image, using the tab bar’s selectedImageTintColor value. To prevent system coloring of an item, provide images using the UIImageRenderingModeAlwaysOriginal rendering mode.

在一些情况,正常状态为白色图片时,真机测试时,白色图片会出现偏色(显示结果为灰色),这是因为系统默认着色导致的,在创建UITabBarItem时,可通过使用UIImageRenderingModeAlwaysOriginal避免。示例代码如下:

 
let imageNormal = UIImage(contentsOfFile: "imageNormal")?.imageWithRenderingMode(UIImageRenderingMode.AlwaysOriginal)let imageSelected = UIImage(contentsOfFile: "imageSelected")let tabBarItem = UITabBarItem(title: "title",         image: imageNormal,         selectedImage: imageSelected)

三、一劳永逸,利用Hook原理通设NavigationBar颜色

IOS应用中,NavigationBar十分常用,它的使用主要包括以下两个场景

1.代码直接构建2.Xib/Storyboard构建

如果是纯代码构建的时候,比较简单,直接使用UIColor的扩展来设置颜色。实际项目中,有些界面是通过Xib/Storyboard来创建的,有些是代码写的,但这也难不到大家,使用继承。创建一个继承自UINavigationController的子类,通过这个子类来统一设置主题色。然后告诉项目中的所有人,强制使用UINavigationController子类,包括Xib/Storyboard等。问题是旧项目怎么办,这种强制要求可以工作,有没有一个更好的办法,让所有人正常使用UINavigationController,而在神不知鬼不觉的情况下,通设所有NavigationBar呢? 先上代码,再解释

1.创建一个UIViewController的扩展

 
extension UIViewController {    func viewDidLoadForChangeTitleColor() {        self.viewDidLoadForChangeTitleColor()        if self.isKindOfClass(UINavigationController.classForCoder()) {           self.changeNavigationBarTextColor(self as UINavigationController)        }    }    func changeNavigationBarTextColor(navController: UINavigationController) {        let nav = navController as UINavigationController        let dic = NSDictionary(object: UIColor.applicationMainColor(),         forKey:NSForegroundColorAttributeName)        nav.navigationBar.titleTextAttributes = dic        nav.navigationBar.barTintColor = UIColor.applicationSecondColor()        nav.navigationBar.tintColor = UIColor.applicationMainColor()    }}

2.编写用于Hook的工具类

 
func swizzlingMethod(clzz: AnyClass, #oldSelector: Selector, #newSelector: Selector) {    let oldMethod = class_getInstanceMethod(clzz, oldSelector)    let newMethod = class_getInstanceMethod(clzz, newSelector)    method_exchangeImplementations(oldMethod, newMethod)}

3.在AppDelegate中调用

 
  func application(application: UIApplication, didFinishLaunchingWithOptions launchOptions: NSDictionary?) -> Bool {        swizzlingMethod(UIViewController.self,        oldSelector: "viewDidLoad",        newSelector: "viewDidLoadForChangeTitleColor")  //do others        return true    }

4.原理说明

在程序入口处,通过运行时机制,动态的替换UIViewController的周期方法 viewDidLoad 为我们指定的方法 viewDidLoadForChangeTitleColor 。在 viewDidLoadChangeTitleColor中,需要做两件事:

  • 调用原来的 viewDidLoad 方法
  • 执行修改主题色相关代码

1.如何调用原来的viewDidLoad方法

在AppDelegate中,通过调用方法 swizzlingMethod 我们将 viewDidLoad 与viewDidLoadForChangeTitleColor 方法体进行了替换,原理如下图:

从上面的图可以看出,当在 viewDidLoadForChangeTitleColor 中执行:

 
self.viewDidLoadForChangeTitleColor()

是不会造成循环调用,反而是调用了我们期望执行的 viewDidLoad 方法体。

三、Xib/Storyboard的处理

一些在Xib/Storyboard中设置的主题色,比如文本颜色,按钮的高亮颜色等,该如何处理呢,以UILabel为例,建立扩展

 
extension UILabel {    var colorString: String {        set(newValue) {            switch newValue {            case "main":                self.textColor = UIColor.applicationMainColor()            case "second":                self.textColor = UIColor.applicationSecondColor()            case "warning":                self.textColor = UIColor.applicationWarningColor()            default:                self.textColor = UIColor.applicationSecondColor()            }        }        get {            return self.colorString        }    }}

在Xib/Storyboard的查检器中进行编辑,如下图: 

4.总结

1.只有一套主题时,上面的方法可以直接复制使用,在更换主题时,只需要更换相应图片及修改UIColor的扩展类

2.在有多套主题,用户可以自由切换主题时,可以按文章中的Hook机制,对 viewWillAppear进行劫持,也可以轻松实现主题的改变

时间: 2024-12-21 22:17:50

iOS开发:Swift主题色顶级解决方案的相关文章

Swift主题色顶级解决方案

一.常规主题色使用点 应用在发布前都会对其主题色进行设置,以统一应用的风格(可能有多套主题).在主题色设置上有几个方面,如下: 1.TabBar部分,设置图片高亮.文本高度颜色 2.NavigationBar部分,设置导航栏颜色及字体颜色 3.应用标签等,设置字体的颜色 4.应用图片主题色 主题色的设置点,大体从上面四个方面着手,图片的主题色我们可通过图片更换的方式进行处理.而通过代码来处理的`1-3`条,有着不同的处理方法.大家常规处理方法如下: 步骤一:变化分离 1.利用Swift扩展语法扩

李洪强iOS开发Swift篇—03_字符串和数据类型

李洪强iOS开发Swift篇—03_字符串和数据类型 一.字符串 字符串是String类型的数据,用双引号""包住文字内容  let website = "http://www.wendingding.com" 1.字符串的常见操作 (1)用加号 + 做字符串拼接 let scheme = "http://" let path = “www.wendingding.com” let website = scheme + path // websi

iOS开发-Swift篇-(1)

iOS开发Swift篇—简单介绍 一.简介 Swift是苹果于2014年WWDC(苹果开发者大会)发布的全新编程语言 Swift在天朝译为“雨燕”,是它的LOGO 是一只燕子,跟Objective-C一样,可以用于开发iOS.Mac应用程序 苹果从2010年7月开始设计Swift语言,耗时4年打造 Swift的语法特点 从它的语法中能看到Objective-C.JavaScript.Python等语言的影子 语法简单.代码简洁.使用方便 可与Objective-C混合使用(相互调用) 为什么要设

iOS开发Swift篇(01) 变量&常量&元组

iOS开发Swift篇(01) 变量&常量&元组 说明: 1)终于要写一写swift了.其实早在14年就已经写了swift的部分博客,无奈时过境迁,此时早已不同往昔了.另外,对于14年部分iOS开发Swift篇专题的博文也不再做任何的校正和更新,特此说明. 2)该博文对应代码可以在https://github.com/HanGangAndHanMeimei/Code获得. 一.变量和常量 01 变量和常量的定义 在swift中变量使用var来修饰,常量使用let来修饰,变量可以修改而常量不

iOS开发Swift篇—(八)函数(2)

iOS开发Swift篇—(八)函数(2) 一.函数类型 函数类型也是数据类型的一种,它由形参类型和返回值类型组成,格式是 (形参类型列表) -> 返回值类型 1 func sum(num1: Int, num2: Int) -> Int { 2 return num1 + num2 3 } sum函数的函数类型是(Int, Int) -> Int 1 func printLine() 2 { 3 println("-----------") 4 } printLine

iOS开发swift语法初级篇—————(swift技术交流群:361513739)

iOS开发之swift语法初级篇:点击打开链接 swift技术交流QQ群361513739 iOS开发swift语法初级篇-----(swift技术交流群:361513739)

iOS开发Swift篇—简单介绍

iOS开发Swift篇—简单介绍 一.简介 Swift是苹果于2014年WWDC(苹果开发者大会)发布的全新编程语言 Swift在天朝译为“雨燕”,是它的LOGO 是一只燕子,跟Objective-C一样,可以用于开发iOS.Mac应用程序 苹果从2010年7月开始设计Swift语言,耗时4年打造 Swift的语法特点 从它的语法中能看到Objective-C.JavaScript.Python等语言的影子 语法简单.代码简洁.使用方便 可与Objective-C混合使用(相互调用) 为什么要设

李洪强iOS开发Swift篇—09_属性

李洪强iOS开发Swift篇—09_属性 一.类的定义 Swift与Objective-C定义类的区别 Objective-C:一般需要2个文件,1个.h声明文件和1个.m实现文件 Swift:只需要1个.swift文件 Swift中类的定义格式 1 class 类名 { 2 // ... 属性和方法 3 } 二.属性 1.什么是属性 Swift中的属性(Properties),就类似于其他面向对象语言中的成员变量 2.属性的分类 按照官方文档的说明,属性可以分为以下几种 (1)存储属性(Sto

李洪强iOS开发Swift篇—02_变量和常量

李洪强iOS开发Swift篇—02_变量和常量 一.语言的性能 (1)根据WWDC的展示 在进行复杂对象排序时Objective-C的性能是Python的2.8倍,Swift的性能是Python的3.9倍 在实现 RC4加密算法时Objective-C的的性能是Python的127倍,Swift的性能是Python的220倍 有持怀疑态度的国外程序员,也对Objective-C和Swift的性能进行了测试 http://www.splasmata.com/?p=2798 (2)说明 目前的性能不