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

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

ただいまの
回答率

88.77%

SwiftでtodoListアプリ

解決済

回答 1

投稿 編集

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

o2ka_cz

score 13

前提・実現したいこと

Swiftでtodoリストのアプリ開発をしています。
+ボタンをタップしたらtodoを入力します。OKボタンでTable View Cellに追加して表示するところまではできました。
そこに、todoにチェックマークをつけることと、UserDefaultを使用したデータ保存機能を実装しようとしています。
独自クラスMyToDoを作成し、viewDidRoadで保存しているtodoのデータを読み取るようにしました。
アプリ自体は起動するのですが、todoを保存することができず、アプリが落ちてしまいます。

発生している問題・エラーメッセージ

MyToDoList[28985:2219996] [User Defaults] Attempt to set a non-property-list object (
"<MyToDoList.MyToDo: 0x6000027f4080>"
) as an NSUserDefaults/CFPreferences value for key todoList

ソースコード

import UIKit

class ViewController: UIViewController, UITableViewDataSource, UITableViewDelegate {

    var todoList = [MyToDo]()
    // UITableViewに対して操作を行うため
    @IBOutlet weak var tableView: UITableView!

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

    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCell(withIdentifier: "todoCell", for: indexPath)
        let myTodo = todoList[indexPath.row]
        cell.textLabel?.text = myTodo.todoTitle
        if myTodo.todoDone {
            cell.accessoryType = UITableViewCell.AccessoryType.checkmark
        } else {
            cell.accessoryType = UITableViewCell.AccessoryType.none
        }
        return cell
    }


    override func viewDidLoad() {
        super.viewDidLoad()

        let userDefaults = UserDefaults.standard
        if let storedToDoList = userDefaults.object(forKey: "todoList") as? Data {
            if let unarchiveToDoList = try! NSKeyedUnarchiver.unarchiveTopLevelObjectWithData(storedToDoList) as? MyToDo {
                todoList.append(unarchiveToDoList)
            }
        }
    }

    @IBAction func tapAddButton(_ sender: Any) {
        // アラートダイアログを生成
        let alertControllor = UIAlertController(title: "ToDo追加", message: "ToDoを追加してください", preferredStyle: UIAlertController.Style.alert)
        // テキストエリアを追加
        alertControllor.addTextField(configurationHandler: nil)
        // OKボタンを生成
        let okAction = UIAlertAction(title: "OK", style: UIAlertAction.Style.default) { (action: UIAlertAction) in
           // OKボタンがタップされた時の処理
            if let textField = alertControllor.textFields?.first {
                // ToDoの配列に入力値を先頭に格納
                let myTodo = MyToDo()
                myTodo.todoTitle = textField.text!
                self.todoList.insert(myTodo, at: 0)

                // テーブルに行が追加されたことをテーブルに通知
                self.tableView.insertRows(at: [IndexPath(row: 0, section: 0)], with: UITableView.RowAnimation.right)

                // ToDoの保存処理
                let userDefaults = UserDefaults.standard
                userDefaults.set(self.todoList, forKey: "todoList")
                userDefaults.synchronize()
            }
        }
        // OKボタンを追加
        alertControllor.addAction(okAction)

        // CANSELボタンがタップされた時の処理
        let cancelButton = UIAlertAction(title: "CANCEL", style: UIAlertAction.Style.cancel, handler: nil)
        // CANSELボタンを追加
        alertControllor.addAction(cancelButton)

        // アラートダイアログを表示
        present(alertControllor, animated: true, completion: nil)
    }

}


class MyToDo: NSObject, NSCoding {

    // ToDoのタイトル
    var todoTitle: String?
    // ToDoを完了したかを表すフラグ
    var todoDone: Bool = false
    override init(){

    }

    // NSCodingプロトコルに宣言されているデシリアライズ処理
    // decode
    required init?(coder aDecoder: NSCoder) {
        todoTitle = (aDecoder.decodeObject(forKey: "todoTitle") as? String)!
        todoDone = aDecoder.decodeBool(forKey: "todoDone")
    }

    //encode
    func encode(with aCoder: NSCoder) {
        aCoder.encode(todoTitle, forKey: "todoTitle")
        aCoder.encode(todoDone, forKey: "todoDone")
    }
}
  • 気になる質問をクリップする

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

質問への追記・修正、ベストアンサー選択の依頼

  • hameji

    2019/12/09 00:08 編集

    コードはスコープ({ } のこと)ごとにインデントの位置を揃え、
    「```」を先頭と末尾に行を変えてつけていただけるとコードがみやすくなります。
    https://teratail.com/help/question-tips#questionTips3-3-1 の アドバイスのとこ見てください

    キャンセル

  • o2ka_cz

    2019/12/09 00:10

    アドバイスありがとうございます。
    変更してみたので確認よろしくお願いします。

    キャンセル

  • hameji

    2019/12/09 00:13

    どの段階で落ちるのでしょうか?
    tapAddButtonでAlertのOKを押した時ですか?

    キャンセル

  • o2ka_cz

    2019/12/09 00:16

    はい、そのときに落ちてしまいます。

    キャンセル

回答 1

checkベストアンサー

0

追記・修正欄でも聞きましたが、
AlertのOKの時に落ちるとの事でしたね。
3つのことを行っている(①新規MyToDoの作成、②tableViewの表示更新、③データuserdefaultsに保存)ので、
それぞれを順にコメントアウト (該当部分を選択状態にし「command」+「/」) し、実行してどの部分が間違っているか確認してください。

おそらく、というかuserdefaultsの保存の仕方だと思います。
https://hitoto28.hatenablog.com/entry/2018/07/09/020000
を参考にされた?かもしれませんが、保存のところで、

// シリアライズ処理
let archiveData = NSKeyedArchiver.archivedData(withRootObject: sampleData)

ということが行われています。
それが抜けているのではないでしょうか?

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2019/12/09 11:23

    シリアライズ処理を追加したらアプリが落ちなくなりました!
    しかし、アプリを再起動したらデータが全て消えていました。
    override func viewDidLoad() {
    super.viewDidLoad()

    let userDefaults = UserDefaults.standard
    if let storedToDoList = userDefaults.object(forKey: "todoList") as? Data {
    if let unarchiveToDoList = try! NSKeyedUnarchiver.unarchiveTopLevelObjectWithData(storedToDoList) as? MyToDo {
    self.todoList.append(unarchiveToDoList)
    }
    }
    }

    データを保存できているのかがわかりません。

    キャンセル

  • 2019/12/09 12:01

    違う質問になっていますが、、、、
    別に立て直してください

    キャンセル

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

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

関連した質問

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