当相机活动返回时,承载这个片段的活动有它的onActivityResult被调用。

我的片段开始一个活动的结果与意图发送给相机拍照。图片应用程序加载正常,拍摄照片并返回。然而onActivityResult从未被击中。我设置了断点,但什么都没有触发。一个片段可以有onActivityResult吗?我想是的,因为它是一个已提供的函数。为什么这个没有被触发?

ImageView myImage = (ImageView)inflatedView.findViewById(R.id.image);
myImage.setOnClickListener(new OnClickListener() {
    @Override
    public void onClick(View view) {
        Intent cameraIntent = new Intent(android.provider.MediaStore.ACTION_IMAGE_CAPTURE);
        startActivityForResult(cameraIntent, 1888);
    }
});

@Override
public void onActivityResult(int requestCode, int resultCode, Intent data) {
    if( requestCode == 1888 ) {
        Bitmap photo = (Bitmap) data.getExtras().get("data");
        ((ImageView)inflatedView.findViewById(R.id.image)).setImageBitmap(photo);
    }
}

当前回答

最简单的方法之一是从你的片段开始一个活动。

startActivity(ActivityName);

然后,添加你调用startActivityForResult(intent,"1");在你的Activity中添加onActivityResult

startActivityForResult(intent,"1");

@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
    Fragment fragment = getSupportFragmentManager().findFragmentById(R.id.dualPane);
    fragment.onActivityResult(requestCode, resultCode, data);
// perform other activity result like fetching from Uris or handling cursors

finish(); // finish the activity will  get to back to your fragment.
}

其他回答

这是最受欢迎的问题之一。我们可以找到很多关于这个问题的帖子。但没有一个对我有用。

所以我用这个方法解决了这个问题。

让我们先来理解为什么会发生这种情况。

我们可以直接从Fragment调用startActivityForResult,但实际上背后的机制都是由Activity处理的。

一旦你从一个Fragment中调用startActivityForResult, requestCode将会被改变以附加Fragment的标识到代码中。这将使Activity能够在收到结果后跟踪发送此请求的人。

一旦Activity被导航回来,结果将被发送到Activity的onActivityResult和修改后的requestCode,它将被解码为原始的requestCode + Fragment的标识。之后,Activity会通过onActivityResult将ActivityResult发送给那个Fragment。一切都完成了。

问题是:

Activity只能将结果发送给直接附加到Activity的Fragment,而不能发送给嵌套的Fragment。这就是为什么嵌套片段的onActivityResult无论如何都不会被调用的原因。

解决方案:

1)通过以下代码开始你的片段的意图:

       /** Pass your fragment reference **/
       frag.startActivityForResult(intent, REQUEST_CODE); // REQUEST_CODE = 12345

2)现在在你的父活动覆盖**onActivityResult():**

    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
    }

你必须在parent activity中调用它才能使它工作。

3)在你的片段调用中:

@Override
    public void onActivityResult(int requestCode, int resultCode, Intent data) {
        super.onActivityResult(requestCode, resultCode, data);
        if (resultCode == Activity.RESULT_OK) {

       }
}

就是这样。 有了这个解决方案,它可以应用于任何单个片段,无论它是否嵌套。是的,它也涵盖了所有的情况!此外,代码也很漂亮和干净。

原来的帖子。

FragmentActivity用修改过的requestCode替换requestCode。之后,当onActivityResult()将被调用时,FragmentActivity将解析较高的16位并恢复原始Fragment的索引。看看这个方案:

如果在根级别上有一些片段,就没有问题。但是如果你有嵌套的片段,比如在ViewPager里面有几个标签的片段,你肯定会遇到问题(或者已经遇到了)。

因为只有一个索引存储在requestCode中。这是Fragment在FragmentManager中的索引。当我们使用嵌套片段时,会有子片段管理器,它们有自己的片段列表。因此,有必要保存从根FragmentManager开始的整个索引链。

我们如何解决这个问题?在这篇文章中有一个常见的变通方案。

GitHub: https://github.com/shamanland/nested-fragment-issue

只需使用下面的代码片段。

@Override
public void onOtherButtonClick(ActionSheet actionSheet, int index) {

    if (index == 1)
    {
        Intent intent = new Intent(Intent.ACTION_PICK,
                                   android.provider.MediaStore.Images.Media.EXTERNAL_CONTENT_URI);
        intent.setType("image/*");
        startActivityForResult(Intent.createChooser(intent,
                                                    "Select Picture"), 1);
     }
}

public void onActivityResult(int requestCode, int resultCode, Intent data) {
    super.onActivityResult(requestCode, resultCode, data);
    if (resultCode == 1) {
        if (requestCode == 1) {
            Uri selectedImageUri = data.getData();
            //selectedImagePath = getPath(selectedImageUri);
        }
    }
}

onActivityResult将调用而不调用它的父类。

有一点没有人提到,确保你的主机活动启动模式不能设置为singleInstance或singleTask。

如果你的启动模式设置为SingleInstance或SingleTask, onActivityResult将不起作用。或者你用这些IntentFilters调用你的activity

标准或单顶启动模式都可以工作。

Kotlin版本(在你的活动onActivityResult()中)

 override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
    //add following lines in your activity
    if(supportFragmentManager?.fragments!=null && supportFragmentManager?.fragments!!.size>0)
     for (i in 0..supportFragmentManager?.fragments!!.size-1) {
         val fragment= supportFragmentManager?.fragments!!.get(i)
         fragment.onActivityResult(requestCode, resultCode, data)
    }
 }