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

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

ただいまの
回答率

90.36%

  • Swift

    8196questions

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

  • Swift 2

    1336questions

    Swift 2は、Apple社が独自に開発を行っている言語「Swift」のアップグレード版です。iOSやOS X、さらにLinuxにも対応可能です。また、throws-catchベースのエラーハンドリングが追加されています。

Alamofire内でTabPageViewControllerを呼ぶと、一瞬表示されない

解決済

回答 3

投稿 編集

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

Y_M

score 243

前提・実現したいこと

遷移した段階でTabPageViewControllerを表示させたい。

発生している問題・エラーメッセージ

Alamofireの処理の中にTabPageViewControllercreate()以降の処理を記述すると
タブをスワイプするなどの動作をしないと設定したタブバーが表示されない。

★通信結果によってはじめに表示されるタブを変更したい。
★受け取ったjson["result"]の結果がnilの場合は0番めのタブを。
★受け取ったjson["result"]の結果がnilの場合は1番めのタブを。

現在の構造は
viewDidLoad
−−check()
−−−tabPageSetting()

該当のソースコード

import UIKit
import Alamofire
import SwiftyJSON
import TabPageViewController

class ViewController: UIViewController {

    let server: String = "http://sample.jp"
    let appDelegate = UIApplication.sharedApplication().delegate as! AppDelegate

    override func viewDidLoad() {
        super.viewDidLoad()

        self.navigationItem.title = "テスト"
        self.navigationController!.interactivePopGestureRecognizer!.enabled = false

        check()

    }

    func tabPageSetting(setIndex: Int) {

        let tabPageViewController = TabPageViewController.create()
        let vc1 = self.storyboard?.instantiateViewControllerWithIdentifier("Tab1") as! Tab1ViewController
        let vc2 = self.storyboard?.instantiateViewControllerWithIdentifier("Tab2") as! Tab2ViewController
        let vc3 = self.storyboard?.instantiateViewControllerWithIdentifier("Tab3") as! Tab3ViewController
        tabPageViewController.tabItems = [(vc1, "タブ1"), (vc2, "タブ2"), (vc3, "タブ3")]
        tabPageViewController.displayControllerWithIndex(setIndex, direction: .Forward, animated: false)

        var option = TabPageOption()
        option.currentColor = UIColor.redColor()
        option.tabWidth = view.frame.width / CGFloat(tabPageViewController.tabItems.count)
        tabPageViewController.option = option

        tabPageViewController.view.frame.origin.y += (self.tabBarController?.tabBar.frame.height)!
        tabPageViewController.view.frame.size.height -= (self.tabBarController?.tabBar.frame.height)!

        self.addChildViewController(tabPageViewController)
        self.view.addSubview(tabPageViewController.view)
        tabPageViewController.didMoveToParentViewController(self)

    }

    func check() {

        Alamofire.request(.GET, "\(server)/api/test", headers: appDelegate.customHeader)
            .responseJSON { response in
                guard let object = response.result.value else {
                    return
                }
                let json = JSON(object)
                if json["result"].int == nil {
                    self.tabPageSetting(0)
                } else {
                    self.tabPageSetting(1)
                }
        }

    }

}

試したこと

override func viewDidLoad() {
        super.viewDidLoad()

        self.navigationItem.title = "テスト"
        self.navigationController!.interactivePopGestureRecognizer!.enabled = false

        tabPageSetting(0)

    }

Alamofire内で処理をしなければ、普通に表示されます。

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

Xcode7.3.1
Swift2
TabPageViewController (0.2.1)

追記

>fuzzballさん

override func viewDidLoad() {
        super.viewDidLoad()

        self.navigationItem.title = "テスト"
        self.navigationController!.interactivePopGestureRecognizer!.enabled = false

        tabPageSetting(0)
        check()

    }

これで再度実行しましたが結果は変わらずでした。

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

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

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

  • fuzzball

    2016/08/16 15:46

    検証ありがとうございました。上記の修正は元に戻しておいて下さい。

    キャンセル

  • fuzzball

    2016/08/16 17:35

    「タブバーが表示されない」のタブバーとは、TabPageViewControllerのでしょうか?それともUITabBarControllerのでしょうか?(スワイプと書いているのでTabPageViewControllerのような気がしますが)

    キャンセル

  • Y_M

    2016/08/16 17:49

    混乱させてしまい申し訳ないです。TabPageViewControllerのタブになります

    キャンセル

回答 3

check解決した方法

0

遷移前のViewCOntroller

import UIKit
import Alamofire
import SwiftyJSON
import TabPageViewController

class ExampleTableViewController: UIViewController, UITableViewDataSource {

    @IBOutlet weak var tableView: UITableView!

    let appDelegate = UIApplication.sharedApplication().delegate as! AppDelegate

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

    func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return 10
    }

    func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {

        let cell = tableView.dequeueReusableCellWithIdentifier("Cell", forIndexPath: indexPath)
        return cell
    }

    func tableView(tableView: UITableView, didSelectRowAtIndexPath indexPath: NSIndexPath) {
        //セルを選択した際に呼びだされ、checkを呼び出し結果を`appDelegate`の`tagIndex`に格納する
        //その後`TabPageViewController`を実装しているViewへ遷移する
        check { response in
            self.appDelegate.tagIndex = response
            let nextVC = self.storyboard?.instantiateViewControllerWithIdentifier("View2") as! ViewController2
            self.navigationController?.pushViewController(nextVC, animated: true)
        }
    }

    func check(apiResponse: (responseData: Int) -> ()) {

        Alamofire.request(.GET, "http://sample.jp/api/test", headers: appDelegate.customHeader)
            .responseJSON { response in
                guard let object = response.result.value else {
                    return
                }
                var result: Int = 0
                let json = JSON(object)
                if json["result"]["member_no"].int == nil {
                    result = 0
                } else {
                    result = 1
                }
                apiResponse(responseData: result)
        }

    }

遷移後のView(TabPageViewControllerを実装するView)

class ViewController2: UIViewController {

    let appDelegate = UIApplication.sharedApplication().delegate as! AppDelegate

    override func viewDidLoad() {
        super.viewDidLoad()

        self.navigationItem.title = "Page2"

        let tabPageViewController = TabPageViewController.create()
        let vc1 = UIViewController()
        vc1.view.backgroundColor = UIColor.grayColor()
        let vc2 = self.storyboard?.instantiateViewControllerWithIdentifier("Tab2") as! Tab2ViewController
        let vc3 = UIViewController()
        vc3.view.backgroundColor = UIColor.greenColor()
        tabPageViewController.tabItems = [(vc1, "タブ1"), (vc2, "タブ2"), (vc3, "タブ3")]
        tabPageViewController.displayControllerWithIndex(appDelegate.tagIndex!, direction: .Forward, animated: false)

        var option = TabPageOption()
        option.currentColor = UIColor.redColor()
        option.tabWidth = self.view.frame.width / CGFloat(tabPageViewController.tabItems.count)
        tabPageViewController.option = option

        tabPageViewController.view.frame.origin.y += (self.tabBarController?.tabBar.frame.height)!
        tabPageViewController.view.frame.size.height -= (self.tabBarController?.tabBar.frame.height)!

        self.addChildViewController(tabPageViewController)
        self.view.addSubview(tabPageViewController.view)
        tabPageViewController.didMoveToParentViewController(self)

    }

}

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

0

Alamofireの戻りでの呼び出しをメインスレッドにしてみるといかがでしょうか?
※そもそも通信しているので、タイミングによってはやはりそのようになるのはしょうがないと思いますが。

dispatch_async(dispatch_get_main_queue(), {

    let json = JSON(object)
    if json["result"].int == nil {
        self.tabPageSetting(0)
    } else {
        self.tabPageSetting(1)
    }
})

投稿

編集

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2016/08/16 14:26

    画面が真っ白で表示されなくなってしまいました。。。

    キャンセル

  • 2016/08/16 14:29 編集

    ちょっと間違えていたのでもう一度以下でお願いします。

    dispatch_async(dispatch_get_main_queue(), {

    })

    キャンセル

  • 2016/08/16 14:35

    画面は再度表示されましたが、タブバーの部分は消えたままになっています。
    やはりスワイプすると表示されます。

    キャンセル

  • 2016/08/16 14:39

    違う原因ですね、確かにその現象には自分も見覚えがあったので、分かるようなら追記します。

    キャンセル

  • 2016/08/16 14:42

    ありがとうございます。
    よろしくお願い致します。

    キャンセル

  • 2016/08/16 16:56

    自分の考えとしては、本家のサンプルも前画面で設定してから遷移しているのでその使い方が一番無難かと思いました。

    気になったのですがサーバーへアクセスして取得するような条件の方です、ABテストの様な設定を変更したいということですか?そうではなくてアプリ内で判断できることであれば通信はしなくてもよいと思いました。

    キャンセル

  • 2016/08/16 17:04

    一つ前の画面にて複数項目の中から一つを選択し、
    その選択された項目が個々にIDを持っていて、それをサーバへ投げ
    その項目に(例えばAというイベント)に参加しているかいないかで

    ・参加している場合は0番めのタブ
    ・参加していない場合は1番目のタブ

    と表示するタブを変えたいのです。

    そのためアプリ単体では判断できないと思われます。
    タブにこだわっている理由としてはどちらの場合も両方のページが見れるようにするためです。

    キャンセル

  • 2016/08/16 17:23

    そうでしたか、それでしたら前の画面でAPIを投げてから遷移しても同じですね。
    前画面でしたらAPIの戻りでタブを選択して遷移するのが一連の流れでできるので検討してもよいかと思います。

    まぁ、APIの話では無くてライブラリの使い方なので、前画面で生成するのが無難な気がします。

    キャンセル

  • 2016/08/17 11:39

    _Kentarouさんの言うとおり、前のViewで情報を取得した後に
    次のビューに値を引き継ぎ初期設定することで無事に意図する動きとなりました。

    本番環境に実装してみて問題がなければ再度ソースを解答に記載したいと思います。
    長い時間の助言本当にありがとうございました。

    キャンセル

0

同期すればいけそうな気がしたのですが、
カッコイイ同期の方法が見つからなかったです。

func check() {

    var keepAlive = true //※追加

    Alamofire.request(.GET, "\(server)/api/test", headers: appDelegate.customHeader)
        .responseJSON { response in
            guard let object = response.result.value else {
                return
            }
            let json = JSON(object)
            if json["result"].int == nil {
                self.tabPageSetting(0)
            } else {
                self.tabPageSetting(1)
            }
            keepAlive = false //※追加
    }

    //sync ※追加
    let runLoop = NSRunLoop.currentRunLoop()
    while true
        && keepAlive
        && runLoop.runMode(NSDefaultRunLoopMode, beforeDate: NSDate(timeIntervalSinceNow: 0.1))
    {}
}

同期に関しては下記の記事からパクりました。
Alamofireでデータ受信が終わるまで待機する方法

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2016/08/16 16:52

    ご回答ありがとうございます。
    回答いただきましたコードを取り入れて実行してみたのですが、
    結果は変わらずということになっています。。。

    キャンセル

  • 2016/08/16 17:07

    むむむ‥同じ環境作って試したいのですが面倒なのでこれにて終了。

    キャンセル

  • 2016/08/16 17:09

    ありがとうございました。

    キャンセル

  • 2016/08/16 17:54

    と言いつつ、同じような環境を作ってみたのですが症状を再現できず‥。
    (Alamofireのリクエスト部分以外は同じソース)

    キャンセル

  • 2016/08/16 18:11

    ここには書いていなかった(忘れていた)条件としては、
    ・TabPageViewControllerを使ったページ①に3つタブを用意し、
     それぞれにテーブルビューを配置
    ・そのテーブルビューのセルの一つを選択した際に
     TabPageViewControllerを使ったページ② <--- これが今回のページ
     に遷移するという仕組みになっています。

    キャンセル

  • 2016/08/16 18:25

    さすがにこれ以上は‥。

    キャンセル

  • 2016/08/16 18:44

    あ、やってくださいというわけではないです(汗
    ここまでお力添えいただきありがとうございました。

    キャンセル

  • 2016/08/16 23:43

    風呂に入っていて思い付いた(思い出した)のですがw、check()をviewWillAppearで呼ぶようにするとどうなるでしょうか?まずは非同期のままで、ダメなら同期させてみて下さい。

    キャンセル

  • 2016/08/17 10:15

    viewWillApperで呼んでも結果は変わらずでした。
    やり方が間違っているのかもしれませんが同期にすると固まってしまいます。

    キャンセル

  • 2016/08/17 11:00

    viewDidLoadでは固まらないのにviewWillAppearでは固まるのでしょうか?

    override func viewWillAppear(animated: Bool) {
    super.viewWillAppear(animated)
    check()
    }

    キャンセル

  • 2016/08/17 11:14

    認識が間違っていたかも知れません。
    その書き方だと固まらずTabPageのタブも表示されません。

    キャンセル

  • 2016/08/17 11:27

    新しくプロジェクトを作り最低限の機能で再現してみても、
    同じ結果になってしまうので、TabPageViewControllerの内部処理が原因なのかもしれないです。

    キャンセル

  • 2016/08/17 11:40

    _Kentarouさんの言うとおり、前のViewで情報を取得した後に
    次のビューに値を引き継ぎ初期設定することで無事に意図する動きとなりました。

    本番環境に実装してみて問題がなければ再度ソースを解答に記載したいと思います。
    同じような環境を作って検証までして頂き、助言いただき本当にありがとうございました。

    キャンセル

  • 2016/08/17 11:48

    その「最低限」のプロジェクトを丸ごと見せてもらえると検証出来るのですが、もういいですね。

    キャンセル

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

  • Swift

    8196questions

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

  • Swift 2

    1336questions

    Swift 2は、Apple社が独自に開発を行っている言語「Swift」のアップグレード版です。iOSやOS X、さらにLinuxにも対応可能です。また、throws-catchベースのエラーハンドリングが追加されています。