更新:
我创建了一个非常简单的独立项目来演示这个bug。如果有人想拉同一个,看看他们是否能找到我出错的地方,我一定很感激。没有太多代码需要查看。公共回购: https://github.com/reidnez/NSAttributedStringBugDemo
我在这里遇到一个非常奇怪的问题:我有一个桌面视图。每个单元格都有一个包含1-3个单词的标题标签,以及一个包含多个CSV关键字的关键字标签。我也有一个搜索栏。要求是当用户键入搜索栏时,每个单元格的标题和关键字的任何部分匹配都会突出显示。屏幕截图:
第一张图片是A-Okay。在第二张图片中," an"应突出显示标题标签。但是,正如你所看到的,不是那么多......
这对"关键字"完全正常。标签,如上所示。这两个标签的属性字符串都是由我写的类别(下面的代码)创建的。在两个字符串上调用相同的方法,并且看起来与调试器告诉我的行为相同。用户界面讲述了另一个故事。
我已多次遍历调试器,并且在所有情况下,已正确配置属性字符串 。我还验证了其他内容没有调用[tableView reloadData]
并且我的代码中没有其他地方覆盖标签的值。这就是" an"的匹配方式。因为" Fang"在cellForRowAtIndexPath
:
(lldb) po customCell.entryTitleLabel.attributedText
F{
}an{
NSBackgroundColor = "UIDeviceRGBColorSpace 0.533333 0.835294 0.156863 1";
}g{
}
对我来说很好......这正是我想要的。但是当细胞渲染时,没有任何亮点可见!更奇怪的是,作为一项实验,我尝试将标签设置为我在cellForRow中创建的完全任意的attributesString:
NSMutableAttributedString *fake = [[NSMutableAttributedString alloc] initWithString:@"Fang"];
[fake addAttribute:NSBackgroundColorAttributeName value:MATCH_TEXT_HILIGHT_COLOR range:NSMakeRange(1, 2)];
customCell.entryTitleLabel.attributedText = fake;
这也失败了。根本没有突出显示...但我可以突出显示{0,1}到{0,fake.length}范围内的任何子字符串,并且它的行为与预期一致。 同样,它似乎拒绝突出显示不在索引0处开始的任何子字符串 - 但仅限于标题标签。
我是否会失去理智?我错过了什么?
下面是我的类别......但我相信这个问题并不存在,因为它对关键字字符串起到了很好的作用,并且(再次)属性似乎在单元格返回之前正确设置:
-(void)hilightMatchingSubstring:(NSString*)substring color:(UIColor*)hilightColor range:(NSRange)range
{
if ([self.string compare:substring options:NSCaseInsensitiveSearch] == NSOrderedSame) {
[self addAttribute:NSBackgroundColorAttributeName value:hilightColor range:NSMakeRange(0, self.length)];
return;
}
// Sanity check. Make sure a valid range has been passed so that we don't get out-of-bounds crashes. Default to return self wrapped in an attributed string with no attributes.
NSRange selfRange = NSMakeRange(0, self.length);
if (NSIntersectionRange(selfRange, range).length == 0) {
NSLog(@" \n\n\n*** Match range {%lu, %lu} does not intersect main string's range {%lu, %lu}. Aborting *** \n\n\n", (unsigned long)range.location, (unsigned long)range.length, (unsigned long)selfRange.location, (unsigned long)selfRange.length);
return;
}
if (substring.length > 0) {
NSRange movingRange = NSMakeRange(range.location, substring.length);
if (NSMaxRange(movingRange) > self.length) {
return;
}
NSString *movingString = [self.string substringWithRange:movingRange];
while (NSMaxRange(movingRange) < NSMaxRange(range)) {
if ([movingString compare:substring options:NSCaseInsensitiveSearch] == NSOrderedSame) {
[self addAttribute:NSBackgroundColorAttributeName value:hilightColor range:movingRange];
}
movingRange = NSMakeRange(movingRange.location + 1, substring.length);
movingString = [self.string substringWithRange:movingRange];
}
} // This is fine...string leaves properly attributed.
}
答案 0 :(得分:2)
感谢你写这篇文章......我以为我也疯了!
我想出了一个解决方法(阅读:hack),同时我们等待来自Apple的官方消息。
NSDictionary *hackAttribute = [NSDictionary dictionaryWithObjectsAndKeys:
[UIColor clearColor], NSBackgroundColorAttributeName, nil];
NSMutableAttributedString *attributedText =
[[NSMutableAttributedString alloc] initWithString:@"some text..."];
[attributedAddressText setAttributes:hackAttribute range:NSMakeRange(0, attributedText.length)];
// Then set your other attributes as per normal
希望有所帮助。