2020-11-18 03:49:12 +01:00
|
|
|
//
|
|
|
|
// WidgetDataEncoder.swift
|
|
|
|
// NetNewsWire
|
|
|
|
//
|
|
|
|
// Created by Stuart Breckenridge on 18/11/20.
|
|
|
|
// Copyright © 2020 Ranchero Software. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
import Foundation
|
|
|
|
import WidgetKit
|
|
|
|
import os.log
|
|
|
|
import UIKit
|
|
|
|
import RSCore
|
|
|
|
import Articles
|
2021-03-24 13:30:21 +01:00
|
|
|
import Account
|
2020-11-18 03:49:12 +01:00
|
|
|
|
2020-12-03 13:32:26 +01:00
|
|
|
|
|
|
|
public final class WidgetDataEncoder {
|
|
|
|
|
|
|
|
private let log = OSLog(subsystem: Bundle.main.bundleIdentifier!, category: "Application")
|
2021-03-24 13:30:21 +01:00
|
|
|
private let fetchLimit = 7
|
2020-12-03 13:32:26 +01:00
|
|
|
|
|
|
|
private lazy var appGroup = Bundle.main.object(forInfoDictionaryKey: "AppGroup") as! String
|
|
|
|
private lazy var containerURL = FileManager.default.containerURL(forSecurityApplicationGroupIdentifier: appGroup)
|
2022-05-16 03:32:38 +02:00
|
|
|
private lazy var imageContainer = containerURL?.appendingPathComponent("widgetImages", isDirectory: true)
|
2020-12-03 13:32:26 +01:00
|
|
|
private lazy var dataURL = containerURL?.appendingPathComponent("widget-data.json")
|
2020-11-18 03:49:12 +01:00
|
|
|
|
2022-10-09 00:10:40 +02:00
|
|
|
private var searchWorkItem: DispatchWorkItem?
|
2022-07-29 00:15:36 +02:00
|
|
|
|
|
|
|
init () {
|
2022-05-16 03:32:38 +02:00
|
|
|
if imageContainer != nil {
|
|
|
|
try? FileManager.default.createDirectory(at: imageContainer!, withIntermediateDirectories: true, attributes: nil)
|
|
|
|
}
|
2022-10-09 00:10:40 +02:00
|
|
|
NotificationCenter.default.addObserver(self, selector: #selector(statusesDidChange(_:)), name: .StatusesDidChange, object: nil)
|
2022-05-16 03:32:38 +02:00
|
|
|
}
|
2020-11-29 10:12:53 +01:00
|
|
|
|
2022-10-09 00:10:40 +02:00
|
|
|
func pause() {
|
|
|
|
searchWorkItem?.cancel()
|
|
|
|
}
|
|
|
|
|
|
|
|
func resume() {
|
|
|
|
dispatchWorkItem()
|
2022-07-29 00:15:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
@objc func statusesDidChange(_ note: Notification) {
|
2022-10-09 00:10:40 +02:00
|
|
|
dispatchWorkItem()
|
2022-07-29 00:15:36 +02:00
|
|
|
}
|
|
|
|
|
2022-10-09 00:10:40 +02:00
|
|
|
func dispatchWorkItem() {
|
|
|
|
if #available(iOS 14, *) {
|
|
|
|
searchWorkItem?.cancel()
|
|
|
|
searchWorkItem = DispatchWorkItem { [weak self] in
|
|
|
|
self?.encodeWidgetData()
|
2022-07-29 00:15:36 +02:00
|
|
|
}
|
2022-10-09 00:10:40 +02:00
|
|
|
DispatchQueue.main.asyncAfter(deadline: .now() + 5, execute: searchWorkItem!)
|
2022-07-29 00:15:36 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@available(iOS 14, *)
|
|
|
|
private func encodeWidgetData() {
|
2022-05-16 03:32:38 +02:00
|
|
|
flushSharedContainer()
|
2020-11-29 10:12:53 +01:00
|
|
|
os_log(.debug, log: log, "Starting encoding widget data.")
|
2020-12-03 13:32:26 +01:00
|
|
|
|
2021-11-13 16:38:57 +01:00
|
|
|
do {
|
|
|
|
let unreadArticles = Array(try AccountManager.shared.fetchArticles(.unread(fetchLimit))).sortedByDate(.orderedDescending)
|
|
|
|
let starredArticles = Array(try AccountManager.shared.fetchArticles(.starred(fetchLimit))).sortedByDate(.orderedDescending)
|
|
|
|
let todayArticles = Array(try AccountManager.shared.fetchArticles(.today(fetchLimit))).sortedByDate(.orderedDescending)
|
|
|
|
|
|
|
|
var unread = [LatestArticle]()
|
|
|
|
var today = [LatestArticle]()
|
|
|
|
var starred = [LatestArticle]()
|
|
|
|
|
|
|
|
for article in unreadArticles {
|
|
|
|
let latestArticle = LatestArticle(id: article.sortableArticleID,
|
|
|
|
feedTitle: article.sortableName,
|
|
|
|
articleTitle: ArticleStringFormatter.truncatedTitle(article).isEmpty ? ArticleStringFormatter.truncatedSummary(article) : ArticleStringFormatter.truncatedTitle(article),
|
|
|
|
articleSummary: article.summary,
|
2022-05-16 03:32:38 +02:00
|
|
|
feedIconPath: writeImageDataToSharedContainer(article.iconImage()?.image.dataRepresentation()),
|
2021-11-13 16:38:57 +01:00
|
|
|
pubDate: article.datePublished?.description ?? "")
|
|
|
|
unread.append(latestArticle)
|
|
|
|
}
|
|
|
|
|
|
|
|
for article in starredArticles {
|
|
|
|
let latestArticle = LatestArticle(id: article.sortableArticleID,
|
|
|
|
feedTitle: article.sortableName,
|
|
|
|
articleTitle: ArticleStringFormatter.truncatedTitle(article).isEmpty ? ArticleStringFormatter.truncatedSummary(article) : ArticleStringFormatter.truncatedTitle(article),
|
|
|
|
articleSummary: article.summary,
|
2022-05-16 03:32:38 +02:00
|
|
|
feedIconPath: writeImageDataToSharedContainer(article.iconImage()?.image.dataRepresentation()),
|
2021-11-13 16:38:57 +01:00
|
|
|
pubDate: article.datePublished?.description ?? "")
|
|
|
|
starred.append(latestArticle)
|
|
|
|
}
|
|
|
|
|
|
|
|
for article in todayArticles {
|
|
|
|
let latestArticle = LatestArticle(id: article.sortableArticleID,
|
|
|
|
feedTitle: article.sortableName,
|
|
|
|
articleTitle: ArticleStringFormatter.truncatedTitle(article).isEmpty ? ArticleStringFormatter.truncatedSummary(article) : ArticleStringFormatter.truncatedTitle(article),
|
|
|
|
articleSummary: article.summary,
|
2022-05-16 03:32:38 +02:00
|
|
|
feedIconPath: writeImageDataToSharedContainer(article.iconImage()?.image.dataRepresentation()),
|
2021-11-13 16:38:57 +01:00
|
|
|
pubDate: article.datePublished?.description ?? "")
|
|
|
|
today.append(latestArticle)
|
|
|
|
}
|
|
|
|
|
|
|
|
let latestData = WidgetData(currentUnreadCount: SmartFeedsController.shared.unreadFeed.unreadCount,
|
|
|
|
currentTodayCount: SmartFeedsController.shared.todayFeed.unreadCount,
|
2022-05-13 06:56:49 +02:00
|
|
|
currentStarredCount: try AccountManager.shared.fetchCountForStarredArticles(),
|
2021-11-13 16:38:57 +01:00
|
|
|
unreadArticles: unread,
|
|
|
|
starredArticles: starred,
|
|
|
|
todayArticles:today,
|
|
|
|
lastUpdateTime: Date())
|
|
|
|
|
|
|
|
|
|
|
|
DispatchQueue.global().async { [weak self] in
|
|
|
|
guard let self = self else { return }
|
2020-12-03 13:32:26 +01:00
|
|
|
|
2021-11-13 16:38:57 +01:00
|
|
|
let encodedData = try? JSONEncoder().encode(latestData)
|
2020-12-03 13:32:26 +01:00
|
|
|
|
|
|
|
os_log(.debug, log: self.log, "Finished encoding widget data.")
|
|
|
|
|
|
|
|
if self.fileExists() {
|
|
|
|
try? FileManager.default.removeItem(at: self.dataURL!)
|
|
|
|
os_log(.debug, log: self.log, "Removed widget data from container.")
|
|
|
|
}
|
|
|
|
if FileManager.default.createFile(atPath: self.dataURL!.path, contents: encodedData, attributes: nil) {
|
|
|
|
os_log(.debug, log: self.log, "Wrote widget data to container.")
|
2020-12-03 13:41:37 +01:00
|
|
|
WidgetCenter.shared.reloadAllTimelines()
|
2020-11-30 03:08:23 +01:00
|
|
|
}
|
2021-11-13 16:38:57 +01:00
|
|
|
|
2020-11-18 03:49:12 +01:00
|
|
|
}
|
2022-07-29 00:15:36 +02:00
|
|
|
} catch {
|
|
|
|
os_log(.error, log: log, "WidgetDataEncoder failed to write the widget data.")
|
2020-11-18 03:49:12 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-03 13:32:26 +01:00
|
|
|
private func fileExists() -> Bool {
|
|
|
|
FileManager.default.fileExists(atPath: dataURL!.path)
|
|
|
|
}
|
2020-11-18 03:49:12 +01:00
|
|
|
|
2022-05-16 03:32:38 +02:00
|
|
|
private func writeImageDataToSharedContainer(_ imageData: Data?) -> String? {
|
|
|
|
if imageData == nil { return nil }
|
|
|
|
// Each image gets a UUID
|
|
|
|
let uuid = UUID().uuidString
|
|
|
|
if let imagePath = imageContainer?.appendingPathComponent(uuid, isDirectory: false) {
|
|
|
|
do {
|
|
|
|
try imageData!.write(to: imagePath)
|
|
|
|
return imagePath.path
|
|
|
|
} catch {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
private func flushSharedContainer() {
|
|
|
|
if let imageContainer = imageContainer {
|
|
|
|
try? FileManager.default.removeItem(atPath: imageContainer.path)
|
|
|
|
try? FileManager.default.createDirectory(at: imageContainer, withIntermediateDirectories: true, attributes: nil)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-18 03:49:12 +01:00
|
|
|
}
|
|
|
|
|