Glitchy在recyclerview和viewpager之间共享元素转换

时间:2018-05-28 06:14:09

标签: java android user-interface animation shared-element-transition

您好我已经创建了一个应用程序,我使用共享元素转换来处理片段中的recyclelerview和处于活动状态的滑块viewpager之间的某些动画。 现在的问题是,当我应用共享元素转换并单击recyclerview时,它会闪烁片刻并且细节活动开始但没有动画,当我按下后退按钮时,共享元素动画会发生,但是太麻烦了。 我的意思是,当我按下cardview的白色背景弹出窗口,然后图像就会越过它。 为此,我在youtube上看到了一些教程以及在谷歌上搜索并发现了一些相关的So question,但仍然无法弄清楚我在哪里出错了,因为我给了两个相同的转换名称。 所以,如果有人可以在这里帮助我。 谢谢

这是我的recyclerview和styles.xml

的代码

styles.xml

<resources xmlns:tools="http://schemas.android.com/tools">

    <!-- Base application theme. -->
    <style name="AppTheme" parent="Theme.AppCompat.Light.NoActionBar">
        <!-- Customize your theme here. -->
        <item name="colorPrimary">@color/colorPrimary</item>
        <item name="colorPrimaryDark">@color/colorPrimaryDark</item>
        <item name="colorAccent">@color/colorAccent</item>
        <item name="android:windowContentTransitions" tools:targetApi="lollipop">true</item>
        <item name="android:windowAllowEnterTransitionOverlap" tools:targetApi="lollipop">true</item>
        <item name="android:windowAllowReturnTransitionOverlap" tools:targetApi="lollipop">true</item>
        <item name="android:windowSharedElementEnterTransition" tools:targetApi="lollipop">@android:transition/move</item>
        <item name="android:windowSharedElementExitTransition" tools:targetApi="lollipop">@android:transition/move</item>
    </style>

</resources>

ViewHolder

 public class VH extends RecyclerView.ViewHolder implements View.OnClickListener{
        ImageView iv;
        TextView tv;
        HWRatioContainer ivc;
        Context context;
        int[] images;

        private AtomicBoolean enterTransitionStarted;
        public static final String EXTRA_TRANSITION_IMAGE = "image";

        CustomButton top_recycler_button;
        public VH(View itemView,Context context,int[] images) {
            super(itemView);
            iv = (ImageView) itemView.findViewById(R.id.iv);
            this.context = context;
            this.images = images;

            this.enterTransitionStarted = new AtomicBoolean();
            top_recycler_button = (CustomButton)itemView.findViewById(R.id.top_recycler_button);
            itemView.setOnClickListener(this);
            //tv = (TextView) itemView.findViewById(R.id.tv);
            ///ivc = (HWRatioContainer) itemView.findViewById(R.id.ivc);
           /*ivc.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
                @Override
                public void onGlobalLayout() {
                    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN) {
                        ivc.getViewTreeObserver().removeOnGlobalLayoutListener(this);
                    } else {
                        ivc.getViewTreeObserver().removeGlobalOnLayoutListener(this);
                    }
                    ivc.setTranslationX(ivc.getWidth() >> 4);
                }
            });*/

        }

        @SuppressLint("RestrictedApi")
        @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
        @Override
        public void onClick(View view) {
            Intent intent = new Intent(context,DetailsActivity.class);
            intent.putExtra("slider_image",images[getAdapterPosition()]);
            iv.setTransitionName(context.getString(R.string.first_page_transition));
            ActivityOptionsCompat compat = ActivityOptionsCompat.makeSceneTransitionAnimation((MainActivity)context,iv,ViewCompat.getTransitionName(iv));
            context.startActivity(intent,compat.toBundle());
        }
    }

和活动中的viewpager适配器的代码

@TargetApi(Build.VERSION_CODES.LOLLIPOP)
    @RequiresApi(api = Build.VERSION_CODES.KITKAT)
    @NonNull
    @Override
    public Object instantiateItem(@NonNull ViewGroup container, int position) {
        layoutInflater = (LayoutInflater)context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
        View view = layoutInflater.inflate(R.layout.custom_dots_layout,null);
        imageView = (ImageView)view.findViewById(R.id.detail_viewpager_image);
        //Intent intent  = new Intent();
        imageView.setImageResource(getImage());

        if(Build.VERSION.SDK_INT>Build.VERSION_CODES.LOLLIPOP){
            imageView.setTransitionName(context.getString(R.string.first_page_transition));

        }
        ViewPager viewPager1 = (ViewPager)container;
        viewPager1.addView(view,0);
        return view;
    }

1 个答案:

答案 0 :(得分:0)

我发布它作为答案,因为它有点长。您是对的,例如,如果要在片段活动之间共享一个图像,则需要在两者中提供相同的名称。但在您的情况下,您有一个RecyclerView,它可以包含许多元素,因此您需要为每个元素提供唯一的转换名称。一个好的地方是onBindViewHolder,您可以通过附加位置来设置转换名称,例如:

@Override
public void onBindViewHolder(ViewHolder holder, int position) {
    imageView.setTransitionName("YourTransitionName" + position);
    // other code
}

这样您就可以确定每个转换名称都是唯一的,因此没有冲突。如您所知,您需要在两个位置设置转换名称,因此您需要将位置传递给ViewPager。另外,请查看它可能有帮助的this文章。