1 大致思路
1) 初始化日历数据,作为list传入到RecyclerView.Adapter
2) 重写RecyclerView的onTouchEvent方法,监听手势的改变,然后更改list数据,重新显示UI

上个效果图

2 关键代码

那么整个项目的重点在于如何获取到正确的日期数据,这个在网上找了一个CalendarTool经过测试确实还不错,省去很多工夫,稍作修改就直接拿来用了。

public class CalendarTool<T extends BaseDateEntity> {private final String TAG = CalendarTool.class.getSimpleName();public static int FLING_MIN_DISTANCE = 100;private final int[] weekDayRow = {0, 1, 2, 3, 4, 5, 6};private ArrayList<DateEntity> mDataList = new ArrayList<>();//日期数组private ArrayList<T> mRecordList;//事件记录数组private DateEntity   mDateEntity;private int          mYear;private int          mMonth;private boolean mEndBelong;private boolean mStartBelong;private int     mStartDay;private int     mEndDay;/*** 当前年月日*/private int mCurrenYear;private int mCurrenMonth;private int mCurrenDay;/*** 平年月天数数组*/int commonYearMonthDay[] = {31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};/*** 闰年月天数数组*/int leapYearMonthDay[]   = {31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};public CalendarTool() {/** 初始化当前系统的日期 */Calendar calendar = Calendar.getInstance();mCurrenYear = calendar.get(Calendar.YEAR);mCurrenMonth = calendar.get(Calendar.MONTH) + 1;mCurrenDay = calendar.get(Calendar.DAY_OF_MONTH);this.mYear = mCurrenYear;this.mMonth = mCurrenMonth;}/*** 获取当前日历的年月 x为年,y为月*/public Point getNowCalendar() {Point p = new Point(mYear, mMonth);return p;}/*** 判断第一天属不属于本月*/public boolean isStartBelong() {return mStartBelong;}/*** 判断最后一天属不属于本月*/public boolean isEndBelong() {return mEndBelong;}/*** 获取日历第一天的日期*/public int getStartDay() {return mStartDay;}/*** 获取日历最后一天的日期*/public int getEndDay() {return mEndDay;}public ArrayList<DateEntity> initDateList() {return initDateList(mYear, mMonth);}public void initRecordList(ArrayList<T> recordList) {mRecordList = recordList;}/*** 通过年月获取当前页面的日期集合*/private ArrayList<DateEntity> initDateList(int year, int month) {Log.i(TAG, "initDateList: year = " + year + " month = " + month);mDataList.clear();/** 修改部分 */int endDate = 0;// 得到上一个月的天数,作为上一个月在本日历的结束日期if ((year - 1) == this.mYear || month == 1) {// 说明向前翻了一年,那么上个月的天数就应该是上一年的12月的天数,或者到翻到一月份的时候,那么上一个月的天数也是上一年的12月份的天数endDate = this.getDays(year - 1, 12);} else {// 得到上一个月的天数,作为上一个月在本日历的结束日期endDate = this.getDays(year, month - 1);}/** 修改部分结束 */this.mYear = year;// 当前日历上显示的年this.mMonth = month;// 当前日历上显示的月int days = this.getDays(year, month);// 得到本月的总共天数int dayOfWeek = this.getWeekDay(year, month);//得到当前年月的第一天为星期几int selfDaysEndWeek = 0;// 本月的最后一天是星期几mStartBelong = true;/** 先添加前面不属于本月的 */if (dayOfWeek != 0) {int startDate = endDate - dayOfWeek + 1;// 当前月的上一个月在本日历的开始日期for (int i = startDate, j = 0; i <= endDate; i++, j++) {mDateEntity = new DateEntity(year, month - 1, i);mDateEntity.date = mDateEntity.year * 10000 + mDateEntity.month * 100 + i;if (startDate == i) {mStartBelong = false;mStartDay = mDateEntity.date;}mDateEntity.isSelfMonthDate = false;mDateEntity.weekDay = weekDayRow[j];mDateEntity.hasRecord = hasRecord(mDateEntity.date);mDataList.add(mDateEntity);}}/** 添加本月的 */for (int i = 1, j = dayOfWeek; i <= days; i++, j++) {mDateEntity = new DateEntity(year, month, i);mDateEntity.date = mDateEntity.year * 10000 + mDateEntity.month * 100 + i;if (mStartBelong && i == 1) {mStartDay = mDateEntity.date;}if (i == days) {mEndDay = mDateEntity.date;}mDateEntity.isSelfMonthDate = true;if (j >= 7) {j = 0;}selfDaysEndWeek = j;mDateEntity.weekDay = weekDayRow[j];if (year == mCurrenYear && month == mCurrenMonth && i == mCurrenDay) {mDateEntity.isNowDate = true;}mDateEntity.hasRecord = hasRecord(mDateEntity.date);mDataList.add(mDateEntity);}mEndBelong = true;/*** 添加后面下一个月的 */for (int i = 1, j = selfDaysEndWeek + 1; i < 7; i++, j++) {if (j >= 7) {break;}mEndBelong = false;mDateEntity = new DateEntity(year, month + 1, i);if (mDateEntity.month > 12) {mDateEntity.year = year + 1;mDateEntity.month = 1;}mDateEntity.date = mDateEntity.year * 10000 + mDateEntity.month * 100 + i;mDateEntity.isSelfMonthDate = false;mDateEntity.weekDay = weekDayRow[j];mDateEntity.hasRecord = hasRecord(mDateEntity.date);mDataList.add(mDateEntity);mEndDay = mDateEntity.year * 10000 + mDateEntity.month * 100 + i;}return mDataList;}/*** 通过年月,获取这个月一共有多少天*/private int getDays(int year, int month) {int days = 0;if ((year % 4 == 0 && (year % 100 != 0)) || (year % 400 == 0)) {if (month > 0 && month <= 12) {days = leapYearMonthDay[month - 1];}} else {if (month > 0 && month <= 12) {days = commonYearMonthDay[month - 1];}}return days;}private boolean hasRecord(int date) {if (mRecordList != null) {for (T baseDateEntity : mRecordList) {if (baseDateEntity.year * 10000 + baseDateEntity.month * 100 + baseDateEntity.day == date) {return true;}}}return false;}/*** 通过年,月获取当前月的第一天为星期几 ,返回0是星期天,1是星期一,依次类推*/private int getWeekDay(int year, int month) {int dayOfWeek;int goneYearDays = 0;int thisYearDays = 0;boolean isLeapYear = false;//闰年int commonYearMonthDay[] = {31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};int leapYearMonthDay[] = {31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};for (int i = 1900; i < year; i++) {// 从1900年开始算起,1900年1月1日为星期一if ((i % 4 == 0 && (i % 100 != 0)) || (i % 400 == 0)) {goneYearDays = goneYearDays + 366;} else {goneYearDays = goneYearDays + 365;}}if ((year % 4 == 0 && (year % 100 != 0)) || (year % 400 == 0)) {isLeapYear = true;for (int i = 0; i < month - 1; i++) {thisYearDays = thisYearDays + leapYearMonthDay[i];}} else {isLeapYear = false;for (int i = 0; i < month - 1; i++) {thisYearDays = thisYearDays + commonYearMonthDay[i];}}dayOfWeek = (goneYearDays + thisYearDays + 1) % 7;Log.d(this.getClass().getName(), "从1990到现在有" + (goneYearDays + thisYearDays + 1) + "天");Log.d(this.getClass().getName(), year + "年" + month + "月" + 1 + "日是星期" + dayOfWeek);return dayOfWeek;}public void flushDate(float distance_x) {if (distance_x < 0) {// Fling rightif (mMonth + 1 > 12) {mDataList = initDateList(mYear + 1, 1);} else {mDataList = initDateList(mYear, mMonth + 1);}} else {// Fling leftif (mMonth - 1 <= 0) {mDataList = initDateList(mYear - 1, 12);} else {mDataList = initDateList(mYear, mMonth - 1);}}}
}

关键部分在于initDateList方法,会根据当前传入的年月数据来计算当前日历该显示的数据,包括显示前一个月的日期以及后一个月的日期。那么APP要做的就只是不断的传入年月信息,重新去计算日历数据
。这里重写了RecyclerView方法的onTouchEvent来判断用户的触摸方向,后来因为要实现每一个日历的点击事件,与监听的滑动事件冲突了,
所以也重写了onInterceptTouchEvent方法来拦截子控件的点击事件。

public class CalendarRecycleView<T extends BaseDateEntity> extends RecyclerView {private CalendarRecycleViewAdapter mAdapter;private Context                    mContext;private CalendarTool               mCalendarTool;private OnCalendarDateListener     mDateListener;private float                      motion_x;public CalendarRecycleView(Context context) {super(context);mContext = context;init();}public CalendarRecycleView(Context context, AttributeSet attrs) {super(context, attrs);mContext = context;init();}public CalendarRecycleView(Context context, AttributeSet attrs, int defStyle) {super(context, attrs, defStyle);mContext = context;init();}private void init() {setLayoutManager(new StaggeredGridLayoutManager(7, StaggeredGridLayoutManager.VERTICAL));mCalendarTool = new CalendarTool();mAdapter = new CalendarRecycleViewAdapter(mContext, mCalendarTool.initDateList());setAdapter(mAdapter);mAdapter.setOnItemListener(new CalendarRecycleViewAdapter.OnItemListener() {@Overridepublic void onItemClick(DateEntity dateEntity) {if (mDateListener != null) {mDateListener.onDateItemClick(dateEntity);}}});}public void initRecordList(ArrayList<T> list) {mCalendarTool.initRecordList(list);mCalendarTool.initDateList();mAdapter.notifyDataSetChanged();}public void setOnCalendarDateListener(OnCalendarDateListener listener) {this.mDateListener = listener;}@Overridepublic boolean onTouchEvent(MotionEvent event) {switch (event.getAction()) {case MotionEvent.ACTION_DOWN:motion_x = event.getX();Log.i("onTouchEvent", "ACTION_DOWN: " + event.getX() + "  " + motion_x);break;case MotionEvent.ACTION_MOVE:Log.i("onTouchEvent", "ACTION_MOVE: " + event.getX() + "  " + motion_x);float x = event.getX() - motion_x;if (Math.abs(x) > CalendarTool.FLING_MIN_DISTANCE && motion_x != 0) {mCalendarTool.flushDate(x);mAdapter.notifyDataSetChanged();motion_x = 0;if (mDateListener != null) {mDateListener.onDateChange(mCalendarTool.getNowCalendar(),mCalendarTool.getStartDay(),mCalendarTool.getEndDay(),mCalendarTool.isStartBelong(),mCalendarTool.isEndBelong());}return true;}break;case MotionEvent.ACTION_UP:break;}return super.onTouchEvent(event);}@Overridepublic boolean onInterceptTouchEvent(MotionEvent event) {switch (event.getAction()) {case MotionEvent.ACTION_DOWN:motion_x = event.getX();break;case MotionEvent.ACTION_MOVE:if (Math.abs(motion_x - event.getX()) >= mCalendarTool.FLING_MIN_DISTANCE) {return true;}break;case MotionEvent.ACTION_UP:break;}return super.onInterceptTouchEvent(event);}
}

3 结论
实现起来很简单,就是细节方面要多注意,然后因为没有用到ViewPager,所以滑动起来没有动画效果,加上去也简单,关键是要把ViewPager fragment的list数据计算好就成。
下载:

http://download.csdn.net/detail/u012521570/9879095

Android 使用RecycleView打造自定义日历相关推荐

  1. android程序日历layout,Android使用GridLayout绘制自定义日历控件

    效果图 思路:就是先设置Gridlayout的行列数,然后往里面放置一定数目的自定义日历按钮控件,最后实现日历逻辑就可以了. 步骤: 第一步:自定义日历控件(初步) 第二步:实现自定义单个日期按钮控件 ...

  2. 农历 Android Java 节气_Android自定义日历,可以点击、标注日期、节气、旧历等

    /****************从此出开始将代码拷贝到一个文件中*******************/package cc.util.android.view; import java.text. ...

  3. android gridview控件使用详解_Android开发实现自定义日历、日期选择控件

    点击上方蓝字关注 ?? 来源: wenzhihao123 https://www.jianshu.com/p/a2f102c728ce 前言 最近项目需要日历效果,考虑用第三方的反而不太适合设计需求, ...

  4. android 日历图,Android中的自定义日历视图

    我正在为我的 Android应用程序构建一个自定义日历视图,允许您在几个月之间滑动.我已经创建了自定义日历方块视图,我已将其嵌入到自定义日历月视图中,并且所有内容都在1个月的范围内完美运行. 不幸的是 ...

  5. android 自定义日历 数据,Android实现自定义日历.pdf

    Android实实现现自自定定义义日日历历 自定义日历类源码,可以自己在里面按照需求增减功能,做成自己想要的日历效果,小伙伴们可以根据自己的需求自 由 改 自定义日历控件,支持旧历.节气.日期标注.点 ...

  6. Android自定义日历控件,自带农历节假日,已经开源,即取即用~

    关注本人的更多博客:http://www.cnblogs.com/liushilin/ 该自定义日历控件已经开源:github地址 可能不少的小伙伴都有看楼主昨天发的自定义日历控件,虽然实现功能不多, ...

  7. android+高仿+日历,android高仿钉钉和小米的自定义日历控件(支持阴历和阳历,左右无限翻页viewpager)...

    收藏 0 简介 这是一个高仿钉钉和小米的日历控件,支持快速滑动,界面缓存.想要定制化UI,使用起来非常简单,就像使用ListView一样 一些特点: 可以自定义日历控件UI 支持快速滑动 支持农历和阳 ...

  8. 自定义日历控android,Android 一个日历控件的实现小记

    先看几张动态的效果图吧! 这里主要记录一下在编写日历控件过程中一些主要的点: 一.主要功能 1.支持农历.节气.常用节假日 2.日期范围设置,默认支持的最大日期范围[1900.1~2049.12] 3 ...

  9. android 获取对话框对象,Android 基本Dialog和自定义Dialog

    Android 基本Dialog和自定义Dialog Dialog类是对话框的基类,但你应该避免直接实例化Dialog ,可以使用子类 1.AlertDialog 此对话框可以显示标题,最多三个按钮, ...

最新文章

  1. OPENGL ES 对象的拾取
  2. python在线课程-开始网上在线深度学习python课程
  3. androidx使用FileProvider适配安卓7
  4. Java se之动态代理
  5. 线性表 - 数据结构和算法06
  6. 矩阵特征值的用matlab,[急求]谁可以用matlab帮我运行求矩阵特征值的命令???...
  7. WEB-INF/views/menu/list.jsp (line: 26, column: 58) equal symbol expected
  8. esp分区创建 linux_Elementary OS - 号称最漂亮的 Linux 发行版
  9. c语言中char有无符号,char代表有符号还是无符号?
  10. hr面试十大经典提问
  11. colorbox加载ajax调用的html页面,ColorBox
  12. 【观察】星环科技重构数据云平台,持续释放数据红利和价值
  13. 安装linux系统提示acpi,安装Linux系统时的ACPI和APIC问题
  14. 工信部信息技术发展司谢少锋司长高度评价华云数据“自主创新”的钻研精神
  15. 发票自动处理——摆脱纸张和数据输入的束缚,自动化工作流程和异常处理,大幅缩短审核准备时间
  16. overview of hevc(一)
  17. 康托尔集:1883年,康托尔构造的一个“分形”,称作康托尔集,从数轴上单位长度线段开始,康托尔取走其中间的1/3而达到第一阶段;然后从每一个余下的1/3线段中取走其中间的1/3而达到第二阶段
  18. css特效1:流光背景?我不允许你还不会
  19. 删除中间和后面的星号
  20. JavaScript 三种输出方式

热门文章

  1. UnityShader凹凸感
  2. sql查询:查询药品的信息和被收藏的次数
  3. pycharm安装到32位操作系统
  4. SLC、MLC、TLC 和 QLC NAND SSD 之间的区别:哪个更好?
  5. SLC MLC TLC QLC闪存颗粒
  6. python 优先队列_python实现最大优先队列 python优先级队列如何最大值优先
  7. [LeetCode]187. 重复的DNA序列(java实现)暴力 + 哈希
  8. 消息在网络中的传输过程
  9. 影响股票涨跌的基本因素
  10. fastdfs应用场景