Android实现日历控件示例代码
发布时间 - 2026-01-10 23:26:10 点击率:次做的是一个酒店的项目,可以选择入住和离开的日期。声明为了省事在网上找的资料,自己修改的逻辑,希望对需要的朋友有帮助。喜欢的给个好评。谢谢啦!祝生活愉快!

先上图
第一步,搭建布局xml
<ScrollView 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">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical">
<FrameLayout
android:layout_width="match_parent"
android:layout_height="wrap_content">
<TextView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="@color/days_detail"
android:gravity="center"
android:padding="16dp"
android:text="选择住店离店日期"
android:textColor="@color/white"
android:textSize="18sp" />
<ImageView
android:id="@+id/back_to_up"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_marginLeft="10dp"
android:layout_marginTop="10dp"
android:src="@drawable/backto" />
</FrameLayout>
<FrameLayout
android:layout_width="match_parent"
android:layout_height="wrap_content">
<View
android:layout_width="match_parent"
android:layout_height="30dp"
android:background="@color/days_detail" />
<android.support.v7.widget.CardView
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_margin="10dp"
android:layout_marginTop="20dp"
android:padding="10dp"
app:cardCornerRadius="10dp"
app:cardElevation="16dp">
<LinearLayout
android:id="@+id/ll"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="vertical" />
</android.support.v7.widget.CardView>
</FrameLayout>
</LinearLayout>
</ScrollView>
第二部在编写逻辑
LinearLayout ll;
MyCalendar c1;
Date date;
String nowday;
long nd = 1000 * 24L * 60L * 60L;//一天的毫秒数
SimpleDateFormat simpleDateFormat, sd1, sd2;
SharedPreferences sp;
SharedPreferences.Editor editor;
private String inday, outday//日期
sp_inday, sp_outday;//周几
Activity extends BaseActivity implements MyCalendar.OnDaySelectListener {
继承BaseActivity实现点击日历的监听回调
private void init() {
List<String> listDate = getDateList();
LinearLayout.LayoutParams params = new LinearLayout.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT,
ViewGroup.LayoutParams.WRAP_CONTENT);
for (int i = 0; i < listDate.size(); i++) {
c1 = new MyCalendar(this);
c1.setLayoutParams(params);
Date date = null;
try {
date = simpleDateFormat.parse(listDate.get(i));
} catch (ParseException e) {
e.printStackTrace();
}
if (!"".equals(sp_inday)) {
c1.setInDay(sp_inday);
}
if (!"".equals(sp_outday)) {
c1.setOutDay(sp_outday);
}
c1.setTheDay(date);
c1.setOnDaySelectListener(this);
ll.addView(c1);
}
}
@Override
public void onDaySelectListener(View view, String date) {
//若日历日期小于当前日期,或日历日期-当前日期超过三个月,则不能点击
try {
if (simpleDateFormat.parse(date).getTime() < simpleDateFormat.parse(nowday).getTime()) {
return;
}
long dayxc = (simpleDateFormat.parse(date).getTime() - simpleDateFormat.parse(nowday).getTime()) / nd;
if (dayxc > 90) {
return;
}
} catch (ParseException e) {
e.printStackTrace();
}
//若以前已经选择了日期,则在进入日历后会显示以选择的日期,该部分作用则是重新点击日历时,清空以前选择的数据(包括背景图案)
if (!"".equals(sp_inday)) {
c1.viewIn.setBackgroundColor(Color.WHITE);
((TextView) c1.viewIn.findViewById(R.id.tv_calendar_day)).setTextColor(Color.BLACK);
((TextView) c1.viewIn.findViewById(R.id.tv_calendar)).setText("");
}
if (!"".equals(sp_outday)) {
c1.viewOut.setBackgroundColor(Color.WHITE);
((TextView) c1.viewOut.findViewById(R.id.tv_calendar_day)).setTextColor(Color.BLACK);
((TextView) c1.viewOut.findViewById(R.id.tv_calendar)).setText("");
}
String dateDay = date.split("-")[2];
if (Integer.parseInt(dateDay) < 10) {
dateDay = date.split("-")[2].replace("0", "");
}
TextView textDayView = (TextView) view.findViewById(R.id.tv_calendar_day);
TextView textView = (TextView) view.findViewById(R.id.tv_calendar);
view.setBackgroundColor(Color.parseColor("#33B5E5"));
textDayView.setTextColor(Color.WHITE);
if (null == inday || inday.equals("")) {
textDayView.setText(dateDay);
textView.setText("入住");
inday = date;
} else {
if (inday.equals(date)) {
view.setBackgroundColor(Color.WHITE);
textDayView.setText(dateDay);
textDayView.setTextColor(Color.BLACK);
textView.setText("");
inday = "";
} else {
try {
if (simpleDateFormat.parse(date).getTime() < simpleDateFormat.parse(inday).getTime()) {
view.setBackgroundColor(Color.WHITE);
textDayView.setTextColor(Color.BLACK);
Toast.makeText(CalendarActivity.this, "离开日期不能小于入住日期", Toast.LENGTH_SHORT).show();
return;
}
} catch (ParseException e) {
e.printStackTrace();
}
textDayView.setText(dateDay);
textView.setText("离开");
outday = date;
editor.putString("dateIn", inday);
editor.putString("dateOut", outday);
editor.commit();
finish();
}
}
}
//根据当前日期,向后数三个月(若当前day不是1号,为满足至少90天,则需要向后数4个月)
@SuppressLint("SimpleDateFormat")
public List<String> getDateList() {
List<String> list = new ArrayList<String>();
Date date = new Date();
int nowMon = date.getMonth() + 1;
String yyyy = sd1.format(date);
String dd = sd2.format(date);
if (nowMon == 9) {
list.add(simpleDateFormat.format(date));
list.add(yyyy + "-10-" + dd);
list.add(yyyy + "-11-" + dd);
if (!dd.equals("01")) {
list.add(yyyy + "-12-" + dd);
}
} else if (nowMon == 10) {
list.add(yyyy + "-10-" + dd);
list.add(yyyy + "-11-" + dd);
list.add(yyyy + "-12-" + dd);
if (!dd.equals("01")) {
list.add((Integer.parseInt(yyyy) + 1) + "-01-" + dd);
}
} else if (nowMon == 11) {
list.add(yyyy + "-11-" + dd);
list.add(yyyy + "-12-" + dd);
list.add((Integer.parseInt(yyyy) + 1) + "-01-" + dd);
if (!dd.equals("01")) {
list.add((Integer.parseInt(yyyy) + 1) + "-02-" + dd);
}
} else if (nowMon == 12) {
list.add(yyyy + "-12-" + dd);
list.add((Integer.parseInt(yyyy) + 1) + "-01-" + dd);
list.add((Integer.parseInt(yyyy) + 1) + "-02-" + dd);
if (!dd.equals("01")) {
list.add((Integer.parseInt(yyyy) + 1) + "-03-" + dd);
}
} else {
list.add(yyyy + "-" + getMon(nowMon) + "-" + dd);
list.add(yyyy + "-" + getMon((nowMon + 1)) + "-" + dd);
list.add(yyyy + "-" + getMon((nowMon + 2)) + "-" + dd);
if (!dd.equals("01")) {
list.add(yyyy + "-" + getMon((nowMon + 3)) + "-" + dd);
}
}
return list;
}
public String getMon(int mon) {
String month = "";
if (mon < 10) {
month = "0" + mon;
} else {
month = "" + mon;
}
return month;
}
第三部 编写监听,自定义的控件
public class MyCalendar extends LinearLayout {
private static Context context;
private Date theInDay;
private String inday = "", outday = "";
public static View viewIn;
public static View viewOut;
public static String positionIn;
public static String positionOut;
public static final int WEEKDAYS = 7;
public static String[] WEEK = {
"周日",
"周一",
"周二",
"周三",
"周四",
"周五",
"周六"
};
static long nd = 1000 * 24L * 60L * 60L;//一天的毫秒数
private List<String> gvList;//存放天
private OnDaySelectListener callBack;//回调函数
private static String nowday = new SimpleDateFormat("yyyy-MM-dd").format(new Date());
private SimpleDateFormat dateFormat = new SimpleDateFormat("yyyyMM");//日期格式化
private static SimpleDateFormat dateFormat2 = new SimpleDateFormat("yyyy-MM-dd");//日期格式化
/**
* 构造函数
*
* @param context
*/
public MyCalendar(Context context) {
super(context);
MyCalendar.context = context;
}
/**
* 日期变量转成对应的星期字符串
*
* @param date
* @return
*/
public static String DateToWeek(Date date) {
Calendar calendar = Calendar.getInstance();
calendar.setTime(date);
int dayIndex = calendar.get(Calendar.DAY_OF_WEEK);
if (dayIndex < 1 || dayIndex > WEEKDAYS) {
return null;
}
return WEEK[dayIndex - 1];
}
/**
* 获得天数差
*
* @param begin
* @param end
* @return
*/
public static long getDayDiff(Date begin, Date end) {
long day = 1;
if (end.getTime() < begin.getTime()) {
day = -1;
} else if (end.getTime() == begin.getTime()) {
day = 1;
} else {
day += (end.getTime() - begin.getTime()) / (24 * 60 * 60 * 1000);
}
return day;
}
/**
* 将yyyy-MM-dd类型转换成MM.dd
*
* @param time
* @return
* @throws ParseException
*/
public static String format1(String time) throws ParseException {
SimpleDateFormat from = new SimpleDateFormat("yyyy-MM-dd", Locale.CHINA);
SimpleDateFormat to = new SimpleDateFormat("MM.dd", Locale.CHINA);
return to.format(from.parse(time));
}
/**
* 获得指定日期的后一天
*
* @param specifiedDay
* @return
*/
public static String getSpecifiedDayAfter(String specifiedDay) {
Calendar c = Calendar.getInstance();
Date date = null;
try {
date = new SimpleDateFormat("yy-MM-dd").parse(specifiedDay);
} catch (ParseException e) {
e.printStackTrace();
}
c.setTime(date);
int day = c.get(Calendar.DATE);
c.set(Calendar.DATE, day + 1);
String dayAfter = new SimpleDateFormat("yyyy-MM-dd")
.format(c.getTime());
return dayAfter;
}
/**
* 构造函数
*
* @param context
*/
public MyCalendar(Context context, AttributeSet attrs) {
super(context, attrs);
MyCalendar.context = context;
}
public void setInDay(String inday) {
this.inday = inday;
}
public void setOutDay(String outday) {
this.outday = outday;
}
public void setTheDay(Date dateIn) {
this.theInDay = dateIn;
init();
}
/**
* 初始化日期以及view等控件
*/
private void init() {
gvList = new ArrayList<String>();//存放天
final Calendar cal = Calendar.getInstance();//获取日历实例
cal.setTime(theInDay);//cal设置为当天的
cal.set(Calendar.DATE, 1);//cal设置当前day为当前月第一天
int tempSum = countNeedHowMuchEmpety(cal);//获取当前月第一天为星期几
int dayNumInMonth = getDayNumInMonth(cal);//获取当前月有多少天
setGvListData(tempSum, dayNumInMonth, cal.get(Calendar.YEAR) + "-" + getMonth((cal.get(Calendar.MONTH) + 1)));
View view = LayoutInflater.from(context).inflate(R.layout.comm_calendar, this, true);//获取布局,开始初始化
TextView tv_year = (TextView) view.findViewById(R.id.tv_year);
if (cal.get(Calendar.YEAR) > new Date().getYear()) {
tv_year.setVisibility(View.VISIBLE);
tv_year.setText(cal.get(Calendar.YEAR) + "年");
}
TextView tv_month = (TextView) view.findViewById(R.id.tv_month);
tv_month.setText(String.valueOf(theInDay.getMonth() + 1));
MyGridView gv = (MyGridView) view.findViewById(R.id.gv_calendar);
calendarGridViewAdapter gridViewAdapter = new calendarGridViewAdapter(gvList, inday, outday);
gv.setAdapter(gridViewAdapter);
gv.setOnItemClickListener(new OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> adapterView, View arg1, int position, long arg3) {
String choiceDay = (String) adapterView.getAdapter().getItem(position);
String[] date = choiceDay.split(",");
String day = date[1];
if (!" ".equals(day)) {
if (Integer.parseInt(day) < 10) {
day = "0" + date[1];
}
choiceDay = date[0] + "-" + day;
if (callBack != null) {//调用回调函数回调数据
callBack.onDaySelectListener(arg1, choiceDay);
}
}
}
});
}
/**
* 为gridview中添加需要展示的数据
*
* @param tempSum
* @param dayNumInMonth
*/
private void setGvListData(int tempSum, int dayNumInMonth, String YM) {
gvList.clear();
for (int i = 0; i < tempSum; i++) {
gvList.add(" , ");
}
for (int j = 1; j <= dayNumInMonth; j++) {
gvList.add(YM + "," + String.valueOf(j));
}
}
private String getMonth(int month) {
String mon = "";
if (month < 10) {
mon = "0" + month;
} else {
mon = "" + month;
}
return mon;
}
/**
* 获取当前月的总共天数
*
* @param cal
* @return cal.getActualMaximum(Calendar.DATE)
*/
private int getDayNumInMonth(Calendar cal) {
return cal.getActualMaximum(Calendar.DATE);
}
/**
* 获取当前月第一天在第一个礼拜的第几天,得出第一天是星期几
*
* @param cal
* @return firstDayInWeek
*/
private int countNeedHowMuchEmpety(Calendar cal) {
int firstDayInWeek = cal.get(Calendar.DAY_OF_WEEK) - 1;
Log.e("MyCalendar", String.valueOf(firstDayInWeek));
return firstDayInWeek;
}
/**
* gridview中adapter的viewholder
*
* @author wx
*/
static class GrideViewHolder {
TextView tvDay, tv;
}
/**
* gridview的adapter
*
* @author Administrator
*/
static class calendarGridViewAdapter extends BaseAdapter {
List<String> gvList;//存放天
String inday, outday;
public calendarGridViewAdapter(List<String> gvList, String inday, String outday) {
super();
this.gvList = gvList;
this.inday = inday;
this.outday = outday;
}
@Override
public int getCount() {
return gvList.size();
}
@Override
public String getItem(int position) {
return gvList.get(position);
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
GrideViewHolder holder;
if (convertView == null) {
holder = new GrideViewHolder();
convertView = inflate(context, R.layout.common_calendar_gridview_item, null);
holder.tv = (TextView) convertView.findViewById(R.id.tv_calendar);
holder.tvDay = (TextView) convertView.findViewById(R.id.tv_calendar_day);
convertView.setTag(holder);
} else {
holder = (GrideViewHolder) convertView.getTag();
}
String[] date = getItem(position).split(",");
holder.tvDay.setText(date[1]);
if ((position + 1) % 7 == 0 || (position) % 7 == 0) {
holder.tvDay.setTextColor(Color.parseColor("#339900"));
}
if (!date[1].equals(" ")) {
String day = date[1];
if (Integer.parseInt(date[1]) < 10) {
day = "0" + date[1];
}
if ((date[0] + "-" + day).equals(nowday)) {
holder.tvDay.setTextColor(Color.parseColor("#FF6600"));
holder.tvDay.setTextSize(15);
holder.tvDay.setText("今天");
}
if (!"".equals(inday) && (date[0] + "-" + day).equals(inday)) {
convertView.setBackgroundColor(Color.parseColor("#33B5E5"));
holder.tvDay.setTextColor(Color.WHITE);
holder.tvDay.setText(date[1]);
holder.tv.setText("入住");
viewIn = convertView;
positionIn = date[1];
}
if (!"".equals(outday) && (date[0] + "-" + day).equals(outday)) {
convertView.setBackgroundColor(Color.parseColor("#33B5E5"));
holder.tvDay.setTextColor(Color.WHITE);
holder.tvDay.setText(date[1]);
holder.tv.setText("离开");
viewOut = convertView;
positionOut = date[1];
}
try {
//若日历日期<当前日期,则不能选择
if (dateFormat2.parse(date[0] + "-" + day).getTime() < dateFormat2.parse(nowday).getTime()) {
holder.tvDay.setTextColor(Color.parseColor("#999999"));
}
//若日历日期-当前日期>90天,则不能选择
long dayxc = (dateFormat2.parse(date[0] + "-" + day).getTime() - dateFormat2.parse(nowday).getTime()) / nd;
if (dayxc > 90) {
holder.tvDay.setTextColor(Color.parseColor("#999999"));
}
} catch (ParseException e) {
e.printStackTrace();
}
}
return convertView;
}
}
/**
* 自定义监听接口
*
* @author Administrator
*/
public interface OnDaySelectListener {
void onDaySelectListener(View view, String date);
}
/**
* 自定义监听接口设置对象
*
* @param o
*/
public void setOnDaySelectListener(OnDaySelectListener o) {
callBack = o;
}
}
在界面显示选择的日期
接下来就是为了显示选择的日期进行逻辑判断,包括字符串的转换以及日期格式的转换,日期的计算等。
simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd");
if ("".equals(home_into_date.getText().toString())
&& "".equals(home_out_date.getText().toString())) {
inday = simpleDateFormat.format(new Date());
try {
//转换时间格式
String changeDate = MyCalendar.format1(inday);
Date dateIn = simpleDateFormat.parse(inday);
//将日期转换成周
String weekIn = MyCalendar.DateToWeek(dateIn);
home_into_date.setText("" + changeDate + weekIn);
String nextDay = MyCalendar.format1(MyCalendar.getSpecifiedDayAfter(inday));
Date dateOut = simpleDateFormat.parse(MyCalendar.getSpecifiedDayAfter(inday));
String weekOut = MyCalendar.DateToWeek(dateOut);
home_out_date.setText("" + nextDay + weekOut);
long days = MyCalendar.getDayDiff(dateIn, dateOut);
home_total_days.setText("共" + (days - 1) + "晚");
} catch (ParseException e) {
e.printStackTrace();
}
} else {
//这里使用sp传的值
inday = pro.getString("dateIn", "");
outday = pro.getString("dateOut", "");
try {
String changeDate = MyCalendar.format1(inday);
Date dateIn = simpleDateFormat.parse(inday);
//将日期转换成周
String weekIn = MyCalendar.DateToWeek(dateIn);
home_into_date.setText("" + changeDate + weekIn);
String outDay = MyCalendar.format1(outday);
Date dateOut = simpleDateFormat.parse(outday);
String weekOut = MyCalendar.DateToWeek(dateOut);
home_out_date.setText("" + outDay + weekOut);
long days = MyCalendar.getDayDiff(dateIn, dateOut);
home_total_days.setText("共" + (days - 1) + "晚");
} catch (ParseException e) {
e.printStackTrace();
}
其中有几个布局,给图自己写吧
comm_calendar
common_calendar_gridview_item
<LinearLayout
android:layout_width="0dp"
android:layout_height="match_parent"
android:layout_weight="1"
android:gravity="center"
android:orientation="vertical">
<LinearLayout
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:gravity="center"
android:orientation="vertical">
<TextView
android:id="@+id/tv_calendar_day"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:gravity="center"
android:textSize="15sp" />
<TextView
android:id="@+id/tv_calendar"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:gravity="center"
android:textColor="#FFFFFF"
android:textSize="15sp" />
</LinearLayout>
<View
android:layout_width="match_parent"
android:layout_height="0.5dp"
android:background="#E4E4E4" />
</LinearLayout>
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
# android日历控件
# android日历控件大全
# android实现日历
# Android 一个日历控件的实现代码
# Android开发之日历CalendarView用法示例
# Android实现自定义日历
# android 开发教程之日历项目实践(一)
# Android自定义控件实现可多选课程日历CalendarView
# android 开发教程之日历项目实践(三)
# android 开发教程之日历项目实践(二)
# Android可签到日历控件的实现方法
# Android 仿日历翻页、仿htc时钟翻页、数字翻页切换效果
# Android Studio简单实现自定义日历
# 回调
# 自定义
# 转换成
# 的是
# 第一个
# 则是
# 几天
# 有多少
# 有几个
# 第二部
# 可以选择
# 后会
# 设置为
# 则在
# 要向
# 第三部
# 给个
# 个月
# 转成
# 大家多多
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
大型企业网站制作流程,做网站需要注册公司吗?
javascript事件捕获机制【深入分析IE和DOM中的事件模型】
google浏览器怎么清理缓存_谷歌浏览器清除缓存加速详细步骤
Laravel辅助函数有哪些_Laravel Helpers常用助手函数大全
香港服务器网站测试全流程:性能评估、SEO加载与移动适配优化
在线教育网站制作平台,山西立德教育官网?
Laravel集合Collection怎么用_Laravel集合常用函数详解
jQuery中的100个技巧汇总
标题:Vue + Vuex + JWT 身份认证的正确实践与常见误区解析
详解一款开源免费的.NET文档操作组件DocX(.NET组件介绍之一)
Laravel如何记录自定义日志?(Log频道配置)
百度浏览器ai对话怎么关 百度浏览器ai聊天窗口隐藏
头像制作网站在线观看,除了站酷,还有哪些比较好的设计网站?
JS去除重复并统计数量的实现方法
佛山网站制作系统,佛山企业变更地址网上办理步骤?
如何在云主机上快速搭建网站?
如何将凡科建站内容保存为本地文件?
node.js报错:Cannot find module 'ejs'的解决办法
宙斯浏览器文件分类查看教程 快速筛选视频文档与图片方法
如何用景安虚拟主机手机版绑定域名建站?
东莞专业网站制作公司有哪些,东莞招聘网站哪个好?
Laravel Seeder怎么填充数据_Laravel数据库填充器的使用方法与技巧
Laravel如何实现API资源集合?(Resource Collection教程)
哪家制作企业网站好,开办像阿里巴巴那样的网络公司和网站要怎么做?
制作电商网页,电商供应链怎么做?
今日头条AI怎样推荐抢票工具_今日头条AI抢票工具推荐算法与筛选【技巧】
Laravel如何配置Horizon来管理队列?(安装和使用)
如何在云主机快速搭建网站站点?
HTML5建模怎么导出为FBX格式_FBX格式兼容性及导出步骤【指南】
制作网站软件推荐手机版,如何制作属于自己的手机网站app应用?
如何用AWS免费套餐快速搭建高效网站?
音响网站制作视频教程,隆霸音响官方网站?
nginx修改上传文件大小限制的方法
如何快速打造个性化非模板自助建站?
Laravel如何记录日志_Laravel Logging系统配置与自定义日志通道
Laravel如何实现API版本控制_Laravel版本化API设计方案
网站页面设计需要考虑到这些问题
如何用手机制作网站和网页,手机移动端的网站能制作成中英双语的吗?
Laravel如何实现用户角色和权限系统_Laravel角色权限管理机制
打造顶配客厅影院,这份100寸电视推荐名单请查收
Laravel怎么连接多个数据库_Laravel多数据库连接配置
惠州网站建设制作推广,惠州市华视达文化传媒有限公司怎么样?
*服务器网站为何频现安全漏洞?
详解免费开源的DotNet二维码操作组件ThoughtWorks.QRCode(.NET组件介绍之四)
Laravel怎么实现验证码(Captcha)功能
如何在香港服务器上快速搭建免备案网站?
如何用好域名打造高点击率的自主建站?
Claude怎样写约束型提示词_Claude约束提示词写法【教程】
Laravel Octane如何提升性能_使用Laravel Octane加速你的应用
Android使用GridView实现日历的简单功能

