79

背景

谷歌宣布了一种名为“ ConstraintLayout ”的新布局,它应该是终极布局,它可以取代所有布局,同时保持平坦(没有嵌套布局)并具有更好的性能。

问题

问题是,除了 Google IO 上的视频之外,我几乎看不到任何可以帮助我解决这个问题的教程。

我想做的是,鉴于我在另一个布局中有一个垂直居中的 LinearLayout - 将它们都转换为单个 ConstraintLayout。

毕竟,这就是这个新布局的目的……

我希望处理的布局如下所示:

在此处输入图像描述

请注意,中心的视图仅垂直居中,并且 2 个 textView 位于 ImageView 的右侧,ImageView 也是垂直居中的。

这一切都适用于具有 2 个 TextView 的 LinearLayout 的 RelativeLayout,但我想知道如何将它们转换为单个 ConstraintLayout。

这是我展示的示例 XML:

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:minHeight="?attr/listPreferredItemHeightSmall">

    <ImageView
        android:id="@+id/appIconImageView"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentLeft="true"
        android:layout_alignParentStart="true"
        android:layout_centerVertical="true"
        android:layout_marginEnd="4dp"
        android:layout_marginLeft="2dp"
        android:layout_marginRight="4dp"
        android:layout_marginStart="2dp"
        android:adjustViewBounds="true"
        android:src="@android:drawable/sym_def_app_icon"
        tools:ignore="ContentDescription"/>

    <LinearLayout
        android:id="@+id/appDetailsContainer"
        android:layout_width="0px"
        android:layout_height="wrap_content"
        android:layout_centerVertical="true"
        android:layout_toEndOf="@+id/appIconImageView"
        android:layout_toLeftOf="@+id/overflowView"
        android:layout_toRightOf="@+id/appIconImageView"
        android:layout_toStartOf="@+id/overflowView"
        android:orientation="vertical">

        <TextView
            android:id="@+id/appLabelTextView"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:ellipsize="marquee"
            android:text="label"
            android:textAppearance="?android:attr/textAppearanceLarge"
            android:textDirection="locale"
            tools:ignore="HardcodedText,UnusedAttribute"/>

        <TextView
            android:id="@+id/appDescriptionTextView"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:ellipsize="marquee"
            android:minLines="3"
            android:text="description"
            android:textAppearance="?android:attr/textAppearanceSmall"
            android:textDirection="locale"
            tools:ignore="HardcodedText,UnusedAttribute"/>
    </LinearLayout>

    <ImageView
        android:id="@+id/overflowView"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentEnd="true"
        android:layout_alignParentRight="true"
        android:layout_alignParentTop="true"
        android:adjustViewBounds="true"
        android:background="?attr/selectableItemBackground"
        android:clickable="true"
        android:padding="10dp"
        app:srcCompat="@drawable/ic_more_vert_black_24dp"

        tools:src="@drawable/ic_more_vert_black_24dp"
        tools:ignore="ContentDescription"/>

    <ImageView
        android:id="@+id/isSystemAppImageView"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignEnd="@+id/overflowView"
        android:layout_alignLeft="@+id/overflowView"
        android:layout_alignParentBottom="true"
        android:layout_alignRight="@+id/overflowView"
        android:layout_alignStart="@+id/overflowView"
        android:adjustViewBounds="true"
        android:scaleType="centerInside"
        app:srcCompat="@drawable/ic_warning_black_24dp"
        tools:ignore="ContentDescription"
        tools:src="@drawable/ic_warning_black_24dp"/>

</RelativeLayout>

我试过的

我尝试阅读一些文章并观看 Google 的一些视频:

那没有帮助,所以我尝试使用它,希望我能自己找出如何使用它。但我不知道该怎么做。我尝试使用该功能来转换布局,但这会使视图变得一团糟,并增加了我不想拥有的额外边距。

问题

如何将 2 个布局转换为单个 ConstraintLayout ?

4

5 回答 5

110

看看我的回答here

ContraintLayout包含一个功能 -Chains可以实现您的要求:

链在单个轴(水平或垂直)上提供类似组的行为。

如果一组小部件通过双向连接链接在一起,则它们被视为链

创建链后,有两种可能性:

  • 在可用空间中散布元素
  • 链也可以“打包”,在这种情况下,元素被组合在一起

至于你的情况,你必须打包你的labeldescriptionTextViews 并将它们垂直居中在你的父级中:

(确保您使用ConstraintLayout支持链的版本)

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


    <TextView
        android:id="@+id/textView"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginStart="8dp"
        android:layout_marginTop="16dp"
        android:text="TextView"
        app:layout_constraintBottom_toTopOf="@+id/button"
        app:layout_constraintLeft_toRightOf="@+id/imageView2"
        app:layout_constraintTop_toTopOf="parent"
        app:layout_constraintHorizontal_chainStyle="packed"/>

    <TextView
        android:id="@+id/button"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginBottom="16dp"
        android:layout_marginStart="8dp"
        android:layout_marginTop="8dp"
        android:text="Button\nMkay"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintLeft_toRightOf="@+id/imageView2"
        app:layout_constraintTop_toBottomOf="@+id/textView"/>

    <ImageView
        android:id="@+id/imageView"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginEnd="16dp"
        android:layout_marginTop="16dp"
        app:layout_constraintRight_toRightOf="parent"
        app:layout_constraintTop_toTopOf="parent"
        app:srcCompat="@mipmap/ic_launcher"/>

    <ImageView
        android:id="@+id/imageView2"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginBottom="16dp"
        android:layout_marginStart="16dp"
        android:layout_marginTop="16dp"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintLeft_toLeftOf="parent"
        app:layout_constraintTop_toTopOf="parent"
        app:srcCompat="@mipmap/ic_launcher"/>

    <ImageView
        android:id="@+id/imageView3"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginBottom="16dp"
        android:layout_marginEnd="16dp"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintRight_toRightOf="parent"
        app:srcCompat="@mipmap/ic_launcher"/>
</android.support.constraint.ConstraintLayout>

2019 年 6 月 25 日更新(@Saeid Z):

现在在约束布局 1.1.3 中,我们必须使用app:layout_constraintHorizontal_chainStyle="packed"而不是app:layout_constraintVertical_chainPacked="true"

于 2016-10-18T07:38:56.010 回答
8

设置app:layout_constraintVertical_bias="0.5"为需要垂直居中的视图,bias 属性仅在您指定边界约束时才有效(例如,top 和 bottom 用于垂直偏差,left 和 right 用于水平偏差)

一个例子:

<android.support.constraint.ConstraintLayout
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:id="@+id/constraintLayout">

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        app:layout_constraintTop_toTopOf="@id/constraintLayout"
        app:layout_constraintBottom_toBottomOf="@id/constraintLayout"
        app:layout_constraintVertical_bias="0.5" />

</android.support.constraint.ConstraintLayout>

让它在我的布局中工作:https ://github.com/hidroh/tldroid/blob/master/app/src/main/res/layout/activity_main.xml ,非常相似的布局,虽然我把东西放在 1/屏幕高度的 3 分之一。

在此处输入图像描述

于 2016-05-30T05:46:31.263 回答
5

要垂直或水平居中,请在布局上设置相反的约束。

垂直居中

    app:layout_constraintTop_toTopOf="parent"
    app:layout_constraintBottom_toBottomOf="parent"

水平居中

    app:layout_constraintRight_toRightOf="parent"
    app:layout_constraintLeft_toLeftOf="parent"
于 2017-07-14T04:33:01.693 回答
5

编辑:这个答案是在链可用之前写的。请立即使用链,请参阅上面的答案:https ://stackoverflow.com/a/40102296/1402641


现在我相信你唯一的选择是将两个文本视图包装在另一个布局中——可能线性布局最适合这种情况。

但是约束布局背后的团队说他们想引入“虚拟容器”,它正好服务于这个用例:你将两个或多个视图组合在一起,并在容器上设置一个约束(比如垂直居中)。这个想法是它不是约束布局内的完整嵌套布局,而只是约束布局用来定位其子级的东西 - 因此它应该提供比嵌套更好的性能。

他们在I/O 演讲中提到了这一点(与确切时间相关)。所以我想请继续关注。

于 2016-06-02T17:10:48.233 回答
2

看看我的例子(ConstraintLayout 中的中心组件

现在您可以将“图像”居中(约束左上-下),将“标签”顶部约束到“图像”顶部,将“描述”顶部约束到“标签”底部。在下面的示例中,我将按钮高度与右侧的两个文本视图匹配(这是您的情况)。其他文本视图受到上述链接的限制。

在此处输入图像描述

更新:在下面回答您的评论:

我已经安装了你的应用程序,这只是我目前能想到的。由于它是一个 ViewHolder 布局,因此 layout_height 可以设置为 wrap_content,或者您可以根据需要对其进行修复。如果您愿意,我可以向您发送 xml,我不想淹没答案。

Top TextView 左约束被约束为 ImageView 的右约束。此外,顶部 TextView -> 顶部约束被限制在容器顶部,所以是正确的。底部 TextView 被限制在容器底部。中间的 TextView 被限制为与顶部 TextView 的宽度匹配,并且与 ImageView 没有连接。

在此处输入图像描述

于 2016-06-09T10:06:31.033 回答