🎄teratailクリスマスプレゼントキャンペーン2024🎄』開催中!

\teratail特別グッズやAmazonギフトカード最大2,000円分が当たる!/

詳細はこちら
Android Studio

Android Studioは、 Google社によって開発された、 Androidのネイティブアプリケーション開発に特化した統合開発ツールです。

Kotlin

Kotlinは、ジェットブレインズ社のアンドリー・ブレスラフ、ドミトリー・ジェメロフが開発した、 静的型付けのオブジェクト指向プログラミング言語です。

Q&A

解決済

1回答

4012閲覧

Kotlin コールバックして呼び出し元のコントロールを触りたいです。

n3mryo

総合スコア1

Android Studio

Android Studioは、 Google社によって開発された、 Androidのネイティブアプリケーション開発に特化した統合開発ツールです。

Kotlin

Kotlinは、ジェットブレインズ社のアンドリー・ブレスラフ、ドミトリー・ジェメロフが開発した、 静的型付けのオブジェクト指向プログラミング言語です。

0グッド

0クリップ

投稿2021/01/28 15:22

##ダイアログが消えたタイミングで呼び出し元のコントロールを変化させたい
Kotlin勉強中の初学者です。Fragmentから表示させたダイアログが消えたタイミングで呼び出し元のListViewを更新するプログラムを作りたいです。自分なりに調べたりこちらでご助力いただいてコールバックといのをすれば良いのかな?というとこまで思い至りました。

ダイアログのDismissのタイミングで呼び出し元に戻ってくるところまではできたんですが、activityが破棄されているみたいでNullPointerExceptionが出たり、TextViewなどのプロパティを触ろうとするとmust not be null とエラーがでてしまいます。activityを再生成?する方法ってありますでしょうか。(それとも何らかの方法で保持しておくものなんでしょうか。)

以下ソースの一部になりますが、どういった処理を加えたらよいか。どういったことを調べたらよいかを教えていただけると嬉しいです。

呼び出し元のソースです。

Kotlin

1import kotlinx.android.synthetic.main.fragment_ranking.* 2 3class RankingFragment : Fragment() ,DialogDeleteCheck.DialogListener { 4 5//省略 6 7  //DialogのDismissのタイミングで戻ってくる。 8 override fun onDialogDismiss(dialog: DialogFragment) { 9 Log.d("tag","main") 10 tvRanking.text = "main" //must not be nullエラーが出る 11 } 12}

DialogFragmentのソースです。

Kotlin

1class DialogDeleteCheck: DialogFragment() { 2 3  //インターフェース 4 interface DialogListener{ 5 fun onDialogDismiss(dialog:DialogFragment) 6 } 7 8 /** 9 * コールバック用リスナー 10 */ 11 private var listener :DialogListener? = null 12 13 14  //省略 15 16 17 /** 18 * Dialog終了時の処理 19 */ 20 override fun onDismiss(dialog: DialogInterface) { 21 super.onDismiss(dialog) 22 Log.d("tag","dismiss") 23 listener?.onDialogDismiss(this) 24 } 25 26 /** 27 *Attach時処理 28 */ 29 override fun onAttach(context: Context) { 30 super.onAttach(context) 31 Log.d("tag","attach") 32 33 try { 34 val targetFrg : Fragment = RankingFragment() 35 listener = targetFrg as DialogListener 36 }catch (e: Exception){ 37 Log.e("ERROR","CANNOT FIND LISTENER") 38 } 39 } 40 41 /** 42 * Detach時処理 43 */ 44 override fun onDetach() { 45 super.onDetach() 46 Log.d("tag","detach") 47 listener = null 48 } 49}

##質問したいこと
・そもそもソースが間違っているせいで呼び出し元のactivityがnullになってしまうのでしょうか?
・呼び出し元のactivityを再生成?する方法はあるのしょうか?もしありましたらどういった処理をしたらよいか、どういうことを調べたらよいかを教えていただけると嬉しいです。

何かわかる方がいらっしゃいましたらよろしくお願いします!

##環境
Android Studio 4.1.1
AndroidEmulator Nexus_5X_API_28

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

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

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

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

hoshi-takanori

2021/01/28 16:39

listener に呼び出し元の fragment ではなく、val targetFrg : Fragment = RankingFragment() で新しく作ったものが設定されているからですね。listener が Activity の場合は getActivity() できますが、fragment の場合は setFragmentResult と setFragmentResultListener を使う必要があるようです。 https://tech.mokelab.com/android/Fragment/result.html
n3mryo

2021/01/29 15:00

すいません。初歩的な質問で申し訳ないのですがparentFragmentMangerというものが予測変換に出てこないのですがどういう環境下で使えるものなんでしょうか?
n3mryo

2021/01/29 15:58 編集

失礼しました。Gradleファイルのdependenciesの方へ implementation "androidx.fragment:fragment-ktx:1.3.0-alpha04"と追加したところ使えるようになりましたm(_ _)m
n3mryo

2021/01/30 09:09

自分なりにいろいろやっていましたがなかなかうまくいきません。。。 呼び出し元のfragmentをlistenerにいれるというのは具体的にどのようにやればよいでしょうか? いただいたURLを参考にして(今回の僕のソースだと) 呼び出し元(RankinguFragment)にsetFragmentResultListener 呼び出された方(DialogDeleteCheck)にsetFragmentResultを加えるという認識でよろしいでしょうか? そのあと、何をどうやってリスナーに入れたら良いのでしょうか?
n3mryo

2021/01/30 13:36

解決しました。ご協力ありがとうございました。
guest

回答1

0

自己解決

難しく考えすぎていましたがコールバックリスナーは不要でした。呼び出し元にserFragmentResultListener、呼び出されるダイアログにparentFragmentManager.setFragmentResultを記述してやればよかったです。

呼び出し元

Kotlin

1class RankingFragment : Fragment() { 2 3//省略 4 5 override fun onCreateView( 6 inflater: LayoutInflater, container: ViewGroup?, 7 savedInstanceState: Bundle? 8 ): View? { 9 10 //Dialogから応答を受け取る 11 setFragmentResultListener("input"){ 12 _, data -> 13 when(data.getString("when","")){ 14 "DISMISS"->{ 15 //DialogDismiss時の処理 16 fncSetRecodeToListView(gModeFlg.ordinal)} 17 else->{} 18 } 19 } 20}

呼び出されるダイアログ

Kotlin

1class DialogDeleteCheck: DialogFragment() { 2 3 //省略 4 //コールバックする必要がなくなったのでインターフェースも消した 5 6 /** 7 * Dialog終了時の処理 8 */ 9 override fun onDismiss(dialog: DialogInterface) { 10 super.onDismiss(dialog) 11 12 //返したい情報をセット 13 val data = Bundle() 14 data.putString("when","DISMISS") 15 //FragmentManger経由で結果を知らせる 16 parentFragmentManager.setFragmentResult("input",data) 17 } 18} 19 20

コンテキストが欲しいだけなのでputStringはあってもなくてもいいですが判定用に文字列を一応入れました。

投稿2021/01/30 14:07

n3mryo

総合スコア1

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.36%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問