TextKit学习(一)可以调整字体大小的FontResizeDemo

昨天看完了苹果官方的IntroduceToTextKit这个Demo,了解了有关TextKit的一些新特性,TextKit的出现的确使得文字的处理变得更加便利和多功能化。个人觉得最有趣的是ExclusionPath这个部分。


之后参照这个Demo我写了FontResizeDemo,顾名思义就是当用户在Settings的Text Size中调整字体大小时,本程序的字体相应地做出调整。

先说说Xcode 5-DP2的一些使用感想:

1.操作便利性没有改进:

在iOS7之前,如果在程序中导入新的框架,必须在Building Phases中的Link Binary with Libraries选项中手动添加对应的框架文件,这种方式的确非常麻烦。Xcode 5则解决了手动添加的麻烦,只需要在代码中添加就可以直接使用。所以在WWDC2013中的关于Obj-C新特性的视频中,说完这一特性以后可以清楚听到下面的鼓掌欢呼声。

但是,昨天使用Xcode 5才发现,为了在自己新建的类中使用框架等文件时,必须要在Building Phases中的Compile Sources中手动添加类文件,如果不这么做(即使在类文件中导入了<UIKit/UIKit.h>),最明显的结果就是在写代码时如UIView这样的名词都不会显示代码高亮状态。同样地,如果要使用外部资源中的内容(如book.txt电子书中的内容),也必须手动在Copy Bundle Resources中添加这些资源文件。如果不这样做,那么在程序运行时如果使用了资源中的内容(如在控制台输出电子书中的内容),内容将返回结果null。添加文件后入下图所示:

TextKit学习(一)可以调整字体大小的FontResizeDemo_第1张图片

这样比较下来,必须手动在编译源(Compile Sources)和资源(Bundle Resources)中添加文件,整个开发过程将变得更加麻烦。

2.iOS7 Simulator存在问题

如果使用iOS7模拟器反复调试或退出模拟器重新run一遍,将发现程序在运行开始时卡住,即使在Debug area中点击continue也没有作用。必须关闭程序重新打开工程文件才能在模拟器中正常运行程序。不知道是模拟器存在bug还是我没找对调试的方法。

另外i7模拟器的外观也放弃了i6模拟器的iPhone外观,在明显的就是Home键没有了,要按Home键只能按Command+Shift+H快捷键或者在模拟器的硬件菜单中选择。如果要打开后台,必须连续按两次Command+Shift+H快捷键。在模拟器的使用上,个人觉得还是iOS6的使用更为便利。


回到Demo中来。这个Demo我只写了一个FontResizeViewController类,下面来看看该类的实现文件:

[cpp]  view plain copy
  1. #import "FontResizeViewController.h"  
  2. #import "UITextView+TextKitDemo.h"  
  3. #import "UIFont+TextKitDemo.h"  
  4. #import "UIFontDescriptor+TextKitDemo.h"  
  5.   
  6. @interface FontResizeViewController ()  
  7. @property (nonatomic, strong) NSMutableAttributedString *content; // 文本内容  
  8. @end  
  9.   
  10. @implementation FontResizeViewController  
  11.   
  12. - (id)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil  
  13. {  
  14.     self = [super initWithNibName:nibNameOrNil bundle:nibBundleOrNil];  
  15.     if (self) {  
  16.         // Custom initialization  
  17.     }  
  18.     return self;  
  19. }  
  20.   
  21. - (void)viewDidLoad  
  22. {  
  23.     [super viewDidLoad];  
  24.       
  25.     self.textView = [[UITextView alloc] initWithFrame:CGRectMake(0.0, 30.0, 320.0, 520.0)];  
  26.     [self.view addSubview:self.textView];  
  27.       
  28.     NSURL *url = nil;  
  29.     // url = [[NSBundle mainBundle] URLForResource:@"Basic Interaction.rtf" withExtension:nil];  
  30.     url = [[NSBundle mainBundle] URLForResource:@"book.txt" withExtension:nil];  
  31.     NSMutableAttributedString *attributedTextHolder = [[NSMutableAttributedString alloc] initWithFileURL:url options:@{} documentAttributes:nil error:nil];  
  32.     [attributedTextHolder addAttribute:NSFontAttributeName value:[UIFont preferredFontForTextStyle:UIFontTextStyleBody] range:NSMakeRange(0, attributedTextHolder.length)];  
  33.     // NSLog(@"%@", attributedTextHolder);  
  34.     self.content = [attributedTextHolder copy];  
  35.      
  36.     self.textView.attributedText = self.content;  
  37.     self.textView.editable = NO;  
  38.       
  39.     [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(preferredContentSizeChanged:) name:UIContentSizeCategoryDidChangeNotification object:nil]; // 当不同类别的字体大小发生变化时发送消息给self  
  40. }  
  41.   
  42. -(void)preferredContentSizeChanged:(NSNotification *)noti  
  43. {  
  44.     static const CGFloat textScaleFactor = .8;  
  45.     NSString *textStyle = [self.textView tkd_textStyle]; // 先设置样式  
  46.     UIFont *font = [UIFont tkd_preferredFontWithTextStyle:textStyle scale:textScaleFactor]; //后设置字体(包括样式和大小)  
  47.       
  48.     self.textView.font = font;  
  49. }  
  50.   
  51. - (void)viewDidUnload  
  52. {  
  53.     [[NSNotificationCenter defaultCenter] removeObserver:self];  
  54.       
  55.     [super viewDidUnload];  
  56. }  
  57.   
  58. - (void)dealloc  
  59. {  
  60.     [[NSNotificationCenter defaultCenter] removeObserver:self];  
  61. }  
  62.   
  63. - (void)didReceiveMemoryWarning  
  64. {  
  65.     [super didReceiveMemoryWarning];  
  66.     // Dispose of any resources that can be recreated.  
  67. }  
  68.   
  69. @end  

非常简单,关键就是在控制器中添加一个事件观察者来接收消息:

[cpp]  view plain copy
  1. [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(preferredContentSizeChanged:) name:UIContentSizeCategoryDidChangeNotification object:nil]; // 当不同类别的字体大小发生变化时发送消息给self  

另外要注意的是如果以self.content = @"abcdefghijklmnopqrstuvwxyz"的方式简单初始化self.content的内容,那么在调整完文字大小回到程序时self.content的内容可能已经被回收,可以看到文字内容显示为空。因此在将self.content赋值给self.textView.attributedText前,self.content必须要retain一次,也就是self.content必须要以init或者copy等方式获得:

[cpp]  view plain copy
  1. NSURL *url = nil;  
  2.    // url = [[NSBundle mainBundle] URLForResource:@"Basic Interaction.rtf" withExtension:nil];  
  3.    url = [[NSBundle mainBundle] URLForResource:@"book.txt" withExtension:nil];  
  4.    NSMutableAttributedString *attributedTextHolder = [[NSMutableAttributedString alloc] initWithFileURL:url options:@{} documentAttributes:nil error:nil];  
  5.    [attributedTextHolder addAttribute:NSFontAttributeName value:[UIFont preferredFontForTextStyle:UIFontTextStyleBody] range:NSMakeRange(0, attributedTextHolder.length)];  
  6.    // NSLog(@"%@", attributedTextHolder);  
  7.    self.content = [attributedTextHolder copy];  

必须重写viewDidUnload和dealloc方法来移除事件观察者:

[cpp]  view plain copy
  1. - (void)viewDidUnload  
  2. {  
  3.     [[NSNotificationCenter defaultCenter] removeObserver:self];  
  4.       
  5.     [super viewDidUnload];  
  6. }  
  7.   
  8. - (void)dealloc  
  9. {  
  10.     [[NSNotificationCenter defaultCenter] removeObserver:self];  
  11. }  

重点看看收到消息后调整字体大小的方法:

[cpp]  view plain copy
  1. -(void)preferredContentSizeChanged:(NSNotification *)noti  
  2. {  
  3.     static const CGFloat textScaleFactor = .8;  
  4.     NSString *textStyle = [self.textView tkd_textStyle]; // 先设置样式  
  5.     UIFont *font = [UIFont tkd_preferredFontWithTextStyle:textStyle scale:textScaleFactor]; //后设置字体(包括样式和大小)  
  6.       
  7.     self.textView.font = font;  
  8. }  

在这里用到的tkd_textStyle和tkd_preferredFontWithTextStyle:scale:是调整字体样式和大小关键。在这里使用了IntroduceTextKitDemo中的Categories中的类别和方法:
TextKit学习(一)可以调整字体大小的FontResizeDemo_第2张图片
下面来看看取得字体样式的整个流程:
(1)在FontResizeViewController.m中调用UITextView类别的tkd_textStyle方法。
[cpp]  view plain copy
  1. NSString *textStyle = [self.textView tkd_textStyle]; // 先设置样式  
(2)在UITextView+TextKitDemo.m中的tkd_textStyle方法调用UIFont类别的tkd_textStyle方法。
[cpp]  view plain copy
  1. - (NSString *)tkd_textStyle  
  2. {  
  3.     return [self.font tkd_textStyle];  
  4. }  
(3)在UIFont+TextKitDemo.m中的tkd_textStyle方法中调用UIFontDecsriptor类别的tkd_textStyle方法。
[cpp]  view plain copy
  1. - (NSString *)tkd_textStyle  
  2. {  
  3.     return [self.fontDescriptor tkd_textStyle];  
  4. }  
(4)在UIFontDescriptor+TextKitDemo.m的tkd_textStyle方法中回调字体样式结果。
[cpp]  view plain copy
  1. - (NSString *)tkd_textStyle  
  2. {  
  3.     return [self objectForKey:@"NSCTFontUIUsageAttribute"];  
  4. }  


在设置后字体样式后调用tkd_preferredFontWithTextStyle:scale:方法来设置字体,textStyle的参数使用了以上获得的tkd_textStyle,scale的参数则使用了textScaleFactor。整个方法的工作流程是相似的:
(1)FontResizeViewController.m
[cpp]  view plain copy
  1. UIFont *font = [UIFont tkd_preferredFontWithTextStyle:textStyle scale:textScaleFactor]; //后设置字体(包括样式和大小)  
(2)UIFont+TextKitDemo.m
[cpp]  view plain copy
  1. + (UIFont *)tkd_preferredFontWithTextStyle:(NSString *)aTextStyle scale:(CGFloat)aScale  
  2. {  
  3.     UIFontDescriptor *newFontDescriptor = [UIFontDescriptor tkd_preferredFontDescriptorWithTextStyle:aTextStyle scale:aScale];  
  4.   
  5.     return [UIFont fontWithDescriptor:newFontDescriptor size:newFontDescriptor.pointSize];  
  6. }  
(3)UIFontDescriptor+TextKitDemo.m
[cpp]  view plain copy
  1. + (UIFontDescriptor *)tkd_preferredFontDescriptorWithTextStyle:(NSString *)aTextStyle scale:(CGFloat)aScale  
  2. {  
  3.     UIFontDescriptor *newBaseDescriptor = [self preferredFontDescriptorWithTextStyle:aTextStyle];  
  4.   
  5.     return [newBaseDescriptor fontDescriptorWithSize:lrint([newBaseDescriptor pointSize] * aScale)];  
  6. }  


这就是整个程序的工作过程,主要还是参照和使用了IntroduceToTextKit这个Demo。
最后看看程序运行结果:
调整字体前:
TextKit学习(一)可以调整字体大小的FontResizeDemo_第3张图片

调整Text Size到最大:
TextKit学习(一)可以调整字体大小的FontResizeDemo_第4张图片
调整后的文本:
TextKit学习(一)可以调整字体大小的FontResizeDemo_第5张图片 

你可能感兴趣的:(TextKit学习(一)可以调整字体大小的FontResizeDemo)