比较或测试NSAttributed字符串的特定颜色属性的正确方法是什么?
作为一个例子,我想知道文本选择是否有红色文本。我已经尝试了几种方法,如下所示,但它们都没有产生匹配。我看到屏幕上的文本变为红色,并且记录该属性返回:UIDeviceRGBColorSpace 1 0 0 1
- (BOOL)isRedWithAttributes:(NSDictionary *)attributes
{
BOOL isRedWithAttributes = NO;
if (attributes != nil)
{
// if ( [attributes objectForKey:NSForegroundColorAttributeName] == [UIColor redColor] )
// if ( [attributes objectForKey:NSForegroundColorAttributeName] == @"UIDeviceRGBColorSpace 1 0 0 1" )
if ( [attributes objectForKey:NSForegroundColorAttributeName] == [UIColor colorWithRed:1 green:0 blue:0 alpha:1] )
{
isRedWithAttributes = YES;
}
else
{
isRedWithAttributes = NO;
}
}
return isRedWithAttributes;
}
以下是我传递测试属性的方法:
NSAttributedString *selectedString = [attributedString attributedSubstringFromRange:selectedRange];
[selectedString enumerateAttributesInRange:NSMakeRange(0, [selectedString length])
options:NSAttributedStringEnumerationLongestEffectiveRangeNotRequired
usingBlock:^(NSDictionary *attributes, NSRange range, BOOL *stop)
{
UIFont *fontFace = [attributes objectForKey:NSFontAttributeName];
BOOL isRedWithAttributes = [fontFace isRedWithAttributes:attributes];
if (isRedWithAttributes)
{
NSLog(@"detected red. set selected");
[self.redButton setSelected:YES];
}
else
{
NSLog(@"detected not red. do not set selected");
[self.redButton setSelected:NO];
}
}
我觉得这不重要,但为了完整起见,我的方法就是将文字设置为红色。
NSMutableAttributedString *attributedString = [[NSMutableAttributedString alloc] initWithAttributedString:self.synopsisTextView.attributedText];
[attributedString addAttribute:NSForegroundColorAttributeName value:[UIColor redColor] range:selectedRange];
self.synopsisTextView.attributedText = attributedString;
答案 0 :(得分:8)
Objective-C是C的严格超集。Objective-C对象存在于堆上并且跟踪via指针。这样的净效果是测试:
[attributes objectForKey:NSForegroundColorAttributeName] ==
[UIColor colorWithRed:1 green:0 blue:0 alpha:1]
测试身份而非平等。也就是说,不是测试两个对象是否具有相同的值,而是测试它们是否是相同的物理对象,位于内存中的同一地址。 C不允许运算符重载,因此==
运算符在Objective-C中的含义与在C中的含义完全相同。您将比较两个指针。他们碰巧指向Objective-C对象既不在这里也不在那里。
你可能想要:
[[attributes objectForKey:NSForegroundColorAttributeName] isEqual:
[UIColor colorWithRed:1 green:0 blue:0 alpha:1]]
允许UIColor
对象应用它认为会建立相等的任何测试。
唯一可能的警告是UIColor
只有在相对于相同颜色空间定义并具有相同系数时才认为颜色相等。假设你总是在RGBA中定义所有颜色,这些颜色应该没什么区别。
答案 1 :(得分:4)
尝试在属性dict中测试NSForegroundColorAttributeName
对象而不是NSFontAttributeName
,如下所示:
NSAttributedString *selectedString = [attributedString attributedSubstringFromRange:selectedRange];
[selectedString enumerateAttributesInRange:NSMakeRange(0, [selectedString length])
options:NSAttributedStringEnumerationLongestEffectiveRangeNotRequired
usingBlock:^(NSDictionary *attributes, NSRange range, BOOL *stop) {
UIColor *fgColor = [attributes objectForKey:NSForegroundColorAttributeName];
if ([fgColor isEqual:[UIColor redColor]]) {
NSLog(@"detected red. set selected");
[self.redButton setSelected:YES];
} else {
NSLog(@"detected not red. do not set selected");
[self.redButton setSelected:NO];
}
}
答案 2 :(得分:0)
您必须为比较创建-isEqualToColor:
方法。 UIColor只从NSObject继承-isEqual:
,如果它们的哈希值相等,则会看到两个UIColor实例相等。
这样的方法相当容易构建,只需获得与色彩空间无关的RGB + A值,如果所有这些值都相同,则颜色相等。
如果你很懒,你会使用CGColorEqualToColor
,但如果一个是灰色(2个组件)而另一个是RGBA颜色(4个组件)则不会报告相等。
这可能就是为什么Apple没有实现-isEqualToColor:
方法的原因,因为如果组件的数量应该起作用或者它是否只是RGB值,意见会大不相同。因此,我建议您找到自己的定义并相应地实施比较。
PS:在某些情况下你可能会发现isEqual有效,因为UIColor会保留一些标准颜色的缓存,例如如果您申请红色,那么您将获得相同的实例。相同的实例意味着相同的散列,因此“相等”。但正如我之前所说,依靠实施并不是一件好事。