我有一个带有适配器的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"/>
答案 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_below
,layout_above
或其他类似属性。
您可以使用LinearLayout weight
,layout_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);
就这些...