2017-07-02 02:22:19 +02:00
|
|
|
|
//
|
|
|
|
|
// 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
|
2018-07-28 21:16:14 +02:00
|
|
|
|
import Articles
|
2018-09-14 07:25:10 +02:00
|
|
|
|
import RSDatabase
|
2017-07-02 02:22:19 +02:00
|
|
|
|
|
2017-09-26 22:26:28 +02:00
|
|
|
|
public final class Feed: DisplayNameProvider, UnreadCountProvider, Hashable {
|
2017-07-02 02:22:19 +02:00
|
|
|
|
|
2018-09-14 07:52:34 +02:00
|
|
|
|
private struct Key {
|
|
|
|
|
static let url = "url"
|
|
|
|
|
static let feedID = "feedID"
|
|
|
|
|
static let homePageURL = "homePageURL"
|
|
|
|
|
static let iconURL = "iconURL"
|
|
|
|
|
static let faviconURL = "faviconURL"
|
|
|
|
|
static let name = "name"
|
|
|
|
|
static let editedName = "editedName"
|
|
|
|
|
static let authors = "authors"
|
|
|
|
|
static let conditionalGetInfo = "conditionalGetInfo"
|
2018-09-15 04:33:47 +02:00
|
|
|
|
static let conditionalGetLastModified = "lastModified"
|
|
|
|
|
static let conditionalGetEtag = "etag"
|
2018-09-14 07:52:34 +02:00
|
|
|
|
static let contentHash = "contentHash"
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-14 07:37:40 +02:00
|
|
|
|
public weak var account: Account?
|
2017-07-02 02:22:19 +02:00
|
|
|
|
public let url: String
|
|
|
|
|
public let feedID: String
|
2018-09-02 21:14:04 +02:00
|
|
|
|
|
|
|
|
|
public var homePageURL: String? {
|
|
|
|
|
get {
|
2018-09-15 07:23:30 +02:00
|
|
|
|
return settingsTable.string(for: Key.homePageURL)
|
2018-09-02 21:14:04 +02:00
|
|
|
|
}
|
|
|
|
|
set {
|
|
|
|
|
if let url = newValue {
|
2018-09-15 07:23:30 +02:00
|
|
|
|
settingsTable.setString(url.rs_normalizedURL(), for: Key.homePageURL)
|
2018-09-02 21:14:04 +02:00
|
|
|
|
}
|
|
|
|
|
else {
|
2018-09-15 07:23:30 +02:00
|
|
|
|
settingsTable.setString(nil, for: Key.homePageURL)
|
2018-09-02 21:14:04 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-15 06:51:05 +02:00
|
|
|
|
public var iconURL: String? {
|
|
|
|
|
get {
|
|
|
|
|
return settingsTable.string(for: Key.iconURL)
|
|
|
|
|
}
|
|
|
|
|
set {
|
|
|
|
|
settingsTable.setString(newValue, for: Key.iconURL)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public var faviconURL: String? {
|
|
|
|
|
get {
|
|
|
|
|
return settingsTable.string(for: Key.faviconURL)
|
|
|
|
|
}
|
|
|
|
|
set {
|
|
|
|
|
settingsTable.setString(newValue, for: Key.faviconURL)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-15 20:45:01 +02:00
|
|
|
|
public var name: String? {
|
|
|
|
|
get {
|
|
|
|
|
return settingsTable.string(for: Key.name)
|
|
|
|
|
}
|
|
|
|
|
set {
|
|
|
|
|
let oldNameForDisplay = nameForDisplay
|
|
|
|
|
settingsTable.setString(newValue, for: Key.name)
|
|
|
|
|
if oldNameForDisplay != nameForDisplay {
|
|
|
|
|
postDisplayNameDidChangeNotification()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-15 20:16:05 +02:00
|
|
|
|
public var authors: Set<Author>? {
|
|
|
|
|
get {
|
|
|
|
|
guard let authorsJSON = settingsTable.string(for: Key.authors) else {
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
return Author.authorsWithJSON(authorsJSON)
|
|
|
|
|
}
|
|
|
|
|
set {
|
|
|
|
|
if let authorsJSON = newValue?.json() {
|
|
|
|
|
settingsTable.setString(authorsJSON, for: Key.authors)
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
settingsTable.setString(nil, for: Key.authors)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-01-24 06:49:33 +01:00
|
|
|
|
|
|
|
|
|
public var editedName: String? {
|
2018-09-15 20:39:33 +02:00
|
|
|
|
get {
|
|
|
|
|
return settingsTable.string(for: Key.editedName)
|
|
|
|
|
}
|
|
|
|
|
set {
|
|
|
|
|
if newValue != editedName {
|
|
|
|
|
settingsTable.setString(newValue, for: Key.editedName)
|
|
|
|
|
postDisplayNameDidChangeNotification()
|
|
|
|
|
}
|
2018-01-24 06:49:33 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-15 04:33:47 +02:00
|
|
|
|
public var conditionalGetInfo: HTTPConditionalGetInfo? {
|
|
|
|
|
get {
|
|
|
|
|
let lastModified = settingsTable.string(for: Key.conditionalGetLastModified)
|
|
|
|
|
let etag = settingsTable.string(for: Key.conditionalGetEtag)
|
|
|
|
|
return HTTPConditionalGetInfo(lastModified: lastModified, etag: etag)
|
|
|
|
|
}
|
|
|
|
|
set {
|
|
|
|
|
settingsTable.setString(newValue?.lastModified, for: Key.conditionalGetLastModified)
|
|
|
|
|
settingsTable.setString(newValue?.etag, for: Key.conditionalGetEtag)
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-09-15 06:51:05 +02:00
|
|
|
|
|
2018-09-14 07:52:34 +02:00
|
|
|
|
public var contentHash: String? {
|
|
|
|
|
get {
|
|
|
|
|
return settingsTable.string(for: Key.contentHash)
|
|
|
|
|
}
|
|
|
|
|
set {
|
|
|
|
|
settingsTable.setString(newValue, for: Key.contentHash)
|
|
|
|
|
}
|
|
|
|
|
}
|
2017-09-17 00:25:38 +02:00
|
|
|
|
|
|
|
|
|
// MARK: - DisplayNameProvider
|
|
|
|
|
|
2017-07-02 02:22:19 +02:00
|
|
|
|
public var nameForDisplay: String {
|
2018-02-14 22:14:25 +01:00
|
|
|
|
if let s = editedName, !s.isEmpty {
|
|
|
|
|
return s
|
2017-07-02 02:22:19 +02:00
|
|
|
|
}
|
2018-02-14 22:14:25 +01:00
|
|
|
|
if let s = name, !s.isEmpty {
|
|
|
|
|
return s
|
|
|
|
|
}
|
|
|
|
|
return NSLocalizedString("Untitled", comment: "Feed name")
|
2017-07-02 02:22:19 +02:00
|
|
|
|
}
|
|
|
|
|
|
2017-09-17 00:25:38 +02:00
|
|
|
|
// MARK: - UnreadCountProvider
|
2017-07-03 19:29:44 +02:00
|
|
|
|
|
2018-09-15 07:06:03 +02:00
|
|
|
|
public var unreadCount: Int {
|
|
|
|
|
get {
|
|
|
|
|
return account?.unreadCount(for: self) ?? 0
|
|
|
|
|
}
|
|
|
|
|
set {
|
|
|
|
|
if unreadCount == newValue {
|
|
|
|
|
return
|
2017-09-17 00:25:38 +02:00
|
|
|
|
}
|
2018-09-15 07:06:03 +02:00
|
|
|
|
account?.setUnreadCount(newValue, for: self)
|
|
|
|
|
postUnreadCountDidChangeNotification()
|
2017-09-17 00:25:38 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-14 07:37:40 +02:00
|
|
|
|
private let settingsTable: ODBRawValueTable
|
2018-09-15 07:15:22 +02:00
|
|
|
|
private let accountID: String // Used for hashing and equality; account may turn nil
|
2018-09-14 07:25:10 +02:00
|
|
|
|
|
2017-09-17 00:25:38 +02:00
|
|
|
|
// MARK: - Init
|
|
|
|
|
|
2018-09-14 07:37:40 +02:00
|
|
|
|
public init(account: Account, url: String, feedID: String) {
|
2017-07-02 02:22:19 +02:00
|
|
|
|
|
2018-09-14 07:37:40 +02:00
|
|
|
|
self.account = account
|
2018-09-15 07:15:22 +02:00
|
|
|
|
self.accountID = account.accountID
|
2017-07-02 02:22:19 +02:00
|
|
|
|
self.url = url
|
|
|
|
|
self.feedID = feedID
|
2018-09-14 07:37:40 +02:00
|
|
|
|
self.settingsTable = account.settingsTableForFeed(feedID: feedID)!
|
2017-07-02 02:22:19 +02:00
|
|
|
|
}
|
|
|
|
|
|
2017-09-26 22:26:28 +02:00
|
|
|
|
// MARK: - Disk Dictionary
|
|
|
|
|
|
2018-09-14 07:37:40 +02:00
|
|
|
|
convenience public init?(account: Account, dictionary: [String: Any]) {
|
2017-09-26 22:26:28 +02:00
|
|
|
|
|
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
|
|
|
|
|
|
2018-09-14 07:37:40 +02:00
|
|
|
|
self.init(account: account, url: url, feedID: feedID)
|
2017-09-26 22:26:28 +02:00
|
|
|
|
self.editedName = dictionary[Key.editedName] as? String
|
2018-09-15 20:45:01 +02:00
|
|
|
|
self.name = dictionary[Key.name] as? String
|
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
|
|
|
|
|
}
|
|
|
|
|
|
2018-09-16 21:42:46 +02:00
|
|
|
|
// public var dictionary: [String: Any] {
|
|
|
|
|
// var d = [String: Any]()
|
|
|
|
|
//
|
|
|
|
|
// d[Key.url] = url
|
|
|
|
|
//
|
|
|
|
|
// // feedID is not repeated when it’s the same as url
|
|
|
|
|
// if (feedID != url) {
|
|
|
|
|
// d[Key.feedID] = feedID
|
|
|
|
|
// }
|
|
|
|
|
//
|
|
|
|
|
// if let name = name {
|
|
|
|
|
// d[Key.name] = name
|
|
|
|
|
// }
|
|
|
|
|
// if let editedName = editedName {
|
|
|
|
|
// d[Key.editedName] = editedName
|
|
|
|
|
// }
|
|
|
|
|
//
|
|
|
|
|
// return d
|
|
|
|
|
// }
|
2017-09-26 22:26:28 +02:00
|
|
|
|
|
2017-11-25 20:13:15 +01:00
|
|
|
|
// MARK: - Debug
|
|
|
|
|
|
|
|
|
|
public func debugDropConditionalGetInfo() {
|
|
|
|
|
|
|
|
|
|
conditionalGetInfo = nil
|
|
|
|
|
contentHash = nil
|
|
|
|
|
}
|
|
|
|
|
|
2018-08-25 20:54:58 +02:00
|
|
|
|
// MARK: - Hashable
|
|
|
|
|
|
|
|
|
|
public func hash(into hasher: inout Hasher) {
|
|
|
|
|
hasher.combine(feedID)
|
2018-09-15 07:15:22 +02:00
|
|
|
|
hasher.combine(accountID)
|
2018-08-25 20:54:58 +02:00
|
|
|
|
}
|
|
|
|
|
|
2017-11-25 20:13:15 +01:00
|
|
|
|
// MARK: - Equatable
|
|
|
|
|
|
2017-07-02 02:22:19 +02:00
|
|
|
|
public class func ==(lhs: Feed, rhs: Feed) -> Bool {
|
|
|
|
|
|
2018-09-15 07:15:22 +02:00
|
|
|
|
return lhs.feedID == rhs.feedID && lhs.accountID == rhs.accountID
|
2017-07-02 02:22:19 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-09-17 00:25:38 +02:00
|
|
|
|
// 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
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-07-28 21:16:14 +02:00
|
|
|
|
extension Set where Element == Feed {
|
|
|
|
|
|
|
|
|
|
func feedIDs() -> Set<String> {
|
|
|
|
|
|
|
|
|
|
return Set<String>(map { $0.feedID })
|
|
|
|
|
}
|
|
|
|
|
}
|