美文网首页
Android插入日历提醒

Android插入日历提醒

作者: 如水至清 | 来源:发表于2021-07-01 17:05 被阅读0次

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插入日历提醒

    Android插入日历提醒 本文用于说明Android日历功能 Android系统提供了插入日功能。第三方应用可以...

  • Android日历提醒软件

    快要毕业答辩了,课题是做一款Android日历提醒软件,最近一直在改论文,代码很久之前就完成了,虽然很多方面写的不...

  • android 本地日历插入事件

    1.权限申请 为了实现在项目中调用系统日历和插入日程事件,我们首先在AndroidManifest.xml文件中添...

  • 日历提醒

    直接上关键函数

  • iOS日历提醒

    最近公司项目要上一个日历提醒的功能,之前没有接触过类似的需求,只得从头开始慢慢研究,今天借机给大家分享一下个...

  • 日历提醒事项

    1.需要在info.plist添加访问日历的权限:Privacy - Calendars Usage Descri...

  • 系统日历 插入重复事件规则 (android/java)

    Recurrence Rule 重复规则 rrule(Recurrence Rule) 属于 icalendar ...

  • android 设置系统闹钟和系统日历提醒

    现在有一个定时提醒的功能, 用 AlarmManager 自己来做,有多少坑做过的都知道。(应用被kill, 应用...

  • iOS日历提醒(Swift)

    最近项目中要用到类似天猫日历提醒功能,网上找了很多资料跟实际需求有出入,换是决定自己写一下,语言描述能力有限直接上...

  • 日历提醒功能实现

    提醒功能主要依赖的是Android的AlarmManager,应用通过CalendarProvider设置提醒,C...

网友评论

      本文标题:Android插入日历提醒

      本文链接:https://www.haomeiwen.com/subject/ktzjultx.html