notifyItemChanged()使RecyclerView滚动并跳转到UP

时间:2016-04-19 17:02:40

标签: android android-recyclerview recycler-adapter

我有一个带有适配器的RecycleView,它显示了服务器列表 并且用户必须选择一个服务器。

当我在onClick()方法中调用notifyItemChanged(previousPosition)时 要取消选择旧服务器并选择新服务器, 这使得RecycleView列表完全跳到列表中间。

当我点击RecycleView列表中最后2或3个服务器之一时,会发生此问题

这是我的RecyclerView.Adapter的代码:

public class ServerAdapter extends RecyclerView.Adapter<ServerAdapter.ServerViewHolder> {

    private List<Server> listServers = new ArrayList<>();
    private int[] icons = new int[]{R.drawable.server1,R.drawable.server2,R.drawable.server3,R.drawable.server4,R.drawable.server5,R.drawable.server6,R.drawable.offline};
    private int selected = 0;
    private int previousSelected = 0;

    public ServerAdapter(List<Server> listServers){
        this.listServers = listServers;
    }

    @Override
    public ServerViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.server_relative_layout,parent,false);
        return new ServerViewHolder(view);
    }

    @Override
    public void onBindViewHolder(final ServerViewHolder holder, final int position) {
        if(position == selected){
            holder.getBackground().setSelected(true);
        }else{
            holder.getBackground().setSelected(false);
        }
        holder.getBackground().setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                if(position != selected){
                    previousSelected = selected;
                    selected = position;
                    holder.getBackground().setSelected(true);
                    notifyItemChanged(previousSelected);
                }
            }
        });
        holder.getImageServer().setImageResource(icons[position%6]);
        holder.getTextNameServer().setText(listServers.get(position).getName());
        holder.getTextConnected().setText(listServers.get(position).getUrl());
    }

    @Override
    public int getItemCount() {
        return listServers.size();
    }

    public class ServerViewHolder extends RecyclerView.ViewHolder{
        private ImageView imageServer;
        private TextView textNameServer;
        private TextView textConnected;
        private View background;
        public ServerViewHolder(View itemView) {
            super(itemView);
            imageServer = (ImageView)itemView.findViewById(R.id.imageServer);
            textNameServer = (TextView)itemView.findViewById(R.id.textNameServer);
            textConnected = (TextView)itemView.findViewById(R.id.textConnected);
            background = itemView;
        }

        public ImageView getImageServer() {
            return imageServer;
        }

        public TextView getTextConnected() {
            return textConnected;
        }

        public TextView getTextNameServer() {
            return textNameServer;
        }

        public View getBackground() {
            return background;
        }
    }
}

解决这个问题的任何解决方案?感谢。

问题恰好发生在我指定布局高度并且不让它为wrap_content

<android.support.v7.widget.RecyclerView
    android:layout_width="wrap_content"
    android:layout_height="400dp"
    android:id="@+id/serverRecyclerView"
    android:layout_margin="10dp"
/>

或者当我把它放在下面这样的例子中时:

<android.support.v7.widget.RecyclerView
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:id="@+id/serverRecyclerView"
    android:layout_margin="10dp"
    android:layout_below="@+id/image"/>

我的代码完全是:

<android.support.v7.widget.RecyclerView
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:id="@+id/serverRecyclerView"
    android:layout_margin="10dp"
    android:layout_alignTop="@+id/imageBall"
    android:layout_alignParentRight="true"
    android:layout_alignParentEnd="true"
    android:layout_toRightOf="@+id/camera"
    android:layout_toEndOf="@+id/camera"/>

10 个答案:

答案 0 :(得分:12)

看起来这是一个错误:https://code.google.com/p/android/issues/detail?id=203574

最好的解决方法似乎是Bart的答案,将RecyclerView的LinearLayoutManager的AutoMeasure属性设置为false。

  LinearLayoutManager llm = new LinearLayoutManager(context);
  llm.setAutoMeasureEnabled(false);       
  recyclerView.setLayoutManager(llm);

将FixedSize设置为true解决方案有太多副作用......

RecyclerView.setHasFixedSize(真)

答案 1 :(得分:4)

$this->db->where_in('brands', (array) $brand_name);

android:descendantFocusability =“ blocksDescendants”

此属性解决了我的错误

答案 2 :(得分:3)

我不知道为什么,但我用过:

RecyclerView.setHasFixedSize(true)

这对我有用。我希望它可以提供帮助。

答案 3 :(得分:1)

RecyclerView.ItemAnimator animator = myRecyclerListView.getItemAnimator();
if (animator instanceof SimpleItemAnimator) {
    ((SimpleItemAnimator)animator).setSupportsChangeAnimations(false);
}

答案 4 :(得分:1)

对于遇到此问题的任何人,请尝试使用

yourRecyclerView.notifyItemChanged(int position, Object payload);

这个对我有用。

使用

setAutoMeasureEnabled(false);

也有效,但在某些边缘情况下,回收者的观点很奇怪。祝你好运!

答案 5 :(得分:0)

我遇到了类似的问题,只需要处理xml布局文件 请勿在RecyclerView或RecyclerView的父视图中使用layout_belowlayout_above或其他类似属性。
您可以使用LinearLayout weightlayout_marginBottom或某事来实现 layout_below或其他。

答案 6 :(得分:0)

我的RecyclerView位于ConstraintLayout内,我也遇到了此类问题,调用RecyclerView的LayoutManager的setAutoMeasureEnabled(false)并没有为我解决问题,此外,此方法在28.0.0版本中已弃用。我所做的就是,我用RelativeLayout包装了RecyclerView,现在它像一个魅力一样工作了。如bugtracker中所述,此“问题”是LinearLayout中的预期行为,不会得到解决。因此,如果可能的话,只需将RecyclerView包装成这样:

<RelativeLayout
    android:id="@+id/container_messages_list"
    android:layout_width="match_parent"
    android:layout_height="0dp"
    android:background="@drawable/chat_back_pattern"
    app:layout_constraintBottom_toTopOf="@+id/bottom_view"
    app:layout_constraintTop_toBottomOf="@+id/toolbar">

    <android.support.v7.widget.RecyclerView
        android:id="@+id/messages_list"
        android:layout_width="match_parent"
        android:layout_height="match_parent">
    </android.support.v7.widget.RecyclerView>
</RelativeLayout>

答案 7 :(得分:0)

最新答案总比没有好,如果您使用NestedScrollView作为RecyclerView的父视图,则应将其删除。

答案 8 :(得分:0)

我有一个类似的问题,我尝试了上面列出的所有解决方案,但没有人工作。 我已经将“有效载荷”填充到“ notifyItemChanged(位置,有效载荷)”,因为我只需要“上传”一个复选框值,所以我在“有效载荷”中传递了该值,而无需调用整个视图持有者的更新。 此解决方案适用于我的回收者视图中的所有视图持有者(最后一个除外)(并且可能适用于所有“回收的”视图持有者,我的意思是那些通过“回收”现有视图来调用“ onBindViewHolder”的人)。 我认为,如果您只有recyclerview,则可以使用“ notifyItemChanged”,并且我也认为嵌套滚动视图和回收器视图会引起“自动滚动”问题。

在这种情况下,我被“ raed”暴露了,所以“ ScroolView-> RecyclerView->“ n” x RecyclerView“。我有一个scroolview,其中包含一个recyclerview,其视图持有者可以包含一个recycler视图。

删除父级ScrollView是一个非常奇怪的解决方案,我无法使用它,因此我将“ onStopNestedScroll”设置为“ ScrollView”而不是RecyclerView。

我个人通过以下方式以编程方式在调用“ notifyItemChanged”方法的代码部分之前使用它:

msvContainer.onStopNestedScroll(mRecyclerView);

其中“ msvContainer”是我的ScrollView,其中包含RecyclerView,而“ mRecyclerView”是我的ScrollView中包含的RecyclerView。 这种方法可以工作99%,因为我第一次调用“ notifyItemChanged”时,该视图仅针对ScrollView向上滚动,因此它在ScrollView中隐藏了一个位于RecyclerView下方的按钮,但不会滚动RecyclerView项。第一次调用后,“ notifyItemChanged”正常运行。

我发现打电话给

msvContainer.stopNestedScroll();

也可以。但是,如果您有多个嵌套滚动视图,我建议对目标视图使用第一种方法。

无论如何,在用完重新更新视图持有人的关键部分用完之后,应该调用“ startNestedScroll”,因为目标视图(在我的情况下为RecyclerView)将不会滚动,直到您调用此方法,这样它才会也要回收他的观察者。

(如果我需要在最内部的Recycler视图内调用“ notifyItemChanged”,则在父Scroll View内的父Recycler视图内有多个Recycler View,我会为每个父对象使用“ stopNestedScroll”方法查看并在滚动关键部分之后重新激活滚动)

希望这对您有所帮助,编码不错! 再见

答案 9 :(得分:0)

就我而言,我所做的就是将recyclerview的高度设置为“ match_parent”。然后在您的MainActivity中执行; recyclerView.smoothScrollToPosition(yourAdapter.getItemCount()-1);

就这些...