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

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

ただいまの
回答率

90.12%

delegate swift nil

解決済

回答 1

投稿

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

sr2000

score 30

tableViewのセルをカスタムセルで作成して、ショッピングサイトのように1つのセルの行にViewを左右に2つ並べてセルのタップイベントをtableView.allowsSelection = falseとして無効にしています。
そこでviewをタップしたときに画面遷移をしたくてUITapGestureRecognizerを作成してselecotorメソッドで画面遷移をしたいのですが、viewをoutlet接続しているのが、tableViewCellクラスなのでpresentなどのメソッドが使えないので、delegateでmainViewControllerに処理を委託したく実装したのですが、delegateメソッドが実行されません。
自分が考えるにSecondViewControllerのviewDidLoad内のdelegateの委託方法だと思いましていろいろ試しましたが、セレクターメソッドのニルがプリントされてしまいます。
どなたかアドバイスいただけませんか?実際のコードを載せます。宜しくおねがいします。

import UIKit

class SecondViewController: UIViewController, TestDelegate {

    let testData: [String] = ["a","b","c","d","e","f","g","h","i","j"]



    @IBOutlet weak var tableView: UITableView!



    override func viewDidLoad() {
        super.viewDidLoad()

        tableView.delegate = self
        tableView.dataSource = self

        //タッチイベントを削除
        tableView.allowsSelection = false

        //ここにdelegateの設定したら動くはず!!
        let a = SecondTableViewCell()
        a.testDelegate = self
        let q = UITableViewCell()
        q.addSubview(a)


        //self.view.addSubview(a)
        //self.tableView.addSubview(a)


        // Do any additional setup after loading the view.
    }

    func test() {
        print("いつになったら実装できますか?")
    }


    /*
    // MARK: - Navigation

    // In a storyboard-based application, you will often want to do a little preparation before navigation
    override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
        // Get the new view controller using segue.destination.
        // Pass the selected object to the new view controller.
    }
    */

}

extension SecondViewController: UITableViewDelegate, UITableViewDataSource {

//    func numberOfSections(in tableView: UITableView) -> Int {
//        return 1
//    }

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


    //ここを設定してなかったからデータの表示がされてなかっただけ!!
    func tableView(_ tableView: UITableView, heightForRowAt indexPath: IndexPath) -> CGFloat {
        return 200
    }

    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCell(withIdentifier: "second", for: indexPath) as! SecondTableViewCell ?? SecondTableViewCell()

        let left = indexPath.row * 2
        let right = left + 1

        cell.leftLabel.text = testData[left]
        cell.rightLabel.text = testData[right]
        return cell

    }

}
import UIKit

class SecondTableViewCell: UITableViewCell {


    @IBOutlet weak var leftView: UIView!

    @IBOutlet weak var leftLabel: UILabel!

    @IBOutlet weak var rightView: UIView!

    @IBOutlet weak var rightLabel: UILabel!

    var testDelegate: TestDelegate?



    override func awakeFromNib() {
        super.awakeFromNib()
        // Initialization code


        //ジェスチャ作成
        let myLeftTap: UITapGestureRecognizer = UITapGestureRecognizer(target: self, action: #selector(changeLeftView(sender:)))
        let myRightTap: UITapGestureRecognizer = UITapGestureRecognizer(target: self, action: #selector(changeRightView(sender:)))


        //Viewに追加
        self.leftView.addGestureRecognizer(myLeftTap)
        self.rightView.addGestureRecognizer(myRightTap)



    }

    override func setSelected(_ selected: Bool, animated: Bool) {
        super.setSelected(selected, animated: animated)

        // Configure the view for the selected state
    }


    //タップ処理
    @objc func changeLeftView(sender: UITapGestureRecognizer) {

        if let dbg = testDelegate {
            print("左通過")
            dbg.test()
        } else {
            print("ニル")
        }
    }


    //タップ処理
    @objc func changeRightView(sender: UITapGestureRecognizer) {
        print("右セレクター通過")
    }

}


protocol TestDelegate: AnyObject {
    func test()
}
  • 気になる質問をクリップする

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

回答 1

checkベストアンサー

0

こうだと思います

func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
    let cell = tableView.dequeueReusableCell(withIdentifier: "second", for: indexPath) as! SecondTableViewCell ?? SecondTableViewCell()

    cell. testDelegate = self

//省略

    return cell
}

//ここにdelegateの設定したら動くはず!!
の部分は不要です

(以下おまけ)

クラス、インスタンス、継承周りで認識違いがあると思います。
クラスの定義として「SecondTableViewCellのアクションは常にSecondViewControllerで行う」という方法は、その方法ではできません(storyboardを使えば可能ですがあまり一般的ではないと思います)
dequeueしたcell(SecondTableViewCellのインスタンス)で起きたイベントをviewController側に移譲しするようにして使います。

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2018/11/16 22:54

    丁寧な回答ありがとうございます。
    特に(以下おまけ)がすごく参考になりました。
    悩んでいたことが解決できました。

    キャンセル

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

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