在开发中可能会遇到某些情况下需要用到日历的功能,并且还要在日历上加标签什么的,最重要的就是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;

    }

  }

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

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

02-02 11:55