5

我制作了一个计算器应用程序(swift),并设置了一些 UIKeyCommands,以便如果用户有蓝牙键盘,他们可以在计算器中输入数字/符号。

它们的工作方式如下:

UIKeyCommand(input: "4", modifierFlags: [], action: "TypeFourInCalculator:")
func TypeFourInCalculator(sender: UIKeyCommand) {
    btn4Press(UIButton)
}

这一切都很好,当用户按下四键时,将四添加到计算器中(即使计算器本身没有文本字段)。但是:我也有几个标准文本字段,我希望 UIKeyCommands 在用户进入其中一个文本字段时停止(这样他们就可以再次使用 BT 键盘定期键入)。在不禁用 UIKeyCommands 的情况下,键入会导致计算器功能,并且不会在文本字段中输入。

所以我尝试了这个尝试在文本字段成为第一响应者时禁用 UIKeyCommands:

let globalKeyCommands = [UIKeyCommand(input: "4", modifierFlags: [], action: "TypeFourInCalculator:"), UIKeyCommand(input: "5", modifierFlags: [], action: "TypeFiveInCalculator:")]

override var keyCommands: [UIKeyCommand]? {
    if powertextfield.isFirstResponder() == false { // if the user isn't typing into that text field
        return globalKeyCommands // use key commands
    } else { // if the user is typing
        return nil // disable those UIKeyCommands
    }

这偶尔会起作用,但通常不起作用。如果用户还没有使用 BT 键盘输入任何内容(即我猜没有激活键盘命令),那么他们可以像往常一样使用 BT 键盘在文本字段中输入内容。但是,如果他们已经通过 UIKeyCommand 在计算器中输入了数字,他们就无法在文本字段中输入(嗯,有时它可以正常工作,有时它会像我添加该预防性代码之前那样失败)。键入的文本不会出现在该文本字段中,而是调用计算器命令。

那么当用户开始在普通文本字段中输入时,我能做些什么来禁用这些 UIKeyCommands 呢?

4

2 回答 2

3

keyCommands您可以使用s 的addKeyCommand(_:)removeKeyCommand(_:)方法,而不是创建计算属性UIViewController。像这样子类化您的文本字段:

class PowerTextField: UITextField {
    var enableKeyCommands: (Bool->())?

    override func becomeFirstResponder() -> Bool {
        super.becomeFirstResponder()
        enableKeyCommands?(false)
        return true
    }

    override func resignFirstResponder() -> Bool {
        super.resignFirstResponder()
        enableKeyCommands?(true)
        return true
    }
}

然后在你的UIViewController

override func viewDidLoad() {
    super.viewDidLoad()

    // globalKeyCommands is a view controller property

    // Add key commands to be on by default
    for command in globalKeyCommands {
        self.addKeyCommand(command)
    }

    // Configure text field to callback when 
    // it should enable key commands
    powerTextField.enableKeyCommands = { [weak self] enabled in
        guard let commands = self?.globalKeyCommands else {
            return
        }
        if enabled {
            for command in globalKeyCommands {
                self?.addKeyCommand(command)
            }
        } else {
            for command in globalKeyCommands {
                self?.removeKeyCommand(command)
            }
        }
    }
}

您可以为将采用UIViewController的委托协议设置一个委托协议,而不是由 配置的可选存储过程。UITextFieldUIViewController

此外,如果您需要在UIAlertController弹出 a 时停止这些键命令,您可以通过调用主视图控制器在其.UIAlertControllerviewWillAppear(_:)viewWillDisappear(_:)becomeFirstResponder()resignFirstResponder()enableKeyCommands(_:)viewDidLoad()

至于为什么会发生这种情况的解释,也许最有可能的解释是它是一个错误。我不确定为什么这在您的测试中是不规则的。我认为您可以尝试隔离它在什么条件下有效或无效。不清楚为什么这只会发生在蓝牙键盘上,但是当您开始引入无线技术时,可能会出现很多边缘情况。

于 2016-06-01T05:52:03.513 回答
2

所以我面临同样的问题,并找到了一个非常好的和简单的解决方案。

我想出了一个办法,如果当前的 firstResponder 是一个文本字段,或者类似的。

extension UIResponder {
    private weak static var _currentFirstResponder: UIResponder? = nil

    public static var isFirstResponderTextField: Bool {
        var isTextField = false
        if let firstResponder = UIResponder.currentFirstResponder {
            isTextField = firstResponder.isKind(of: UITextField.self) || firstResponder.isKind(of: UITextView.self) || firstResponder.isKind(of: UISearchBar.self)
        }

        return isTextField
    }

    public static var currentFirstResponder: UIResponder? {
        UIResponder._currentFirstResponder = nil
        UIApplication.shared.sendAction(#selector(findFirstResponder(sender:)), to: nil, from: nil, for: nil)
        return UIResponder._currentFirstResponder
    }

    @objc internal func findFirstResponder(sender: AnyObject) {
        UIResponder._currentFirstResponder = self
    }
}

然后简单地使用这个布尔值来确定在覆盖 keyCommands var 时要返回哪些 UIKeyCommands:

    override var keyCommands: [UIKeyCommand]? {
        var commands = [UIKeyCommand]()

        if !UIResponder.isFirstResponderTextField {
            commands.append(UIKeyCommand(title: "Some title", image: nil, action: #selector(someAction), input: "1", modifierFlags: [], propertyList: nil, alternates: [], discoverabilityTitle: "Some title", attributes: [], state: .on))

        }

        commands.append(UIKeyCommand(title: "Some other title", image: nil, action: #selector(someOtherAction), input: "2", modifierFlags: [.command], propertyList: nil, alternates: [], discoverabilityTitle: "Some title", attributes: [], state: .on))

        return commands
    }
于 2019-10-11T13:41:06.720 回答