~~~~~~ Android把所有能够显示的图形都抽象为Drawable类(可绘制的)。这里的图形不止是图片,还包括色块、画板、背景等。
包含图片在内的图形文件放在res目录的各个drawable目录下,其中drawable目录一般保存描述性的
XML文件,而图片文件一般放在具体分辨率的drawable目录下。例如:
Shape图形又称形状图形,可以用它绘制矩形、圆角矩形、圆形、椭圆等。
形状图形的定义文件放在drawable目录下,以shape为根标签,描述了当前是哪种几何图形,shape标签中有个shape属性,用来指定图形的形状。
shape节点下面有6个子标签:
下面是绘制一个矩形的代码:
<shape xmlns:android="http://schemas.android.com/apk/res/android">
<solid android:color="#ffdd66"/>
<stroke android:width="1dp" android:color="#aaaaaa"/>
<corners android:radius="10dp"/>
shape>
状态图形一般会用到按钮控件的背景,例如一个按钮被按下,就会有凹下去的感觉,这时就饿可以用Drawable的一个子类StateListDrawable(状态列表图形),在xml中规定不同状态时候呈现的图形列表。
下面演示如何制作一个状态列表:
在drawable目录创建一个xml文件,里面的根标签是selector
<selector xmlns:android="http://schemas.android.com/apk/res/android">
<item android:drawable="@drawable/shape_oval_rose" android:state_pressed="true"/>
<item android:drawable="@drawable/shape_rect_gold"/>
selector>
上面演示的是当按钮被按下时显示shape_oval_rose这个图形,默认显示shape_rect_gold图形
状态列表图形不仅用于按钮控件,还可用于其他拥有多种状态的控件,这取决于开发者在XML文件中指定了哪种状态类型。
CompoundButton类是抽象的复合按钮,所以不能直接使用,派生类主要有复选框CheckBox、单选按钮RadioButton以及开关按钮Switch,因为CompoundButton又继承了Button类,所以上面几个按钮也可也使用Button类的属性和方法。继承关系如下:
CompoundButton在XML文件中主要使用下面两个属性。
CompoundButton在Java代码中主要使用下列4种方法。
复选框CheckBox,点击勾选,再次点击取消勾选,复选框对象调用setOnCheckedChangeListener方法设置勾选监听器。
定义一个复选框:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
<CheckBox
android:id="@+id/ck_system"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="这是系统默认的样式"
/>
<CheckBox
android:id="@+id/ck_custom"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:button="@drawable/checkbox_selector"
android:text="这是自己定制的样式"
android:checked="true"
/>
LinearLayout>
java代码:
public class CheckBoxActivity extends AppCompatActivity implements CompoundButton.OnCheckedChangeListener {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_check_box);
CheckBox ck_system = findViewById(R.id.ck_system);
CheckBox ck_custom = findViewById(R.id.ck_custom);
ck_system.setOnCheckedChangeListener(this);
ck_custom.setOnCheckedChangeListener(this);
}
@Override
public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
String desc = String.format("你%s了这个CheckBox",isChecked?"勾选":"取消勾选");
buttonView.setText(desc);
}
}
Switch是开关按钮,Switch控件新添加的XML属性说明如下:
定义一个开关按钮:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
<Switch
android:id="@+id/sw_status"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="Switch开关:"
android:textSize="17sp"
/>
<TextView
android:id="@+id/tv_result"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="10dp"
/>
LinearLayout>
java代码:
public class SwitchDefaultActivity extends AppCompatActivity implements CompoundButton.OnCheckedChangeListener {
private TextView tv_result;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_switch_default);
tv_result = findViewById(R.id.tv_result);
Switch sw_status = findViewById(R.id.sw_status);
sw_status.setOnCheckedChangeListener(this);
}
@Override
public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {
String s = String.format("Switch的状态是%s",isChecked?"开":"关");
tv_result.setText(s);
}
}
所谓单选按钮,指的是在一组按钮中选择其中一项,并且不能多选,这要求有个容器确定这组按钮的范围,这个容器便是单选组RadioGroup。单选组实质上是个布局,同一组RadioButton都要放在同一个RadioGroup节点下。RadioGroup提供了orientation属性指定下级控件的排列方向,该属为horizontal时,单选按钮在水平方向排列;该属性为vertical时,单选按钮在垂直方向排列。RadioGroup下面除了RadioButton,还可以挂载其他子控件(如TextView、ImageView等)。如此看来,单选组相当于特殊的线性布局。
RadioGroup在Java代码中的3个常用方法:
与CheckBox不同的是,RadioButton默认未选中,点击后显示选中,但是再次点击不会取消选中。只有点击同组的其他单选按钮时,原来选中的单选按钮才会取消选中。另需注意,单选按钮的选中事件不是由RadioButton处理,而是由RadioGroup处理。
定义一个单选按钮:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="请选择你的性别"
/>
<RadioGroup
android:id="@+id/rg_gender"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
>
<RadioButton
android:id="@+id/rb_male"
android:layout_width="0dp"
android:layout_weight="1"
android:layout_height="wrap_content"
android:text="男"
/>
<RadioButton
android:id="@+id/rb_female"
android:layout_width="0dp"
android:layout_weight="1"
android:layout_height="wrap_content"
android:text="女"
/>
RadioGroup>
<TextView
android:id="@+id/tv_result"
android:layout_width="match_parent"
android:layout_height="wrap_content"
/>
LinearLayout>
java代码:
public class RadioHorizontalActivity extends AppCompatActivity implements RadioGroup.OnCheckedChangeListener {
private TextView tv_result;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_radio_horizontal);
RadioGroup rg_gender = findViewById(R.id.rg_gender);
tv_result = findViewById(R.id.tv_result);
rg_gender.setOnCheckedChangeListener(this);
}
@Override
public void onCheckedChanged(RadioGroup group, int checkedId) {
switch (checkedId){
case R.id.rb_male:
tv_result.setText("你好帅气的男孩");
break;
case R.id.rb_female:
tv_result.setText("你好漂亮的女孩");
break;
}
}
}
编辑框EditText上高效地输入文本
编辑框EditText用于接收软键盘输入的文字,例如用户名、密码、评价内容等,它由文本视图派生而
来,除了TextView已有的各种属性和方法,EditText还支持下列XML属性。
maxLength:指定文本允许输入的最大长度。
hint:指定提示文本的内容。
textColorHint:指定提示文本的颜色
定义一个编辑框:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical">
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:text="下面是登录信息"
/>
<EditText
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:hint="请输入用户名"
android:maxLength="6"
android:inputType="text"
/>
<EditText
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:hint="请输入密码"
android:inputType="textPassword"
/>
LinearLayout>
当然EditText默认的下划线背景不甚好看,下面将利用状态列表图形将编辑框背景改为更加美观的圆角矩形。
首先编写圆角矩形的形状图形文件,它的XML定义文件示例如下:
<shape xmlns:android="http://schemas.android.com/apk/res/android" >
<solid android:color="#ffffff" />
<stroke
android:width="1dp"
android:color="#aaaaaa" />
<corners android:radius="5dp" />
<padding
android:bottom="2dp"
android:left="2dp"
android:right="2dp"
android:top="2dp" />
shape>
上面定义的是在未输入时显示的样式,点击输入时的样式在shape_edit_focus.xml自行定义,和上面一样的代码,只需要改变一下线条的颜色即可。
接下来接着编写编辑框背景的状态列表图形文件,主要在selector节点下添加两个item,一个是聚焦时的图形,另一个是未聚焦时的图形。
<selector xmlns:android="http://schemas.android.com/apk/res/android">
<item android:state_focused="true"
android:drawable="@drawable/shape_edit_focus"/>
<item android:drawable="@drawable/shape_edit_normal"/>
selector>
EditText将background属性值设为@drawable/editext_selector,其背景由editext_selector.xml所定义的状态列表图形决定。
定义xml布局:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:padding="5dp"
android:orientation="vertical" >
<EditText
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="5dp"
android:inputType="text"
android:hint="我的边框是圆角"
android:background="@drawable/editext_selector"
android:textColor="@color/black"
android:textSize="17sp" />
LinearLayout>
~~~~~~ 焦点变更监听器来自于接口View.OnFocusChangeListener,若想注册该监听器,就要调用编辑框对象的setOnFocusChangeListener方法,即可在光标切换之时(获得光标和失去光标)触发焦点变更事件。下面是给密码框注册焦点变更监听器的代码例子:
布局效果,两个输入框,一个按钮:
java代码:
public class EditFocusActivity extends AppCompatActivity implements View.OnFocusChangeListener{
private EditText et_phone;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_edit_focus);
et_phone = findViewById(R.id.et_phone);
EditText et_password = findViewById(R.id.et_password);
et_password.setOnFocusChangeListener(this);
}
@Override
public void onFocusChange(View v, boolean hasFocus) {
if (hasFocus){
String phone = et_phone.getText().toString();
//手机号不足11位
if (TextUtils.isEmpty(phone) || phone.length() <11){
//手机号码编辑框请求焦点,也就是把光标移回手机号码编辑框
et_phone.requestFocus();
Toast.makeText(this,"请输入11位手机号码",Toast.LENGTH_SHORT).show();
}
}
}
}
当用户输入完密码,登录按钮有时候就被软键盘挡着了,下面演示如何在输入完密码之后自动关闭软键盘。
按下返回键就会关闭软键盘,但这是系统自己关闭的,而非开发者在代码中关闭。因为输入法软键盘由系统服务INPUT_METHOD_SERVICE管理,所以关闭软键盘也要由该服务处理,下面是使用系统服务关闭软键盘的代码例子
定义一个工具类:
public class ViewUtil {
public static void hideOneInputMethod(Activity act, View v){
//从系统服务中获取输入法管理器
InputMethodManager imm = (InputMethodManager) act.getSystemService(Context.INPUT_METHOD_SERVICE);
//关闭屏幕上的输入法软键盘
imm.hideSoftInputFromWindow(v.getWindowToken(),0);
}
}
注意上述代码里面的视图对象v,虽然控件类型为View,但它必须是EditText类型才能正常关闭软键盘。
该功能点要求实时监控当前已输入的文本长度,这个监控操作用到文本监听器接口TextWatcher,该接
口提供了3个监控方法,具体说明如下:
使用addTextChangedListener方法注册文本监听器,代码实现:
public class EditHideActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_edit_hide);
EditText et_phone = findViewById(R.id.et_phone);
EditText et_password = findViewById(R.id.et_password);
//注册监听器
et_phone.addTextChangedListener(new HideTextWatcher(et_phone,11));
et_password.addTextChangedListener(new HideTextWatcher(et_password,6));
}
//定义一个编辑框监听器, 在输入文本达到指定长度时自动隐藏输入法
private class HideTextWatcher implements TextWatcher{
private EditText mView;
private int mMaxLength;
public HideTextWatcher(EditText v,int maxLength){
this.mView = v;
this.mMaxLength = maxLength;
}
@Override
public void beforeTextChanged(CharSequence s, int start, int count, int after) {
}
@Override
public void onTextChanged(CharSequence s, int start, int before, int count) {
}
//在编辑框的输入文本变化后触发
@Override
public void afterTextChanged(Editable s) {
//获得已输入的文本字符串
String str = s.toString();
//输入文本达到11位(如手机号码),或者达到6位(如登录密码)时关闭输入法
if (str.length() == mMaxLength){
//隐藏软键盘
ViewUtil.hideOneInputMethod(EditHideActivity.this,mView);
}
}
}
}
AlertDialog没有公开的构造方法,因此必须借助建造器AlertDialog.Builder才能完成参数设置AlertDialog.Builder的常用方法说明如下。
通过AlertDialog.Builder设置完对话框参数,还需调用建造器的create方法才能生成对话框实例。最后
调用对话框实例的show方法,在页面上弹出提醒对话框。
代码示例:
public class AlertDialogActivity extends AppCompatActivity implements View.OnClickListener{
private TextView tv_result;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_alert_dialog);
findViewById(R.id.btn_alert).setOnClickListener(this);
tv_result = findViewById(R.id.tv_result);
}
@Override
public void onClick(View v) {
//创建提醒对话框的建造器
AlertDialog.Builder builder = new AlertDialog.Builder(this);
//设置对话框标题文本
builder.setTitle("尊敬的用户");
//设置对话框内容文本
builder.setMessage("您真的要卸载我吗");
//设置对话框的肯定按钮文本及其点击监听器
builder.setPositiveButton("残忍卸载", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
tv_result.setText("虽然依依不舍,但也只能离开了");
}
});
//设置对话框的否定按钮文本及其点击监听器
builder.setNegativeButton("我再想想",(dialog,which)->{
tv_result.setText("让我再陪你365天");
});
//根据建造器构建提醒对话框对象
AlertDialog dialog = builder.create();
//显示提醒对话框
dialog.show();
}
}
DatePickerDialog相当于在AlertDialog上装载了DatePicker,编码时只需调用构造方法设置当前的年、
月、日,然后调用show方法即可弹出日期对话框。日期选择事件则由监听器OnDateSetListener负责.
代码示例:
public class DatePickerActivity extends AppCompatActivity implements View.OnClickListener, DatePickerDialog.OnDateSetListener {
private DatePicker dp_date;
private TextView tv_date;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_date_picker);
findViewById(R.id.btn_ok).setOnClickListener(this);
findViewById(R.id.btn_date).setOnClickListener(this);
dp_date = findViewById(R.id.dp_date);
tv_date = findViewById(R.id.tv_date);
}
@Override
public void onClick(View v) {
switch (v.getId()){
case R.id.btn_ok:
String s = String.format("你选择的日期是%s年%s月%s日",dp_date.getYear(),dp_date.getMonth()+1,dp_date.getDayOfMonth());
tv_date.setText(s);
break;
case R.id.btn_date:
DatePickerDialog dialog = new DatePickerDialog(this,this,2020,11,25);
//显示日期对话框
dialog.show();
break;
}
}
@Override
public void onDateSet(DatePicker view, int year, int month, int dayOfMonth) {
String s = String.format("你选择的日期是%s年%s月%s日",year,month+1,dayOfMonth);
tv_date.setText(s);
}
}
TimePickerDialog为时间对话框。
代码示例:
public class TimePickerActivity extends AppCompatActivity implements View.OnClickListener, TimePickerDialog.OnTimeSetListener {
private TextView tv_time;
private TimePicker tp_time;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_time_picker);
tv_time = findViewById(R.id.tv_time);
tp_time = findViewById(R.id.tp_time);
tp_time.setIs24HourView(true);
findViewById(R.id.btn_ok).setOnClickListener(this);
findViewById(R.id.btn_time).setOnClickListener(this);
}
@Override
public void onClick(View v) {
switch (v.getId()){
case R.id.btn_ok:
String s = String.format("你选择的时间是%s时%s分",tp_time.getHour(),tp_time.getMinute());
tv_time.setText(s);
break;
case R.id.btn_time:
//构建一个时间对话框,该对话框已经集成了时间选择器。
TimePickerDialog dialog = new TimePickerDialog(this, android.R.style.Theme_Holo_Light_Dialog,this,8,0,true);
dialog.show();
break;
}
}
@Override
public void onTimeSet(TimePicker view, int hourOfDay, int minute) {
String s = String.format("你选择的时间是%s时%s分",hourOfDay,minute);
tv_time.setText(s);
}
}
TimePickerDialog dialog = new TimePickerDialog(this, android.R.style.Theme_Holo_Light_Dialog,this,8,0,true);
最后一个参数的true代码采用24小时制,false为12小时制。