html - 如何从 iOS UILabel 对象中的 NSAttributedString(从 HTML 转换)中消除多余的换行符?
问题描述
我有一些表格视图单元格需要从我们的服务器显示一些简单的 html。我将 html 转换为 NSAttributedString 以便它可以显示在单元格上的 UILabel 对象中。但是当表格视图绘制单元格时,UILabel 对象似乎在它们的末尾添加了一些换行符。注意这里的额外空间:
(顺便说一句,图片提供了两个表格视图单元格,因为我尝试了两种不同形式的 html,并且在这两种情况下似乎都插入了换行符)我想也许是由于我的布局,也许 UILabel 被强制进入它的高度布局,并且不能根据分配的文本自由调整大小/缩小自身。但是,当我向同一个标签提供一个简单的 NSAttributedString 时,会以这种方式创建:
sRet = [[NSAttributedString alloc] initWithString:@"[Dummy text string 111]"];
(黄色) UILabel 确实会像这样缩小响应:
几乎证明了我的布局允许 UILabel 根据分配给它的文本自由调整大小。
下面是分配给标签的 html 以及当我将它们记录到控制台时生成的 NSAttributedString 值。这些对应于您在上面第一张图片中的黄色 UILabel 对象中看到的内容。这是被转换为属性字符串并分配的 html:
<h2>Student did poorly</h2>
<p><span style="font-family:comic sans ms,cursive;"><span style="font-size:14px;">Student did ok</span></span></p>
以下是相应的属性字符串:
Student did poorly
{
NSColor = "kCGColorSpaceModelRGB 0 0 0 1 ";
NSFont = "<UICTFont: 0x7fce3f8798e0> font-family: \"TimesNewRomanPS-BoldMT\"; font-weight: bold; font-style: normal; font-size: 18.00pt";
NSKern = 0;
NSParagraphStyle = "Alignment 4, LineSpacing 0, ParagraphSpacing 0, ParagraphSpacingBefore 0, HeadIndent 0, TailIndent 0, FirstLineHeadIndent 0, LineHeight 22/0, LineHeightMultiple 0, LineBreakMode 0, Tabs (\n), DefaultTabInterval 36, Blocks (null), Lists (null), BaseWritingDirection 0, HyphenationFactor 0, TighteningForTruncation NO, HeaderLevel 2";
NSStrokeColor = "kCGColorSpaceModelRGB 0 0 0 1 ";
NSStrokeWidth = 0;
}
Student did ok{
NSColor = "kCGColorSpaceModelRGB 0 0 0 1 ";
NSFont = "<UICTFont: 0x7fce3d5a96c0> font-family: \"Snell Roundhand\"; font-weight: normal; font-style: normal; font-size: 14.00pt";
NSKern = 0;
NSParagraphStyle = "Alignment 4, LineSpacing 0, ParagraphSpacing 0, ParagraphSpacingBefore 0, HeadIndent 0, TailIndent 0, FirstLineHeadIndent 0, LineHeight 19/0, LineHeightMultiple 0, LineBreakMode 0, Tabs (\n), DefaultTabInterval 36, Blocks (\n), Lists (\n), BaseWritingDirection 0, HyphenationFactor 0, TighteningForTruncation NO, HeaderLevel 0";
NSStrokeColor = "kCGColorSpaceModelRGB 0 0 0 1 ";
NSStrokeWidth = 0;
}
{
NSColor = "kCGColorSpaceModelRGB 0 0 0 1 ";
NSFont = "<UICTFont: 0x7fce3d7959e0> font-family: \"Times New Roman\"; font-weight: normal; font-style: normal; font-size: 12.00pt";
NSKern = 0;
NSParagraphStyle = "Alignment 4, LineSpacing 0, ParagraphSpacing 0, ParagraphSpacingBefore 0, HeadIndent 0, TailIndent 0, FirstLineHeadIndent 0, LineHeight 15/0, LineHeightMultiple 0, LineBreakMode 0, Tabs (\n), DefaultTabInterval 36, Blocks (\n), Lists (\n), BaseWritingDirection 0, HyphenationFactor 0, TighteningForTruncation NO, HeaderLevel 0";
NSStrokeColor = "kCGColorSpaceModelRGB 0 0 0 1 ";
NSStrokeWidth = 0;
}
在 html 和属性字符串中,我都没有看到 UILabel 对象具有额外垂直空间的原因。Fwiw,每个单元格上的一对 UILabel 对象(黄色和白色)都包含在 UIStackView 中,所以也许它可能会以某种方式做恶作剧。但是只有四个约束,尾随、前导、底部和顶部,并且如上所述,标签可以很好地调整大小,因此这些约束似乎对此没有影响。
解决方案
我遇到了同样的问题,需要删除我的<p>
标签添加的不需要的换行符,并且无法在服务器端进行更改。所以我首先提出了这个解决方案:
迅速
while !attributedString.string.isEmpty //Validates if the attributed string has at least one character
&& CharacterSet.newlines.contains(attributedString.string.unicodeScalars.last!) //Compares if the last unicode character of the attributed string is inside the `CharacterSet` of newlines
{
attributedString.deleteCharacters(in: NSRange(location: attributedString.length - 1, length: 1)) //Deletes the last character of the attributed string
}
Obj-C(未测试)
while ([attributedString.string length] > 0
&& [[attributedString.string substringFromIndex:[attributedString.string length] - 1] rangeOfCharacterFromSet:NSCharacterSet.newlineCharacterSet].location != NSNotFound)
{
[attributedString deleteCharactersInRange:NSMakeRange([attributedString length] - 1, 1)];
}
(我在将 html 代码转换为属性字符串后立即执行此代码。)
但我可以通过答案看到<h#>
标签也添加了换行符。因此,对于扩展解决方案,可能类似于为每对标签创建一个属性字符串;应用它们上面的过滤器;之后,将它们加入一个属性字符串或显示在不同的标签中。
编辑
我使用扩展String
名将 HTML 代码转换为NSMutableAttributedString
:
extension String {
/// Returns a new attributed string loading any HTML tags that the receiver contains.
///
/// If the HTML code is malformed or can not format it, this method returns the receiver but as a `NSMutableAttributedString`. If the formated resulting string contains one or more newlines at the end, they are removed.
func htmlFormat() -> NSMutableAttributedString {
var attributedString = NSMutableAttributedString(string: self, attributes: [.font : UIFont.preferredFont(forTextStyle: .body)])
if let data = data(using: .utf8),
let formatedString = try? NSMutableAttributedString(data: data,
options: [.documentType: NSAttributedString.DocumentType.html,
.characterEncoding: String.Encoding.utf8.rawValue],
documentAttributes: nil) {
attributedString = formatedString
}
while !attributedString.string.isEmpty
&& CharacterSet.newlines.contains(attributedString.string.unicodeScalars.last!) {
attributedString.deleteCharacters(in: NSRange(location: attributedString.length - 1, length: 1))
}
return attributedString
}
}
推荐阅读
- python - 如何在帧之间添加延迟以及如何在全屏模式下保存 gif 文件?
- mongodb - 如何在特殊条件下在 MongoDb 中多次插入文档
- python - 如何使用 python flask 将修改后的图像直接上传到 s3 存储桶
- android - RxJava - 将 delaySubscription 与 withLatestFrom 一起使用
- bootstrap-4 - 应用引导程序时如何将两个表单字段放在一行中?
- python - 我可以有 Flask-SQLAlchemy 的数组枚举列吗?
- javascript - JavaScript 数组动态过滤
- python - 如何使用postgresql在链表python中显示数据库表中的所有值
- python - 如何从 csv 读取块数据并添加列,然后使用 pandas 写入同一文件中的同一行
- java - 如何使用 Java 8 Streams 将列表中的对象与地图中的数据匹配并保存到另一个地图