Android自定義日曆控制元件的實現過程詳解

zhikaizhang發表於2016-05-23

為什麼要自定義控制元件

有時,原生控制元件不能滿足我們對於外觀和功能的需求,這時候可以自定義控制元件來定製外觀或功能;有時,原生控制元件可以通過複雜的編碼實現想要的功能,這時候可以自定義控制元件來提高程式碼的可複用性。

如何自定義控制元件

下面我通過我在github上開源的Android-CalendarView專案為例,來介紹一下自定義控制元件的方法。該專案中自定義的控制元件類名是CalendarView。這個自定義控制元件覆蓋了一些自定義控制元件時常需要重寫的一些方法。

建構函式

為了支援本控制元件既能使用xml佈局檔案宣告,也可在java檔案中動態建立,實現了三個建構函式。

public CalendarView(Context context, AttributeSet attrs, int defStyle);
public CalendarView(Context context, AttributeSet attrs);
public CalendarView(Context context);

可以在引數列表最長的第一個方法中寫上你的初始化程式碼,下面兩個建構函式呼叫第一個即可。

public CalendarView(Context context, AttributeSet attrs) {
    this(context, attrs, 0);
}
public CalendarView(Context context) {
    this(context, null);
}

那麼在建構函式中做了哪些事情呢?

1 讀取自定義引數

讀取佈局檔案中可能設定的自定義屬性(該日曆控制元件僅自定義了一個mode引數來表示日曆的模式)。程式碼如下。只要在attrs.xml中自定義了屬性,就會自動建立一些R.styleable下的變數。

TypedArray typedArray = context.obtainStyledAttributes(attrs, R.styleable.CalendarView);
mode = typedArray.getInt(R.styleable.CalendarView_mode, Constant.MODE_SHOW_DATA_OF_THIS_MONTH);

然後附上res目錄下values目錄下的attrs.xml檔案,需要在此檔案中宣告你自定義控制元件的自定義引數。

<?xml version="1.0" encoding="utf-8"?>
<resources>
    <declare-styleable name="CalendarView">
        <attr name="mode" format="integer" />
    </declare-styleable>
</resources>

2 初始化關於繪製控制元件的相關引數

如字型的顏色、尺寸,控制元件各個部分尺寸。

3 初始化關於邏輯的相關引數

對於日曆來說,需要能夠判斷對應於當前的年月,日曆中的每個單元格是否合法,以及若合法,其表示的day的值是多少。未設定年月之前先用當前時間來初始化。實現如下。

/**
 * calculate the values of date[] and the legal range of index of date[]
 */
private void initial() {
    int dayOfWeek = calendar.get(Calendar.DAY_OF_WEEK);
    int monthStart = -1;
    if(dayOfWeek >= 2 && dayOfWeek <= 7){
        monthStart = dayOfWeek - 2;
    }else if(dayOfWeek == 1){
        monthStart = 6;
    }
    curStartIndex = monthStart;
    date[monthStart] = 1;
    int daysOfMonth = daysOfCurrentMonth();
    for (int i = 1; i < daysOfMonth; i++) {
        date[monthStart + i] = i + 1;
    }
    curEndIndex = monthStart + daysOfMonth;
    if(mode == Constant.MODE_SHOW_DATA_OF_THIS_MONTH){
        Calendar tmp = Calendar.getInstance();
        todayIndex = tmp.get(Calendar.DAY_OF_MONTH) + monthStart - 1;
    }
}

其中date[]是一個整型陣列,長度為42,因為一個日曆最多需要6行來顯示(6*7=42),curStartIndex和curEndIndex決定了date[]陣列的合法下標區間,即前者表示該月的第一天在date[]陣列的下標,後者表示該月的最後一天在date[]陣列的下標。

4 繫結了一個OnTouchListener監聽器

監聽控制元件的觸控事件。

onMeasure方法

該方法對控制元件的寬和高進行測量。CalendarView覆蓋了View類的onMeasure()方法,因為某個月的第一天可能是星期一到星期日的任何一個,而且每個月的天數不盡相同,因此日曆控制元件的行數會有多變化,也導致控制元件的高度會有變化。因此需要根據當前的年月計算控制元件顯示的高度(寬度設為螢幕寬度即可)。實現如下。

@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    widthMeasureSpec = View.MeasureSpec.makeMeasureSpec(screenWidth, View.MeasureSpec.EXACTLY);
    heightMeasureSpec = View.MeasureSpec.makeMeasureSpec(measureHeight(), View.MeasureSpec.EXACTLY);
    setMeasuredDimension(widthMeasureSpec, heightMeasureSpec);
    super.onMeasure(widthMeasureSpec, heightMeasureSpec);
}

其中screenWidth是建構函式中已經獲取的螢幕寬度,measureHeight()則是根據年月計算控制元件所需要的高度。實現如下,已經寫了非常詳細的註釋。

/**
 * calculate the total height of the widget
 */
private int measureHeight(){
    /**
     * the weekday of the first day of the month, Sunday's result is 1 and Monday 2 and Saturday 7, etc.
     */
    int dayOfWeek = calendar.get(Calendar.DAY_OF_WEEK);
    /**
     * the number of days of current month
     */
    int daysOfMonth = daysOfCurrentMonth();
    /**
     * calculate the total lines, which equals to 1 (head of the calendar) + 1 (the first line) + n/7 + (n%7==0?0:1)
     * and n means numberOfDaysExceptFirstLine
     */
    int numberOfDaysExceptFirstLine = -1;
    if(dayOfWeek >= 2 && dayOfWeek <= 7){
        numberOfDaysExceptFirstLine = daysOfMonth - (8 - dayOfWeek + 1);
    }else if(dayOfWeek == 1){
        numberOfDaysExceptFirstLine = daysOfMonth - 1;
    }
    int lines = 2 + numberOfDaysExceptFirstLine / 7 + (numberOfDaysExceptFirstLine % 7 == 0 ? 0 : 1);
    return (int) (cellHeight * lines);
}

onDraw方法

該方法實現對控制元件的繪製。其中drawCircle給定圓心和半徑繪製圓,drawText是給定一個座標x,y繪製文字。

/**
 * render
 */
@Override
protected void onDraw(Canvas canvas) {
	super.onDraw(canvas);
	/**
	 * render the head
	 */
	float baseline = RenderUtil.getBaseline(0, cellHeight, weekTextPaint);
	for (int i = 0; i < 7; i++) {
		float weekTextX = RenderUtil.getStartX(cellWidth * i + cellWidth * 0.5f, weekTextPaint, weekText[i]);
		canvas.drawText(weekText[i], weekTextX, baseline, weekTextPaint);
	}
	if(mode == Constant.MODE_CALENDAR){
		for (int i = curStartIndex; i < curEndIndex; i++) {
			drawText(canvas, i, textPaint, "" + date[i]);
		}
	}else if(mode == Constant.MODE_SHOW_DATA_OF_THIS_MONTH){
		for (int i = curStartIndex; i < curEndIndex; i++) {
			if(i < todayIndex){
				if(data[date[i]]){
					drawCircle(canvas, i, bluePaint, cellHeight * 0.37f);
					drawCircle(canvas, i, whitePaint, cellHeight * 0.31f);
					drawCircle(canvas, i, blackPaint, cellHeight * 0.1f);
				}else{
					drawCircle(canvas, i, grayPaint, cellHeight * 0.1f);
				}
			}else if(i == todayIndex){
				if(data[date[i]]){
					drawCircle(canvas, i, bluePaint, cellHeight * 0.37f);
					drawCircle(canvas, i, whitePaint, cellHeight * 0.31f);
					drawCircle(canvas, i, blackPaint, cellHeight * 0.1f);
				}else{
					drawCircle(canvas, i, grayPaint, cellHeight * 0.37f);
					drawCircle(canvas, i, whitePaint, cellHeight * 0.31f);
					drawCircle(canvas, i, blackPaint, cellHeight * 0.1f);
				}
			}else{
				drawText(canvas, i, textPaint, "" + date[i]);
			}
		}
	}
}

需要說明的是,繪製文字時的這個x表示開始位置的x座標(文字最左端),這個y卻不是文字最頂端的y座標,而應傳入文字的baseline。因此若想要將文字繪製在某個區域居中部分,需要經過一番計算。本專案將其封裝在了RenderUtil類中。實現如下。

/**
 * get the baseline to draw between top and bottom in the middle
 */
public static float getBaseline(float top, float bottom, Paint paint){
	Paint.FontMetrics fontMetrics = paint.getFontMetrics();
	return (top + bottom - fontMetrics.bottom - fontMetrics.top) / 2;
}
/**
 *	get the x position to draw around the middle
 */
public static float getStartX(float middle, Paint paint, String text){
	return middle - paint.measureText(text) * 0.5f;
}

自定義監聽器

控制元件需要自定義一些監聽器,以在控制元件發生了某種行為或互動時提供一個外部介面來處理一些事情。本專案的CalendarView提供了兩個介面,OnRefreshListener和OnItemClickListener,均為自定義的介面。onItemClick只傳了day一個引數,年和月可通過CalendarView物件的getYear和getMonth方法獲取。

interface OnItemClickListener{
	void onItemClick(int day);
}
interface OnRefreshListener{
	void onRefresh();
}

先介紹一下兩種mode,CalendarView提供了兩種模式,第一種普通日曆模式,日曆每個位置簡單顯示了day這個數字,第二種本月計劃完成情況模式,繪製了一些圖形來表示本月的某一天是否完成了計劃(模仿自悅跑圈,用一個圈表示本日跑了步)。

OnRefreshListener用於重新整理日曆資料後進行回撥。兩種模式定義了不同的重新整理方法,都對OnRefreshListener進行了回撥。refresh0用於第一種模式,refresh1用於第二種模式。

/**
 * used for MODE_CALENDAR
 * legal values of month: 1-12
 */
@Override
public void refresh0(int year, int month) {
	if(mode == Constant.MODE_CALENDAR){
		selectedYear = year;
		selectedMonth = month;
		calendar.set(Calendar.YEAR, selectedYear);
		calendar.set(Calendar.MONTH, selectedMonth - 1);
		calendar.set(Calendar.DAY_OF_MONTH, 1);
		initial();
		invalidate();
		if(onRefreshListener != null){
			onRefreshListener.onRefresh();
		}
	}
}

/**
 * used for MODE_SHOW_DATA_OF_THIS_MONTH
 * the index 1 to 31(big month), 1 to 30(small month), 1 - 28(Feb of normal year), 1 - 29(Feb of leap year)
 * is better to be accessible in the parameter data, illegal indexes will be ignored with default false value
 */
@Override
public void refresh1(boolean[] data) {
	/**
	 * the month and year may change (eg. Jan 31st becomes Feb 1st after refreshing)
	 */
	if(mode == Constant.MODE_SHOW_DATA_OF_THIS_MONTH){
		calendar = Calendar.getInstance();
		selectedYear = calendar.get(Calendar.YEAR);
		selectedMonth = calendar.get(Calendar.MONTH) + 1;
		calendar.set(Calendar.DAY_OF_MONTH, 1);
		for(int i = 1; i <= daysOfCurrentMonth(); i++){
			if(i < data.length){
				this.data[i] = data[i];
			}else{
				this.data[i] = false;
			}
		}
		initial();
		invalidate();
		if(onRefreshListener != null){
			onRefreshListener.onRefresh();
		}
	}
}

OnItemClickListener用於響應點選了日曆上的某一天這個事件。點選的判斷在onTouch方法中實現。實現如下。在同一位置依次接收到ACTION_DOWNACTION_UP兩個事件才認為完成了點選。

@Override
public boolean onTouch(View v, MotionEvent event) {
	float x = event.getX();
	float y = event.getY();
	switch (event.getAction()) {
		case MotionEvent.ACTION_DOWN:
			if(coordIsCalendarCell(y)){
				int index = getIndexByCoordinate(x, y);
				if(isLegalIndex(index)) {
					actionDownIndex = index;
				}
			}
			break;
		case MotionEvent.ACTION_UP:
			if(coordIsCalendarCell(y)){
				int actionUpIndex = getIndexByCoordinate(x, y);
				if(isLegalIndex(actionUpIndex)){
					if(actionDownIndex == actionUpIndex){
						actionDownIndex = -1;
						int day = date[actionUpIndex];
						if(onItemClickListener != null){
							onItemClickListener.onItemClick(day);
						}
					}
				}
			}
			break;
	}
	return true;
}

關於該日曆控制元件

日曆控制元件demo效果圖如下,分別為普通日曆模式和本月計劃完成情況模式。

需要說明的是CalendarView控制元件部分只包括日曆頭與下面的日曆,該控制元件上方的是其他控制元件,這裡僅用作展示一種使用方法,你完全可以自定義這部分的樣式。

此外,日曆頭的文字支援多種選擇,比如週一有四種表示:一、週一、星期一、Mon。此外還有其他一些控制樣式的介面,詳情見github專案主頁。

github專案主頁: Android-CalendarView

相關文章