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

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

ただいまの
回答率

88.91%

SegmentedControl tableView delete機能が上手くいきません。

受付中

回答 1

投稿 編集

  • 評価
  • クリップ 1
  • VIEW 129

Ytan

score 34

前回の質問からの変化

前回の質問 リンク内容
こちらでの問題であった
最後に取得した行数が現在のtodos.countとして取得されていてdeleteされたらnumberOfRowsで整合性が保たれていないと言うエラーが出ていたので直しました。

//todosから要素を削除して保持しておく
        let deletedItem = self.todos.remove(at: indexPath.row)

        self.allTodo.removeAll {$0 === deletedItem}
        self.selectedTodo.removeAll {$0 === deletedItem}

        //削除した結果も保存
        let data = try! NSKeyedArchiver.archivedData(withRootObject: self.allTodo, requiringSecureCoding: false)
        UserDefaults.standard.set(data, forKey: "todoList")

        let data2 = try! NSKeyedArchiver.archivedData(withRootObject: self.selectedTodo, requiringSecureCoding: false)
        UserDefaults.standard.set(data2, forKey: "todoShare")

        tableView.deleteRows(at: [indexPath], with: .automatic)

        completionHandler(true)

ソースコード

import UIKit

class ViewController: UIViewController,UITableViewDelegate,UITableViewDataSource {

    //registered cell
    var todos:[Item] = []
    //to show in First case
    var allTodo:[Item] = []
    //to show in second case
    var selectedTodo:[Item] = []

    @IBOutlet weak var Table:UITableView!

    @IBAction func segmentselected(_ sender: UISegmentedControl) {

        switch sender.selectedSegmentIndex {

        case 0:
            todos = allTodo
        case 1:
            todos = selectedTodo
        default:
            fatalError("case でカバーできていません")
        }
        Table.reloadData()
    }

    override func viewDidLoad() {
        super.viewDidLoad()
        // Do any additional setup after loading the view
        Table.delegate = self
        Table.dataSource = self

        //UD読み込み
        if let data = UserDefaults.standard.data(forKey: "todoList"){
            self.allTodo = try! NSKeyedUnarchiver.unarchiveTopLevelObjectWithData(data) as!
                [Item]
        }

        if let data2 = UserDefaults.standard.data(forKey: "todoShare"){
            self.selectedTodo = try! NSKeyedUnarchiver.unarchiveTopLevelObjectWithData(data2) as!
                [Item]
        }

    }

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

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

        let cell:UITableViewCell = tableView.dequeueReusableCell(withIdentifier: "Cell", for: indexPath)
        let item = todos[indexPath.row]
        cell.textLabel!.text = item.title
        return cell
    }

    func tableView(_ tableView: UITableView, heightForRowAt indexPath: IndexPath) -> CGFloat {
        return view.frame.size.height/10
    }

    @IBAction func addNewTodo(_sender: Any){
        var textField = UITextField()

        let alert = UIAlertController(title: "新しくやることを追加", message: "", preferredStyle: .alert)

        let action = UIAlertAction(title: "追加", style: .default) { (action) in

            let newItem:Item = Item(title: textField.text!)
            print("追加されました")

            self.allTodo.append(newItem)

            //UD保存
            let data = try! NSKeyedArchiver.archivedData(withRootObject: self.allTodo, requiringSecureCoding: false)
            UserDefaults.standard.set(data, forKey: "todoList")

            let data2 = try! NSKeyedArchiver.archivedData(withRootObject: self.selectedTodo, requiringSecureCoding: false)
            UserDefaults.standard.set(data2, forKey: "todoShare")

            //コピー
            self.todos = self.allTodo

            self.Table.reloadData()
        }

        let cancelaction = UIAlertAction(title: "キャンセル", style: .cancel) { (cancelaction) in
            print("canceled")
        }

        alert.addTextField { (alertTextField) in

            alertTextField.placeholder = "やることを記入してください"
            textField = alertTextField
        }

        alert.addAction(action)
        alert.addAction(cancelaction)
        present(alert, animated: true,completion: nil)

    }

    //swipe action
    func tableView(_ tableView: UITableView, trailingSwipeActionsConfigurationForRowAt indexPath: IndexPath) -> UISwipeActionsConfiguration? {

    // シェアのアクションを設定する
        let shareAction = UIContextualAction(style: .normal  , title: "完了") {
        (ctxAction, view, completionHandler) in
         print("シェアを実行する")

            //add cell to second case
            self.selectedTodo.append(self.allTodo[indexPath.row])

            //保存
            let data2 = try! NSKeyedArchiver.archivedData(withRootObject: self.selectedTodo, requiringSecureCoding: false)
            UserDefaults.standard.set(data2, forKey: "todoShare")

            completionHandler(true)
            self.Table.reloadData()
    }

//    // シェアボタンのデザインを設定する
//        let shareImage = UIImage(systemName: "arrowshape.turn.up.right.fill")?.withTintColor(UIColor.white, renderingMode: .alwaysTemplate)
//    shareAction.image = shareImage
    shareAction.backgroundColor = UIColor(red: 0/255, green: 125/255, blue: 255/255, alpha: 1)

    // 削除のアクションを設定する
        let deleteAction = UIContextualAction(style: .destructive, title:"削除") {
        (ctxAction, view, completionHandler) in

        //todosから要素を削除して保持しておく
        let deletedItem = self.todos.remove(at: indexPath.row)

        self.allTodo.removeAll {$0 === deletedItem}
        self.selectedTodo.removeAll {$0 === deletedItem}

        //削除した結果も保存
        let data = try! NSKeyedArchiver.archivedData(withRootObject: self.allTodo, requiringSecureCoding: false)
        UserDefaults.standard.set(data, forKey: "todoList")
        let data2 = try! NSKeyedArchiver.archivedData(withRootObject: self.selectedTodo, requiringSecureCoding: false)
        UserDefaults.standard.set(data2, forKey: "todoShare")

        tableView.deleteRows(at: [indexPath], with: .automatic)

        completionHandler(true)
    }

    // スワイプでの削除を無効化して設定する
    let swipeAction = UISwipeActionsConfiguration(actions:[deleteAction, shareAction])
    swipeAction.performsFirstActionWithFullSwipe = false

    return swipeAction
    }
}

class Item: NSObject,NSCoding{

    var title:String

    init(title:String) {
        self.title = title
    }

    func encode(with coder: NSCoder) {
        coder.encode(self.title,forKey: "title")
    }

    required init?(coder: NSCoder) {
        self.title = coder.decodeObject(forKey: "title") as! String
    }

}


こちらで削除ともに完成に近いものができたと思ったのですが

問題点

case0=allTodo
case1=selectedTodo

  1. share(完了)を押すとselectedTodoにコピーされるのだが同時にallTodoからデータを消したい
  2. selectedTodoでのshareが有効になっているためselectedTodoでShareのactionを無効にすることは可能ですか?
  3. 2でshareが有効になっているのでselectedTodoでshareを押すとselectedTodoにrowが追加されるわけです

gyazoによる問題の動画です>> gyazo.GIF

上記の動画のようにselectedTodoで3を複製した後3(row4)を削除するとallTodoにはrow4が存在していないため?かallTodoではrow3から削除されてしまいます。
こちらはどのように改善したら良いと思われますか?

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

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

回答 1

0

これで解決でますか?修正された上で追加の質問ということでしょうか?

投稿

編集

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2020/07/12 22:28

    修正した上で追加の質問と言うことです。
    返信が遅れてすみません。

    キャンセル

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

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

関連した質問

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