查了一天半,总算有点大概了。以下是自己的理解,有错误的地方望指正。

  android系统有日历功能,应用程序可以根据一些接口开发自己的功能,即使是日历app也是根据这些接口开发的,所以我们可以利用程序向系统日历写入事件,然后用手机上的日历软件就可以看到我们添加的事件。网上这方面资料也不少,也有demo,但是我没找到一个可以正确运行的,有的是缺少参数,有的是版本的原因,4.0以上的系统这方面变动比较大,所以只能一边查资料一边修改。

  大体思路就是:先查看系统日历是否有账户,如果没有必须要添加一个,然后才可以添加事件。其中有好多参数是必须要填的,还有什么 sync adapter。事件的保存是按账户为单位的。

  目前还不完善,删除时会把系统日历中的所有账户都删除,请注意!!!

大体框架利用:http://blog.csdn.net/Android_Tutor/article/details/6165470 ,感谢原作者的分享。为了体现原作者的向往情怀,事件内容维持不变!!

在此基础上做了完善以及进行了主要的注释。

用户权限:

01.<uses-permission android:name="android.permission.READ_CALENDAR"/>
02.<uses-permission android:name="android.permission.WRITE_CALENDAR"/>

XML:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="fill_parent"
android:layout_height="fill_parent"
android:orientation="vertical" > <TextView
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:text="向系统日历中添加事件" /> <Button
android:id="@+id/inputaccount"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:onClick="onClick"
android:text="添加账户" /> <Button
android:id="@+id/readUserButton"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:onClick="onClick"
android:text="查看账户" /> <Button
android:id="@+id/readEventButton"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:onClick="onClick"
android:text="查看事件" /> <Button
android:id="@+id/writeEventButton"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:onClick="onClick"
android:text="插入事件" /> <Button
android:id="@+id/delEventButton"
android:layout_width="fill_parent"
android:layout_height="wrap_content"
android:onClick="onClick"
android:text="删除事件" /> </LinearLayout>

主程序代码:

package com.example.canlendertesttwo;

import java.util.Calendar;
import java.util.TimeZone; import android.app.Activity;
import android.content.ContentValues;
import android.database.Cursor;
import android.net.Uri;
import android.os.Bundle;
import android.provider.CalendarContract;
import android.provider.CalendarContract.Calendars;
import android.provider.CalendarContract.Events;
import android.view.View;
import android.widget.Toast; public class MainActivity extends Activity { //Android2.2版本以后的URL,之前的就不写了
private static String calanderURL = "content://com.android.calendar/calendars";
private static String calanderEventURL = "content://com.android.calendar/events";
private static String calanderRemiderURL = "content://com.android.calendar/reminders"; @Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main); } public void onClick(View v) {
if (v.getId() == R.id.readUserButton) { //读取系统日历账户,如果为0的话先添加
Cursor userCursor = getContentResolver().query(Uri.parse(calanderURL), null, null, null, null); System.out.println("Count: " + userCursor.getCount());
Toast.makeText(this, "Count: " + userCursor.getCount(), Toast.LENGTH_LONG).show(); for (userCursor.moveToFirst(); !userCursor.isAfterLast(); userCursor.moveToNext()) {
System.out.println("name: " + userCursor.getString(userCursor.getColumnIndex("ACCOUNT_NAME"))); String userName1 = userCursor.getString(userCursor.getColumnIndex("name"));
String userName0 = userCursor.getString(userCursor.getColumnIndex("ACCOUNT_NAME"));
Toast.makeText(this, "NAME: " + userName1 + " -- ACCOUNT_NAME: " + userName0, Toast.LENGTH_LONG).show();
}
}
else if (v.getId() == R.id.inputaccount) { //添加日历账户
initCalendars(); }
else if (v.getId() == R.id.delEventButton) { //删除事件 int rownum = getContentResolver().delete(Uri.parse(calanderURL), "_id!=-1", null); //注意:会全部删除所有账户,新添加的账户一般从id=1开始,
//可以令_id=你添加账户的id,以此删除你添加的账户
Toast.makeText(MainActivity.this, "删除了: " + rownum, Toast.LENGTH_LONG).show(); }
else if (v.getId() == R.id.readEventButton) { //读取事件
Cursor eventCursor = getContentResolver().query(Uri.parse(calanderEventURL), null, null, null, null);
if (eventCursor.getCount() > 0) {
eventCursor.moveToLast(); //注意:这里与添加事件时的账户相对应,都是向最后一个账户添加
String eventTitle = eventCursor.getString(eventCursor.getColumnIndex("title"));
Toast.makeText(MainActivity.this, eventTitle, Toast.LENGTH_LONG).show();
}
}
else if (v.getId() == R.id.writeEventButton) {
// 获取要出入的gmail账户的id
String calId = "";
Cursor userCursor = getContentResolver().query(Uri.parse(calanderURL), null, null, null, null);
if (userCursor.getCount() > 0) {
userCursor.moveToLast(); //注意:是向最后一个账户添加,开发者可以根据需要改变添加事件 的账户
calId = userCursor.getString(userCursor.getColumnIndex("_id"));
}
else {
Toast.makeText(this, "没有账户,请先添加账户", 0).show();
return;
} ContentValues event = new ContentValues();
event.put("title", "与苍井空小姐动作交流");
event.put("description", "Frankie受空姐邀请,今天晚上10点以后将在Sheraton动作交流.lol~");
// 插入账户
event.put("calendar_id", calId);
System.out.println("calId: " + calId);
event.put("eventLocation", "地球-华夏"); Calendar mCalendar = Calendar.getInstance();
mCalendar.set(Calendar.HOUR_OF_DAY, 11);
mCalendar.set(Calendar.MINUTE, 45);
long start = mCalendar.getTime().getTime();
mCalendar.set(Calendar.HOUR_OF_DAY, 12);
long end = mCalendar.getTime().getTime(); event.put("dtstart", start);
event.put("dtend", end);
event.put("hasAlarm", 1); event.put(Events.EVENT_TIMEZONE, "Asia/Shanghai"); //这个是时区,必须有,
//添加事件
Uri newEvent = getContentResolver().insert(Uri.parse(calanderEventURL), event);
//事件提醒的设定
long id = Long.parseLong(newEvent.getLastPathSegment());
ContentValues values = new ContentValues();
values.put("event_id", id);
// 提前10分钟有提醒
values.put("minutes", 10);
getContentResolver().insert(Uri.parse(calanderRemiderURL), values); Toast.makeText(MainActivity.this, "插入事件成功!!!", Toast.LENGTH_LONG).show();
}
} //添加账户
private void initCalendars() { TimeZone timeZone = TimeZone.getDefault();
ContentValues value = new ContentValues();
value.put(Calendars.NAME, "yy"); value.put(Calendars.ACCOUNT_NAME, "mygmailaddress@gmail.com");
value.put(Calendars.ACCOUNT_TYPE, "com.android.exchange");
value.put(Calendars.CALENDAR_DISPLAY_NAME, "mytt");
value.put(Calendars.VISIBLE, 1);
value.put(Calendars.CALENDAR_COLOR, -9206951);
value.put(Calendars.CALENDAR_ACCESS_LEVEL, Calendars.CAL_ACCESS_OWNER);
value.put(Calendars.SYNC_EVENTS, 1);
value.put(Calendars.CALENDAR_TIME_ZONE, timeZone.getID());
value.put(Calendars.OWNER_ACCOUNT, "mygmailaddress@gmail.com");
value.put(Calendars.CAN_ORGANIZER_RESPOND, 0); Uri calendarUri = Calendars.CONTENT_URI;
calendarUri = calendarUri.buildUpon()
.appendQueryParameter(CalendarContract.CALLER_IS_SYNCADAPTER, "true")
.appendQueryParameter(Calendars.ACCOUNT_NAME, "mygmailaddress@gmail.com")
.appendQueryParameter(Calendars.ACCOUNT_TYPE, "com.android.exchange")
.build(); getContentResolver().insert(calendarUri, value);
} }

运行结果:  

  

Android 向系统日历中添加事件的更多相关文章

  1. Android向系统日历中添加日程事件

    转自Android向系统日历中添加日程事件 总结 在项目开发中,我们有预约提醒.定时提醒需求时,可以使用系统日历来辅助提醒: 通过向系统日历中写入事件.设置提醒方式(闹钟),实现到时间自动提醒的功能: ...

  2. Android向系统日历添加日程提醒事件

    在项目开发过程中,有时会有预约提醒.定时提醒等需求,这时我们可以使用系统日历来辅助提醒.通过向系统日历中写入事件.设置提醒方式(闹钟),实现到达某个特定的时间自动提醒的功能.这样做的好处是由于提醒功能 ...

  3. 【定制Android系统】Android O 在ROM中添加自己的 so 库(1)——Android.mk 与 Android.bp 的区别【转】

    本文转载自: 版权声明:本文为博主原创文章,转载时请注明原作者及出处.    https://blog.csdn.net/u014248312/article/details/82020204需求:在 ...

  4. Android 获取系统相册中的所有图片

    Android 提供了API可获取到系统相册中的一些信息,主要还是通过ContentProvider 来获取想要的内容. 代码很简单,只要熟悉ContentProvider 就可以了. public ...

  5. Android Studio] Gradle项目中添加JNI生成文件(.so文件)

    转:http://blog.csdn.net/qiujuer/article/details/24209457 为了适应潮流使用Android Studio还是有半年多了! 对于从Eclipse迁移项 ...

  6. js中添加事件 attachEvent 与 addEventListener

    给元素添加事件时,使用js进行实现时产生了疑惑,有关事件浏览器兼容的问题,在此记录如下. <!DOCTYPE html> <html> <head> <met ...

  7. mfc 小程序---在系统菜单中添加菜单项

    1建立一个对话框工程:在dlg类里定义一个菜单指针m_pMenu,在对话框OnInitDialog函数里添加代码: m_pMenu=GetSystemMenu(FALSE);//获取系统菜单的指针 m ...

  8. Android Tips: 在给drawable中添加图片资源时,文件名必须全小写

    在给drawable中添加图片资源时,文件名必须全小写

  9. Android向系统相册中插入图片,相册中会出现两张 一样的图片(只是图片大小不一致)

    向系统相册中插入图片调用此方法时,相册中会出现两张一样的图片 MediaStore.Images.Media.insertImage 一张图片是原图一张图片是缩略图.表现形式为:android4.4. ...

随机推荐

  1. Oracle中的正则表达式

    检查约束 --密码的长度必须在3-6 --年龄必须在1-120 --性别只能是男或女 --电话号码必须满足电话的格式: 手机格式,座机格式 drop table test; select * from ...

  2. pod导入融云路径报错解决办法

    build Settings中搜索sear Search Patchs下点开Library Search Paths 将$(inherited)"$(SRCROOT)/Pods"分 ...

  3. iOS 自带二维码扫描功能的实现

    自从iOS7以后中新增了二维码扫描功能.因此可以在不借助第三方类库的情况下简单的写出二维码的扫描功能: 原生的二维码扫描功能在AVFoundation框架下,所以在使用原生的二维码扫描功能时要先导入A ...

  4. draw: Could not use program error=0x505

    原因:Android的模拟器在ADT中调试运行AVD时速度太慢.也就是说创建的虚拟手机配置太好,电脑带不动. 解决办法:从新创建虚拟手机,是手机配置低一点,具体创建方法如下: 开始先点运行,也就是下图 ...

  5. Calculating Stereo Pairs

    Calculating Stereo Pairs Written by Paul BourkeJuly 1999 Introduction The following discusses comput ...

  6. 深入浅出Symfony2 - 结合MongoDB开发LBS应用

    简介 随着近几年各类移动终端的迅速普及,基于地理位置的服务(LBS)和相关应用也越来越多,而支撑这些应用的最基础技术之一,就是基于地理位置信息的处理.我所在的项目也正从事相关系统的开发,我们使用的是S ...

  7. Redis与Memcached的区别

    传统MySQL+ Memcached架构遇到的问题 实际MySQL是适合进行海量数据存储的,通过Memcached将热点数据加载到cache,加速访问,很多公司都曾经使用过这样的架构,但随着业务数据量 ...

  8. [[UIScreen mainScreen] bounds] 返回的屏幕尺寸不对

    在使用cocos2d-iphone 2.0生成项目的时候,用5s测试时全屏中上下一直有黑条,发现[[UIScreen mainScreen] bounds]返回的屏幕尺寸不是320*568的,而是32 ...

  9. 下拉tableView实现类似微信中带图的灰色背景

    UIView *topView = [[UIView alloc]initWithFrame:CGRectMake(, -, ScreenWidth, )]; UIImageView *iconIma ...

  10. Java多线程简析

    一.线程的状态: 线程共有下面4种状态: 1.新建状态(New): 新创建了一个线程对象,当你用new创建一个线程时,该线程尚未运行. 2.就绪状态(Runnable): 线程对象创建后,其他线程调用 ...