iOS 11 拖放:拖放到标准日历应用程序上

Tho*_*rie 2 calendar drag-and-drop ios11

我正在尝试将 iOS 11 的拖放功能集成到 Xcode 9 beta 的应用程序中。我对创建一个标准日历应用程序可以理解的 NSItemProvider 很感兴趣。我的拖动从 UITableView 开始,因此只实现了一个 dragDelegate 方法。

到目前为止,我已经尝试了以下方法:

    let text = "Rendez-vous at \(clientName)"
    let data = text.data(using: .utf8)

    let itemProvider = NSItemProvider()
    itemProvider.registerDataRepresentation(forTypeIdentifier: kUTTypeCalendarEvent as String, visibility: .all) { completion in
        completion(data, nil)
        return nil
    }

    let dragItem = UIDragItem(itemProvider: itemProvider)
Run Code Online (Sandbox Code Playgroud)

并且还尝试使用类型标识符 kUTTypePlainText。不走运,日历应用程序不会注册下降。

我找不到任何关于此的官方文档。我希望日历应用程序正在寻找一些标准日历数据,这不仅限于标准应用程序相互通信。例如,您可以将笔记应用程序中的文本拖放到日历中以创建事件。

有谁知道我可以尝试什么?

提前致谢。

Luc*_*cas 5

TL;DR:让默认 iOS 日历应用程序识别符合NSItemProviderWriting协议的自定义对象的秘密成分是"com.apple.ical.ics"类型标识符并将自定义对象导出为 iCalendar 字符串 ( vEvent)。


我已经玩了几天了,我终于设法让它工作了!我会尽量分解我的步骤。

  1. 创建一个符合NSItemProviderWriting的自定义对象:
    • 为了符合协议,你需要实现 writableTypeIdentifiersForItemProviderloadData(withTypeIdentifier:, forItemProviderCompletionHandler:)
    • 标识符数组是您可以导出的类型标识符列表,按保真度顺序,最高保真度在前。因此,将您的自定义类型标识符放在首位,然后再包括在内"com.apple.ical.ics"
    • loadData上面提到的函数将尝试找到拖动目的地(本场景中的日历应用程序)支持且您的应用程序也支持的标识符,尝试首先获得最高保真度。

例如:

public static var writableTypeIdentifiersForItemProvider: [String] {
    return ["com.yourcompany.myEvent", "com.apple.ical.ics"]
}

public func loadData(withTypeIdentifier typeIdentifier: String, forItemProviderCompletionHandler completionHandler: @escaping (Data?, Error?) -> Void) -> Progress? {
    switch typeIdentifier {
    case "com.yourcompany.myEvent": // Your custom handling goes here
    case "com.apple.ical.ics": completionHandler(createVEvent(), nil)
    default: completionHandler(nil, YourAppError.someCustomError)
    }

    return nil
}
Run Code Online (Sandbox Code Playgroud)
  1. vEvent使用您的自定义事件信息创建一个 iCalendar字符串:
    • 此步骤与WWDC 2017 的Session 227 中的演示非常相似,它们vCard为自定义联系人创建字符串。从该会话签出演示项目。
    • 本质上,日历应用程序支持几种特定类型的类型标识符,iCal 格式就是其中之一。因此,您只需要创建一个辅助方法将您的自定义事件类型转换为iCalendar字符串。
    • 我不会详细介绍vEvents 但这里这里有很多很好的信息。

这是一个建议:

private func createVEvent() -> Data? {
    let today = Date()

    let dateFormatter = DateFormatter()
    dateFormatter.dateFormat = "yyyyMMdd'T'HHmmss'Z'"
    dateFormatter.timeZone = TimeZone(abbreviation: "UTC")

    let startDateString = dateFormatter.string(from: startDate)
    let endDateString = dateFormatter.string(from: endDate)
    let todayString = dateFormatter.string(from: today)

    var vCalendarText = "BEGIN:VCALENDAR\n"
    vCalendarText +=    "VERSION:2.0\n"
    vCalendarText +=    "PRODID:-//Your Company//App Name//EN\n"
    vCalendarText +=    "BEGIN:VEVENT\n"

    vCalendarText +=    "UID:\(identifier.uuidString)\n"
    vCalendarText +=    "SUMMARY:\(name)\n"
    vCalendarText +=    "DTSTAMP:\(todayString)\n"
    vCalendarText +=    "DTSTART:\(startDateString)\n"
    vCalendarText +=    "DTEND:\(endDateString)\n"

    vCalendarText +=    "END:VEVENT\n"
    vCalendarText +=    "END:VCALENDAR"

    return vCalendarText.data(using: .utf8)
}
Run Code Online (Sandbox Code Playgroud)

? 类似于createVCard()我之前提到的那个会话演示中的功能......

就是这样!这几乎就是你所需要的。困难的部分是知道日历应用程序支持什么标识符以及如何制作vEvent. 感谢quiker找到支持的标识符。

最后一个提示:您可以在此处使用这个漂亮的工具验证您的 iCalendar 字符串。

请注意,这仅允许删除您的自定义对象。事件从日历拖到您的应用程序是另一回事,但您可以从我的提示中弄清楚。:)

快乐编码!