【问题标题】:App Crashes Due to Binding to Table Cell View由于绑定到表格单元格视图,应用程序崩溃
【发布时间】:2020-04-10 06:25:12
【问题描述】:

所以我创建了一个NSOutlineView 以分层方式显示文件和目录列表。我正在构建一个 BitTorrent 客户端(说明类名是有意义的)。

如您所见,大纲视图的外观大致如下:

问题与 名称 列有关。在名称列中,对于每一行,我都有一个复选框和一个并排的文本字段。这将帮助您获得更清晰的想法:

现在,我使用绑定来获取每个文本字段的值。但是,由于有 2 个视图(复选框和文本字段)需要绑定到同一个 NSTableCellView,因此我从数据源返回一个结构,其中包含 2 个值:文本字段的字符串(保存文件/目录名称),以及一个用于启用/禁用复选框的布尔值。

为了处理大纲视图(尤其是其数据),我将其类设置为TorrentContent,其定义如下:

import Cocoa

struct Name {
    let value: String
    let enabled: Bool
}

class TorrentContent: NSOutlineView, NSOutlineViewDelegate, NSOutlineViewDataSource {
    var content: [TorrentContentItem]

    required init?(coder: NSCoder) {
        let srcDir = TorrentContentItem("src")

        let mainJava = TorrentContentItem("main.java")
        let mainCpp = TorrentContentItem("main.cpp")

        srcDir.children.append(mainJava)
        srcDir.children.append(mainCpp)

        content = [srcDir]

        super.init(coder: coder)

        delegate = self
        dataSource = self
    }

    func outlineView(_: NSOutlineView, isItemExpandable item: Any) -> Bool {
        if let _item = item as? TorrentContentItem {
            if _item.children.count > 0 {
                return true
            } else {
                return false
            }
        } else {
            return false
        }
    }

    func outlineView(_: NSOutlineView, numberOfChildrenOfItem item: Any?) -> Int {
        if item == nil {
            return content.count
        } else {
            if let _item = item as? TorrentContentItem {
                return _item.children.count
            }
        }

        return 0
    }

    func outlineView(_: NSOutlineView, child: Int, ofItem item: Any?) -> Any {
        if item != nil {
            if let _item = item as? TorrentContentItem {
                return _item.children[child]
            }
        }

        return content[child]
    }

    func outlineView(_: NSOutlineView, objectValueFor col: NSTableColumn?, byItem item: Any?) -> Any? {
        if item != nil, col != nil {
            if let _item = item as? TorrentContentItem {
                switch col!.title {
                case "Name":
                    return Name(value: _item.name, enabled: false)
                default:
                    return nil
                }
            }
        }

        return nil
    }
}

我已经对数据进行了硬编码,以便您更容易理解发生了什么。

只关注名称列,这是上面代码中处理该列的部分:

func outlineView(_: NSOutlineView, objectValueFor col: NSTableColumn?, byItem item: Any?) -> Any? {
    if item != nil, col != nil {
        if let _item = item as? TorrentContentItem {
            switch col!.title {
            case "Name":
                return Name(value: _item.name, enabled: false)
            default:
                return nil
            }
        }
    }

    return nil
}

如您所见,它返回Name 结构,其中包含两个视图的值。我已将 enabled 值硬编码为 false,仅用于测试目的。

现在要将其绑定到文本字段的 value 属性,我已经这样做了:

我的逻辑是,由于 objectValueName 结构的一个实例,objectValue.value 应该是Name struct 的实例,是一个字符串。

我想以类似的方式绑定复选框的 enabled 属性。但是,任何绑定都不起作用。它们导致应用程序崩溃。这是每次我尝试在运行时查看大纲视图时 XCode 在崩溃后向我显示的内容:

在控制台中只有“(lldb)”。

我做错了什么,如何实现我想要的?即从数据源类中设置多个视图的属性值。

【问题讨论】:

    标签: swift xcode macos xcode11 appkit


    【解决方案1】:

    Cocoa Bindings 使用 Key Value Observing (KVO) 并且被观察的对象必须是 KVO 兼容的。见Using Key-Value Observing in Swift

    您只能对从 NSObject 继承的类使用键值观察。

    使用@objc 属性和动态修饰符标记您要通过键值观察来观察的属性。

    解决方案 A:从 outlineView(_:objectValueFor:byItem:) 返回一个 KVO 兼容对象

    解决方案 B:不要使用 Cocoa 绑定。创建NSTableCellView 的子类并添加enabledCheckbox 出口。设置outlineView(_:viewFor:item:)中的值。

    【讨论】:

      猜你喜欢
      • 2011-02-10
      • 2011-10-16
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多