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

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

ただいまの
回答率

87.48%

TableViewCellを保存する方法(UserDefaults)

解決済

回答 1

投稿

  • 評価
  • クリップ 1
  • VIEW 1,649

score 39

実行したいこと

TableviewcellにtextFieldで出力した値を配列に入れTodoとして取り出しているのですが、アプリを閉じるとcellが消えてしまうのでUserDefaultsに保存したいのですがいまいちわかりません。

また,遷移先にtextViewを設け一言書けるようにもしたいのですが、こちらもcellの保存は第一にcellとは別で保存する必要があるともうのですがどうですか?

該当コード(ViewController)

import UIKit

class ViewController: UIViewController,UITableViewDelegate,UITableViewDataSource,UITextFieldDelegate {


    @IBOutlet weak var textField: UITextField!
    @IBOutlet weak var tableView: UITableView!


    var textArray = [String]()

    override func viewDidLoad() {
        super.viewDidLoad()
        // Do any additional setup after loading the view.

        tableView.delegate = self
        tableView.dataSource = self
        textField.delegate = self

    }

    override func viewWillAppear(_ animated: Bool) {
        super.viewWillAppear(animated)

        navigationController?.isNavigationBarHidden = true

        if UserDefaults.standard.object(forKey: "todo") != nil{

            textArray = UserDefaults.standard.object(forKey: "todo") as! [String]
        }

    }

    func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {

        return textArray.count

    }

    func numberOfSections(in tableView: UITableView) -> Int {

        return 1
    }

    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {

        let cell = tableView.dequeueReusableCell(withIdentifier: "Cell", for: indexPath)

        cell.selectionStyle = .none
        cell.textLabel?.text = textArray[indexPath.row]
        cell.imageView?.image = UIImage(named: "profile")
        cell.textLabel?.textColor = .white

        return cell

    }

    func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {

        let nextVC = storyboard?.instantiateViewController(identifier: "next") as! NextViewController

        nextVC.todoString = textArray[indexPath.row]

        navigationController?.pushViewController(nextVC, animated: true)

    }

    func tableView(_ tableView: UITableView, heightForRowAt indexPath: IndexPath) -> CGFloat {

        return view.frame.size.height/7

    }

    func textFieldShouldReturn(_ textField: UITextField) -> Bool {

        textArray.append(textField.text!)
        textField.resignFirstResponder()
        textField.text = ""
        tableView.reloadData()

        return true

    }
}

該当コード(NextViewController)

import UIKit

class NextViewController: UIViewController {

    var todoString = String()


    @IBOutlet weak var textView: UITextView!
    @IBOutlet weak var Todolabel: UILabel!

    override func viewDidLoad() {
        super.viewDidLoad()

        // Do any additional setup after loading the view.

        Todolabel.text = todoString

    }

    override func viewWillAppear(_ animated: Bool) {
        super.viewWillAppear(animated)

        navigationController?.isNavigationBarHidden = false
    }
}

補足

if UserDefaults.standard.object(forKey: "todo"と一応呼び出しているのですがうまくは実証されませんでした。保存元がtextArrayではないのでしょうか?また保存したものを取り出すと思うのですがどのようにすれば良いですか?

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

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

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

  • TsukubaDepot

    2020/04/06 00:22

    > また,遷移先にtextViewを設け一言書けるようにもしたいのですが、こちらもcellの保存は第一にcellとは別で保存する必要があるともうのですがどうですか?

    ここの記述がよくわからないのですが、具体的にはどのようなことを行いたいのでしょうか。

    キャンセル

  • Ytan

    2020/04/06 08:27

    cellひとつひとつにTextViewの文字を保存してcellを戻した時やアプリを閉じた時も保存されていて欲しいということですね。

    個人的にTextViewはcellの配列ではなくTextView単体として保存するのか?と思ったので質問しました!

    キャンセル

回答 1

checkベストアンサー

0

[String]型のUserDefaultsを使った保存と呼び出しはこんな感じになります。

import Foundation

let todos = ["食う", "寝る", "遊ぶ"]

// 保存
UserDefaults.standard.set(todos, forKey: "todo")

// 読み込み
if let getData = UserDefaults.standard.stringArray(forKey: "todo") {
    print(getData)
} else {
    print("error")
}

// object(forKey:)でも取得できるが、型変換が必要
if let getData = UserDefaults.standard.object(forKey: "todo") as? [String] {
    print(getData)
} else {
    print("error")
}


適切な場所で保存し、適切な場所で読み出せばいいかと思います。

上記のサンプルはPlaygroundでも動くので、まずは簡単なサンプルを作り動かしてみることで動作を確認されてはいかがでしょうか。

--

ToDo のデータを保存したいということであれば、Cell に保存するというか、1件のToDoに対応するクラスを作り、それを配列にして扱った方が簡単です。

たとえば、

class Todo {
    var title: String
    var todoDescription: String

    init(title: String, description: String = "") {
        self.title = title
        self.todoDescription = description
    }
}


のような感じでクラスを作り、適切なところで

var todoArray: [Todo] = []

` といった感じでTodoクラスの配列を作り、

    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        // 主要なところのみ掲載
        let cell = tableView.dequeueReusableCell(withIdentifier: "Cell", for: indexPath)
        cell.textLabel?.text = todoArray[indexPath.row].title
        return cell
    }

でテーブルセルに表示させたり、

    func textFieldShouldReturn(_ textField: UITextField) -> Bool {
        let text = textField.text ?? ""
        todoArray.append(Todo(title: text))

        tableView.reloadData()
        // その他必要な処理
        return true

    }

で新規データを追加すればいいと思います。

同じような感じで、Todoの詳細(Todo.todoDescription)は遷移先でセットすればいいのではないでしょうか。

ちなみに、カスタムクラス(正確にはプロパティリスト)以外のクラスをUserDefaultsに保存するためには、さらにひと工夫必要です。

これについては過去の回答の後半に例がありますので、そちらも参考にしてみてください。

投稿

編集

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2020/04/06 11:15

    いつも丁寧な返答ありがとうございます。こちらで試してみたいと思います!

    キャンセル

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

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

関連した質問

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