Android插入日历提醒

Android插入日历提醒

本文用于说明Android日历功能

Android系统提供了插入日功能。第三方应用可以自行插入日历事件,系统日历会根据日历事件在设定的时间弹出通知提醒用户。

Android系统日历提醒原理

Android提供了一个provider:com.android.providers.calendar  该provider提供一套接口供开发者使用。

日历事件存储

日历事件存储在数据库里,数据库的路径是:

/data/data/com.android.providers.calendar/databases/calendar.db

日历事件数据格式

日历事件主要涉及三个数据结构,也是上面说的数据库里的三个表:Calendars、Events、Reminders

Calendars

存储的是日历帐号信息。每个app都可以往日历数据库里注册多个帐号,也可以使用数据库里已有的帐号。一般来讲,每个app注册一个帐号即可。当不需要使用这个日的时候,删除这个日历。与这个日历绑定的事件和提醒都会被删除。

表结构为:



Events

日历事件,比如今天下午三点要去读书。这就是一个事件。在Events表里,每一行数据都是一个事件

表结构为:


表中有一个字段是calendar_id。用于标识该事件属于哪个帐号,该值对应着Calendars表中的_id字段

Reminders

日历提醒,有了日历事件。并不代表就会产生提醒。如果某个日历事件需要提醒用户。需要对这个日历事件创建一个提醒,告诉系统,该事件需要以什么方式提醒用户

表结构:


表中event_id用于标识该提醒属于哪个日历事件。该值对应着Events表中的_id字段


以上三个表的字段含义这里就不一一解释了。具体可以看https://developer.android.com/reference/android/provider/CalendarContract.Calendars

https://developer.android.com/reference/android/provider/CalendarContract.Events

https://developer.android.com/reference/android/provider/CalendarContract.Reminders

日历开发API

1. 存储日历帐号

• 存储帐号信息的URI:android.provider.CalendarContract.Calendars#CONTENT_URI

• DEMO CODE

//Calendar 为自定义日历数据格式,用于存储用户创建的帐号信息

fun addCalendar(calendar: Calendar, resolver:

  ContentResolver): Long {

    val  calendars = queryCalendar(calendar.ACCOUNT_NAME, calendar.ACCOUNT_TYPE,  calendar.OWNER_ACCOUNT, resolver)

    if  (calendars != null && calendars.size > 0) {


  Log.d("lzqtest", "CalendarWrapper.addCalendar: already  has a same calendar "+calendars)


  return CalendarStatus.ALREADY_EXIST.value()

    }

    val  contentValues = ContentValues()

    if  (!calendar.ACCOUNT_TYPE.isNullOrEmpty())  {


  contentValues.put(Calendars.ACCOUNT_TYPE,  calendar.ACCOUNT_TYPE)

    }

    if  (!calendar.ACCOUNT_NAME.isNullOrEmpty())  {


  contentValues.put(Calendars.ACCOUNT_NAME,  calendar.ACCOUNT_NAME)

    }

    if  (!calendar.NAME.isNullOrEmpty()) {


  contentValues.put(Calendars.NAME,  calendar.NAME)

    }

    if  (calendar.DIRTY != null) {


  contentValues.put(Calendars.DIRTY,  calendar.DIRTY)

    }

    if  (!calendar.MUTATORS.isNullOrEmpty())  {


  contentValues.put(Calendars.MUTATORS,  calendar.MUTATORS)

    }

    if  (!calendar.CALENDAR_DISPLAY_NAME.isNullOrEmpty())  {


  contentValues.put(Calendars.CALENDAR_DISPLAY_NAME,  calendar.CALENDAR_DISPLAY_NAME)

    }

    if  (calendar.CALENDAR_COLOR != null) {


  contentValues.put(Calendars.CALENDAR_COLOR,  calendar.CALENDAR_COLOR)

    }

    if  (calendar.CALENDAR_ACCESS_LEVEL != null) {


  contentValues.put(Calendars.CALENDAR_ACCESS_LEVEL,  calendar.CALENDAR_ACCESS_LEVEL)

    }

    if  (!calendar.CALENDAR_LOCATION.isNullOrEmpty())  {


  contentValues.put(Calendars.CALENDAR_LOCATION,  calendar.CALENDAR_LOCATION)

    }

    if  (!calendar.CALENDAR_TIME_ZONE.isNullOrEmpty())  {


  contentValues.put(Calendars.CALENDAR_TIME_ZONE,  calendar.CALENDAR_TIME_ZONE)

    }

    if  (!calendar.OWNER_ACCOUNT.isNullOrEmpty())  {


  contentValues.put(Calendars.OWNER_ACCOUNT,  calendar.OWNER_ACCOUNT)

    }

    if  (calendar.MAX_REMINDERS != null) {


  contentValues.put(Calendars.MAX_REMINDERS,  calendar.MAX_REMINDERS)

    }

    if  (!calendar.ALLOWED_REMINDERS.isNullOrEmpty())  {


  contentValues.put(Calendars.ALLOWED_REMINDERS,  calendar.ALLOWED_REMINDERS)

    }

    if  (!calendar.ALLOWED_AVAILABILITY.isNullOrEmpty())  {


  contentValues.put(Calendars.ALLOWED_AVAILABILITY,  calendar.ALLOWED_AVAILABILITY)

    }

    if  (!calendar.ALLOWED_ATTENDEE_TYPES.isNullOrEmpty())  {


  contentValues.put(Calendars.ALLOWED_ATTENDEE_TYPES,  calendar.ALLOWED_ATTENDEE_TYPES)

    }

    var uri  = Calendars.CONTENT_URI

    //uri需要通过appendQueryParameter 追加参数,否则会报插入失败的错误

    uri= uri.buildUpon()


  .appendQueryParameter(CalendarContract.CALLER_IS_SYNCADAPTER, "true")


  .appendQueryParameter(Calendars.ACCOUNT_NAME,  calendar.ACCOUNT_NAME)


  .appendQueryParameter(Calendars.ACCOUNT_TYPE,


  calendar.ACCOUNT_TYPE)


  .build()

//插入日历帐号信息

     val  ret=resolver.insert(uri, contentValues)


    return  if (ret!=null) ContentUris.parseId(ret) else CalendarStatus.FAIL.value()


}

2. 存储日历事件

• 存储日历事件URI:android.provider.CalendarContract.Events#CONTENT_URI

• DEMO CODE

// EVENT 为自定义对象,用于存储用户日历事件信息

fun addEvent(event: Event, resolver:

  ContentResolver): Long {

    val  events = queryEventByCalendarId(event.CALENDAR_ID, resolver)

    if  (events != null && events.size > 0) {

        for  (data in events) {


  if (data.TITLE?.equals(event.TITLE) == true

                    && data.DTSTART  == event.DTSTART

                    && data.DTEND ==  event.DTEND) {


  Log.d("lzqtest", "EventWrapper.addEvent: alread exist  event $events ")


  return CalendarStatus.ALREADY_EXIST.value()


  }

        }

    }

    if  (event.CALENDAR_ID == null || event.CALENDAR_ID!! < 0) {


  return CalendarStatus.PARAMETER_ERROR.value()

    }

    val  contentValues = ContentValues()


  contentValues.put(CalendarContract.Events.CALENDAR_ID, event.CALENDAR_ID)

    if  (!event.TITLE.isNullOrEmpty())


  contentValues.put(CalendarContract.Events.TITLE, event.TITLE)

    if  (!event.EVENT_LOCATION.isNullOrEmpty())


  contentValues.put(CalendarContract.Events.EVENT_LOCATION, event.EVENT_LOCATION)

    if  (!event.DESCRIPTION.isNullOrEmpty())  {


  contentValues.put(CalendarContract.Events.DESCRIPTION, event.DESCRIPTION)

    }

    if  (event.EVENT_COLOR != null) {


  contentValues.put(CalendarContract.Events.EVENT_COLOR, event.EVENT_COLOR)

    }

    if  (event.STATUS != null) {


  contentValues.put(CalendarContract.Events.STATUS, event.STATUS)

    }

    if  (event.SELF_ATTENDEE_STATUS != null) {


  contentValues.put(CalendarContract.Events.SELF_ATTENDEE_STATUS, event.SELF_ATTENDEE_STATUS)

    }

    if  (event.DTSTART != null) {


  contentValues.put(CalendarContract.Events.DTSTART, event.DTSTART)

    }

    if  (!event.EVENT_TIMEZONE.isNullOrEmpty())  {


  contentValues.put(CalendarContract.Events.EVENT_TIMEZONE, event.EVENT_TIMEZONE)

    }

    if  (event.ALL_DAY != null) {


  contentValues.put(CalendarContract.Events.ALL_DAY, event.ALL_DAY)

    }

    if  (event.ACCESS_LEVEL != null) {


  contentValues.put(CalendarContract.Events.ACCESS_LEVEL, event.ACCESS_LEVEL)

    }

    if  (event.AVAILABILITY != null) {


  contentValues.put(CalendarContract.Events.AVAILABILITY, event.AVAILABILITY)

    }

    if  (event.HAS_ALARM != null) {


  contentValues.put(CalendarContract.Events.HAS_ALARM, event.HAS_ALARM)

    }

    if (event.HAS_EXTENDED_PROPERTIES  != null) {


  contentValues.put(CalendarContract.Events.HAS_EXTENDED_PROPERTIES, event.HAS_EXTENDED_PROPERTIES)

    }

    if  (!event.RRULE.isNullOrEmpty()) {


  contentValues.put(CalendarContract.Events.RRULE, event.RRULE)

    }

    if  (!event.RDATE.isNullOrEmpty()) {


  contentValues.put(CalendarContract.Events.RDATE, event.RDATE)

    }

    if  (!event.EXRULE.isNullOrEmpty()) {


  contentValues.put(CalendarContract.Events.EXRULE, event.EXRULE)

    }

    if  (!event.EXDATE.isNullOrEmpty()) {


  contentValues.put(CalendarContract.Events.EXDATE, event.EXDATE)

    }

    if  (event.ORIGINAL_ID != null) {


  contentValues.put(CalendarContract.Events.ORIGINAL_ID, event.ORIGINAL_ID)

    }

    if  (event.LAST_DATE != null) {


  contentValues.put(CalendarContract.Events.LAST_DATE, event.LAST_DATE)

    }

    if  (event.HAS_ATTENDEE_DATA != null) {


  contentValues.put(CalendarContract.Events.HAS_ATTENDEE_DATA, event.HAS_ATTENDEE_DATA)

    }

    if  (event.GUESTS_CAN_MODIFY != null) {


  contentValues.put(CalendarContract.Events.GUESTS_CAN_MODIFY, event.GUESTS_CAN_MODIFY)

    }

    if  (event.GUESTS_CAN_INVITE_OTHERS != null) {


  contentValues.put(CalendarContract.Events.GUESTS_CAN_INVITE_OTHERS, event.GUESTS_CAN_INVITE_OTHERS)

    }

    if  (event.GUESTS_CAN_SEE_GUESTS != null) {


  contentValues.put(CalendarContract.Events.GUESTS_CAN_SEE_GUESTS, event.GUESTS_CAN_SEE_GUESTS)

    }

    if  (!event.ORGANIZER.isNullOrEmpty())  {


  contentValues.put(CalendarContract.Events.ORGANIZER, event.ORGANIZER)

    }

    if  (event.DELETED != null) {


  contentValues.put(CalendarContract.Events.DELETED, event.DELETED)

    }

    if  (!event.EVENT_END_TIMEZONE.isNullOrEmpty())  {


  contentValues.put(CalendarContract.Events.EVENT_END_TIMEZONE, event.EVENT_END_TIMEZONE)

    }


    if  (event.DTEND != null) {


  contentValues.put(CalendarContract.Events.DTEND, event.DTEND)

    }


  Log.d("lzqtest", "EventWrapper.addEvent: insert event  now ")

    val ret  = resolver.insert(CalendarContract.Events.CONTENT_URI,  contentValues)


  Log.d("lzqtest", "EventWrapper.addEvent: insert result  ret=$ret ")

    return  if (ret != null) ContentUris.parseId(ret) else CalendarStatus.FAIL.value()

}

3. 存储日历提醒

• 存储日历提醒URI:android.provider.CalendarContract.Reminders#CONTENT_URI

• DEMO CODE


fun addReminder(reminder: Reminder, resolver:

  ContentResolver): Long? {

    val  reminders = queryMinderByEventId(reminder.EVENT_ID, resolver)

    if  (reminders != null && reminders.size > 0) {

        for  (data in reminders) {


  if (data.MINUTES == (reminder.MINUTES)


  && data.METHOD == reminder.METHOD


  ) {


  Log.d(

                    "lzqtest",


  "ReminderWrapper.addReminder: alread exist reminder=$reminders  "


  )


  return CalendarStatus.ALREADY_EXIST.value()


  }

        }

    }

    if  (reminder.EVENT_ID == null) {


  Log.d("lzqtest", "ReminderWrapper.addReminder: EVENT_ID  is null ")


  return CalendarStatus.PARAMETER_ERROR.value()

    }

    val  contentValues = ContentValues()


  contentValues.put(CalendarContract.Reminders.EVENT_ID, reminder.EVENT_ID)


    if  (reminder.METHOD != null) {


  contentValues.put(CalendarContract.Reminders.METHOD, reminder.METHOD)

    }

    if  (reminder.MINUTES != null) {


  contentValues.put(CalendarContract.Reminders.MINUTES, reminder.MINUTES)

    }


  Log.d("lzqtest", "ReminderWrapper.addReminder: add  reminder now ")

    val ret  = resolver.insert(CalendarContract.Reminders.CONTENT_URI, contentValues)


  Log.d("lzqtest", "ReminderWrapper.addReminder: result  ret=$ret ")

    return  if (ret == null) CalendarStatus.FAIL.value()  else ContentUris.parseId(ret)

}

按以上方式插入日历数据到provider后,系统日历app即可显示插入的日历提醒。第三方日历app不一定会显示。这取决于第三方日历app会不会去读取系统日历数据库

demo code已上传至https://github.com/lzqnet/CalendarManager

你可能感兴趣的:(Android插入日历提醒)