使用 startActivityForResult 时的共享元素

eme*_*ais 5 android android-intent activity-transition shared-element-transition

我有一个带有两个的应用程序activities,其中第二个用作图像“选择器”并向第一个提供一些信息,以便ImageView可以相应地更新。

为此,我activity通过调用第二个startActivityForResult(),并ImageView根据收到的信息更新onActivityResult()

当我尝试为选取器和主元素之间的共享元素设置动画时,问题出现了Activity:过渡动画似乎是第一个“快照” Activity,即使恢复后ImageView立即更新Activity,过渡动画也会“闪烁”较旧的版本View显示当前版本之前。

这是我制作的一个示例应用程序,用于说明我在说什么:

错误的视觉描述

无缝实现此动画的最佳方法是什么?是否应该使用另一种方法?

这是用于示例的代码:

主要活动

public class MainActivity extends AppCompatActivity {

    ImageView mainImageView;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        getSupportActionBar().setTitle("Main Activity");

        mainImageView = (ImageView) findViewById(R.id.imageView);
        findViewById(R.id.button).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent intent = new Intent(MainActivity.this, PickerActivity.class);
                ActivityOptionsCompat options = ActivityOptionsCompat.
                        makeSceneTransitionAnimation(MainActivity.this, mainImageView, "shape_transition");
                startActivityForResult(intent, 1, options.toBundle());
            }
        });
    }

    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent intent) {
        super.onActivityResult(requestCode, resultCode, intent);
        int shapeNumber = intent.getIntExtra("shape_number", -1);

        switch (shapeNumber) {
            case 1:
                mainImageView.setImageResource(R.drawable.blue);
                break;
            case 2:
                mainImageView.setImageResource(R.drawable.green);
                break;
            default:
                //something else
                break;
        }


    }
}
Run Code Online (Sandbox Code Playgroud)

选择器活动:

public class PickerActivity extends AppCompatActivity {

    ImageView blueImageView;
    ImageView greenImageView;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_picker);
        getSupportActionBar().setTitle("Picker Activity");

        blueImageView = (ImageView) findViewById(R.id.imageView2);

        blueImageView.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                blueImageView.setTransitionName("shape_transition");
                Intent intent = new Intent(PickerActivity.this, MainActivity.class);
                intent.putExtra("shape_number", 1);
                setResult(RESULT_OK, intent);
                finishAfterTransition();
            }
        });

        greenImageView = (ImageView) findViewById(R.id.imageView3);

        greenImageView.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                greenImageView.setTransitionName("shape_transition");
                Intent intent = new Intent(PickerActivity.this, MainActivity.class);
                intent.putExtra("shape_number", 2);
                setResult(RESULT_OK, intent);
                finishAfterTransition();
            }
        });
    }
}
Run Code Online (Sandbox Code Playgroud)

主要活动xml:

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    tools:context="demonstration.sharedelementproblemexample.MainActivity">

    <ImageView
        android:layout_width="128dp"
        android:layout_height="128dp"
        android:id="@+id/imageView"
        android:transitionName="shape_transition"
        android:layout_alignParentTop="true"
        android:layout_alignParentStart="true"
        android:src="@drawable/red" />

    <Button
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Pick"
        android:id="@+id/button"
        android:layout_below="@+id/imageView"
        android:layout_alignParentStart="true" />
</RelativeLayout>
Run Code Online (Sandbox Code Playgroud)

选择器活动 xml:

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    tools:context="demonstration.sharedelementproblemexample.PickerActivity">

    <ImageView
        android:layout_width="128dp"
        android:layout_height="128dp"
        android:id="@+id/imageView2"
        android:layout_alignParentBottom="true"
        android:layout_alignParentEnd="true"
        android:src="@drawable/blue" />

    <ImageView
        android:layout_width="128dp"
        android:layout_height="128dp"
        android:id="@+id/imageView3"
        android:src="@drawable/green"
        android:layout_alignParentStart="true"
        android:layout_alignTop="@+id/imageView2" />
</RelativeLayout>
Run Code Online (Sandbox Code Playgroud)

eme*_*ais 3

在从事另一个项目时,我找到了一种防止这种闪烁发生的方法。

在第一个 Activity(发送请求的 Activity)中,onActivityResult(int requestCode, int resultCode, Intent intent)您应该重写onActivityReenter(int resultCode, Intent data)并更新该视图,而不是更新最初通过该方法转换的视图。

Activity该方法在过渡动画开始之前调用,您仍然可以通过参数访问对方的信息Intent

这是使用此解决方案的同一示例应用程序:

显示修复的图片

这是重写的方法:

@Override
public void onActivityReenter(int resultCode, Intent data) {
    int shapeNumber = data.getIntExtra("shape_number", -1);

    switch (shapeNumber) {
        case 1:
            mainImageView.setImageResource(R.drawable.blue);
            break;
        case 2:
            mainImageView.setImageResource(R.drawable.green);
            break;
        default:
            //something else
            break;
    }
    super.onActivityReenter(resultCode, data);
}
Run Code Online (Sandbox Code Playgroud)