Android自定义控件之日期选择控件使用详解
作者:MarcoReus 时间:2021-07-11 09:39:52
Android日期选择控件效果如下:
调用的代码:
@OnClick(R.id.btn0)
public void btn0() {
final AlertDialog dialog = new AlertDialog.Builder(context).create();
dialog.show();
Window window = dialog.getWindow();
window.setContentView(R.layout.dialog_change_date);
window.setBackgroundDrawable(new ColorDrawable(0x00000000)); // 处理5.0以上对话框的白边问题
window.setGravity(Gravity.BOTTOM);
final DatePickerView datePickerView = (DatePickerView) window.findViewById(R.id.datePickerView);
//打开页面时需要选中的日期 TODO
datePickerView.setDate(2015, 5, 11);
// datePickerView.setDate(birthdayArray[0], birthdayArray[1], birthdayArray[2]);
final int[] birthdayArray = new int[3];
datePickerView.addOnSelectedChangingListener(new DatePickerView.OnSelectedChangedListener() {
@Override
public void OnSelectedChanged(int[] oldValue, int[] newValue) {
birthdayArray[0] = newValue[0];
birthdayArray[1] = newValue[1];
birthdayArray[2] = newValue[2];
}
});
window.findViewById(R.id.tvCancel).setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
dialog.dismiss();
}
});
window.findViewById(R.id.tvOK).setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
dialog.dismiss();
btn0.setText(birthdayArray[0] + "年" + birthdayArray[1] + "月" + birthdayArray[2] + "日");
}
});
}
1.WheelView 源码(有修改)
2.xml布局文件
<?xml version="1.0" encoding="utf-8"?>
<!--widget_date_picker.xml-->
<!--注意修改页面自定义控件的包名-->
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal">
<de.bvb.rxdemo.widget.DateSelectWidget.wheelview.WheelView
android:id="@+id/wheelViewYear"
android:layout_width="0dp"
android:layout_height="match_parent"
android:layout_gravity="center"
android:layout_weight="1"/>
<de.bvb.rxdemo.widget.DateSelectWidget.wheelview.WheelView
android:id="@+id/wheelViewMonth"
android:layout_width="0dp"
android:layout_height="match_parent"
android:layout_gravity="center"
android:layout_weight="1"/>
<de.bvb.rxdemo.widget.DateSelectWidget.wheelview.WheelView
android:id="@+id/wheelViewDay"
android:layout_width="0dp"
android:layout_height="match_parent"
android:layout_gravity="center"
android:layout_weight="1"/>
</LinearLayout>
<?xml version="1.0" encoding="utf-8"?>
<!--dialog_change_date.xml-->
<!--注意修改页面自定义控件的包名-->
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:background="@android:color/transparent"
android:gravity="bottom"
android:orientation="vertical">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="@android:color/white"
android:orientation="vertical">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="48dp"
android:orientation="horizontal">
<TextView
android:id="@+id/tvCancel"
android:layout_width="0dp"
android:layout_height="match_parent"
android:layout_weight="1"
android:background="#F9F9F9"
android:gravity="center"
android:text="取消"
android:textColor="#43AAFC"/>
<View
android:layout_width="1px"
android:layout_height="match_parent"
android:background="#D7D7D7"/>
<TextView
android:id="@+id/tvOK"
android:layout_width="0dp"
android:layout_height="match_parent"
android:layout_weight="1"
android:background="#F9F9F9"
android:gravity="center"
android:text="确定"
android:textColor="#43AAFC"/>
</LinearLayout>
<View
android:layout_width="match_parent"
android:layout_height="1px"
android:background="#D7D7D7"/>
<de.bvb.rxdemo.widget.DateSelectWidget.DatePickerView
android:id="@+id/datePickerView"
android:layout_width="match_parent"
android:layout_height="wrap_content"/>
</LinearLayout>
</LinearLayout>
3.java文件
package de.bvb.rxdemo.widget.DateSelectWidget;
import android.content.Context;
import android.util.AttributeSet;
import android.view.LayoutInflater;
import android.view.View;
import android.widget.LinearLayout;
import android.widget.TextView;
import java.util.ArrayList;
import de.bvb.rxdemo.R;
import de.bvb.rxdemo.widget.DateSelectWidget.wheelview.OnWheelChangedListener;
import de.bvb.rxdemo.widget.DateSelectWidget.wheelview.OnWheelScrollListener;
import de.bvb.rxdemo.widget.DateSelectWidget.wheelview.WheelView;
import de.bvb.rxdemo.widget.DateSelectWidget.wheelview.adapter.AbstractWheelTextAdapter1;
public class DatePickerView extends LinearLayout {
private static final int YEAR_MIN = 1950;
private static final int YEAR_MAX = 2020;
private int year = YEAR_MIN;
private int month = 1;
private int day = 1;
private ArrayList<Integer> yearList = new ArrayList<>(YEAR_MAX - YEAR_MIN + 1);
private ArrayList<Integer> monthList = new ArrayList<>(12);
private ArrayList<Integer> dayList = new ArrayList<>(31);
private DateTextAdapter yearAdapter;
private DateTextAdapter monthAdapter;
private DateTextAdapter dayAdapter;
private Context context;
private WheelView wheelViewYear;
private WheelView wheelViewMonth;
private WheelView wheelViewDay;
public DatePickerView(Context context) {
this(context, null);
}
public DatePickerView(Context context, AttributeSet attrs) {
this(context, attrs, 0);
}
public DatePickerView(Context context, AttributeSet attrs, int defStyleAttr) {
super(context, attrs, defStyleAttr);
this.context = context;
init();
}
private void init() {
for (int i = YEAR_MIN; i < YEAR_MAX + 1; i++) {
yearList.add(i);
}
for (int i = 1; i < 13; i++) {
monthList.add(i);
}
for (int i = 1; i < 32; i++) {
dayList.add(i);
}
LayoutInflater.from(context).inflate(R.layout.widget_date_picker, this);
// View.inflate(context, R.layout.widget_date_picker, this);
wheelViewYear = (WheelView) findViewById(R.id.wheelViewYear);
wheelViewMonth = (WheelView) findViewById(R.id.wheelViewMonth);
wheelViewDay = (WheelView) findViewById(R.id.wheelViewDay);
wheelViewYear.setCyclic(true);// 可循环滚动
wheelViewMonth.setCyclic(true);// 可循环滚动
wheelViewDay.setCyclic(true);// 可循环滚动
yearAdapter = new DateTextAdapter(context);
monthAdapter = new DateTextAdapter(context);
dayAdapter = new DateTextAdapter(context);
yearAdapter.setList(yearList);
monthAdapter.setList(monthList);
dayAdapter.setList(dayList);
wheelViewYear.setViewAdapter(yearAdapter);
wheelViewMonth.setViewAdapter(monthAdapter);
wheelViewDay.setViewAdapter(dayAdapter);
wheelViewYear.addChangingListener(new OnWheelChangedListener() {
@Override
public void onChanged(WheelView wheel, int oldValue, int newValue) {
year = YEAR_MIN + newValue;
int days = calcDay(year, month); // days=28
dayList = getDayList(days);
dayAdapter.setList(dayList);
if (day > days) {
dayAdapter.currentIndex = days - 1;
wheelViewDay.setCurrentItem(dayAdapter.currentIndex);
} else {
dayAdapter.currentIndex = day - 1; // day = 30
}
if (onSelectedChangedListener != null) {
onSelectedChangedListener.OnSelectedChanged(parseInt2Array(YEAR_MIN + oldValue, month, day), getSelectDate());
}
}
});
wheelViewMonth.addChangingListener(new OnWheelChangedListener() {
@Override
public void onChanged(WheelView wheel, int oldValue, int newValue) {
month = 1 + newValue;
int days = calcDay(year, month); // days=28
dayList = getDayList(days);
dayAdapter.setList(dayList);
if (day > days) {
dayAdapter.currentIndex = days - 1;
wheelViewDay.setCurrentItem(dayAdapter.currentIndex);
} else {
dayAdapter.currentIndex = day - 1; // day = 30
}
if (onSelectedChangedListener != null) {
onSelectedChangedListener.OnSelectedChanged(parseInt2Array(year, 1 + oldValue, day), getSelectDate());
}
}
});
wheelViewDay.addChangingListener(new OnWheelChangedListener() {
@Override
public void onChanged(WheelView wheel, int oldValue, int newValue) {
day = 1 + newValue;
if (onSelectedChangedListener != null) {
onSelectedChangedListener.OnSelectedChanged(parseInt2Array(year, month, oldValue + 1), getSelectDate());
}
}
});
wheelViewYear.addScrollingListener(onWheelScrollListener);
wheelViewMonth.addScrollingListener(onWheelScrollListener);
wheelViewDay.addScrollingListener(onWheelScrollListener);
}
OnWheelScrollListener onWheelScrollListener = new OnWheelScrollListener() {
@Override
public void onScrollingStarted(WheelView wheel) {
}
@Override
public void onScrollingFinished(WheelView wheel) {
setTextViewStyle();
}
};
private void setTextViewStyle() {
setTextViewSize(year + "", yearAdapter);
setTextViewSize(month + "", monthAdapter);
setTextViewSize(day + "", dayAdapter);
}
private void setTextViewSize(String currentItemText, AbstractWheelTextAdapter1 adapter) {
ArrayList<View> arrayList = adapter.getTextViews();
int size = arrayList.size();
String currentText;
TextView textView;
boolean current;
for (int i = 0; i < size; i++) {
textView = (TextView) arrayList.get(i);
currentText = textView.getText().toString();
current = currentItemText.equals(currentText);
textView.setTextColor(current ? adapter.selected_text_color : adapter.un_selected_text_color);
textView.setTextSize(current ? adapter.selected_text_size : adapter.un_selected_text_size);
}
}
/**
* 设置控件的初始值
*
* @param year
* @param month
* @param day
*/
public void setDate(int year, int month, int day) {
//验证合法性
if (year > YEAR_MAX || year < YEAR_MIN) {
year = YEAR_MIN;
// throw new RuntimeException("年份必须在[" + YEAR_MIN + "," + YEAR_MAX + "]之间");
}
if (month > 12 || month < 1) {
month = 1;
// throw new RuntimeException("月份份必须在[" + 1 + "," + 12 + "]之间");
}
final int days = calcDay(year, month);
if (day > days || day < 1) {
day = 1;
// throw new RuntimeException("日期份必须在[" + 1 + "," + days + "]之间");
}
this.year = year;
this.month = month;
this.day = day;
yearAdapter.currentIndex = DatePickerView.this.year - YEAR_MIN;
monthAdapter.currentIndex = DatePickerView.this.month - 1;
dayAdapter.currentIndex = DatePickerView.this.day - 1;
wheelViewYear.setCurrentItem(yearAdapter.currentIndex);
wheelViewMonth.setCurrentItem(monthAdapter.currentIndex);
wheelViewDay.setCurrentItem(dayAdapter.currentIndex);
}
public void addOnSelectedChangingListener(OnSelectedChangedListener onSelectedChangedListener) {
this.onSelectedChangedListener = onSelectedChangedListener;
}
private OnSelectedChangedListener onSelectedChangedListener;
public interface OnSelectedChangedListener {
void OnSelectedChanged(int[] oldValue, int[] newValue);
}
private int[] parseInt2Array(int year, int month, int day) {
return new int[]{year, month, day};
}
private int[] getSelectDate() {
return new int[]{year, month, day};
}
private ArrayList<Integer> getDayList(int days) {
if (days <= 0) {
return null;
}
ArrayList<Integer> list = new ArrayList(days);
for (int i = 1; i < days + 1; i++) {
list.add(i);
}
return list;
}
private int calcDay(int year, int month) {
int days = 0;
switch (month) {
case 1:
case 3:
case 5:
case 7:
case 8:
case 10:
case 12:
days = 31;
break;
case 4:
case 6:
case 9:
case 11:
days = 30;
break;
case 2:
days = (year % 400 == 0 || (year % 100 != 0 && year % 4 == 0)) ? 29 : 28;
break;
}
return days;
}
private class DateTextAdapter extends AbstractWheelTextAdapter1 {
ArrayList<Integer> list;
public DateTextAdapter(Context context) {
super(context, android.R.layout.simple_list_item_1);
// super(context, R.layout.item_birth_year);
// setItemTextResource(R.id.tempValue);
// item_birth_year.xml 内容如下
// <?xml version="1.0" encoding="utf-8"?>
// <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
// android:layout_width="match_parent"
// android:layout_height="35dip"
// android:gravity="center"
// android:orientation="horizontal">
// <TextView
// android:id="@+id/tempValue"
// android:layout_width="match_parent"
// android:layout_height="match_parent"
// android:gravity="center"
// android:textColor="#ffff0000"/>
// </LinearLayout>
}
public void setList(ArrayList<Integer> list) {
this.list = list;
notifyDataChangedEvent();
}
@Override
protected CharSequence getItemText(int index) {
return list == null ? "" : String.valueOf(list.get(index));
}
@Override
public int getItemsCount() {
return list == null ? 0 : list.size();
}
}
}
标签:Android,日期,控件
![](/images/zang.png)
![](/images/jiucuo.png)
猜你喜欢
SpringCloud Alibaba项目实战之nacos-server服务搭建过程
2022-02-28 02:43:26
![](https://img.aspxhome.com/file/2023/2/85542_0s.jpg)
Android实现拍照添加时间水印
2023-10-02 14:23:51
![](https://img.aspxhome.com/file/2023/5/130345_0s.jpg)
为何Java8需要引入新的日期与时间库
2023-12-16 06:02:23
![](https://img.aspxhome.com/file/2023/9/71659_0s.png)
docker 的java编译环境构建详细介绍
2023-02-10 04:08:30
Java三大特性之多态详解
2022-11-07 14:24:40
Android开发之DatePickerDialog、TimePickerDialog时间日期对话框用法示例
2022-10-25 21:21:12
![](https://img.aspxhome.com/file/2023/8/106038_0s.gif)
spring boot配置读写分离的完整实现步骤
2022-02-15 21:23:11
![](https://img.aspxhome.com/file/2023/3/96133_0s.png)
Java实现图片倒影的源码实例内容
2022-08-30 02:39:24
![](https://img.aspxhome.com/file/2023/2/65162_0s.jpg)
Springboot定时任务Scheduled重复执行操作
2021-10-08 00:35:54
C#实现改变DataGrid某一行和单元格颜色的方法
2022-08-23 20:45:22
Android自定义View之圆形进度条式按钮
2023-09-07 22:51:28
![](https://img.aspxhome.com/file/2023/7/138127_0s.png)
Spring使用AspectJ注解和XML配置实现AOP
2023-02-01 09:45:03
AjaxControlToolkit AjaxFileUpload 显示英文改成中文的解决方法
2021-06-01 21:09:57
![](https://img.aspxhome.com/file/2023/5/106425_0s.gif)
一文详解如何在控制台显示MyBatis的SQL语句
2023-01-09 06:43:38
![](https://img.aspxhome.com/file/2023/8/99868_0s.png)
SpringBoot配置Email发送功能实例
2022-03-07 20:52:20
![](https://img.aspxhome.com/file/2023/3/61943_0s.jpg)
Springboot-admin整合Quartz实现动态管理定时任务的过程详解
2023-08-26 03:01:07
![](https://img.aspxhome.com/file/2023/0/94200_0s.png)
Android简单实现启动画面的方法
2022-08-25 17:54:55
基于WPF实现验证码控件
2021-08-15 21:44:36
![](https://img.aspxhome.com/file/2023/5/92975_0s.png)
Android使用ViewPager加载图片和轮播视频
2023-10-26 08:20:04
![](https://img.aspxhome.com/file/2023/8/119098_0s.png)
老生常谈设计模式之动态代理
2021-06-12 06:15:50