iOS利用NSAttributeString实现不同颜色大小显示的方法

前言

最近开发需求遇到一个比较简单但又棘手的问题.先看需求

一个UILabel显示不同大小颜色的字符串,当然我们首先的想到属性字符串,但是注意: 我们这里要处理国际化完成的字符串也就是说:

必须在国际化完成以后才能追加我们的逻辑,而不是一上来就加属性字符串

比如: 2分14秒 or 2min14secs

也就是给我们的是一个 "2分14秒"字符串 我们需要匹配range来修改或者替换. 带着这个疑问开始今天的文章?

实现思路

孔圣贤有云:”举一隅不以三隅反,则不复也。”

出自《论语·第七章·述而篇》

为了不愧对圣贤对我的期待我把 这个问题定位升级成 4个等级

  • Level 1 最优解,时间复杂度最低,效率最高
  • Level 2 非最优解,时间复杂度最低,效率高
  • Level 3 都一般
  • Level 4 简单粗暴

我想到了以下至少两种方法

  • 通过计算出来的时间 eg: 分 秒 字符串 range去国际化处理完的字符串去匹配修改
  • 用正则匹配数字
  • 用谓词匹配数字
  • level4太业余了不敢想向一个工作好几年的开发者还写出这么打脸的代码

准备工作

在工程中拖拽了一个label

@interface ViewController ()
@property (weak, nonatomic) IBOutlet UILabel *label;
@end
@implementation ViewController
- (void)viewDidLoad {
 [super viewDidLoad];

 //调用
 NSAttributedString *resultTime = [self formattedCurrentTime:133];
 self.label.attributedText = resultTime;
}

方案1: 字符串range匹配

/**
 返回当前时间格式
 @return 返回组装好的字符串
 */
- (NSAttributedString *)formattedCurrentTime:(NSTimeInterval)timeInterval {

 NSUInteger time = (NSUInteger)timeInterval;
 NSInteger minutes = (time / 60) % 60;
 NSInteger seconds = time % 60;
 NSString *minStr = [NSString stringWithFormat:@" %zd ",minutes];
 NSString *secStr = [NSString stringWithFormat:@" %zd ",seconds];
 //假设这就是我们国际化后的字符串
 NSString *localizedFormatString = [NSString stringWithFormat:@"%@分%@秒",minStr,secStr];
 NSMutableAttributedString *attributeStr = [[NSMutableAttributedString alloc] initWithString:localizedFormatString];
 NSRange minRange, secRange;
 if (@available(iOS 9.0, *)) {
 minRange = [localizedFormatString localizedStandardRangeOfString:minStr];
 secRange = [localizedFormatString localizedStandardRangeOfString:secStr];
 } else {
 minRange = [localizedFormatString rangeOfString:minStr];
 secRange = [localizedFormatString rangeOfString:secStr];
 }
 NSDictionary *timeAttrs = @{ NSForegroundColorAttributeName : [UIColor redColor],
     NSFontAttributeName : [UIFont systemFontOfSize:40.0f]};
 [attributeStr addAttributes:timeAttrs range:minRange];
 [attributeStr addAttributes:timeAttrs range:secRange];
 return [[NSAttributedString alloc] initWithAttributedString:attributeStr];;
}

看下显示结果

是不是看上去很好

但我认为这并不完美,这种搞法虽然简单直接,但是过于依赖minStr和secStr的原始range,基于iOS9之后提供的API计算range

if (@available(iOS 9.0, *)) {
 minRange = [localizedFormatString localizedStandardRangeOfString:minStr];
 secRange = [localizedFormatString localizedStandardRangeOfString:secStr];
} else {
 minRange = [localizedFormatString rangeOfString:minStr];
 secRange = [localizedFormatString rangeOfString:secStr];
}

注意:API平台区分

但是这么实现有个Bug 当遇到同样字符串的时候就会匹配错位, 如图

错误的原因显然大家都了解

字符串 “0” 的range相同了,但就解决这个问题而言,简单判断一下range然后截取字符串向后跳跃length继续截取获取能实现,但这显然很啰嗦,万一有一天 你遇到的是 “0小时0分12秒“这种字符串那该如何写呢?

是不是要递归的遍历一遍然后挨个取Range 做属性修改?

这样的结果显然不但代码啰嗦 实现起来成本还是比较高的,对代码阅读性都有很大影响(写得好的代码除外哈).

那怎么不啰嗦呢?

有一种搞法就是 用两个不同的字符占位.然后 国际化完成之后取Range,再然后替换文字,搞法虽然low点,但是时间复杂度降低了不少,还是可以考虑的.代码我就不写了 我怕小伙伴review代码的时候会虐我.继续往下看

评级: Level 2

那如何不依赖range解决这种问题呢?

方案2: 正则匹配

/**
 返回当前时间格式
 @return 返回组装好的字符串
 */
- (NSAttributedString *)formattedCurrentTime:(NSTimeInterval)timeInterval {

 NSUInteger time = (NSUInteger)timeInterval;
 NSInteger minutes = (time / 60) % 60;
 NSInteger seconds = time % 60;
 NSString *minStr = [NSString stringWithFormat:@" %zd ",minutes];
 NSString *secStr = [NSString stringWithFormat:@" %zd ",seconds];
 //假设这就是我们国际化后的字符串
 NSString *localizedFormatString = [NSString stringWithFormat:@"%@分%@秒",minStr,secStr];
 NSMutableAttributedString *attributeStr = [[NSMutableAttributedString alloc] initWithString:localizedFormatString];
 NSDictionary *timeAttrs = @{ NSForegroundColorAttributeName : [UIColor redColor],
     NSFontAttributeName : [UIFont systemFontOfSize:40.0f]};
 /** 方案2 **/
 NSError *error = nil;
 NSRegularExpression *reg = [NSRegularExpression regularExpressionWithPattern:@"[0-9]+" options:NSRegularExpressionCaseInsensitive error:&error];
 if (error == nil) {
 NSArray *matches = [reg matchesInString:localizedFormatString options:NSMatchingReportCompletion range:NSMakeRange(0, localizedFormatString.length)];
 for (NSTextCheckingResult *match in matches) {
  for (NSUInteger i = 0; i < match.numberOfRanges; i++) {
  NSRange range = [match rangeAtIndex:i];
  if (range.location != NSNotFound) {
   [attributeStr addAttributes:timeAttrs range:range];
  }
  }
 }
 }
 return [[NSAttributedString alloc] initWithAttributedString:attributeStr];;
}

看下显示结果

完美实现

这种方案缺点就是,时间复杂度高了一些,需要每次正则遍历
有点是扩展性好一点,万一有一天PM又提了需求要做成 A1 B2 C3 XXX#话题这种,那一定会出坑

但我第一次这么实现被小伙伴嘲笑很业余.确实很业余,但它能避免方案1中的bug.而且相当精确.

评级: Level 2

方案3: 谓词匹配

这种搞法我没尝试,估计会比 方案1和方案2都快一些和简单直接一些,时间太紧张算了,期待评论轻喷吧!

方案4: 简单粗暴

就搞 4个label. 我都想象到了被实习生嘲讽+打脸的搞法发生在一个工作好几年开发者身上是多么惨痛的画面. 放弃这种low的搞法

总结

最终解决问题的方案还是方案2:正则匹配比较靠谱,而且一劳永逸

本篇主要蛋疼的问题是 国际化后的字符串返回结果后,对返回的结果进行加工处理.

没有做到Level 1级的做法很是遗憾,愧对圣贤. 希望小伙伴多提提建议.

Demo在这里找到

补充

格式化时间的代码

/**
 返回时间格式 HH:mm:ss
 @return 返回组装好的字符串
 */
- (NSString *)formattedCurrentTime {
 NSUInteger time = (NSUInteger)self.recorder.currentTime;
 NSInteger hours = (time / 3600);
 NSInteger minutes = (time / 60) % 60;
 NSInteger seconds = time % 60;

 NSString *format = @"%02i:%02i:%02i";
 return [NSString stringWithFormat:format, hours, minutes, seconds];
}

全文完

好了,以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对我们的支持。

(0)

相关推荐

  • iOS中Label实现显示不同颜色与字体的方法

    前言 iOS中Label是我们经常遇到的一个控件,我们大家应该都会简单的使用它,像下面这个代码,就能简单的创建一个label // 1.创建 CGRectrect =CGRectMake(100,100,100,100); UILabel* label = [[UILabelalloc]initWithFrame:rect]; 引言 然而我们在开发中,经常会遇到一行字,但是显示不同颜色和字体的情况,话不多说,直接上代码. 1.显示不同颜色,有两种方式 (1)通过 range 来设置 NSMuta

  • iOS利用NSAttributeString实现不同颜色大小显示的方法

    前言 最近开发需求遇到一个比较简单但又棘手的问题.先看需求 一个UILabel显示不同大小颜色的字符串,当然我们首先的想到属性字符串,但是注意: 我们这里要处理国际化完成的字符串也就是说: 必须在国际化完成以后才能追加我们的逻辑,而不是一上来就加属性字符串 比如: 2分14秒 or 2min14secs 也就是给我们的是一个 "2分14秒"字符串 我们需要匹配range来修改或者替换. 带着这个疑问开始今天的文章? 实现思路 孔圣贤有云:"举一隅不以三隅反,则不复也.&quo

  • PHP实现格式化文件数据大小显示的方法

    本文实例讲述了PHP实现格式化文件数据大小显示的方法.分享给大家供大家参考.具体分析如下: 有时候我们需要在网页上显示某个文件的大小,或者是其它数据的大小数字. 这个数字往往从跨度很大,如果以B为单位的话可能是个位,如果1G则长达1073741824的数字,这个时候我们就需要根据大小来格式化,比如小于1K则以B为单位显示,小于1M则以KB为单位显示,小于1G则以MB为单位显示,以此类推... 格式化函数参考如下: 复制代码 代码如下: //格式化size显示 function formatSiz

  • iOS利用摄像头获取环境光感参数的方法

    本文介绍了iOS利用摄像头获取环境光感参数的方法,分享给大家,具体如下: 不多说,代码如下: #import "LightSensitiveViewController.h" @import AVFoundation; #import <ImageIO/ImageIO.h> @interface LightSensitiveViewController ()< AVCaptureVideoDataOutputSampleBufferDelegate> @prop

  • iOS 缩小打包项目ipa大小的实现方法

    之前项目上线完全由技术老大搞,这次独立开发自己来,觉得自己的打包项目体积略大,网上搜索了一些比较不错的方法,这里总结下. 1.配置编译选项 (Levels选项内)Genetate Debug Symbols 设置为NO,这个配置选项应该会让你减去小半的体积. 2.舍弃架构armv7,因为armv7用于支持4s和3gs,4s是2011年11月正式上线,虽然还有小部分人在使用,但是追求包体大小的完全可以舍弃了. 3.编译的版本必须是发布版本, 4.查找内部使用到的第三方库,一方面可以进行删减代码,用

  • iOS利用Label实现的简单高性能标签TagView

    前言 我相信很多人在开发者都有这样的需求,标签展示(如下图) 很多人都可以自己实现(网上别人写的也很多,但是别人写的总有不满足自己需求的点),实现的方法也很多种,比如动态添加view,使用UICollectionView等等.这种实现方法不是不好,但是当列表比较复杂,数据比较多的时候,可曾想过性能会怎么样呢? 在一次深入了解富文本的时候,突发其想,好像富文本能达到这种效果,也就是一个label就可以实现这种标签的效果了,效果性能就不用多说了,再加上YYLabel的异步绘制,真是锦上添花啊. XW

  • iOS利用余弦函数实现卡片浏览工具

    本文实例为大家分享了iOS利用余弦函数实现卡片浏览工具的具体代码,供大家参考,具体内容如下 一.实现效果 通过拖拽屏幕实现卡片移动,左右两侧的卡片随着拖动变小,中间的变大.效果如下: 二.原理说明 1.上面的动画效果是根据余弦函数的曲线特性实现的,先看一下函数曲线y=cos(x),在区间-π/2 到 π/2的范围内,y的值在x的0的是后是最大的,左右则越来越小. 2.可以将被滚动的卡片的高度按照0.0~1.0的比例放大缩小,效果如下: 3.放置到手机屏幕上的效果如下: 三.代码 封装每个卡片为C

  • python画柱状图--不同颜色并显示数值的方法

    用python画柱状图容易,但是如何对不同柱子使用不同颜色呢?同时在柱子顶端显示精确数值? 主要用的方法为: atplotlib.pyplot.bar(left, height, width=0.8, bottom=None, hold=None, data=None, **kwargs) 参数说明: left: 每一个柱形左侧的X坐标 height:每一个柱形的高度 width: 柱形之间的宽度 bottom: 柱形的Y坐标 color: 柱形的颜色 下面是代码示例(首先请先安装numpy以及

  • Python中利用pyqt5制作指针钟表显示实时时间(指针时钟)

    文末附完整源代码实现过程... 想实现这样一个功能,然后pyqt5中又没有现成的组件可以使用,于是就想着只能通过绘图的方式来实现.说到绘图的话,turtle框架无疑是最常见的选择,但其实通过pyqt5的QPainter组件也是可以实现的.而且最后呈现出来的效果还是挺漂亮的. 实现思路:通过使用pyqt5的QPainter组件来绘制好时钟的图表,最后通过定时器不断的改变当前当前时间在图表上面的显示位置.这样最终就实现了一个指针时钟在不断的走动的过程. 和前面的UI应用一样,我们用到的UI相关的组件

  • iOS利用UIBezierPath + CAAnimation实现路径动画效果

    前言 上次给大家介绍了iOS利用UIBezierPath + CAAnimation实现路径动画效果的相关内容,今天实现一个根据心跳路径实现一个路径动画,让某一视图沿着路径进行运动.. 效果图如下: 核心代码 1-首先通过 drawRect 绘制心形路径 - (void)drawRect:(CGRect)rect { // Drawing code // 初始化UIBezierPath UIBezierPath *path = [UIBezierPath bezierPath]; // 首先设置

  • iOS利用UIScrollView实现图片的缩放实例代码

    本文介绍了iOS利用UIScrollView实现图片的缩放实例代码,分享给大家: 第一步:添加scrollView到控制器中 UIScrollView *scrollView = [[UIScrollView alloc] init]; scrollView.frame = CGRectMake(40, 250, 300, 200); self.scrollView = scrollView; [self.view addSubview:scrollView]; 第二步:添加图片控件到scrol

随机推荐