mirror of
https://github.com/Ranchero-Software/NetNewsWire
synced 2025-08-12 06:26:36 +00:00
261 lines
10 KiB
Swift
261 lines
10 KiB
Swift
//
|
|
// Feed+Scriptability.swift
|
|
// Evergreen
|
|
//
|
|
// Created by Olof Hellman on 1/10/18.
|
|
// Copyright © 2018 Olof Hellman. All rights reserved.
|
|
//
|
|
|
|
import Foundation
|
|
import RSParser
|
|
import Account
|
|
import Data
|
|
|
|
@objc(ScriptableFeed)
|
|
class ScriptableFeed: NSObject, UniqueIdScriptingObject, ScriptingObjectContainer{
|
|
|
|
let feed:Feed
|
|
let container:ScriptingObjectContainer
|
|
|
|
init (_ feed:Feed, container:ScriptingObjectContainer) {
|
|
self.feed = feed
|
|
self.container = container
|
|
}
|
|
|
|
@objc(objectSpecifier)
|
|
override var objectSpecifier: NSScriptObjectSpecifier? {
|
|
let scriptObjectSpecifier = self.container.makeFormUniqueIDScriptObjectSpecifier(forObject:self)
|
|
return (scriptObjectSpecifier)
|
|
}
|
|
|
|
@objc(scriptingSpecifierDescriptor)
|
|
func scriptingSpecifierDescriptor() -> NSScriptObjectSpecifier {
|
|
return (self.objectSpecifier ?? NSScriptObjectSpecifier() )
|
|
}
|
|
|
|
// MARK: --- ScriptingObject protocol ---
|
|
|
|
var scriptingKey: String {
|
|
return "feeds"
|
|
}
|
|
|
|
// MARK: --- UniqueIdScriptingObject protocol ---
|
|
|
|
// I am not sure if account should prefer to be specified by name or by ID
|
|
// but in either case it seems like the accountID would be used as the keydata, so I chose ID
|
|
|
|
var scriptingUniqueId:Any {
|
|
return feed.feedID
|
|
}
|
|
|
|
// MARK: --- ScriptingObjectContainer protocol ---
|
|
|
|
var scriptingClassDescription: NSScriptClassDescription {
|
|
return self.classDescription as! NSScriptClassDescription
|
|
}
|
|
|
|
|
|
// MARK: --- Create Element Handlers ---
|
|
|
|
class func parsedFeedForURL(_ urlString:String, _ completionHandler: @escaping (_ parsedFeed: ParsedFeed?) -> Void) {
|
|
guard let url = URL(string: urlString) else {
|
|
completionHandler(nil)
|
|
return
|
|
}
|
|
InitialFeedDownloader.download(url) { (parsedFeed) in
|
|
completionHandler(parsedFeed)
|
|
}
|
|
}
|
|
|
|
class func urlForNewFeed(arguments:[String:Any]) -> String? {
|
|
var url:String?
|
|
if let withDataParam = arguments["ObjectData"] {
|
|
if let objectDataDescriptor = withDataParam as? NSAppleEventDescriptor {
|
|
url = objectDataDescriptor.stringValue
|
|
}
|
|
} else if let withPropsParam = arguments["ObjectProperties"] as? [String:Any] {
|
|
url = withPropsParam["url"] as? String
|
|
}
|
|
return url
|
|
}
|
|
|
|
class func accountAndFolderForNewFeed(appleEvent:NSAppleEventDescriptor?) -> (Account, Folder?) {
|
|
var account = AccountManager.shared.localAccount
|
|
var folder:Folder? = nil
|
|
if let appleEvent = appleEvent {
|
|
var descriptorToConsider:NSAppleEventDescriptor?
|
|
if let insertionLocationDescriptor = appleEvent.paramDescriptor(forKeyword:keyAEInsertHere) {
|
|
print("insertionLocation : \(insertionLocationDescriptor)")
|
|
// insertion location can be a typeObjectSpecifier, e.g. 'in account "Acct"'
|
|
// or a typeInsertionLocation, e.g. 'at end of folder "
|
|
if (insertionLocationDescriptor.descriptorType == "insl".FourCharCode()) {
|
|
descriptorToConsider = insertionLocationDescriptor.forKeyword("kobj".FourCharCode())
|
|
} else if ( insertionLocationDescriptor.descriptorType == "obj ".FourCharCode()) {
|
|
descriptorToConsider = insertionLocationDescriptor
|
|
}
|
|
} else if let subjectDescriptor = appleEvent.attributeDescriptor(forKeyword:"subj".FourCharCode()) {
|
|
descriptorToConsider = subjectDescriptor
|
|
}
|
|
|
|
if let descriptorToConsider = descriptorToConsider {
|
|
guard let newContainerSpecifier = NSScriptObjectSpecifier(descriptor:descriptorToConsider) else {return (account, folder)}
|
|
let newContainer = newContainerSpecifier.objectsByEvaluatingSpecifier
|
|
if let scriptableAccount = newContainer as? ScriptableAccount {
|
|
account = scriptableAccount.account
|
|
} else if let scriptableFolder = newContainer as? ScriptableFolder {
|
|
if let folderAccount = scriptableFolder.folder.account {
|
|
folder = scriptableFolder.folder
|
|
account = folderAccount
|
|
}
|
|
}
|
|
}
|
|
print("found account : \(account)")
|
|
print("found folder : \(folder)")
|
|
}
|
|
return (account, folder)
|
|
}
|
|
|
|
class func handleCreateElement(command:NSCreateCommand) -> Any? {
|
|
let appleEventManager = NSAppleEventManager.shared()
|
|
if let receivers = command.receiversSpecifier {
|
|
print("receivers : \(receivers)")
|
|
}
|
|
if let evaluatedReceivers = command.evaluatedReceivers {
|
|
print("evaluatedReceivers : \(evaluatedReceivers)")
|
|
}
|
|
if let evaluatedArguments = command.evaluatedArguments {
|
|
print("evaluatedArguments : \(evaluatedArguments)")
|
|
}
|
|
if let directObject = command.directParameter {
|
|
print("directObject : \(directObject)")
|
|
}
|
|
if let appleEvent = command.appleEvent { // keyDirectObject
|
|
print("appleEvent : \(appleEvent)")
|
|
if let subjectDescriptor = appleEvent.attributeDescriptor(forKeyword:"subj".FourCharCode()) {
|
|
print("subjectDescriptor : \(subjectDescriptor)")
|
|
let subjectObjectSpecifier = NSScriptObjectSpecifier(descriptor:subjectDescriptor)
|
|
let subjects = subjectObjectSpecifier?.objectsByEvaluatingSpecifier
|
|
print("resolvedSubjects : \(subjects)")
|
|
}
|
|
}
|
|
let commandDescription = command.commandDescription
|
|
print("commandDescription : \(commandDescription)")
|
|
|
|
let (account, folder) = self.accountAndFolderForNewFeed(appleEvent:command.appleEvent)
|
|
let scriptableAccount = ScriptableAccount(account)
|
|
guard let arguments = command.arguments else {return nil}
|
|
guard let newObjectClass = arguments["ObjectClass"] as? Int else {return nil}
|
|
guard (newObjectClass.FourCharCode() == "Feed".FourCharCode()) else {return nil}
|
|
guard let url = self.urlForNewFeed(arguments:arguments) else {return nil}
|
|
|
|
if let existingFeed = account.existingFeed(withURL:url) {
|
|
return ScriptableFeed(existingFeed, container:scriptableAccount)
|
|
}
|
|
|
|
// at this point, we have to download the feed and parse it.
|
|
// RS Parser does the callback for the download on the main thread
|
|
// because we can't wait here (on the main thread, maybe) for the callback, we have to return from this function
|
|
// generally, the means handling the appleEvent is over, but to prevent the apple event from returning
|
|
// we call suspendExecution here. When we get the callback, we can resume execution
|
|
command.suspendExecution()
|
|
|
|
self.parsedFeedForURL(url, { (parsedFeedOptional) in
|
|
if let parsedFeed = parsedFeedOptional {
|
|
let titleFromFeed = parsedFeed.title
|
|
let titleFromArgs = arguments["name"] as? String
|
|
|
|
guard let feed = account.createFeed(with: titleFromFeed, editedName: titleFromArgs, url: url) else {
|
|
command.resumeExecution(withResult:nil)
|
|
return
|
|
}
|
|
account.update(feed, with:parsedFeed, {})
|
|
|
|
// add the feed, puttin git in a folder if needed
|
|
if account.addFeed(feed, to: folder) {
|
|
NotificationCenter.default.post(name: .UserDidAddFeed, object: self, userInfo: [UserInfoKey.feed: feed])
|
|
}
|
|
|
|
let resolvedKeyDictionary = command.resolvedKeyDictionary
|
|
print("resolvedKeyDictionary : \(resolvedKeyDictionary)")
|
|
let scriptableFeed = ScriptableFeed(feed, container:ScriptableAccount(account))
|
|
command.resumeExecution(withResult:scriptableFeed.objectSpecifier)
|
|
} else {
|
|
command.resumeExecution(withResult:nil)
|
|
}
|
|
})
|
|
return nil
|
|
}
|
|
|
|
|
|
|
|
|
|
// MARK: --- Scriptable properties ---
|
|
|
|
@objc(url)
|
|
var url:String {
|
|
return self.feed.url
|
|
}
|
|
|
|
@objc(uniqueId)
|
|
var uniqueId:String {
|
|
return self.feed.feedID
|
|
}
|
|
|
|
@objc(name)
|
|
var name:String {
|
|
return self.feed.name ?? ""
|
|
}
|
|
|
|
@objc(homePageURL)
|
|
var homePageURL:String {
|
|
return self.feed.homePageURL ?? ""
|
|
}
|
|
|
|
@objc(iconURL)
|
|
var iconURL:String {
|
|
return self.feed.iconURL ?? ""
|
|
}
|
|
|
|
@objc(faviconURL)
|
|
var faviconURL:String {
|
|
return self.feed.faviconURL ?? ""
|
|
}
|
|
|
|
@objc(opmlRepresentation)
|
|
var opmlRepresentation:String {
|
|
return self.feed.OPMLString(indentLevel:0)
|
|
}
|
|
|
|
// MARK: --- scriptable elements ---
|
|
|
|
@objc(authors)
|
|
var authors:NSArray {
|
|
let feedAuthors = feed.authors ?? []
|
|
return feedAuthors.map { ScriptableAuthor($0, container:self) } as NSArray
|
|
}
|
|
|
|
@objc(valueInAuthorsWithUniqueID:)
|
|
func valueInAuthors(withUniqueID id:String) -> ScriptableAuthor? {
|
|
guard let author = feed.authors?.first(where:{$0.authorID == id}) else { return nil }
|
|
return ScriptableAuthor(author, container:self)
|
|
}
|
|
|
|
@objc(articles)
|
|
var articles:NSArray {
|
|
let feedArticles = feed.fetchArticles()
|
|
// the articles are a set, use the sorting algorithm from the viewer
|
|
let sortedArticles = feedArticles.sorted(by:{
|
|
return $0.logicalDatePublished > $1.logicalDatePublished
|
|
})
|
|
return sortedArticles.map { ScriptableArticle($0, container:self) } as NSArray
|
|
}
|
|
|
|
@objc(valueInArticlesWithUniqueID:)
|
|
func valueInArticles(withUniqueID id:String) -> ScriptableArticle? {
|
|
let articles = feed.fetchArticles()
|
|
guard let article = articles.first(where:{$0.uniqueID == id}) else { return nil }
|
|
return ScriptableArticle(article, container:self)
|
|
}
|
|
|
|
}
|