Android使用GridLayout绘制自定义日历控件

效果图

思路:就是先设置Gridlayout的行列数,然后往里面放置一定数目的自定义日历按钮控件,最后实现日历逻辑就可以了。

步骤:

第一步:自定义日历控件(初步)

第二步:实现自定义单个日期按钮控件

第三步:将第二步得到的控件动态添加到第一步的布局中,并实现日期逻辑

第四步:编写单个日期点击监听器接口

第一步:自定义日历控件(初步)

<?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="match_parent"
  android:orientation="vertical" >

  <LinearLayout
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:orientation="vertical" >

    <LinearLayout
      android:layout_width="match_parent"
      android:layout_height="55dp"
      android:orientation="vertical"
      android:background="@color/lightGreen" >

      <RelativeLayout
        android:layout_width="match_parent"
        android:layout_height="match_parent" >

        <TextView
          android:id="@+id/textView1"
          android:layout_width="wrap_content"
          android:layout_height="wrap_content"
          android:layout_centerHorizontal="true"
          android:layout_centerVertical="true"
          android:textColor="@color/white"
          android:textSize="9pt"
          android:text="2016年10月" />

        <ImageButton
          android:id="@+id/imageButton1"
          android:layout_width="30dp"
          android:layout_height="20dp"
          android:layout_alignParentLeft="true"
          android:layout_centerVertical="true"
          android:layout_marginLeft="16dp"
          android:background="@drawable/back" />

        <ImageButton
          android:id="@+id/imageButton2"
          android:layout_width="20dp"
          android:layout_height="20dp"
          android:layout_alignTop="@+id/imageButton1"
          android:layout_centerVertical="true"
          android:layout_marginRight="16dp"
          android:layout_toLeftOf="@+id/textView1"
          android:background="@drawable/pre" />

        <ImageButton
          android:id="@+id/imageButton3"
          android:layout_width="18dp"
          android:layout_height="18dp"
          android:layout_centerVertical="true"
          android:layout_alignTop="@+id/textView1"
          android:layout_marginLeft="16dp"
          android:layout_toRightOf="@+id/textView1"
          android:background="@drawable/back11" />

      </RelativeLayout>

    </LinearLayout>

    <LinearLayout
      android:layout_width="match_parent"
      android:layout_height="wrap_content"
      android:orientation="vertical"
      android:layout_marginTop="10dp" >

      <GridLayout
        android:id="@+id/gridLayout01"
        android:layout_width="match_parent"
        android:layout_height="match_parent" >
      <--!自定义的单个日历按钮控件就放在这里-->
      </GridLayout>

    </LinearLayout>

    <ListView
      android:id="@+id/listView1"
      android:layout_width="match_parent"
      android:layout_height="wrap_content" >
    </ListView>

  </LinearLayout>

</LinearLayout>
package com.包名.myCalendarView;

import java.util.Calendar;
import java.util.Date;

import android.content.Context;
import android.graphics.Color;
import android.util.AttributeSet;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.WindowManager;
import android.widget.GridLayout;
import android.widget.ImageButton;
import android.widget.LinearLayout;
import android.widget.TextView;

import com.xuy849.utils.Info;
import com.xuy849.weightapp.R;

public class MyCalendar extends LinearLayout implements OnClickListener{
 Context context;
 TextView tv_yearAndMonth;
 ImageButton ib_pre;
 ImageButton ib_next;
 ImageButton ib_back;
 GridLayout gl_calendar;
 View view;
 CalendarButton buttons[];
 int sideLength;
 String week[] = {"日","一","二","三","四","五","六"};
 Date date;
 Calendar calendar;
 int year,month,day;
 int res;

 public MyCalendar(Context context, AttributeSet attrs, int defStyle) {
 super(context, attrs, defStyle);

 this.context = context;
 // TODO Auto-generated constructor stub

 init();
 }

 public MyCalendar(Context context, AttributeSet attrs) {
 super(context, attrs);

 this.context = context;
 // TODO Auto-generated constructor stub

 init();
 }

 public MyCalendar(Context context) {
 super(context);

 this.context = context;
 // TODO Auto-generated constructor stub

 init();
 }

 private void init(){
 this.view = LayoutInflater.from(context).inflate(R.layout.my_calendar_view, this);

 }
}

效果图:(请无视ListView)

第二步:自定义单个日期按钮控件

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  android:layout_width="wrap_content"
  android:layout_height="wrap_content"
  android:background="@drawable/background01"
  android:orientation="vertical" >

  <LinearLayout
    android:layout_width="wrap_content"
    android:id="@+id/linearLayout01"
    android:layout_height="wrap_content"
    android:background="@color/calendarBackground"
    android:orientation="vertical"
    android:layout_marginBottom="1dp">

    <FrameLayout
      android:layout_width="match_parent"
      android:layout_height="match_parent" >

      <ImageView
        android:id="@+id/imageView1"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:visibility="invisible"
        android:src="@drawable/selected_date" />

      <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:orientation="vertical"
        android:gravity="center">

        <TextView
          android:id="@+id/textView1"
          android:layout_width="wrap_content"
          android:layout_height="wrap_content"
          android:textSize="8pt"
          android:text="20" />

        <TextView
          android:id="@+id/textView2"
          android:layout_width="wrap_content"
          android:layout_height="wrap_content"
          android:layout_marginTop="3dp"
          android:textColor="@color/blue"
          android:visibility="invisible"
          android:text="65.1" />

      </LinearLayout>

    </FrameLayout>

  </LinearLayout>

</LinearLayout>
package com.包名.myCalendarView;

import java.util.Date;

import android.content.Context;
import android.graphics.Color;
import android.util.AttributeSet;
import android.view.LayoutInflater;
import android.view.View;
import android.widget.ImageView;
import android.widget.LinearLayout;
import android.widget.TextView;

import com.xuy849.utils.Info;
import com.xuy849.weightapp.R;

class CalendarButton extends LinearLayout{

 Context context;
 TextView tv_date;
 TextView tv_data;
 ImageView iv_note;
 LinearLayout ll_container;
 View view;
 Date date;

 public CalendarButton(Context context, AttributeSet attrs, int defStyle) {
 super(context, attrs, defStyle);

 this.context = context;
 // TODO Auto-generated constructor stub

 init();
 }

 public CalendarButton(Context context, AttributeSet attrs) {
 super(context, attrs);

 this.context = context;
 // TODO Auto-generated constructor stub

 init();
 }

 public void setTotalEnable(boolean b){
 this.setEnabled(b);
 }

 public CalendarButton(Context context) {
 super(context);

 this.context = context;
 // TODO Auto-generated constructor stub

 init();

 }

 public void setDate(Date date){
 this.date = date;
 }

 public Date getDate(){
 return this.date;
 }

 private void init(){
 view = LayoutInflater.from(context).inflate(R.layout.date_button, this);

 //findViewById
 tv_date =(TextView)view.findViewById(R.id.textView1);
 tv_data = (TextView)view.findViewById(R.id.textView2);
 iv_note = (ImageView)view.findViewById(R.id.imageView1);
 ll_container = (LinearLayout)view.findViewById(R.id.linearLayout01);

 }

 public void setDateText(String text){
 tv_date.setText(text);
 }

 public void setDateTextColor(int color){
 tv_date.setTextColor(color);
 }
 }

}

效果图:

第三步:将第二步得到的控件在java代码中添加到第一步的布局中,并添加相关逻辑

package com.包名.myCalendarView;

import java.util.Calendar;
import java.util.Date;

import android.content.Context;
import android.graphics.Color;
import android.util.AttributeSet;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.WindowManager;
import android.widget.GridLayout;
import android.widget.ImageButton;
import android.widget.LinearLayout;
import android.widget.TextView;

import com.xuy849.utils.Info;
import com.xuy849.weightapp.R;

public class MyCalendar extends LinearLayout implements OnClickListener{
 Context context;
 TextView tv_yearAndMonth;
 ImageButton ib_pre;
 ImageButton ib_next;
 ImageButton ib_back;
 GridLayout gl_calendar;
 View view;
 CalendarButton buttons[];
 int sideLength;
 String week[] = {"日","一","二","三","四","五","六"};
 Date date;
 Calendar calendar;
 int year,month,day;
 int res;

 public MyCalendar(Context context, AttributeSet attrs, int defStyle) {
 super(context, attrs, defStyle);

 this.context = context;
 // TODO Auto-generated constructor stub

 init();
 }

 public MyCalendar(Context context, AttributeSet attrs) {
 super(context, attrs);

 this.context = context;
 // TODO Auto-generated constructor stub

 init();
 }

 public MyCalendar(Context context) {
 super(context);

 this.context = context;
 // TODO Auto-generated constructor stub

 init();
 }

 private void init(){
 this.view = LayoutInflater.from(context).inflate(R.layout.my_calendar_view, this);

 WindowManager ww = (WindowManager)getContext().getSystemService(Context.WINDOW_SERVICE);
 sideLength = ww.getDefaultDisplay().getWidth()/Info.COLUMN_COUNT;

 //
 myFindViewById();

 //初始化日历按钮布局
 initCalendarLayout();

 //初始化按钮点击事件
 initButton();

 }

 private void initButton(){
 ib_pre.setOnClickListener(this);
 ib_next.setOnClickListener(this);
 ib_back.setOnClickListener(this);
 }

 private void myFindViewById(){
 tv_yearAndMonth = (TextView)view.findViewById(R.id.textView1);
 ib_pre = (ImageButton)view.findViewById(R.id.imageButton2);
 ib_next = (ImageButton)view.findViewById(R.id.imageButton3);
 ib_back = (ImageButton)view.findViewById(R.id.imageButton1);
 gl_calendar =(GridLayout)view.findViewById(R.id.gridLayout01);

 //
 buttons = new CalendarButton[Info.COLUMN_COUNT*Info.ROW_COUNT];

 }

  //根据传递过来的calendar,绘制当月的日历视图
  private void initCalendar(Calendar calendar){
 int year = calendar.get(Calendar.YEAR);
 int month = (calendar.get(Calendar.MONTH)+1);
 int date = calendar.get(Calendar.DATE);

 //设置标题
 String todayStr = String.format("%04d年%02d月", calendar.get(Calendar.YEAR),(calendar.get(Calendar.MONTH)+1));
 tv_yearAndMonth.setText(todayStr);

 //
 calendar.set(Calendar.DAY_OF_MONTH, 1);
 int currentMonthFirstDateInWeek = calendar.get(Calendar.DAY_OF_WEEK)-1;
 calendar.set(Calendar.DAY_OF_MONTH, date);
 int currentMonthDaysSum = calendar.getActualMaximum(Calendar.DAY_OF_MONTH);

 calendar.roll(Calendar.MONTH, -1);
 int lastMonthDaysSum = calendar.getActualMaximum(Calendar.DAY_OF_MONTH);
 int i;
 calendar.roll(Calendar.MONTH, 1);

 Log.w("月", currentMonthDaysSum+"天");
 Log.w("上月", lastMonthDaysSum +"天");

 /*
  * 设置日期
  */
 //设置本月
 Log.w("currentMonthFirstDateInWeek",currentMonthFirstDateInWeek+"");
 Log.w("currentMonthFirstDateInWeek%7111",currentMonthDaysSum+currentMonthFirstDateInWeek%7+"");

 for(i=currentMonthFirstDateInWeek%7+7;i<=(currentMonthDaysSum+currentMonthFirstDateInWeek%7+7-1)&&i<Info.COLUMN_COUNT*Info.ROW_COUNT;i++){
  buttons[i].changeToState(Info.STATE_NORMAL_NORECORED);
  buttons[i].setDateText((i-currentMonthFirstDateInWeek%7-7+1)+"");
  buttons[i].setDateTextColor(Color.BLACK);
  buttons[i].setEnabled(true);
  buttons[i].setDate(new Date(year,month,i-(currentMonthFirstDateInWeek%7+7)+1));
 }

 //设置上一个月
 for(i=7;i<(currentMonthFirstDateInWeek%7+7)&&i<Info.COLUMN_COUNT*Info.ROW_COUNT;i++){
  buttons[i].changeToState(Info.STATE_NORMAL_NORECORED);
  buttons[i].setDateTextColor(Color.GRAY);
  buttons[i].setDateText(lastMonthDaysSum-(currentMonthFirstDateInWeek%7-i%7)+1+"");
  buttons[i].setEnabled(false);
 }

 //设置下一个月
 for(i =currentMonthDaysSum+currentMonthFirstDateInWeek%7+7;i<Info.ROW_COUNT*Info.COLUMN_COUNT;i++){
  buttons[i].changeToState(Info.STATE_NORMAL_NORECORED);
  buttons[i].setDateTextColor(Color.GRAY);
  buttons[i].setDateText((i-(currentMonthDaysSum+currentMonthFirstDateInWeek%7+7)+1)+"");
  buttons[i].setEnabled(false);
 }

 //设置当天
 if(year==this.year&&month==this.month&&day==this.day)
  buttons[date+currentMonthFirstDateInWeek%7+7-1].changeToState(Info.STATE_TODAY_NORECORED);

 //初始化界面
 if(res!=0)
  initLayout(res);

 }

 private void initCalendarLayout(){
 //设置行数,列数
 gl_calendar.setRowCount(Info.ROW_COUNT);
 gl_calendar.setColumnCount(Info.COLUMN_COUNT);

 /*
  * 添加按钮到布局
  */
 int i;
 int sum = Info.ROW_COUNT*Info.COLUMN_COUNT;

 //设置星期
 for(i=0;i<Info.COLUMN_COUNT;i++){
  buttons[i] = new CalendarButton(context);
  buttons[i].setDateText(week[i]);
  buttons[i].setLayoutParams(new LayoutParams(sideLength, sideLength));
  gl_calendar.addView(buttons[i], i);
  buttons[i].setEnabled(false);
 }

 for(i = Info.COLUMN_COUNT;i<sum;i++){
  buttons[i] = new CalendarButton(context);
  buttons[i].setDateText("55");
  buttons[i].setLayoutParams(new LayoutParams(sideLength, sideLength));
  gl_calendar.addView(buttons[i], i);
 }

 //根据当月设置情况
 calendar = Calendar.getInstance();
 year = calendar.get(Calendar.YEAR);
 month = calendar.get(Calendar.MONTH)+1;
 day = calendar.get(Calendar.DATE);
 initCalendar(calendar);

 }

 @Override
 public void onClick(View v) {
 // TODO Auto-generated method stub
 switch(v.getId()){

 //上一个月
 case R.id.imageButton2:{
  calendar.roll(Calendar.MONTH, -1);
  initCalendar(calendar);
  break;
 }

 //下一个月
 case R.id.imageButton3:{
  calendar.roll(Calendar.MONTH, 1);
  initCalendar(calendar);
  break;
 }
 }
 }

 public void setOnClickButtonListener(OnClickListener l,int index){
 buttons[index].setOnClickListener(l);
 }

 public Date getDate(int index){
 return buttons[index].getDate();
 }

 public void setData(float f,int index){
 buttons[index].tv_data.setText(String.format("%.1f", f));
 buttons[index].tv_data.setVisibility(TextView.VISIBLE);
 }

 public void initLayout(int res){
 switch(res){
 case Info.VIEW_WEIHT:{

  break;
 }

 default:{
  break;
 }
 }
 }

 public void setRes(int res){
 this.res = res;
 }
}

第四步:编写单个日期点击监听器接口

在第三步中添加方法:

//设置下标是index的日期按钮的点击事件监听器
public void setOnClickButtonListener(OnClickListener l,int index){
 buttons[index].setOnClickListener(l);
 }

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • android 开发教程之日历项目实践(一)

    前言:决定开始学习 Android 平台下的软件开发,以日历作为实践项目,进行一周后,基本完成. 为了总结及笔记,并给有需要的朋友借鉴,开始整理本教程. 开始之前: 在编写程序之前,需要进行项目设计,因为是练习项目,主要是确定软件 UI 界面,这是已经完成的屏幕截图: 对这个画面,进一步作分解: 这里总共分解为三个 View 文件: 1:activity_main.xml 作为启动的主画面,新建项目时,首先生成. 2:view_calendar_table.xml 定义月历视图,头部固定,其它行

  • Android自定义控件实现可多选课程日历CalendarView

    可多选课程日历CalendarView的效果图 开发环境 IDE版本:AndroidStudio2.0 物理机版本:Win7旗舰版(64位) 前言 最近的项目中用到了一个课程选择的日历View,于是在网上搜了搜自定义日历View,发现基本上都是单选的,不能够满足项目中的需求.于是自己重新造了个轮子,写了个可以被多选的自定义日历View.最后面会给出GitHub地址. 代码实现 package widget; import android.content.Context; import andro

  • Android自定义日历Calender代码实现

    产品要做签到功能,签到功能要基于一个日历来进行,所以就根据 要求自定义了一个日历 自定义控件相信做android都知道: (1)首先创建一个类,继承一个容器类或者是一个控件 (2)然后就是你需要设置的属性等的,在attrs文件夹中 (3)然后就是在类里边进行属性的设置以及布局等等功能的添加 其实自定义一个日历问题都不多,很多人都会想到通过一个gridView然后填充就可以,确实是这样,主要是在显示每个月的第一天的位置以及每个月显示多少天有点绕. 思路:通过判断当前星期几然后进行日历的填充,但是填

  • Android自定义可标记日历效果

    先直接看看效果吧 初始情况 点击一个作为标记 再次点击后删除 3.这里还要感谢前辈的代码作为参考,毕竟以前也没有写过关于日历方面的东西,别人确实写得不错,我在原基础上加入了数据库操作等补充,以完成自己实际需求,作为尊重首先给出原作者的连接 就是这里–>Android自定义控件实现可多选课程日历CalendarView 4.然后贴出来关于数据库操作的代码,给大家作为参考 DatabaseHelper .java 这是关于简单数据库操作的部分 package com.xugongming38.edi

  • Android实现自定义日历

    自定义日历控件,支持旧历.节气.日期标注.点击操作 (参考网络上的日历控件改写) 注:将下面的四张资源图片拷贝到所建包的下一个image目录中,如Calendar.java 所在包为 cc.util.android.view,则需要再创建一个包cc.util.android.view.image 然后将图片拷贝进去 /****************从此出开始将代码拷贝到一个文件中*******************/ package cc.util.android.view; import

  • Android自定义日历控件实例详解

    为什么要自定义控件 有时,原生控件不能满足我们对于外观和功能的需求,这时候可以自定义控件来定制外观或功能:有时,原生控件可以通过复杂的编码实现想要的功能,这时候可以自定义控件来提高代码的可复用性. 如何自定义控件 下面我通过我在github上开源的Android-CalendarView项目为例,来介绍一下自定义控件的方法.该项目中自定义的控件类名是CalendarView.这个自定义控件覆盖了一些自定义控件时常需要重写的一些方法. 构造函数 为了支持本控件既能使用xml布局文件声明,也可在ja

  • Android实现日历控件示例代码

    做的是一个酒店的项目,可以选择入住和离开的日期.声明为了省事在网上找的资料,自己修改的逻辑,希望对需要的朋友有帮助.喜欢的给个好评.谢谢啦!祝生活愉快! 先上图 第一步,搭建布局xml <ScrollView xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" android:layout_w

  • java制作android 日历代码分享

    代码很简单,就不多废话了 复制代码 代码如下: //读取日历事件     public static void getCalendarInfo(Activity activity,String tag){         String[] projection = new String[]{CalendarContract.Events._ID,CalendarContract.Events.TITLE};         ContentResolver cr = activity.getCon

  • android 开发教程之日历项目实践(二)

    一.创建 Android Project 在新建对话框中输入 App 属性,SDK版本全部选最新的,不作版本兼容.主题选择 Holo Dark. 下一步,使用默认设置 下一步,使用默认设置 下一步,使用默认配置 下一步,使用默认设置 创建完成后的初始画面 在上面的步骤中,我们选择了创建 MainActivity,ADT 帮我们在 src 目录下生成了 MainActivity.java 文件,在 res/layout/ 目录下生成了 activity_main.xml 文件,并在编辑窗口打开,如

  • Android使用GridView实现日历的简单功能

    简单的日历实现,只是显示了每一个月,没有显示当天和记事这些功能 主要是计算月初是周几,月末是周几,然后相应的显示上一月多少天和下一月多少天. 先看一下关于日期的用到的几个工具类 /** * 获取该月天数 */ public static int getCurrentMonthDay(long millSec) { Calendar calendar = Calendar.getInstance(); calendar.setTimeInMillis(millSec); calendar.set(

随机推荐