UITextField
を含む約20行のUITableView
があります。最初にテキストフィールドをタップするとキーボードが開き、このテキストフィールドを編集する準備が整いました。次のテキストフィールドをタップすると(キーボードは常に表示されます)、キーボードはまだ表示されていますが、青色のカーソルは新しいテキストフィールドにはなく、テキストを入力することはできません。しかし、もう一度別のテキストフィールドをタップするとうまくいきます。この現象は交互に発生します。もう1回は動作しません。UITableViewの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
からです。私はクラスProfileData
からであるsetProperty
方法から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()
}
抽出のためのテキストフィールドの境界線に
// Extract from FixableTextField class
func enableEditing() {
self.isEnabled = true
self.borderStyle = .roundedRect
}
func disableEditing() {
self.isEnabled = false
self.borderStyle = .none
}
コードを見ることができるので、私はテキストフィールドが常に有効になっていることがわかります。 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
}
}
私たちがあなたを助けるためにコードを投稿してください。特に、cellForRowAt indexPath関数。あなたのtextFieldsにリスナーをどのように追加しますか? –
ありがとう、もちろん私はします。 – Codey
'textFieldShouldBeginEditing(_ :)'にフィルタリングロジックがありますか?あなたのVCがすべてのテキストフィールドの代理人になるように設定されていることがわかります。デリゲートメソッドの実装内のセルを区別するロジックがありますか? TextFieldデリゲートをセル内で移動して、各セルがそれ自身の責任を負うようにすることができます。 – kr45ko