Android: startActivityForResult не вызывает onActivityResult

Моя установка

  • Служба, работающая в собственном процессе, CentralService
  • Активность, которая вызывает startActivityForResult(), MainActivity
  • Действие, которое запускается для результата, ReturnResultActivity

Что я пытаюсь сделать

  1. Запустите ReturnResultActivity и привяжите его к службе (зарегистрируйте его обработчик)
  2. Пусть любые другие действия хотят работать
  3. When it receives a message from the service:
    • Unbind from the Service
    • финиш()
    • setResult()
      1. Have the MainActivity's onActivityResult() method called

Используя Log.i, я подтвердил, что шаги с 1 по 3 выполняются. Однако, когда должен быть вызван метод onActivityResult(), я получаю в журнале следующее:

V/ActivityManager(   60): Finishing activity: token=HistoryRecord{442cc518 com.myActivity/.primitives.ReturnResultActivity}, result=3, data=Intent { (has extras) }
V/ActivityManager(   60): Adding result to HistoryRecord{442a3988 com.mainActivity.sample/.MainActivity} who=null req=500 res=3 data=Intent { (has extras) }

Дополнительная информация

Каждый объект находится в отдельном проекте (у меня есть 3 проекта, взаимодействующих друг с другом), и поэтому все они работают в своем собственном процессе.

MainActivity запускается из службы со следующим кодом:

Intent intent = new Intent(Intent.ACTION_MAIN);
intent.addCategory(Intent.CATEGORY_LAUNCHER);
intent.setClassName(activityInfo.packageName, activityInfo.name);
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK 
          | Intent.FLAG_ACTIVITY_RESET_TASK_IF_NEEDED);
startActivity(intent);

Кое-что из Кодекса

Активность возврата результата:

public class ReturnResultActivity extends SomeActivity {
    private static final boolean DEBUG = true;
    private static final String TAG = "ReturnResultActivity";

    protected void onBind() {
        // We want to monitor the service for as long as we are
        // connected to it.
        Message msg = Message.obtain(null,
                CentralService.MSG_LISTEN);
        msg.replyTo = mMessenger;

        try {
            mService.send(msg);
        } catch (RemoteException e) {
            // In this case the service has crashed before we could even
            // do anything with it; we can count on soon being
            // disconnected (and then reconnected if it can be restarted)
            // so there is no need to do anything here.
            e.printStackTrace();
        }
    }

    /** this method is called eventually after doUnbindService is called **/
    protected void onUnbind() {
        if (DEBUG) Log.i(TAG, "Unbinding and finishing"); //I can tell this gets called when I run it
        if (data != null) {
            setResult(CentralService.MSG_SPEECH_RECOGNIZED, data);
        }
        finish();
    }

    Intent data;
    protected boolean receiveMessage(Message msg) {
        if (DEBUG) Log.i(TAG, "Incoming Message to Listener: " + msg.what);

        switch (msg.what) {
            case ASRManager.MSG_SPEECH_RECOGNIZED:
                data = new Intent();
                Bundle bundle = msg.getData();
                bundle.setClassLoader(getClassLoader());
                data.putExtra(ASRManager.TOKEN_PARCEL_KEY, bundle.getParcelable(ASRManager.TOKEN_PARCEL_KEY));
                if (DEBUG) Log.i(TAG, "Received speech, setting result.");
                doUnbindService();

                return true;
            default:
                return false;
        }
    }


}

Основная деятельность

public class MainActivity extends CustomActivity {
    private static final boolean DEBUG = true;
    private static final String TAG = "MainActivity";

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        Intent intent = new Intent("com.custom.returnresultaction");
        startActivityForResult(listenIntent, 500);

    }

    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        if (DEBUG) Log.i(TAG, "Got activity result " + resultCode); //this doesn't get called

    }
}

В моем окне LogCat не отображаются трассировки стека. Пожалуйста помоги!

EDIT: Интересная новая информация. Если я затем запускаю MainActivity из панели запуска, я вижу следующее в logcat:

I/ActivityManager(   60): Starting activity: Intent { act=android.intent.action.MAIN cat=[android.intent.category.LAUNCHER] flg=0x10200000 cmp=com.mainActivity.sample/.MainActivity }
V/ActivityManager(   60): Sending result to HistoryRecord{44253568 com.android.launcher/com.android.launcher2.Launcher} (index -1)
I/ActivityManager(   60): Starting Activity Unchecked Locked
V/ActivityManager(   60): com.mainActivity.sample.MainActivity Audio Activity Found
V/ActivityManager(   60): Delivering results to HistoryRecord{442c0310 com.mainActivity.sample/.MainActivity}: [ResultInfo{who=null, request=500, result=3, data=Intent { (has extras) }}]
V/ActivityThread(  327): Handling send result to ActivityRecord{440de270 token=android.os.BinderProxy@440ddca0 {com.mainActivity.sample/com.mainAcitivty.sample.MainActivity}}
V/ActivityThread(  327): Delivering result to activity ActivityRecord{440de270 token=android.os.BinderProxy@440ddca0 {com.mainActivity.sample/com.mainActivity.sample.MainActivity}} : ResultInfo{who=null, request=500, result=3, data=Intent { (has extras) }}
I/MainActivity(  327): Got activity result 3

Моя теория заключается в том, что задача, содержащая MainActivity, не выполняется, потому что сообщение получено от CentralService. Какие-нибудь мысли? Любая идея, как переключиться на правильную задачу. (обратите внимание, хотя это может показаться плохой практикой и разрушительным переключением задач на пользователя и выведением другого действия на передний план, это то, что я хочу сделать. Это потому, что это в конечном итоге будет работать на пользовательской версии Android, которая сделает все это не мешает)


person m3hughes    schedule 27.10.2011    source источник
comment
Это, вероятно, один из лучших отформатированных вопросов, которые я когда-либо видел от нового пользователя. Огромный респект тебе, мой друг. Я не вижу, что не так с вашим кодом, но я очень надеюсь, что вы получите ответ!   -  person Kurtis Nusbaum    schedule 27.10.2011


Ответы (2)


Вернувшись к этому вопросу через 2 года, проблема заключается в том, что действие запускается в новой задаче. (Я установил это в намерении, которое я запускаю). Запуск действия для результата, который немедленно запускается в новой задаче, вернет -1.

person m3hughes    schedule 10.06.2014
comment
как не запускать с новой задачи? - person temirbek; 12.10.2017

После того, как вы позвоните

startActivityForResult(listenIntent, 500);

Вы включаете этот код в класс com.custom.returnresultaction?

setResult(RESULT_CANCELED, i);

or setResult(RESULT_OK, i);

and call finish();

Вам нужно установить результат и вызвать finish().

person dcanh121    schedule 27.10.2011
comment
Да, если вы посмотрите на метод onUnbind, этот код вызывается. doUnbindService() вызывается в суперклассе, который затем вызывает onUnbind() (я проверил это, потому что вызывается сообщение журнала). Данные не нулевые, поэтому вызывается setResult, а затем неизбежно вызывается finish. - person m3hughes; 27.10.2011
comment
Учитывая то, что вы хотите сделать, если вы еще не решили эту проблему, может быть, было бы лучше отправлять широковещательные сообщения между различными действиями и процессами, а не использовать завершение действия? - person Neil Townsend; 15.07.2013