RecyclerView项layout_weight,大小不更新

时间:2016-08-18 20:45:26

标签: android android-layout android-recyclerview

我的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]                |
+----------------------------------------+

14 个答案:

答案 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>


结果:

enter image description here

答案 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

Usage of forceLayout(), requestLayout() and invalidate()

答案 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)

我认为你期待像这样enter image description here

的Out put simile 顶部容器中的

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>