swift - Swift KVO 绑定未更新
问题描述
在我的 macOS 应用程序中,我想将文本与可视 NSDatePicker 结合起来。因此,我制作了一个自定义的 NSDatePicker,单击时会显示一个弹出窗口。
我的问题是,虽然在弹出窗口中选择某些内容时文本选择器会在视觉上更新,但绑定值(prop_dateTime)没有更新。
我尝试过的一些方法:
class TextualAndVisualDatePickerElement: NSDatePicker {
[...]
func showPopover(sender: NSDatePicker) {
//using Bond
self.reactive.objectValue.bidirectionalBind(to: visualDatePicker.reactive.objectValue)
//using KVO
visualDatePicker.bind(.value, to: self, withKeyPath: "dateValue", options: [:])
//using setValue
_ = self.reactive.objectValue.observeNext(with: {self.setValue($0, forKey: "dateValue")})
_ = visualDatePicker.reactive.objectValue.observeNext(with: {sender.setValue($0, forKey: "dateValue")})
//using willChange
_ = self.reactive.objectValue.observeNext(with: {visualDatePicker.willChangeValue(forKey: "dateValue"); visualDatePicker.dateValue = $0 as! Date; visualDatePicker.didChangeValue(forKey: "dateValue")})
_ = visualDatePicker.reactive.objectValue.observeNext(with: {self.willChangeValue(forKey: "dateValue"); self.dateValue = $0 as! Date; self.didChangeValue(forKey: "dateValue")})
}
[...]
//using keyPaths
override public class func keyPathsForValuesAffectingValue(forKey key: String) -> Set<String> {
if key == "dateValue" {
return Set(["visualDatePicker.dateValue"])
}
if key == "visualDatePicker.dateValue" {
return Set(["dateValue"])
}
return Set([])
}
}
class ViewController: NSViewController {
@objc dynamic var objectControllerContent : ObjectControllerContent = ObjectControllerContent()
@IBOutlet weak var label: NSTextField!
@IBAction func btnClicked(_ sender: Any) {
label.stringValue = "prop_dateTime is \(objectControllerContent.prop_dateTime)"
}
}
class ObjectControllerContent : NSObject {
@objc dynamic var prop_dateTime : Date?
}
解决方案
设置dateValue
或objectValue
aNSDatePicker
或 anyNSControl
不会触发绑定。绑定在 中触发sendAction(_:to:)
。
解决方案:
call sendAction(_:to:)
,动作和/或目标可以是nil
.
datePicker.dateValue = date
datePicker.sendAction(datePicker.action, to: datePicker.target)
或模仿绑定
datePicker.dateValue = date
guard
let bindingInfo = datePicker.infoForBinding(NSBindingName.value),
let observedObject = bindingInfo[NSBindingInfoKey.observedObject] as? NSObject,
let observedKeyPath = bindingInfo[NSBindingInfoKey.observedKeyPath] as? String else {
return
}
observedObject.setValue(self.dateValue, forKeyPath: observedKeyPath)
推荐阅读
- firebase - Firebase Passwordless auth 在本机反应中没有按预期工作
- ios - 无法在框架项目的 Objective-C 代码中导入 Swift 类
- jmeter - 如何在 Jmeter 分布式测试中从 CSV 文件中选择唯一值
- sql - BIRT 报告:Oracle 查询:如何获得一个 Personid 的单行输出
- asp.net-core - 当有未经授权的请求时,适当的事件处理程序可以更改响应
- data-structures - 是否可以将堆作为具有父引用的树?
- sql - 在 phpmyadmin 中创建 Sql 过程时遇到问题
- oracle-apex - Oracle Apex - 执行 DML 命令时此处不允许使用虚拟列
- tfs - 以编程方式在我的 ASP .net Core Web API 中获取 TFS 工作项详细信息?
- javascript - PHP 返回具有相似值的行 - 必须匹配第一个单词