嵌套共享元素的转换不是从预期位置开始

rnr*_*ies 5 android android-5.0-lollipop

TL;DR为什么这种转变不是从预期的地方开始?

也就是CardView所在的地方。

在此处输入图片说明

平台/Deps: Android Studio 1.0-RC2、TargetSdk 21、MinSdk 21、AppCompat-21.0.2、CardView-21.0.2、RecyclerView-21.0.2。

测试对象: Genymotion Nexus 5 Lollipop Preview

代码(简化版)

我有三个共享元素:

来源:

<CardView
    android:transitionName="cardView">
    <ImageView
      android:transitionName="imageView">
    <TextView
      android:transitionName="textView">
Run Code Online (Sandbox Code Playgroud)

目标:

<RelativeLayout
    android:transitionName="cardView">
    <ImageView
      android:transitionName="imageView">
    <TextView
      android:transitionName="textView">
Run Code Online (Sandbox Code Playgroud)

请注意,以下过渡名称是以编程方式添加的。这是因为 CardView 保存在 RecyclerView 上,并且有很多副本。我试过使用相同的名称(如在 XML 实例中),但也不起作用。

RecillerView.Adapter 这样做:

@Override
public void onBindViewHolder(ViewHolder viewHolder, int position) {
    //...
    ViewCompat.setTransitionName(viewHolder.mCardView, "cardViewTransition" + position);
    ViewCompat.setTransitionName(viewHolder.mImageView, "imageTransition" + position);
    ViewCompat.setTransitionName(viewHolder.mTextView, "textTransition" + position);
    //...
}
Run Code Online (Sandbox Code Playgroud)

HomeActivity 这样做:

@Override
protected void onCreate(Bundle savedInstanceState) {
    getWindow().requestFeature(android.view.Window.FEATURE_CONTENT_TRANSITIONS);
    getWindow().requestFeature(android.view.Window.FEATURE_ACTIVITY_TRANSITIONS);
    // ..
    mRecyclerView.addOnItemTouchListener(
        new RecyclerItemClickListener(this, new RecyclerItemClickListener.OnItemClickListener() {
            @Override public void onItemClick(View view, int position) {
                Intent intent = new Intent(selfContext, DetailActivity.class);

                View imageView = view.findViewById(R.id.imageView);
                TextView textView = (TextView)view.findViewById(R.id.textView);

                Bundle extras = new Bundle();
                extras.putInt("position", position);
                extras.putString("text", textView.getText().toString());
                intent.putExtras(extras);
                String name = ViewCompat.getTransitionName(view);
                ActivityOptionsCompat options =
                    ActivityOptionsCompat.makeSceneTransitionAnimation(selfContext
                            ,Pair.create(view, ViewCompat.getTransitionName(view))
                            ,Pair.create(imageView, ViewCompat.getTransitionName(imageView))
                            ,Pair.create((View)textView, ViewCompat.getTransitionName(textView))
                    );
                ActivityCompat.startActivity(selfContext, intent, options.toBundle());
            }
        })
    );
    ///...
}
Run Code Online (Sandbox Code Playgroud)

DetailActivity 这样做:

@Override
protected void onCreate(Bundle savedInstanceState) {
    getWindow().requestFeature(android.view.Window.FEATURE_CONTENT_TRANSITIONS);
    getWindow().requestFeature(android.view.Window.FEATURE_ACTIVITY_TRANSITIONS);
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_detail);

    // ...

    Bundle extras = getIntent().getExtras();
    int mPositionRef = extras.getInt("position");
    View base = findViewById(R.id.detail_layout);
    View image = findViewById(R.id.imageView);
    TextView text = (TextView) findViewById(R.id.txtLabel);
    text.setText(extras.getString("text"));

    ViewCompat.setTransitionName(base, "cardViewTransition" + mPositionRef);
    ViewCompat.setTransitionName(image, "imageTransition" + mPositionRef);
    ViewCompat.setTransitionName(text, "textTransition" + mPositionRef);

}
Run Code Online (Sandbox Code Playgroud)

我已经调试和共享元素在源和目标中具有相同的名称。

问题(再次)

为什么这种转变不是从预期的地方开始?也就是CardView所在的地方。

GitHub 中的完整示例

Ale*_*ood 2

Lollipop 中有一个与嵌套共享元素相关的已知错误(即同时添加视图组及其子元素之一作为共享元素),因此它可能与此相关(来源:Glitch when animating Nested Views in a共享元素 Activity 转换?)。

尝试仅将 ImageView 和 TextView 添加为共享元素(而不是父相对布局容器)。