質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

90.50%

  • Swift

    7448questions

    Swiftは、アップルのiOSおよびOS Xのためのプログラミング言語で、Objective-CやObjective-C++と共存することが意図されています

  • iOS

    4078questions

    iOSとは、Apple製のスマートフォンであるiPhoneやタブレット端末のiPadに搭載しているオペレーションシステム(OS)です。その他にもiPod touch・Apple TVにも搭載されています。

  • Realm

    202questions

    RealmとはSQLiteやCore Dataに代わるモバイルデータベースです。iOSとAndroidの両方でサポートされています。

Realmで値の保存を遅らせたい

解決済

回答 1

投稿 編集

  • 評価
  • クリップ 0
  • VIEW 548

tarofess

score 115

現在以下の画面のようにtableviewのセルに3つtextfieldがあります。

screen説明

そして画面右上のOKが押されるとtextfieldの値がrealmに保存されるようにしていたのですが、コードを書いている時にtextfieldの編集が終わった時点でrealmに値が保存されることに気づきました。コードは以下のようになっています。

func textFieldShouldEndEditing(_ textField: UITextField) -> Bool {
        let realm = try! Realm()

        switch textField.tag {
        case CellTextField.name.rawValue:
            try! realm.write {
                self.product?.name = textField.text
            }

        case CellTextField.category.rawValue:
            try! realm.write {
                self.product?.category = textField.text
            }

        case CellTextField.price.rawValue:
            try! realm.write {
                if !textField.text!.isEmpty {
                    self.product?.price.value = Int(textField.text!)
                }
            }

        default:
            break
        }
        return true
    }

textfieldを編集しても画面左上のCencelが押されれば値は保存されないようにしたいので上記のコードではダメだと思いました。
なのでOKボタンのアクションのハンドラ内でrealmへの保存を行いたいのですが、realmでは値の保存を遅らせたりすることはできるのでしょうか?
その他例えばダミーのProductクラスを一つ作って、上記コードのtextFieldShouldEndEditingでそのダミーProductのインスタンスに値を入れつつ、OKが押された時にダミーのProductをrealmのProductに流しこもうかなと思ったのですが、手間がかかって面倒だと思いました......。
realmへの値の保存を遅らせるだけでなく、その他何かいい方法があれば教えていただきたいです。
よろしくお願いします。

///回答を受けての追記///

        let realm = try! Realm()
        self.view.subviews.forEach({ view in
            if let textField = view as? NameEditTableViewCell {
                try! realm.write {
                    self.product?.name = textField.textLabel?.text
                }
            }
            if let textField = view as? CategoryEditTableViewCell {
                try! realm.write {
                    self.product?.name = textField.textLabel?.text
                }
            }
            if let textField = view as? PriceEditTableViewCell {
                try! realm.write {
                    self.product?.name = textField.textLabel?.text
                }
            }
        })

///更なる追記///

1:

class NameEditTableViewCell: UITableViewCell {

    @IBOutlet weak var nameTextField: UITextField!

}

class CategoryEditTableViewCell: UITableViewCell {

    @IBOutlet weak var categoryTextField: UITextField!
}

class PriceEditTableViewCell: UITableViewCell {

    @IBOutlet weak var priceTextField: UITextField!

}

2: はい、viewcontrollerのフィールドにstoryboardのtableviewから引っ張ってきたtableviewのインスタンスがあります。

3:

func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let type = self.displayCells()[indexPath.row]
        let id = type.rawValue
        let cell = self.tableView.dequeueReusableCell(withIdentifier: id, for: indexPath)
        self.configureCell(targetCell: cell, type: type)

        return cell
    }

func configureCell(targetCell: UITableViewCell, type: CellType) {
        switch type {
        case .name:
            if let cell = targetCell as? NameEditTableViewCell {
                cell.nameTextField.text = self.product?.name
                cell.nameTextField.delegate = self
            }
        case .category:
            if let cell = targetCell as? CategoryEditTableViewCell {
                cell.categoryTextField.text = self.product?.category
                cell.categoryTextField.delegate = self
            }
        case .price:
            if let cell = targetCell as? PriceEditTableViewCell {
                if let price = self.product?.price.value {
                    cell.priceTextField.text = String(describing: price)
                }
                cell.priceTextField.delegate = self
            }
        }
    }
  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 1

checkベストアンサー

0

textFieldShouldEndEditing(_:)デリゲートメソッドはキーボードの「改行」をタップした時に呼ばれるメソッドなので、たしかにこのコードでしたらrealmに3つのテキストフィールドの文字が保存されてしまいますね。
realmの問題よりも保存するタイミングの問題だと思います。
textFieldShouldEndEditing(_:)ではキーボードを閉じるだけにして、OKボタンのIBActionを設定してそこでテキストフィールドの文字をRealmで保存すればいいと思います。

追記:okActionメソッドを変更し、tableviewのvisibleCellsプロパティからcellを取得するようにしました。

//キーボードを閉じる
func textFieldShouldEndEditing(_ textField: UITextField) -> Bool {
//キーボードでEnterボタンを押したら、キーボードを閉じる
textField.resignFirstResponder()
return true
}

OKボタンのIBActionを設定で各テキストフィールドの文字列をRealmに保存する。
Storyboard(またはXib)から以下のokAction(_:)メソッドに接続してください。

//ViewController.swift
 @IBAction func okAction(_ sender: Any) {

        //selfはViewControllerの想定。UITableviewの変数名をtableviewと想定。
     //tableviewのvisibleCellsプロパティで今表示されているcellを全て取得できます。
        self.tableview.visibleCells.forEach({
            cell in
            if let cell = cell as? NameEditTableViewCell {
               //NameEditTableViewCellを取得する
         try! realm.write {
                    self.product?.name = cell.nameTextField.text//cellはNameEditTableViewCellにキャストされたので、nameTextFieldが取得できる
                }
            }

       if let cell = cell as? CategoryEditTableViewCell {
               //CategoryEditTableViewCellを取得する
         try! realm.write {
                    self.product?.category = cell.categoryTextField.text
                }
            }
       if let cell = cell as? PriceEditTableViewCell {
               //PriceEditTableViewCellを取得する
         try! realm.write {
                    self.product?.price.value = Int(textField.text!)
                }
            }
        })
/*
        //self.view.subviewsでViewControllerのサブビューを全て取得
        self.view.subviews.forEach({
            view in

            if let textField = view as? UITextField {//viewをUITextFieldにキャストする。こちら独自のUITextFieldのサブクラスを使用していましたらそちらに変えてください。
                case CellTextField.name.rawValue:
                try! realm.write {
                    self.product?.name = textField.text
                }

                case CellTextField.category.rawValue:
                try! realm.write {
                    self.product?.category = textField.text
                }

                case CellTextField.price.rawValue:
                try! realm.write {
                    if !textField.text!.isEmpty {
                        self.product?.price.value = Int(textField.text!)
                    }
                }

                default:
                break
            }
        }


    })
*/
}

投稿

編集

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2017/01/10 21:52

    ご回答ありがとうございます。if let textField = view as? UITextFieldでキャストするクラスは複数あるので、少しご提示のコードを書き換えてみたのですが、うまく動きません。質問に追記しているのですが、self.view.subviews.forEach内のif let textField = view as? NameEditTableViewCellといった行でうまくキャストできていないようです。。。

    キャンセル

  • 2017/01/10 22:25

    あーなるほど、よく見たらTableViewにカスタムのCellをそれぞれ作っていらっしゃるんですね。。
    するとちょっと情報が足りないのでいろいろお聞きしたいです。
    `okAction `メソッドはやりたいことは各CellのUITextFiledを取得してそのtextをrealmに保存することです。各CellのUITextFiledの取得ができていないので、取得するために、Viewの構造を知りたいです。以下の実装を教えていただけないでしょうか?
    (1と2が特に知りたいです。3は参考程度なイメージです!)
    ---------
    1.NameEditTableViewCell、CategoryEditTableViewCell、PriceEditTableViewCellの実装
    2.ViewControllerとUITableView関係(ViewControllerのプロパティでUITableView保持してます?)
    3.`func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell `の実装

    キャンセル

  • 2017/01/11 07:56

    はい、それぞれのcellにカスタムcellを割り当てています。
    質問に1と2と3について追記しました。
    よろしくお願いします。

    キャンセル

  • 2017/01/11 10:38

    回答を更新しました。UITableViewのvisibleCellsというプロパティで「今表示されているセル」が取得できるので、それをforEachで回して各セルをキャストしてUITextFieldを取得、realmに保存する処理に変更しています。ご確認ください

    キャンセル

  • 2017/01/11 21:08

    visibleCellsでtableviewのセルを取得できました!おかげで目的の動作が確認できて大変嬉しいです。最後まで丁寧に教えてくださり、誠にありがとうございましたmm

    キャンセル

  • 2017/01/11 21:10

    良かったです!私も嬉しいです!

    キャンセル

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 90.50%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る

  • Swift

    7448questions

    Swiftは、アップルのiOSおよびOS Xのためのプログラミング言語で、Objective-CやObjective-C++と共存することが意図されています

  • iOS

    4078questions

    iOSとは、Apple製のスマートフォンであるiPhoneやタブレット端末のiPadに搭載しているオペレーションシステム(OS)です。その他にもiPod touch・Apple TVにも搭載されています。

  • Realm

    202questions

    RealmとはSQLiteやCore Dataに代わるモバイルデータベースです。iOSとAndroidの両方でサポートされています。