首页 > 系统 > Android > 正文

Android使用GridView实现日历的方法

2019-12-12 02:07:39
字体:
来源:转载
供稿:网友

在开发中可能会遇到某些情况下需要用到日历的功能,并且还要在日历上加标签什么的,最重要的就是android自带的日历由于各个系统版本不同导致日历的样式也不同,这样就会导致使用起来比较麻烦..而且在日历中加标签也不好实现...所以很多时候日历都是自己去实现的...由于自定义日历会比较麻烦...这里就教大家使用GridView来实现,主要是我们比较熟悉这个控件...到时候也可以根据自己的情况进行封装为自定义View

下面就先看看效果图.由于是从项目中抽取出来的,某些地方定制性比较强, 可以根据需求自行修改

效果图

图中的红点就是标签,蓝色背景就是选中的意思.

下面开始撸代码:

先上核心的GridView的适配器:

CalendarAdapter.java

/** * 日历gridview中的每一个item显示的textview */public class CalendarAdapter extends BaseAdapter {  private static String TAG = "CalendarAdapter";  private boolean isLeapyear = false; //是否为闰年   private int daysOfMonth = 0;   //某月的天数   private int dayOfWeek = 0;    //具体某一天是星期几   private int lastDaysOfMonth = 0; //上一个月的总天数   private Context context;  private String[] dayNumber = new String[42]; //一个gridview中的日期存入此数组中   private SpecialCalendar sc = null;  private int currentYear = 0;  private int currentMonth = 0;  /**   * 当前选中的日期位置   */  private int currentFlag = -1;  /**   * 当前选中天的字符串 例:20170830   */  private String currentDayStr;  private SimpleDateFormat sdf = new SimpleDateFormat("yyyy-M-d");  private Set<Integer> schDateTagFlag = new ArraySet<>(); //存储当月所有的日程日期(标签)  private String showYear = "";  //用于在头部显示的年份  private String showMonth = ""; //用于在头部显示的月份   private String animalsYear = "";  private String leapMonth = "";  //闰哪一个月   private Set<String> mSet = null;  /**   * 距离当前月的差(上一个月-1,当前月0,下一个月+1)   */  private int jumpMonth = 0;  public CalendarAdapter(Context context, int year, int month, String currentDayStr) {    this.context = context;    sc = new SpecialCalendar();    currentYear = year;    currentMonth = month; //得到跳转到的月份    this.currentDayStr = currentDayStr;    getCalendar(currentYear, currentMonth);  }  @Override  public int getCount() {    return dayNumber.length;  }  @Override  public Object getItem(int position) {    return position;  }  @Override  public long getItemId(int position) {    return position;  }  @Override  public View getView(int position, View convertView, ViewGroup parent) {    ViewHolder myViewHolder = null;    if (convertView == null || convertView.getTag() == null) {      convertView = LayoutInflater.from(context).inflate(R.layout.item_select_time, null);      myViewHolder = new ViewHolder(convertView);      convertView.setTag(myViewHolder);    } else {      myViewHolder = (ViewHolder) convertView.getTag();    }    myViewHolder.mIdTvItemSelectTimeDay.setText(dayNumber[position]);    myViewHolder.mIdTvItemSelectTimeDay.setTextColor(Color.GRAY);//不是当前月为灰    if (position < daysOfMonth + dayOfWeek && position >= dayOfWeek) {      // 当前月信息显示       myViewHolder.mIdTvItemSelectTimeDay.setTextColor(Color.BLACK);// 当月字体设黑      myViewHolder.mIdTvItemSelectTimeDay.setTag(true);// 当月字体设黑    }else {      myViewHolder.mIdTvItemSelectTimeDay.setTag(false);// 当月字体设黑    }    if (currentFlag != -1 && currentFlag == position) {      //设置当天的背景      myViewHolder.mIdTvItemSelectTimeDay.setBackgroundResource(R.color.mainMenu);      myViewHolder.mIdTvItemSelectTimeDay.setTextColor(Color.WHITE);    } else {      myViewHolder.mIdTvItemSelectTimeDay.setBackgroundColor(0);    }    //显示小圆点    if (schDateTagFlag != null && schDateTagFlag.size() > 0) {      if (schDateTagFlag.contains(position)) {        if (myViewHolder.mIdImgItemSelectTimeLogo.getVisibility()!=View.VISIBLE) {          myViewHolder.mIdImgItemSelectTimeLogo.setVisibility(View.VISIBLE);        }      } else {        if (myViewHolder.mIdImgItemSelectTimeLogo.getVisibility()!=View.GONE) {          myViewHolder.mIdImgItemSelectTimeLogo.setVisibility(View.GONE);        }      }    } else {      if (myViewHolder.mIdImgItemSelectTimeLogo.getVisibility()!=View.GONE) {        myViewHolder.mIdImgItemSelectTimeLogo.setVisibility(View.GONE);      }    }    return convertView;  }  /**   * 下一个月   */  public void addMonth() {    jumpMonth++;  }  /**   * 上一个月   */  public void lessMonth() {    jumpMonth--;  }  /**   * 更新日历数据   */  public void upDataMonth() {    int stepYear;    int stepMonth = currentMonth + jumpMonth;    if (stepMonth > 0) {      //下一个月      if (stepMonth % 12 == 0) {        stepYear = currentYear + stepMonth / 12 - 1;        stepMonth = 12;      } else {        stepYear = currentYear + stepMonth / 12;        stepMonth = stepMonth % 12;      }    } else {      //上一个月      stepYear = currentYear - 1 + stepMonth / 12;      stepMonth = stepMonth % 12 + 12;    }    getCalendar(stepYear, stepMonth);  }  /**   * 得到某年的某月的天数且这月的第一天是星期几   *   * @param year   * @param month   */  private void getCalendar(int year, int month) {    isLeapyear = sc.isLeapYear(year);       //是否为闰年     daysOfMonth = sc.getDaysOfMonth(isLeapyear, month); //某月的总天数     dayOfWeek = sc.getWeekdayOfMonth(year, month);   //某月第一天为星期几     lastDaysOfMonth = sc.getDaysOfMonth(isLeapyear, month - 1); //上一个月的总天数    getWeek(year, month);  }  /**   * 将一个月中的每一天的值添加入数组dayNuber中   *   * @param year   * @param month   */  private void getWeek(int year, int month) {    schDateTagFlag.clear();    currentFlag = -1;    int j = 1;    //得到当前月的所有日程日期(这些日期需要标记)    for (int i = 0; i < dayNumber.length; i++) {      if (i < dayOfWeek) { //前一个月        int temp = lastDaysOfMonth - dayOfWeek + 1;        dayNumber[i] = (temp + i) + "";      } else if (i < daysOfMonth + dayOfWeek) {//本月        int day = i - dayOfWeek + 1;  //得到的日期        dayNumber[i] = i - dayOfWeek + 1 + "";        //对于当前月才去标记当前日期         String yearStr = String.valueOf(year);        String monthStr =getStr(String.valueOf(month),2);        String dayStr =getStr(String.valueOf(day),2);        String timeAll = yearStr + monthStr + dayStr;        if (timeAll.equals(currentDayStr)) {//判断选中的位置          currentFlag = i;        }        if (mSet != null && mSet.size() > 0) {          for (String s : mSet) {//迭代器遍历判断是否需要带标签            if (timeAll.equals(s)) {              schDateTagFlag.add(i);            }          }        }        setShowYear(yearStr);        setShowMonth(String.valueOf(month));      } else {  //下一个月        dayNumber[i] = j + "";        j++;      }    }  }  /**   * 获取当前时间 样式:20170830   * @param position   * @return   */  public String getItemTime(int position) {    String month = getStr(getShowMonth(), 2);    String day = getStr(getDateByClickItem(position), 2);    return getShowYear() + month + day;  }  /**   * 保留N位整数,不足前面补0   *   * @param file String   * @param bit 位数   * @return   */  public static String getStr(String file,int bit) {    while (file.length() < bit)      file = "0" + file;    return file;  }  /**   * 点击每一个item时返回item中的日期   *   * @param position   * @return   */  public String getDateByClickItem(int position) {    return dayNumber[position];  }  /**   * 在点击gridView时,得到这个月中第一天的位置   *   * @return   */  public int getStartPositon() {    return dayOfWeek + 7;  }  /**   * 在点击gridView时,得到这个月中最后一天的位置   *   * @return   */  public int getEndPosition() {    return (dayOfWeek + daysOfMonth + 7) - 1;  }  public String getShowYear() {    return showYear;  }  public void setShowYear(String showYear) {    this.showYear = showYear;  }  public String getShowMonth() {    return showMonth;  }  public void setShowMonth(String showMonth) {    this.showMonth = showMonth;  }  public String getAnimalsYear() {    return animalsYear;  }  public void setAnimalsYear(String animalsYear) {    this.animalsYear = animalsYear;  }  public String getLeapMonth() {    return leapMonth;  }  public void setLeapMonth(String leapMonth) {    this.leapMonth = leapMonth;  }  public Set<String> getSet() {    return mSet;  }  public void setSet(Set<String> set) {    mSet = set;  }  static class ViewHolder {    @BindView(R.id.id_img_item_select_time_logo)    ImageView mIdImgItemSelectTimeLogo;    @BindView(R.id.id_tv_item_select_time_day)    TextView mIdTvItemSelectTimeDay;    ViewHolder(View view) {      ButterKnife.bind(this, view);    }  }}

日历工具类:

/** * 日历工具类 */public class SpecialCalendar {   private int daysOfMonth = 0;   //某月的天数   private int dayOfWeek = 0;    //具体某一天是星期几   /**   * 判断是否为闰年   * @param year   * @return   */  public boolean isLeapYear(int year) {    if (year % 100 == 0 && year % 400 == 0) {       return true;     } else if (year % 100 != 0 && year % 4 == 0) {       return true;     }     return false;   }  /**   * 得到某月有多少天数   * @param isLeapyear   * @param month   * @return   */  public int getDaysOfMonth(boolean isLeapyear, int month) {    switch (month) {     case 1:     case 3:     case 5:     case 7:     case 8:     case 10:     case 12:       daysOfMonth = 31;       break;     case 4:     case 6:     case 9:     case 11:       daysOfMonth = 30;       break;     case 2:       if (isLeapyear) {         daysOfMonth = 29;       } else {         daysOfMonth = 28;       }      }     return daysOfMonth;   }  /**   * 指定某年中的某月的第一天是星期几   * @param year   * @param month   * @return   */  public int getWeekdayOfMonth(int year, int month){    Calendar cal = Calendar.getInstance();    cal.set(year, month-1, 1);     dayOfWeek = cal.get(Calendar.DAY_OF_WEEK)-1;     return dayOfWeek;   }       } 

布局文件:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"       android:layout_width="match_parent"       android:layout_height="match_parent"       android:orientation="vertical">  <include    layout="@layout/layout_public_finish_menu"    />  <LinearLayout    android:layout_width="match_parent"    android:layout_height="40dp"    android:background="@color/bg_home_gone_menu"    android:gravity="center"    android:orientation="horizontal"    >    <ImageView      android:id="@+id/id_img_select_time_less"      android:layout_width="wrap_content"      android:layout_height="match_parent"      android:layout_weight="1"      android:padding="5dp"      android:src="@mipmap/ic_sd_time_less"      android:background="@drawable/selector_public_btn_bg"      />    <TextView      android:id="@+id/id_tv_select_time_show"      android:layout_width="wrap_content"      android:layout_height="wrap_content"      android:layout_weight="4"      android:gravity="center"      android:text="年月"      android:textColor="@color/white"      android:textSize="@dimen/default_big"      />    <ImageView      android:id="@+id/id_img_select_time_add"      android:layout_width="wrap_content"      android:layout_height="match_parent"      android:layout_weight="1"      android:padding="5dp"      android:src="@mipmap/ic_sd_time_add"      android:background="@drawable/selector_public_btn_bg"      />  </LinearLayout>  <GridView    android:id="@+id/id_gv_select_item"    android:layout_width="wrap_content"    android:layout_height="wrap_content"    android:background="@drawable/bg_rectangle_null_black_1"    android:clickable="true"    android:gravity="center"    android:layout_gravity="center"    android:clipChildren="true"    android:listSelector="@null"    android:numColumns="7"    android:padding="1dp"    android:layout_margin="5dp"    android:stretchMode="columnWidth"    /></LinearLayout>

Item布局

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"        android:layout_width="match_parent"        android:layout_height="match_parent"        android:background="@drawable/bg_rectangle_null_black_1">  <TextView    android:id="@+id/id_tv_item_select_time_day"    android:layout_width="match_parent"    android:layout_height="match_parent"    android:gravity="center"    android:text="天"    android:textSize="@dimen/default_big"    android:layout_margin="1dp"    android:textStyle="bold"    />  <ImageView    android:id="@+id/id_img_item_select_time_logo"    android:layout_width="5dp"    android:layout_height="5dp"    android:layout_margin="3dp"    android:src="@drawable/shap_doorbell_oval_red"    android:visibility="gone"    /></RelativeLayout>

布局只供参考...可以根据需求进行修改

下面就看看简单的调用

 //传入当前的年,月..已经选中的时间(20170830) mAdapter = new CalendarAdapter(mContext, year_c, month_c, currentDayStr); mIdGvSelectItem.setAdapter(mAdapter); /**   * GridView Item的点击事件   */  private class MyGvListener implements AdapterView.OnItemClickListener {    @Override    public void onItemClick(AdapterView<?> parent, View view, int position, long id) {      TextView mTv = (TextView) view.findViewById(R.id.id_tv_item_select_time_day);      boolean isOnClick = (boolean) mTv.getTag();      if (isOnClick) {        String time = mAdapter.getItemTime(position);        Intent mIntent = getIntent();        mIntent.putExtra("fileDate", time);        setResult(AppStart.SDVA_SDTA, mIntent);        finish();        Log.i(TAG,"当前选择的时间:" + time);      }    }  } /**   * 点击事件逻辑处理   */  private class MyListener implements View.OnClickListener {    @Override    public void onClick(View v) {      switch (v.getId()) {        //上一个月        case R.id.id_img_select_time_less:          mAdapter.lessMonth();          mHandler.sendEmptyMessage(UPDATA_TIME);          addTextToTopTextView(mIdTvSelectTimeShow);          break;        //下一个月        case R.id.id_img_select_time_add:          mAdapter.addMonth();          mHandler.sendEmptyMessage(UPDATA_TIME);          addTextToTopTextView(mIdTvSelectTimeShow);          break;      }    }  }private Set<String> dayEventCount = new HashSet<>();//设置需要显示标签的实际 mAdapter.setSet(dayEventCount);//更新  @Override  protected void uiHandlerMessage(Message msg) {    switch (msg.what) {      case UPDATA_TIME:        mAdapter.upDataMonth();        mAdapter.notifyDataSetChanged();        break;    }  }

调用部分的代码由于是从项目中直接复制出来的..代码的前后没有什么关联性,主要是说明功能的..请根据自己的项目进行调整..

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

发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表