1.前段时间要求做个定制的日历,由于UI给的图与传统的不一致,所以需要手绘一个
将编写代码给出
/**
* 日历控件 功能:获得点选的日期区间
*
*/
public class CalendarView extends View implements View.OnTouchListener {
private final static String TAG = "anCalendar";
private Date selectedStartDate;
private Date selectedEndDate;
public static Date curDate; // 当前日历显示的月
private Date today; // 今天的日期文字显示红色
private Date downDate; // 手指按下状态时临时日期
private Date showFirstDate, showLastDate; // 日历显示的第一个日期和最后一个日期
private int downIndex; // 按下的格子索引
private Calendar calendar;
private Surface surface;
private int[] date = new int[42]; // 日历显示数字
private int curStartIndex, curEndIndex; // 当前显示的日历起始的索引
//private boolean completed = false; // 为false表示只选择了开始日期,true表示结束日期也选择了
//给控件设置监听事件
private OnItemClickListener onItemClickListener;
private Canvas canvass;
private int[] visitDate;
private Context context;
public static boolean isItemChecked = false;
public int[] getVisitDate() {
return visitDate;
}
public void setVisitDate(int[] visitDate) {
invalidate();
this.visitDate = visitDate;
}
public CalendarView(Context context) {
super(context);
init();
}
public CalendarView(Context context, AttributeSet attrs) {
super(context, attrs);
this.context = context;
init();
}
private void init() {
curDate = selectedStartDate = selectedEndDate = today = new Date();
calendar = Calendar.getInstance();
calendar.setTime(curDate);
surface = new Surface();
surface.density = getResources().getDisplayMetrics().density;
setBackgroundColor(surface.bgColor);
setOnTouchListener(this);
}
@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
surface.width = getResources().getDisplayMetrics().widthPixels;
surface.height = (int) (getResources().getDisplayMetrics().heightPixels*2/5);
surface.height = (int) surface.width - surface.width*2/35;
// if (View.MeasureSpec.getMode(widthMeasureSpec) == View.MeasureSpec.EXACTLY) {
// surface.width = View.MeasureSpec.getSize(widthMeasureSpec);
// }
// if (View.MeasureSpec.getMode(heightMeasureSpec) == View.MeasureSpec.EXACTLY) {
// surface.height = View.MeasureSpec.getSize(heightMeasureSpec);
// }
widthMeasureSpec = View.MeasureSpec.makeMeasureSpec(surface.width,
View.MeasureSpec.EXACTLY);
heightMeasureSpec = View.MeasureSpec.makeMeasureSpec(surface.height,
View.MeasureSpec.EXACTLY);
setMeasuredDimension(widthMeasureSpec, heightMeasureSpec);
super.onMeasure(widthMeasureSpec, heightMeasureSpec);
}
@Override
protected void onLayout(boolean changed, int left, int top, int right,
int bottom) {
Log.d(TAG, "[onLayout] changed:"
+ (changed ? "new size" : "not change") + " left:" + left
+ " top:" + top + " right:" + right + " bottom:" + bottom);
if (changed) {
surface.init();
}
super.onLayout(changed, left, top, right, bottom);
}
@Override
protected void onDraw(Canvas canvas) {
canvass = canvas;
Log.d(TAG, "onDraw");
drawBg(0,surface.weekHeight,surface.width,surface.cellHeight+surface.weekHeight);
drawBg(0,surface.weekHeight+surface.cellHeight*2,surface.width,surface.cellHeight*3+surface.weekHeight);
drawBg(0,surface.weekHeight+surface.cellHeight*4,surface.width,surface.cellHeight*5+surface.weekHeight);
//drawBg(0,surface.cellHeight,surface.width,surface.cellHeight);
//drawBg(150,surface.cellHeight*3,surface.width,surface.cellHeight);
// 画框
canvas.drawPath(surface.boxPath, surface.borderPaint);
// 年月
//String monthText = getYearAndmonth();
//float textWidth = surface.monthPaint.measureText(monthText);
//canvas.drawText(monthText, (surface.width - textWidth) / 2f,
// surface.monthHeight * 3 / 4f, surface.monthPaint);
// 上一月/下一月
//canvas.drawPath(surface.preMonthBtnPath, surface.monthChangeBtnPaint);
//canvas.drawPath(surface.nextMonthBtnPath, surface.monthChangeBtnPaint);
// 星期
float weekTextY = surface.monthHeight + surface.weekHeight * 3 / 4f;
// 星期背景
surface.cellBgPaint.setColor(surface.weekBgColor);
canvas.drawRect(surface.weekHeight, surface.width, surface.weekHeight, surface.width, surface.cellBgPaint);
for (int i = 0; i < surface.weekText.length; i++) {
float weekTextX = i
* surface.cellWidth
+ (surface.cellWidth - surface.weekPaint
.measureText(surface.weekText[i])) / 2f;
canvas.drawText(surface.weekText[i], weekTextX, weekTextY,
surface.weekPaint);
}
// 计算日期
calculateDate();
// 按下状态,选择状态背景色
drawDownOrSelectedBg(canvas);
// write date number
// today index
int todayIndex = -1;
calendar.setTime(curDate);
String curYearAndMonth = calendar.get(Calendar.YEAR) + ""
+ calendar.get(Calendar.MONTH);
calendar.setTime(today);
String todayYearAndMonth = calendar.get(Calendar.YEAR) + ""
+ calendar.get(Calendar.MONTH);
if (curYearAndMonth.equals(todayYearAndMonth)) {
int todayNumber = calendar.get(Calendar.DAY_OF_MONTH);
todayIndex = curStartIndex + todayNumber - 1;
}
int start = 0;
int end = 0;
boolean isToday = false;
for (int i = 0; i < 42; i++) {
int color = surface.dateTextColor;
if (isLastMonth(i)) {
color = surface.borderColor;
} else if (isNextMonth(i)) {
color = surface.borderColor;
}
if (todayIndex != -1 && i == todayIndex) {
drawCellBg(canvas, todayIndex, surface.cellDownColor, true);
color = surface.todayTextColor;
isToday = true;
}else{
isToday = false;
}
drawCellText(canvas, i, date[i] + "", color);
int y = (i)/7+2;
int x = (i)%7;
if(date[i] == 1 && i < 14){
start = i;
}
if(date[i] == 1 && i > 20){
end = i-1;
}
if(getVisitDate()!=null && getVisitDate().length > 0){
for(int j=0;j < getVisitDate().length; j++){
if(start > 0 && end ==0){
if(date[i] == getVisitDate()[j]){
drawCirculer(x, y, isToday);
}
}else if(end > 0 && i < end){
if(date[i] == getVisitDate()[j]){
drawCirculer(x, y, isToday);
}
}
}
}
}
//drawCirculer(surface.cellHeight/2+10, surface.cellHeight*2-20, surface.cellHeight/2+20, surface.cellHeight*2-10);
//drawCirculer(surface.cellWidth*2+surface.cellHeight/2+10, surface.cellHeight*2-20, surface.cellWidth*2+surface.cellHeight/2+20, surface.cellHeight*2-10);
//drawCirculer(surface.cellWidth+surface.cellHeight/2+10, surface.cellHeight*2-20, surface.cellWidth+surface.cellHeight/2+20, surface.cellHeight*2-10);
super.onDraw(canvas);
}
private void calculateDate() {
calendar.setTime(curDate);
calendar.set(Calendar.DAY_OF_MONTH, 1);
int dayInWeek = calendar.get(Calendar.DAY_OF_WEEK);
Log.d(TAG, "day in week:" + dayInWeek);
int monthStart = dayInWeek;
if (monthStart == 1) {
monthStart = 8;
}
monthStart -= 1; //以日为开头-1,以星期一为开头-2
curStartIndex = monthStart;
date[monthStart] = 1;
// last month
if (monthStart > 0) {
calendar.set(Calendar.DAY_OF_MONTH, 0);
int dayInmonth = calendar.get(Calendar.DAY_OF_MONTH);
for (int i = monthStart - 1; i >= 0; i--) {
date[i] = dayInmonth;
dayInmonth--;
}
calendar.set(Calendar.DAY_OF_MONTH, date[0]);
}
showFirstDate = calendar.getTime();
// this month
calendar.setTime(curDate);
calendar.add(Calendar.MONTH, 1);
calendar.set(Calendar.DAY_OF_MONTH, 0);
// Log.d(TAG, "m:" + calendar.get(Calendar.MONTH) + " d:" +
// calendar.get(Calendar.DAY_OF_MONTH));
int monthDay = calendar.get(Calendar.DAY_OF_MONTH);
for (int i = 1; i < monthDay; i++) {
date[monthStart + i] = i + 1;
}
curEndIndex = monthStart + monthDay;
// next month
for (int i = monthStart + monthDay; i < 42; i++) {
date[i] = i - (monthStart + monthDay) + 1;
}
if (curEndIndex < 42) {
// 显示了下一月的
calendar.add(Calendar.DAY_OF_MONTH, 1);
}
calendar.set(Calendar.DAY_OF_MONTH, date[41]);
showLastDate = calendar.getTime();
}
/**
*
* @param canvas
* @param index
* @param text
*/
private void drawCellText(Canvas canvas, int index, String text, int color) {
int x = getXByIndex(index);
int y = getYByIndex(index);
surface.datePaint.setColor(color);
float cellY = surface.monthHeight + surface.cellWidth + (y - 1)
* surface.cellHeight + surface.cellWidth*0.3f;
float cellX = (surface.cellWidth * (x - 1))
+ (surface.cellWidth - surface.datePaint.measureText(text))
/ 2f;
canvas.drawText(text, cellX, cellY, surface.datePaint);
}
/**
*
* @param canvas
* @param index
* @param color
*/
private void drawCellBg(Canvas canvas, int index, int color, boolean isFill) {
int x = getXByIndex(index);
int y = getYByIndex(index);
surface.cellBgPaint.setColor(color);
if(isFill){
surface.cellBgPaint.setStyle(Style.FILL);
} else{
surface.cellBgPaint.setStyle(Style.STROKE);
surface.cellBgPaint.setStrokeWidth(2);
}
float left = surface.cellWidth * (x - 1) + surface.borderWidth;
float top = surface.monthHeight + surface.weekHeight + (y - 1)
* surface.cellHeight + surface.borderWidth;
canvas.drawRect(left, top, left + surface.cellWidth
- surface.borderWidth, top + surface.cellHeight
- surface.borderWidth, surface.cellBgPaint);
}
private boolean isFirsttime = false;
private int timeFirst = 1;
private void drawDownOrSelectedBg(Canvas canvas) {
// down and not up
if (downDate != null) {
//drawCellBg(canvas, downIndex, surface.cellDownColor, true);
}
// selected bg color
if (!selectedEndDate.before(showFirstDate)
&& !selectedStartDate.after(showLastDate)) {
int[] section = new int[] { -1, -1 };
calendar.setTime(curDate);
calendar.add(Calendar.MONTH, -1);
findSelectedIndex(0, curStartIndex, calendar, section);
if (section[1] == -1) {
calendar.setTime(curDate);
findSelectedIndex(curStartIndex, curEndIndex, calendar, section);
}
if (section[1] == -1) {
calendar.setTime(curDate);
calendar.add(Calendar.MONTH, 1);
findSelectedIndex(curEndIndex, 42, calendar, section);
}
if (section[0] == -1) {
section[0] = 0;
}
if (section[1] == -1) {
section[1] = 41;
}
for (int i = section[0]; i <= section[1]; i++) {
if(isItemChecked){
drawCellBg(canvas, i, surface.cellSelectedColor, false);
isItemChecked = false;
}
}
if(timeFirst > 2)
isFirsttime = true;
timeFirst ++;
}
}
private void findSelectedIndex(int startIndex, int endIndex,
Calendar calendar, int[] section) {
for (int i = startIndex; i < endIndex; i++) {
calendar.set(Calendar.DAY_OF_MONTH, date[i]);
Date temp = calendar.getTime();
// Log.d(TAG, "temp:" + temp.toLocaleString());
if (temp.compareTo(selectedStartDate) == 0) {
section[0] = i;
}
if (temp.compareTo(selectedEndDate) == 0) {
section[1] = i;
return;
}
}
}
public Date getSelectedStartDate() {
return selectedStartDate;
}
public Date getSelectedEndDate() {
return selectedEndDate;
}
private boolean isLastMonth(int i) {
if (i < curStartIndex) {
return true;
}
return false;
}
private boolean isNextMonth(int i) {
if (i >= curEndIndex) {
return true;
}
return false;
}
private int getXByIndex(int i) {
return i % 7 + 1; // 1 2 3 4 5 6 7
}
private int getYByIndex(int i) {
return i / 7 + 1; // 1 2 3 4 5 6
}
// 获得当前应该显示的年月
public String getYearAndmonth() {
calendar.setTime(curDate);
int year = calendar.get(Calendar.YEAR);
int month = calendar.get(Calendar.MONTH);
return year + "-" + surface.monthText[month];
}
//上一月
public String clickLeftMonth(){
calendar.setTime(curDate);
calendar.add(Calendar.MONTH, -1);
curDate = calendar.getTime();
invalidate();
return getYearAndmonth();
}
//下一月
public String clickRightMonth(){
calendar.setTime(curDate);
calendar.add(Calendar.MONTH, 1);
curDate = calendar.getTime();
invalidate();
return getYearAndmonth();
}
private void setSelectedDateByCoor(float x, float y) {
// change month
// if (y < surface.monthHeight) {
// // pre month
// if (x < surface.monthChangeWidth) {
// calendar.setTime(curDate);
// calendar.add(Calendar.MONTH, -1);
// curDate = calendar.getTime();
// }
// // next month
// else if (x > surface.width - surface.monthChangeWidth) {
// calendar.setTime(curDate);
// calendar.add(Calendar.MONTH, 1);
// curDate = calendar.getTime();
// }
// }
// cell click down
if (y > surface.monthHeight + surface.weekHeight) {
int m = (int) (Math.floor(x / surface.cellWidth) + 1);
int n = (int) (Math
.floor((y - (surface.monthHeight + surface.weekHeight))
/ Float.valueOf(surface.cellHeight)) + 1);
downIndex = (n - 1) * 7 + m - 1;
Log.d(TAG, "downIndex:" + downIndex);
calendar.setTime(curDate);
if (isLastMonth(downIndex)) {
calendar.add(Calendar.MONTH, -1);
} else if (isNextMonth(downIndex)) {
calendar.add(Calendar.MONTH, 1);
}
calendar.set(Calendar.DAY_OF_MONTH, date[downIndex]);
downDate = calendar.getTime();
}
invalidate();
}
@Override
public boolean onTouch(View v, MotionEvent event) {
switch (event.getAction()) {
case MotionEvent.ACTION_DOWN:
setSelectedDateByCoor(event.getX(), event.getY());
break;
case MotionEvent.ACTION_UP:
if (downDate != null) {
// if (!completed) {
// if (downDate.before(selectedStartDate)) {
// selectedEndDate = selectedStartDate;
// selectedStartDate = downDate;
// } else {
// selectedEndDate = downDate;
// }
// completed = true;
// } else {
// selectedStartDate = selectedEndDate = downDate;
// completed = false;
// }
selectedStartDate = selectedEndDate = downDate;
//响应监听事件
onItemClickListener.OnItemClick(selectedStartDate);
// Log.d(TAG, "downdate:" + downDate.toLocaleString());
//Log.d(TAG, "start:" + selectedStartDate.toLocaleString());
//Log.d(TAG, "end:" + selectedEndDate.toLocaleString());
downDate = null;
invalidate();
}
break;
}
return true;
}
//给控件设置监听事件
public void setOnItemClickListener(OnItemClickListener onItemClickListener){
this.onItemClickListener = onItemClickListener;
}
//监听接口
public interface OnItemClickListener {
void OnItemClick(Date date);
}
/**
*
* 1. 布局尺寸 2. 文字颜色,大小 3. 当前日期的颜色,选择的日期颜色
*/
private class Surface {
public float density;
public int width; // 整个控件的宽度
public int height; // 整个控件的高度
public float monthHeight; // 显示月的高度
//public float monthChangeWidth; // 上一月、下一月按钮宽度
public float weekHeight; // 显示星期的高度
public float cellWidth; // 日期方框宽度
public float cellHeight; // 日期方框高度
public float borderWidth;
public float circleSize; //圆点尺寸
public int bgColor = Color.parseColor("#FFFFFF");
private int dateTextColor = Color.BLACK;
private int todayTextColor = Color.parseColor("#FFFFFF");
private int weekTextColor = Color.parseColor("#9b9b9b");
private int monthTextColor = Color.BLACK;;
private int weekBgColor = Color.parseColor("#f8f8f8");
//private int textColorUnimportant = Color.parseColor("#666666");
private int btnColor = Color.parseColor("#666666");
private int borderColor = Color.parseColor("#CCCCCC");
public int todayNumberColor = Color.parseColor("#19b3ed");
public int cellDownColor = Color.parseColor("#19b3ed");
public int cellSelectedColor = Color.parseColor("#19b3ed");
public Paint borderPaint;
public Paint monthPaint;
public Paint weekPaint;
public Paint datePaint;
public Paint monthChangeBtnPaint;
public Paint cellBgPaint;
public Path boxPath; // 边框路径
//public Path preMonthBtnPath; // 上一月按钮三角形
//public Path nextMonthBtnPath; // 下一月按钮三角形
public String[] weekText = { "S","M", "T", "W", "T", "F", "S"};
//public String[] weekText = { "Sun","Mon", "Tue", "Wed", "Thu", "Fri", "Sat"};
//public String[] monthText = {"Jan","Feb","Mar","Apr","May","Jun","Jul","Aug","Sep","Oct","Nov","Dec"};
public String[] monthText = {"01","02","03","04","05","06","07","08","09","10","11","12"};
public void init() {
float temp = height / 7f;
monthHeight = 0;//(float) ((temp + temp * 0.3f) * 0.6);
//monthChangeWidth = monthHeight * 1.5f;
//weekHeight = (float) ((temp + temp * 0.3f) * 0.7);
cellWidth = width / 7f;
//cellHeight = (height - monthHeight - weekHeight) / 6f;
cellHeight = cellWidth;
circleSize = (float) (cellWidth*0.1);
weekHeight = (float)(cellWidth*0.6);
borderPaint = new Paint();
borderPaint.setColor(borderColor);
borderPaint.setStyle(Paint.Style.STROKE);
borderWidth = (float) (0.5 * density);
// Log.d(TAG, "borderwidth:" + borderWidth);
borderWidth = borderWidth < 1 ? 1 : borderWidth;
borderPaint.setStrokeWidth(borderWidth);
monthPaint = new Paint();
monthPaint.setColor(monthTextColor);
monthPaint.setAntiAlias(true);
float textSize = (float) (cellHeight*0.4);
Log.d(TAG, "text size:" + textSize);
monthPaint.setTextSize(textSize);
//monthPaint.setTypeface(Typeface.DEFAULT_BOLD);
weekPaint = new Paint();
weekPaint.setColor(weekTextColor);
weekPaint.setAntiAlias(true);
float weekTextSize = (float) (cellHeight*0.4);
weekPaint.setTextSize(weekTextSize);
//weekPaint.setTypeface(Typeface.DEFAULT_BOLD);
datePaint = new Paint();
datePaint.setColor(dateTextColor);
datePaint.setAntiAlias(true);
float cellTextSize = (float) (cellHeight*0.4);
datePaint.setTextSize(cellTextSize);
//datePaint.setTypeface(Typeface.DEFAULT_BOLD);
boxPath = new Path();
//boxPath.addRect(0, 0, width, height, Direction.CW);
//boxPath.moveTo(0, monthHeight);
//boxPath.rLineTo(width, 0);
// boxPath.moveTo(0, monthHeight + weekHeight);
// boxPath.rLineTo(width, 0);
// for (int i = 1; i < 6; i++) {
// boxPath.moveTo(0, monthHeight + weekHeight + i * cellHeight);
// boxPath.rLineTo(width, 0);
// boxPath.moveTo(i * cellWidth, monthHeight);
// boxPath.rLineTo(0, height - monthHeight);
// }
// boxPath.moveTo(6 * cellWidth, monthHeight);
// boxPath.rLineTo(0, height - monthHeight);
//preMonthBtnPath = new Path();
//int btnHeight = (int) (monthHeight * 0.6f);
//preMonthBtnPath.moveTo(monthChangeWidth / 2f, monthHeight / 2f);
//preMonthBtnPath.rLineTo(btnHeight / 2f, -btnHeight / 2f);
//preMonthBtnPath.rLineTo(0, btnHeight);
//preMonthBtnPath.close();
//nextMonthBtnPath = new Path();
//nextMonthBtnPath.moveTo(width - monthChangeWidth / 2f,
// monthHeight / 2f);
//nextMonthBtnPath.rLineTo(-btnHeight / 2f, -btnHeight / 2f);
//nextMonthBtnPath.rLineTo(0, btnHeight);
//nextMonthBtnPath.close();
monthChangeBtnPaint = new Paint();
monthChangeBtnPaint.setAntiAlias(true);
monthChangeBtnPaint.setStyle(Paint.Style.FILL_AND_STROKE);
monthChangeBtnPaint.setColor(btnColor);
cellBgPaint = new Paint();
cellBgPaint.setAntiAlias(true);
cellBgPaint.setStyle(Paint.Style.FILL);
cellBgPaint.setColor(cellSelectedColor);
}
}
private void drawBg(int left, float top, float cellHeight2, float cellBotom){
// 首先定义一个paint
Paint paint = new Paint();
// 绘制矩形区域-实心矩形
// 设置颜色
paint.setColor(getResources().getColor(R.color.calendar_gray1));
// 设置样式-填充
paint.setStyle(Paint.Style.FILL);
// 绘制一个矩形
canvass.drawRect(new Rect(left, (int)top, (int) cellHeight2, (int) cellBotom), paint);
}
private void drawCirculer(int x, int y, boolean isTosay){
Paint paint = new Paint();
if(isTosay){
paint.setColor(getResources().getColor(R.color.white));
}else{
paint.setColor(getResources().getColor(R.color.calendar_blue));
}
RectF rf2 = new RectF(surface.cellWidth*x+(surface.cellHeight-surface.circleSize)/2, surface.weekHeight+surface.cellHeight*(y-1)-surface.circleSize*2,
surface.cellWidth*x+(surface.cellHeight-surface.circleSize)/2+surface.circleSize, surface.weekHeight+surface.cellHeight*(y-1)-surface.circleSize);
//画圆
canvass.drawOval(rf2, paint);
}
}
ui层通过调用该控件显示日历即可。
上一个月
String leftYearAndmonth = viewPagercurrent.clickLeftMonth();
String[] lya = leftYearAndmonth.split("-");
dateTxt.setText(lya[0]+"-"+lya[1]);
下一个月
String rightYearAndmonth = viewPagercurrent.clickRightMonth();
String[] rya = rightYearAndmonth.split("-");
dateTxt.setText(rya[0]+"-"+rya[1]);
点击事件
class calendarItemClickListener implements OnItemClickListener {
@Override
public void OnItemClick(Date date) {
showDateVisitCalendar(date);
CalendarView.isItemChecked = true;
//Toast.makeText(getActivity(), date + "", Toast.LENGTH_SHORT).show();
}