如何从 message:// URL in OSX Swift 获取邮件主题

How to get e-mail subject from message:// URL in OSX Swift

我有一个桌面应用程序可以从拖放粘贴板接收电子邮件 URLs("message://" 方案),我想从相关消息中获取主题。到目前为止,我唯一的线索是 QuickLook 库可能会给我一个信息对象,我可以从中检索此信息。

由于 QuickLook API 目前似乎在不断变化,并且大多数示例都显示了如何在 iOS 中使用它,所以我根本找不到设置 [=24= 的方法] 对象使用 URL 并从那里获取信息。

我想避免将我的项目设置为 QuickLook 插件,或设置整个预览窗格/视图脚手架;目前我只想在 QuickLook 开始显示之前弄清楚它加载了什么,但我无法理解 Apple 要我在这里实现的范例。

XCode 7.3.1.

事实证明,我将 draggingInfo.draggingPasteboard().types 的内容误解为仅包含一种信息类型的分层列表(在本例中为 URL)。 必须订阅拖动事件类型 kUTTypeMessage as String 并使用 stringForType("public.url-name")

从粘贴板中检索电子邮件主题

编辑:请注意当前的 Mail.app 将 有时 在您拖动电子邮件线程时创建一堆邮件。虽然上面的方法仍然可以获取堆栈的主题,但是拖动信息中没有 URL 并且由于也没有可用的 Message-ID 列表,我不得不求助于抓取用户的 mbox 目录:

        // See if we can resolve e-mail message meta data
        if let mboxPath = pboard.stringForType("com.apple.mail.PasteboardTypeMessageTransfer") {
            if let automatorPlist = pboard.propertyListForType("com.apple.mail.PasteboardTypeAutomator") {
                // Get the latest e-mail in the thread
                if let maxID = (automatorPlist.allObjects.flatMap({ [=10=]["id"]! }) as AnyObject).valueForKeyPath("@max.self") as? Int {
                    // Read its meta data in the background
                    let emailItem = draggingEmailItem
                    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0)) {
                        // Find the e-mail file
                        if let path = Util.findEmlById(searchPath: mboxPath, id: maxID) {
                            // Read its contents
                            emailItem.properties = Util.metaDataFromEml(path)
                            dispatch_async(dispatch_get_main_queue(), {
                                // Update UI
                            });
                        }
                    }
                }
            }
        }

实用函数:

/* Searches the given path for <id>.eml[x] and returns its URL if found
 */
static func findEmlById(searchPath searchPath: String, id: Int)-> NSURL? {
    let enumerator = NSFileManager.defaultManager().enumeratorAtPath(searchPath)
    while let element = enumerator?.nextObject() as? NSString {
        switch (element.lastPathComponent, element.pathExtension) {
            case (let lpc, "emlx") where lpc.hasPrefix("\(id)"):
                return NSURL(fileURLWithPath: searchPath).URLByAppendingPathComponent(element as String)!
            case (let lpc, "eml") where lpc.hasPrefix("\(id)"):
                return NSURL(fileURLWithPath: searchPath).URLByAppendingPathComponent(element as String)!
            default: ()
        }
    }
    return nil
}

/* Reads an eml[x] file and parses it, looking for e-mail meta data
 */
static func metaDataFromEml(path: NSURL)-> Dictionary<String, AnyObject> {

    // TODO Support more fields

    var properties: Dictionary<String, AnyObject> = [:]
    do {
        let emlxContent = try String(contentsOfURL: path, encoding: NSUTF8StringEncoding)
        // Parse message ID from "...\nMessage-ID: <...>"
        let messageIdStrMatches = emlxContent.regexMatches("[\n\r].*Message-ID:\s*<([^\n\r]*)>")
        if !messageIdStrMatches.isEmpty {
            properties["messageId"] = messageIdStrMatches[0] as String
        }
    }
    catch {
        print("ERROR: Failed to open emlx file")
    }
    return properties
}

注意:如果您的应用是沙盒应用,您需要将 com.apple.security.temporary-exception.files.home-relative-path.read-only 权利设置为一个数组,其中包含一个字符串:/Library/