美文网首页
iOS—EventKit实现app日程同步到本地日历

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

作者: 程序媛的程 | 来源:发表于2018-12-21 12:33 被阅读0次

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

    和产品经理一起研究了一下市面上该功能的实现,绝大多数都是把本地手机日历的日程单向同步到自己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。后台或者前端返回来数据我就处理数据就好了。不到一天时间就能搞定,不过前期一定要做好预研工作,把问题尽快的暴露给项目组,然后大家一起讨论解决方案,后面才能水到渠成。

    相关文章

      网友评论

          本文标题:iOS—EventKit实现app日程同步到本地日历

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