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

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

ただいまの
回答率

90.86%

  • Swift

    6297questions

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

  • Xcode

    3630questions

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

作成した関数がCannot use instance memberで使えない

解決済

回答 1

投稿 編集

  • 評価
  • クリップ 0
  • VIEW 1,180

kifu

score 18

日付の配列が空の場合には、当日の日付を配列に加えるという関数を作成し使おうとしたところ、下記コードのエラー部分で次のエラーメッセージが出てしまいました。プレイグランドではちゃんと動きます。どのようにすれば良いのでしょうか。ご教示いただけますか。
Cannot use instance member '...' within property initializer; property initializers run before 'self' is available

import UIKit

class ViewController : UIViewController, UITableViewDelegate, UITableViewDataSource {



    //今日と日付の配列の最後の日差分の日付を配列にするメソッド
    func getDaysArrayToToday(start:String,max:Int) -> [String] {


        var result:[String] = []
        let formatter = DateFormatter()
        formatter.locale = NSLocale(localeIdentifier: "ja_JP") as Locale
        formatter.dateFormat = "yyyy-MM-dd"
        // 今日
        let todayStr = formatter.string(from: Date())

        // 配列中の最大日が当日以下の場合は終了
        if todayStr <= start {
            return result
        }

        let startDate = formatter.date(from: start)!
        var components = DateComponents()
        let calendar = Calendar(identifier: Calendar.Identifier.gregorian)

        //for i in 0 ..< max { 当日を含まないようにした
        for i in 1 ..< max {
            components.setValue(i,for: Calendar.Component.day)
            let wk = calendar.date(byAdding: components, to: startDate)!
            let wkStr = formatter.string(from: wk)
            if wkStr > todayStr {
                break
            } else {

                result.append(wkStr)
            }
         }

        return result
    }
    //テーブルビューのsectionに表示する日付の配列
//    var sectionTitle = ["2016-01-23","2015-12-31","2015-01-12","2016-02-21","2016-12-20"]
    var preSectionTitle:[String] = []

//作成した関数   
    func getSectionTitle() -> Array<String>? {
        let formatter = DateFormatter()
        formatter.locale = NSLocale(localeIdentifier: "ja_JP") as Locale
        formatter.dateFormat = "yyyy-MM-dd"
        let todayStr = formatter.string(from: Date())

        if preSectionTitle.isEmpty{
            preSectionTitle.append(todayStr)}
        return preSectionTitle
    }

**__//エラー部分__**
    var sectionTitle = getSectionTitle()


    //今日と日付の配列の最後の日差分の日付配列をテーブルビューのsectionに表示する

    func numberOfSections(in tableView: UITableView) -> Int // Default is 1 if not implemented
    {
        return sectionTitle.count
    }
    ////
    func tableView(_ tableView: UITableView, titleForHeaderInSection section: Int) -> String? // fixed font style. use custom view (UILabel) if you want something different
    {
        return sectionTitle[section]
    }


    //
    let statusBarHeight = UIApplication.shared.statusBarFrame.height

    //セクションの項目



    // チェックリストの項目とチェック状態
    var checkListItem1: [String : Bool] = [
        "アイテム1" : true,
        "アイテム2" : false,
        "アイテム3" : true,
        "アイテム4" : true,
        "アイテム5" : false
    ]

    // 新規追加日用
    var checkListItem: [String : Bool] = [
        "###アイテム1" : false,
        "###アイテム2" : false,
        "###アイテム3" : false,
        "###アイテム4" : false,
        "###アイテム5" : false
    ]

    //
    var tableData: [[String: Bool]] = []

    let tableView = UITableView()

    //
    override func viewDidLoad() {
        super.viewDidLoad()

        let lastLaunch = sectionTitle.max()!

//        tableData = [checkListItem1, checkListItem2, checkListItem3, checkListItem4, checkListItem5]
        tableData = [checkListItem1]

        let appendDays = getDaysArrayToToday(start: lastLaunch, max: 1200)

        // 追加された日付分のcheckListItemをtableDataに追加
        for _ in 0..<appendDays.count {
            tableData.insert(checkListItem, at: 0)
        }
        //修正後発生したエラー箇所
        sectionTitle.append(contentsOf: appendDays)

        //    sectionを日付降順にソートする
        sectionTitle = sectionTitle.sorted { $0 > $1 }

        //tableData = [checkListItem1, checkListItem2, checkListItem3, checkListItem4, checkListItem5] 上に移動

        // UITableView の作成
        tableView.frame = CGRect(
            x: 0,
            y: statusBarHeight,
            width: self.view.frame.width,
            height: self.view.frame.height - statusBarHeight
        )
        tableView.delegate = self
        tableView.dataSource = self


        self.view.addSubview(tableView)
    }

    // セルの作成
    //
    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {

        // tableDataの中から抽出
        let sectionData = tableData[indexPath.section]

        // キーで並び替え
        let keys = sectionData.keys.sorted()

        // キーの文字列を取得
        let cellText = keys[indexPath.row]

        // セルの作成とテキストの設定
        let cell = UITableViewCell(style: .default, reuseIdentifier: "cell")
        cell.textLabel?.text = cellText

        ///
        let cellIscheckd = sectionData[cellText]

        // チェック状態が true なら、初めからチェック状態にする
        if cellIscheckd == true {
            cell.imageView?.image = UIImage(named: "checked")
        } else {
            cell.imageView?.image = UIImage(named: "unchecked")
        }

        return cell
    }

    // セルがタップされた時の処理
    func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {

        if let cell = tableView.cellForRow(at: indexPath) {

            // タップしたセルのテキストを取得
            let cellText = cell.textLabel?.text ?? ""

            // 画像を切り替えと Dictonary の値を変更
            if cell.imageView?.image == UIImage(named: "checked") {

                self.tableData[indexPath.section][cellText] = false

                cell.imageView?.image = UIImage(named: "unchecked")
            } else {

                self.tableData[indexPath.section][cellText] = true
                cell.imageView?.image = UIImage(named: "checked")
            }

            // 選択状態を解除
            cell.isSelected = false
        }
    }


    func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        let sectionData = tableData[section]
        return sectionData.count
    }
}
  • 気になる質問をクリップする

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

回答 1

checkベストアンサー

+1

インスタンス変数の初期化時に、他のインスタンス変数の値を使用することはできません。
※ getSectionTitle()メソッドの中でpreSectionTitleにアクセスしている

対応方法は以下の2つ

① lazyプロパティにすることで遅延して初期化することができます。
参考URL: 【Swift】レイジープロパティ(lazy)の使い方。最初にアクセスされたときに初期値が決まる。

lazy var sectionTitle: Array<String>? = self.getSectionTitle()

② viewDidLoad()の中で初期化する

var sectionTitle: Array<String>?

    override func viewDidLoad() {
    super.viewDidLoad()

    sectionTitle = getSectionTitle()

}

返却される値がオプショナルなのでその他の場所で!?を付けないといけなくなりますね。

投稿

編集

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2017/01/07 21:17 編集

    御回答に感謝します。①②の方法で修正し必要箇所に!?をつけたところ、上記コードに追記した//修正後発生したエラー箇所「sectionTitle.append(contentsOf: appendDays)」に
    「Value of type 'Array<String>?' has no member 'append'」
    というエラーメッセージが①②とも出ました。続いての質問で恐縮ですがご教示いただけますか。

    キャンセル

  • 2017/01/07 20:46

    !をつければ問題ないと思いますが、そもそもsectionTitleはオプショナルになる可能性がないのでArray<String>?の?を外せば良いと思います。

    キャンセル

  • 2017/01/07 21:23

    Array<String>?の?を外したところ解決しました。心から感謝します。

    キャンセル

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

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

関連した質問

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

  • Swift

    6297questions

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

  • Xcode

    3630questions

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