Suspend and resume account resources when moving between the background and foreground.
This commit is contained in:
parent
4f3f56bda5
commit
d9f2e13732
Frameworks/Account
Shared/Timeline
iOS
@ -24,6 +24,7 @@ public final class AccountManager: UnreadCountProvider {
|
|||||||
private let defaultAccountFolderName = "OnMyMac"
|
private let defaultAccountFolderName = "OnMyMac"
|
||||||
private let defaultAccountIdentifier = "OnMyMac"
|
private let defaultAccountIdentifier = "OnMyMac"
|
||||||
|
|
||||||
|
public var isSuspended = false
|
||||||
public var isUnreadCountsInitialized: Bool {
|
public var isUnreadCountsInitialized: Bool {
|
||||||
for account in activeAccounts {
|
for account in activeAccounts {
|
||||||
if !account.isUnreadCountsInitialized {
|
if !account.isUnreadCountsInitialized {
|
||||||
@ -164,10 +165,12 @@ public final class AccountManager: UnreadCountProvider {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public func suspendAll() {
|
public func suspendAll() {
|
||||||
|
isSuspended = true
|
||||||
accounts.forEach { $0.suspend() }
|
accounts.forEach { $0.suspend() }
|
||||||
}
|
}
|
||||||
|
|
||||||
public func resumeAll() {
|
public func resumeAll() {
|
||||||
|
isSuspended = false
|
||||||
accounts.forEach { $0.resume() }
|
accounts.forEach { $0.resume() }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -46,6 +46,12 @@ final class FetchRequestOperation {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// The account manager may have been suspended while we were queued up
|
||||||
|
if AccountManager.shared.isSuspended {
|
||||||
|
callCompletionIfNeeded()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
if isCanceled {
|
if isCanceled {
|
||||||
callCompletionIfNeeded()
|
callCompletionIfNeeded()
|
||||||
return
|
return
|
||||||
|
@ -51,6 +51,18 @@ class AppDelegate: UIResponder, UIApplicationDelegate, UNUserNotificationCenterD
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var isSyncArticleStatusRunning = false
|
||||||
|
var isWaitingForSyncTasks = false
|
||||||
|
|
||||||
|
var isAnySceneInForeground: Bool {
|
||||||
|
for scene in UIApplication.shared.connectedScenes {
|
||||||
|
if scene.activationState != .background {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
override init() {
|
override init() {
|
||||||
super.init()
|
super.init()
|
||||||
appDelegate = self
|
appDelegate = self
|
||||||
@ -108,7 +120,6 @@ class AppDelegate: UIResponder, UIApplicationDelegate, UNUserNotificationCenterD
|
|||||||
|
|
||||||
func applicationWillTerminate(_ application: UIApplication) {
|
func applicationWillTerminate(_ application: UIApplication) {
|
||||||
shuttingDown = true
|
shuttingDown = true
|
||||||
AccountManager.shared.suspendAll()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// MARK: Notifications
|
// MARK: Notifications
|
||||||
@ -128,11 +139,15 @@ class AppDelegate: UIResponder, UIApplicationDelegate, UNUserNotificationCenterD
|
|||||||
func prepareAccountsForBackground() {
|
func prepareAccountsForBackground() {
|
||||||
syncTimer?.invalidate()
|
syncTimer?.invalidate()
|
||||||
scheduleBackgroundFeedRefresh()
|
scheduleBackgroundFeedRefresh()
|
||||||
waitForProgressToFinish()
|
|
||||||
syncArticleStatus()
|
syncArticleStatus()
|
||||||
|
waitForSyncTasksToFinish()
|
||||||
}
|
}
|
||||||
|
|
||||||
func prepareAccountsForForeground() {
|
func prepareAccountsForForeground() {
|
||||||
|
if AccountManager.shared.isSuspended {
|
||||||
|
AccountManager.shared.resumeAll()
|
||||||
|
}
|
||||||
|
|
||||||
if let lastRefresh = AppDefaults.lastRefresh {
|
if let lastRefresh = AppDefaults.lastRefresh {
|
||||||
if Date() > lastRefresh.addingTimeInterval(15 * 60) {
|
if Date() > lastRefresh.addingTimeInterval(15 * 60) {
|
||||||
AccountManager.shared.refreshAll(errorHandler: ErrorHandler.log)
|
AccountManager.shared.refreshAll(errorHandler: ErrorHandler.log)
|
||||||
@ -216,46 +231,58 @@ private extension AppDelegate {
|
|||||||
// MARK: Go To Background
|
// MARK: Go To Background
|
||||||
private extension AppDelegate {
|
private extension AppDelegate {
|
||||||
|
|
||||||
func waitForProgressToFinish() {
|
func waitForSyncTasksToFinish() {
|
||||||
let completeProcessing = { [unowned self] in
|
guard !isAnySceneInForeground && !isWaitingForSyncTasks else { return }
|
||||||
UIApplication.shared.endBackgroundTask(self.waitBackgroundUpdateTask)
|
|
||||||
self.waitBackgroundUpdateTask = UIBackgroundTaskIdentifier.invalid
|
|
||||||
}
|
|
||||||
|
|
||||||
self.waitBackgroundUpdateTask = UIApplication.shared.beginBackgroundTask {
|
isWaitingForSyncTasks = true
|
||||||
completeProcessing()
|
|
||||||
|
self.waitBackgroundUpdateTask = UIApplication.shared.beginBackgroundTask { [weak self] in
|
||||||
|
guard let self = self else { return }
|
||||||
|
self.completeProcessing(true)
|
||||||
os_log("Accounts wait for progress terminated for running too long.", log: self.log, type: .info)
|
os_log("Accounts wait for progress terminated for running too long.", log: self.log, type: .info)
|
||||||
}
|
}
|
||||||
|
|
||||||
DispatchQueue.main.async { [weak self] in
|
DispatchQueue.main.async { [weak self] in
|
||||||
self?.waitToComplete() {
|
self?.waitToComplete() { [weak self] suspend in
|
||||||
completeProcessing()
|
self?.completeProcessing(suspend)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func waitToComplete(completion: @escaping () -> Void) {
|
func waitToComplete(completion: @escaping (Bool) -> Void) {
|
||||||
guard UIApplication.shared.applicationState != .active else {
|
guard !isAnySceneInForeground else {
|
||||||
os_log("App came back to forground, no longer waiting.", log: self.log, type: .info)
|
os_log("App came back to forground, no longer waiting.", log: self.log, type: .info)
|
||||||
completion()
|
completion(false)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
if AccountManager.shared.refreshInProgress {
|
if AccountManager.shared.refreshInProgress || isSyncArticleStatusRunning {
|
||||||
os_log("Waiting for refresh progress to finish...", log: self.log, type: .info)
|
os_log("Waiting for sync to finish...", log: self.log, type: .info)
|
||||||
DispatchQueue.main.asyncAfter(deadline: .now() + 1.0) { [weak self] in
|
DispatchQueue.main.asyncAfter(deadline: .now() + 1.0) { [weak self] in
|
||||||
self?.waitToComplete() {
|
self?.waitToComplete(completion: completion)
|
||||||
completion()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
os_log("Refresh progress complete.", log: self.log, type: .info)
|
os_log("Refresh progress complete.", log: self.log, type: .info)
|
||||||
completion()
|
completion(true)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func completeProcessing(_ suspend: Bool) {
|
||||||
|
if suspend {
|
||||||
|
AccountManager.shared.suspendAll()
|
||||||
|
}
|
||||||
|
UIApplication.shared.endBackgroundTask(self.waitBackgroundUpdateTask)
|
||||||
|
self.waitBackgroundUpdateTask = UIBackgroundTaskIdentifier.invalid
|
||||||
|
isWaitingForSyncTasks = false
|
||||||
|
}
|
||||||
|
|
||||||
func syncArticleStatus() {
|
func syncArticleStatus() {
|
||||||
|
guard !isSyncArticleStatusRunning else { return }
|
||||||
|
|
||||||
|
isSyncArticleStatusRunning = true
|
||||||
|
|
||||||
let completeProcessing = { [unowned self] in
|
let completeProcessing = { [unowned self] in
|
||||||
|
self.isSyncArticleStatusRunning = false
|
||||||
UIApplication.shared.endBackgroundTask(self.syncBackgroundUpdateTask)
|
UIApplication.shared.endBackgroundTask(self.syncBackgroundUpdateTask)
|
||||||
self.syncBackgroundUpdateTask = UIBackgroundTaskIdentifier.invalid
|
self.syncBackgroundUpdateTask = UIBackgroundTaskIdentifier.invalid
|
||||||
}
|
}
|
||||||
@ -312,8 +339,12 @@ private extension AppDelegate {
|
|||||||
os_log("Woken to perform account refresh.", log: self.log, type: .info)
|
os_log("Woken to perform account refresh.", log: self.log, type: .info)
|
||||||
|
|
||||||
DispatchQueue.main.async { [weak task] in
|
DispatchQueue.main.async { [weak task] in
|
||||||
|
if AccountManager.shared.isSuspended {
|
||||||
|
AccountManager.shared.resumeAll()
|
||||||
|
}
|
||||||
AccountManager.shared.refreshAll(errorHandler: ErrorHandler.log) {
|
AccountManager.shared.refreshAll(errorHandler: ErrorHandler.log) {
|
||||||
AccountManager.shared.saveAll()
|
AccountManager.shared.saveAll()
|
||||||
|
AccountManager.shared.suspendAll()
|
||||||
os_log("Account refresh operation completed.", log: self.log, type: .info)
|
os_log("Account refresh operation completed.", log: self.log, type: .info)
|
||||||
task?.setTaskCompleted(success: true)
|
task?.setTaskCompleted(success: true)
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user