在項目開發過程中,有時會有預約提醒、定時提醒等需求,這時我們可以使用系統日歷來輔助提醒。通過向系統日歷中寫入事件、設置提醒方式(鬧鍾),實現到達某個特定的時間自動提醒的功能。這樣做的好處是由於提醒功能是交付給系統日歷來做,不會出現應用被殺情況,能夠做到准時提醒。
一般來說實現向系統日歷中讀寫事件一般有以下幾個步驟:
(1)需要有讀寫日歷權限;
(2)如果沒有日歷賬戶需要先創建賬戶;
(3)實現日歷事件增刪改查、提醒功能;
一般來說實現向系統日歷中讀寫事件一般有以下幾個步驟:
(1)需要有讀寫日歷權限;
(2)如果沒有日歷賬戶需要先創建賬戶;
(3)實現日歷事件增刪改查、提醒功能;
1.權限申請
<uses-permission android:name="android.permission.READ_CALENDAR" /> <uses-permission android:name="android.permission.WRITE_CALENDAR" />
2.日歷相關uri
private static String CALENDER_URL = "content://com.android.calendar/calendars"; private static String CALENDER_EVENT_URL = "content://com.android.calendar/events"; private static String CALENDER_REMINDER_URL = "content://com.android.calendar/reminders";
3.具體實現
import android.content.ContentUris; import android.content.ContentValues; import android.content.Context; import android.database.Cursor; import android.graphics.Color; import android.net.Uri; import android.os.Build; import android.provider.CalendarContract; import android.support.annotation.RequiresApi; import android.text.TextUtils; import java.util.Calendar; import java.util.TimeZone; public class CalendarReminderUtils { private static String CALENDER_URL = "content://com.android.calendar/calendars"; private static String CALENDER_EVENT_URL = "content://com.android.calendar/events"; private static String CALENDER_REMINDER_URL = "content://com.android.calendar/reminders"; private static String CALENDARS_NAME = "boohee"; private static String CALENDARS_ACCOUNT_NAME = "BOOHEE@boohee.com"; private static String CALENDARS_ACCOUNT_TYPE = "com.android.boohee"; private static String CALENDARS_DISPLAY_NAME = "BOOHEE賬戶"; /** * 檢查是否已經添加了日歷賬戶,如果沒有添加先添加一個日歷賬戶再查詢 * 獲取賬戶成功返回賬戶id,否則返回-1 */ @RequiresApi(api = Build.VERSION_CODES.N) private static int checkAndAddCalendarAccount(Context context) { int oldId = checkCalendarAccount(context); if( oldId >= 0 ){ return oldId; }else{ long addId = addCalendarAccount(context); if (addId >= 0) { return checkCalendarAccount(context); } else { return -1; } } } /** * 檢查是否存在現有賬戶,存在則返回賬戶id,否則返回-1 */ private static int checkCalendarAccount(Context context) { Cursor userCursor = context.getContentResolver().query(Uri.parse(CALENDER_URL), null, null, null, null); try { if (userCursor == null) { //查詢返回空值 return -1; } int count = userCursor.getCount(); if (count > 0) { //存在現有賬戶,取第一個賬戶的id返回 userCursor.moveToFirst(); return userCursor.getInt(userCursor.getColumnIndex(CalendarContract.Calendars._ID)); } else { return -1; } } finally { if (userCursor != null) { userCursor.close(); } } } /** * 添加日歷賬戶,賬戶創建成功則返回賬戶id,否則返回-1 */ private static long addCalendarAccount(Context context) { TimeZone timeZone = TimeZone.getDefault(); ContentValues value = new ContentValues(); value.put(CalendarContract.Calendars.NAME, CALENDARS_NAME); value.put(CalendarContract.Calendars.ACCOUNT_NAME, CALENDARS_ACCOUNT_NAME); value.put(CalendarContract.Calendars.ACCOUNT_TYPE, CALENDARS_ACCOUNT_TYPE); value.put(CalendarContract.Calendars.CALENDAR_DISPLAY_NAME, CALENDARS_DISPLAY_NAME); value.put(CalendarContract.Calendars.VISIBLE, 1); value.put(CalendarContract.Calendars.CALENDAR_COLOR, Color.BLUE); value.put(CalendarContract.Calendars.CALENDAR_ACCESS_LEVEL, CalendarContract.Calendars.CAL_ACCESS_OWNER); value.put(CalendarContract.Calendars.SYNC_EVENTS, 1); value.put(CalendarContract.Calendars.CALENDAR_TIME_ZONE, timeZone.getID()); value.put(CalendarContract.Calendars.OWNER_ACCOUNT, CALENDARS_ACCOUNT_NAME); value.put(CalendarContract.Calendars.CAN_ORGANIZER_RESPOND, 0); Uri calendarUri = Uri.parse(CALENDER_URL); calendarUri = calendarUri.buildUpon() .appendQueryParameter(CalendarContract.CALLER_IS_SYNCADAPTER, "true") .appendQueryParameter(CalendarContract.Calendars.ACCOUNT_NAME, CALENDARS_ACCOUNT_NAME) .appendQueryParameter(CalendarContract.Calendars.ACCOUNT_TYPE, CALENDARS_ACCOUNT_TYPE) .build(); Uri result = context.getContentResolver().insert(calendarUri, value); long id = result == null ? -1 : ContentUris.parseId(result); return id; } /** * 添加日歷事件 */ @RequiresApi(api = Build.VERSION_CODES.N) public static void addCalendarEvent(Context context, String title, String description, long reminderTime, int previousDate) { if (context == null) { return; } int calId = checkAndAddCalendarAccount(context); //獲取日歷賬戶的id if (calId < 0) { //獲取賬戶id失敗直接返回,添加日歷事件失敗 return; } //添加日歷事件 Calendar mCalendar = Calendar.getInstance(); mCalendar.setTimeInMillis(reminderTime);//設置開始時間 long start = mCalendar.getTime().getTime(); mCalendar.setTimeInMillis(start + 10 * 60 * 1000);//設置終止時間,開始時間加10分鍾 long end = mCalendar.getTime().getTime(); ContentValues event = new ContentValues(); event.put("title", title); event.put("description", description); event.put("calendar_id", calId); //插入賬戶的id event.put(CalendarContract.Events.DTSTART, start); event.put(CalendarContract.Events.DTEND, end); event.put(CalendarContract.Events.HAS_ALARM, 1);//設置有鬧鍾提醒 event.put(CalendarContract.Events.EVENT_TIMEZONE, "Asia/Shanghai");//這個是時區,必須有 Uri newEvent = context.getContentResolver().insert(Uri.parse(CALENDER_EVENT_URL), event); //添加事件 if (newEvent == null) { //添加日歷事件失敗直接返回 return; } //事件提醒的設定 ContentValues values = new ContentValues(); values.put(CalendarContract.Reminders.EVENT_ID, ContentUris.parseId(newEvent)); values.put(CalendarContract.Reminders.MINUTES, previousDate * 24 * 60);// 提前previousDate天有提醒 values.put(CalendarContract.Reminders.METHOD, CalendarContract.Reminders.METHOD_ALERT); Uri uri = context.getContentResolver().insert(Uri.parse(CALENDER_REMINDER_URL), values); if(uri == null) { //添加事件提醒失敗直接返回 return; } } /** * 刪除日歷事件 */ public static void deleteCalendarEvent(Context context,String title) { if (context == null) { return; } Cursor eventCursor = context.getContentResolver().query(Uri.parse(CALENDER_EVENT_URL), null, null, null, null); try { if (eventCursor == null) { //查詢返回空值 return; } if (eventCursor.getCount() > 0) { //遍歷所有事件,找到title跟需要查詢的title一樣的項 for (eventCursor.moveToFirst(); !eventCursor.isAfterLast(); eventCursor.moveToNext()) { String eventTitle = eventCursor.getString(eventCursor.getColumnIndex("title")); if (!TextUtils.isEmpty(title) && title.equals(eventTitle)) { int id = eventCursor.getInt(eventCursor.getColumnIndex(CalendarContract.Calendars._ID));//取得id Uri deleteUri = ContentUris.withAppendedId(Uri.parse(CALENDER_EVENT_URL), id); int rows = context.getContentResolver().delete(deleteUri, null, null); if (rows == -1) { //事件刪除失敗 return; } } } } } finally { if (eventCursor != null) { eventCursor.close(); } } }
4.測試添加事件
CalendarReminderUtils.addCalendarEvent(this,"學校讀書","吃了飯再去",System.currentTimeMillis()+3600*24*1000*2+10000,2);
5,效果

注意:6.0 以上需要申請權限才可以使用哦