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

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

ただいまの
回答率

90.52%

  • Swift

    7224questions

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

  • Xcode

    4087questions

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

  • Firebase

    601questions

    Firebaseは、Googleが提供するBasSサービスの一つ。リアルタイム通知可能、並びにアクセス制御ができるオブジェクトデータベース機能を備えます。さらに認証機能、アプリケーションのログ解析機能などの利用も可能です。

セルにdictionaryの値を渡し、表示したい。 Firestore

解決済

回答 2

投稿 編集

  • 評価
  • クリップ 0
  • VIEW 207
退会済みユーザー

退会済みユーザー

ストーリーボード未使用で開発しています。

 実現したいこと

UITableViewControllerのcellForRowAt内のcell.textLabel?.textの値として、
コード①のdictionary内のvalueを持たせ、セルに表示させたい。

 分からない点

下記コード①をFirestoreのSetDataで保存しています。
ここのdictionary内のuserNameのvalueを、
UITableViewControllerのセルに表示したいです。

コード① Prof1です

import Foundation
import Firebase


protocol DocumentSerializable2 {

    init?(dictionary:[String:Any])
}


struct Prof1 {
    var userName :Any

    var dictionary:[String:Any] {
        return [

            "userName" : (userName as AnyObject).text ?? ""
        ]}
}



extension Prof1 : DocumentSerializable2 {
    init?(dictionary: [String:Any]) {

        guard let userName = dictionary["userName"] as? Any
            else {return nil}

        self.init(userName: userName)
    }
}

下記コード②でhoge.contentとhoge.timeStampはセルに表示できています。

この様に、コード①のuserNameも セルに表示したいのですが、
値のイニシャライズが分かりません。(let profff = Prof1()の書き方自体間違ってるかもしれませんが...)

コード② UITableViewControllerです

//省略してます


 override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCell(withIdentifier: "cell", for: indexPath)


         let hoge = hogeArray[hogeArray.count - indexPath.row - 1]

     //⚠️ここで値を渡したいですのですが、イニシャライズが分かりません。
     let profff = Prof1()        


        cell.textLabel?.text = "\(profff.userName) \(hoge.content)"
        cell.detailTextLabel?.text = "\(hoge.timeStamp)"

        return cell
    }

//省略してます

数日調べていますが、分からない為質問致します。
よろしくお願いします。

//追記

reloadDataもできています。

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

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

回答 2

checkベストアンサー

+1

addSnapshotListenerviewWillAppearで呼び出すのは変わらないですよね。
それで取得できたデータからProf1を作って、その作ったProf1cell.textLabel?.textを更新するだけです。

  var prof1 : Prof1?
   var dataListener : ListenerRegistration!

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

       let userRef = db.collection("users").document("user")

       dataListener = userRef.addSnapshotListener { [unowned self] (docSnapshot, error) in
           guard let docSnapshot = docSnapshot, docSnapshot.exists, else { return }

           if let prof1 = Prof1(dictionary: docSnapshot.data()) {
               self.prof1 = prof1
               self.tableView.reloadData()
           }
       }
   }

   override func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCell(withIdentifier: "cell", for: indexPath)

        let hoge = hogeArray[hogeArray.count - indexPath.row - 1]

        if let profff = self.prof1 {
            cell.textLabel?.text = "\(profff.userName) \(hoge.content)"
        }
        cell.detailTextLabel?.text = "\(hoge.timeStamp)"

        return cell
    }

 追記

userNameAnyだと取り出すことが出来なくなるので、普通にStringで持つようにしましょう。

struct Prof1 {
    var userName : String

    var dictionary:[String:Any] {
        return [

            "userName" : userName
        ]}
}



extension Prof1 : DocumentSerializable2 {
    init?(dictionary: [String:Any]) {

        guard let userName = dictionary["userName"] as? String
            else {return nil}

        self.init(userName: userName)
    }
}

投稿

編集

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2018/07/08 01:40 編集

    f-miyu様
    本当にありがとうございます。表示できました。userNameの値を表示するにはListenerRegistrationは必要だったのですね お手数おかけしました。

    分からない箇所があります。
    Optional(Huga)と表示されてしまうのですが、Hugaだけ表示したいのでOptionalを消すにはどうすれば良いのでしょうか??

    キャンセル

  • 2018/07/08 02:31

    回答に追記しました。

    キャンセル

  • 2018/07/08 13:53

    SetDataの時、テキストをUITextFieldに入力してSetDataをするので、
    struct Prof1のdictionaryは "userName" : (userName as AnyObject).text ?? ""としていました。
    Anyではなく、Stringだとどうしても自分ではエラーを出してしまいます。

    キャンセル

  • 2018/07/08 14:30

    何が問題なのかよくわからないのですが、SetDataする時でもUITextFieldのtextの値でdictionaryを作るだけだと思うのですが。

    キャンセル

  • 2018/07/08 22:46

    今の所AnyでもOptional()無しで表示できていますが、
    >>SetDataする時でもUITextFieldのtextの値でdictionaryを作る
    Stringで書けるように確認してみます。
    f-miyu様 本当にありがとうございます。助かりました!

    キャンセル

+1

cellForRowAtで取得したCellに対して値を渡せばいいのではないでしょうか。
addSnapshotListenerでは、取得した値をプロパティで受け取った後、reloadDataを呼び出し、テーブルの更新を行えばいいと思います。

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2018/07/07 17:21

    ありがとうございます。
    UIlabelではなくCellに対して値を渡したいと思います。

    >>addSnapshotListenerでは、取得した値をプロパティで受け取った後
    プロパティで受け取るというのは、どういうことでしょうか??docSnapshot.data()のdata()の中でしょうか?

    キャンセル

  • 2018/07/07 17:40 編集

    今回の場合、必要なデータはuserNameだけっぽいので、この値をUITableViewControllのプロパティとして持っておくということです。
    このプロパティを使って、cellForRowAt時に取得できるHogeCellのUIlabelを更新します。
    ちなみに、全てのセルに対して、同じuserNameを表示するということでいいんですよね?

    キャンセル

  • 2018/07/07 22:50

    f-miyu様 何度も申し訳ございません。

    大変恐れ入りますが、試行錯誤した結果UILabelではなく、tableviewCell内の cell.textLabel?.textに表示したいと思いました。
    この質問の本文を全て書き換えました。
    見ていただけないでしょうか?
    どうぞ、よろしくお願いいたします。

    キャンセル

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

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

関連した質問

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

  • Swift

    7224questions

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

  • Xcode

    4087questions

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

  • Firebase

    601questions

    Firebaseは、Googleが提供するBasSサービスの一つ。リアルタイム通知可能、並びにアクセス制御ができるオブジェクトデータベース機能を備えます。さらに認証機能、アプリケーションのログ解析機能などの利用も可能です。