NetNewsWire/Frameworks/Account/Feed.swift

204 lines
4.8 KiB
Swift
Raw Normal View History

//
// Feed.swift
// DataModel
//
// Created by Brent Simmons on 7/1/17.
// Copyright © 2017 Ranchero Software, LLC. All rights reserved.
//
import Foundation
import RSCore
2017-09-17 21:08:50 +02:00
import RSWeb
import Articles
2017-09-26 22:26:28 +02:00
public final class Feed: DisplayNameProvider, UnreadCountProvider, Hashable {
public let accountID: String
public let url: String
public let feedID: String
public var homePageURL: String?
2017-11-20 07:37:59 +01:00
public var iconURL: String?
2017-11-20 06:24:19 +01:00
public var faviconURL: String?
public var name: String?
2017-12-02 23:20:58 +01:00
public var authors: Set<Author>?
public var editedName: String? {
didSet {
postDisplayNameDidChangeNotification()
}
}
2017-09-17 21:08:50 +02:00
public var conditionalGetInfo: HTTPConditionalGetInfo?
public var contentHash: String?
// MARK: - DisplayNameProvider
public var nameForDisplay: String {
if let s = editedName, !s.isEmpty {
return s
}
if let s = name, !s.isEmpty {
return s
}
return NSLocalizedString("Untitled", comment: "Feed name")
}
// MARK: - UnreadCountProvider
public var unreadCount = 0 {
didSet {
if unreadCount != oldValue {
postUnreadCountDidChangeNotification()
}
}
}
// MARK: - Init
public init(accountID: String, url: String, feedID: String) {
self.accountID = accountID
self.url = url
self.feedID = feedID
}
2017-09-26 22:26:28 +02:00
// MARK: - Disk Dictionary
2017-09-28 22:16:47 +02:00
private struct Key {
2017-09-26 22:26:28 +02:00
static let url = "url"
static let feedID = "feedID"
static let homePageURL = "homePageURL"
2017-11-20 07:37:59 +01:00
static let iconURL = "iconURL"
2017-11-20 06:24:19 +01:00
static let faviconURL = "faviconURL"
2017-09-26 22:26:28 +02:00
static let name = "name"
static let editedName = "editedName"
2017-12-02 23:20:58 +01:00
static let authors = "authors"
2017-09-26 22:26:28 +02:00
static let conditionalGetInfo = "conditionalGetInfo"
static let contentHash = "contentHash"
static let unreadCount = "unreadCount"
}
convenience public init?(accountID: String, dictionary: [String: Any]) {
2017-10-06 06:08:27 +02:00
guard let url = dictionary[Key.url] as? String else {
2017-09-26 22:26:28 +02:00
return nil
}
2017-10-06 06:08:27 +02:00
let feedID = dictionary[Key.feedID] as? String ?? url
2017-09-26 22:26:28 +02:00
self.init(accountID: accountID, url: url, feedID: feedID)
self.homePageURL = dictionary[Key.homePageURL] as? String
2017-11-20 07:37:59 +01:00
self.iconURL = dictionary[Key.iconURL] as? String
2017-11-20 06:24:19 +01:00
self.faviconURL = dictionary[Key.faviconURL] as? String
2017-09-26 22:26:28 +02:00
self.name = dictionary[Key.name] as? String
self.editedName = dictionary[Key.editedName] as? String
self.contentHash = dictionary[Key.contentHash] as? String
if let conditionalGetInfoDictionary = dictionary[Key.conditionalGetInfo] as? [String: String] {
self.conditionalGetInfo = HTTPConditionalGetInfo(dictionary: conditionalGetInfoDictionary)
}
if let savedUnreadCount = dictionary[Key.unreadCount] as? Int {
self.unreadCount = savedUnreadCount
}
2017-12-02 23:20:58 +01:00
if let authorsDiskArray = dictionary[Key.authors] as? [[String: Any]] {
self.authors = Author.authorsWithDiskArray(authorsDiskArray)
}
2017-09-26 22:26:28 +02:00
}
2017-09-28 22:16:47 +02:00
public static func isFeedDictionary(_ d: [String: Any]) -> Bool {
return d[Key.url] != nil
}
2017-09-26 22:26:28 +02:00
public var dictionary: [String: Any] {
var d = [String: Any]()
d[Key.url] = url
// feedID is not repeated when its the same as url
if (feedID != url) {
d[Key.feedID] = feedID
}
if let homePageURL = homePageURL {
d[Key.homePageURL] = homePageURL
}
if let iconURL = iconURL {
d[Key.iconURL] = iconURL
}
if let faviconURL = faviconURL {
d[Key.faviconURL] = faviconURL
2017-09-26 22:26:28 +02:00
}
if let name = name {
d[Key.name] = name
}
if let editedName = editedName {
d[Key.editedName] = editedName
}
if let authorsArray = authors?.diskArray() {
d[Key.authors] = authorsArray
}
if let contentHash = contentHash {
d[Key.contentHash] = contentHash
}
if unreadCount > 0 {
d[Key.unreadCount] = unreadCount
}
if let conditionalGetInfo = conditionalGetInfo {
d[Key.conditionalGetInfo] = conditionalGetInfo.dictionary
}
return d
2017-09-26 22:26:28 +02:00
}
// MARK: - Debug
public func debugDropConditionalGetInfo() {
conditionalGetInfo = nil
contentHash = nil
}
// MARK: - Hashable
public func hash(into hasher: inout Hasher) {
hasher.combine(feedID)
}
// MARK: - Equatable
public class func ==(lhs: Feed, rhs: Feed) -> Bool {
return lhs === rhs
}
}
// MARK: - OPMLRepresentable
extension Feed: OPMLRepresentable {
public func OPMLString(indentLevel: Int) -> String {
let escapedName = nameForDisplay.rs_stringByEscapingSpecialXMLCharacters()
var escapedHomePageURL = ""
if let homePageURL = homePageURL {
escapedHomePageURL = homePageURL.rs_stringByEscapingSpecialXMLCharacters()
}
let escapedFeedURL = url.rs_stringByEscapingSpecialXMLCharacters()
var s = "<outline text=\"\(escapedName)\" title=\"\(escapedName)\" description=\"\" type=\"rss\" version=\"RSS\" htmlUrl=\"\(escapedHomePageURL)\" xmlUrl=\"\(escapedFeedURL)\"/>\n"
s = s.rs_string(byPrependingNumberOfTabs: indentLevel)
return s
}
}
extension Set where Element == Feed {
func feedIDs() -> Set<String> {
return Set<String>(map { $0.feedID })
}
}