将Swift中的属性绑定到NSTextFields

问题描述 投票:0回答:2

我正在研究用Swift编写的单元转换器,它将在适当的NSTextFields中自动显示更新的单元。对于此示例,如果用户在minutesField中输入分钟,则secondsFieldhoursField应自动更新以显示属性的转换值。下面是我的视图控制器中的代码示例:

import Cocoa

class ViewController: NSViewController {

    @IBOutlet weak var secondsField: NSTextField!
    @IBOutlet weak var minutesField: NSTextField!
    @IBOutlet weak var hoursField: NSTextField!

    var sec: Double = 1
    var min: Double = 1
    var hr: Double = 1

    let conv = [[1,     0.0166666666666667, 0.000277777777777778],
                [60,    1,                  0.0166666666666667],
                [3600,  60,                 1]]

    func updateTextFields() {
        secondsField.stringValue = "\(sec)"
        minutesField.stringValue = "\(min)"
        hoursField.stringValue = "\(hr)"
    }
}

extension ViewController: NSTextFieldDelegate {

    override func controlTextDidChange(obj: NSNotification) {

        let tag = obj.object?.tag() ?? 0
        let text = obj.object?.stringValue ?? "1"
        let value = Double(text) ?? 1.0

        switch tag {
        case 0:
            // base unit = seconds
            sec = value
            min = value * conv[0][1]
            hr = value * conv[0][2]
        case 1:
            // base unit = minutes
            sec = value * conv[1][0]
            min = value
            hr = value * conv[1][2]
        case 2:
            // base unit = hours
            sec = value * conv[2][0]
            min = value * conv[2][1]
            hr = value
        default:
            "none"
        }
        updateTextFields()
    }    
}

上面的代码更新了文本字段,但是在第一个键击之后输入冻结的活动字段。例如,如果您尝试在秒字段中输入60,则下一个键击不会执行任何操作。在第一次击键后出现6.0数字(见下图)。这可能是由于在更新属性后调用的函数。根据活动文本字段的标记更新属性。

是否可以使用一个函数来更新当前活动字段以外的所有其他文本字段?

enter image description here

swift cocoa cocoa-bindings nstextfield
2个回答
0
投票

是否可以使用一个函数来更新当前活动字段以外的所有其他文本字段?

是。您可以将所有文本字段连接到同一操作。

首先将所有文本字段相应地连接到视图控制器:

@IBOutlet weak var hourField: NSTextField!
@IBOutlet weak var minuteField: NSTextField!
@IBOutlet weak var secondField: NSTextField!

其次创建一个var来表示您的时间间隔。您还可以添加setter / getter以将其自动存储到USerDefaults:

var timeInterval: TimeInterval {
    get {
        return UserDefaults.standard.double(forKey: "timeInterval")
    }
    set {
        UserDefaults.standard.set(newValue, forKey: "timeInterval")
    }
}

第三,创建一个TimeInterval扩展,将时间从秒转换为小时/分钟

extension TimeInterval {
    var second: Double { return self }
    var minute: Double { return self / 60 }
    var hour:   Double { return self / 3600 }
    var hourToMinute:   Double { return self * 60 }
    var hourToSecond:   Double { return self * 3600 }
    var minuteToHour:   Double { return self / 60 }
    var minuteToSecond: Double { return self * 60 }
}

第四,创建将更新其他文本字段的操作:

@IBAction func timeAction(sender: NSTextField) {
    // use guard to convert the field string to Double
    guard let time = Double(sender.stringValue) else { return }
    // switch the field  
    switch sender {
    case hourField:
        // convert / update secondary fields
        minuteField.stringValue = time.hourToMinute.description
        secondField.stringValue = time.hourToSecond.description
        // make it persist through launches
        timeInterval = time * 3600
    case minuteField:
        hourField.stringValue = time.minuteToHour.description
        secondField.stringValue = time.minuteToSecond.description
        timeInterval = time * 60
    default:
        hourField.stringValue = time.hour.description
        minuteField.stringValue = time.minute.description
        timeInterval = time
    }
}

最后但并非最不重要的是确保您下次出现视图时加载值:

override func viewWillAppear() {
    hourField.stringValue = timeInterval.hour.description
    minuteField.stringValue = timeInterval.minute.description
    secondField.stringValue = timeInterval.second.description
}

Sample Project


0
投票

您需要将文本字段值的设置从viewDidLoad()移动到单独的方法。从viewDidLoad()和你的didSet财产上的seconds财产观察员调用该方法。

您还需要为更改秒,分钟或小时字段时创建操作方法。在seconds字段的action方法中,将seconds属性设置为secondsField.floatValue。对于其他两个领域也是如此。在NIB /故事板中,将文本字段的操作连接到视图控制器上的相应操作方法。

© www.soinside.com 2019 - 2024. All rights reserved.