iOS—EventKit实现app日程同步到本地日历

最近在做日历同步的需求,趁着已经提测,整理一些坑坑洼洼的地方。

和产品经理一起研究了一下市面上该功能的实现,绝大多数都是把本地手机日历的日程单向同步到自己app中去。而我们产品经理反其道而行,同步依然是单向的,但是是要从app的日程业务中把数据同步到日历中去。其实这样的需求对于我们本身的业务来讲是说得通的,而且不需要通过push消息就能让用户自己去定义提醒的时间。但是在预研阶段就发现了问题,所以正式开发以前,给她们总结了一些解决思路和不可避免的问题。


问题一:EKEvent对象的eventIdentifier属性是只读的

@property(null_unspecified, nonatomic, readonly) NSString *eventIdentifier;

看到readonly我瞬间明白了其他APP为什么是手机本地日历单向同步到应用中,因为业务对象的唯一标识不能和本地的日历建立有效的连接。这样就会导致如果我们对APP内日程的数据进行的更改或者删除操作,本地日历中的日程就没有办法同步更新,因为匹配不上~

解决方案:将我们的日程id带入到本地日程中去。

eventIdentifier用不了了,只能找其他属性,并且是string类型,最后决定用url,当然做了一些其他处理。不过问题依然是有的,比如该属性是暴露给用户的,用户可以自己去编辑。那么就会导致有新的编辑过的数据过来以后,我只能把该日程处理成新增日程。


问题二:获取本地日历中的日程数据数据量可能会很大,导致与服务端返回的新数据进行匹配的时候双重for循环影响效率(虽然用户感知不到)

解决方案:使用allowsContentModifications属性

-(NSMutableArray*)getLocalSchedules{

    NSMutableArray *allowsModifyEvents = [NSMutableArray array];

    NSDate *startDate = startdate;

    NSDate *endDate = enddate;

    NSPredicate *pre = [self.store predicateForEventsWithStartDate:startDate endDate:endDate calendars:nil];

    NSArray *events = [self.store eventsMatchingPredicate:pre];

    events = [eventssortedArrayUsingSelector:@selector(compareStartDateWithEvent:)];

    for(EKEvent*eventinevents) {

        if (event.calendar.allowsContentModifications == YES) {

            [allowsModifyEventsaddObject:event];

        }

    }

    returnallowsModifyEvents;

}

是的,由于我们手动添加的数据都是可以手动编辑的,所以event的allowsContentModifications这一只读属性刚好可以用到。可以减少很多系统日历自带的event对象,比如节假日、节气等等。


问题三:日程需要分账户

解决方案:使用EKCalendar

//为日程添加日历分类

EKSource*myLocalSource =nil;

EKCalendar*myLocalCalendar;

NSArray *calendars = [self.store calendarsForEntityType:EKEntityTypeEvent];

NSString*calendarTitle = [NSStringstringWithFormat:@"%@",userName];//这里使用username是因为我们的APP可以进行用户切换,产品经理希望不同用户的日程保存到不同的分类下。但是由于EKCalendar的calendarIdentifier属性也是只读的,所以目前只能用username进行本地和服务端返回数据的匹配。

//日历优先取本地已有的

for(EKCalendar*calendar in calendars) {

        if([calendar.title isEqualToString: calendarTitle]) {

            myLocalCalendar = calendar;

            break;

        }

}

    //本地没有则新建

    if(myLocalCalendar ==nil) {

        //先取已经存在本地的个人source

        for(EKSource*calendarSource in self.store.sources) {

            if(calendarSource.sourceType==EKSourceTypeLocal) {

                myLocalSource = calendarSource;

                break;

            }

        }

        //EKSource的类型有很多种,Local类型在用户打开日历的cloud同步以后会变成CalDAV类型

        if(myLocalSource ==nil) {

            for(EKSource*calendarSource in self.store.sources) {

                if(calendarSource.sourceType==EKSourceTypeCalDAV&&

                    [calendarSource.titleisEqualToString:@"iCloud"]) {//该判断条件不知道有没有更好的方案,也是在网上找到的。

                    myLocalSource = calendarSource;

                    break;

                }

            }

        }


        myLocalCalendar = [EKCalendar calendarForEntityType:EKEntityTypeEvent eventStore:self.store];

        myLocalCalendar.title= calendarTitle;

        CGColorSpaceRef rgbSapceRef = CGColorSpaceCreateDeviceRGB();

        CGFloatrgbComponents[] = {1,0,0,1};// RGBA 颜色组件

        CGColorRefrgbColorRef =CGColorCreate(rgbSapceRef, rgbComponents);

        myLocalCalendar.CGColor= rgbColorRef;

        myLocalCalendar.source= myLocalSource;

        NSError*err;

        [self.store saveCalendar:myLocalCalendar commit:YES error:&err];

        if (err) {//新建日历失败的话则将日程的日历分类到默认日历下

            [newEventsetCalendar:[self.store defaultCalendarForNewEvents]];

        }else{

            [newEventsetCalendar:myLocalCalendar];

        }

    }else{

        [newEventsetCalendar:myLocalCalendar];

    }


结论:EventKit框架中有太多的只读属性的对象,其实正确的做法是把已经存到本地的EKEvent对象的eventIdentifier属性返回给我们自己的服务器,让后台与业务日程进行关联。但是目前该方案由于种种原因没有最终拍死,所以只能原生负责第一期的需求先实现。后面再慢慢埋坑吧~基本上一些重要的代码也就上面一点点,就不上demo了。其实我蛮喜欢做这样的需求的,没有UI\UE。后台或者前端返回来数据我就处理数据就好了。不到一天时间就能搞定,不过前期一定要做好预研工作,把问题尽快的暴露给项目组,然后大家一起讨论解决方案,后面才能水到渠成。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,242评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,769评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,484评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,133评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,007评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,080评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,496评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,190评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,464评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,549评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,330评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,205评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,567评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,889评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,160评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,475评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,650评论 2 335

推荐阅读更多精彩内容