Android开发权威指南(第二版)
上QQ阅读APP看书,第一时间看更新

13.3 按钮(Button)和复选框(Checkbox)控件

本节将介绍Android SDK中的按钮和复选框控件。按钮可分为多种,例如,普通按钮(Button)、图像按钮(ImageButton)、选项按钮(RadioButton)等。除此之外,复选框(CheckBox)也是Android SDK中非常重要的控件,通常用于多选的应用中。

13.3.1 Button(普通按钮控件)

源代码目录:src/ch13/Button

Button控件在前面的章节已经多次使用到了。Button控件的基本使用方法与TextView、EditText并无太大的差异,例如,下面的代码在布局文件中定义了一个按钮。

<Button android:id="@+id/button1" android:layout_width="wrap_content"

    android:layout_height="wrap_content" android:text="我的按钮1" />

最常用的按钮事件是单击事件,可以通过Button.setOnClickListener方法设置处理单击事件的监听器,也可以直接通过<Button>标签的android:onClick属性指定单击事件方法名。如果当前的类实现了android.view.View.OnClickListener接口,可以直接将this传入setOnClickListener方法,代码如下:

Button button1 = (Button) findViewById(R.id.button1);

button1.setOnClickListener(this);

在本节的例子中包含两个按钮,并在单击事件中通过value变量控制按钮放大或缩小(value=1为放大,value=1为缩小),代码如下:

源代码文件:src/ch13/Button/src/mobile/android/button/Main.java

// “我的按钮1”按钮的单击事件方法

public void onClick(View view)

{

  Button button = (Button) view;

  // 如果按钮宽度等于屏幕宽度,按钮开始缩小

  if (value == 1

      && button.getWidth() == getWindowManager().getDefaultDisplay().getWidth())

    value = -1;

  // 如果按钮宽度小于100,按钮开始放大

  else if (value == -1 && button.getWidth() < 100)

    value = 1;

  // 以按钮宽度和高度的10%放大或缩小按钮

  button.setWidth(button.getWidth() + (int) (button.getWidth() * 0.1)* value);

  button.setHeight(button.getHeight() + (int) (button.getHeight() * 0.1)* value);

}

运行上面的代码后,将显示两个按钮,单击任何一个按钮后,该按钮都会放大,当按钮宽度等于屏幕宽度时,再次单击按钮时,按钮开始缩小。

为了使按钮更绚丽,还可以为按钮加上背景图。通过设置透明背景图,可以做出任何形状的按钮,这种按钮也可以称为“异形按钮”。异形按钮需要处理3个事件 ,这3个事件及各自的处理逻辑如下。

触摸事件(onTouch):当触摸按钮时,应该显示按钮被触摸后的状态。

焦点变化事件(onFocusChange):当焦点从一个按钮切换到另一个按钮时,应该显示按钮被触摸时的状态(image2.png),并且将上一个焦点按钮设为未被触摸时的状态。

键盘事件(onKey):当某一个按钮获得焦点后,按下手机或模拟器上的“确认”按钮后,当前按钮应该被设置成被按下的状态,也就是这个按钮的第3张图,因此,需要为每一个按钮准备3张图,分别是正常状态、触摸状态、按钮被按下的状态。

现在将本例的第2个按钮改成异形按钮,所以需要准备按钮的 3 个状态的png图(button1.png、button2.png、button3.png)。

先来编写触摸事件(onTouch)中的代码。

源代码文件:src/ch13/Button/src/mobile/android/button/Main.java

@Override

public boolean onTouch(View view, MotionEvent event)

{

  if (event.getAction() == MotionEvent.ACTION_UP)

  {

    // 当手指或鼠标抬起时恢复按钮的默认状态

    view.setBackgroundResource(R.drawable.button1);

  }

  else if (event.getAction() == MotionEvent.ACTION_DOWN)

  {

    // 当手指或鼠标按下时将按钮置为按下状态

    view.setBackgroundResource(R.drawable.button2);

  }

  return false;

}

在上面的代码中根据getAction方法获得了触摸的状态,并使用setBackgroundResource方法为按钮设置了不同的背景图像。

当异形按钮处于焦点状态时,需要将异形按钮的背景图设为被触摸状态,代码如下:

源代码文件:src/ch13/Button/src/mobile/android/button/Main.java

// 当控件焦点变化时调用该事件方法,当控件获得焦点时,hasFocus参数值为true,否则为false

@Override

public void onFocusChange(View view, boolean hasFocus)

{

  if (hasFocus)

  {

    // 设置异形按钮获得焦点时的背景图

    imageButton.setBackgroundResource(R.drawable.button2);

  }

  else

  {

    // 设置异形按钮失去焦点时的背景图

    imageButton.setBackgroundResource(R.drawable.button1);

  }

}

当按手机或模拟器上的“确认”键时,需要将当前获得焦点的异形按钮设为按键被按下的状态,当松开“确认”键时,又恢复到获得焦点的状态,这个功能需要在处理键盘事件的onKey中完成,代码如下:

源代码文件:src/ch13/Button/src/mobile/android/button/Main.java

@Override

public boolean onKey(View view, int keyCode, KeyEvent event)

{

  if (KeyEvent.ACTION_DOWN == event.getAction())

  {

    // 设置异形按钮被按下时的背景图

    view.setBackgroundResource(R.drawable.button3);

  }

  else if (KeyEvent.ACTION_UP == event.getAction())

  {

    // 设置异形按钮被抬起时的背景图(焦点状态的背景图)

    view.setBackgroundResource(R.drawable.button2);

  }

  return false;

}

本例中两个按钮在默认情况下的显示效果如图13-12所示。当不断按屏幕上的两个按钮时,两个按钮会不断放大,当遇到屏幕边界时,会不断缩小,如图13-13所示。

 

▲图13-12 默认状态的两个按钮

 

▲图13-13 单击放大的两个按钮

13.3.2 图文混排的按钮

源代码目录:src/ch13/ImageTextButton

上一节介绍的只是在按钮上显示文字或图像,在本节将介绍如下两种同时在按钮上显示文字和图像的方法。

使用<Button>标签的android:drawableXxx属性,其中Xxx表示Top、Bottom、Left、Right。这4个属性的值都是资源类型,需要指定图像资源的ID,分别表示在按钮上、下、左、右插入一个图像,同时还可配合android:drawablePadding属性来设置图像到文字的距离。

Button和EditText一样,也是TextView的子类,因此,也可以采用与TextView、EditText同样的方法来实现图文混排。

首先在布局文件中放5个按钮,其中前4个按钮分别使用android:drawableXxx属性在文字上、下、左、右插入一个图像,第5个按钮通过ImageSpan和SpannableString在文字两侧分别插入一个图像。

布局文件的代码如下:

源代码文件:src/ch13/ImageTextButton/res/layout/main.xml

<?xml version="1.0" encoding="utf-8"?>

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"

  android:orientation="vertical" android:layout_width="fill_parent"

  android:layout_height="fill_parent">

  <LinearLayout android:orientation="horizontal"

    android:layout_width="fill_parent" android:layout_height="120dp">

    <Button android:layout_width="wrap_content"

      android:layout_height="wrap_content" android:drawableTop="@drawable/star"

      android:text="按钮1" />

    <Button android:layout_width="wrap_content"

      android:layout_height="wrap_content" android:drawableBottom="@drawable/star"

      android:text="按钮2" android:drawablePadding="30dp" />

    <Button android:layout_width="wrap_content"

      android:layout_height="wrap_content" android:drawableLeft="@drawable/star"

      android:text="按钮3" />

    <Button android:layout_width="wrap_content"

      android:layout_height="wrap_content" android:drawableRight="@drawable/star"

      android:text="按钮4" android:drawablePadding="20dp" />

  </LinearLayout>

  <Button android:id="@+id/button" android:layout_width="200dp"

    android:layout_height="200dp" android:layout_marginTop="10dp"  />

</LinearLayout>

下面的代码向第5个按钮上的文字两侧分别插入了两个图像。

源代码文件:src/ch13/ImageTextButton/src/mobile/android/image/text/button/Main.java

public class Main extends Activity

{  

  @Override

  public void onCreate(Bundle savedInstanceState)

  {

    super.onCreate(savedInstanceState);

    setContentView(R.layout.main);

    Button button = (Button) findViewById(R.id.button);

    // 处理文字左侧的图像

    SpannableString spannableStringLeft = new SpannableString("left");

    Bitmap bitmapLeft = BitmapFactory.decodeResource(getResources(),

        R.drawable.image_left);

    ImageSpan imageSpanLeft = new ImageSpan(bitmapLeft,

                    DynamicDrawableSpan.ALIGN_BOTTOM);

    // 将left替换成ImageSpan对象

    spannableStringLeft.setSpan(imageSpanLeft, 0, 4,

        Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);

    // 处理文字右侧的图像

    SpannableString spannableStringRight = new SpannableString("right");

    Bitmap bitmapRight = BitmapFactory.decodeResource(getResources(),

        R.drawable.image_right);

    ImageSpan imageSpanRight = new ImageSpan(bitmapRight);

    // 将right替换成ImageSpan对象

    spannableStringRight.setSpan(imageSpanRight, 0, 5,

        Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);

    // 插入文字左侧的图像

    button.append(spannableStringLeft);

    // 插入文字

    button.append("我的按钮");

    // 插入文字右侧的图像

    button.append(spannableStringRight);    

  }

}

运行本例后,在窗口上会显示如图13-14所示的5个按钮。

13.3.3 ImageButton(图像按钮控件)

源代码目录:src/ch13/ImageButton

ImageButton控件可以作为图像按钮来使用。例如,下面的代码配置了两个带背景的图像按钮。

源代码文件:src/ch13/ImageButton/res/layout/main.xml

<ImageButton android:layout_width="wrap_content"

  android:layout_height="wrap_content" android:src="@drawable/button1_1" />

<ImageButton android:layout_width="wrap_content"

  android:layout_height="wrap_content" android:src="@drawable/button2_1" />

如果想在代码中修改ImageButton的图像,可以使用ImageButton.setImageResource或其他类似的方法。本例的显示效果如图13-15所示。

 

▲图13-14 图文混排的按钮

 

▲图13-15 图像按钮

注意

ImageButton并不是TextView的子类,而是ImageView的子类,因此并没有android:text属性。如果要在ImageButton上输出文字,可以自定义一个控件,并在onDraw方法中将文字画在ImageButton上。

13.3.4 RadioButton(选项按钮控件)

源代码目录:src/ch13/RadioButton

选项按钮可用于多选一的应用中。如果想在选中某一个选项按钮后,其他的选项按钮都被设为未选中状态,需要将<RadioButton>标签放在<RadioGroup>标签中。由于RadioButton是Button的子类(实际上,RadioButton是ComponentButton的直接子类,而ComponentButton又是Button的直接子类),因此,在<RadioButton>标签中同样可以使用android:drawableXxx及android:drawablePadding属性。例如,下面的代码在屏幕上放置了3个选项按钮,其中第3个选项按钮周围显示了4个图像。

源代码文件:src/ch13/RadioButton/res/layout/main.xml

<RadioGroup android:layout_width="wrap_content" android:layout_height="wrap_content">

  <RadioButton android:layout_width="wrap_content"

    android:layout_height="wrap_content" android:text="选项1" />

  <RadioButton android:layout_width="wrap_content"

    android:layout_height="wrap_content" android:text="选项2" />

  <RadioButton android:layout_width="wrap_content"

    android:layout_height="wrap_content" android:text="选项3"

    android:drawableLeft="@drawable/star" android:drawableTop="@drawable/circle"

    android:drawableRight="@drawable/star"

    android:drawableBottom="@drawable/circle" android:drawablePadding="20dp" />

</RadioGroup>

本例的显示效果如图13-16所示。

 

▲图13-16 选项按钮

13.3.5 ToggleButton(开关状态按钮控件)

源代码目录:src/ch13/ToggleButton

ToggleButton控件与Button控件的功能基本相同,但ToggleButton控件还提供了可以表示“开/关”状态的功能,这种功能非常类似于复选框(CheckBox)。ToggleButton控件通过在按钮文字的下方显示一个绿色的指示条来表示“开/关”状态。至于绿色的指示条是表示“开”还是“关”,完全由开发人员自己决定。当指示条在绿色状态时,再次单击按钮,指示条就会变成白色。ToggleButton控件的基本使用方法与Button控件相同,代码如下:

<ToggleButton android:layout_width="wrap_content" android:layout_height="wrap_content" />

虽然ToggleButton是Button的子类,但android:text属性并不起作用。在默认情况下,根据ToggleButton控件的不同状态,会在按钮上显示“关闭”或“开启”。如果要更改默认的按钮文本,可以使用android:textOff和android:textOn属性,代码如下:

源代码文件:src/ch13/ToggleButton/res/layout/main.xml

<ToggleButton android:id="@+id/toggleButton" android:layout_width="wrap_content"

    android:layout_height="wrap_content" android:layout_marginLeft="30dp"

    android:textOff="打开电灯" android:textOn="关闭电灯" />

默认情况下,按钮上的指示条是白色的,如果要在XML布局文件中修改默认状态,可以使用android:checked属性,在代码中使用ToggleButton.setChecked方法。当checked属性值或setChecked方法的参数值为true时,指示条显示为绿色。

本例的显示效果如图13-17所示。

 

▲图13-17 ToggleButton 控件

13.3.6 CheckBox(复选框控件)

源代码目录:src/ch13/DynamicCheckbox

复选框通常用于多选的应用。基本的使用方法如下:

<CheckBox android:id="@+id/checkbox" android:layout_width="fill_parent"

    android:layout_height="wrap_content" />

CheckBox默认情况下是未选中状态。如果想修改这个默认值,可以将<CheckBox>标签的android:checked属性值设为true,或使用CheckBox.setChecked方法设置CheckBox的状态。在代码中可以使用CheckBox.isChecked方法判断CheckBox是否被选中。如果isChecked方法返回true,则表示CheckBox处于选中状态。

本节给出一个动态创建CheckBox控件的例子,也许很多读者会首先想到在代码中创建CheckBox对象。这样做虽然从技术上说没有任何问题,也属于比较常用的动态创建控件的方法,但在实际应用中,往往会为CheckBox设置很多属性,如果在代码中直接创建CheckBox对象,就需要手工设置CheckBox对象的属性,这样做是很麻烦的。

如果单纯考虑设置控件的属性,布局文件无疑是最简单的方法。我们自然会想到是否可以在布局文件中先配置一个或若干CheckBox,然后以这些配置为模板来动态创建CheckBox对象呢?如果能想到这些,将会大大减少动态创建CheckBox对象的代码。

由于同一个控件不能拥有两个及以上的父控件,因此,不能直接在窗口中使用findViewById方法来获得CheckBox对象。如何解决这个问题呢?

实际上,XML布局文件的根节点不仅可以是像<LinearLayout>、<RelativeLayout>一样的ViewGroup(容器控件),也可以直接使用View,例如,<CheckBox>标签。因此,我们先来编写一个checkbox.xml文件设置复选框。

源代码文件:src/ch13/DynamicCheckbox/res/layout/checkbox.xml

<?xml version="1.0" encoding="utf-8"?>

<CheckBox xmlns:android="http://schemas.android.com/apk/res/android"

  android:id="@+id/checkbox" android:layout_width="fill_parent"

  android:layout_height="wrap_content" />

为了单击“确认”按钮后显示被选中的复选框的文本,需要在复选框下方显示一个“确认”按钮。因此,在本例中还需要一个main.xml布局文件,用于定义“确认”按钮。

源代码文件:src/ch13/DynamicCheckbox/res/layout/main.xml

<?xml version="1.0" encoding="utf-8"?>

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"

  android:orientation="vertical" android:layout_width="fill_parent"

  android:layout_height="fill_parent">

  <Button android:id="@+id/button" android:layout_width="wrap_content"

    android:layout_height="wrap_content" android:text="确定" />

</LinearLayout>

动态创建CheckBox对象的方法是定义一个String类型的数组,数组元素是CheckBox的文本,然后根据String数组的元素个数来动态创建CheckBox对象。动态创建CheckBox对象的步骤如下。

第1步:使用getLayoutInflater().inflate方法装载main.xml布局文件,并返回一个LinearLayout对象。

第2步:使用getLayoutInflater().inflate方法装载checkbox.xml布局文件,并返回一个CheckBox对象。

第3步:根据String数组中的值设置CheckBox的文本。

第4步:调用linearLayout.addView方法将CheckBox对象添加到LinearLayout对象中。

第5步:根据String数组的元素重复执行第2步、第3步和第4步,直到处理完String数组中的最后一个元素为止。

动态创建CheckBox对象是在onCreate方法中完成的,代码如下:

源代码文件:src/ch13/DynamicCheckbox/src/mobile/android/dynamic/checkbox/Main.java

public void onCreate(Bundle savedInstanceState)

{

  String[] checkboxText = new String[]

  { "是学生吗?", "是否喜欢Android?", "买车了吗?", "打算出国吗?" };

  super.onCreate(savedInstanceState);

  LinearLayout linearLayout = (LinearLayout) getLayoutInflater().inflate(

      R.layout.main, null);

  for (int i = 0; i < checkboxText.length; i++)

  {

    // 根据checkbox.xml文件创建CheckBox对象

    CheckBox checkbox = (CheckBox) getLayoutInflater().inflate(R.layout.checkbox, null);

    // 将CheckBox对象添加到checkboxs数组中,用于对CheckBox对象的检索

    checkboxs.add(checkbox);

    // 设置CheckBox对象的文本

    checkboxs.get(i).setText(checkboxText[i]);

    // 将CheckBox对象添加到界面的主布局中(main.xml),在按钮前面显示复选框

    linearLayout.addView(checkbox, i);

  }

  setContentView(linearLayout);

  Button button = (Button) findViewById(R.id.button);

  button.setOnClickListener(this);

}

其中checkboxs是在Main类中定义的一个List<CheckBox>类型的变量,用来保存动态创建的CheckBox对象。该变量需要在“确认”按钮的单击事件方法中使用,代码如下:

源代码文件:src/ch13/DynamicCheckbox/src/mobile/android/dynamic/checkbox/Main.java

public void onClick(View view)

{

  String s = "";

  // 扫描所有的CheckBox,以便获得被选中的复选框的文本

  for (CheckBox checkbox : checkboxs)

  {

    if (checkbox.isChecked())

      s += checkbox.getText() + "\n";

  }

  if ("".equals(s))

    s = "您还没选呢!";

  new AlertDialog.Builder(this).setMessage(s).setPositive Button("关闭", null).show();

}

运行本例并选中相应的复选框,然后单击“确认”按钮,显示如图13-18所示的效果。

 

▲图13-18 动态创建复选框