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

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

ただいまの
回答率

90.51%

  • Swift

    7272questions

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

  • Xcode

    4115questions

    Xcodeはソフトウェア開発のための、Appleの統合開発環境です。Mac OSXに付随するかたちで配布されています。

  • Realm

    197questions

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

Realmのデータの追加について

解決済

回答 1

投稿

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

haruka-t

score 77

import UIKit
import RealmSwift

extension UIColor {
    class func rgb(r: Int, g: Int, b: Int, alpha: CGFloat) -> UIColor{
        return UIColor(red: CGFloat(r) / 255.0, green: CGFloat(g) / 255.0, blue: CGFloat(b) / 255.0, alpha: alpha)
    }
}

class TopViewViewController: UIViewController,UITableViewDataSource,UITableViewDelegate {


    @IBOutlet weak var TopViewTabelView: UITableView!
    let ToDo = TopTodo()
    var todoItem: Results<TopTodo>!


    override func viewDidLoad() {
        super.viewDidLoad()

        // 永続化されているデータを取りだす
        do{
            let realm = try Realm()
            self.todoItem = realm.objects(TopTodo.self)
            self.TopViewTabelView.reloadData()
        }catch{

        }

        navigationController?.navigationBar.prefersLargeTitles = true //これがやりたかった設定やメモ帳のナビゲーションの可変のやつ
        navigationItem.title = "データベース"
        TopViewTabelView.tableFooterView = UIView(frame: .zero)
        self.view.backgroundColor = UIColor.rgb(r: 250, g: 204, b: 125, alpha: 1)
        TopViewTabelView.frame = CGRect(x: 0, y: 64, width: 375, height: 667)

    }




    @IBAction func addBtr(_ sender: Any) {
        let alert = UIAlertController(title: "タイトル", message: "メッセージ", preferredStyle: .alert)

        // OKボタンの設定
        let okAction = UIAlertAction(title: "OK", style: .default, handler: {
            (action:UIAlertAction!) -> Void in

            // OKを押した時入力されていたテキストを表示
            if let textFields = alert.textFields {

                // アラートに含まれるすべてのテキストフィールドを調べる
                for textField in textFields {
                    //self.item.insert(textField.text!, at: 0)

                    let newTodo = self.ToDo
                    do{
                        let realm = try Realm()
                        try realm.write({ () -> Void in
                            realm.add(newTodo)
                            print("ToDo Saved")
                            self.ToDo.item.append(textField.text!)
                            self.TopViewTabelView.insertRows(at: [IndexPath(row: 0, section: 0)],with: UITableViewRowAnimation.automatic)
                            print(textField.text!)
                        })
                    }catch{
                        print("Save is Faild")
                    }
                }
                self.TopViewTabelView.reloadData()
            }
        })
        alert.addAction(okAction)

        // キャンセルボタンの設定
        let cancelAction = UIAlertAction(title: "Cancel", style: .cancel, handler: nil)
        alert.addAction(cancelAction)

        // テキストフィールドを追加
        alert.addTextField(configurationHandler: {(textField: UITextField!) -> Void in
            textField.placeholder = "テキスト"
        })
        alert.view.setNeedsLayout() // シミュレータの種類によっては、これがないと警告が発生

        self.present(alert, animated: true, completion: nil)
    }

    func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return todoItem.count
    }

    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let Cell = tableView.dequeueReusableCell(withIdentifier: "cell", for: indexPath)
        /*let now = NSDate() // 現在日時の取得

        let dateFormatter = DateFormatter()
        dateFormatter.locale = NSLocale(localeIdentifier: "en_US") as Locale? // ロケールの設定
        dateFormatter.dateFormat = "yyyy年MM月dd日 HH:mm"//:ss" // 日付フォーマットの設定

        let dateString = dateFormatter.string(from: now as Date)
        print(dateString) // -> 2014/06/25 02:13:18*/

        let object = todoItem[indexPath.row]
        Cell.textLabel?.text = object.item
        Cell.textLabel?.textAlignment = NSTextAlignment.center
        /*Cell.detailTextLabel?.text = dateString
        Cell.detailTextLabel?.textAlignment = NSTextAlignment.right*/
        return Cell
    }

    func tableView(_ tableView: UITableView, commit editingStyle: UITableViewCellEditingStyle, forRowAt indexPath: IndexPath) {


        if(editingStyle == UITableViewCellEditingStyle.delete) {
            do{
                let realm = try Realm()
                try realm.write {
                    realm.delete(self.todoItem[indexPath.row])
                }
                TopViewTabelView.deleteRows(at: [indexPath], with: UITableViewRowAnimation.fade)
            }catch{
            }
            TopViewTabelView.reloadData()
        }
    }



    override func didReceiveMemoryWarning() {
        super.didReceiveMemoryWarning()
        // Dispose of any resources that can be recreated.
    }

}
import RealmSwift

class TopTodo: Object{
    @objc dynamic  var item = ""
}

 実現したいこと

tableViewに追加したデータを永続的に保存したい。

 困っていること

現状ではtableViewにalertでデータを追加しているのですが一つしかデータを追加できなくて困っています。2つ目を追加しようとボタンをタップすると落ちていまう。(理由が分からない)
また、todoの一覧部分の永続保存の処理までは一応出来たが遷移先のtableViewのデータの保存の仕方が分からなくて困っている。(labelなど複雑な処理がある為)

gitに上げているので見て頂けると分かりやすいと思います
https://github.com/haruka22/new-project

  • 気になる質問をクリップする

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

回答 1

checkベストアンサー

+1

ご提示いただいたGithubのソースを試してみたところ
原因としてはinsertRowsをしたあとのtableViewの更新でDatasourceが持っているデータ数と表示するセルの数に不整合が生じているからだと思います。

// アラートに含まれるすべてのテキストフィールドを調べる
for textField in textFields {

    let newTodo = self.ToDo
    do{
        let realm = try Realm()
        try realm.write({ () -> Void in
            realm.add(newTodo)
            print("ToDo Saved")
            self.ToDo.item.append(textField.text!)
            self.TopViewTabelView.insertRows(at: [IndexPath(row: 0, section: 0)],with: UITableViewRowAnimation.automatic)
            print(textField.text!)
        })
    }catch{
        print("Save is Faild")
    }
}
self.TopViewTabelView.reloadData()

try realm.write中でself.TopViewTabelView.insertRowsを呼んでいますが、
この時点だとRealmにはまだデータが追加されていない(=まだコミットされていないため)
一度トランザクションを抜けてコミットをする必要があります。

また、todoItemの更新が行われていないため、rowを追加してもTableViewの更新時点で
todoItemの数(numberOfRowsInSectionで使われている数)はTableViewでinsertしている分の数足りなくなっています。

下記のようにしたらどうなりますでしょうか?

// アラートに含まれるすべてのテキストフィールドを調べる
for textField in textFields {

    // 変数で持つ必要はないのではないでしょうか?
    let newTodo = TopTodo()
    newTodo.item = textField.text!
    do{
        let realm = try Realm()

        // トランザクション開始
        try realm.write({ () -> Void in
            realm.add(newTodo)
            print("ToDo Saved")
        })
        // ここでトランザクションが終了

     // ここでtodoItemの更新
        self.todoItem = realm.objects(TopTodo.self)
        self.TopViewTabelView.insertRows(at: [IndexPath(row: 0, section: 0)],with: UITableViewRowAnimation.automatic)
        print(textField.text!)

    }catch{
        print("Save is Faild")
    }
}
self.TopViewTabelView.reloadData()

あまり回答できる時間がなくて恐縮ですが、ご検討いただけましたら幸いです。
何卒よろしくお願い致します。

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2018/06/21 13:50

    ありがとうございます。見事に解決しました。処理を書く順番が問題だったんですね。
    それと差し支えが無けれあば教えて頂きたいんですが、Objectのクラスに追加で@objc dynamic var date = NSDate()を追加したところnumberOfRowsInSectionでエラーが出てしまいました。
    以下がエラーコードになります。
    Fatal error: Unexpectedly found nil while unwrapping an Optional value
    色々調べて見たのですがどこにもこのようなエラーで困っている人がいなく参考に出来る記事がなくて困っています。もしお時間ありましたらご教授して頂けると幸いです。

    キャンセル

  • 2018/06/25 03:05

    どうしても分からない箇所がありご教授願いたいです。
    Objectクラスにて、追加するセルの名前の他にサブタイトルで日時を追加したく、クラスの中に@objc dynamic var date = NSDate()を追加したところnumberOfRowsInSectionでアンラップエラーが起きてしまいその対処に苦慮しています。
    エラーコードとしては、オプショル型が問題なのかなと思うのですがどこがいけないのかが分からないです。
    差し支えなければご教授して頂けないでしょうか?
    よろしくお願いします。
    https://teratail.com/questions/132346

    キャンセル

  • 2018/07/11 02:41

    お忙し中すいません。先日は懇切に教えて頂きありがとうございました。
    また、分からないところがあり色々ネットなどでも調べて見たのですが解決方法が見つからず苦慮しています。もし宜しければお時間ある時にご教授願いたいです。
    具体的に説明しますと、現在カスタムアラートを製作しており、その過程でアラート内に数字キーボードを用意しています。
    そして、用意したキーボードからtextFieldsに値を入力していくのですがその際の条件分岐の仕方が分からなくて困っています。
    やりたい事としては、セグメントコントロールで選択したテキストフィールドのみ入力を行えるようにしたいのですがそのやり方についてどうしても分からなくて困っています。
    そのため、お時間のある時に御教授して頂けたら幸いです。何卒宜しくお願いいたします。
    https://teratail.com/questions/135363

    キャンセル

  • 2018/08/04 23:24

    お久しぶりです。お忙し中すいません。現在ios-chartsにて入力したデータを折れ線グラフにするグラフを製作しています。
    大まかな部分は出来たのですが、入力した時の日付を横軸に表示させたいのですがどうも上手くいかなくて困っています。
    そのため、もしお時間がありましたら見ていただけないでしょうか?

    キャンセル

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

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

関連した質問

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

  • Swift

    7272questions

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

  • Xcode

    4115questions

    Xcodeはソフトウェア開発のための、Appleの統合開発環境です。Mac OSXに付随するかたちで配布されています。

  • Realm

    197questions

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