美文网首页
ConstraintLayout从入门到精通

ConstraintLayout从入门到精通

作者: 码农来搬砖 | 来源:发表于2019-05-28 14:56 被阅读0次

1、介绍

在Android 开发最常用的是:线性布局LinearLayout和相对布局RelativeLayout,在使用的过程中都是各种嵌套,严重影响了代码的可读性。
约束布局ConstraintLayout的出现主要是为了解决布局嵌套过多的问题,以灵活的方式定位和调整小部件。可以在Api9以上的Android系统使用它,从Android Studio 2.3起,官方的模板默认使用ConstraintLayout。

2、如何使用ConstraintLayout

对于刚上手的童鞋,看着各种不习惯,等使用熟练以后就会发现它的各种好处了,代码终于不用在像以前那样嵌套到恶心的地步。废话不多说,下面看看具体怎么用的。
当然英语不错的也可以去读读官方文档,介绍的还是很详细的:ConstraintLayout官方文档
(1)相对定位,我个人理解用法上和相对布局也差不多吧

图一、B在A的右侧
<Button android:id="@+id/buttonA" ... />
<Button android:id="@+id/buttonB" ...
        app:layout_constraintLeft_toRightOf="@+id/buttonA" />

约束条件如下:

layout_constraintLeft_toLeftOf
layout_constraintLeft_toRightOf
layout_constraintRight_toLeftOf
layout_constraintRight_toRightOf
layout_constraintTop_toTopOf
layout_constraintTop_toBottomOf
layout_constraintBottom_toTopOf
layout_constraintBottom_toBottomOf
layout_constraintBaseline_toBaselineOf
layout_constraintStart_toEndOf
layout_constraintStart_toStartOf
layout_constraintEnd_toStartOf
layout_constraintEnd_toEndOf

这里的属性对比上面的一个例子,相信大家都看的懂了,其中有一个比较特殊:
layout_constraintBaseline_toBaselineOf,指的是文本基线对齐

图二、基线

(2)边距,其实这个就没什么好说的了,和之前的相对布局和线下布局都一样,然后就列举一下吧!

android:layout_marginStart
android:layout_marginEnd
android:layout_marginLeft
android:layout_marginTop
android:layout_marginRight
android:layout_marginBottom

(3)goneMargin,主要用于约束的控件可见性被设置为gone的时候使用的margin值,这个就比较人性化了,虽然不是很常用,但是用起来确实方便太多了,具体属性如下:

layout_goneMarginStart
layout_goneMarginEnd
layout_goneMarginLeft
layout_goneMarginTop
layout_goneMarginRight
layout_goneMarginBottom

举个栗子:如果Button2的左面约束在Button1的右面,Button1设有10dp的左边距,Button2没有左边距,但是给Button2设置10dp的goneMarginStart,现在把Button1可见性设置为GONE,就会如下图显示Button2会有10dp的左边距。

图三

(4)居中和偏移

<android.support.constraint.ConstraintLayout ...>
             <Button android:id="@+id/button" ...
                 app:layout_constraintLeft_toLeftOf="parent"
                 app:layout_constraintRight_toRightOf="parent/>
         </>

这里是左右居中,上下居中也同理。

图四、居中样式

然后就是偏移,可以设置边距同样也能达到偏移的效果,这个也是大家都知道的,所以再多介绍也没什么意义,这里要介绍的是另外两个新的属性:

layout_constraintHorizontal_bias
layout_constraintVertical_bias

例如,下面将使左侧具有30%的偏移而不是默认的50%,这样左侧将更短,小部件将更倾向于左侧。上下偏移这里就不再举例介绍了。

<android.support.constraint.ConstraintLayout ...>
             <Button android:id="@+id/button" ...
                 app:layout_constraintHorizontal_bias="0.3"
                 app:layout_constraintLeft_toLeftOf="parent"
                 app:layout_constraintRight_toRightOf="parent/>
         </>
图五、左右偏移样式

(5)圆形定位,这个就显得就比较高端了,虽然感觉用到的地方不是很多,但就是感觉很吊的样子。可以以一个角度和一个距离约束一个小部件中心相对于另一个小部件中心。这允许您在一个圆上定位一个小部件。可以使用以下属性:

layout_constraintCircle : 引用另一个小部件ID
layout_constraintCircleRadius : 到其他窗口小部件中心的距离
layout_constraintCircleAngle : 小部件应该处于哪个角度(度,从0到360)
图六、圆形定位
<Button android:id="@+id/buttonA" ... />
  <Button android:id="@+id/buttonB" ...
      app:layout_constraintCircle="@+id/buttonA"
      app:layout_constraintCircleRadius="100dp"
      app:layout_constraintCircleAngle="45" />

(6)可见性行为,其实就是View.GONE,和之前的一样,需要注意的就是当有控件设置了goneMargin时的情况,这个上面有说过。
(7)限制尺寸,约束布局的最小或最大尺寸,可以为约束布局本身定义最小和最大大小,一般设置宽度使用WRAP_CONTENT :

android:minwidth 设置布局的最小宽度
android:minheight 设置布局的最小高度
android:maxwidth 设置布局的最大宽度
android:maxheight 设置布局的最大高度

当ConstraintLayout为1.1版本以下时,使用以上属性需要设置:

app:layout_constrainedWidth=”true|false”
app:layout_constrainedHeight=”true|false”

官方不推荐使用MATCH_CONSTRAINT,建议使用0dp去代替

图七

a、是包裹内容,b、设置了0dp,c、设置了0dp并设置了左边距
下面是c的例子代码:

<TextView
        android:layout_width="0dp"
        android:layout_height="wrap_content"
        android:layout_marginLeft="50dp"
        app:layout_constraintLeft_toLeftOf="parent"
        app:layout_constraintRight_toRightOf="parent" />

在约束布局中宽高的维度 match_parent 被 0dp 代替,默认生成的大小占所有的可用空间。那么有以下几个属性可以使用:
layout_constraintWidth_min and layout_constraintHeight_min //设置最小尺寸
layout_constraintWidth_max and layout_constraintHeight_max //设置最大尺寸
layout_constraintWidth_percent and layout_constraintHeight_percent //设置相对于父类的百分比
(8)宽高比,当宽或高至少有一个尺寸被设置为0dp时,可以通过属性layout_constraintDimensionRatio设置宽高比。

<Button android:layout_width="wrap_content"
        android:layout_height="0dp"
        app:layout_constraintDimensionRatio="1:1" />

在设置宽高比的值的时候,还可以在前面加W或H,分别指定宽度或高度限制。

<Button android:layout_width="0dp"
        android:layout_height="0dp"
        app:layout_constraintDimensionRatio="H,16:9"
        app:layout_constraintBottom_toBottomOf="parent"
        app:layout_constraintTop_toTopOf="parent"/>

app:layout_constraintDimensionRatio="H,16:9"指的是 高:宽=16:9
app:layout_constraintDimensionRatio="W,16:9"指的是 宽:高=16:9

(9)链,如果两个或以上控件通过下图的方式约束在一起,就可以认为是他们是一条链(图为横向的链,纵向同理)。

图八、最小链

链头,链由链的第一个元素(链的“头”)上设置的属性控制:

图九、链头
    <TextView
        android:id="@+id/TextView1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        app:layout_constraintLeft_toLeftOf="parent"
        app:layout_constraintRight_toLeftOf="@+id/TextView2" />

    <TextView
        android:id="@+id/TextView2"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        app:layout_constraintLeft_toRightOf="@+id/TextView1"
        app:layout_constraintRight_toLeftOf="@+id/TextView3" />

    <TextView
        android:id="@+id/TextView3"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        app:layout_constraintLeft_toRightOf="@+id/TextView2"
        app:layout_constraintRight_toRightOf="parent" />

在链的第一个元素上设置属性layout_constrainthorizontal_chain style或layout_constraintvertical_chainstyle时,链的行为将根据指定的样式更改(默认为chain_spread)。
1、CHAIN_SPREAD — 元素将展开(默认样式)
2、CHAIN_SPREAD_INSIDE —— 展开元素,但链的两端贴近parent;
3、通过layout_constrainthorizontal_weight属性设置了权重
4、CHAIN_PACKED —— 链的元素将被打包在一起。
5、如果设置了边距,排列链的时候将从总的分配的空间中扣除。

图十、链的样式

3、辅助工具

1、Optimizer优化器
在1.1中,我们公开了约束优化器。可以通过将layout-optimizationlevel添加到constraintlayout元素来决定应用哪些优化。
none:无优化
standard:默认。仅优化直接约束和屏障约束
direct:优化直接约束
barrier:优化屏障约束
chain:优化链约束(实验)
dimensions:优化尺寸测量,减少匹配约束元素的测量数
2、Barrier

图十一

假设有3个控件ABC,C在AB的右边,但是AB的宽是不固定的,这个时候C无论约束在A的右边或者B的右边都不对。当出现这种情况可以用Barrier来解决。Barrier可以在多个控件的一侧建立一个屏障,如下所示:

图十二

这个时候C只要约束在Barrier的右边就可以了,代码如下:

    <TextView
        android:id="@+id/TextView1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content" />

    <TextView
        android:id="@+id/TextView2"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        app:layout_constraintTop_toBottomOf="@+id/TextView1" />

    <android.support.constraint.Barrier
        android:id="@+id/barrier"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        app:barrierDirection="right"
        app:constraint_referenced_ids="TextView1,TextView2" />

    <TextView
        android:id="@+id/TextView3"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        app:layout_constraintLeft_toRightOf="@+id/barrier" />

app:barrierDirection为屏障所在的位置,可设置的值有:bottom、end、left、right、start、top
app:constraint_referenced_ids为屏障引用的控件,可设置多个(用“,”隔开)
3、Group
Group可以把多个控件归为一组,方便隐藏或显示一组控件,举个例子:

<TextView
        android:id="@+id/TextView1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content" />

    <TextView
        android:id="@+id/TextView2"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        app:layout_constraintLeft_toRightOf="@+id/TextView1" />

    <TextView
        android:id="@+id/TextView3"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        app:layout_constraintLeft_toRightOf="@id/TextView2" />

    <android.support.constraint.Group
        android:id="@+id/group"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:visibility="invisible"
        app:constraint_referenced_ids="TextView1,TextView3" />

用Group把TextView1和TextView3归为一组,再设置这组控件的可见性
4、Placeholder
Placeholder指的是占位符。在Placeholder中可使用setContent()设置另一个控件的id,使这个控件移动到占位符的位置。举个例子:

<android.support.constraint.Placeholder
        android:id="@+id/placeholder"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        app:content="@+id/textview"
        app:layout_constraintLeft_toLeftOf="parent"
        app:layout_constraintTop_toTopOf="parent" />

    <TextView
        android:id="@+id/textview"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:background="#cccccc"
        android:padding="16dp"
        android:text="TextView"
        android:textColor="#000000"
        app:layout_constraintRight_toRightOf="parent"
        app:layout_constraintTop_toTopOf="parent" />

新建一个Placeholder约束在屏幕的左上角,新建一个TextView约束在屏幕的右上角,在Placeholder中设置 app:content="@+id/textview",这时TextView会跑到屏幕的左上角。效果如下:
5、Guideline
Guildline像辅助线一样,在预览的时候帮助你完成布局(不会显示在界面上)。
Guildline的主要属性:
android:orientation 垂直vertical,水平horizontal
layout_constraintGuide_begin 开始位置
layout_constraintGuide_end 结束位置
layout_constraintGuide_percent 距离顶部的百分比(orientation = horizontal时则为距离左边)
举个例子:

   <android.support.constraint.Guideline
        android:id="@+id/guideline1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:orientation="horizontal"
        app:layout_constraintGuide_begin="50dp" />

    <android.support.constraint.Guideline
        android:id="@+id/guideline2"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:orientation="vertical"
        app:layout_constraintGuide_percent="0.5" />

guideline1为水平辅助线,开始位置是距离顶部50dp,guideline2位垂直辅助线,开始位置为屏幕宽的0.5(中点位置),效果如下:


图十三

4、总结

本篇文章主要介绍了ConstraintLayout的用法,刚开始可能会感觉到用的各种不习惯,用着用着就习惯了。ConstraintLayout的初衷是为了实现托拉拽完成布局,但本人还是比较喜欢直接用代码写,所以才专门研究一下,纯粹个人喜好问题,至于拖拽大家自己研究了。

相关文章

网友评论

      本文标题:ConstraintLayout从入门到精通

      本文链接:https://www.haomeiwen.com/subject/rhprtctx.html