UITableView中的UITextField不会成为第一响应者

时间:2017-06-22 19:27:46

标签: ios swift uitableview uitextfield first-responder

我有UITableView,大约有20行,每行包含UITextField。我第一次点击文本字段会打开键盘,我准备编辑这个文本字段。如果我点击下一个文本字段(注意键盘一直显示),键盘仍然显示但蓝色光标不在新文本字段中,我无法输入任何文本。但是,如果我再次点击另一个文本字段,它就可以正常工作。这种行为交替发生,有一次它不起作用。

总是调用委托方法textFieldShouldBeginEditing(_:),我可以编辑或不编辑。委托方法textFieldDidBeginEditing(_:)仅在编辑工作时调用。

这是cellForRowAt

的代码
func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
    let cell = tableView.dequeueReusableCell(withIdentifier: "TextFieldCell")!
    let titleLabel = cell.viewWithTag(1) as! UILabel
    let contentTextField = cell.viewWithTag(2) as! FixableTextField
    contentTextField.delegate = self
    contentTextField.inputAccessoryView = doneToolbar

    // Enable/disable editing for text fields
    if isEditing {
        contentTextField.enableEditing()
    } else {
        contentTextField.disableEditing()
    }

    // Present Profile Data
    if profileUpdateBuffer != nil {

        switch indexPath.row {
        case 0:
            titleLabel.text = "Count"
            contentTextField.text = "\(profileUpdateBuffer!.count)"
            contentTextField.purposeID = "count"
            contentTextField.keyboardType = .numberPad

        case 1:
            titleLabel.text = "City"
            contentTextField.text = "\(profileUpdateBuffer!.city)"
            contentTextField.purposeID = "city"
            contentTextField.keyboardType = .default

        // ...

        case 20:
            titleLabel.text = "Name"
            contentTextField.text = "\(profileUpdateBuffer!.name)"
            contentTextField.purposeID = "name"
            contentTextField.keyboardType = .default

        default:
            titleLabel.text = ""
            contentTextField.text = ""
        }

        return cell
    }

    // No data available -> show info in first row
    else {
        if indexPath.row == 0 {
            titleLabel.text = "No data"
            contentTextField.text = "No data"
        }
        else {
            titleLabel.text = ""
            contentTextField.text = ""
        }
        return cell
    }
}

enableEditing()disableEditing()方法来自班级FixableTextField。我可以看到文本字段始终处于启用状态,因为我可以看到文本字段边框

// Extract from FixableTextField class
func enableEditing() {
    self.isEnabled = true
    self.borderStyle = .roundedRect
}

func disableEditing() {
    self.isEnabled = false
    self.borderStyle = .none
}

UITextField

的代码
func textFieldShouldBeginEditing(_ textField: UITextField) -> Bool {

    // Delete empty field indicator "-"
    if textField.text == "-" {
        textField.text = ""
    }

    //Move profileTable's contentView to correct position
    if textField is FixableTextField {
        let path = IndexPath(row: rowMap[(textField as! FixableTextField).purposeID]!, section: 0)
        moveContentViewUp(indexPath: path)
    }
    return true
}



func textFieldDidEndEditing(_ textField: UITextField) {

    // Save new value to profileUpdateBuffer
    do {
        try self.profileUpdateBuffer?.setProperty(value: textField.text!, key: (textField as! FixableTextField).purposeID)
    } catch ProfileError.PropertySettingWrongType {
        let falseInputAlert = UIAlertController(title: "False Input", message: "The input for this field is not valid.", preferredStyle: .alert)
        falseInputAlert.addAction(UIAlertAction(title: "OK", style: .cancel, handler: nil))
        self.present(falseInputAlert, animated: true, completion: nil)
    } catch {
        print("Error when trying to set property for profileUpdateBuffer in ProfileViewController")
    }

    // Display new data in table
    profileTable.reloadData()
}

setPropertyProfileData方法中提取。 profileUpdateBuffer的类型为ProfileData

func setProperty(value:String, key:String) throws {
    switch key {
    case "count":
        count = value

    case "city":
        count = value

    // ...

    case "name":
        name = value

    default:
        throw ProfileError.PropertySettingWrongType
    }
}

2 个答案:

答案 0 :(得分:1)

我制作了一个小程序来模仿你描述的行为。 似乎问题是由textFieldDidEndEditing(_:)

末尾的表格视图数据重新加载引起的
func textFieldDidEndEditing(_ textField: UITextField) {

    // Save new value to profileUpdateBuffer
    do {
        try self.profileUpdateBuffer?.setProperty(value: textField.text!, key: (textField as! FixableTextField).purposeID)
    } catch ProfileError.PropertySettingWrongType {
        let falseInputAlert = UIAlertController(title: "False Input", message: "The input for this field is not valid.", preferredStyle: .alert)
        falseInputAlert.addAction(UIAlertAction(title: "OK", style: .cancel, handler: nil))
        self.present(falseInputAlert, animated: true, completion: nil)
    } catch {
        print("Error when trying to set property for profileUpdateBuffer in ProfileViewController")
    }

    // Display new data in table
    profileTable.reloadData()
}

尝试删除profileTable.reloadData()以便进行实验以确认问题的根本原因(是的,您的其他单元格不会更新)。

解决此问题的一种方法是在visibleCells textFieldDidEndEditing(_:)上使用直接小区更新。我看到profileUpdateBuffer?是您的数据模型。如果它们位于表视图的可见单元属性中,只需从模型中手动更新单元格的titleLabel和textField属性。

如果您想相应调整单元格的大小,请使用自动布局和UITableViewAutomaticDimension表格视图行高与beginUpdates() / endUpdates()来电相结合。

有关如何在不失去键盘焦点的情况下实现直接细胞操作和/或动态像元大小更新的更多详细信息check the accepted answer on this question我已经回答过了。

希望这会有所帮助!

答案 1 :(得分:0)

我认为问题是你调用profileTable.reloadData()......你应该只重新加载一个单元格。也许使用旧的TextField作为参数调用textFieldDidEndEditing( :),然后使用textFieldShouldBeginEditing( :)作为新的TextField。问题是你刷新textFieldDidEndEditing( :)末尾的所有单元格,这意味着系统作为textFieldShouldBeginEditing( :)的参数传递的TextField可能不一定是与相应的单元格包含...可能是那些新的键击被发送到属于可重用单元格队列中的单元格的TextField,即不可见但仍存在于内存中的某个位置。