Android自定義日曆控制元件的實現過程詳解
為什麼要自定義控制元件
有時,原生控制元件不能滿足我們對於外觀和功能的需求,這時候可以自定義控制元件來定製外觀或功能;有時,原生控制元件可以通過複雜的編碼實現想要的功能,這時候可以自定義控制元件來提高程式碼的可複用性。
如何自定義控制元件
下面我通過我在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_DOWN
和ACTION_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
相關文章
- VirtualView Android 實現詳解(三)—— 新增一個自定義控制元件ViewAndroid控制元件
- iOS 自定義日曆(日期選擇)控制元件iOS控制元件
- Android 控制元件架構與自定義控制元件詳解Android控制元件架構
- 【android】自定義佈局控制控制元件的位置可以通過繼承FrameLayout實現Android控制元件繼承
- 【朝花夕拾】Android自定義View篇之(四)自定義View的三種實現方式及自定義屬性詳解AndroidView
- 案例展示自定義C函式的實現過程函式
- Qt實現自定義控制元件QT控制元件
- 縱享絲滑滑動切換的周月日曆,水滴效果,豐富自定義日曆樣式,仿小米日曆(ViewDragHelper實現)View
- Android自定義拍照實現Android
- Android自定義多宮格解鎖控制元件Android控制元件
- 深入mysql建立自定義函式與儲存過程的詳解MySql函式儲存過程
- Flutter日曆,可以自定義風格UIFlutterUI
- 【Node】詳解模組的實現過程
- 【Android】自定義樹形控制元件Android控制元件
- Feign通過自定義註解實現路徑的轉義
- Spring Data JPA框架的Repository自定義實現詳解Spring框架
- 自定義 Behavior,實現巢狀滑動、平滑切換周月檢視的日曆巢狀
- 微信小程式 vue 自定義日曆 sku微信小程式Vue
- VirtualView Android實現詳解(二)—— 虛擬控制元件的設計與實現ViewAndroid控制元件
- Android自定義控制元件 帶文字提示的SeekBarAndroid控制元件
- QT實現可拖動自定義控制元件QT控制元件
- Springboot AOP 自定義註解實現系統日誌Spring Boot
- 使用自定義註解透過BeanPostProcessor實現策略模式Bean模式
- RN自定義元件封裝 – 拖拽選擇日期的日曆元件封裝
- RN自定義元件封裝 - 拖拽選擇日期的日曆元件封裝
- Android自定義view之實現帶checkbox的SnackbarAndroidView
- 微信小程式Tree自定義控制元件實現微信小程式控制元件
- android 螢幕適配一:通過自定義View的方式實現適配AndroidView
- 帶農曆日曆的DatePicker控制元件!Xamarin控制元件開發小記控制元件
- Android自定義View--翻書控制元件(一)AndroidView控制元件
- Android自定義控制元件(神級)+MediaRecoder錄音Android控制元件
- 基於 RecyclerView 實現的歌詞滾動自定義控制元件View控制元件
- MySQL使用之五_自定義函式和自定義過程MySql函式
- Android Studio通過style和layer-list實現自定義進度條Android
- 筆記3:自定義註解的實現筆記
- 聊聊如何通過自定義註解實現springmvc和sentinel整合SpringMVC
- 用java實現日曆demo。Java
- Vue日曆的編寫,可顯示周和月的模式(其中可以自定義日曆裡內容的顯示)Vue模式
- EventSource的自定義實現