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

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

ただいまの
回答率

87.36%

Swiftでデータ保存機能(ToDoList)

解決済

回答 1

投稿

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

score 13

前提・実現したいこと

SwiftでToDoリストのアプリを開発しています。
画面の+ボタンをタップすると、ToDoを追加するアラートが表示されます。
OKボタンで保存し、アプリを再起動すると保存したデータが表示されるようにしたいです。

発生している問題

データを保存して、アプリ再起動時に保存してあるデータを読み取る処理を書いたのですが、データが消えてしまいます。
データが保存されていないのか、読み取りができていないのかがわかりません。

該当のソースコード

//
//  ViewController.swift
//  MyToDoList
//

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 {
                self.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
                let archiveData = try! NSKeyedArchiver.archivedData(withRootObject: self.todoList, requiringSecureCoding: false)
                // userDefaults.set(self.todoList, forKey: "todoList")
                userDefaults.set(archiveData, 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ページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

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

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

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

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

    詳細な説明はこちら

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

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

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

回答 1

checkベストアンサー

0

printなどを用いて検証するのが、バグの修正では常套手段です。
今回の場合は2点あり、
①保存されてるかどうか、②表示ように変換ができているかが検証点ですね。

コードの意味を理解しているなら、どこに書けばいいか考えれば、
すぐわかると思います。

検証方法の答えを示してしまうと、

override func viewDidLoad() {
    super.viewDidLoad()

    let userDefaults = UserDefaults.standard
    if let storedToDoList = userDefaults.object(forKey: "todoList") as? Data { // userDefaultsにtodoListキーで保存されているかどうかなので、
        if let unarchiveToDoList = try! NSKeyedUnarchiver.unarchiveTopLevelObjectWithData(storedToDoList) as? MyToDo { // 同様に、DataからMyToDoに変換し、nil(失敗)しないでデータを得た場合
            self.todoList.append(unarchiveToDoList)
        } else {
            print("Dataの変換に失敗しています。")
        }
    } else {
        print("todoListキーでdataが保存されていません。")
    }
}

また、変換前のデータを直接print()するのもヒントになるかもしれません。

print(storedToDoList)

// as? MyToDoと型変換せずに、printしてみる
let unarchiveToDoList = try! NSKeyedUnarchiver.unarchiveTopLevelObjectWithData(storedToDoList)
print(unarchiveToDoList)

よくコードの中身を検証していませんが、
根本的な問題点としては、
MyToDoに変換しているのがおかしい気がします。

保存する時は self.todoList を保存しているので、
var todoList = [MyToDo]()だから、、つまり、MyToDo 1個ではなく、
MyToDoの配列に変換すべきなんじゃないですか???
なので、appendも不要で、直接MyToDoの配列に変換し、
代入するすべきだと思います。

よくコードの中身を追って、順に考え、整合性を考えてみてください。

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2019/12/10 23:00

    初心者なのでとても参考になります。
    ありがとうございました!!

    キャンセル

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

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

関連した質問

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