我的RecyclerView项目中有两个文本视图。其中一个视图有layout_weight =“1”,如果视图很宽,则可以看到另一个视图。
问题是当视图被回收时视图大小没有更新,所以当我滚动列表时,textView id:name width不正确。
我应该以某种方式强制视图更新布局吗?
我的RecyclerView项目:
<LinearLayout
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginLeft="16dp"
android:gravity="center_vertical"
android:orientation="horizontal">
<TextView
android:id="@+id/name"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_marginRight="8dp"
android:layout_weight="1"
android:ellipsize="end"
android:maxLines="1" />
<LinearLayout
android:id="@+id/code_container"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center_vertical"
android:background="@drawable/background_code"
android:orientation="vertical">
<TextView
android:id="@+id/code"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:ellipsize="end"
android:paddingBottom="1dp"
android:paddingLeft="7dp"
android:paddingRight="7dp"
android:paddingTop="1dp"
android:singleLine="true"/>
</LinearLayout>
</LinearLayout>
我正在使用RecyclerView v.23.4.0。
更新
1。我试图在适配器onBindViewHolder()的末尾调用我的RecyclerView项目布局根requestLayout。
这没有任何影响,我的观点没有更新。
@Override
public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {
...
viewHolder.mLayoutRoot.requestLayout();
}
2。也许问题是当我调用requestLayout时视图还没有完全绘制出来?所以我尝试设置GlobalLayoutListener并在那里调用requestLayout。我认为这有点帮助,但如果我有足够的项目,在我的列表中并不是所有的视图都会更新。在我的名单中的Owerall我有大约10件物品。
ViewTreeObserver vto = viewHolder.mLayoutRoot.getViewTreeObserver();
vto.addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
@Override
public void onGlobalLayout() {
viewHolder.mLayoutRoot.requestLayout();
}
});
3。我尝试使用带有shrinkColumns属性的TableLayout来接近类似的布局,但结果是一样的。名称文本视图widht并不总是正确设置。此外,看起来requestLayout根本没有任何影响(或者我在错误的地方调用它?)
4。 (2016年8月28日)
我将我的代码剥离到最低限度以重现这一点。
我设法使用setIsRecyclable(false)修复此问题,就像dosssik建议的那样,但我认为它在性能方面不是很好的解决方案?
这是我的完整清单项目:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="wrap_content"
android:layout_height="150dp"
android:layout_marginLeft="16dp"
android:orientation="horizontal">
<TextView
android:id="@+id/name"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_marginRight="8dp"
android:layout_weight="1"
android:ellipsize="end"
android:maxLines="1" />
<TextView
android:id="@+id/code"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:background="@drawable/background_stop_code_small"
android:ellipsize="end"
android:paddingBottom="1dp"
android:paddingLeft="7dp"
android:paddingRight="7dp"
android:paddingTop="1dp"
android:singleLine="true"
android:textSize="11sp"/>
</LinearLayout>
我的适配器代码:
我也尝试调用invalidate(),但看不到任何影响。
@Override
public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {
Item item = getItem(position);
NormalViewHolder viewHolder = (NormalViewHolder) holder;
viewHolder.mName.setText(item.getName());
//viewHolder.mName.invalidate(); // No help
viewHolder.mCode.setText(item.getCode());
//viewHolder.mCode.invalidate(); // No help
}
预期行为:
+----------------------------------------+
|[Name][Code] |
+----------------------------------------+
|[Name qwertyuiopasdfghjklzxcvb...][Code]|
+----------------------------------------+
如果我向上/向下滚动列表,这就是现在的工作方式。因此布局不会更新,代码也不是alignet的名称。名字也没有占据足够的空间。
+----------------------------------------+
|[Name][Code] |
+----------------------------------------+
|[Name qwer...] [Code] |
+----------------------------------------+
|[Name qwertyuiopa][Code] |
+----------------------------------------+
|[Name qwe] [Code] |
+----------------------------------------+
答案 0 :(得分:3)
这个xml在不使用setIsRecyclable(false);
的情况下适合我
在一些Android版本(如kitkat)中,代码:android:ellipsize="end"
仅适用于android:singleLine="true"
,因此将其添加到第一个TextView“@ + id / name”中它将起作用!
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="wrap_content"
android:layout_height="150dp"
android:layout_marginLeft="16dp"
android:orientation="horizontal">
<TextView
android:id="@+id/name"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_marginRight="8dp"
android:layout_weight="1"
android:ellipsize="end"
android:singleLine="true"
android:maxLines="1"
/>
<TextView
android:id="@+id/code"
android:layout_weight="0"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:background="@drawable/background_stop_code_small"
android:ellipsize="end"
android:paddingBottom="1dp"
android:paddingLeft="7dp"
android:paddingRight="7dp"
android:paddingTop="1dp"
android:singleLine="true"
android:textSize="11sp"/>
</LinearLayout>
答案 1 :(得分:2)
As it seems code is of constant length, you can use following UI deisgn;
<RelativeLayout
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginLeft="16dp"
android:gravity="center_vertical">
<TextView
android:id="@+id/name"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:align_parentLeft="true"
android:layout_toLeftOf="@+id/code"
android:layout_marginRight="8dp"
android:ellipsize="end"
android:maxLines="1" />
<TextView
android:id="@+id/code"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:background="@drawable/background_code"
android:gravity="center"
android:align_ParentRight="true"
android:ellipsize="end"
android:paddingBottom="1dp"
android:paddingLeft="7dp"
android:paddingRight="7dp"
android:paddingTop="1dp"
android:singleLine="true"/>
</RelativeLayout>
答案 2 :(得分:1)
RecyclerView
将重用ViewHolder
个对象,因此这些对象的布局会被夸大一次,然后在列表的生命周期内重复使用。尝试拨打根requestLayout
上的LinearLayout
。
答案 3 :(得分:1)
您需要使两个TextView
视图无效,并确保在为其设置新文本值后执行此操作:
@Override
public void onBindViewHolder(ItemHolder holder, int position)
{
...
holder.nameTextView.invalidate();
holder.codeTextView.invalidate();
}
作为替代方案,如果您不介意将代码视图与右侧对齐,则可以将LinearLayout
替换为RelativeLayout
并在名称视图和代码容器之间设置适当的约束:
<TextView
android:id="@+id/name"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginRight="8dp"
android:layout_centerVertical="true"
android:layout_alignParentStart="true"
android:layout_alignParentLeft="true"
android:layout_toLeftOf="@+id/code_container"
android:layout_toStartOf="@+id/code_container"
android:ellipsize="end"
android:maxLines="1" />
<LinearLayout
android:id="@+id/code_container"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerVertical="true"
android:layout_alignParentEnd="true"
android:layout_alignParentRight="true"
android:background="@drawable/background_code"
android:orientation="vertical">
<TextView
android:id="@+id/code"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:ellipsize="end"
android:paddingBottom="1dp"
android:paddingLeft="7dp"
android:paddingRight="7dp"
android:paddingTop="1dp"
android:singleLine="true"/>
</LinearLayout>
答案 4 :(得分:1)
我终于得到了正确的答案。作为我的测试,您只需在方法onBindViewHolder
中添加两行,如下所示:
@Override
public void onBindViewHolder(ItemHolder holder, int position)
{
holder.mName.setSingleLine(false); //first line
holder.mName.setText("name");
holder.mCode.setText("");
holder.mName.setSingleLine(true); //second line
}
希望对你有所帮助。
答案 5 :(得分:1)
你应该增加内部线性布局的重量然后它也会
<TextView
android:id="@+id/name"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_marginRight="8dp"
android:layout_weight="2"
android:ellipsize="end"
android:maxLines="1" />
<LinearLayout
android:id="@+id/code_container"
android:layout_width="0dp"
android:layout_weight="1"
android:layout_height="wrap_content"
android:layout_gravity="center_vertical"
android:background="@drawable/background_code"
android:orientation="vertical">
<TextView
android:id="@+id/code"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:ellipsize="end"
android:paddingBottom="1dp"
android:paddingLeft="7dp"
android:paddingRight="7dp"
android:paddingTop="1dp"
android:singleLine="true"/>
</LinearLayout>
试试这会对你有用
答案 6 :(得分:0)
我在使用RecyclerView的聊天活动中遇到了布局刷新问题:我在onBindViewHolder中调用了一个方法来调整任何聊天消息的边距,在代码片段下方: .. RecyclerView.LayoutParams lp =(RecyclerView.LayoutParams)this.mItemView.getLayoutParams(); lp.setMargins(0,0,50,0);
答案 7 :(得分:0)
我在recyclerView上遇到了非常类似的问题。 有时,当物品回收时,它不能正常工作。
对我来说有助于关闭可回收利用。
setIsRecyclable(false);
你可以在ViewHolder的构造器中这样做:
class Holder extends RecyclerView.ViewHolder {
public Holder(View itemView) {
super(itemView);
setIsRecyclable(false);
}
}
当然,这不是很棒的解决方案,但它是帮助我的唯一解决方案。
希望它会对你有所帮助。
答案 8 :(得分:0)
尝试这个我不是使用编辑器但是如果xml存在任何问题则会自动更正
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent" //i changed this
android:layout_height="150dp"
android:layout_marginLeft="16dp"
android:weightsum="1"//added this
android:orientation="horizontal">
<TextView
android:id="@+id/name"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_marginRight="8dp"
android:layout_weight="1"
android:ellipsize="end"
android:singleLine="true" />//added this
<TextView
android:id="@+id/code"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:background="@drawable/background_stop_code_small"
android:ellipsize="end"
android:paddingBottom="1dp"
android:paddingLeft="7dp"
android:paddingRight="7dp"
android:paddingTop="1dp"
android:singleLine="true"
android:textSize="11sp"/>
</LinearLayout>
以上将起作用
如果您仍然坚持更新布局,请始终知道您需要的更新是否为测量值,然后您需要与View.invalidate()
一起调用View.requestLayout()
答案 9 :(得分:0)
尝试使用此功能,我有同样的问题,我使用此解决了它:
@Override
public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
View rootView = LayoutInflater.from(context).inflate(R.layout.review_data, null, false);
RecyclerView.LayoutParams lp = new RecyclerView.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT);
rootView.setLayoutParams(lp);
return new ViewHolder(rootView);
}
在onCreateViewHolder中设置布局参数,如上所述。
答案 10 :(得分:0)
仅对其中一个设置权重是错误的,因为当code
很长时,name
将会消失。
我不知道你想要实现什么,但即使有一个简单的ListView和一个简单的布局,如下面的项目,它们永远不会消失,我认为它们看起来更好:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="wrap_content"
android:layout_height="150dp"
android:gravity="left"
android:layout_marginLeft="16dp"
android:orientation="horizontal">
<TextView
android:id="@+id/name"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_weight=".5"
android:layout_marginRight="8dp"
android:ellipsize="end"
android:text="dfsdd45te5sf"
android:maxLines="1" />
<TextView
android:id="@+id/code"
android:layout_weight=".5"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:background="@android:color/holo_green_dark"
android:ellipsize="end"
android:paddingBottom="1dp"
android:paddingLeft="7dp"
android:paddingRight="7dp"
android:paddingTop="1dp"
android:layout_marginRight="1dp"
android:text="ertertete5tte5tte5e5te5t"
android:singleLine="true"
android:textSize="12sp"/>
</LinearLayout>
答案 11 :(得分:0)
LinearLayout将“wrap_content”更改为“match_parent”
查看本教程 http://wiki.workassis.com/android-recyclerview-example/
答案 12 :(得分:0)
我分享了我不需要使用的解决方案
setIsRecyclable(false)
在xml上使用以下自定义TextView。
public class TextViewForRecyclerView extends android.support.v7.widget.AppCompatTextView {
public TextViewForRecyclerView(Context context) {
super(context);
}
public TextViewForRecyclerView(Context context, @Nullable AttributeSet attrs) {
super(context, attrs);
}
public TextViewForRecyclerView(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
}
@Override
public void setText(CharSequence text, BufferType type) {
super.setText(text, type);
requestLayout();
}
}
答案 13 :(得分:-1)
好吧,我没有得到你所面对的确切问题......但我可以解决一个问题 在你的代码中看到的是重量,你是以错误的方式使用重量。如果你在一个小部件中使用权重将无效,至少你必须使用两个小部件在它们之间给予适当的权重,并使用其父级中的权重和来正常工作。例如,见下面修改后的代码: -
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginLeft="16dp"
android:gravity="center_vertical"
android:orientation="horizontal"
android:weightSum="3">
<TextView
android:id="@+id/name"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_marginRight="8dp"
android:layout_weight="1"
android:ellipsize="end"
android:text="qwertyuioopasdfsd" />
<LinearLayout
android:id="@+id/code_container"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_gravity="center_vertical"
android:layout_weight="2"
android:background="@drawable/background_code"
android:orientation="vertical">
<TextView
android:id="@+id/code"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:ellipsize="end"
android:paddingBottom="1dp"
android:paddingLeft="7dp"
android:paddingRight="7dp"
android:paddingTop="1dp" />
</LinearLayout>
</LinearLayout>