前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >安卓的日历_公认不卡的安卓手机

安卓的日历_公认不卡的安卓手机

作者头像
全栈程序员站长
发布于 2022-08-03 01:42:44
发布于 2022-08-03 01:42:44
5.1K00
代码可运行
举报
运行总次数:0
代码可运行

大家好,又见面了,我是你们的朋友全栈君。

最近写了一款日历,包含周日历、月日历以及滑动切换视图,先上效果图:

代码已上传到github:https://github.com/yannecer/NCalendar 项目主要用到了自定义View,ViewPager,RecyclerView和NestedScrollingParent。

本篇文章主要说一下月日历数据、月视图绘制以及点击日期的实现。

数据

数据部分,网上能找到比较完整的工具类,主要是根据本月和上月的天数以及本月第一天是周几来计算。

首先计算上月日期: 由本月的第一天是周几和上个月的天数,得出上月的日期的显示

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
int temp = lastMonthDays - firstDayOfWeek + 1;//上个月的天数减去本月第一天周几再加上1

再计算本月日期:本月内的数据根据该月的天数跑循环。

再计算下月计算上月日期的显示: 下月的天数显示可以看本月最后一天是周几,根据距离一周最后一天的间隔天数,从1开始直接加上就可以了。

这里要分情况了,有的月份跨5个周,有的月份能跨6个周。计算上没有区别,但是显示的时候会有区别,为了简单,统一成6周,共42个元素,一月多余的用下月日期补充。日期计算肯定使用joda-time了,天数、月份、年份计算都非常简单,有一点,这个库每周是周一开始的,周日历要注意一下。

一月的数据:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
List<Integer> date = new ArrayList();
 int j = 1;
 for (int i = 0; i < 42; i++) {
     if (i < firstDayOfWeek) { // 前一个月
         int temp = lastMonthDays - firstDayOfWeek + 1;
         date .add(temp + i);
      } else if (i < days + firstDayOfWeek) { // 本月
        int temp = i - firstDayOfWeek + 1;
          date .add(temp)
       } else { // 下一个yue 
          date .add();
         j++;
       }        
  }

这里简化了操作,项目中我把每个数据都转化成了joda-time中的DateTime对象,方便后面操作。 数据有了,接着就是绘制这些数据。

MonthView

MonthView继承于View,重写onDraw(canva)方法。 首先在构造方法中根据颜色和字体大小初始化画笔:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
  mSorlarPaint = getPaint(mSolarTextColor, mSolarTextSize);

    .....
  private Paint getPaint(int paintColor, float paintSize) {
        Paint paint = new Paint();
        paint.setColor(paintColor);
        paint.setTextSize(paintSize);
        paint.setAntiAlias(true);
        paint.setTextAlign(Paint.Align.CENTER);
        return paint;
    }

接着就是在onDraw(canva)方法中绘制。 我们先考虑一下我们都需要做哪些事情。需要绘制公历、农历、小圆点、选中的圆环包括后面的点击操作,这些元素确定位置都需要一个矩形(Rect),那么就可以先在这个View里面绘制42个矩形。四个点确定一个矩形,可以在纸上画一下大致的图案,大致画个一两行矩形,应该就找到规律了,感觉有点像以前上学时做的找规律的数学题。

6行7列的一个矩形阵

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        mWidth = getWidth();//view的宽度
        mHeight = getHeight();//view的高度

        for (int i = 0; i < 6; i++) {
            for (int j = 0; j < 7; j++) {
                Rect rect = new Rect(j * mWidth / 7, i * mHeight / 6, j * mWidth / 7 + mWidth / 7, i * mHeight / 6 + mHeight / 6);
                canvas.drawRect(rect,mSorlarPaint);
            }
        }
    }

有了这42个矩形,我们做后面的事情就简单了。

绘制文字

绘制文字 canvas.drawText()会发现,可能会出现文字不在矩形的中心,解决办法参看这篇博客,Android Canvas drawText实现中文垂直居中

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
Paint.FontMetricsInt fontMetrics = mSorlarPaint.getFontMetricsInt();
int baseline = (rect.bottom + rect.top - fontMetrics.bottom - fontMetrics.top) / 2;
canvas.drawText(dateTime.getDayOfMonth() + "", rect.centerX(), baseline, mSorlarPaint);

我们需要在绘制的循环里面要判断这些内容:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
1、是不是本月的数据(用颜色区分本月和其他月的数据)
2、是不是今天
3、有没有选中的日期
4、显示不显示农历

其中今天和选中的日期用圆环表示,就需要在当天和选中的日期的矩形中绘制圆环。 已今天为例:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
 //是今天,且是当月的今天才绘制今天的标识  
 if (Utils.isToday(dateTime) && Utils.isEqualsMonth(dateTime, mInitialDateTime)) {
     mSorlarPaint.setColor(mSelectCircleColor);//画笔设置选中的颜色
     int radius = Math.min(Math.min(rect.width() / 2, rect.height() / 2), mSelectCircleRadius);//圆环半径取矩形宽、高和设置半径的最小值
     canvas.drawCircle(rect.centerX(), rect.centerY(), radius, mSorlarPaint);
     mSorlarPaint.setColor(Color.WHITE);//当天的文本设置成白色
     canvas.drawText(dateTime.getDayOfMonth() + "", rect.centerX(), baseline, mSorlarPaint);
  }

完整的onDraw(Canvas canvas)里面的代码

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        mWidth = getWidth();
        mHeight = getHeight();
        mRectList.clear();
        //6行7列
        for (int i = 0; i < 6; i++) { 
   
            for (int j = 0; j < 7; j++) { 
   
                Rect rect = new Rect(j * mWidth / 7, i * mHeight / 6, j * mWidth / 7 + mWidth / 7, i * mHeight / 6 + mHeight / 6);
                mRectList.add(rect);
                DateTime dateTime = monthDateTimeList.get(i * 7 + j);
                Paint.FontMetricsInt fontMetrics = mSorlarPaint.getFontMetricsInt();
                int baseline = (rect.bottom + rect.top - fontMetrics.bottom - fontMetrics.top) / 2;
                //判断是不是当月,当月和上下月的颜色不同
                if (Utils.isEqualsMonth(dateTime, mInitialDateTime)) {
                    //当天和选中的日期不绘制农历
                    if (Utils.isToday(dateTime)) {
                        mSorlarPaint.setColor(mSelectCircleColor);
                        int radius = Math.min(Math.min(rect.width() / 2, rect.height() / 2), mSelectCircleRadius);
                        canvas.drawCircle(rect.centerX(), rect.centerY(), radius, mSorlarPaint);
                        mSorlarPaint.setColor(Color.WHITE);
                        canvas.drawText(dateTime.getDayOfMonth() + "", rect.centerX(), baseline, mSorlarPaint);
                        //选中日期不为null绘制空心圆
                    } else if (mSelectDateTime != null && dateTime.toLocalDate().equals(mSelectDateTime.toLocalDate())) {
                        mSorlarPaint.setColor(mSelectCircleColor);
                        int radius = Math.min(Math.min(rect.width() / 2, rect.height() / 2), mSelectCircleRadius);
                        canvas.drawCircle(rect.centerX(), rect.centerY(), radius, mSorlarPaint);
                        mSorlarPaint.setColor(mHollowCircleColor);
                        canvas.drawCircle(rect.centerX(), rect.centerY(), radius - mHollowCircleStroke, mSorlarPaint);
                        mSorlarPaint.setColor(mSolarTextColor);
                        canvas.drawText(dateTime.getDayOfMonth() + "", rect.centerX(), baseline, mSorlarPaint);
                    } else {
                        mSorlarPaint.setColor(mSolarTextColor);
                        canvas.drawText(dateTime.getDayOfMonth() + "", rect.centerX(), baseline, mSorlarPaint);
                        drawLunar(canvas, rect, mLunarTextColor, i, j);
                    }
                } else {
                    mSorlarPaint.setColor(mHintColor);
                    canvas.drawText(dateTime.getDayOfMonth() + "", rect.centerX(), baseline, mSorlarPaint);
                    drawLunar(canvas, rect, mHintColor, i, j);
                }
                    //绘制提示的小圆点
                    if (mPointList.contains(dateTime.toLocalDate().toString())) {
                        mSorlarPaint.setColor(mPointColor);
                        canvas.drawCircle(rect.centerX(), rect.bottom-mPointSize, mPointSize, mSorlarPaint);
                    }
            }
        }
    }

......
    //绘制农历
    private void drawLunar(Canvas canvas, Rect rect, int color, int i, int j) {
        if (isShowLunar) {
            mLunarPaint.setColor(color);
            String lunar = lunarList.get(i * 7 + j);
            canvas.drawText(lunar, rect.centerX(), rect.bottom - Utils.dp2px(getContext(), 5), mLunarPaint);
        }
    }

里面的一些工具类可参见github上的项目:https://github.com/yannecer/NCalendar

点击事件

点击操作使用了GestureDetector,这个类里面已经定义好了单级,双击,长按等操作,只需要我们重写相应的方法就可以,不用我们在去定义一个点击操作了。 重写MonthViewonTouchEvent(MotionEvent event)方法,交给GestureDetector处理

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
 @Override
 public boolean onTouchEvent(MotionEvent event) {
    return mGestureDetector.onTouchEvent(event);
 }

触摸事件交给GestureDetector,当发生单击时,循环刚才绘制文本时的矩形,根据用户点击的XY坐标值判断是在哪个矩形内,我们就知道用户点击的是哪个日期了。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
    private GestureDetector mGestureDetector = new GestureDetector(getContext(), new GestureDetector.SimpleOnGestureListener() {
        @Override
        public boolean onDown(MotionEvent e) {
            return true;
        }
        @Override
        public boolean onSingleTapUp(MotionEvent e) {
            for (int i = 0; i < mRectList.size(); i++) {
                Rect rect = mRectList.get(i);
                //判断
                if (rect.contains((int) e.getX(), (int) e.getY())) {
                    DateTime selectDateTime = monthDateTimeList.get(i);
                    //点击的是上个月
                    if (Utils.isLastMonth(selectDateTime, mInitialDateTime)) {
                        onClickMonthViewListener.onClickLastMonth(selectDateTime);
                    } else if (Utils.isNextMonth(selectDateTime, mInitialDateTime)) {
                        //点击的是下个月
                        onClickMonthViewListener.onClickNextMonth(selectDateTime);
                    } else {
                         //点击的是本月
                        onClickMonthViewListener.onClickCurrentMonth(selectDateTime);
                    }
                    break;
                }
            }
            return true;
        }
    });

里面写了一些回调,方便在ViewPager中跳转到相应的月份。剩下的操作放到了ViewPager中完成,如果不是本月就跳转再设置选中的日期,如果是本月,就直接设置选中的日期:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
    @Override
    public void onClickCurrentMonth(DateTime dateTime) {
        doClickEvent(dateTime, getCurrentItem());
    }

    @Override
    public void onClickLastMonth(DateTime dateTime) {
        int currentItem = getCurrentItem() - 1;
        doClickEvent(dateTime, currentItem);
    }

    @Override
    public void onClickNextMonth(DateTime dateTime) {
        int currentItem = getCurrentItem() + 1;
        doClickEvent(dateTime, currentItem);
    }

    ........
    //处理点击
    private void doClickEvent(DateTime dateTime, int currentItem) {
        MonthCalendar.this.setCurrentItem(currentItem);
        MonthView monthView = (MonthView) calendarAdapter.getCalendarViews().get(currentItem);
        monthView.setSelectDateTime(dateTime);
        if (onClickMonthCalendarListener != null) {
            onClickMonthCalendarListener.onClickMonthCalendar(dateTime);
        }

    }

doClickEvent(DateTime dateTime, int currentItem)方法中,得到当前的MonthView ,设置选中日期monthView.setSelectDateTime(dateTime); 而在setSelectDateTime(DateTime dateTime)中就是赋值和重绘页面:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
    public void setSelectDateTime(DateTime dateTime) {
        this.mSelectDateTime = dateTime;
        invalidate();
    }

这样在onDraw(Canvas canvas)mSelectDateTime!=null,就会绘制选中的圆环了。

MonthView没有重写onMeasure(int widthMeasureSpec, int heightMeasureSpec)方法,是因为这个MonthView是放在继承自ViewPagerMonthCalendar中使用的,只需在布局文件中设置MonthCalendarlayout_widthlayout_height即可:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
 <com.necer.ncalendar.calendar.MonthCalendar
        android:id="@+id/monthCalendar"
        android:layout_width="match_parent"
        android:layout_height="240dp"
        android:background="@color/white"
        app:selectCircleColor= "@android:color/holo_red_light"
        app:pointcolor="#00c8aa"
        app:pointSize="1dp"
        app:solarTextSize= "15sp"/>

发布者:全栈程序员栈长,转载请注明出处:https://javaforall.cn/125033.html原文链接:https://javaforall.cn

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2022年4月1,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
Spring,hibernate,struts的面试笔试题及答案
Hibernate工作原理及为什么要用? 原理: 读取并解析配置文件 读取并解析映射信息,创建SessionFactory 打开Sesssion 创建事务Transation 持久化操作 提交事务 关闭Session 关闭SesstionFactory 为什么要用: 对JDBC访问数据库的代码做了封装,大大简化了数据访问层繁琐的重复性代码。 Hibernate是一个基于JDBC的主流持久化框架,是一个优秀的ORM实现。他很
WindWant
2020/09/11
7660
JavaWeb(2)
JSP是Servlet技术的扩展,本质上就是Servlet的简易方式。JSP编译后是“类servlet”。Servlet和JSP最主要的不同点在于,Servlet的应用逻辑是在Java文件中,并且完全从表示层中的HTML里分离开来。而JSP的情况是Java和HTML可以组合成一个扩展名为.jsp的文件。JSP侧重于视图,Servlet主要用于控制逻辑。在struts框架中,JSP位于MVC设计模式的视图层,而Servlet位于控制层.
一朵灼灼华
2022/08/05
6380
Java面试分享(题目+答案)
1.说下Struts的设计模式 MVC模式: web应用程序启动时 就会加载并初始化ActionServler。 用户提交表单时, 一个配置好的ActionForm对象被创建, 并被填入表单相应的数据, ActionServler根据Struts-config.xml文件 配置好的设置决定是否需要表单验证, 如果需要就调用ActionForm的Validate() 验证后选择将请求发送到哪个Action, 如果Action不存在, ActionServlet会先创建这个对象, 然后调用Action的exe
Java高级架构
2018/12/18
1.3K0
[ Java面试题 ]框架篇二
1.Hibernate工作原理及为什么要使用Hibernate? 工作原理:     1.读取并解析配置文件     2.读取并解析映射信息,创建SessionFactory     3.打开Session     4.创建事务Transation     5.持久化操作     6.提交事务     7.关闭Session     8.关闭SesstionFactory 为什么要使用Hibernate(即它的优点):     1. 对JDBC访问数据库的代码做了封装,大大简化了数据访问层繁琐的重复性代码。
Kevin_Zhang
2018/07/05
5800
[ Java面试题 ] 框架篇
  1. struts是一个按MVC模式设计的Web层框架,其实它就是一个Servlet,这个Servlet名为ActionServlet,或是ActionServlet的子类。我们可以在web.xml文件中将符合某种特征的所有请求交给这个Servlet处理,这个Servlet再参照一个配置文件将各个请求分别分配给不同的action去处理。
Kevin_Zhang
2018/09/20
7590
struts、hibernate、spring、 mybatis、 spring boot 等面试题汇总
1. struts是一个按MVC模式设计的Web层框架,其实它就是一个大大的servlet,这个Servlet名为ActionServlet,或是ActionServlet的子类。我们可以在web.xml文件中将符合某种特征的所有请求交给这个Servlet处理,这个Servlet再参照一个配置文件(通常为/WEB-INF/struts-config.xml)将各个请求分别分配给不同的action去处理。
张哥编程
2024/12/17
1220
进阶spring/Hibernate*框架精选面试题
,今天给大家带来的是一些Java框架的面试题,这些面试题涵盖Hibernate框架,spring框架等,建议收藏
框架师
2019/09/19
6570
进阶spring/Hibernate*框架精选面试题
SSH框架总结
首先,SSH不是一个框架,而是多个框架(struts+spring+hibernate)的集成,是目前较流行的一种Web应用程序开源集成框架,用于构建灵活、易于扩展的多层Web应用程序。
哲洛不闹
2018/09/19
1K0
SSH框架总结
SSH到底是什么?(彻底搞懂SSH,SSM)
什么是SSH? SSH在J2EE项目中表示了3种框架,即 Spring + Struts +Hibernate。 Struts对Model,View和Controller都提供了对应的组件。 Sp
葆宁
2019/04/19
2.4K0
SSH到底是什么?(彻底搞懂SSH,SSM)
SSH框架(三) 常用WEB框架Struts1、Struts2和Spring MVC三者的区别
SSH框架(三) 常用WEB框架Struts1、Struts2和Spring MVC三者的区别
Java架构师必看
2021/05/27
1.1K0
struts2和struts1认识
Struts 2框架本身可以大致分3部分:核心控制器FilterDispatcher、业务总监Action与用户实现企业业务逻辑组件。
全栈程序员站长
2022/07/06
6180
java三大框架介绍
常听人提起三大框架,关于三大框架,做了如下了解: 三大框架:Struts+Hibernate+Spring java三大框架主要用来做WEN应用。 Struts主要负责表示层的显示 Spring利用它的IOC和AOP来处理控制业务(负责对数据库的操作) Hibernate主要是数据持久化到数据库 再用jsp的servlet做网页开发的时候有个web.xml的映射文件,里面有一个mapping的标签就是用来做文件映射的。当你在浏览器上输入URL得知的时候,文件就会根据你写的名称对应到一个JAVA文件
程序员互动联盟
2018/03/16
9460
【SSH进阶之路】Struts + Spring + Hibernate 进阶开端(一)
Long Long ago,就听说过SSH,起初还以为是一个东东,具体内容更是不详,总觉得高端大气上档次,经过学习之后才发现,不仅仅是高大上,更是低调奢华有内涵,经过一段时间的研究和学习SSH框架的基本原理与思想,总算接地气了。作为初学者,有点小小收获,想通过以博文的形式和大家一起分享,共同进步,在更新博文的过程中难免有认识不足的地方,还请各位大牛提出宝贵的建议,对于好的建议一定虚心接受,认真学习。
程序猿小亮
2021/01/29
4590
SSH 学习杂记
Web 开发是一个很有意思的事情。Struts+Spring+Hibernate 作为一种当今流行的开发模式,我很荣幸地在一周左右的时间里,疯狂地边学边琢磨了一番,甚有感触。
四火
2022/07/15
6420
SSH 学习杂记
SSH学习(一)—— 基础概念篇
最近有个小面试需要复习以前用到的SSH框架,忘得差不多了当然当时也差不多就是不会的状态,现在花了三四天的时间进行一个简单的学习,总结一些概念性的东西放在这儿。
全栈程序员站长
2022/09/12
6820
Java面试宝典4.0版
Jdk 是 java 开发人员在开发过程使用的软件开发包,他提供了 java 的开发环境和运行环境 JRE 是 Java Runtime Enviroment 是指 Java 的运行环境
全栈程序员站长
2022/11/05
1.2K0
Java理论知识及面试题
1、客户端发出一个http请求给web服务器,web服务器对http请求进行解析,如果匹配DispatcherServlet的请求映射路径(在web.xml中指定),web容器将请求转交给DispatcherServlet.
小语雀网
2022/05/06
5600
Java企业面试——SSH框架
SSH框架阶段SSH的优缺点,使用场景? Hibernate优点: (1) 对象/关系数据库映射(ORM) 它使用时只需要操纵对象,使开发更对象化,抛弃了数据库中心的思想,完全的面向对象思想 (2) 透明持久化(persistent) 带有持久化状态的、具有业务功能的单线程对象,此对象生存期很短。这些对象可能是普通的JavaBeans/POJO,这个对象没有实现第三方框架或者接口,唯一特殊的是他们正与(仅仅一个)Session相关联。一旦这个Session被关闭,这些对象就会脱离持久化状态,这样就可被应用
Java帮帮
2018/03/16
1K0
Java企业面试——SSH框架
Spring工作原理
      内部最核心的就是IOC了,动态注入,让一个对象的创建不用new了,可以自动的生产,这其实就是利用java里的反射,反射其实就是在运行时动态的去创建、调用对象,Spring就是在运行时,跟xml Spring的配置文件来动态的创建对象,和调用对象里的方法的 。
Java架构师历程
2018/09/26
2.2K0
Struts 1基础入门
在上一阶段的学习中,我们通过使用MVC模式进行了多个项目的开发。在MVC模式中,JavaBean实现业务逻辑,Servlet实现流程控制,JSP负责数据显示,这样的设计使程序的层次更加清晰,项目分工更加明确,代码耦合度也大大降低。但是在实际应用中,MVC模式也逐渐暴露出了诸多缺陷:开发效率低、结构不一致、难以重用等等。
张哥编程
2024/12/17
1520
Struts 1基础入门
相关推荐
Spring,hibernate,struts的面试笔试题及答案
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档