iOS开发笔记--使用blend改变图片颜色

2024-03-24 08:40

本文主要是介绍iOS开发笔记--使用blend改变图片颜色,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

最近对Core AnimationCore Graphics的内容东西比较感兴趣,自己之前也在这块相对薄弱,趁此机会也想补习一下这块的内容,所以之后几篇可能都会是对CA和CG学习的记录的文章。

在应用里一个很常见的需求是主题变换:同样的图标,同样的素材,但是需要按照用户喜爱变为不同的颜色。在iOS5和6的SDK里部分标准控件引入了tintColor,来满足个性化界面的需求,但是Apple在这方面还远远做的不够。一是现在用默认控件根本难以做出界面优秀的应用,二是tintColor所覆盖的并不够全面,在很多情况下开发者都无法使用其来完成个性化定义。解决办法是什么?最简单当然是拜托设计师大大出图,想要蓝色主题?那好,开PS盖个蓝色图层出一套蓝色的UI;又想加粉色UI,那好,再出一套粉色的图然后导入Xcode。代码上的话根据颜色需求使用image-blue或者image-pink这样的名字来加载图片。

如果有一丁点重构的意识,就会知道这不是一个很好的解决方案。工程中存在大量的冗余和重复(就算你要狡辩这些图片颜色不同不算重复,你也会在内心里知道这种狡辩是多么无力),这是非常致命的。想象一下如果你有10套主题界面,先不论应用的体积会膨胀到多少,光是想做一点修改就会痛苦万分,比如希望改一下某个按钮的形状,很好,设计师大大请重复地修改10遍,并出10套UI,然后一系列的重命名,文件移动和导入…一场灾难。

当然有其他办法,因为说白了就是tint不同的颜色到图片上而已,如果我们能实现改变UIImage的颜色,那我们就只需要一套UI,然后用代码来改变UI的颜色就可以了,生活有木有一下光明起来呀。嗯,让我们先从一张图片开始吧~下面是一张带有alpha通道的图片,原始颜色是纯的灰色(当然什么颜色都可以,只不过我这个人现在暂时比较喜欢灰色而已)。



我们将用blending给这张图片加上另一个纯色作为tint,并保持原来的alpha通道。用Core Graphics来做的话,大概的想法很直接:

  1. 创建一个上下文用以画新的图片
  2. 将新的tintColor设置为填充颜色
  3. 将原图片画在创建的上下文中,并用新的填充色着色(注意保持alpha通道)
  4. 从当前上下文中取得图片并返回

最麻烦的部分可能就是保持alpha通道了。UIImage的文档中提供了使用blend绘图的方法drawInRect:blendMode:alpha:rectalpha都没什么问题,但是blendMode是个啥玩意儿啊…继续看文档,关于CGBlendMode的文档,里面有一大堆看不懂的枚举值,比如这样:

[objc]  view plain copy
在CODE上查看代码片 派生到我的代码片
  1. kCGBlendModeDestinationOver  
  2. R = S*(1 - Da) + D  
  3. Available in iOS 2.0 and later.  
  4. Declared in CGContext.h.  

完全不懂..直接看之后的Discussion部分:

The blend mode constants introduced in OS X v10.5 represent the Porter-Duff blend modes. The symbols in the equations for these blend modes are:
R is the premultiplied result
S is the source color, and includes alpha
D is the destination color, and includes alpha
Ra, Sa, and Da are the alpha components of R, S, and D

原来如此,R表示结果,S表示包含alpha的原色,D表示包含alpha的目标色,Ra,Sa和Da分别是三个的alpha。明白了这些以后,就可以开始寻找我们所需要的blend模式了。相信你可以和我一样,很快找到这个模式:

[objc]  view plain copy
在CODE上查看代码片 派生到我的代码片
  1. kCGBlendModeDestinationIn  
  2. R = D*Sa  
  3. Available in iOS 2.0 and later.  
  4. Declared in CGContext.h.  

结果 = 目标色和原色透明度的加成,看起来正式所需要的。啦啦啦,还等什么呢,开始动手实现看看对不对吧~

为了以后使用方便,当然是祭出Category,先创建一个UIImage的类别:

[objc]  view plain copy
在CODE上查看代码片 派生到我的代码片
  1. //  UIImage+Tint.h  
  2.   
  3. #import <UIKit/UIKit.h>  
  4.   
  5. @interface UIImage (Tint)  
  6.   
  7. - (UIImage *) imageWithTintColor:(UIColor *)tintColor;  
  8.   
  9. @end  

暂时先这样,当然我们也可以创建一个类方法直接完成从bundle读取图片然后加tintColor,但是很多时候并不如上面一个实例方法方便(比如想要从非bundle的地方获取图片),这个问题之后再说。那么就按照之前设想的步骤来实现吧:

[objc]  view plain copy
在CODE上查看代码片 派生到我的代码片
  1. //  UIImage+Tint.m  
  2.   
  3. #import "UIImage+Tint.h"  
  4.   
  5. @implementation UIImage (Tint)  
  6. - (UIImage *) imageWithTintColor:(UIColor *)tintColor  
  7. {  
  8.     //We want to keep alpha, set opaque to NO; Use 0.0f for scale to use the scale factor of the device’s main screen.  
  9.     UIGraphicsBeginImageContextWithOptions(self.sizeNO0.0f);  
  10.     [tintColor setFill];  
  11.     CGRect bounds = CGRectMake(00self.size.widthself.size.height);  
  12.     UIRectFill(bounds);  
  13.   
  14.     //Draw the tinted image in context  
  15.     [self drawInRect:bounds blendMode:kCGBlendModeDestinationIn alpha:1.0f];  
  16.   
  17.     UIImage *tintedImage = UIGraphicsGetImageFromCurrentImageContext();  
  18.     UIGraphicsEndImageContext();  
  19.   
  20.     return tintedImage;  
  21. }  
  22. @end  

简单明了,没有任何难点。测试之: [[UIImage imageNamed:@"image"] imageWithTintColor:[UIColor orangeColor]]; ,得到的结果为:


嗯…怎么说呢,虽然tintColor的颜色是变了,但是总觉得怪怪的。仔细对比一下就会发现,原来灰色图里星星和周围的灰度渐变到了橙色的图里好像都消失了:星星整个变成了橙色,周围的一圈漂亮的光晕也没有了,这是神马情况啊…这种图能交差的话那算见鬼了,会被设计和产品打死的吧。对于无渐变的纯色图的图来说直接用上面的方法是没问题的,但是现在除了Metro的大色块以外哪里无灰度渐变的设计啊…检查一下使用的blend,R = D * Sa,恍然大悟,我们虽然保留了原色的透明度,但是却把它的所有的灰度信息弄丢了。怎么办?继续刨CGBlendMode的文档吧,那么多blend模式应该总有我们需要的。功夫不负有心人,kCGBlendModeOverlay一副嗷嗷待选的样子:

[objc]  view plain copy
在CODE上查看代码片 派生到我的代码片
  1. kCGBlendModeOverlay  
  2. Either multiplies or screens the source image samples with the background image samples, depending on the background color. The result is to overlay the existing image samples while preserving the highlights and shadows of the background. The background color mixes with the source image to reflect the lightness or darkness of the background.  
  3. Available in iOS 2.0 and later.  
  4. Declared in CGContext.h.  

kCGBlendModeOverlay可以保持背景色的明暗,也就是灰度信息,听起来正是我们需要的。加入到声明中,并且添加相应的实现( 顺便重构一下原来的代码 :) ):

[objc]  view plain copy
在CODE上查看代码片 派生到我的代码片
  1. //  UIImage+Tint.h  
  2.   
  3. #import <UIKit/UIKit.h>  
  4.   
  5. @interface UIImage (Tint)  
  6.   
  7. - (UIImage *) imageWithTintColor:(UIColor *)tintColor;  
  8. - (UIImage *) imageWithGradientTintColor:(UIColor *)tintColor;  
  9.   
  10. @end  

[objc]  view plain copy
在CODE上查看代码片 派生到我的代码片
  1. //  UIImage+Tint.m  
  2.   
  3. #import "UIImage+Tint.h"  
  4.   
  5. @implementation UIImage (Tint)  
  6. - (UIImage *) imageWithTintColor:(UIColor *)tintColor  
  7. {  
  8.     return [self imageWithTintColor:tintColor blendMode:kCGBlendModeDestinationIn];  
  9. }  
  10.   
  11. - (UIImage *) imageWithGradientTintColor:(UIColor *)tintColor  
  12. {  
  13.     return [self imageWithTintColor:tintColor blendMode:kCGBlendModeOverlay];  
  14. }  
  15.   
  16. - (UIImage *) imageWithTintColor:(UIColor *)tintColor blendMode:(CGBlendMode)blendMode  
  17. {  
  18.     //We want to keep alpha, set opaque to NO; Use 0.0f for scale to use the scale factor of the device’s main screen.  
  19.     UIGraphicsBeginImageContextWithOptions(self.sizeNO0.0f);  
  20.     [tintColor setFill];  
  21.     CGRect bounds = CGRectMake(00self.size.widthself.size.height);  
  22.     UIRectFill(bounds);  
  23.   
  24.     //Draw the tinted image in context  
  25.     [self drawInRect:bounds blendMode:blendMode alpha:1.0f];  
  26.   
  27.     UIImage *tintedImage = UIGraphicsGetImageFromCurrentImageContext();  
  28.     UIGraphicsEndImageContext();  
  29.   
  30.     return tintedImage;  
  31. }  
  32.   
  33. @end  

完成,测试之…好吧,好尴尬,虽然颜色和周围的光这次对了,但是透明度又没了啊魂淡..一点不奇怪啊,因为 kCGBlendModeOverlay 本来就没承诺给你保留原图的透明度的说。


那么..既然我们用kCGBlendModeOverlay能保留灰度信息,用kCGBlendModeDestinationIn能保留透明度信息,那就两个blendMode都用不就完事儿了么~尝试之,如果在blend绘图时不是kCGBlendModeDestinationIn模式的话,则再用kCGBlendModeDestinationIn画一次:

[objc]  view plain copy
在CODE上查看代码片 派生到我的代码片
  1. //  UIImage+Tint.m  
  2.   
  3. #import "UIImage+Tint.h"  
  4.   
  5. @implementation UIImage (Tint)  
  6. - (UIImage *) imageWithTintColor:(UIColor *)tintColor  
  7. {  
  8.     return [self imageWithTintColor:tintColor blendMode:kCGBlendModeDestinationIn];  
  9. }  
  10.   
  11. - (UIImage *) imageWithGradientTintColor:(UIColor *)tintColor  
  12. {  
  13.     return [self imageWithTintColor:tintColor blendMode:kCGBlendModeOverlay];  
  14. }  
  15.   
  16. - (UIImage *) imageWithTintColor:(UIColor *)tintColor blendMode:(CGBlendMode)blendMode  
  17. {  
  18.     //We want to keep alpha, set opaque to NO; Use 0.0f for scale to use the scale factor of the device’s main screen.  
  19.     UIGraphicsBeginImageContextWithOptions(self.sizeNO0.0f);  
  20.     [tintColor setFill];  
  21.     CGRect bounds = CGRectMake(00self.size.widthself.size.height);  
  22.     UIRectFill(bounds);  
  23.   
  24.     //Draw the tinted image in context  
  25.     [self drawInRect:bounds blendMode:blendMode alpha:1.0f];  
  26.   
  27.     if (blendMode != kCGBlendModeDestinationIn) {  
  28.         [self drawInRect:bounds blendMode:kCGBlendModeDestinationIn alpha:1.0f];  
  29.     }  
  30.   
  31.     UIImage *tintedImage = UIGraphicsGetImageFromCurrentImageContext();  
  32.     UIGraphicsEndImageContext();  
  33.   
  34.     return tintedImage;  
  35. }  
  36.   
  37. @end  

结果如下:


已经很完美了,这样的话只要在代码里设定一下颜色,我们就能够很轻易地使用同样一套UI,将其blend为需要的颜色,来实现素材的重用了。唯一需要注意的是,因为每次使用UIImage+Tint的方法绘图时,都使用了CG的绘制方法,这就意味着每次调用都会是用到CPU的Offscreen drawing,大量使用的话可能导致性能的问题(主要对于iPhone 3GS或之前的设备,可能同时处理大量这样的绘制调用的能力会有不足)。关于CA和CG的性能的问题,打算在之后用一篇文章来介绍一下。对于这里的UIImage+Tint的实现,可以写一套缓存的机制,来确保大量重复的元素只在load的时候blend一次,之后将其缓存在内存中以快速读取。当然这是一个权衡的问题,在时间和空间中做出正确的平衡和选择,也正是程序设计的乐趣所在。

这篇文章中作为示例的工程和UIImage+Tint可以在Github上找到,您可以随意玩弄..我相信也会是个来研究每种blend的特性的好机会~


转自:http://onevcat.com/2013/04/using-blending-in-iOS/

这篇关于iOS开发笔记--使用blend改变图片颜色的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/841050

相关文章

C++11右值引用与Lambda表达式的使用

《C++11右值引用与Lambda表达式的使用》C++11引入右值引用,实现移动语义提升性能,支持资源转移与完美转发;同时引入Lambda表达式,简化匿名函数定义,通过捕获列表和参数列表灵活处理变量... 目录C++11新特性右值引用和移动语义左值 / 右值常见的左值和右值移动语义移动构造函数移动复制运算符

Python对接支付宝支付之使用AliPay实现的详细操作指南

《Python对接支付宝支付之使用AliPay实现的详细操作指南》支付宝没有提供PythonSDK,但是强大的github就有提供python-alipay-sdk,封装里很多复杂操作,使用这个我们就... 目录一、引言二、准备工作2.1 支付宝开放平台入驻与应用创建2.2 密钥生成与配置2.3 安装ali

C#中lock关键字的使用小结

《C#中lock关键字的使用小结》在C#中,lock关键字用于确保当一个线程位于给定实例的代码块中时,其他线程无法访问同一实例的该代码块,下面就来介绍一下lock关键字的使用... 目录使用方式工作原理注意事项示例代码为什么不能lock值类型在C#中,lock关键字用于确保当一个线程位于给定实例的代码块中时

PyQt5 GUI 开发的基础知识

《PyQt5GUI开发的基础知识》Qt是一个跨平台的C++图形用户界面开发框架,支持GUI和非GUI程序开发,本文介绍了使用PyQt5进行界面开发的基础知识,包括创建简单窗口、常用控件、窗口属性设... 目录简介第一个PyQt程序最常用的三个功能模块控件QPushButton(按钮)控件QLable(纯文本

MySQL 强制使用特定索引的操作

《MySQL强制使用特定索引的操作》MySQL可通过FORCEINDEX、USEINDEX等语法强制查询使用特定索引,但优化器可能不采纳,需结合EXPLAIN分析执行计划,避免性能下降,注意版本差异... 目录1. 使用FORCE INDEX语法2. 使用USE INDEX语法3. 使用IGNORE IND

C# $字符串插值的使用

《C#$字符串插值的使用》本文介绍了C#中的字符串插值功能,详细介绍了使用$符号的实现方式,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起学习学习吧... 目录$ 字符使用方式创建内插字符串包含不同的数据类型控制内插表达式的格式控制内插表达式的对齐方式内插表达式中使用转义序列内插表达式中使用

flask库中sessions.py的使用小结

《flask库中sessions.py的使用小结》在Flask中Session是一种用于在不同请求之间存储用户数据的机制,Session默认是基于客户端Cookie的,但数据会经过加密签名,防止篡改,... 目录1. Flask Session 的基本使用(1) 启用 Session(2) 存储和读取 Se

Java Thread中join方法使用举例详解

《JavaThread中join方法使用举例详解》JavaThread中join()方法主要是让调用改方法的thread完成run方法里面的东西后,在执行join()方法后面的代码,这篇文章主要介绍... 目录前言1.join()方法的定义和作用2.join()方法的三个重载版本3.join()方法的工作原

Spring AI使用tool Calling和MCP的示例详解

《SpringAI使用toolCalling和MCP的示例详解》SpringAI1.0.0.M6引入ToolCalling与MCP协议,提升AI与工具交互的扩展性与标准化,支持信息检索、行动执行等... 目录深入探索 Spring AI聊天接口示例Function CallingMCPSTDIOSSE结束语

Linux系统之lvcreate命令使用解读

《Linux系统之lvcreate命令使用解读》lvcreate是LVM中创建逻辑卷的核心命令,支持线性、条带化、RAID、镜像、快照、瘦池和缓存池等多种类型,实现灵活存储资源管理,需注意空间分配、R... 目录lvcreate命令详解一、命令概述二、语法格式三、核心功能四、选项详解五、使用示例1. 创建逻