【问题标题】:RecyclerView with CardView Items, notifyItemChanged and shadow problemRecyclerView with CardView Items, notifyItemChanged 和 shadow 问题
【发布时间】:2019-10-01 17:37:14
【问题描述】:

有什么方法可以在项目更改时从回收站视图中禁用 CardView 高程动画,但保留原始阴影和项目动画器?

我尝试禁用 CardView 状态动画器 android:stateListAnimator="@null" 但没有结果。

这是重现问题的一种方法:

package com.w.cardviewrecyclerviewanim;

import android.support.annotation.NonNull;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.support.v7.widget.CardView;
import android.support.v7.widget.GridLayoutManager;
import android.support.v7.widget.LinearLayoutManager;
import android.support.v7.widget.RecyclerView;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;

public class MainActivity extends AppCompatActivity {

    RecyclerView items;
    Adapter adapter = new Adapter();
    GridLayoutManager layoutManager;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        items = findViewById(R.id.items);
        layoutManager = new GridLayoutManager(this, 2, LinearLayoutManager.VERTICAL, false);
        items.setLayoutManager(layoutManager);
        items.setAdapter(adapter);
    }

    class Holder extends RecyclerView.ViewHolder{
        CardView cardView;

        public Holder(@NonNull View itemView) {
            super(itemView);
            cardView = itemView.findViewById(R.id.card_view);
        }

        void bind (){
            cardView.setOnClickListener(new View.OnClickListener() {
                @Override
                public void onClick(View v) {
                    adapter.notifyItemChanged(getAdapterPosition());
                }
            });
            cardView.setOnLongClickListener(new View.OnLongClickListener() {
                @Override
                public boolean onLongClick(View v) {
                    adapter.notifyItemRangeChanged(0, adapter.getItemCount());
                    return true;
                }
            });
        }
    }

    class Adapter extends RecyclerView.Adapter<Holder>{

        @NonNull
        @Override
        public Holder onCreateViewHolder(@NonNull ViewGroup viewGroup, int position) {
            LayoutInflater layoutInflater = LayoutInflater.from(viewGroup.getContext());
            return new Holder(layoutInflater.inflate(R.layout.item, viewGroup, false));
        }

        @Override
        public void onBindViewHolder(@NonNull Holder holder, int position) {
            holder.bind();
        }

        @Override
        public int getItemCount() {
            return 30;
        }
    }
}

这里是 item.xml

<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="160dp"
    >

    <android.support.v7.widget.CardView
        android:id="@+id/card_view"

        android:layout_width="0dp"
        android:layout_height="0dp"
        android:layout_margin="10dp"
        app:layout_constraintTop_toTopOf="parent"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintLeft_toLeftOf="parent"
        app:layout_constraintRight_toRightOf="parent"

        app:cardCornerRadius="8dp"
        app:cardElevation="5dp"
        android:stateListAnimator="@null"
        >

    </android.support.v7.widget.CardView>

</android.support.constraint.ConstraintLayout>

我有一个简单的 RecyclerViewer,它有 30 个项目,每个项目都有一个具有 10dp 边距的 CardView 和一个 5dp 的 cardElevation,只是为了有一些漂亮的阴影。 当一个项目被点击时,recyclerview 适配器会收到一个 notifyItemChanged(getAdapterPosition()) 并且在长按时所有项目都会通过一个 notifyItemRangeChanged(0, adapter.getItemCount()) 调用适配器收到一个更改事件;

使用 DefaultItemAnimator 时,每当项目“更改”时,CardView 阴影在过渡动画期间会变得更大。我想保持过渡,因为我正在对项目进行一些更新,但我不想拥有阴影/高度动画。 你可以在这里看到结果: https://gph.is/g/ajmNGen

认为在长按时,应用程序会进入“多选”模式,其中所有项目的布局都有很大的变化,我希望这个变化是动画的(通过 DefaulItemAnimator),我想保持阴影不变原来。

非常感谢任何帮助,因为我在处理这个问题时正在拉扯我的头发。

【问题讨论】:

  • 你说要“保持过渡”;这是什么意思?您想对卡片内容进行淡入淡出吗?还有什么?
  • 嗨,Ben,确实是淡入淡出。这是 ItemAnimator (DefaultItemAnimator) 的默认行为。我想保持那个交叉淡入淡出。 “阴影”问题与 DefaultItemAnimator 相关,它只发生在过渡中,在交叉淡入淡出中。如果我禁用动画器 [items.setItemAnimator (null)],阴影问题就会“消失”,但交叉淡入淡出也是如此,项目只是从当前模式弹出到多选并进一步向下流水线。
  • 也许这会有所启发。假设在每次绑定时,我们都会随机更改卡片颜色。通过在 Holder 的 bind() 函数的末尾添加这条小线:cardView.setCardBackgroundColor(0xff000000 | (int)(Math.random() * (double)0xffffff)); 现在,每次触摸卡片时,项目都会再次绑定并具有新的颜色。交叉淡入淡出将在旧颜色和新颜色之间平滑变化,而不是新颜色突然出现。问题是在过渡/交叉淡入淡出期间,阴影正在播放不需要的动画。
  • 我相信您将不得不实现自己的项目动画师。您可以传递notifyItemChanged() 有效负载来禁用整个默认动画(请参阅stackoverflow.com/a/47355363/8298909),但是您不会在卡片内容上获得交叉淡入淡出。这个 Google 演示文稿youtube.com/watch?v=imsr8NrIAMs 介绍了如何制作自定义动画(包括褪色)。
  • 谢谢,好电话本。但是,我已经实现了一个新的 ItemAnimator。几乎只是 android.googlesource.com/platform/frameworks/support/+/8cf399b/… 有一个小的变化,我不做交叉淡入淡出,而是将 newHolder 保持为 alpha 1,并将 oldHolder(位于顶部)淡出到 alpha 0 以避免背景闪烁(白色)朝向 alpha 0.5。问题在于 DefaultItemAnimator 和新的。有趣的是,他们都没有碰过 translateZ 或海拔。

标签: android animation android-recyclerview shadow cardview


【解决方案1】:

cardView里面设置一个约束布局,然后就可以清晰的看到阴影了:-

<?xml version="1.0" encoding="utf-8"?>
<android.support.v7.widget.CardView 
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:elevation="6dp"
    tools:context=".MainActivity">

    <android.support.constraint.ConstraintLayout
        android:layout_width="match_parent"
        android:layout_height="match_parent">
    </android.support.constraint.ConstraintLayout>

</android.support.v7.widget.CardView>

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多