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

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

ただいまの
回答率

90.04%

Container Viewに入れたTable View Controllerを外からreloadData()したい

解決済

回答 1

投稿

  • 評価
  • クリップ 2
  • VIEW 5,286
退会済みユーザー

退会済みユーザー

前提・問題点

現在、ContainerViewにTableViewControllerを入れて、
テーブルにデータを表示しています。

ContainerViewを入れているViewControllerのボタンを押して、

ContainerViewの中のTableViewControllerに
tableView.reloadData()の命令を送りたいのですが、
まったくうまくいきません。

試したこと

Reload TableViewController from parent
このページなどを参考に色々試してみましたが、
Swiftに対する知識不足から、まったく上手くいきませんでした。

実現したいこと

想像としては、親ViewControllerの値を監視して、値が変化したら子TableViewControllerがtableView.reloadData()を実行する、というようなことが実現できたらと思います。

他にもっと良い方法があれば、その方法もぜひ教えていただきたいです。

どうかよろしくお願いします。

補足情報(言語/FW/ツール等のバージョンなど)

Xcode 7.3
Swift 2

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

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

回答 1

checkベストアンサー

+2

試していた通知の方法でも大丈夫かと思いますが、、、

自分が試してうまくいったコードを載せて起きます。

ViewControllerを作成するときにprepareForSegueが呼ばれるので、そこでTableViewControllerのインスタンスを自身に保持するやり方です。

import UIKit

class ViewController: UIViewController {

    var tableViewController: TableViewController!

    override func viewDidLoad() {
        super.viewDidLoad()

    }

    @IBAction func containerTableReload(sender: UIButton) {
        tableViewController.dataArray += ["Sample Data"]
        tableViewController.reloadTable()
    }

    override func prepareForSegue(segue: UIStoryboardSegue, sender: AnyObject?) {

        if let tableVC = segue.destinationViewController as? TableViewController {
            // TableViewController Instance を保持
            self.tableViewController = tableVC
        }
    }
}

// --------------------------------------

class TableViewController: UITableViewController {

    override func viewDidLoad() {
        super.viewDidLoad()

        tableView.estimatedRowHeight = 20
        tableView.rowHeight = UITableViewAutomaticDimension
    }

    func reloadTable()  {
        tableView.reloadData()
    }

    // Data Array
    var dataArray = ["Sample Data"]

    // MARK: - TableView Delegate & DataSource

    // Row Count
    override func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return dataArray.count
    }

    // Generate Cell
    override func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCellWithIdentifier("Cell", forIndexPath: indexPath)
        cell.textLabel?.numberOfLines = 0
        cell.textLabel?.text = dataArray[indexPath.row]
        return cell
    }
}

s

Containerの子配列からオブジェクトを取得する場合  

import UIKit

class ViewController: UIViewController {

    var tableViewController: TableViewController!

    override func viewDidLoad() {
        super.viewDidLoad()

        // Containerの子配列からTableViewControllerオブジェクトを取得
        tableViewController = self.childViewControllers[0] as! TableViewController
    }

    @IBAction func containerTableReload(sender: UIButton) {

        tableViewController.dataArray += ["Sample Data"]
        tableViewController.reloadTable()
    }
}

// --------------------------------------

class TableViewController: UITableViewController {

    override func viewDidLoad() {
        super.viewDidLoad()

        tableView.estimatedRowHeight = 20
        tableView.rowHeight = UITableViewAutomaticDimension
    }

    func reloadTable()  {
        tableView.reloadData()
    }

    // Data Array
    var dataArray = ["Sample Data"]

    // MARK: - TableView Delegate & DataSource

    // Row Count
    override func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return dataArray.count
    }

    // Generate Cell
    override func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCellWithIdentifier("Cell", forIndexPath: indexPath)
        cell.textLabel?.numberOfLines = 0
        cell.textLabel?.text = dataArray[indexPath.row]
        return cell
    }
}

通知を使用する場合

import UIKit

class ViewController: UIViewController {

    override func viewDidLoad() {
        super.viewDidLoad()

    }

    @IBAction func containerTableReload(sender: UIButton) {
        // 通知を送信
        NSNotificationCenter.defaultCenter().postNotificationName("TableReloadNotification", object: ["Sample Data"])
    }
}

// --------------------------------------

class TableViewController: UITableViewController {

    override func viewDidLoad() {
        super.viewDidLoad()

        // 通知を登録
        NSNotificationCenter.defaultCenter().addObserver(self, selector: #selector(self.reloadTable(_:)), name: "TableReloadNotification", object: nil)

        tableView.estimatedRowHeight = 20
        tableView.rowHeight = UITableViewAutomaticDimension
    }

    deinit {
        // 通知を解除
        NSNotificationCenter.defaultCenter().removeObserver(self)
    }

    func reloadTable(notification: NSNotification)  {
        let data = notification.object as! [String]
        dataArray += data
        tableView.reloadData()
    }

    // Data Array
    var dataArray = ["Sample Data"]

    // MARK: - TableView Delegate & DataSource

    // Row Count
    override func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return dataArray.count
    }

    // Generate Cell
    override func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCellWithIdentifier("Cell", forIndexPath: indexPath)
        cell.textLabel?.numberOfLines = 0
        cell.textLabel?.text = dataArray[indexPath.row]
        return cell
    }
}

投稿

編集

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2016/08/07 07:53

    _Kentarouさん、ありがとうございます。

    とてもきれいに動くようになりました!

    通知の方法も理解できるように、
    Swiftを勉強していきたいと思います。

    わかりやすい回答、ありがとうございました、

    キャンセル

  • 2016/08/07 07:55 編集

    Containerの子配列からオブジェクトを取得する方法と、通知を使用した場合のコードも回答に追記しました、参考にしてみてください。

    キャンセル

  • 2016/08/07 17:41

    _Kentarouさん、ありがとうございます!
    わかりやすいコードの例は、本当に助かります。
    「オブジェクトを取得」や「通知」は、できるようになると
    自由度が広がると思うので、このコード例を参考に
    じっくりと勉強したいと思います。
    本当にありがとうございました。

    キャンセル

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

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