- objective-c - iOS 5 : Can you override UIAppearance customisations in specific classes?
- iphone - 如何将 CGFontRef 转换为 UIFont?
- ios - 以编程方式关闭标记的信息窗口 google maps iOS
- ios - Xcode 5 - 尝试验证存档时出现 "No application records were found"
我的应用程序从API中提取HTML,将其转换为NSAttributedString
(以便允许可点击的链接),并将其写入AutoLayout表中的一行。麻烦的是,每当我调用这种类型的单元格时,高度的计算都会错误,并且内容会被截断。我尝试了行高计算的不同实现,但都无法正常工作。
我如何准确,动态地计算这些行之一的高度,同时仍保持点击HTML链接的功能?
Example of undesired behavior
我的代码如下。
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
switch(indexPath.section) {
...
case kContent:
{
FlexibleTextViewTableViewCell* cell = (FlexibleTextViewTableViewCell*)[TableFactory getCellForIdentifier:@"content" cellClass:FlexibleTextViewTableViewCell.class forTable:tableView withStyle:UITableViewCellStyleDefault];
[self configureContentCellForIndexPath:cell atIndexPath:indexPath];
[cell.contentView setNeedsLayout];
[cell.contentView layoutIfNeeded];
cell.selectionStyle = UITableViewCellSelectionStyleNone;
cell.desc.font = [UIFont fontWithName:[StringFactory defaultFontType] size:14.0f];
return cell;
}
...
default:
return nil;
}
}
- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath {
UIFont *contentFont = [UIFont fontWithName:[StringFactory defaultFontType] size:14.0f];
switch(indexPath.section) {
...
case kContent:
return [self textViewHeightForAttributedText:[self convertHTMLtoAttributedString:myHTMLString] andFont:contentFont andWidth:self.tappableCell.width];
break;
...
default:
return 0.0f;
}
}
-(NSAttributedString*) convertHTMLtoAttributedString: (NSString *) html {
return [[NSAttributedString alloc] initWithData:[html dataUsingEncoding:NSUTF8StringEncoding]
options:@{NSDocumentTypeDocumentAttribute: NSHTMLTextDocumentType,
NSCharacterEncodingDocumentAttribute: @(NSUTF8StringEncoding)}
documentAttributes:nil
error:nil];
}
- (CGFloat)textViewHeightForAttributedText:(NSAttributedString*)text andFont:(UIFont *)font andWidth:(CGFloat)width {
NSMutableAttributedString *mutableText = [[NSMutableAttributedString alloc] initWithAttributedString:text];
[mutableText addAttribute:NSFontAttributeName value:font range:NSMakeRange(0, text.length)];
UITextView *calculationView = [[UITextView alloc] init];
[calculationView setAttributedText:mutableText];
CGSize size = [self text:mutableText.string sizeWithFont:font constrainedToSize:CGSizeMake(width,FLT_MAX)];
CGSize sizeThatFits = [calculationView sizeThatFits:CGSizeMake(width, FLT_MAX)];
return sizeThatFits.height;
}
最佳答案
在我正在使用的应用程序中,该应用程序从其他人编写的糟糕的API中提取了可怕的HTML字符串,并将HTML字符串转换为NSAttributedString
对象。我别无选择,只能使用这个糟糕的API。很伤心任何必须解析可怕的HTML字符串的人都知道我的痛苦。我使用Text Kit
。方法如下:
NSAttributedString
对象,使用自定义属性标记链接,使用NSTextAttachment
标记图像。我称其为富文本。 Text Kit
对象。即NSLayoutManager
,NSTextStorage
,NSTextContainer
。分配后将它们连接起来。 NSTextStorage
[NSTextStorage setAttributedString:]
对象。
[NSLayoutManager ensureLayoutForTextContainer:]
强制布局发生[NSLayoutManager usedRectForTextContainer:]
方法计算绘制富文本格式所需的框架。如果需要,添加填充或边距。 [tableView: heightForRowAtIndexPath:]
[NSLayoutManager drawGlyphsForGlyphRange:atPoint:]
绘制富文本。我在这里使用屏幕外绘画技术,因此结果是一个UIImage
对象。 UIImageView
渲染最终结果图像。或将结果图像对象传递给contents
中layer
对象的contentView
属性的UITableViewCell
属性的[tableView:cellForRowAtIndexPath:]
属性。 [NSLayoutManager glyphIndexForPoint:inTextContainer:fractionOfDistanceThroughGlyph]
和[NSAttributedString attribute:atIndex:effectiveRange:]
轻拍了链接或图像。 CGPoint location = [tap locationInView:self.tableView];
// tap is a tap gesture recognizer
NSIndexPath *indexPath = [self.tableView indexPathForRowAtPoint:location];
if (!indexPath) {
return;
}
CustomDataModel *post = [self getPostWithIndexPath:indexPath];
// CustomDataModel is a subclass of NSObject class.
UITableViewCell *cell = [self.tableView cellForRowAtIndexPath:indexPath];
location = [tap locationInView:cell.contentView];
// the rich text is drawn into a bitmap context and rendered with
// cell.contentView.layer.contents
// The `Text Kit` objects can be accessed with the model object.
NSUInteger index = [post.layoutManager
glyphIndexForPoint:location
inTextContainer:post.textContainer
fractionOfDistanceThroughGlyph:NULL];
CustomLinkAttribute *link = [post.content.richText
attribute:CustomLinkAttributeName
atIndex:index
effectiveRange:NULL];
// CustomLinkAttributeName is a string constant defined in other file
// CustomLinkAttribute is a subclass of NSObject class. The instance of
// this class contains information of a link
if (link) {
// handle tap on link
}
// same technique can be used to handle tap on image
[NSAttributedString initWithData:options:documentAttributes:error:]
更快,更可自定义。即使没有概要分析,我也可以告诉
Text Kit
方法更快。即使我必须自己解析html并构造属性字符串,它也非常快速且令人满意。
NSDocumentTypeDocumentAttribute
方法太慢,因此是 Not Acceptable 。使用
Text Kit
,我还可以创建复杂的布局,例如带有可变缩进的文本块,边框,任意深度的嵌套文本块等。但是它确实需要编写更多代码来构造
NSAttributedString
并控制布局过程。我不知道如何计算使用
NSDocumentTypeDocumentAttribute
创建的属性字符串的边界rect。我相信用
NSDocumentTypeDocumentAttribute
创建的属性字符串是由
Web Kit
而不是
Text Kit
处理的。因此,这并不意味着可变高度表 View 单元格。
NSDocumentTypeDocumentAttribute
,我认为您必须弄清楚布局过程是如何发生的。也许您可以设置一些断点,以查看哪个对象负责布局过程。然后,您可以查询该对象或使用另一种方法来模拟布局过程以获取布局信息。有些人使用即席单元格或
UITextView
对象来计算高度,我认为这不是一个好的解决方案。因为以这种方式,应用程序必须布局相同的文本块至少两次。不管您是否知道,在应用程序中的某个位置,某些对象都必须对文本进行布局,以便您可以获取布局信息(例如边界矩形)。由于您提到了
NSAttributedString
类,所以最好的解决方案是iOS 7之后的
Text Kit
。如果您的应用程序针对较早的iOS版本,则为
Core Text
。
Text Kit
,因为这样,对于从API中提取的每个html字符串,布局过程只会发生一次,并且
NSLayoutManager
对象会缓存布局信息(例如边界rect和每个字形的位置)。只要保留
Text Kit
对象,您就可以随时重用它们。当使用表格 View 呈现任意长度的文本时,这是非常有效的,因为文本仅布局一次,并且每次需要显示一个单元格时才绘制一次。我还建议使用
Text Kit
而不使用
UITextView
,如官方的Apple文档建议的那样。因为如果一个人想重用该
UITextView
附带的
Text Kit
对象,则必须缓存每个
UITextView
。像我一样将
Text Kit
对象附加到模型对象,并且仅当从API中提取新的html字符串时,才更新
NSTextStorage
并强制
NSLayoutManager
进行布局。如果表 View 的行数是固定的,则还可以使用固定位置的占位符模型对象列表,以避免重复分配和配置。并且因为
drawRect:
导致
Core Animation
创建了必须避免的无用的支持位图,所以请勿使用
UIView
和
drawRect:
。使用
CALayer
绘制技术或将文本绘制到位图上下文中。我使用后一种方法,因为可以在后台线程中使用
GCD
进行此操作,因此主线程可以自由响应用户的操作。我的应用程序中的结果确实令人满意,快速,排版很好,表格 View 的滚动非常流畅(60 fps),因为所有绘制过程都是在后台线程中使用
GCD
完成的。每个应用程序都需要使用表 View 绘制一些文本,应使用
Text Kit
。
关于ios - NSLayoutdString的AutoLayout行高计算错误,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27803453/
IO 设备如何知道属于它的内存中的值在memory mapped IO 中发生了变化? ? 例如,假设内存地址 0 专用于保存 VGA 设备的背景颜色。当我们更改 memory[0] 中的值时,VGA
我目前正在开发一个使用Facebook sdk登录(通过FBLoginView)的iOS应用。 一切正常,除了那些拥有较旧版本的facebook的人。 当他们按下“使用Facebook登录”按钮时,他
假设我有: this - is an - example - with some - dashesNSRange将使用`rangeOfString:@“-”拾取“-”的第一个实例,但是如果我只想要最后
Card.io SDK提供以下详细信息: 卡号,有效期,月份,年份,CVV和邮政编码。 如何从此SDK获取国家名称。 - (void)userDidProvideCreditCardInfo:(Car
iOS 应用程序如何从网络服务下载图片并在安装过程中将它们安装到用户的 iOS 设备上?可能吗? 最佳答案 您无法控制应用在用户设备上的安装,因此无法在安装过程中下载其他数据。 只需在安装后首次启动应
我曾经开发过一款企业版 iOS 产品,我们公司曾将其出售给大型企业,供他们的员工使用。 该应用程序通过 AppStore 提供,企业用户获得了公司特定的配置文件(包含应用程序配置文件)以启用他们有权使
我正在尝试将 Card.io SDK 集成到我的 iOS 应用程序中。我想为 CardIO ui 做一个简单的本地化,如更改取消按钮标题或“在此保留信用卡”提示文本。 我在 github 上找到了这个
我正在使用 CardIOView 和 CardIOViewDelegate 类,没有可以设置为 YES 的 BOOL 来扫描 collectCardholderName。我可以看到它在 CardIOP
我有一个集成了通话工具包的 voip 应用程序。每次我从我的 voip 应用程序调用时,都会在 native 电话应用程序中创建一个新的最近通话记录。我在 voip 应用程序中也有自定义联系人(电话应
iOS 应用程序如何知道应用程序打开时屏幕上是否已经有键盘?应用程序运行后,它可以接收键盘显示/隐藏通知。但是,如果应用程序在分屏模式下作为辅助应用程序打开,而主应用程序已经显示键盘,则辅助应用程序不
我在模拟器中收到以下错误: ImageIO: CGImageReadSessionGetCachedImageBlockData *** CGImageReadSessionGetCachedIm
如 Apple 文档所示,可以通过 EAAccessory Framework 与经过认证的配件(由 Apple 认证)进行通信。但是我有点困惑,因为一些帖子告诉我它也可以通过 CoreBluetoo
尽管现在的调试器已经很不错了,但有时找出应用程序中正在发生的事情的最好方法仍然是古老的 NSLog。当您连接到计算机时,这样做很容易; Xcode 会帮助弹出日志查看器面板,然后就可以了。当您不在办公
在我的 iOS 应用程序中,我定义了一些兴趣点。其中一些有一个 Kontakt.io 信标的名称,它绑定(bind)到一个特定的 PoI(我的意思是通常贴在信标标签上的名称)。现在我想在附近发现信标,
我正在为警报提示创建一个 trigger.io 插件。尝试从警报提示返回数据。这是我的代码: // Prompt + (void)show_prompt:(ForgeTask*)task{
您好,我是 Apple iOS 的新手。我阅读并搜索了很多关于推送通知的文章,但我没有发现任何关于 APNS 从 io4 到 ios 6 的新更新的信息。任何人都可以向我提供 APNS 如何在 ios
UITabBar 的高度似乎在 iOS 7 和 8/9/10/11 之间发生了变化。我发布这个问题是为了让其他人轻松找到答案。 那么:在 iPhone 和 iPad 上的 iOS 8/9/10/11
我想我可以针对不同的 iOS 版本使用不同的 Storyboard。 由于 UI 的差异,我将创建下一个 Storyboard: Main_iPhone.storyboard Main_iPad.st
我正在写一些东西,我将使用设备的 iTunes 库中的一部分音轨来覆盖 2 个视频的组合,例如: AVMutableComposition* mixComposition = [[AVMutableC
我创建了一个简单的 iOS 程序,可以顺利编译并在 iPad 模拟器上运行良好。当我告诉 XCode 4 使用我连接的 iPad 设备时,无法编译相同的程序。问题似乎是当我尝试使用附加的 iPad 时
我是一名优秀的程序员,十分优秀!