mirror of
https://github.com/Ranchero-Software/NetNewsWire.git
synced 2025-02-02 20:16:54 +01:00
Merge branch 'ios-release'
This commit is contained in:
commit
715804fe64
@ -7,42 +7,116 @@
|
|||||||
//
|
//
|
||||||
|
|
||||||
import Foundation
|
import Foundation
|
||||||
|
import RSCore
|
||||||
import WebKit
|
import WebKit
|
||||||
|
|
||||||
/// WKWebView has an awful behavior of a flash to white on first load when in dark mode.
|
/// WKWebView has an awful behavior of a flash to white on first load when in dark mode.
|
||||||
/// Keep a queue of WebViews where we've already done a trivial load so that by the time we need them in the UI, they're past the flash-to-shite part of their lifecycle.
|
/// Keep a queue of WebViews where we've already done a trivial load so that by the time we need them in the UI, they're past the flash-to-shite part of their lifecycle.
|
||||||
class WebViewProvider: NSObject {
|
class WebViewProvider: NSObject {
|
||||||
|
|
||||||
let articleIconSchemeHandler: ArticleIconSchemeHandler
|
private let articleIconSchemeHandler: ArticleIconSchemeHandler
|
||||||
|
private let operationQueue = MainThreadOperationQueue()
|
||||||
private let minimumQueueDepth = 3
|
|
||||||
private let maximumQueueDepth = 6
|
|
||||||
private var queue = UIView()
|
private var queue = UIView()
|
||||||
|
|
||||||
init(coordinator: SceneCoordinator, viewController: UIViewController) {
|
init(coordinator: SceneCoordinator, viewController: UIViewController) {
|
||||||
articleIconSchemeHandler = ArticleIconSchemeHandler(coordinator: coordinator)
|
articleIconSchemeHandler = ArticleIconSchemeHandler(coordinator: coordinator)
|
||||||
super.init()
|
super.init()
|
||||||
viewController.view.insertSubview(queue, at: 0)
|
viewController.view.insertSubview(queue, at: 0)
|
||||||
|
|
||||||
replenishQueueIfNeeded()
|
replenishQueueIfNeeded()
|
||||||
}
|
}
|
||||||
|
|
||||||
func flushQueue() {
|
func flushQueue() {
|
||||||
queue.subviews.forEach { $0.removeFromSuperview() }
|
operationQueue.add(WebViewProviderFlushQueueOperation(queue: queue))
|
||||||
}
|
}
|
||||||
|
|
||||||
func replenishQueueIfNeeded() {
|
func replenishQueueIfNeeded() {
|
||||||
while queue.subviews.count < minimumQueueDepth {
|
operationQueue.add(WebViewProviderReplenishQueueOperation(queue: queue, articleIconSchemeHandler: articleIconSchemeHandler))
|
||||||
enqueueWebView(PreloadedWebView(articleIconSchemeHandler: articleIconSchemeHandler))
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func dequeueWebView(completion: @escaping (PreloadedWebView) -> ()) {
|
func dequeueWebView(completion: @escaping (PreloadedWebView) -> ()) {
|
||||||
|
operationQueue.add(WebViewProviderDequeueOperation(queue: queue, articleIconSchemeHandler: articleIconSchemeHandler, completion: completion))
|
||||||
|
operationQueue.add(WebViewProviderReplenishQueueOperation(queue: queue, articleIconSchemeHandler: articleIconSchemeHandler))
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
class WebViewProviderFlushQueueOperation: MainThreadOperation {
|
||||||
|
|
||||||
|
// MainThreadOperation
|
||||||
|
public var isCanceled = false
|
||||||
|
public var id: Int?
|
||||||
|
public weak var operationDelegate: MainThreadOperationDelegate?
|
||||||
|
public var name: String? = "WebViewProviderFlushQueueOperation"
|
||||||
|
public var completionBlock: MainThreadOperation.MainThreadOperationCompletionBlock?
|
||||||
|
|
||||||
|
private var queue: UIView
|
||||||
|
|
||||||
|
init(queue: UIView) {
|
||||||
|
self.queue = queue
|
||||||
|
}
|
||||||
|
|
||||||
|
func run() {
|
||||||
|
queue.subviews.forEach { $0.removeFromSuperview() }
|
||||||
|
self.operationDelegate?.operationDidComplete(self)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
class WebViewProviderReplenishQueueOperation: MainThreadOperation {
|
||||||
|
|
||||||
|
// MainThreadOperation
|
||||||
|
public var isCanceled = false
|
||||||
|
public var id: Int?
|
||||||
|
public weak var operationDelegate: MainThreadOperationDelegate?
|
||||||
|
public var name: String? = "WebViewProviderReplenishQueueOperation"
|
||||||
|
public var completionBlock: MainThreadOperation.MainThreadOperationCompletionBlock?
|
||||||
|
|
||||||
|
private let minimumQueueDepth = 3
|
||||||
|
|
||||||
|
private var queue: UIView
|
||||||
|
private var articleIconSchemeHandler: ArticleIconSchemeHandler
|
||||||
|
|
||||||
|
init(queue: UIView, articleIconSchemeHandler: ArticleIconSchemeHandler) {
|
||||||
|
self.queue = queue
|
||||||
|
self.articleIconSchemeHandler = articleIconSchemeHandler
|
||||||
|
}
|
||||||
|
|
||||||
|
func run() {
|
||||||
|
while queue.subviews.count < minimumQueueDepth {
|
||||||
|
let webView = PreloadedWebView(articleIconSchemeHandler: articleIconSchemeHandler)
|
||||||
|
queue.insertSubview(webView, at: 0)
|
||||||
|
webView.preload()
|
||||||
|
}
|
||||||
|
self.operationDelegate?.operationDidComplete(self)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
class WebViewProviderDequeueOperation: MainThreadOperation {
|
||||||
|
|
||||||
|
// MainThreadOperation
|
||||||
|
public var isCanceled = false
|
||||||
|
public var id: Int?
|
||||||
|
public weak var operationDelegate: MainThreadOperationDelegate?
|
||||||
|
public var name: String? = "WebViewProviderFlushQueueOperation"
|
||||||
|
public var completionBlock: MainThreadOperation.MainThreadOperationCompletionBlock?
|
||||||
|
|
||||||
|
private var queue: UIView
|
||||||
|
private var articleIconSchemeHandler: ArticleIconSchemeHandler
|
||||||
|
private var completion: (PreloadedWebView) -> ()
|
||||||
|
|
||||||
|
init(queue: UIView, articleIconSchemeHandler: ArticleIconSchemeHandler, completion: @escaping (PreloadedWebView) -> ()) {
|
||||||
|
self.queue = queue
|
||||||
|
self.articleIconSchemeHandler = articleIconSchemeHandler
|
||||||
|
self.completion = completion
|
||||||
|
}
|
||||||
|
|
||||||
|
func run() {
|
||||||
if let webView = queue.subviews.last as? PreloadedWebView {
|
if let webView = queue.subviews.last as? PreloadedWebView {
|
||||||
webView.ready { preloadedWebView in
|
webView.ready { preloadedWebView in
|
||||||
preloadedWebView.removeFromSuperview()
|
preloadedWebView.removeFromSuperview()
|
||||||
self.replenishQueueIfNeeded()
|
self.completion(preloadedWebView)
|
||||||
completion(preloadedWebView)
|
self.operationDelegate?.operationDidComplete(self)
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@ -50,18 +124,11 @@ class WebViewProvider: NSObject {
|
|||||||
assertionFailure("Creating PreloadedWebView in \(#function); queue has run dry.")
|
assertionFailure("Creating PreloadedWebView in \(#function); queue has run dry.")
|
||||||
|
|
||||||
let webView = PreloadedWebView(articleIconSchemeHandler: articleIconSchemeHandler)
|
let webView = PreloadedWebView(articleIconSchemeHandler: articleIconSchemeHandler)
|
||||||
webView.ready { preloadedWebView in
|
|
||||||
self.replenishQueueIfNeeded()
|
|
||||||
completion(preloadedWebView)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func enqueueWebView(_ webView: PreloadedWebView) {
|
|
||||||
guard queue.subviews.count < maximumQueueDepth else {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
queue.insertSubview(webView, at: 0)
|
|
||||||
webView.preload()
|
webView.preload()
|
||||||
|
webView.ready { preloadedWebView in
|
||||||
|
self.completion(preloadedWebView)
|
||||||
|
self.operationDelegate?.operationDidComplete(self)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
|
|
||||||
// High Level Settings common to both the iOS application and any extensions we bundle with it
|
// High Level Settings common to both the iOS application and any extensions we bundle with it
|
||||||
MARKETING_VERSION = 5.0.1
|
MARKETING_VERSION = 5.0.1
|
||||||
CURRENT_PROJECT_VERSION = 44
|
CURRENT_PROJECT_VERSION = 45
|
||||||
|
|
||||||
ALWAYS_EMBED_SWIFT_STANDARD_LIBRARIES = YES
|
ALWAYS_EMBED_SWIFT_STANDARD_LIBRARIES = YES
|
||||||
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon
|
ASSETCATALOG_COMPILER_APPICON_NAME = AppIcon
|
||||||
|
Loading…
x
Reference in New Issue
Block a user