iOS 图文混排 链接 可点击
对于这个话题 我想到
1 第一个解决方法就是使用 webView 比较经典
把所有复杂工作都交给控件本身去处理了, 但是好像好多需要自定义的地方 没法从 webView获得响应回调 :(估计也可以实现 也比较复杂,而且 这个需要对 html编码进行分析理解剥离等)
2 富文本方式 核心框架 coretext
图文混排 一点问题都没有 关键是怎么对 目标图片 或者链接 进行触发响应
要点:
(1)首先要封装的要相对独立 拓展也方便 首当其冲就是 和服务端约定的 数据模型 CoreTextModel 要敲定:
我们约定是 一段文字中间用 "[1]" 做标记 [我是图片id] ,同时传一个图片 和相应描述文字 "图片数组" 图片数组元素中主键就是这个图片id . 使用正则 去组织 CoreTextModel 数组集合
我们开发过程中是默认一张图片 点击这个图 展示解释 所以 直接知道这张图片大小 进行绘制, 如果是网络图片 那么就应该预留这张图片的大小 因为 绘制如果 不知道大小 线性绘制过程 就会卡在这 如果预知大小 那么就会继续往下绘制, 变成异步的
所以图片大小(宽高) 当然 也是这个图片数组元素 作为数据模型中的.属性元素而存在
如果是链接呢?? 所以 [1] 也可以认为是 [我是链接的id] 然后模型中有个参数 区分 当前 需要插入的是图片 还是 链接 或者其他需求 然后进行枚举判断
那么有几种枚举 就要有几个数据结构 比如这里 就有coreTextImageData && coreTextLinkData 和 绘制的过程 就需要 有几种可变数组 枚举的过程 分别进行add
(2)其次 绘制
首先来说 绘制 就是 直接动用底层CoreText
绘制图片\链接 都是动用底层 我现在的能力 是 调用\理解\修改\拓展 还不能自如的 写任意 自己想写东西 我想 也许当我到了这种程度 才能成为 大神吧 哈哈 路漫漫修远兮 吾将上下求索
1,遍历数据模型 CoreTextModel 数组 对 文本 夹杂"[1]"的重新组织
2,最终得到的富文本NSMutableAttributedString *result 都会通过 result.length 知道当前的图片/链接coreTextImageData/ coreTextLinkData的起始位置.
3,绘制 draw
a,绘制图片 (1)CTRunDelegateCallbacks callbacks;创建内存存储空间(怎么使用 百度 一大堆) (2) 使用0xFFFC作为空白的占位符 拼入 add 在result里
b,绘制链接 或者是一段连续子串 触发 给他指定颜色 下划线 大小等 同样 也add在result里
c,最终得到的 result 通过 创建CTFramesetterRef实例 获得 获得要缓制的区域的高度 再将生成好的CTFrameRef实例
ctFrame 和计算好的缓制高度height保存到CoreTextData实例中,最后返回CoreTextData实例(CoreTextData 这个结构体里面至少有ctFrame,height,result, coreTextImageData , coreTextLinkData )
ctFrame 这个 相当于 OC中 的 CGFrame , 但是成镜像且上下颠倒
盗个图:
CGFrame

Core Text CTFrameRef

所以如果触发点击事件 那么 还需要把这个绘制坐标 转成UI坐标
4,那么 现在 得到的result
在指定点击的UIView 要使用iPhone重绘机制drawRect 触发方法[self setNeedsDisplay] 调用- (void)drawRect:(CGRect)rect
就可展示出 这个视图了.
- (void)drawRect:(CGRect)rect
{
[super drawRect:rect];
if (self.data == nil) {//self.data 是 CoreTextData
return;
}
CGContextRef context = UIGraphicsGetCurrentContext();
CGContextSetTextMatrix(context, CGAffineTransformIdentity);
CGContextTranslateCTM(context, , self.bounds.size.height);
CGContextScaleCTM(context, 1.0, -1.0);
CTFrameDraw(self.data.ctFrame, context);
//对所有指定位置的图片填充
for (CoreTextImageData * imageData in self.data.imageArray) {
UIImage *image = [UIImage imageNamed:imageData.name];
if (image) {
CGContextDrawImage(context, imageData.imagePosition, image.CGImage);
}
}
}
(4)触发点击事件
原理: 触发点击UIView 通过 UIGestureRecognizer * tapRecognizer.
点击事件中 可以获得 CGPoint point = [recognizer locationInView:self]; 点击焦点 . 那么 判断 这个 点击焦点 是不是 图 或者 是不是可触发的链接 或者字符串 就可以了
a,遍历 self.data.imageArray 判断 是否存在满足条件 如果满足 则 用 通知post 去触发点击响应

我使用了一个参数 focusRect 把焦点区域放大 一圈 防止 图片 太小点击 不上 这个是我开发过程中 需要也是必要的 大家用不用 看心情
链接 就举一反三吧 网上讲解也一大堆 目的达到了
好的链接参考
下一篇 接着讲 触发弹出 一个对话框的逻辑
iOS 图文混排 链接 可点击的更多相关文章
- Unity插件之NGUI学习(5)—— 创建Label图文混排及文字点击
创建一个新的Scene,并按 Unity插件之NGUI学习(2)创建UI Root. 准备工作,制作Font.如今Project窗体创建一个Font目录.然后从系统自带字体目录中选择自己须要的字体,我 ...
- ios图文混排
图文混排的形式 1. 富文本形式 2. core Text(文字排版) 3. TextKit 4. UIWebView 一.富文本 我们可以采用attributeString来进行图文混排.例如一个文 ...
- iOS 图文混排
使用系统自带的NSAttributedString来处理,对于一般的图文混排已经足够了,但是,有一个缺点就是NSAttributedString并不支持gif动画.实际上,使用gif动画还是挺卡的. ...
- TextView + Spanned实现图文混排以及图片点击交互
最近要实现图文混排的需求,webview过大,所以想到了用SpannableStringBuilder来实现. 不过参考了大量国内文章,大多数是教你如何实现图文混排,并没有提及图片点击交互的.有翻阅了 ...
- iOS图文混排的几种方式
最近优化升级了之前做的一个项目,现在这一期已接近尾声了,今天可以腾出些时间总结一下最近项目中用的比较多的图片文字混排显示的内容.现在遇到比较多的图文混排的基本有三种:一种是在标签中显示 价格符号+价格 ...
- [UGUI]图文混排(六):点击区域
点击区域可以分成两部分来分析: 0.Rect 搜索api:Rect和Rect.Rect,可以知道: 在GUI和GUILayout中,Rect的原点在左上角,向右为x轴正方向,向下为y轴正方向: 除此之 ...
- iOS 图文混排 (Swift版)
// 0> 图片附件 let attachment = NSTextAttachment() attachment.image = #imageLiteral(resourceName: &qu ...
- CoreText实现图文混排之文字环绕及点击算法
系列文章: CoreText实现图文混排:http://www.jianshu.com/p/6db3289fb05d CoreText实现图文混排之点击事件:http://www.jianshu.co ...
- 高性能图文混排框架,构架顺滑的iOS应用-b
About GallopGallop是一个功能强大.性能优秀的图文混排框架. Features主要用于解决以下需求: 滚动列表的性能优化.Gallop使用异步绘制.视图层级合并.观察mainRunlo ...
随机推荐
- html:HTML元素分类
参考博客:http://www.cnblogs.com/polk6/p/3185692.html#Menu3-Display HTML元素大题可分为内联(inline)元素和块(block)元素. 1 ...
- mysql 随机取数据
SELECT * FROM table WHERE id >= (SELECT FLOOR(RAND()*MAX(id)) FROM table ) ORDER BY idLIMIT 1; 这样 ...
- ASP.NET MVC深入浅出(被替换) 第一节: 结合EF的本地缓存属性来介绍【EF增删改操作】的几种形式 第三节: EF调用普通SQL语句的两类封装(ExecuteSqlCommand和SqlQuery ) 第四节: EF调用存储过程的通用写法和DBFirst模式子类调用的特有写法 第六节: EF高级属性(二) 之延迟加载、立即加载、显示加载(含导航属性) 第十节: EF的三种追踪
ASP.NET MVC深入浅出(被替换) 一. 谈情怀-ASP.NET体系 从事.Net开发以来,最先接触的Web开发框架是Asp.Net WebForm,该框架高度封装,为了隐藏Http的无状态 ...
- 高抛低吸T+0操作要领(目前行情短线炒作的必备技能)
最近的行情只能用操蛋来形容,但是危机中不乏机会.现在已经不是之前行情的思路,那着一个股票长线抱着,即使是好的牛股,也经不起目前行情的这 么折腾.所以,现在最适合的操作方式就是高抛低吸.今天低吸保不准明 ...
- Nginx日志配置与切割
访问日志主要记录客户端访问Nginx的每一个请求,格式可以自定义.通过访问日志,你可以得到用户地域来源.跳转来源.使用终端.某个URL访问量等相关信息. Nginx中访问日志相关指令主要有两条,一条是 ...
- vi编辑器命令大全
>> from zhuhaiqing.info
- Unity2D实现人物三连击
之前写过一个系列<HTML5 2D平台游戏开发>,在此过程中发现有很多知识点没有掌握,而且用纯JavaScript来开发一个游戏效率极低,因为调试与地图编辑都没有可视化的工具,开发起来费时 ...
- VMware网络连接 桥接、NAt、host-only模式
如果你想利用VMWare安装虚拟机,或想创建一个与网内其他机器相隔离的虚拟系统,进行特殊的调试工作.此时,对虚拟系统网络连接模式的选择就非常重要了.如果你选择的工作模式不正确,就无法实现上述目的,也就 ...
- 使用MongoDB 记录业务日志
最近公司有个需求,要对业务日志进行记录并根据日志排查问题,以前都是使用log4net之类的日志组件来记录到文件,这种方式已经不能满足业务的需要,因为日志文件会很大,即使进行分割后,查找也是很不方便,何 ...
- dynamic_cast, RTTI, 整理
主要是参考下图,了解内存布局,然后写个实例程序就差不多明白了,但是需要熟悉指针转换. 1) 只有多态类才有RTTI信息,dynamic_cast正是运用RTTI进行转换,属于运行时类型检查. 2) d ...