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

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

ただいまの
回答率

87.49%

DialogFragmentでどのボタンがクリックされたか、呼び出し元のActivityに知らせたい。

解決済

回答 2

投稿

  • 評価
  • クリップ 0
  • VIEW 3,042

score 341

DialogFragmentでどのボタンがクリックされたか、呼び出し元のActivityに知らせたい。

私はノベルゲームを作っているのですが、いわゆる「章選択」の実装でつまっています。

私は以下のようなワークフローで章選択画面からストーリー画面への遷移を実現したいと思いました。
1:章選択画面ので、リストの中の一つの章をクリック
2:章を始めるか確認するAlertDialog表示
3:OKだったらストーリー画面に遷移、NOだったらそのままAlertDialogを消す。

この3の部分。私はAlertDialogから直接の遷移ではなく、章選択画面からの遷移をしたいと思っています。
理由はActivityのほうが、フェードアウトアニメーションなどをつけるのに都合がいいからです。

そこでタイトルの質問に行き着きました。DialogFragmentでどのボタンがクリックされたか、呼び出し元のActivityに知らせるにはどうすればよいのでしょうか?それがわかれば遷移もできそうな気が、なんとなくしています・・・

一応ネットで転がっている情報も漁ったのですが、DialogFragmentの実装方法は千差万別らしく、なかなか自分と似たことをしている人は見つけられませんでした・・・

以下にコードを示します。
どうかご教授いただけると助かります。

//選択アクティビティ
class SelectActivity : FragmentActivity() {

    /*リスナをメンバー変数に。物語確認ダイアログを表示*/
    private inner class ListStoryClickListener: AdapterView.OnItemClickListener {
        override fun onItemClick(parent: AdapterView<*>?, view: View?, position: Int, id: Long) {
            //・・・省略    
      //確認ダイアログを表示。
            val storyConf = StoryConfirmDialog()
            storyConf.show(supportFragmentManager,"Confirm Story")
            StoryConfirmDialog.DialogButtonClickListener()
      //おそらくここでなにかする・・・?
        }
    }

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_select)
        //章の選択画面を作り、リスナを追加
    val lvStory:ListView=findViewById(R.id.list_Story)
       lvStory.onItemClickListener = ListStoryClickListener()
       //・・・省略

    }



}
//確認ダイアログ
class StoryConfirmDialog : DialogFragment() {
    /*リスナをメンバー変数に。*/
    class DialogButtonClickListener : DialogInterface.OnClickListener{
        override fun onClick(dialog: DialogInterface?, which: Int) {
            when(which){
                DialogInterface.BUTTON_POSITIVE->{
                    //アニメーションや分岐の関係で、ここで直接の遷移はしたくない。
           //クリックされたことをAcitivityに知らせるには?
                }
                DialogInterface.BUTTON_NEGATIVE->{
   
                }
            }
        }
    }

    override fun onCreateDialog(savedInstanceState: Bundle?): Dialog {
   /*ここでアラートダイアログを作り、それぞれのボタンにリスナを実装*/
        /*ダイアログビルダーを生成。どのアクティビティから呼ばれるかわからないから引数にはactivity*/
        val builder = AlertDialog.Builder(activity).apply {
            setTitle("選択確認")
            setMessage("選択した物語を開始します。よろしいですか?")
            setPositiveButton("はい",DialogButtonClickListener())
            setNegativeButton("いいえ",DialogButtonClickListener())
        }
        /*ダイアログオブジェクトを生成アンドリターン*/
        return builder.create()
    }

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

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

回答 2

+1

センスないかもしれませんが、助けになればと思い回答させていただきます。

章選択アクティビティで、例えば「1章」「2章」などを選択・押下した時点で
1・ダイアログを表示する
2・フラグを変更する(例えば var iを 1章なら1、2章なら2、…)
の2点を実行する

開始確認ダイアログでyesを押下した場合は、
1.ダイアログを閉じる
2.章選択アクティビティのメソッドを呼ぶ
このメソッドは、フラグiを参照してwhenなどで分岐し、画面遷移を行う。

ダイアログでcancelを押下した場合はダイアログを閉じるだけ。

kotlin詳しくないので申し訳ないのですが、
このままだとダイアログでyesを押した瞬間に
閉じる&アクティビティからの画面遷移が同時に見えるので

Handler().postDelayed(Runnable {
            val intent = Intent(this, Test2Activity::class.java)
            startActivity(intent)
        }, 1000)


のように時間差で画面遷移を実行すればイケるかと思います。

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2019/01/16 14:44

    回答ありがとうございます!!
    アクティビティからの画面遷移については、今後どうしようか考えようと思っていました。
    なので回答者さまに教えていただいたコードはありがたく、今後大いに参考にさせていただきます!

    キャンセル

checkベストアンサー

0

FragmentからActivityにイベントを知らせるには、 getActivity() メソッド経由で通知をするのがスタンダードな方法です。
Fragment側でイベント通知用のinterfaceを定義して、Activity側はそれを実装してあげるとよいでしょう。今回の場合は、DialogInteface.OnClickListenerがそのまま使えそうなのでそれを利用すると楽です。

class StoryConfirmDialog : DialogFragment() {
    inner class DialogButtonClickListener : DialogInterface.OnClickListener {
        override fun onClick(dialog: DialogInterface?, which: Int) {
            val a = activity
            if (a is DialogInterface.OnClickListener) {
                a.onClick(dialog, which)
            }
        }
    }

    ...
}
class SelectActivity : FragmentActivity(), DialogInterface.OnClickListener {
    ...

    override fun onClick(dialog: DialogInterface?, which: Int) {
        when (which) {
            DialogInterface.BUTTON_POSITIVE -> {
            }
            DialogInterface.BUTTON_NEGATIVE -> {
            }
        }
    }
}

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2019/01/16 14:41

    回答ありがとうございます!
    具体的なコードを書いてくださり、よく理解ができました。
    また、getActivityがスタンダードな方法というのを知れたのは、とても大きなことで、本当に助かりました!!!

    キャンセル

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

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

関連した質問

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