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

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

ただいまの
回答率

89.19%

【Swift4】tableViewCellの編集について

解決済

回答 2

投稿

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

aiskw

score 16

前提・実現したいこと

簡易的なToDoアプリを制作しています。下記を実現したく、色々と調べてみたのですが一部解決できません。

1.一覧画面上でtextViewにテキストを入力
2.tableViewCellに入力されたテキストが表示(UserDefalutsで保存)
3.入力済みのtableViewCellをタップ
4.編集画面(textView)に値を保持した状態で遷移
5.内容を編集後、編集完了ボタンタップで一覧画面に戻る
6.tabeViewCellが編集した内容に更新される 

上記手順のうち、6が実装できないため、解決策をご教示いただけませんでしょうか。
よろしくお願いいたします。

該当のソースコード

import UIKit
import AVFoundation

class ViewController: UIViewController, UITableViewDataSource, UITableViewDelegate {

    var todoList = [String]()

    var talker = AVSpeechSynthesizer()

    var senderText: String?

    @IBOutlet weak var inputTextView: UITextView!

    @IBOutlet weak var tableView: UITableView!

    override func viewDidLoad() {
        super.viewDidLoad()

        let userDefaults = UserDefaults.standard
        if let storedTodoList = userDefaults.array(forKey: "saveTodo") as? [String] {
            todoList.append(contentsOf: storedTodoList)
        }

    }

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

        tableView.reloadData()
    }

    @IBAction func tapAddButton(_ sender: Any) {
        if let inputText = inputTextView.text {
            self.todoList.insert(inputText, at: 0)
            self.tableView.insertRows(at: [IndexPath(row: 0, section: 0)], with: UITableView.RowAnimation.right)
        }

        inputTextView.endEditing(true)
        inputTextView.text = ""

        let userDefaults = UserDefaults.standard
        userDefaults.set(self.todoList, forKey: "saveTodo")
        userDefaults.synchronize()

        print("save")
    }

    override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {
        self.view.endEditing(true)
    }

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

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


        let cell = tableView.dequeueReusableCell(withIdentifier: "todoCell", for: indexPath)
        let todoText = todoList[indexPath.row]
        cell.textLabel?.text = todoText


        cell.textLabel?.numberOfLines = 0
        cell.accessoryType = .detailButton

        return cell
    }

    func tableView(_ tableView: UITableView, commit editingStyle: UITableViewCell.EditingStyle, forRowAt indexPath: IndexPath) {
        if editingStyle == UITableViewCell.EditingStyle.delete {
            todoList.remove(at: indexPath.row)
            tableView.deleteRows(at: [indexPath], with: UITableView.RowAnimation.fade)
            let userDefaults = UserDefaults.standard
            userDefaults.set(self.todoList, forKey: "saveTodo")
            userDefaults.synchronize()

            print("delete")
        }
    }

    func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
        tableView.deselectRow(at: indexPath, animated: false)
        performSegue(withIdentifier: "goEdit", sender: indexPath)
        senderText = todoList[indexPath.row]

        print(todoList[indexPath.row])
    }

    func tableView(_ tableView: UITableView, didHighlightRowAt indexPath: IndexPath) {
        senderText = todoList[indexPath.row]
    }

    override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
        let secondVC = segue.destination as! SecondViewController
        secondVC.receivedText = senderText
    }

    func tableView(_ tableView: UITableView, accessoryButtonTappedForRowWith indexPath: IndexPath) {
        let utterance = AVSpeechUtterance(string: todoList[indexPath.row])
        utterance.voice = AVSpeechSynthesisVoice(language: "en-US")
        utterance.preUtteranceDelay = 0;
        self.talker.speak(utterance)
        print("accessoryTapped")
    }
}
import UIKit

class SecondViewController: UIViewController {

    var receivedText: String?

    @IBOutlet weak var editngTextView: UITextView!

    override func viewDidLoad() {
        super.viewDidLoad()

        editngTextView.text = receivedText
    }



    @IBAction func tapOKButton(_ sender: Any) {

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

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

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

  • takabosoft

    2019/06/07 10:15

    Segueがどうなってるかわかりませんが、SecondViewControllerってモーダルで出るんでしょうかね?それともViewControllerと入れ替えですかね?

    キャンセル

  • aiskw

    2019/06/07 15:10

    ご回答いただきありがとうございます!
    SecondViewControllerはSegueで繋いでモーダルで出しています。

    拙い情報で申し訳ないですが、ご教示いただけますと幸いです。

    キャンセル

  • aiskw

    2019/06/08 18:06

    度々の質問すみません。

    昨日ご教示をいただいた内容にて試行錯誤していたのですが、
    viewController.todoList[(編集対象の行番号)] = editngTextView.text!
    上記の編集対象の行番号の取得がうまくいかず、先に進めず困っております。

    編集元のindexPath.rowを編集先に渡し、編集後に元のビューに編集した内容と行番号を返して更新するのだと想定しているのですが、もしよろしければアドバイスを頂けませんでしょうか。
    よろしくお願いいたします。

    キャンセル

回答 2

checkベストアンサー

+1

※私はStoryboardを使わない派なので、頓珍漢な事を言っていたらすみません。

モーダルで呼び出されたUIViewControllerから、呼び出し元のUIViewControllerへ何か値を戻すような事はアプリを開発していると頻繁にでます。

もしモーダルとして呼び出されたUIViewControllerが汎用的なもので、いろんな種類のUIViewControllerから呼び出される場合は、デリゲートパターンやクロージャを使って値を戻します(呼び出し元が誰だろうと関係なく値を戻せるような仕組みを作ります)。

しかし、もし汎用的なものではなく、SecondViewControllerはViewControllerからしか呼ばれることはないのであれば、以下のような方法を採っても大丈夫です(強結合なのでもしかすると嫌う人もいるかもしれませんが)。

OKボタンが押されたら

    let viewController = presentingViewController as! ViewController // fuzzballさんの指摘によりこちらの方法に変更(多用するのであればcomputed propertyにする方がよいかも)
    viewController.todoList[(編集対象の行番号)] = editngTextView.text!
    dismiss(animated: true)

のように、直接書き換えてしまいます。

prepareはそもそも不要かもです。

    /*override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
        let secondVC = segue.destination as! SecondViewController

    }*/

あとは編集対象の行番号をうまいことやりくりする方法を考えて実装してみてください。

投稿

編集

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2019/06/07 17:05

    あら、そんなものが・・・フォローありがとうございます。
    var ownerViewController: ViewController {
    return presentingViewController as! ViewController
    }
    ってやっちゃえば、ストアド変数要らなさそうですかね。

    キャンセル

  • 2019/06/07 17:11

    回答を改造しました。

    キャンセル

  • 2019/06/07 17:21

    fuzzball様
    ご回答いただきありがとうございます!
    色々な方法があるのですね。アドバイスを元に色々と調べてみようと思います!
    取り急ぎお礼申し上げます。

    takabosoft様
    回答を編集していただきありがとうございます!
    帰宅してから色々と触ってみるのが楽しみです!

    takabosoft様

    キャンセル

0

詳細かつご丁寧なご回答ありがとうございます!
上記アドバイスを元に取り組んでみたいと思います。
手を動かしつつ理解していくつもりですが、もし詰まってしまった場合はまたご相談させていただくかもしれません。。
取り急ぎお礼申し上げます。

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2019/06/07 16:06

    上記、返信を書く箇所を間違えて記入したものになります。。

    キャンセル

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

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