Android在屏幕中控件的组织上,可以将各个视图(控件)组成一个视图组(ViewGroup),视图组是一个包含了其他视图的视图。
▲图书封面
1.视图组(ViewGroup抽象类)
android.view包中ViewGroup类继承了View,因此它本身也具有View的特性。ViewGroup主要的功能在于它可以包含其他控件,作为其他控件的容器。
ViewGroup实现了android.view包中的ViewParent接口,这个类表示一个可以作为其他的View的容器的职责。
ViewGroup也实现了android.view包中的ViewManager接口,因此包含ViewManager中的以下几个方法:
public abstract void addView(View view, ViewGroup.LayoutParams params)
public abstract void removeView(View view)
public abstract void updateViewLayout(View view, ViewGroup.LayoutParams params)
addView()以View作为参数,用于将这个View增加为当前视图组的“孩子”;removeView()用于将一个View从视图组中移除;updateViewLayout()用于更新某个View的布局。
ViewGroup.OnHierarchyChangeListener是一个接口,用于监听ViewGroup中的View的层次变化。这个接口中包含了两个监听方法:
public abstract void onChildViewAdded (View parent, View child)
public abstract void onChildViewRemoved (View parent, View child)
实现一个OnHierarchyChangeListener接口后,这两个方法可以监听包括ViewGroup(包括其继承者)之中的增加和删除孩子的情况。通过ViewGroup的setOnHierarchyChangeListener()方法可以将其设置给一个ViewGroup。
ViewGroup是一个抽象类,其中包含了以下的一个抽象方法:
protected abstract void onLayout (boolean changed, int l, int t, int r, int b)
ViewGroup中的onLayout()方法将在ViewGroup为它的孩子们分配尺寸和位置的时候被调用,在这个类的实现中,需要调用每一个控件的布局方法为其布局。
提示:onLayout()在View中是一个public的方法,在ViewGroup为protected类型,并且为abstract,由于这个方法在ViewGroup中没有实现,因此ViewGroup本身不可以直接使用。
2.Android的屏幕元素体系
Android UI程序的屏幕体系结构的组织遵循以下原则:
一个屏幕可以包含一个视图;
视图组本身也是一个视图;
视图组可以包含若干个视图。
Android视图和视图组的关系如图3-12所示。
图3-12中左右两图表示的一个屏幕中View组织的结构,左图是各个视图组和控件在一个屏幕之中的示意图;右图为视图组的层次结构图(View Hierarchy),这也是Android中一种常用的表示屏幕中布局的方式。
如图3-12所示,外部最大的框表示整个屏幕,其中包含一个视图组ViewGroup0,ViewGroup0包含三个子视图,即View1、ViewGroup1、ViewGroup2。ViewGroup1本身也是视图组,包含了View2和View3;ViewGroup2本身也是视图组,包含了View4、ViewGroup3和ViewGroup4;ViewGroup4本身也是视图组,包含了View5和View6。
▲图3-12 视图和视图组的关系(上:屏幕示意图;下:View的层次结构图)
根据以上的原则,当屏幕需要包含多个视图时,必须组织在一个视图组中。由于视图组本身也是一个视图,因此视图组还可以包含视图组。
一个主要的限制是:在没有视图组的情况下,两个以上的视图(也包括视图组)是不能够并列的。例如,在布局文件中,类似下面的写法是不可以的。
<?xml version="1.0" encoding="utf-8"?>
<Button android:id="@+id/button"/>
<EditText android:id="@+id/edit"/>
3.ViewGroup的继承结构
ViewGroup是所有视图组的基类,它本身是一个不能直接使用的抽象类。在程序中使用的主要是ViewGroup的继承者。
ViewGroup的继承者大部分在android.widget包中,其直接继承者包括:AdapterView、AbsoluteLayout、FrameLayout、LinearLayout、RelativeLayout。这些继承者各自又具有一些继承者。
ViewGroup继承者的体系结构如图3-13所示。
图3-13中所示的ViewGroup继承者都是可以作为容器使用的。这些继承者可以为它们的孩子们提供不同的布局方法,用以确定孩子们相互之间的位置和尺寸关系。
这些继承者又具有一些继承者,有些继承者是不作为容器使用的,仅仅是像一个普通控件一样使用。
▲图3-13 视图组的继承结构
4.布局参数类
在Android中每个控件在布局文件中能使用的XML属性其实有三类:
其自己的XML属性;
其祖先类的XML属性;
其容器的布局参数。
其中,布局参数是包含这个控件的容器(一个ViewGroup的继承者)所提供的参数。在Android中,每一个ViewGroup的继承者都有一个相对应的名称为{XXX}.LayoutParams的静态子类,表示这个ViewGroup的孩子们中可以使用的布局参数。
这些布局参数类的基类为ViewGroup.LayoutParams,这是一个ViewGroup的静态子类,表示ViewGroup的子对象中可以使用参数。
ViewGroup.LayoutParams包含有两个重要的XML属性:android:layout_width和android:layout_height,它们表示布局对象的宽和高。除了使用实际的尺寸数值外,还有如下两个常用的选项。
“match_parent”或者“fill_parent”:表示能匹配父视图的最大尺寸;
“wrap_content”:表示仅包裹孩子的最小尺寸。
ViewGroup.MarginLayoutParams是ViewGroup.LayoutParams的一个继承者,主要用于定义和边缘的空白,它具有android:layout_marginTop,android:layout_marginLeft,android:layout_marginBottom,android:layout_marginRight几个XML属性,表示四个方向的边缘空白。
在ViewGroup的继承者中,都具有一个名称为.LayoutParams的静态子类,这些子类继承关系和ViewGroup子类的继承关系具有相似性。布局参数LayoutParams使用方式如图3-14所示。
▲图3-14 视图组布局参数的结构
从图3-14中可见,LinearLayout的布局参数LinearLayout.LayoutParams可以被这个布局的孩子们使用,RelativeLayout的布局参数RelativeLayout.LayoutParams可以被这个布局的孩子们使用。但是这里“孩子”的含义,只包括这个容器直接的孩子。例如,图3-14中LinearLayout的孩子有一个是RelativeLayout,RelativeLayout的孩子中能使用的参数就不包括线性布局的参数LinearLayout.LayoutParams。
提示:由于LayoutParams也具有继承关系,因此LinearLayout的孩子们除了可以使用LinearLayout.LayoutParams自己的XML属性,还可以使用其祖先类ViewGroup.LayoutParams的XML属性。