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

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

新規登録して質問してみよう
ただいま回答率
85.49%
iOS

iOSとは、Apple製のスマートフォンであるiPhoneやタブレット端末のiPadに搭載しているオペレーションシステム(OS)です。その他にもiPod touch・Apple TVにも搭載されています。

Xcode

Xcodeはソフトウェア開発のための、Appleの統合開発環境です。Mac OSXに付随するかたちで配布されています。

Swift

Swiftは、アップルのiOSおよびOS Xのためのプログラミング言語で、Objective-CやObjective-C++と共存することが意図されています

iPhone

iPhoneとは、アップル社が開発・販売しているスマートフォンです。 同社のデジタルオーディオプレーヤーiPodの機能、電話機能、インターネットやメールなどのWeb通信機能の3つをドッキングした機器です。

Q&A

解決済

1回答

1412閲覧

二フクラを使ってXcodeでSNSの投稿削除ボタンを作りたい。

suma0402

総合スコア9

iOS

iOSとは、Apple製のスマートフォンであるiPhoneやタブレット端末のiPadに搭載しているオペレーションシステム(OS)です。その他にもiPod touch・Apple TVにも搭載されています。

Xcode

Xcodeはソフトウェア開発のための、Appleの統合開発環境です。Mac OSXに付随するかたちで配布されています。

Swift

Swiftは、アップルのiOSおよびOS Xのためのプログラミング言語で、Objective-CやObjective-C++と共存することが意図されています

iPhone

iPhoneとは、アップル社が開発・販売しているスマートフォンです。 同社のデジタルオーディオプレーヤーiPodの機能、電話機能、インターネットやメールなどのWeb通信機能の3つをドッキングした機器です。

0グッド

0クリップ

投稿2019/07/28 02:14

問題

二フクラを使ってXcodeでSNSの投稿削除ボタンを作りたい。

カスタムセルを使って、テーブルビューを作っています。
カスタムセルの中に削除ボタンを押すと削除画面に進み、
二フクラのデータベースから指定のデータを削除できる仕様を考えています。

そのためにテーブルビューのコントローラでカスタムセルのコントローラーに

swift

1timeLineCell.deleteButton.tag = indexPath.row 2

でobjectIdを渡して

カスタムセルで

ViewController

1 2 @IBAction func deleteButton(_ sender: Any) { 3 4 let vc = self.storyboard?.instantiateViewController(withIdentifier: "deleteViewController") as! deleteViewController 5 6 vc.objectId = self.deleteButton.tag 7 8 self.show(vc, sender: nil) 9 10 } 11

として削除画面に
objectIdを持って遷移しようと考えているのですが

"Value of type 'cusTableViewCell' has no member 'storyboard'"

というエラーが出て遷移できません。
セルにはstorybboardが無いという意味だとは思い、
テーブルのあるコントローラーからの遷移を考えたのですが、
今度はカスタムセルのボタンのアクションの設定の仕方がどうしてもわかりません。

どちらの方法でもいいのでobjectIDの変数を持って画面遷移できたらと考えています。
以下コードです。

swift

1import UIKit 2 3class ViewController: UIViewController,UITableViewDelegate,UITableViewDataSource { 4 5 @IBOutlet var timeLineTable: UITableView! 6 7 var data = [NCMBObject]() 8 9 var imgData = [String:Data]() 10 11 var refreshControl = UIRefreshControl() 12 13 override func viewDidLoad() { 14 super.viewDidLoad() 15 // Do any additional setup after loading the view. 16 17 self.timeLineTable.delegate = self 18 self.timeLineTable.dataSource = self 19 20 let xib = UINib(nibName: "cusTableViewCell", bundle: nil) 21 self.timeLineTable.register(xib, forCellReuseIdentifier: "cusTableViewCell") 22 23 self.refreshControl.addTarget(self, action: #selector(ViewController.refreshData), for: .valueChanged) 24 self.timeLineTable.addSubview(self.refreshControl) 25 26 27 28 } 29 30 override func viewWillAppear(_ animated: Bool) { 31 // mBaasからデータを取得してくれる人を呼んでくる 32 33 let query = NCMBQuery(className: "timeLine") 34 35 query?.order(byDescending: "createDate") 36 37 query?.findObjectsInBackground({ (values, error) in 38 39 if error == nil { 40 41 self.data = values as! [NCMBObject] 42 43 self.timeLineTable.reloadData() 44 } 45 }) 46 } 47 48 49 func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int { 50 51 return self.data.count 52 53 } 54 55 56 func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell { 57 58 59 let timeLineCell = self.timeLineTable.dequeueReusableCell(withIdentifier: "cusTableViewCell", for: indexPath) as! cusTableViewCell 60 61 timeLineCell.timeLineText.text = self.data[indexPath.row].object(forKey: "timeLineMessage")as? String 62 63 64 let file = NCMBFile.file(withName: "(self.data[indexPath.row].objectId!).jpg", data: nil) as! NCMBFile 65 66 timeLineCell.timeLineImageView.image = nil 67 68 timeLineCell.deleteButton.tag = indexPath.row 69 70 71 let id = self.data[indexPath.row].objectId! 72 if self.imgData[id] != nil { 73 timeLineCell.timeLineImageView.image = UIImage(data: self.imgData[id]!) 74 return timeLineCell 75 } 76 77 // データを取得してもらう 78 file.getDataInBackground { (data, error) in 79 80 if error == nil { 81 82 timeLineCell.timeLineImageView.image = UIImage(data: data!) 83 84 self.imgData[self.data[indexPath.row].objectId!] = data! 85 } 86 } 87 88 return timeLineCell 89 } 90 91 92 93 func tableView(_ tableView: UITableView, heightForRowAt indexPath: IndexPath) -> CGFloat { 94 // 300でお願いします 95 return 300 96 } 97 98 @objc func refreshData() { 99 100 let query = NCMBQuery(className: "timeLine") 101 102 query?.order(byDescending: "createDate") 103 104 query?.findObjectsInBackground({ (values, error) in 105 106 self.refreshControl.endRefreshing() 107 108 self.data = [NCMBObject]() 109 self.imgData = [String:Data]() 110 111 if error == nil { 112 113 self.data = values as! [NCMBObject] 114 115 116 self.timeLineTable.reloadData() 117 } 118 }) 119 } 120 121 122 123 124// @IBAction func deleteButton(sender: AnyObject) { 125// // testクラスへのNCMBObjectを設定 126// let obj6 = NCMBObject(className: "timeLine") 127// // objectIdプロパティを設定 128// obj6?.objectId = self.deleteButton.tag 129// // 設定されたobjectIdを元にデータストアからデータを取得 130// obj6?.fetchInBackground({ (error) in 131// if error != nil { 132// // 取得に失敗した場合の処理 133// }else{ 134// // 取得に成功した場合の処理 135// obj6?.deleteInBackground({ (error) in 136// if error != nil { 137// // 削除に失敗した場合の処理 138// }else{ 139// // 削除に成功した場合の処理 140// let query = NCMBQuery(className: "timeLine") 141// 142// query?.order(byDescending: "createDate") 143// 144// query?.findObjectsInBackground({ (values, error) in 145// 146// self.refreshControl.endRefreshing() 147// 148// self.data = [NCMBObject]() 149// self.imgData = [String:Data]() 150// 151// if error == nil { 152// 153// self.data = values as! [NCMBObject] 154// 155// 156// self.timeLineTable.reloadData() 157// } 158// }) 159// 160// } 161// }) 162// } 163// }) 164// 165// } 166 167 168}

cusTableViewCell

1 2import UIKit 3 4class cusTableViewCell: UITableViewCell { 5 6 @IBOutlet var timeLineImageView: UIImageView! 7 8 var openBrowser: ((String)->())! 9 10 @IBOutlet var timeLineText: UITextView! 11 12 @IBOutlet var deleteButton: UIButton! 13 14 override func awakeFromNib() { 15 super.awakeFromNib() 16 // Initialization code 17 } 18 19 override func setSelected(_ selected: Bool, animated: Bool) { 20 super.setSelected(selected, animated: animated) 21 22 // Configure the view for the selected state 23 } 24 25 @IBAction func deleteButton(_ sender: Any) { 26 27 let vc = self.storyboard?.instantiateViewController(withIdentifier: "deleteViewController") as! deleteViewController 28 29 vc.objectId = self.deleteButton.tag 30 31 self.show(vc, sender: nil) 32 33 } 34 35}

ご教授いただけらありがたいです。
よろしくお願いいたします。

イメージ説明

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

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

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

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答1

0

ベストアンサー

TableViewのあるViewController側からの遷移という考え方は合ってます。
この場合,セルの中にあるボタンをタップした場合の処理をViewController側に移譲すれば良いです。

swift

1// セル側の処理 2import UIKit 3 4// プロトコルを実装 5protocol DeleteButtonDelegate: class { 6 // 削除ボタンをタップした際にdelegateメソッドを実行 7 func moveToDeleteScreen(tag: Int) 8} 9 10 11class cusTableViewCell: UITableViewCell { 12 13 weak var deleteButtonDelegate: DeleteButtonDelegate? 14 15 // 略 16 17 @IBAction func deleteButton(_ sender: Any) { 18 // 削除ボタンがタップされた際の処理を移譲 19 self.deleteButtonDelegate.moveToDeleteScreen(tag: self.deleteButton.tag) 20 } 21}

swift

1// ViewController側の処理 2class ViewController: UIViewController,UITableViewDelegate,UITableViewDataSource { 3 4 // 略 5 6 func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell { 7 8 let timeLineCell = self.timeLineTable.dequeueReusableCell(withIdentifier: "cusTableViewCell", for: indexPath) as! cusTableViewCell 9 10 // ボタンを押した際の処理を移譲されるの私です 11 timeLineCell.deleteButtonDelegate = self 12 13 // 略 14 15 return timeLineCell 16 } 17} 18 19extension ViewController: DeleteButtonDelegate { 20 21 /// 削除ボタンが押された時の処理を実行 22 func moveToDeleteScreen(tag: Int) { 23 let vc = self.storyboard?.instantiateViewController(withIdentifier: "deleteViewController") as! deleteViewController 24 vc.objectId = tag 25 self.show(vc, sender: nil) 26 } 27}

こんな感じでいかがでしょうか?

投稿2019/07/28 03:02

編集2019/07/28 04:10
TakuyaAso

総合スコア1361

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

suma0402

2019/07/28 03:21

回答ありがとうございます 以下のエラーが出てしまい動きませんでした。 extension ViewController: DeleteButtonDelegate部分に Declaration is only valid at file scope vc.objectId = tag部分に Value of type 'deleteViewController' has no member 'objectId' protocol等、 まだ勉強しきれていないところなので、自分でも再度調べさせてもらいます!
TakuyaAso

2019/07/28 04:13

ViewController の extension なので class の中に書いてはいけません。 回答を少し修正しました。class {} の外に出してみてください。 DeleteViewController.swift の方に下記のプロパティを定義すれば値渡しできると思いますー class DeleteViewController: UIViewController { var objectId: Int? // 略 }
suma0402

2019/07/28 04:37 編集

xcodeでエラーが出ずにbuild出来るようになりました! ただ削除ボタンを押すと AppDelegate.swiftファイルで class AppDelegate: UIResponder, UIApplicationDelegate { Thread 1: signal SIGABRT とエラーが出るのですが これはボタンが うまく変数に設定しきれてないという事なのでしょうか? 今までも何度かこのエラーが出てきていて、 その度にググっては分からず作り直しを繰り返しており悩んでいます。 少し質問内容とずれている気がしますが もしよろしかったら回答行っただけたら幸いです。
TakuyaAso

2019/07/28 04:49

Xcodeの下のデバッグエリア(ログがみれる)ところのエラー内容がわかればという感じですね。 Reason とかで検索すると引っかかると思うのですがどんなエラーでしょう?
suma0402

2019/07/28 10:35

返信ありがとうございます。 そのような調べ方があるとは知りませんでした。。 下記のようなエラーが出ました。 2019-07-28 19:33:49.800983+0900 pictweet2[35417:2003243] *** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: 'Storyboard (<UIStoryboard: 0x600003fc4400>) doesn't contain a view controller with identifier 'deleteViewController'' (lldb)
TakuyaAso

2019/07/28 14:14

現在のStoryboardにDeleteViewControllerの画面がないか,Storyboard ID が間違っているかのエラーですね。確認してみてください。参考記事はこちらです(最後のエラーの説明が参考になるかと)。 https://qiita.com/colorrabbit/items/c8cb6f815e6dc0b6beba
suma0402

2019/07/29 10:57

Storyboard ID が間違っているのが原因でした。 初歩的なミスでした。 修正すると無事変数を持って遷移しました。 ありがとうございます!
TakuyaAso

2019/07/29 11:48

おお,解決して良かったです!
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.49%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問