我需要使用 Kern attribute 的 NSAttributedString
。正如我在文档中看到的,该属性的默认值为 0.0。但是我遇到了短语 Hello, world
的奇怪行为(对于短语“你好”一切正常):
NSDictionary<NSString*, id>* attributes = @{NSFontAttributeName: [UIFont systemFontOfSize:12]};
NSString* text = @"Hello, World";
NSAttributedString* string = [[NSAttributedString alloc] initWithString:text attributes:attributes];
CGSize size1 = [string size];
NSMutableDictionary<NSString*, id>* attributesWithKernel = [attributes mutableCopy];
attributesWithKernel[NSKernAttributeName] = @(0.0);
NSAttributedString* stringWithKern = [[NSAttributedString alloc] initWithString:text attributes:attributesWithKernel];
CGSize size2 = [stringWithKern size];
XCTAssertTrue(CGSizeEqualToSize(size1, size2)); //here test falls
//size1 = size1 = (width = 68.8125, height = 14.3203125)
//size2 = (width = 69.515625, height = 14.3203125)
为了使 size1 和 size2 相等,字距调整应该相等
-7.105427357601002e-15
,我知道这非常接近 0.0,但很奇怪,因为这几乎改变了一个像素的宽度。NSAttributedString
在 Objective-C
和 Swift
中具有相同的行为,例如 swift: let text = "Hello, World"
let attributes : [NSAttributedString.Key : Any] = [NSAttributedString.Key.font: UIFont.systemFont(ofSize: UIFont.systemFontSize)]
let str = NSAttributedString(string: text, attributes: attributes)
let size = str.size()
var attributesWithKern = attributes
attributesWithKern[NSAttributedString.Key.kern] = NSNumber(value: 0.0)
let strWithKern = NSAttributedString(string: text, attributes: attributesWithKern)
let sizeWithKern = strWithKern.size()
XCTAssertTrue(size == sizeWithKern)
我该如何解决这种行为?
P.S.
现在,如果键等于 0.0,我只是从属性字符串中删除
NSKernAttributeKey
,但我认为这不是一个好的解决方案。 最佳答案
我相信这里的文档是错误的,值得为此打开一个 radar。如果未设置任何值,则将其解释为“正常字距调整”。当设置为 0 时,它被解释为“禁用字距调整”,这就是为什么宽度会稍微宽一些(字距调整通常是稍微负数,带来字距对字符,如这种字体中的“W”和“o”,有点更近)。我认为没有任何方法可以在不删除属性的情况下明确请求“默认字距调整”。
为了您的目的,我相信您通过在为零时删除该值来做正确的事情,因为您需要默认字距调整,而不是禁用字距调整。
您的微小负值起作用的原因是因为它不是零,所以它不会禁用字距调整,但它是如此之小以至于行为非常非常接近默认值,并且您在中间计算中遇到了 Double 的精度(或者可能是 Float 的精度,取决于它是如何在内部实现的)。您应该发现您的测试通过任何小于(接近零)的值,而不仅仅是该值。例如,在我的测试中,阳性 7e-15 也有效。
关于ios - NSAttributedString 中字距调整的不一致行为,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/54948326/