2017-05-27 10:43:27 -07:00
|
|
|
//
|
|
|
|
// ArticleUtilities.swift
|
2018-08-28 22:18:24 -07:00
|
|
|
// NetNewsWire
|
2017-05-27 10:43:27 -07:00
|
|
|
//
|
|
|
|
// Created by Brent Simmons on 7/25/15.
|
|
|
|
// Copyright © 2015 Ranchero Software, LLC. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
import Foundation
|
2019-08-24 14:57:51 -05:00
|
|
|
import RSCore
|
2018-07-23 18:29:08 -07:00
|
|
|
import Articles
|
2017-09-17 12:34:10 -07:00
|
|
|
import Account
|
2017-05-27 10:43:27 -07:00
|
|
|
|
|
|
|
// These handle multiple accounts.
|
|
|
|
|
2017-10-29 11:14:10 -07:00
|
|
|
@discardableResult
|
|
|
|
func markArticles(_ articles: Set<Article>, statusKey: ArticleStatus.Key, flag: Bool) -> Set<Article>? {
|
2017-05-27 10:43:27 -07:00
|
|
|
|
2017-09-17 12:54:08 -07:00
|
|
|
let d: [String: Set<Article>] = accountAndArticlesDictionary(articles)
|
2017-10-29 11:14:10 -07:00
|
|
|
var updatedArticles = Set<Article>()
|
|
|
|
|
2017-10-08 21:06:25 -07:00
|
|
|
for (accountID, accountArticles) in d {
|
2017-05-27 10:43:27 -07:00
|
|
|
|
2017-10-08 21:06:25 -07:00
|
|
|
guard let account = AccountManager.shared.existingAccount(with: accountID) else {
|
2017-10-29 11:14:10 -07:00
|
|
|
continue
|
2017-05-27 10:43:27 -07:00
|
|
|
}
|
2017-10-29 11:14:10 -07:00
|
|
|
|
|
|
|
if let accountUpdatedArticles = account.markArticles(accountArticles, statusKey: statusKey, flag: flag) {
|
|
|
|
updatedArticles.formUnion(accountUpdatedArticles)
|
|
|
|
}
|
|
|
|
|
2017-05-27 10:43:27 -07:00
|
|
|
}
|
2017-10-29 11:14:10 -07:00
|
|
|
|
|
|
|
return updatedArticles
|
2017-05-27 10:43:27 -07:00
|
|
|
}
|
|
|
|
|
2017-09-17 12:54:08 -07:00
|
|
|
private func accountAndArticlesDictionary(_ articles: Set<Article>) -> [String: Set<Article>] {
|
2017-05-27 10:43:27 -07:00
|
|
|
|
2017-10-08 21:06:25 -07:00
|
|
|
let d = Dictionary(grouping: articles, by: { $0.accountID })
|
|
|
|
return d.mapValues{ Set($0) }
|
2017-05-27 10:43:27 -07:00
|
|
|
}
|
|
|
|
|
2017-09-17 12:54:08 -07:00
|
|
|
extension Article {
|
2017-05-27 10:43:27 -07:00
|
|
|
|
2019-11-14 20:11:41 -06:00
|
|
|
var webFeed: WebFeed? {
|
|
|
|
return account?.existingWebFeed(withWebFeedID: webFeedID)
|
2017-09-17 17:03:58 -07:00
|
|
|
}
|
|
|
|
|
2017-09-17 16:30:45 -07:00
|
|
|
var preferredLink: String? {
|
2019-09-07 18:27:48 -07:00
|
|
|
if let url = url, !url.isEmpty {
|
|
|
|
return url
|
|
|
|
}
|
|
|
|
if let externalURL = externalURL, !externalURL.isEmpty {
|
|
|
|
return externalURL
|
|
|
|
}
|
|
|
|
return nil
|
2017-09-17 16:30:45 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
var body: String? {
|
2018-02-14 13:14:25 -08:00
|
|
|
return contentHTML ?? contentText ?? summary
|
2017-05-27 10:43:27 -07:00
|
|
|
}
|
2017-09-17 17:03:58 -07:00
|
|
|
|
|
|
|
var logicalDatePublished: Date {
|
2018-02-14 13:14:25 -08:00
|
|
|
return datePublished ?? dateModified ?? status.dateArrived
|
2017-09-17 17:03:58 -07:00
|
|
|
}
|
2020-02-18 13:49:29 -08:00
|
|
|
|
|
|
|
var isAvailableToMarkUnread: Bool {
|
|
|
|
guard let markUnreadWindow = account?.behaviors.compactMap( { behavior -> Int? in
|
|
|
|
switch behavior {
|
|
|
|
case .disallowMarkAsUnreadAfterPeriod(let days):
|
|
|
|
return days
|
|
|
|
default:
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}).first else {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
if logicalDatePublished.byAdding(days: markUnreadWindow) > Date() {
|
|
|
|
return true
|
|
|
|
} else {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
2019-08-24 14:57:51 -05:00
|
|
|
|
2019-11-05 18:05:57 -06:00
|
|
|
func iconImage() -> IconImage? {
|
2019-11-07 14:29:16 -06:00
|
|
|
if let authors = authors, authors.count == 1, let author = authors.first {
|
|
|
|
if let image = appDelegate.authorAvatarDownloader.image(for: author) {
|
|
|
|
return image
|
2019-08-24 14:57:51 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-14 20:11:41 -06:00
|
|
|
if let authors = webFeed?.authors, authors.count == 1, let author = authors.first {
|
2019-11-07 14:29:16 -06:00
|
|
|
if let image = appDelegate.authorAvatarDownloader.image(for: author) {
|
|
|
|
return image
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-14 20:11:41 -06:00
|
|
|
guard let webFeed = webFeed else {
|
2019-08-24 14:57:51 -05:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-11-14 20:11:41 -06:00
|
|
|
let feedIconImage = appDelegate.webFeedIconDownloader.icon(for: webFeed)
|
2019-08-24 14:57:51 -05:00
|
|
|
if feedIconImage != nil {
|
|
|
|
return feedIconImage
|
|
|
|
}
|
|
|
|
|
2019-11-14 20:11:41 -06:00
|
|
|
if let faviconImage = appDelegate.faviconDownloader.faviconAsIcon(for: webFeed) {
|
2019-08-24 14:57:51 -05:00
|
|
|
return faviconImage
|
|
|
|
}
|
|
|
|
|
2019-11-14 20:11:41 -06:00
|
|
|
return FaviconGenerator.favicon(webFeed)
|
2019-08-24 14:57:51 -05:00
|
|
|
}
|
2019-11-25 19:43:43 -06:00
|
|
|
|
|
|
|
func byline() -> String {
|
|
|
|
guard let authors = authors ?? webFeed?.authors, !authors.isEmpty else {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the author's name is the same as the feed, then we don't want to display it.
|
|
|
|
// This code assumes that multiple authors would never match the feed name so that
|
|
|
|
// if there feed owner has an article co-author all authors are given the byline.
|
|
|
|
if authors.count == 1, let author = authors.first {
|
|
|
|
if author.name == webFeed?.nameForDisplay {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var byline = ""
|
|
|
|
var isFirstAuthor = true
|
|
|
|
|
|
|
|
for author in authors {
|
|
|
|
if !isFirstAuthor {
|
|
|
|
byline += ", "
|
|
|
|
}
|
|
|
|
isFirstAuthor = false
|
|
|
|
|
|
|
|
if let emailAddress = author.emailAddress, emailAddress.contains(" ") {
|
|
|
|
byline += emailAddress // probably name plus email address
|
|
|
|
}
|
|
|
|
else if let name = author.name, let emailAddress = author.emailAddress {
|
|
|
|
byline += "\(name) <\(emailAddress)>"
|
|
|
|
}
|
|
|
|
else if let name = author.name {
|
|
|
|
byline += name
|
|
|
|
}
|
|
|
|
else if let emailAddress = author.emailAddress {
|
|
|
|
byline += "<\(emailAddress)>"
|
|
|
|
}
|
|
|
|
else if let url = author.url {
|
|
|
|
byline += url
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return byline
|
|
|
|
}
|
|
|
|
|
2017-05-27 10:43:27 -07:00
|
|
|
}
|
2019-09-08 16:48:50 -05:00
|
|
|
|
2019-11-14 15:06:32 -06:00
|
|
|
// MARK: Path
|
2019-10-03 09:53:21 -05:00
|
|
|
|
2019-11-14 15:06:32 -06:00
|
|
|
struct ArticlePathKey {
|
|
|
|
static let accountID = "accountID"
|
|
|
|
static let accountName = "accountName"
|
2019-11-14 20:11:41 -06:00
|
|
|
static let webFeedID = "webFeedID"
|
2019-11-14 15:06:32 -06:00
|
|
|
static let articleID = "articleID"
|
|
|
|
}
|
|
|
|
|
|
|
|
extension Article {
|
2019-10-03 09:53:21 -05:00
|
|
|
|
2019-11-14 15:06:32 -06:00
|
|
|
public var pathUserInfo: [AnyHashable : Any] {
|
2019-10-03 09:53:21 -05:00
|
|
|
return [
|
2019-11-14 15:06:32 -06:00
|
|
|
ArticlePathKey.accountID: accountID,
|
|
|
|
ArticlePathKey.accountName: account?.nameForDisplay ?? "",
|
2019-11-14 20:11:41 -06:00
|
|
|
ArticlePathKey.webFeedID: webFeedID,
|
2019-11-14 15:06:32 -06:00
|
|
|
ArticlePathKey.articleID: articleID
|
2019-10-03 09:53:21 -05:00
|
|
|
]
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2019-09-08 16:48:50 -05:00
|
|
|
// MARK: SortableArticle
|
|
|
|
|
|
|
|
extension Article: SortableArticle {
|
|
|
|
|
|
|
|
var sortableName: String {
|
2019-11-14 20:11:41 -06:00
|
|
|
return webFeed?.name ?? ""
|
2019-09-08 16:48:50 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
var sortableDate: Date {
|
|
|
|
return logicalDatePublished
|
|
|
|
}
|
|
|
|
|
2019-09-13 07:43:28 -05:00
|
|
|
var sortableArticleID: String {
|
2019-09-08 16:48:50 -05:00
|
|
|
return articleID
|
|
|
|
}
|
|
|
|
|
2019-11-14 20:11:41 -06:00
|
|
|
var sortableWebFeedID: String {
|
|
|
|
return webFeedID
|
2019-09-13 07:43:28 -05:00
|
|
|
}
|
|
|
|
|
2019-09-08 16:48:50 -05:00
|
|
|
}
|