2

我正在使用 Swift 4.1。我想编写一个函数来收集 iOS 日历应用程序中所有日历中的所有事件。感谢stackoverflow上的这个答案:How to get all Events out of a Calendar (Swift)我能够自己编写class并调用它Cale。请看这个:

import UIKit
import EventKit

class Cale {

    private func createDate(year: Int) -> Date {
        var components = DateComponents()
        components.year = year
        components.timeZone = TimeZone(secondsFromGMT: 0)

        return Calendar.current.date(from: components)!
    }

    private let eventStore = EKEventStore()

    private func get() {
        let calendars = eventStore.calendars(for: .event)

        for calendar in calendars {
            // This checking will remove Birthdays and Hollidays callendars
            guard calendar.allowsContentModifications else {
                continue
            }

            let start = createDate(year: 2016)
            let end = createDate(year: 2025)

            print("start: \(start)")
            print("  end: \(end)")

            let predicate = eventStore.predicateForEvents(withStart: start, end: end, calendars: [calendar])

            print("predicate: \(predicate)")

            let events = eventStore.events(matching: predicate)

            for event in events {
                print("    title: \(event.title!)")
                print("startDate: \(event.startDate!)")
                print("  endDate: \(event.endDate!)")
            }
        }
    }

    func checkStatusAndGetAllEvents() {
        let currentStatus = EKEventStore.authorizationStatus(for: EKEntityType.event)

        switch currentStatus {
        case .authorized:
            //print("authorized")
            self.get()
        case .notDetermined:
            //print("notDetermined")
            eventStore.requestAccess(to: .event) { accessGranted, error in
                if accessGranted {
                    self.get()
                } else {
                    print("Change Settings to Allow Access")
                }
            }
        case .restricted:
            print("restricted")
        case .denied:
            print("denied")
        }
    }
}

非常简单的类,您可以使用它,它可以工作,但有一个例外。主要功能get()在此功能中,我根据两个日期创建谓词:开始和结束。如您所见,开始日期是:

2016-01-01 00:00:00 +0000

结束日期是:

2025-01-01 00:00:00 +0000

但是如果我们运行程序,我们会看到谓词是这样的:

CADEventPredicate 开始:01/01/2016, 03:00; 结束:01/01/2020, 03:00; 卡尔斯:(2)

仅从2016年到2020年,4年!我已经在不同的日期对其进行了测试,但我可以得到最大间隔为 4 年的谓词。这意味着,它不会给我所有的事件!所以问题是:如何从日历中获取所有事件?如果可能,不使用日期!

感谢您对未来的任何帮助或建议!

4

2 回答 2

2

如果有人仍然想知道,Apple 有意将其限制为四年。

predicateForEvents(withStart:end:calendars:)

出于性能原因,此方法仅匹配四年时间跨度内的那些事件。如果 startDate 和 endDate 之间的日期范围大于四年,则会缩短到前四年。

如果您想要一个更长的范围,我想您必须将其包装在您自己的函数中以将其分成四年的块。

于 2020-05-31T20:08:40.897 回答
0

花了这么多时间后,我找到了解决问题的方法。我的问题如下,

event start date 2019-03-15 09:00:00 +0000
event end date 2019-03-15 14:00:00 +0000

谓词如下,

CADEventPredicate start:15/03/19, 1:30 PM; end:15/03/19, 7:30 PM; cals:(null)

因为我来自印度,所以谓词将 GMT+5:30 添加到我的实际活动开始和结束时间。我从堆栈溢出中获得了一种方法,可以转换为本地和全球时区,如下所示。

    extension Date {
    // Convert local time to UTC (or GMT)
    func toGlobalTime() -> Date {
        let timezone = TimeZone.current
        let seconds = -TimeInterval(timezone.secondsFromGMT(for: self))
        return Date(timeInterval: seconds, since: self)
    }

    // Convert UTC (or GMT) to local time
    func toLocalTime() -> Date {
        let timezone = TimeZone.current
        let seconds = TimeInterval(timezone.secondsFromGMT(for: self))
        return Date(timeInterval: seconds, since: self)
    }
}

在将开始和结束日期与时间一起传递给谓词时,我正在转换为 globalTimeZone,如下所示。

let predicate = eventStore.predicateForEvents(withStart: eventStartDateTime.toGlobalTime(), end: eventEndDateTime.toGlobalTime(), calendars: nil)

如果你没有时间,你只有开始和结束日期,然后使用如下谓词,

event start date 2019-03-15 00:00:00 +0000
event end date 2019-03-15 00:00:00 +0000

let predicate = eventStore.predicateForEvents(withStart: eventStartDateTime, end: eventEndDateTime, calendars: nil)

因为如果您没有时间并转换为 globalTimeZone,您可能会在删除事件时遇到问题。

我希望它会帮助一些人。

于 2019-03-15T07:24:17.600 回答