NSMutableAttributedString不会显示以非零索引开头的属性

时间:2014-09-20 22:10:06

标签: ios uitableview nsmutableattributedstring

更新:

我创建了一个非常简单的独立项目来演示这个bug。如果有人想拉同一个,看看他们是否能找到我出错的地方,我一定很感激。没有太多代码需要查看。公共回购: https://github.com/reidnez/NSAttributedStringBugDemo

我在这里遇到一个非常奇怪的问题:我有一个桌面视图。每个单元格都有一个包含1-3个单词的标题标签,以及一个包含多个CSV关键字的关键字标签。我也有一个搜索栏。要求是当用户键入搜索栏时,每个单元格的标题和关键字的任何部分匹配都会突出显示。屏幕截图:

Everything looks as it should searching on "Fa"

"an" highlights keyword, but not title

第一张图片是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.
}

1 个答案:

答案 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

希望有所帮助。