ConstraintLayout 使用手册

2020-06-25  本文已影响0人  tandeneck

背景

之前我们在这边文章中 Android 优化之布局优化 了解到可以通过使用 ConstraintLayout 来构建我们的布局,这也是 Android 官方推荐首要使用的,手动拖拽的方式习惯后也大大提高了我们的开发效率,如果你还没了解过 ConstraintLayout ,那就继续往下看吧。如果你已经熟练使用的话,不妨扫一眼,说不定有意外的收获。

添加约束条件

1.常规约束

创建约束条件时,每个视图都必须有两个约束条件:一个水平约束条件,一个垂直约束条件,如果我们什么约束条件都没有添加,控件就会位于ConstraintLayout 的左上角。添加约束条件非常简单,我们可以选择手动拖拽的方式或者直接手动编码的方式,个人喜欢拖拽的方式,如果有误差再在布局文件中进行微调。下面示例为 TextView 添加了上下左右四个约束条件:


可以看到,我们在拖动的过程中,布局文件也会生成相应的代码。上图演示的是链接到父布局,除此之外,我们也可以链接到其他控件中,这里不再做演示,以下是常用的约束条件:

这些约束条件应该都可以顾名思义,一个比较特别的是 layout_constraintBaseline_toBaselineOf,它用于将一个视图的文本基线与另一视图的文本基线对齐,要创建基线约束条件,可以右键点击要约束的文本视图,然后点击 show Baseline,接着点击文本基线并将其拖到另一基线上。

有些同学可能会对 start 和 left、end 和 right 有困惑。其实如果应用只是面向国内市场的话, start 等价于 left,end 等价于 right ,因为中文的书写方向是从左到右的,但是有些语言是从右到左的书写方式,典型的就是阿拉伯语,所以 Android 从 4.2 开始推荐使用 start 、end 来代替 left 、 right,这样在切换到 RTL 语言时,UI 会自动进行镜像翻转,可以保持一致的用户体验。

2.Guideline 约束

我们可以添加垂直或水平的 Guideline 来约束视图,相当于辅助线一样,用户是看不到 Guideline 的。

通过请点击工具栏中的 Guideline


然后点击 Add Vertical Guideline 或 Add Horizontal Guideline,拖动虚线将其重新定位,然后点击引导线边缘的圆圈以切换测量模式,有 固定数值百分比 两种模式。
3.Barrier 约束

与 Guileline 类似,Barrier 是一条隐藏的线,Barrier 的位置是根据其所包含的视图的位置而移动,包含视图的属性是 constraint_referenced_ids,Barrier 可以是垂直或水平的,可以创建到引用视图的顶部、底部、左侧和右侧。以下示例,Barrier 包含了 id 为 tv_1,tv_2 的 TextView,而 id 为 tv_3 的 TextView 在 Barrier 的右侧。

代码:

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout 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="match_parent"
    tools:context=".MainActivity">


    <TextView
        android:id="@+id/tv_1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginTop="16dp"
        android:text="我是最长长长长的"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent" />

    <TextView
        android:id="@+id/tv_2"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginTop="8dp"
        android:text="短小如我"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toBottomOf="@+id/tv_1" />

    <androidx.constraintlayout.widget.Barrier
        android:id="@+id/barrier"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        app:barrierDirection="end"
        app:constraint_referenced_ids="tv_1,tv_2"
        tools:layout_editor_absoluteX="113dp" />


    <TextView
        android:id="@+id/tv_3"
        android:layout_width="0dp"
        android:layout_height="wrap_content"
        android:layout_marginStart="8dp"
        android:layout_marginTop="16dp"
        android:layout_marginRight="10dp"
        android:text="我是在 Barrie 的右边"
        app:layout_constraintLeft_toRightOf="@+id/barrier"
        app:layout_constraintRight_toRightOf="parent"
        app:layout_constraintTop_toTopOf="parent" />
</androidx.constraintlayout.widget.ConstraintLayout>

效果:


调整视图尺寸

我们一般是通过 layout_width 和 layout_height 来为视图指定尺寸,可供的选择有 match_parent、wrap_content 和 具体的数值,ContraintLayout 亦如此,不过它多了一个选择叫作 Match Constraints,在代码中的体现是 0dp,除了可以在代码中直接设置外,我们还可以在编辑器右侧的 Attributes 窗口,点击相应的位置更改,以宽度为例,操作如下:

从动图中可以看到,当我们切换到


layout_width 变为 0dp 了,代表当前的模式为 match_contrains,此时视图是撑满的,这和 layout_constraintWidth_default 的值有关:

调整约束偏差

当我们对某个视图两侧添加约束条件(并且同一维度的视图尺寸为 fixed 或者 wrap_cotent)时,该视图在两个约束条件之间居中,默认偏差为 0.5,对应的属性是 layout_constraintVertical_biaslayout_constraintHorizontal_bias,可以进行对其调整满足业务需求:

将尺寸设置为比例

如果视图至少有一个尺寸设置为 match_constraints(0dp),我么就可以把视图设置为比例的形式,对应的属性是 layout_constraintDimensionRatio ,如下,我们设置了宽充满屏幕,比例为 1:1 的 TextView:

当然,我们可以把宽高都设置为 match_constraints(0dp),这种情况下视图会先满足约束条件,然后把视图指定为该比例的最大尺寸。我们也可以在比例前面加一个 W,或者 H,来约束宽和高,如下:

       <TextView
        android:layout_width="160dp"
        android:layout_height="0dp"
        app:layout_constraintDimensionRatio="W,2:1" />
    <!--假设有足够的空间,最终的宽为 160dp,高为 320dp-->
       <TextView
        android:layout_width="160dp"
        android:layout_height="0dp"
        app:layout_constraintDimensionRatio="H,2:1" />
    <!--假设有足够的空间,最终的宽为 160dp,高为 80dp-->

边距(Margin)

ContraintLayout 的边距只有在有约束条件的情况下才会生效,比如下面这段代码中TextView 没有添加任何约束条件,最后它会显示 ConstraintLayout 的左上角,设置的 margin 不会生效。

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout 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="match_parent"
    tools:context=".MainActivity">
    <TextView
        android:background="#fff000"
        android:id="@+id/textView2"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="TextView"
        android:layout_margin="16dp"/>
</androidx.constraintlayout.widget.ConstraintLayout>

需要注意的是,ConstraintLayout 的边距设置为负值并不会生效,这点和其他传统布局是有区别的。

此外,ContraintLayoutMargin 还提供了 GONE_MARGIN:

当约束目标被设置为 View.GONE 后,设置的 GONE_MARGIN 就会生效。

圆形(角度)定位(Circular positioning)

我们可以通过一个角度和距离来约束两个视图的位置,引用官方的一张图:


对应的属性是 :

<?xml version="1.0" encoding="utf-8"?>
<androidx.constraintlayout.widget.ConstraintLayout 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="match_parent"
    tools:context=".MainActivity">
    
    <View
        android:background="#fff000"
        android:id="@+id/center"
        android:layout_width="50dp"
        android:layout_height="50dp"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent" />
   
    <View
        android:layout_width="50dp"
        android:layout_height="50dp"
        android:background="@drawable/bg_circle"
        app:layout_constraintCircle="@+id/center"
        app:layout_constraintCircleAngle="60"
        app:layout_constraintCircleRadius="100dp"
        tools:ignore="MissingConstraints" />
    
</androidx.constraintlayout.widget.ConstraintLayout>

上面实现的效果如下:


Chain,链控制

链是控制一组视图的,视图可以是水平链和垂直链的一部分,但是使用链并不会使链中的视图对齐在同一方向上,因此,我们要指定额外的约束条件。以下示例是创建一个水平链并以视图的顶端对齐:


创建链之后,会有一个 "链头"(Chain Head),链头是链中的第一个元素(水平链中最左侧的视图,垂直链中最顶部的视图)。链最重要是它的样式,我们可以通过选择链中的元素,右键点击 Cycle Chain mode 进行样式切换,当然也可以在链头里设置 layout_constraintHorizontal_chainStyle , 链的样式取值有以下几种(不会忽略 margin 的取值):

当链的样式设置为 spread 或者 spread inside 时,且我们把一个或多个视图设置为 match_constraints(0dp)。默认情况下,设置了 match_constraints 的属性会把剩余空间均匀分配,但是我们可以使用 layout_constraintHorizontal_weight 和 ayout_constraintVertical_weight 属性来分配权重。这和 LinearLayout 的 layout_weight 的原理是一样的。这种方式也叫做 weight chain

另外,当链设置为 packed 的样式之后,我们可以通过链头的视图偏差 layout_constraintHorizontal_bias 属性来调整整条链的偏差。这种方式称作 packed chain with bias

下面这张来自官方的图可以帮助我们理解链的不同样式之间的区别:


Group

Group 可以把多个控件归为一组,方便隐藏或显示一组控件,相比我们在外面包一层 ViewGroup 的方法,性能上有优势。使用方式如下:

    <androidx.constraintlayout.widget.Group
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        app:constraint_referenced_ids="id1,id2,id3"/>

不过需要注意的是如果控件被包含在 Group 中,单独一个控件设置 View.GONE 并不会使自己隐藏掉(在 Group 为 View.VISIBLE 的情况下),这是因为设置 View.GONE 会导致重绘,调用 Group 的 updatePreLayout 方法,具体逻辑如下:

    public void updatePreLayout(ConstraintLayout container) {
        int visibility = this.getVisibility();
        float elevation = 0.0F;
        if (VERSION.SDK_INT >= 21) {
            elevation = this.getElevation();
        }

        for(int i = 0; i < this.mCount; ++i) {
            int id = this.mIds[i];
            View view = container.getViewById(id);
            if (view != null) {
                view.setVisibility(visibility);
                if (elevation > 0.0F && VERSION.SDK_INT >= 21) {
                    view.setElevation(elevation);
                }
            }
        }
    }

可以看出,updatePreLayout 方法会把 Group 内的视图的可见性设置为和 Group 的一样。

总结

随着官方的不断完善和优化,与刚出来的时候相比,ConstraintLayout 无论是使用上还是性能上都有了很大的提升,如果还没在项目中使用的同学,是时候上车了。

参考

上一篇 下一篇

猜你喜欢

热点阅读