知乎的ios app里面的图文混排是怎么实现的,是加载的html吗

 我来答
百度网友ed5c075e
2014-11-03 · TA获得超过6631个赞
知道小有建树答主
回答量:1172
采纳率:100%
帮助的人:1055万
展开全部

  iOS实现图文混排的两个方法

  如果你想自定义文本的布局,例如像QQ、微信这样的应用中使用表情,那你多半会用到CoreText,CoreText是iOS、OSX平台的文本处理低层的框架, 可以实现任意的文字编排,更多详细信息请戳官方文档,一般来说, 我们们用下面的代码来实现图文混排:


  text = [[NSMutableAttributedString alloc] initWithString:@""];

  NSAttributedString *txt1 = [[NSAttributedString alloc] initWithString:@"测试"];

  [text appendAttributedString:txt1];

  [txt1 release];

  CTRunDelegateCallbacks callback;

  callback.version = kCTRunDelegateVersion1; //必须指定,否则不会生效,没有回调产生。

  callback.dealloc = deallocCallback;

  callback.getAscent = getAscent;

  callback.getDescent = getDescent;

  callback.getWidth = getWidth;

  NSDictionary *imgAttr = [[NSDictionary dictionaryWithObjectsAndKeys:@100, @"width", nil] retain];

  CTRunDelegateRef delegate = CTRunDelegateCreate(&callback, imgAttr);

  NSDictionary *txtDelegate = [NSDictionary dictionaryWithObjectsAndKeys:(id)delegate, (NSString*)kCTRunDelegateAttributeName, @100, @"width", nil];

  NSAttributedString *imgField = [[[NSAttributedString alloc] initWithString:@" " attributes:txtDelegate] autorelease];

  [text appendAttributedString:imgField];

  [text appendAttributedString:[[[NSAttributedString alloc] initWithString: @"结束"] autorelease]];

  CGMutablePathRef pathRef = CGPathCreateMutable();

  CGPathAddRect(pathRef, NULL, CGRectMake(0, 0, self.frame.size.width, self.frame.size.height));

  framesetter = CTFramesetterCreateWithAttributedString((CFAttributedStringRef)text);

  ctFrame = CTFramesetterCreateFrame(framesetter, CFRangeMake(0, 0), pathRef, NULL);

  CFArrayRef lines = CTFrameGetLines(ctFrame);

  CGPoint origins[CFArrayGetCount(lines)];

  CTFrameGetLineOrigins(ctFrame, CFRangeMake(0, 0), origins);

  for (int i = 0; i < CFArrayGetCount(lines); i++) {

  CTLineRef line = CFArrayGetValueAtIndex(lines, i);

  CFArrayRef runs = CTLineGetGlyphRuns(line);

  for (int j = 0; j < CFArrayGetCount(runs); j++) {

  CTRunRef run = CFArrayGetValueAtIndex(runs, j);

  CGPoint lineOrigin = origins[i];

  NSDictionary *meta = (NSDictionary*)CTRunGetAttributes(run);

  if (meta && ([meta valueForKey:@"width"] != nil)) {

  imageLocation.y = lineOrigin.y;

  CGFloat offset = CTLineGetOffsetForStringIndex(line, CTRunGetStringRange(run).location, NULL);

  imageLocation.x = lineOrigin.x + offset + self.frame.origin.x;

  }

  }

  }

  CFRelease(pathRef);

  [self setNeedsDisplay];



  一直以来,我认为只有这种方法实现。好吧,其实我没有想过有没有其它实现方法的问题。直到有一天看类似效果的代码时惊奇的发现:怎么 没有CTRunDelegate? 于是就仔细想了一下这个问题,创建CTFrame的时候会指定一个path,通常这个path我会使用一个CGRect完事,然后在 有图片的地方使用CTRunDelegate处理一下,但其实完全可以使用CGMutablePath来画出一块不规则的文本路径,比如:这样,就可以在预定的位置画图片了,而不用会CTRunDelegate来特殊处理,这种方式比较适合图片位置固定的应用。


转自madongsheng

亚远景信息科技
2024-12-11 广告
上海亚远景信息科技有限公司是国内汽车行业咨询及评估领军机构之一,深耕于ASPICE、敏捷SPICE、ISO26262功能安全、ISO21434车辆网络安全领域,拥有20年以上的行业经验,专精于培训、咨询及评估服务,广受全球车厂及供应商赞誉,... 点击进入详情页
本回答由亚远景信息科技提供
推荐律师服务: 若未解决您的问题,请您详细描述您的问题,通过百度律临进行免费专业咨询

为你推荐:

下载百度知道APP,抢鲜体验
使用百度知道APP,立即抢鲜体验。你的手机镜头里或许有别人想知道的答案。
扫描二维码下载
×

类别

我们会通过消息、邮箱等方式尽快将举报结果通知您。

说明

0/200

提交
取消

辅 助

模 式