Android约束布局:如何动态添加一个视图并使其位于另一个视图下方

9
我正在尝试在运行时将TextView一个接一个地添加到Constraint Layout中。但我总是只得到一个TextView,其余的都被隐藏在它后面。我尝试了几种方法,包括链接视图,但似乎没有什么作用。
private void method(int position)
    {
        ConstraintSet set = new ConstraintSet();
        TextView textView = new TextView(getContext());
        int textViewId = 100 + position;
        //previousTextViewId = textViewId;
        textView.setId(textViewId);
        ConstraintLayout.LayoutParams layoutParams = new ConstraintLayout.LayoutParams(0, WRAP_CONTENT);
        layoutParams.rightToRight = PARENT_ID;
        layoutParams.leftToLeft = guideline_60.getId(); //Vertical GuideLine of 60%
        layoutParams.rightMargin = 8;
        textView.setLayoutParams(layoutParams);
        if (Build.VERSION.SDK_INT < 23)
        {
            textView.setTextAppearance(getContext(), R.style.textStyle);
        }
        else
        {
            textView.setTextAppearance(R.style.textStyle);
        }
        textView.setBackgroundColor(backgroundColor);
        textView.setText(categoryName);
        textView.setGravity(Gravity.CENTER);
//markerLayout is the ConstraintLayout 
        markerLayout.addView(textView, position);
        set.clone(markerLayout);
        //set.addToVerticalChain(textView.getId(),previousTextViewId,PARENT_ID);
        set.connect(textView.getId(), ConstraintSet.TOP, markerLayout.getId(), ConstraintSet.TOP, 60);
        set.applyTo(markerLayout);
    }

我期望看到像这样的内容 -

在此输入图片描述


该图形界面涉及IT技术相关内容,但具体细节不在文本范围内。
2个回答

11

你的意思是说所有TextView的顶部都连接在父视图的顶部:

set.connect(textView.getId(), ConstraintSet.TOP, 
        markerLayout.getId(), ConstraintSet.TOP, 60);
你想要表达的是,一个顶部连接到另一个底部 - 就像这样:

一个顶部连接到另一个底部 - 就像这样:

set.connect(textView.getId(), ConstraintSet.TOP, 
    previousTextView.getId(), ConstraintSet.BOTTOM, 60);

你的方法被称为方法吗?


3
我试图将代码泛化,因此将方法重命名为“method” :)。如果 (position == 0),下面的代码仍然存在同样的问题: { Timber.d("Position - "+ position); set.connect(textView.getId(), ConstraintSet.TOP, markerLayout.getId(), ConstraintSet.TOP, 16); } 否则 { Timber.d("Position - "+ position); set.connect(textView.getId(), ConstraintSet.TOP, previousTextViewId, ConstraintSet.BOTTOM, 16); } - sku
1
我的错,你的解决方案完美地运作了。在 set.Connect 之前,我曾经写过 previousTextViewId = textViewId; ,而现在将它移动到了后面,并且它也运行良好。感谢 Nick。 - sku

1
我知道这是一个老的回答,但@Temporary拯救了我的一天!!我使用行布局来做同样的事情。我没有得到完美的解决方案来动态添加行布局,所以这可能对其他人有所帮助。
这里是一个简短的例子: activity_main
<androidx.core.widget.NestedScrollView
        android:layout_width="0dp"
        android:layout_height="0dp"
        android:fillViewport="true"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintEnd_toStartOf="@id/guideline_end"
        app:layout_constraintHorizontal_bias="1.0"
        app:layout_constraintStart_toEndOf="@id/guideline_start"
        app:layout_constraintTop_toBottomOf="@id/cvFDToolbar">


        <androidx.constraintlayout.widget.ConstraintLayout
            android:id="@+id/clItem"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"/>


    </androidx.core.widget.NestedScrollView>

Here is row file will add dynamically.

row_main

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:android="http://schemas.android.com/apk/res/android">


    <TextView
        android:id="@+id/txtTitle"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginTop="@dimen/_5sdp"
        android:fontFamily="@font/raleway_medium"
        android:text="TextView"
        android:textColor="@color/black"
        android:textSize="@dimen/_14ssp"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent" />

    <EditText
        android:id="@+id/edtValue"
        android:layout_width="0dp"
        android:layout_height="wrap_content"
        android:layout_marginTop="@dimen/_5sdp"
        android:background="@color/gray"
        android:inputType="textPersonName"
        android:padding="@dimen/_5sdp"
        android:text="Name"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toBottomOf="@+id/txtTitle" />
</androidx.constraintlayout.widget.ConstraintLayout>

这段代码可以动态地添加行布局。将代码添加到MainActivity文件中即可。

val set = ConstraintSet()

        for (i in 0..10) {

            val inflater = LayoutInflater.from(this)

            inflatedLayout = inflater.inflate(
                com.doctor24_7.R.layout.row_pre_consultation_form,
                clItem as ViewGroup,
                false
            )

            inflatedLayout.id = 100 + i
            clItem.addView(inflatedLayout, i)
            set.clone(clItem)
            if (i ==0) {
                set.connect(
                    ConstraintSet.PARENT_ID,
                    ConstraintSet.LEFT,
                    ConstraintSet.PARENT_ID,
                    ConstraintSet.LEFT,
                    60
                )
                set.connect(
                    ConstraintSet.PARENT_ID,
                    ConstraintSet.RIGHT,
                    ConstraintSet.PARENT_ID,
                    ConstraintSet.RIGHT
                )
                set.connect(
                    inflatedLayout.id,
                    ConstraintSet.TOP,
                    ConstraintSet.PARENT_ID,
                    ConstraintSet.TOP
                )
            } else {
                set.connect(
                    ConstraintSet.PARENT_ID,
                    ConstraintSet.LEFT,
                    ConstraintSet.PARENT_ID,
                    ConstraintSet.LEFT
                )
                set.connect(
                    ConstraintSet.PARENT_ID,
                    ConstraintSet.RIGHT,
                    ConstraintSet.PARENT_ID,
                    ConstraintSet.RIGHT
                )
                set.connect(
                    inflatedLayout.id,
                    ConstraintSet.TOP,
                    previousid,
                    ConstraintSet.BOTTOM, resources.getDimension(R.dimen._10sdp).toInt()
                )
            }
            set.applyTo(clItem)
            previousid = inflatedLayout.id

        }

1
谢谢你的回答。这个很好用,我想提一件事,即在for循环之前添加变量var previousid = 100 - Zeeshan Akhtar
你不应该只使用"100"作为Id。而是应该使用这个方法:View.generateViewId()。 - android developer

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接