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

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

新規登録して質問してみよう
ただいま回答率
85.50%
Android

Androidは、Google社が開発したスマートフォンやタブレットなど携帯端末向けのプラットフォームです。 カーネル・ミドルウェア・ユーザーインターフェイス・ウェブブラウザ・電話帳などのアプリケーションやソフトウェアをひとつにまとめて構成。 カーネル・ライブラリ・ランタイムはほとんどがC言語/C++、アプリケーションなどはJavaSEのサブセットとAndroid環境で書かれています。

Kotlin

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

Q&A

解決済

1回答

2153閲覧

Android Kotlin registerForActivityResultのコールバックが呼び出されない

Tai4

総合スコア49

Android

Androidは、Google社が開発したスマートフォンやタブレットなど携帯端末向けのプラットフォームです。 カーネル・ミドルウェア・ユーザーインターフェイス・ウェブブラウザ・電話帳などのアプリケーションやソフトウェアをひとつにまとめて構成。 カーネル・ライブラリ・ランタイムはほとんどがC言語/C++、アプリケーションなどはJavaSEのサブセットとAndroid環境で書かれています。

Kotlin

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

0グッド

0クリップ

投稿2022/03/19 11:01

質問

起動するとスクリーンショットを取る処理を実装しています。mediaProjectionManagerの使用許可ダイアログを出すところまでは順調、つまり permissionIntentLauncher.launch(intent) までは動いています。しかし、ダイアログのボタンを押してもregisterForActivityResultのコールバックが呼び出されません。これは何が原因か教えて頂きたいです。

試したこと

下記をもとに、registerForActivityResultをrequestScreenCapturePermission内ではなくMainActivityの初期化時に呼び出されるようにしました。

Register a request to start an activity for result, designated by the given contract. This creates a record in the registry associated with this caller, managing request code, as well as conversions to/from Intent under the hood. This must be called unconditionally, as part of initialization path, typically as a field initializer of an Activity or Fragment.

コード

Kotlin

1class MainActivity : AppCompatActivity() { 2 private lateinit var permissionCallbackWaiter: CallbackWaiter<ActivityResult> 3 private val permissionIntentLauncher = registerForActivityResult(ActivityResultContracts.StartActivityForResult()) { result: ActivityResult -> 4 permissionCallbackWaiter.resume(result) 5 } 6 7 override fun onCreate(savedInstanceState: Bundle?) { 8 super.onCreate(savedInstanceState) 9 10 runBlocking { 11 val screenshot = takeScreenshot() 12 } 13 } 14 15 private suspend fun takeScreenshot(): Image { 16 val capture = DeviceCapture(this) 17 val mediaProjection = createMediaProjection() 18 19 return suspendCoroutine { continuation -> 20 capture.run(mediaProjection) { result: Image -> 21 capture.stop() 22 continuation.resume(result) 23 } 24 } 25 } 26 27 private suspend fun requestScreenCapturePermission(intent: Intent): ActivityResult { 28 val activityResult = suspendCoroutine<ActivityResult> { continuation -> 29 permissionIntentLauncher.launch(intent) 30 permissionCallbackWaiter = CallbackWaiter(continuation) 31 } 32 33 if (activityResult.resultCode == RESULT_OK) { 34 return activityResult 35 } 36 37 throw Exception("Not authorized to capture") 38 } 39 40 private suspend fun createMediaProjection(): MediaProjection { 41 val mediaProjectionManager = getSystemService(MEDIA_PROJECTION_SERVICE) as MediaProjectionManager 42 val intent = mediaProjectionManager.createScreenCaptureIntent() 43 val permissionRequestResult = requestScreenCapturePermission(intent) 44 return mediaProjectionManager.getMediaProjection(permissionRequestResult.resultCode, intent) 45 } 46} 47 48// CallbackWaiter 49class CallbackWaiter<T> (private val continuation: Continuation<T>) { 50 fun resume(value: T) { 51 continuation.resume(value) 52 } 53}

補足

Android API 24

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

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

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

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

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

jimbe

2022/03/19 15:43 編集

Logcat には何も出力されていないのでしょうか。 DeviceCapture クラスもご提示願えますでしょうか。
jimbe

2022/03/19 15:57

過去に次のようなご質問がありました。参考になりますでしょうか。 UIスレッドからcoroutineで別スレッドの処理を行っている場合の待ち合わせ方法 https://teratail.com/questions/210702
Tai4

2022/03/20 00:48

ありがとうございます。その質問の回答をもとにlaunchでコルーチンを生成するようにしたところ、コールバックが呼び出されました。より詳細に記述して、自己解決として閉じたほうがよろしいでしょうか。
jimbe

2022/03/20 03:09

その様にして頂けると助かります。
guest

回答1

0

自己解決

どうやら、registerForActivityResultのコールバックは、main thread(おそらくUIスレッド)で呼び出されるようです。(下記

the callback to be called on the main thread when activity result is available

今回、元のコードではrunBlockingを使ってコルーチンを生成していたため、UIスレッドがブロックされている状態でした。そのため、コールバックが呼びされなかったと考えられます。jimbeさん、ありがとうございました。

よって、launchでコルーチンを生成するようにしたところ問題が改善しました。一応、コードも載せておきます。

Kotlin

1class MainActivity : AppCompatActivity() { 2 // registerForActivityResultがMainActivityの初期化時に呼び出される必要があるのは変わりません。 3 private val permissionIntentLauncher = registerForActivityResult(ActivityResultContracts.StartActivityForResult()) { ... } 4 private val coroutineScope = MainScope() 5 6 ... 7 8 override fun onCreate(savedInstanceState: Bundle?) { 9 super.onCreate(savedInstanceState) 10 11 coroutineScope.launch { 12 val screenshot = takeScreenshot() 13 } 14 } 15}

投稿2022/03/20 08:21

編集2022/03/20 08:35
Tai4

総合スコア49

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

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

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.50%

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

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

質問する

関連した質問