使用 ConstraintLayout 构建响应式 UI 作为 Android Jetpack 的一部分。
ConstraintLayout
允许你使用扁平的视图层次结构创建大型、复杂的布局——无需嵌套视图组。它类似于 RelativeLayout
,因为所有视图都是根据兄弟视图和父布局之间的关系进行布局的,但它比 RelativeLayout
更灵活,并且更容易与 Android Studio 的布局编辑器一起使用。
ConstraintLayout
的所有功能都可直接从布局编辑器的可视化工具中获得,因为布局 API 和布局编辑器是为彼此专门构建的。你可以完全通过拖动来构建你的 ConstraintLayout
布局,而无需编辑 XML。
此页面演示如何在 Android Studio 3.0 或更高版本中使用 ConstraintLayout
构建布局。有关布局编辑器的更多信息,请参阅 使用布局编辑器构建 UI。
要查看你可以使用 ConstraintLayout
创建的各种布局,请参阅 GitHub 上的 Constraint 布局示例项目。
约束概述
要在 ConstraintLayout
中定义视图的位置,你需要为该视图添加至少一个水平约束和一个垂直约束。每个约束代表与另一个视图、父布局或不可见的参考线的连接或对齐。每个约束定义视图沿垂直或水平轴的位置。每个视图至少必须在一个轴上有一个约束,但通常需要更多约束。
当你将视图拖放到布局编辑器中时,即使它没有约束,它也会停留在你放置的位置。这仅仅是为了使编辑更容易。如果视图在设备上运行布局时没有约束,则它会在位置 [0,0](左上角)绘制。
在图 1 中,布局在编辑器中看起来不错,但视图 C 没有垂直约束。当此布局在设备上绘制时,视图 C 与视图 A 的左右边缘水平对齐,但它出现在屏幕顶部,因为它没有垂直约束。
虽然缺少约束不会导致编译错误,但布局编辑器会将缺少约束作为工具栏中的错误指示出来。要查看错误和其他警告,请点击显示警告和错误 。为了帮助你避免缺少约束,布局编辑器会使用 自动连接和推断约束 功能自动为你添加约束。
将 ConstraintLayout 添加到你的项目
要在你的项目中使用 ConstraintLayout
,请按如下步骤操作
- 确保你的
settings.gradle
文件中声明了maven.google.com
存储库Groovy
dependencyResolutionManagement { ... repositories { google() } )
Kotlin
dependencyResolutionManagement { ... repositories { google() } }
- 在模块级
build.gradle
文件中添加库作为依赖项,如下例所示。最新版本可能与示例中显示的版本不同。Groovy
dependencies { implementation "androidx.constraintlayout:constraintlayout:2.2.0" // To use constraintlayout in compose implementation "androidx.constraintlayout:constraintlayout-compose:1.1.0" }
Kotlin
dependencies { implementation("androidx.constraintlayout:constraintlayout:2.2.0") // To use constraintlayout in compose implementation("androidx.constraintlayout:constraintlayout-compose:1.1.0") }
- 在工具栏或同步通知中,点击使用 Gradle 文件同步项目。
现在,你已准备好使用 ConstraintLayout
构建你的布局。
转换布局
要将现有布局转换为约束布局,请按照以下步骤操作
- 在 Android Studio 中打开你的布局,然后点击编辑器窗口底部的设计选项卡。
- 在组件树窗口中,右键点击布局,然后点击将 LinearLayout 转换为 ConstraintLayout。
创建新布局
要启动一个新的约束布局文件,请按照以下步骤操作
- 在项目窗口中,点击模块文件夹,然后选择文件 > 新建 > XML > 布局 XML。
- 输入布局文件的名称,然后为根标签输入“androidx.constraintlayout.widget.ConstraintLayout”。
- 点击完成。
添加或移除约束
要添加约束,请执行以下操作
将视图从调色板窗口拖到编辑器中。
当你在
ConstraintLayout
中添加视图时,它会显示在一个边界框中,每个角上都有正方形调整大小的句柄,每侧都有圆形的约束句柄。- 点击视图以选择它。
- 执行以下操作之一
- 点击约束句柄并将其拖动到可用的锚点。这个点可以是另一个视图的边缘、布局的边缘或参考线。请注意,当你拖动约束句柄时,布局编辑器会显示潜在的连接锚点和蓝色叠加层。
点击属性窗口布局部分中的创建连接 按钮之一,如图 4 所示。
创建约束后,编辑器会为其提供 默认边距 以分离这两个视图。
创建约束时,请记住以下规则
- 每个视图必须至少有两个约束:一个水平约束和一个垂直约束。
- 你只能在共享同一平面的约束句柄和锚点之间创建约束。视图的垂直平面(左侧和右侧)只能约束到另一个垂直平面,基线只能约束到其他基线。
- 每个约束句柄只能用于一个约束,但你可以从不同的视图创建多个约束到同一个锚点。
你可以通过执行以下任一操作来删除约束
- 点击约束以选择它,然后点击 Delete。
Control-点击(在 macOS 上为 Command-点击)约束锚点。约束会变为红色,表示你可以点击将其删除,如图 5 所示。
在属性窗口的布局部分,点击约束锚点,如图 6 所示。
如果你在视图上添加相反的约束,约束线会像弹簧一样卷曲,以指示相反的力,如视频 2 所示。当视图大小设置为“固定”或“包裹内容”时,效果最为明显,在这种情况下,视图位于约束之间居中。如果你希望视图拉伸其大小以满足约束,请将大小 切换 为“匹配约束”。如果你希望保留当前大小,但移动视图使其不居中,请 调整约束偏差。
你可以使用约束来实现不同类型的布局行为,如下节所述。
父级位置
将视图的一侧约束到布局的相应边缘。
在图 7 中,视图的左侧连接到父布局的左边缘。你可以使用边距定义与边缘的距离。
顺序位置
定义两个视图的出现顺序,可以垂直或水平排列。
在图 8 中,B 始终约束在 A 的右侧,C 约束在 A 的下方。但是,这些约束并不意味着对齐,因此 B 仍然可以上下移动。
对齐
将一个视图的边缘与另一个视图的相同边缘对齐。
在图 9 中,B 的左侧与 A 的左侧对齐。如果要对齐视图中心,请对两侧都创建约束。
您可以通过从约束内向拖动视图来偏移对齐。例如,图 10 显示 B 具有 24dp 的偏移对齐。偏移由受约束视图的边距定义。
您也可以选择所有要对齐的视图,然后单击工具栏中的对齐 以选择对齐类型。
基线对齐
将一个视图的文本基线与另一个视图的文本基线对齐。
在图 11 中,B 的第一行与 A 中的文本对齐。
要创建基线约束,请右键单击要约束的文本视图,然后单击显示基线。然后单击文本基线并将线条拖动到另一个基线。
约束到参考线
您可以添加垂直或水平参考线,让您约束视图,并且对应用程序用户不可见。您可以根据 dp 单位或相对于布局边缘的百分比来定位布局中的参考线。
要创建参考线,请单击工具栏中的参考线 ,然后单击添加垂直参考线或添加水平参考线。
拖动虚线重新定位它,并单击参考线边缘的圆圈以切换测量模式。
约束到屏障
与参考线类似,屏障是一条不可见的线,您可以将视图约束到它,但屏障不定义它自己的位置。相反,屏障位置根据包含在其内的视图的位置而移动。当您想要将视图约束到一组视图而不是一个特定视图时,这很有用。
例如,在图 13 中,视图 C 约束到屏障的右侧。屏障设置为视图 A 和视图 B 的“末端”(或在从左到右的布局中为右侧)。屏障根据视图 A 或视图 B 的右侧哪个最靠右而移动。
要创建屏障,请按照以下步骤操作
- 单击工具栏中的参考线 ,然后单击添加垂直屏障或添加水平屏障。
- 在组件树窗口中,选择要放在屏障内的视图并将它们拖到屏障组件中。
- 从组件树中选择屏障,打开属性 窗口,然后设置barrierDirection。
现在您可以从另一个视图创建到屏障的约束。
您还可以将位于屏障内的视图约束到屏障。这样,即使您不知道哪个视图最长或最高,您也可以将屏障中的所有视图彼此对齐。
您还可以将参考线包含在屏障内,以确保屏障的“最小”位置。
调整约束偏差
当您向视图的两侧添加约束时,并且同一维度的视图大小为“固定”或“包裹内容”时,视图默认情况下会在两个约束之间以 50% 的偏差居中。您可以通过拖动属性窗口中的偏差滑块或拖动视图来调整偏差,如视频 3 所示。
如果您希望视图拉伸其大小以满足约束,请切换大小到“匹配约束”。
调整视图大小
您可以使用角手柄调整视图大小,但这会硬编码大小——视图不会根据不同的内容或屏幕大小调整大小。要选择不同的尺寸模式,请单击一个视图并打开编辑器右侧的属性 窗口。
在属性窗口的顶部附近是视图检查器,其中包含几个布局属性的控件,如图 14 所示。这仅适用于约束布局中的视图。
您可以通过单击图 14 中用标注 3 指示的符号来更改计算高度和宽度的方式。这些符号表示尺寸模式如下。单击该符号可在这些设置之间切换
- 固定:在下面的文本框中或通过在编辑器中调整视图大小来指定特定尺寸。
- 包裹内容:视图仅扩展到足以容纳其内容的程度。
- layout_constrainedWidth
-
匹配约束:在考虑视图边距后,视图尽可能扩展以满足每一侧的约束。但是,您可以使用以下属性和值修改该行为。只有在您将视图宽度设置为“匹配约束”时,这些属性才会生效
- layout_constraintWidth_min
这采用
dp
尺寸作为视图的最小宽度。 - layout_constraintWidth_max
这采用
dp
尺寸作为视图的最大宽度。
但是,如果给定尺寸只有一个约束,则视图会扩展以适应其内容。在高度或宽度上使用此模式还可以让您设置尺寸比例。
- layout_constraintWidth_min
将其设置为true
以使水平尺寸更改以遵守约束。默认情况下,设置为WRAP_CONTENT
的小部件不受约束限制。
设置尺寸比例
如果至少一个视图尺寸设置为“匹配约束”(0dp
),您可以将视图尺寸设置为比例,例如 16:9。要启用比例,请单击切换纵横比约束(图 14 中的标注 1)并在出现的输入中输入宽度:高度比例。
如果宽度和高度都设置为“匹配约束”,您可以单击切换纵横比约束来选择哪个尺寸基于另一个尺寸的比例。视图检查器通过用实线连接相应的边缘来指示哪个尺寸设置为比例。
例如,如果将两侧都设置为“匹配约束”,请单击切换纵横比约束两次以将宽度设置为高度的比例。整个大小由视图的高度决定,这可以用任何方式定义,如图 15 所示。
调整视图边距
要使视图均匀分布,请单击工具栏中的边距 以选择添加到布局的每个视图的默认边距。您对默认边距所做的任何更改仅适用于您从那时起添加的视图。
您可以通过单击代表每个约束的行的数字来控制属性窗口中每个视图的边距。在图 14 中,标注 4 显示底部边距设置为 16dp。
工具提供的全部边距都是 8dp 的倍数,这有助于您的视图符合 Material Design 的 8dp 方形网格建议。
使用链控制线性组
链是一组通过双向位置约束相互链接的视图。链中的视图可以垂直或水平分布。
链可以用以下方式之一进行样式设置
- 分散:在考虑边距后,视图均匀分布。这是默认设置。
- 内部分散:第一个和最后一个视图固定在链两端的约束上,其余视图均匀分布。
- 加权:当链设置为分散或内部分散时,您可以通过将一个或多个视图设置为“匹配约束”(
0dp
)来填充剩余空间。默认情况下,空间在每个设置为“匹配约束”的视图之间均匀分布,但您可以使用layout_constraintHorizontal_weight
和layout_constraintVertical_weight
属性为每个视图分配重要性权重。这与线性布局中的layout_weight
的工作方式相同:权重值最高的视图获得最多的空间,具有相同权重的视图获得相同数量的空间。 - 打包:在考虑边距后,视图打包在一起。您可以通过更改链的“头部”视图偏差来调整整个链的偏差——左或右,或上或下。
链的“头部”视图——水平链中最左侧的视图(在从左到右的布局中)和垂直链中最顶部的视图——在 XML 中定义链的样式。但是,您可以通过选择链中的任何视图并点击视图下方显示的链按钮在分散、内部分散和紧凑之间切换。
如视频 4 所示,创建链的方法如下:
- 选择要包含在链中的所有视图。
- 右键单击其中一个视图。
- 选择链。
- 选择水平居中或垂直居中。
使用链时,请考虑以下几点:
- 一个视图可以同时属于水平链和垂直链的一部分,因此您可以构建灵活的网格布局。
- 只有当链的每一端都约束到同一轴上的另一个对象时,链才能正常工作,如图 14 所示。
- 尽管链的方向是垂直或水平的,但使用链并不会沿该方向对齐视图。为了达到链中每个视图的正确位置,请包含其他约束,例如对齐约束。
自动创建约束
无需在将视图放置到布局中时为每个视图添加约束,您可以将每个视图移动到布局编辑器中所需的位置,然后单击推断约束来自动创建约束。
推断约束会扫描布局以确定所有视图最有效的约束集。它会将视图约束到其当前位置,同时提供灵活性。您可能需要进行调整,以使布局能够根据您的意图响应不同的屏幕尺寸和方向。
自动连接到父级是一项您可以启用的单独功能。启用此功能后,当您向父级添加子视图时,此功能会在您将视图添加到布局时自动为每个视图创建两个或多个约束——但只有在将视图约束到父布局合适的情况下才会这样做。自动连接不会创建到布局中其他视图的约束。
自动连接默认情况下是禁用的。点击布局编辑器工具栏中的启用自动连接到父级来启用它。
关键帧动画
在ConstraintLayout
中,您可以使用ConstraintSet
和TransitionManager
来动画化元素大小和位置的更改。
ConstraintSet
是一个轻量级对象,它表示ConstraintLayout
内所有子元素的约束、边距和填充。当您将ConstraintSet
应用于显示的ConstraintLayout
时,布局会更新其所有子元素的约束。
要使用ConstraintSet
构建动画,请指定两个用作动画起始和结束关键帧的布局文件。然后,您可以从第二个关键帧文件加载ConstraintSet
,并将其应用于显示的ConstraintLayout
。
以下代码示例显示了如何动画化将单个按钮移动到屏幕底部的过程。
// MainActivity.kt
fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.keyframe_one)
constraintLayout = findViewById(R.id.constraint_layout) // member variable
}
fun animateToKeyframeTwo() {
val constraintSet = ConstraintSet()
constraintSet.load(this, R.layout.keyframe_two)
TransitionManager.beginDelayedTransition()
constraintSet.applyTo(constraintLayout)
}
// layout/keyframe1.xml // Keyframe 1 contains the starting position for all elements in the animation // as well as final colors and text sizes. <?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" android:layout_width="match_parent" android:layout_height="match_parent"> <Button android:id="@+id/button2" android:layout_width="0dp" android:layout_height="wrap_content" android:text="Button" app:layout_constraintEnd_toEndOf="parent" app:layout_constraintStart_toStartOf="parent" app:layout_constraintTop_toTopOf="parent" /> </androidx.constraintlayout.widget.ConstraintLayout>
// layout/keyframe2.xml // Keyframe 2 contains another ConstraintLayout with the final positions. <?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" android:layout_width="match_parent" android:layout_height="match_parent"> <Button android:id="@+id/button2" android:layout_width="0dp" android:layout_height="wrap_content" android:text="Button" app:layout_constraintEnd_toEndOf="parent" app:layout_constraintStart_toStartOf="parent" app:layout_constraintBottom_toBottomOf="parent" /> </androidx.constraintlayout.widget.ConstraintLayout>
其他资源
ConstraintLayout
在Sunflower演示应用程序中使用。