iOS实现日历行程的增删改查

前言

我们可以使用系统提供的EventKit框架来访问和操作用户的日历日程和提醒(虽然日历和提醒是两个独立的app,但是是用同一个框架来处理数据)。同样地,日历和提醒的数据的数据,都是存储在同一个叫做Calendar Database的数据库中:

EventKit不仅能获取已存在的日程和提醒,还能在自己的app中创建、编辑、删除用户的日程和提醒,还能添加提醒、监听变化等。

在iOS10+中,若要访问用户日程或提醒,需要在info.plist中分别添加NSRemindersUsageDescription和NSCalendarsUsageDescription

一. 读取日程

我们可以通过EKEventStore类来对用户的Calendar database进行查询、创建、编辑、删除等操作。我们可以使用条件来获取符合条件的一组日程,也可以用唯一标识来获取指定的一条日程。获取到的每一条日程都是一个EKEvent的实例对象,因此我们修改EKEvent对象的属性即可实现修改日程信息。

1.创建连接

#import <EventKit/EventKit.h>
...
EKEventStore *store = [[EKEventStore alloc] init];

EKEventStore对象的创建和释放会比较耗时,因此我们一般会在app加载后只创建一个event store对象。

2.通过条件获取日程

如果要获取一个时间段内的日程,可以使用EKEventStore对象的eventsMatchingPredicate: 方法。下面代码演示如何获取昨天至一年后的所有日程:

// 获取日历对象
NSCalendar *calendar = [NSCalendar currentCalendar];

// 创建开始时间
NSDateComponents *oneDayAgoComponents = [[NSDateComponents alloc] init];
oneDayAgoComponents.day = -1;
NSDate *oneDayAgo = [calendar dateByAddingComponents:oneDayAgoComponents
                                              toDate:[NSDate date]
                                             options:0];

// 创建结束时间
NSDateComponents *oneYearFromNowComponents = [[NSDateComponents alloc] init];
oneYearFromNowComponents.year = 1;
NSDate *oneYearFromNow = [calendar dateByAddingComponents:oneYearFromNowComponents
                                                   toDate:[NSDate date]
                                                  options:0];

// 创建条件
NSPredicate *predicate = [store predicateForEventsWithStartDate:oneDayAgo endDate:oneYearFromNow calendars:nil];

// 获得符合条件的所有日程
NSArray *events = [store eventsMatchingPredicate:predicate];

3.批量处理日程

如果需要批量处理得到的日程,可以用EKEventStore
实例的enumerateEventsMatchingPredicate:usingBlock:方法(同步方法,为了不阻塞主线程建议在其它线程中执行),例如打印出所有符合条件的日程标题:

[store enumerateEventsMatchingPredicate:predicate usingBlock:^(EKEvent * _Nonnull event, BOOL * _Nonnull stop) {
        NSLog(@"event:%@",event.title);
}];

4.通过唯一标识获取日程

每一个日程都有只读的唯一标识属性eventIdentifier,我们可以通过EKEventStore对象的eventWithIdentifier:方法,传入唯一标识获取指定的一个日程(这个标识是只读属性,由系统指定,可以通过前面的条件查询获取,也可以在创建新的日程时保存这个唯一标识),例如我们已经知道一个日程的eventIdentifier值为”D8574A98-A929-4A92-8E9F-048F46FB5DE7:717c8b40-44e3-31ab-8243-2d5918e266ef”:

EKEvent *event = [store eventWithIdentifier:@"D8574A98-A929-4A92-8E9F-048F46FB5DE7:717c8b40-44e3-31ab-8243-2d5918e266ef"];
NSLog(@"event:%@",event);

二.创建日程

1.通过代码创建

通过EKEvent对象的eventWithEventStore:来创建一个日程,并通过对应的属性编辑日程详细信息,部分属性如:

title - 日程的标题
startDate - 日程的开始日期
endDate - 日程的结束日期
calendar - 日程对应的日历
alarms - 日程的提醒时间
recurrenceRules - 重复规则

实例代码:

EKEvent *event = [EKEvent eventWithEventStore:store];
event.title = @"代码创建的日程";
event.calendar = [store defaultCalendarForNewEvents];
NSCalendar *calendar = [NSCalendar currentCalendar];
NSDateComponents *components = [[NSDateComponents alloc] init];
components.hour = 1;
NSDate *endTime = [calendar dateByAddingComponents:components toDate:[NSDate date] options:0];
event.startDate = [NSDate date];
event.endDate = endTime;
event.notes = @"档期详情:hyaction://hunyu-music";
[event addAlarm:[EKAlarm alarmWithRelativeOffset:-10*60]];

NSError *error;
[store saveEvent:event span:EKSpanFutureEvents commit:YES  error:&error];
if (!error) {
    NSLog(@"添加成功!");
}else{
    NSLog(@"添加失败:%@",error);
}

2.通过系统日历ui添加日程

#import <EventKitUI/EventKitUI.h>
...

EKEventEditViewController *vc = [[EKEventEditViewController alloc] init];
vc.eventStore = store;
vc.editViewDelegate = self;
[self presentViewController:vc animated:YES completion:nil];

通过实现EKEventEditViewDelegate代理事件获得结果:

- (void)eventEditViewController:(EKEventEditViewController *)controller didCompleteWithAction:(EKEventEditViewAction)action{
    NSLog(@"添加日程结果:%zd",action);
    [self dismissViewControllerAnimated:YES completion:nil];
}

三.编辑和删除日程

我们可以通过修改event的属性值来对日程进行编辑,最后需要调用EKEventStore的实例方法saveEvent:span:commit:error:进行持久化保存:

event.title = @"修改后的标题";
NSError *error;
[store saveEvent:event span:EKSpanFutureEvents commit:YES error:&error];

通过EKEventStore的实例方法removeEvent:span:commit:error:.来删除日程:

NSError *error;
[store removeEvent:event span:EKSpanFutureEvents error:&error];

四.添加提醒

我们可以给日程添加本地推送提醒,在指定的时间或地点给用户进行提醒。

1.基于时间的提醒

我们可以通过event的 addAlarm:方法为一个日程添加提醒。我们可以指定一个确切时间或一个相对时间(只能是日程开始时间之前)。通过removeAlarm: 方法可将提醒移除。
如在开始时间前10分钟提醒:

[event addAlarm:[EKAlarm alarmWithRelativeOffset:-10*60]];

2.基于地理位置的提醒

我们可以设定当用户进入或离开指定的地理位置区域时,触发日程提醒。例如当用户离开公司,提醒用户需要到超市购买日用品,作为开发者,需要确定一个经纬度以及一个半径范围。

EKAlarm *alarm = [[EKAlarm alloc] init];
    EKStructuredLocation *location = [EKStructuredLocation
                                      locationWithTitle:@"Current Location"];
location.geoLocation = [[CLLocation alloc] initWithLatitude:23.1754700000 longitude:113.4147400000];
alarm.structuredLocation = location;
alarm.proximity = EKAlarmProximityEnter;
[event addAlarm:alarm];

参考苹果官方文档

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

时间: 2021-08-23

iOS自定义日历控件的简单实现过程

因为程序要求要插入一个日历控件,该空间的要求是从当天开始及以后的六个月内的日历,上网查资料基本上都说只要获取两个条件(当月第一天周几和本月一共有多少天)就可以实现一个简单的日历,剩下的靠自己的简单逻辑就OK了,下面开始自己从开始到完成的整个过程 1.首先做NSDate类目,扩展一些方法让日期之间转换更加方便 #import <Foundation/Foundation.h> @interface NSDate (LYWCalendar) #pragma mark - 获取日 - (NSInte

iOS实现一个简易日历代码

日历一般都是用UICollectionView进行开发的,相关demo也很多,这里就讲一个我最近写的玩的demo,由于时间原因没来得及加年历和周历,一个月历的小demo,随着月份天数的不同,自动改变日历的高. 代理部分: @protocol KJCalendarDelegate <NSObject> /** 随着每个月的天数不一样而改变高度 @param height 日历高度 */ - (void)calendarViewHeightChange:(CGFloat)height; /** 当

iOS 简约日历控件EBCalendarView的实现代码

本文介绍了iOS 简约日历控件EBCalendarView的实现代码,分享给大家,具体如下: EBCalendarView日历控件,调用简单,代码简洁. github地址:https://github.com/woheduole/EBCalendarView 效果图 调用示例 EBCalendarView *calendarView = [[EBCalendarView alloc] initWithFrame:CGRectMake(0, 64, CGRectGetWidth(self.view

iOS My97DatePicker日历使用详解

一.效果图 二.工程图 三.代码 RootViewController.h #import <UIKit/UIKit.h> @interface RootViewController : UIViewController @end RootViewController.m #import "RootViewController.h" //加入头文件 #import "My97DatePicker.h" @interface RootViewControl

iOS实现日历翻页动画

本文我主要描述两方面: 1.日历(简单描述原理) 2.翻页动画(重点) 最终的效果如下图:     图中沿四个对角的翻页动画,代表对应方向手势的滑动 1. 日历 要实现一个日历,其实原理很简单,我们只要知道三个数据: 1.今天是哪一天 2.这个月的第一天是星期几(哪天) 3.这个月总共有多少天 根据这个三个数据,就可以把得到的日期显示在日历上了,至于日历用什么来显示,我个人比较喜欢用UICollectionView,一个cell代表一天,当然也可以用很多个label,button来显示. 1.获

iOS Swift开发之日历插件开发示例

本文介绍了iOS Swift开发之日历插件开发示例,分享给大家,具体如下: 效果图 0x01 如何获取目前日期 关于日期,苹果给出了 Date 类,初始化一个 Date 类 let date = Date() 打印出来就是当前系统的日期和时间 那么如何单独获得当前年份,月份呢? var date: [Int] = [] let calendar: Calendar = Calendar(identifier: .gregorian) var comps: DateComponents = Dat

iOS Swift控制器转场动画示例代码

前言 在IOS开发中,我们model另外一个控制器的时候,一般都使用默认的转场动画.本文将给大家详细介绍关于iOS Swift控制器转场动画的相关内容,分享出来供大家参考学习,下面话不多说了,来一起看看详细的介绍吧. 返回效果也可更改 四种转场动画 1. move:源图片位置移动到目标图片位置: 2. circle:根据源控件大小创建圆形或者椭圆形path路径,放大展示目标: 3. tier:源左右,目标由小到大缩放: 4. middle:源的中心点开始放大,返回是缩回到中心. 代码解析 给UI

IOS Swift 开发QRCore(二维码)实例详解

1.搭个界面 2.写代码 // // ViewController.swift // GeneratorQRCode // // Created by targetcloud on 2016/12/3. // Copyright © 2016年 targetcloud. All rights reserved. // import UIKit class ViewController: UIViewController { @IBOutlet weak var qrImg: UIImageVie

iOS Swift创建代理协议的多种方式示例

前言 本文主要给大家介绍了iOS Swift创建代理协议的各种方式,分享出来供大家参考学习,下面话不多说了,来一起看看详细的介绍吧. 假如有一个类为 LXFView,现在要为这个类创建一个代理协议,我们该如何做呢? 首先,代理协议的命名方式:类名 + Delegatev protocol LXFViewDelegate { func view(_ view: LXFView) } 当我们创建的协议遵守其它协议的情况下,只是这样写并不会报错,接下来我们在LXFView中添加一个代理属性,为避免循环

Android开发之日历CalendarView用法示例

本文实例讲述了Android开发之日历CalendarView用法.分享给大家供大家参考,具体如下: 简介: 1.CalendarView是安卓自带的一个日历控件 2.在主活动中 通过设置setOnDataChangeListener() 来为其添加监听事件 可在其中获得 洪湖所选择的年月日的 详细信息 实例: 基本设置方法: 1. 日历的整体背景颜色 android:selectedWeekBackgroundColor="#aff" 2. 月份选择部分的背景色 android:fo

IOS多线程开发之线程的状态

大家都知道,在开发过程中应该尽可能减少用户等待时间,让程序尽可能快的完成运算.可是无论是哪种语言开发的程序最终往往转换成汇编语言进而解释成机器码来执行.但是机器码是按顺序执行的,一个复杂的多步操作只能一步步按顺序逐个执行.改变这种状况可以从两个角度出发:对于单核处理器,可以将多个步骤放到不同的线程,这样一来用户完成UI操作后其他后续任务在其他线程中,当CPU空闲时会继续执行,而此时对于用户而言可以继续进行其他操作:对于多核处理器,如果用户在UI线程中完成某个操作之后,其他后续操作在别的线程中继续

iOS程序开发之使用PlaceholderImageView实现优雅的图片加载效果

说明 1. PlaceHolderImageView基于SDWebImage编写 2. 给定一个图片的urlString,以及一个placeholderImage就可以优雅的显示图片加载效果 效果 源码 PlaceholderImageView.h/.m // // PlaceholderImageView.h // SDWebImageViewPlaceHorder // // Created by YouXianMing on 16/9/14. // Copyright © 2016年 Yo

IOS游戏开发之五子棋OC版

先上效果图 - 功能展示 - 初高级棋盘切换效果 实现思路及主要代码详解 1.绘制棋盘 利用Quartz2D绘制棋盘.代码如下 - (void)drawBackground:(CGSize)size{ self.gridWidth = (size.width - 2 * kBoardSpace) / self.gridCount; //1.开启图像上下文 UIGraphicsBeginImageContext(size); //2.获取上下文 CGContextRef ctx = UIGraph

iOS swift 总结NavigationController出现问题及解决方法

IOS swift 总结NavigationController出现问题及解决方法 最近用Swift语言做了一些iOS项目,颇有些心得,记下一些深刻的问题造福自己,服务大家 1.以NavigationController做为容器后状态栏的字体颜色就会不在受系统的控制,要在NavigationController中的根ViewController中设置方可生效,代码如下: self.navigationController!.navigationBar.barStyle = UIBarStyle.

IOS Swift基础之switch用法详解

IOS  Swift基础之switch用法详解 概述 Swift中的switch语句与Java等语言中的switch有很大的相似点,但是也有不同的地方,并且更加灵活. Swift中switch的case语句中不需要添加break Swift中需要考虑所有情况,default是必要的. case分支可以添加多个条件,用,分割 case不局限与常量,可以使使用范围 switch里可以使用元组 switch默认不需要添加break,执行一个case之后就跳出语句,如果想要继续下面的语句可以使用fall