一尘不染

UIColor返回深色模式颜色的错误值

swift

我有一个自定义UITextField子类,当在其中键入某些内容时,它会更改其边框颜色。我正在通过打电话监听变化

self.addTarget(self, action: #selector(textFieldDidChange(_:)), for: .editingChanged)

然后,textFieldDidChange(_:)我在做:

self.layer.borderColor = UIColor(named: "testColor")?.cgColor

testColor在Assets.xcassets中定义的颜色是哪里,带有亮和暗模式的变体。问题是UIColor(named: "testColor")?.cgColor似乎总是返回灯光模式的颜色。

这是iOS 13
Beta中的错误,还是我做错了什么?有一个GitHub存储库,其中包含展示此行为的代码。运行项目,从XCode切换到暗模式,然后开始在文本字段中键入内容。


阅读 352

收藏
2020-07-07

共1个答案

一尘不染

简短答案

在这种情况下,您需要指定使用哪个特征集合来解析动态颜色。

self.traitCollection.performAsCurrent {
    self.layer.borderColor = UIColor(named: "testColor")?.cgColor
}

要么

self.layer.borderColor = UIColor(named: "testColor")?.resolvedColor(with: self.traitCollection).cgColor

更长的答案

cgColordynamic上调用方法时UIColor,它需要解析动态颜色的值。可以通过参考当前特征集来完成UITraitCollection.current

当调用某些方法的重写时,UIKit将设置当前特征集合,尤其是:

  • UIView
    • 画()
    • layoutSubviews()
    • traitCollectionDidChange()
    • tintColorDidChange()
  • UIViewController
    • viewWillLayoutSubviews()
    • viewDidLayoutSubviews()
    • traitCollectionDidChange()
  • UIPresentationController
    • containerViewWillLayoutSubviews()
    • containerViewDidLayoutSubviews()
    • traitCollectionDidChange()

但是,在这些方法的替代范围之外,当前特征集不必设置为任何特定值。因此,如果您的代码未覆盖这些方法之一,而您想解析一种动态颜色,则您有责任告诉我们要使用的特征集。

(这是因为可以覆盖userInterfaceStyle任何视图或视图控制器的特征,因此即使将设备设置为亮模式,您也可能拥有处于暗模式的视图。)

您可以通过使用UIColor方法直接解析动态颜色来实现resolvedColor(with:)。或使用UITraitCollection方法performAsCurrent,然后将用于解析颜色的代码放入闭包中。上面的简短答案显示了两种方式。

您也可以将代码移入这些方法之一。在这种情况下,我认为您可以将其放入layoutSubviews()。如果执行此操作,则当明/暗样式更改时,它将自动被调用,因此您无需执行其他任何操作。

参考

WWDC 2019,在iOS中实现黑暗模式

从19:00开始,我讨论了动态色彩的解析方式,然后在23:30,我举了一个示例,说明如何CALayer像您一样将边框颜色设置为动态颜色。

2020-07-07