首页 > 代码库 > iOS开发--从TQRichTextViewDemo中学会分析工程
iOS开发--从TQRichTextViewDemo中学会分析工程
下载地址:
http://code4app.com/ios/TQRichTextView/5244fe9c6803fa0862000000
1.首先找到AppDelegate类,无论一个工程有多么复杂,多么多的类,但入口只有一个就是AppDelegate类的didFinishLaunchingWithOptions方法。我们阅读别人源代码的时候可以从这里入手。在TQRichTextViewDemo工程中,这个方法中仅创建了一个TQViewController并设置为window的rootViewController。
2.去看TQViewController的实现。首先也是从初始化方法开始,没有initWithNib,那么就看viewDidLoad。在这个方法中,只创建了一个TQRichTextView,设置了frame,text,backgroundColor,delegate等属性,然后添加到了self.view上。看完viewDidLoad方法,发现这是系统调用的最后一个方法,那么结合模拟器的运行效果,我们可以猜测,文字的解析和绘制过程被封装到了TQRichTextView中。
3.去看TQRichTextView。
- 浏览一下头文件,了解有哪几个属性和可调用的方法。
- 阅读initWithFrame方法,发现这里只是对几个属性的初始化。
- 寻找是否有重写的set,get方法。发现setText,setFont,setTextColor,setLineSpacing方法。阅读发现,每次更新完值后会调用setNeedsDisplay方法。说明,每次对这四个对象赋值后都会调用drawRect。
- 继续阅读.m文件,发现drawRect方法,这个也是有系统调用的,那么绘制的工作应该是在这里完成。阅读drawRect方法。
- 在drawRect中首先调用了analyzeText方法,传入了参数_text,并把返回值赋给了_textAnalyzed。
- 去看analyzeText方法。首先清空了数组和字典,然后定义了一个NSString类型的result变量和NSMutableArray类型的array并赋值为richTextRunsArray。
- 执行TQRichTextEmojiRun的一个方法,传入string和array的指针。
- 因为这个方法是TQRichTextEmojiRun的类方法,所有我们直接去看该方法,而不用去看TQRichTextEmojiRun的初始化方法。连蒙带猜的我们可以确定这个方法应该是解析字符串中表情的方法。看重点,发现如果解析到表情,那么就创建一个TQRichTextEmojiRun的队形,range=表情字符串在string中的range,originalText=表情字符串,把这个创建的对象放入传进来的数组指针中,最后替换传入字符串中的表情字符串为空格,并返回被替换过表情的字符串。
- 通过看TQRichTextEmojiRun的init方法,发现初始化时设置了两个变量,来自父类TQRichTextBaseRun,分别设置了TQRichTextRunType和是否响应触摸。发现TQRichTextURLRun也继承自TQRichTextBaseRun。
- 跳出这个方法,回到analyzeText中,继续往下看。
- 执行TQRichTextURLRun的一个方法,传入result和array的指针。
- 同上,直接去看这个方法的实现。从和TQRichTextEmojiRun的命名格式类似,调用的方法名类似,我们可以猜测这个方法是解析URL的,那么就直接看重点。发现,如果找到匹配的字符串,就创建一个TQRichTextURLRun的对象,设置range=URL的range,originText = URL,然后把对象添加到传进来的数组中,最后返回string。
- 通过看TQRichTextURLRun的init方法,发现同上初始化时设置了两个变量,来自父类TQRichTextBaseRun,分别设置了TQRichTextRunType和是否响应触摸。发现TQRichTextEmojiRun继承自TQRichTextImageRun,TQRichTextImageRun继承自TQRichTextBaseRun。
- 跳出这个方法,回到analyzeText中,继续往下看。
- 遍历richTextRunsArray中的对象,每个对象调用setOriginalFont方法,传入参数self.font。通过阅读上面的两个方法的内部实现明白richTextRunsArray中保存的是TQRichTextEmojiRun和TQRichTextURLRun对象,所以去看这两个对象的setOriginalFont方法。
- 这个设置的是继承自父类TQRichTextBaseRun的属性。
- 回到analyzeText继续往下看。
- 返回result,跳出这个方法,继续看drawRect。
- 接下来是创建NSAttrbutedString,并赋值一些属性。
- 文本处理,遍历richTextRunsArray中的对象,每个对象调用replaceTextWithAttributedString方法。
- 在TQRichTextEmojiRun没有找到replaceTextWithAttributedString方法,那就去TQRichTextImageRun中找,阅读该方法。
- 传入的attrString删除占位的空格字符。
- 创建CTRunDelegateCallbacks,返现设定的宽高为OriginalFont的高度的1.1倍。
- 创建空格NSAttrbutedString对象,添加CTRunDelegateRef属性。
- 把空格NSAttrbutedString对象插入至传入的attrString。
- 查看super方法,发现在这段range上还设置了一个键值对,key为TQRichTextAttribute,value为自己。
- 同上阅读TQRichTextURLRun的replaceTextWithAttributedString方法。
- 为该段range的文字添加蓝色字体。
- 调用super方法,为该段range添加同上的key-value键值对。
- 返回继续阅读drawRect。
- 在TQRichTextEmojiRun没有找到replaceTextWithAttributedString方法,那就去TQRichTextImageRun中找,阅读该方法。
- 下面是绘制的准备工作,不想研究咋绘制的粗读就行。
- 清空richTextRunRectDic
- 绘制,不想研究的也可粗读略过。
- 找重点,绘制替换过的特殊文本单元。
- 遍历取出每一个run,取出run的attributes中key为TQRichTextAttribute的值。
- 如果值存在,则说明,这个run是特殊的run。
- 求run的frame,粗读略过。
- 调用drawRunWithRect方法,并将返回值赋给idDraw变量。
- TQRichTextEmojiRun的drawRunWithRect方法中绘制了表情图片,返回YES。
- TQRichTextURLRun的drawRunWithRect方法返回NO。
- TQRichTextBaseRun的drawRunWithRect方法返回NO。
- 判断run的isResponseTouch属性,查找后发现是TQRichTextBaseRun的属性,注释说是是否响应触摸
- 如果响应触摸
- 在richTextRunRectDic中添加键值对,key为rect,value为run。
- 设置循环的条件,释放CF类型的变量。CF类型的变量不支持ARC。
- 在drawRect中首先调用了analyzeText方法,传入了参数_text,并把返回值赋给了_textAnalyzed。
- 继续寻找TQRichTextView中重载系统的方法。返现touchesBegan和touchesEnd方法。阅读这两个方法。
- 拿到触摸的点,转换点的坐标为以左下角为原点是的坐标。
- 判断是否能只想delegate
- 遍历richTextRunRectDic,判断点是否在rect中。如果在则调用delegate方法。
- 主要功能分析完毕。
- 通读TQRichTextBaseRun,了解其它属性和一些属性的默认值。
- 至此,整个TQRichTextViewDemo基本分析完毕。可得出如下结论。
- 类的结构关系
- TQRichTextEmojiRun—> TQRichTextImageRun—> TQRichTextBaseRun。
- TQRichTextURLRun—> TQRichTextBaseRun。
- TQRichTextEmojiRun和TQRichTextURLRun分别实现了analyzeText:runsArray方法用来从字符串中取出所需要的文字,并创建TQRichTextBaseRun对象添加进数组保存。
- TQRichTextEmojiRun和TQRichTextURLRun分别重载了replaceTextWithAttributedString方法来在字符串的特定range处添加Attributed属性。
- TQRichTextEmojiRun和TQRichTextURLRun分别重载了drawRunWithRect方法实现了自定义位置,并返回是否绘制了内容。
- 流程
- 给TQRichTextView更改属性。
- 调用drawRect。
- 调用TQRichTextEmojiRun和TQRichTextURLRun的analyzeText:runsArray方法解析字符串。
- 创建NSAttributedString。
- 为NSAttributedString添加属性。
- 调用TQRichTextEmojiRun和TQRichTextURLRun的replaceTextWithAttributedString方法为字符串添加属性。
- CoreText绘制NSAttributedString。
- 调用TQRichTextEmojiRun和TQRichTextURLRun的drawRunWithRect方法实现自定义绘制。
- 保存run和rect进字典,从字典取值判断是否能点击。
- 有以上可以得出结论,若要更换表情和文字的解析规则,则只需去TQRichTextEmojiRun或TQRichTextURLRun修改analyzeText:runsArray,replaceTextWithAttributedString,drawRunWithRect三个方法。
- analyzeText:runsArray修改解析规则。
- replaceTextWithAttributedString修改添加属性。
- drawRunWithRect修改自定义绘制。
- 若要实现点击,只需设置isResponseTouch为YES并实现TQRichTextViewDelegate。
- 类的结构关系
- 由以上阅读代码发现如下缺陷
- TQRichTextView的awakeFromNib方法没实现,如过使用xib拖拽,则无法为TQRichTextView添加默认属性。
- 解决方法,添加awakeFromNib方法,并在其中为默认属性赋值。
- sizeWithFont和boundingRectWithSize:options:attributes:context:方法计算时不会计算行间距,TQRichTextView的lineSpacing若不设置为0会导致两方法计算的高度比TQRichTextView实际显示需要的高度小。
- 解决方法,TQRichTextView的lineSpacing设置为0。
- TQRichTextView的awakeFromNib方法没实现,如过使用xib拖拽,则无法为TQRichTextView添加默认属性。
iOS开发--从TQRichTextViewDemo中学会分析工程
声明:以上内容来自用户投稿及互联网公开渠道收集整理发布,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任,若内容有误或涉及侵权可进行投诉: 投诉/举报 工作人员会在5个工作日内联系你,一经查实,本站将立刻删除涉嫌侵权内容。