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

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

ただいまの
回答率

90.12%

collectionViewCellで、xib生成されたあとに呼び出されるメソッド

解決済

回答 1

投稿 編集

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

KazutakaShimizu

score 158

現在iosアプリを作っているのですが、表題のようにcollectionViewCellで、xib生成されたあとに呼び出されるメソッドを教えていただきたいです。

やりたいこととしては、collectionViewのcellForItemAtメソッドからcell内の変数に値を渡して、その値をcellのラベルに表示するということなのですが、下記のようにcell内の変数のdidSetで処理を行うと、labelがnilの状態なのでエラーとなってしまいます。

//セルの処理
class MenuBarViewCell: UICollectionViewCell {
    @IBOutlet var label: UILabel!
    var isCompletedSetUpUI: Bool = false
    var title:String!{
        didSet{
            label.text = title
        }
    }
}
//collectionViewの処理
class MenuBarView: UIView {
    var pageTabItemsWidth: CGFloat = 0.0
    var collectionView:UICollectionView!
    var titles: [String] = Constant.menuTitle
    override init(frame: CGRect) {
        super.init(frame: frame)
        setUpCollectionView()
    }

    required init?(coder aDecoder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }
}

extension MenuBarView: UICollectionViewDelegate, UICollectionViewDataSource, UICollectionViewDelegateFlowLayout {
    fileprivate func setUpCollectionView(){
        let layout = UICollectionViewFlowLayout()
        layout.itemSize = CGSize(width:100, height:30)
        layout.scrollDirection = .horizontal
        collectionView = UICollectionView(frame: CGRect(x: 0, y: 0, width: frame.size.width, height: frame.size.height), collectionViewLayout: layout)
        // Cellに使われるクラスを登録.
        collectionView.delegate = self
        collectionView.dataSource = self
        collectionView.showsHorizontalScrollIndicator = false
        collectionView.register(MenuBarViewCell.self, forCellWithReuseIdentifier: "MyCell")
        self.addSubview(collectionView)

    }

    func collectionView(_ collectionView: UICollectionView, numberOfItemsInSection section: Int) -> Int {
        return titles.count * 3
    }

    /*
     Cellに値を設定する
     */
    func collectionView(_ collectionView: UICollectionView, cellForItemAt indexPath: IndexPath) -> UICollectionViewCell {
        let cell : MenuBarViewCell = collectionView.dequeueReusableCell(withReuseIdentifier: "MyCell",for: indexPath) as! MenuBarViewCell
        configureCell(cell: cell, indexPath: indexPath)
        return cell
    }

    private func configureCell(cell:MenuBarViewCell, indexPath: IndexPath){
        let fixedIndex = indexPath.item % titles.count
        cell.title = titles[fixedIndex]
    }
}

extension MenuBarView: UIScrollViewDelegate {
    func scrollViewDidScroll(_ scrollView: UIScrollView) {
        if pageTabItemsWidth == 0.0 {
            pageTabItemsWidth = floor(scrollView.contentSize.width / 3.0) // 表示したい要素群のwidthを計算
        }

        if (scrollView.contentOffset.x <= 0.0) || (scrollView.contentOffset.x > pageTabItemsWidth * 2.0) { // スクロールした位置がしきい値を超えたら中央に戻す
            scrollView.contentOffset.x = pageTabItemsWidth
        }
    }
}

イニシャライザで行っても同様です。

どなたかご回答いただけると助かります。

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

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

質問への追記・修正、ベストアンサー選択の依頼

  • fuzzball

    2017/06/15 10:09

    nilになること自体がおかしいです。セルの登録、セルの取得(生成)、titleの設定、以上のコードを書いて下さい。

    キャンセル

  • KazutakaShimizu

    2017/06/15 15:56

    なるほど、ありがとうございます。collectionViewの処理を追記したのでご確認いただけると幸いです。

    キャンセル

回答 1

checkベストアンサー

0

セルの登録をするところ、

collectionView.register(MenuBarViewCell.self, forCellWithReuseIdentifier: "MyCell")

これだとxibが読み込まれていないと思うので、

let nib = UINib(nibName: String(describing: MenuBarViewCell.self), bundle: nil)
collectionView.register(nib, forCellWithReuseIdentifier: "MyCell")

に変更してみて下さい。

ところで、なぜStoryboardではなくxibを使っているのでしょうか?

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2017/06/15 17:56

    ありがとうございます。
    ご指摘いただいた点を修正したところ、問題が解決いたしました。

    story boardを使わない理由ですが、一緒に開発しているエンジニアからstory boardは開発に関わる人が増えれば増えるほど不便になると聞いたため、それでは最初から使わず開発しようと思ったためです。
    逆に何かstory boardを使うことはどういうったメリットがあるのでしょうか?

    キャンセル

  • 2017/06/15 18:07

    前回の質問、今回の質問ともに、Storyboardを使っていれば起こらないバグだったので聞いてみただけです。

    キャンセル

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

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