diff --git a/Multiplatform/Shared/AppAssets.swift b/Multiplatform/Shared/AppAssets.swift index ef22695f5..32811e2c9 100644 --- a/Multiplatform/Shared/AppAssets.swift +++ b/Multiplatform/Shared/AppAssets.swift @@ -110,6 +110,18 @@ struct AppAssets { return Image(systemName: "info.circle") }() + #if os(macOS) + static var iconBackgroundColor: NSColor = { + return NSColor(named: "IconBackgroundColor")! + }() + #endif + + #if os(iOS) + static var iconBackgroundColor: UIColor = { + return UIColor(named: "IconBackgroundColor")! + }() + #endif + static var nextArticleImage: Image = { return Image(systemName: "chevron.down") }() diff --git a/Multiplatform/Shared/AppDefaults.swift b/Multiplatform/Shared/AppDefaults.swift index 9cee5fd2e..52452c851 100644 --- a/Multiplatform/Shared/AppDefaults.swift +++ b/Multiplatform/Shared/AppDefaults.swift @@ -193,6 +193,8 @@ final class AppDefaults: ObservableObject { // MARK: Articles @AppStorage(wrappedValue: false, Key.articleFullscreenAvailable, store: store) var articleFullscreenAvailable: Bool + @AppStorage(wrappedValue: false, Key.articleFullscreenEnabled, store: store) var articleFullscreenEnabled: Bool + // MARK: Refresh var lastRefresh: Date? { set { diff --git a/Multiplatform/Shared/Article/ArticleContainerView.swift b/Multiplatform/Shared/Article/ArticleContainerView.swift index 50f247015..dbf6820b6 100644 --- a/Multiplatform/Shared/Article/ArticleContainerView.swift +++ b/Multiplatform/Shared/Article/ArticleContainerView.swift @@ -16,13 +16,8 @@ struct ArticleContainerView: View { var article: Article @ViewBuilder var body: some View { - ArticleView() + ArticleView(sceneModel: sceneModel, articleModel: articleModel, article: article) .modifier(ArticleToolbarModifier()) - .environmentObject(articleModel) - .onAppear { - sceneModel.articleModel = articleModel - articleModel.delegate = sceneModel - } } } diff --git a/Multiplatform/Shared/Article/ArticleModel.swift b/Multiplatform/Shared/Article/ArticleModel.swift index 7e2248d0b..8dc100836 100644 --- a/Multiplatform/Shared/Article/ArticleModel.swift +++ b/Multiplatform/Shared/Article/ArticleModel.swift @@ -14,12 +14,37 @@ import Account import Articles protocol ArticleModelDelegate: class { - func timelineRequestedWebFeedSelection(_: TimelineModel, webFeed: WebFeed) + #if os(iOS) + var webViewProvider: WebViewProvider? { get } + #endif + func findPrevArticle(_: ArticleModel, article: Article) -> Article? + func findNextArticle(_: ArticleModel, article: Article) -> Article? + func selectArticle(_: ArticleModel, article: Article) } class ArticleModel: ObservableObject { weak var delegate: ArticleModelDelegate? + #if os(iOS) + var webViewProvider: WebViewProvider? { + return delegate?.webViewProvider + } + #endif + + // MARK: API + + func findPrevArticle(_ article: Article) -> Article? { + return delegate?.findPrevArticle(self, article: article) + } + + func findNextArticle(_ article: Article) -> Article? { + return delegate?.findNextArticle(self, article: article) + } + + func selectArticle(_ article: Article) { + delegate?.selectArticle(self, article: article) + } + } diff --git a/Multiplatform/Shared/Article/blank.html b/Multiplatform/Shared/Article/blank.html new file mode 100644 index 000000000..6e02cf3a6 --- /dev/null +++ b/Multiplatform/Shared/Article/blank.html @@ -0,0 +1,11 @@ + + + + + + + diff --git a/Multiplatform/Shared/Article/main_multiplatform.js b/Multiplatform/Shared/Article/main_multiplatform.js new file mode 100644 index 000000000..2de36d94b --- /dev/null +++ b/Multiplatform/Shared/Article/main_multiplatform.js @@ -0,0 +1,481 @@ +var activeImageViewer = null; + +class ImageViewer { + constructor(img) { + this.img = img; + this.loadingInterval = null; + this.activityIndicator = ""; + } + + isLoaded() { + return this.img.classList.contains("nnwLoaded"); + } + + clicked() { + this.showLoadingIndicator(); + if (this.isLoaded()) { + this.showViewer(); + } else { + var callback = () => { + if (this.isLoaded()) { + clearInterval(this.loadingInterval); + this.showViewer(); + } + } + this.loadingInterval = setInterval(callback, 100); + } + } + cancel() { + clearInterval(this.loadingInterval); + this.hideLoadingIndicator(); + } + + showViewer() { + this.hideLoadingIndicator(); + + var canvas = document.createElement("canvas"); + var pixelRatio = window.devicePixelRatio; + do { + canvas.width = this.img.naturalWidth * pixelRatio; + canvas.height = this.img.naturalHeight * pixelRatio; + pixelRatio--; + } while (pixelRatio > 0 && canvas.width * canvas.height > 16777216) + canvas.getContext("2d").drawImage(this.img, 0, 0, canvas.width, canvas.height); + + const rect = this.img.getBoundingClientRect(); + const message = { + x: rect.x, + y: rect.y, + width: rect.width, + height: rect.height, + imageTitle: this.img.title, + imageURL: canvas.toDataURL(), + }; + + var jsonMessage = JSON.stringify(message); + window.webkit.messageHandlers.imageWasClicked.postMessage(jsonMessage); + } + + hideImage() { + this.img.style.opacity = 0; + } + + showImage() { + this.img.style.opacity = 1 + } + + showLoadingIndicator() { + var wrapper = document.createElement("div"); + wrapper.classList.add("activityIndicatorWrap"); + this.img.parentNode.insertBefore(wrapper, this.img); + wrapper.appendChild(this.img); + + var activityIndicatorImg = document.createElement("img"); + activityIndicatorImg.classList.add("activityIndicator"); + activityIndicatorImg.style.opacity = 0; + activityIndicatorImg.src = this.activityIndicator; + wrapper.appendChild(activityIndicatorImg); + + activityIndicatorImg.style.opacity = 1; + } + + hideLoadingIndicator() { + var wrapper = this.img.parentNode; + if (wrapper.classList.contains("activityIndicatorWrap")) { + var wrapperParent = wrapper.parentNode; + wrapperParent.insertBefore(this.img, wrapper); + wrapperParent.removeChild(wrapper); + } + } + + static init() { + cancelImageLoad(); + + // keep track of when an image has finished downloading for ImageViewer + document.querySelectorAll("img").forEach(element => { + element.onload = function() { + this.classList.add("nnwLoaded"); + } + }); + + // Add the click listener for images + window.onclick = function(event) { + if (event.target.matches("img") && !event.target.classList.contains("nnw-nozoom")) { + if (activeImageViewer && activeImageViewer.img === event.target) { + cancelImageLoad(); + } else { + cancelImageLoad(); + activeImageViewer = new ImageViewer(event.target); + activeImageViewer.clicked(); + } + } + } + } +} + +function cancelImageLoad() { + if (activeImageViewer) { + activeImageViewer.cancel(); + activeImageViewer = null; + } +} + +function hideClickedImage() { + if (activeImageViewer) { + activeImageViewer.hideImage(); + } +} + +// Used to animate the transition from a fullscreen image +function showClickedImage() { + if (activeImageViewer) { + activeImageViewer.showImage(); + } + window.webkit.messageHandlers.imageWasShown.postMessage(""); +} + +function showFeedInspectorSetup() { + document.getElementById("nnwImageIcon").onclick = function(event) { + window.webkit.messageHandlers.showFeedInspector.postMessage(""); + } +} + +function postRenderProcessing() { + ImageViewer.init(); + showFeedInspectorSetup(); +} + + +function makeHighlightRect({left, top, width, height}, offsetTop=0, offsetLeft=0) { + const overlay = document.createElement('a'); + + Object.assign(overlay.style, { + position: 'absolute', + left: `${Math.floor(left + offsetLeft)}px`, + top: `${Math.floor(top + offsetTop)}px`, + width: `${Math.ceil(width)}px`, + height: `${Math.ceil(height)}px`, + backgroundColor: 'rgba(200, 220, 10, 0.4)', + pointerEvents: 'none' + }); + + return overlay; +} + +function clearHighlightRects() { + let container = document.getElementById('nnw:highlightContainer') + if (container) container.remove(); +} + +function highlightRects(rects, clearOldRects=true, makeHighlightRect=makeHighlightRect) { + const article = document.querySelector('article'); + let container = document.getElementById('nnw:highlightContainer'); + + article.style.position = 'relative'; + + if (container && clearOldRects) + container.remove(); + + container = document.createElement('div'); + container.id = 'nnw:highlightContainer'; + article.appendChild(container); + + const {top, left} = article.getBoundingClientRect(); + return Array.from(rects, rect => + container.appendChild(makeHighlightRect(rect, -top, -left)) + ); +} + +FinderResult = class { + constructor(result) { + Object.assign(this, result); + } + + range() { + const range = document.createRange(); + range.setStart(this.node, this.offset); + range.setEnd(this.node, this.offsetEnd); + return range; + } + + bounds() { + return this.range().getBoundingClientRect(); + } + + rects() { + return this.range().getClientRects(); + } + + highlight({clearOldRects=true, fn=makeHighlightRect} = {}) { + highlightRects(this.rects(), clearOldRects, fn); + } + + scrollTo() { + scrollToRect(this.bounds(), this.node); + } + + toJSON() { + return { + rects: Array.from(this.rects()), + bounds: this.bounds(), + index: this.index, + matchGroups: this.match + }; + } + + toJSONString() { + return JSON.stringify(this.toJSON()); + } +} + +Finder = class { + constructor(pattern, options) { + if (!pattern.global) { + pattern = new RegExp(pattern, 'g'); + } + + this.pattern = pattern; + this.lastResult = null; + this._nodeMatches = []; + this.options = { + rootSelector: '.articleBody', + startNode: null, + startOffset: null, + } + + this.resultIndex = -1 + + Object.assign(this.options, options); + + this.walker = document.createTreeWalker(this.root, NodeFilter.SHOW_TEXT); + } + + get root() { + return document.querySelector(this.options.rootSelector) + } + + get count() { + const node = this.walker.currentNode; + const index = this.resultIndex; + this.reset(); + + let result, count = 0; + while ((result = this.next())) ++count; + + this.resultIndex = index; + this.walker.currentNode = node; + + return count; + } + + reset() { + this.walker.currentNode = this.options.startNode || this.root; + this.resultIndex = -1; + } + + [Symbol.iterator]() { + return this; + } + + next({wrap = false} = {}) { + const { startNode } = this.options; + const { pattern, walker } = this; + + let { node, matchIndex = -1 } = this.lastResult || { node: startNode }; + + while (true) { + if (!node) + node = walker.nextNode(); + + if (!node) { + if (!wrap || this.resultIndex < 0) break; + + this.reset(); + + continue; + } + + let nextIndex = matchIndex + 1; + let matches = this._nodeMatches; + + if (!matches.length) { + matches = Array.from(node.textContent.matchAll(pattern)); + nextIndex = 0; + } + + if (matches[nextIndex]) { + this._nodeMatches = matches; + const m = matches[nextIndex]; + + this.lastResult = new FinderResult({ + node, + offset: m.index, + offsetEnd: m.index + m[0].length, + text: m[0], + match: m, + matchIndex: nextIndex, + index: ++this.resultIndex, + }); + + return { value: this.lastResult, done: false }; + } + + this._nodeMatches = []; + node = null; + } + + return { value: undefined, done: true }; + } + + /// TODO Call when the search text changes + retry() { + if (this.lastResult) { + this.lastResult.offsetEnd = this.lastResult.offset; + } + + } + + toJSON() { + const results = Array.from(this); + } +} + +function scrollParent(node) { + let elt = node.nodeType === Node.ELEMENT_NODE ? node : node.parentElement; + + while (elt) { + if (elt.scrollHeight > elt.clientHeight) + return elt; + elt = elt.parentElement; + } +} + +function scrollToRect({top, height}, node, pad=20, padBottom=60) { + const scrollToTop = top - pad; + + let scrollBy = scrollToTop; + + if (scrollToTop >= 0) { + const visible = window.visualViewport; + const scrollToBottom = top + height + padBottom - visible.height; + // The top of the rect is already in the viewport + if (scrollToBottom <= 0 || scrollToTop === 0) + // Don't need to scroll up--or can't + return; + + scrollBy = Math.min(scrollToBottom, scrollBy); + } + + scrollParent(node).scrollBy({ top: scrollBy }); +} + +function withEncodedArg(fn) { + return function(encodedData, ...rest) { + const data = encodedData && JSON.parse(atob(encodedData)); + return fn(data, ...rest); + } +} + +function escapeRegex(s) { + return s.replace(/[.?*+^$\\()[\]{}]/g, '\\$&'); +} + +class FindState { + constructor(options) { + let { text, caseSensitive, regex } = options; + + if (!regex) + text = escapeRegex(text); + + const finder = new Finder(new RegExp(text, caseSensitive ? 'g' : 'ig')); + this.results = Array.from(finder); + this.index = -1; + this.options = options; + } + + get selected() { + return this.index > -1 ? this.results[this.index] : null; + } + + toJSON() { + return { + index: this.index > -1 ? this.index : null, + results: this.results, + count: this.results.length + }; + } + + selectNext(step=1) { + const index = this.index + step; + const result = this.results[index]; + if (result) { + this.index = index; + result.highlight(); + result.scrollTo(); + } + return result; + } + + selectPrevious() { + return this.selectNext(-1); + } +} + +CurrentFindState = null; + +const ExcludeKeys = new Set(['top', 'right', 'bottom', 'left']); +updateFind = withEncodedArg(options => { + // TODO Start at the current result position + // TODO Introduce slight delay, cap the number of results, and report results asynchronously + + let newFindState; + if (!options || !options.text) { + clearHighlightRects(); + return + } + + try { + newFindState = new FindState(options); + } catch (err) { + clearHighlightRects(); + throw err; + } + + if (newFindState.results.length) { + let selected = CurrentFindState && CurrentFindState.selected; + let selectIndex = 0; + if (selected) { + let {node: currentNode, offset: currentOffset} = selected; + selectIndex = newFindState.results.findIndex(r => { + if (r.node === currentNode) { + return r.offset >= currentOffset; + } + + let relation = currentNode.compareDocumentPosition(r.node); + return Boolean(relation & Node.DOCUMENT_POSITION_FOLLOWING); + }); + } + + newFindState.selectNext(selectIndex+1); + } else { + clearHighlightRects(); + } + + CurrentFindState = newFindState; + return btoa(JSON.stringify(CurrentFindState, (k, v) => (ExcludeKeys.has(k) ? undefined : v))); +}); + +selectNextResult = withEncodedArg(options => { + if (CurrentFindState) + CurrentFindState.selectNext(); +}); + +selectPreviousResult = withEncodedArg(options => { + if (CurrentFindState) + CurrentFindState.selectPrevious(); +}); + +function endFind() { + clearHighlightRects() + CurrentFindState = null; +} diff --git a/Multiplatform/Shared/Article/page.html b/Multiplatform/Shared/Article/page.html new file mode 100644 index 000000000..0d7d9676d --- /dev/null +++ b/Multiplatform/Shared/Article/page.html @@ -0,0 +1,22 @@ + + + [[title]] + + + + + + + + + + [[body]] + + diff --git a/Multiplatform/Shared/Article/styleSheet.css b/Multiplatform/Shared/Article/styleSheet.css new file mode 100644 index 000000000..98e09e971 --- /dev/null +++ b/Multiplatform/Shared/Article/styleSheet.css @@ -0,0 +1,66 @@ +body { + margin-top: 3px; + margin-bottom: 20px; + padding-left: 20px; + padding-right: 20px; + + word-break: break-word; + -webkit-hyphens: auto; + -webkit-text-size-adjust: none; +} + +:root { + color-scheme: light dark; + font: -apple-system-body; + font-size: [[font-size]]px; + --primary-accent-color: #086AEE; + --secondary-accent-color: #086AEE; + --block-quote-border-color: rgba(8, 106, 238, 0.75); +} + +@media(prefers-color-scheme: dark) { + :root { + --primary-accent-color: #2D80F1; + --secondary-accent-color: #5E9EF4; + --block-quote-border-color: rgba(94, 158, 244, 0.75); + --header-table-border-color: rgba(255, 255, 255, 0.2); + } +} + +body a, body a:link, body a:visited { + color: var(--secondary-accent-color); +} +body .header { + font: -apple-system-body; + font-size: [[font-size]]px; +} +body .header a:link, body .header a:visited { + color: var(--primary-accent-color); +} + +.avatar img { + border-radius: 4px; +} + +pre { + border: 1px solid var(--secondary-accent-color); + padding: 5px; +} + +.nnw-overflow table { + border: 1px solid var(--secondary-accent-color); +} + +.activityIndicatorWrap { + position: relative; +} + +.activityIndicator { + z-index: 1; + width: 64px; + height: 64px; + position: absolute; + top: 50%; + left: 50%; + transform: translate(-50%, -50%); +} diff --git a/Multiplatform/Shared/Assets.xcassets/IconBackgroundColor.colorset/Contents.json b/Multiplatform/Shared/Assets.xcassets/IconBackgroundColor.colorset/Contents.json new file mode 100644 index 000000000..49db4ebbe --- /dev/null +++ b/Multiplatform/Shared/Assets.xcassets/IconBackgroundColor.colorset/Contents.json @@ -0,0 +1,38 @@ +{ + "colors" : [ + { + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0.929", + "green" : "0.922", + "red" : "0.922" + } + }, + "idiom" : "universal" + }, + { + "appearances" : [ + { + "appearance" : "luminosity", + "value" : "dark" + } + ], + "color" : { + "color-space" : "srgb", + "components" : { + "alpha" : "1.000", + "blue" : "0.220", + "green" : "0.220", + "red" : "0.220" + } + }, + "idiom" : "universal" + } + ], + "info" : { + "author" : "xcode", + "version" : 1 + } +} diff --git a/Multiplatform/Shared/SceneModel.swift b/Multiplatform/Shared/SceneModel.swift index 250df5031..67f198928 100644 --- a/Multiplatform/Shared/SceneModel.swift +++ b/Multiplatform/Shared/SceneModel.swift @@ -8,6 +8,7 @@ import Foundation import Account +import Articles final class SceneModel: ObservableObject { @@ -18,11 +19,25 @@ final class SceneModel: ObservableObject { var articleModel: ArticleModel? private var refreshProgressModel: RefreshProgressModel? = nil + #if os(iOS) + private var _webViewProvider: WebViewProvider? = nil + #endif + // MARK: API + func startup() { self.refreshProgressModel = RefreshProgressModel() self.refreshProgressModel!.$state.assign(to: self.$refreshProgressState) + + #if os(iOS) + self._webViewProvider = WebViewProvider(sceneModel: self) + #endif } + + func articleFor(_ articleID: String) -> Article? { + return timelineModel?.articleFor(articleID) + } + } // MARK: SidebarModelDelegate @@ -48,6 +63,29 @@ extension SceneModel: TimelineModelDelegate { // MARK: ArticleModelDelegate extension SceneModel: ArticleModelDelegate { + + #if os(iOS) + var webViewProvider: WebViewProvider? { + return _webViewProvider + } + #endif + func findPrevArticle(_: ArticleModel, article: Article) -> Article? { + return timelineModel?.findPrevArticle(article) + } + + func findNextArticle(_: ArticleModel, article: Article) -> Article? { + return timelineModel?.findNextArticle(article) + } + + func selectArticle(_: ArticleModel, article: Article) { + timelineModel?.selectArticle(article) + } + +} + +// MARK: Private + +private extension SceneModel { } diff --git a/Multiplatform/Shared/Timeline/TimeilneToolbarModifier.swift b/Multiplatform/Shared/Timeline/TimeilneToolbarModifier.swift index e23cdc368..24fb9ae45 100644 --- a/Multiplatform/Shared/Timeline/TimeilneToolbarModifier.swift +++ b/Multiplatform/Shared/Timeline/TimeilneToolbarModifier.swift @@ -6,8 +6,6 @@ // Copyright © 2020 Ranchero Software. All rights reserved. // -import Foundation - import SwiftUI struct TimelineToolbarModifier: ViewModifier { diff --git a/Multiplatform/Shared/Timeline/TimelineContainerView.swift b/Multiplatform/Shared/Timeline/TimelineContainerView.swift index 92b371cbb..8329e14d7 100644 --- a/Multiplatform/Shared/Timeline/TimelineContainerView.swift +++ b/Multiplatform/Shared/Timeline/TimelineContainerView.swift @@ -18,6 +18,7 @@ struct TimelineContainerView: View { @ViewBuilder var body: some View { if let feed = feed { TimelineView() + .modifier(TimelineTitleModifier(title: feed.nameForDisplay)) .modifier(TimelineToolbarModifier()) .environmentObject(timelineModel) .onAppear { diff --git a/Multiplatform/Shared/Timeline/TimelineModel.swift b/Multiplatform/Shared/Timeline/TimelineModel.swift index 307bf0f03..3d448761a 100644 --- a/Multiplatform/Shared/Timeline/TimelineModel.swift +++ b/Multiplatform/Shared/Timeline/TimelineModel.swift @@ -27,8 +27,21 @@ class TimelineModel: ObservableObject { private var exceptionArticleFetcher: ArticleFetcher? private var isReadFiltered = false - private var articles = [Article]() + private var articles = [Article]() { + didSet { + articleDictionaryNeedsUpdate = true + } + } + private var articleDictionaryNeedsUpdate = true + private var _idToArticleDictionary = [String: Article]() + private var idToAticleDictionary: [String: Article] { + if articleDictionaryNeedsUpdate { + rebuildArticleDictionaries() + } + return _idToArticleDictionary + } + private var sortDirection = AppDefaults.shared.timelineSortDirection { didSet { if sortDirection != oldValue { @@ -63,6 +76,28 @@ class TimelineModel: ObservableObject { } } + func articleFor(_ articleID: String) -> Article? { + return idToAticleDictionary[articleID] + } + + func findPrevArticle(_ article: Article) -> Article? { + guard let index = articles.firstIndex(of: article), index > 0 else { + return nil + } + return articles[index - 1] + } + + func findNextArticle(_ article: Article) -> Article? { + guard let index = articles.firstIndex(of: article), index + 1 != articles.count else { + return nil + } + return articles[index + 1] + } + + func selectArticle(_ article: Article) { + // TODO: Implement me! + } + } // MARK: Private @@ -82,6 +117,15 @@ private extension TimelineModel { // restoreSelection(savedSelection) } + func rebuildArticleDictionaries() { + var idDictionary = [String: Article]() + articles.forEach { article in + idDictionary[article.articleID] = article + } + _idToArticleDictionary = idDictionary + articleDictionaryNeedsUpdate = false + } + // MARK: Article Fetching func fetchAndReplaceArticlesAsync() { diff --git a/Multiplatform/Shared/Timeline/TimelineTitleModifier.swift b/Multiplatform/Shared/Timeline/TimelineTitleModifier.swift new file mode 100644 index 000000000..a1956c77a --- /dev/null +++ b/Multiplatform/Shared/Timeline/TimelineTitleModifier.swift @@ -0,0 +1,23 @@ +// +// TimelineTitleModifier.swift +// NetNewsWire +// +// Created by Maurice Parker on 7/6/20. +// Copyright © 2020 Ranchero Software. All rights reserved. +// + +import SwiftUI + +struct TimelineTitleModifier: ViewModifier { + + var title: String + + func body(content: Content) -> some View { + #if os(macOS) + return content + #endif + #if os(iOS) + return content.navigationBarTitle(Text(verbatim: title), displayMode: .inline) + #endif + } +} diff --git a/Multiplatform/iOS/Article/ArticleExtractorButtonState.swift b/Multiplatform/iOS/Article/ArticleExtractorButtonState.swift new file mode 100644 index 000000000..6c9f6d04a --- /dev/null +++ b/Multiplatform/iOS/Article/ArticleExtractorButtonState.swift @@ -0,0 +1,16 @@ +// +// ArticleExtractorButtonState.swift +// Multiplatform iOS +// +// Created by Maurice Parker on 7/6/20. +// Copyright © 2020 Ranchero Software. All rights reserved. +// + +import Foundation + +enum ArticleExtractorButtonState { + case error + case animated + case on + case off +} diff --git a/Multiplatform/iOS/Article/ArticleIconSchemeHandler.swift b/Multiplatform/iOS/Article/ArticleIconSchemeHandler.swift new file mode 100644 index 000000000..4e18cccb7 --- /dev/null +++ b/Multiplatform/iOS/Article/ArticleIconSchemeHandler.swift @@ -0,0 +1,60 @@ +// +// ArticleIconSchemeHandler.swift +// Multiplatform iOS +// +// Created by Maurice Parker on 7/6/20. +// Copyright © 2020 Ranchero Software. All rights reserved. +// + +import Foundation +import WebKit +import Articles + +class ArticleIconSchemeHandler: NSObject, WKURLSchemeHandler { + + weak var sceneModel: SceneModel? + + init(sceneModel: SceneModel) { + self.sceneModel = sceneModel + } + + func webView(_ webView: WKWebView, start urlSchemeTask: WKURLSchemeTask) { + + guard let url = urlSchemeTask.request.url, let sceneModel = sceneModel else { + urlSchemeTask.didFailWithError(URLError(.fileDoesNotExist)) + return + } + + guard let components = URLComponents(url: url, resolvingAgainstBaseURL: false) else { + return + } + let articleID = components.path + guard let iconImage = sceneModel.articleFor(articleID)?.iconImage() else { + urlSchemeTask.didFailWithError(URLError(.fileDoesNotExist)) + return + } + + let iconView = IconView(frame: CGRect(x: 0, y: 0, width: 48, height: 48)) + iconView.iconImage = iconImage + let renderedImage = iconView.asImage() + + guard let data = renderedImage.dataRepresentation() else { + urlSchemeTask.didFailWithError(URLError(.fileDoesNotExist)) + return + } + + let headerFields = ["Cache-Control": "no-cache"] + if let response = HTTPURLResponse(url: url, statusCode: 200, httpVersion: nil, headerFields: headerFields) { + urlSchemeTask.didReceive(response) + urlSchemeTask.didReceive(data) + urlSchemeTask.didFinish() + } + + } + + func webView(_ webView: WKWebView, stop urlSchemeTask: WKURLSchemeTask) { + urlSchemeTask.didFailWithError(URLError(.unknown)) + } + +} + diff --git a/Multiplatform/iOS/Article/ArticleView.swift b/Multiplatform/iOS/Article/ArticleView.swift index c9a807d2b..045c29501 100644 --- a/Multiplatform/iOS/Article/ArticleView.swift +++ b/Multiplatform/iOS/Article/ArticleView.swift @@ -7,15 +7,43 @@ // import SwiftUI +import Articles -struct ArticleView: View { - var body: some View { - Text(/*@START_MENU_TOKEN@*/"Hello, World!"/*@END_MENU_TOKEN@*/) - } +final class ArticleView: UIViewControllerRepresentable { + + var sceneModel: SceneModel + var articleModel: ArticleModel + var article: Article + + init(sceneModel: SceneModel, articleModel: ArticleModel, article: Article) { + self.sceneModel = sceneModel + self.articleModel = articleModel + self.article = article + sceneModel.articleModel = articleModel + articleModel.delegate = sceneModel + } + + func makeUIViewController(context: Context) -> ArticleViewController { + let controller = ArticleViewController() + controller.articleModel = articleModel + controller.article = article + return controller + } + + func updateUIViewController(_ uiViewController: ArticleViewController, context: Context) { + + } + } -struct ArticleView_Previews: PreviewProvider { - static var previews: some View { - ArticleView() - } -} +//struct ArticleView: View { +// +// var sceneModel: SceneModel +// var articleModel: ArticleModel +// var article: Article +// +// var body: some View { +// ArticleViewControllerAdapter(sceneModel: sceneModel, articleModel: articleModel, article: article) +// } +// +//} diff --git a/Multiplatform/iOS/Article/ArticleViewController.swift b/Multiplatform/iOS/Article/ArticleViewController.swift new file mode 100644 index 000000000..4eedc9f5e --- /dev/null +++ b/Multiplatform/iOS/Article/ArticleViewController.swift @@ -0,0 +1,172 @@ +// +// ArticleViewController.swift +// Multiplatform iOS +// +// Created by Maurice Parker on 7/6/20. +// Copyright © 2020 Ranchero Software. All rights reserved. +// + +import UIKit +import WebKit +import Account +import Articles +import SafariServices + +class ArticleViewController: UIViewController { + + weak var articleModel: ArticleModel? + + private var pageViewController: UIPageViewController! + + private var currentWebViewController: WebViewController? { + return pageViewController?.viewControllers?.first as? WebViewController + } + + var article: Article? { + didSet { + if let controller = currentWebViewController, controller.article != article { + controller.setArticle(article) + DispatchQueue.main.async { + // You have to set the view controller to clear out the UIPageViewController child controller cache. + // You also have to do it in an async call or you will get a strange assertion error. + self.pageViewController.setViewControllers([controller], direction: .forward, animated: false, completion: nil) + } + } + } + } + + override func viewDidLoad() { + super.viewDidLoad() + + pageViewController = UIPageViewController(transitionStyle: .scroll, navigationOrientation: .horizontal, options: [:]) + pageViewController.delegate = self + pageViewController.dataSource = self + + pageViewController.view.translatesAutoresizingMaskIntoConstraints = false + view.addSubview(pageViewController.view) + addChild(pageViewController!) + NSLayoutConstraint.activate([ + view.leadingAnchor.constraint(equalTo: pageViewController.view.leadingAnchor), + view.trailingAnchor.constraint(equalTo: pageViewController.view.trailingAnchor), + view.topAnchor.constraint(equalTo: pageViewController.view.topAnchor), + view.bottomAnchor.constraint(equalTo: pageViewController.view.bottomAnchor) + ]) + + let controller = createWebViewController(article, updateView: true) + self.pageViewController.setViewControllers([controller], direction: .forward, animated: false, completion: nil) + + } + + // MARK: API + + func focus() { + currentWebViewController?.focus() + } + + func canScrollDown() -> Bool { + return currentWebViewController?.canScrollDown() ?? false + } + + func scrollPageDown() { + currentWebViewController?.scrollPageDown() + } + + func stopArticleExtractorIfProcessing() { + currentWebViewController?.stopArticleExtractorIfProcessing() + } + +} + + +// MARK: WebViewControllerDelegate + +extension ArticleViewController: WebViewControllerDelegate { + + func webViewController(_ webViewController: WebViewController, articleExtractorButtonStateDidUpdate buttonState: ArticleExtractorButtonState) { + if webViewController === currentWebViewController { +// articleExtractorButton.buttonState = buttonState + } + } + +} + +// MARK: UIPageViewControllerDataSource + +extension ArticleViewController: UIPageViewControllerDataSource { + + func pageViewController(_ pageViewController: UIPageViewController, viewControllerBefore viewController: UIViewController) -> UIViewController? { + guard let webViewController = viewController as? WebViewController, + let currentArticle = webViewController.article, + let article = articleModel?.findPrevArticle(currentArticle) else { + return nil + } + return createWebViewController(article) + } + + func pageViewController(_ pageViewController: UIPageViewController, viewControllerAfter viewController: UIViewController) -> UIViewController? { + guard let webViewController = viewController as? WebViewController, + let currentArticle = webViewController.article, + let article = articleModel?.findNextArticle(currentArticle) else { + return nil + } + return createWebViewController(article) + } + +} + +// MARK: UIPageViewControllerDelegate + +extension ArticleViewController: UIPageViewControllerDelegate { + + func pageViewController(_ pageViewController: UIPageViewController, didFinishAnimating finished: Bool, previousViewControllers: [UIViewController], transitionCompleted completed: Bool) { + guard finished, completed else { return } + guard let article = currentWebViewController?.article else { return } + + articleModel?.selectArticle(article) +// articleExtractorButton.buttonState = currentWebViewController?.articleExtractorButtonState ?? .off + + previousViewControllers.compactMap({ $0 as? WebViewController }).forEach({ $0.stopWebViewActivity() }) + } + +} + +// MARK: UIGestureRecognizerDelegate + +extension ArticleViewController: UIGestureRecognizerDelegate { + + func gestureRecognizerShouldBegin(_ gestureRecognizer: UIGestureRecognizer) -> Bool { + return true + } + + func gestureRecognizer(_ gestureRecognizer: UIGestureRecognizer, shouldRecognizeSimultaneouslyWith otherGestureRecognizer: UIGestureRecognizer) -> Bool { + let point = gestureRecognizer.location(in: nil) + if point.x > 40 { + return true + } + return false + } + +} + +// MARK: Private + +private extension ArticleViewController { + + func createWebViewController(_ article: Article?, updateView: Bool = true) -> WebViewController { + let controller = WebViewController() + controller.articleModel = articleModel + controller.delegate = self + controller.setArticle(article, updateView: updateView) + return controller + } + + func resetWebViewController() { + articleModel?.webViewProvider?.flushQueue() + articleModel?.webViewProvider?.replenishQueueIfNeeded() + if let controller = currentWebViewController { + controller.fullReload() + self.pageViewController.setViewControllers([controller], direction: .forward, animated: false, completion: nil) + } + } + +} diff --git a/Multiplatform/iOS/Article/IconView.swift b/Multiplatform/iOS/Article/IconView.swift new file mode 100644 index 000000000..ebd24e5be --- /dev/null +++ b/Multiplatform/iOS/Article/IconView.swift @@ -0,0 +1,124 @@ +// +// IconView.swift +// Multiplatform iOS +// +// Created by Maurice Parker on 7/6/20. +// Copyright © 2020 Ranchero Software. All rights reserved. +// + +import UIKit + +final class IconView: UIView { + + var iconImage: IconImage? = nil { + didSet { + if iconImage !== oldValue { + imageView.image = iconImage?.image + + if self.traitCollection.userInterfaceStyle == .dark { + if self.iconImage?.isDark ?? false { + self.isDisconcernable = false + self.setNeedsLayout() + } else { + self.isDisconcernable = true + self.setNeedsLayout() + } + } else { + if self.iconImage?.isBright ?? false { + self.isDisconcernable = false + self.setNeedsLayout() + } else { + self.isDisconcernable = true + self.setNeedsLayout() + } + } + self.setNeedsLayout() + } + } + } + + private var isDisconcernable = true + + private let imageView: UIImageView = { + let imageView = UIImageView(image: AppAssets.faviconTemplateImage) + imageView.contentMode = .scaleAspectFit + imageView.clipsToBounds = true + imageView.layer.cornerRadius = 4.0 + return imageView + }() + + private var isVerticalBackgroundExposed: Bool { + return imageView.frame.size.height < bounds.size.height + } + + private var isSymbolImage: Bool { + return imageView.image?.isSymbolImage ?? false + } + + override init(frame: CGRect) { + super.init(frame: frame) + commonInit() + } + + required init?(coder: NSCoder) { + super.init(coder: coder) + commonInit() + } + + convenience init() { + self.init(frame: .zero) + } + + override func didMoveToSuperview() { + setNeedsLayout() + } + + override func layoutSubviews() { + imageView.setFrameIfNotEqual(rectForImageView()) + if (iconImage != nil && isVerticalBackgroundExposed && !isSymbolImage) || !isDisconcernable { + backgroundColor = AppAssets.iconBackgroundColor + } else { + backgroundColor = nil + } + } + +} + +private extension IconView { + + func commonInit() { + layer.cornerRadius = 4 + clipsToBounds = true + addSubview(imageView) + } + + func rectForImageView() -> CGRect { + guard let image = iconImage?.image else { + return CGRect.zero + } + + let imageSize = image.size + let viewSize = bounds.size + if imageSize.height == imageSize.width { + if imageSize.height >= viewSize.height * 0.75 { + // Close enough to viewSize to scale up the image. + return CGRect(x: 0.0, y: 0.0, width: viewSize.width, height: viewSize.height) + } + let offset = floor((viewSize.height - imageSize.height) / 2.0) + return CGRect(x: offset, y: offset, width: imageSize.width, height: imageSize.height) + } + else if imageSize.height > imageSize.width { + let factor = viewSize.height / imageSize.height + let width = imageSize.width * factor + let originX = floor((viewSize.width - width) / 2.0) + return CGRect(x: originX, y: 0.0, width: width, height: viewSize.height) + } + + // Wider than tall: imageSize.width > imageSize.height + let factor = viewSize.width / imageSize.width + let height = imageSize.height * factor + let originY = floor((viewSize.height - height) / 2.0) + return CGRect(x: 0.0, y: originY, width: viewSize.width, height: height) + } + +} diff --git a/Multiplatform/iOS/Article/ImageScrollView.swift b/Multiplatform/iOS/Article/ImageScrollView.swift new file mode 100644 index 000000000..31fe3289c --- /dev/null +++ b/Multiplatform/iOS/Article/ImageScrollView.swift @@ -0,0 +1,361 @@ +// +// ImageScrollView.swift +// Multiplatform iOS +// +// Created by Maurice Parker on 7/6/20. +// Copyright © 2020 Ranchero Software. All rights reserved. +// + +import UIKit + +@objc public protocol ImageScrollViewDelegate: UIScrollViewDelegate { + func imageScrollViewDidGestureSwipeUp(imageScrollView: ImageScrollView) + func imageScrollViewDidGestureSwipeDown(imageScrollView: ImageScrollView) +} + +open class ImageScrollView: UIScrollView { + + @objc public enum ScaleMode: Int { + case aspectFill + case aspectFit + case widthFill + case heightFill + } + + @objc public enum Offset: Int { + case begining + case center + } + + static let kZoomInFactorFromMinWhenDoubleTap: CGFloat = 2 + + @objc open var imageContentMode: ScaleMode = .widthFill + @objc open var initialOffset: Offset = .begining + + @objc public private(set) var zoomView: UIImageView? = nil + + @objc open weak var imageScrollViewDelegate: ImageScrollViewDelegate? + + var imageSize: CGSize = CGSize.zero + private var pointToCenterAfterResize: CGPoint = CGPoint.zero + private var scaleToRestoreAfterResize: CGFloat = 1.0 + var maxScaleFromMinScale: CGFloat = 3.0 + + var zoomedFrame: CGRect { + return zoomView?.frame ?? CGRect.zero + } + + override open var frame: CGRect { + willSet { + if frame.equalTo(newValue) == false && newValue.equalTo(CGRect.zero) == false && imageSize.equalTo(CGSize.zero) == false { + prepareToResize() + } + } + + didSet { + if frame.equalTo(oldValue) == false && frame.equalTo(CGRect.zero) == false && imageSize.equalTo(CGSize.zero) == false { + recoverFromResizing() + } + } + } + + override public init(frame: CGRect) { + super.init(frame: frame) + + initialize() + } + + required public init?(coder aDecoder: NSCoder) { + super.init(coder: aDecoder) + + initialize() + } + + private func initialize() { + showsVerticalScrollIndicator = false + showsHorizontalScrollIndicator = false + bouncesZoom = true + decelerationRate = UIScrollView.DecelerationRate.fast + delegate = self + } + + @objc public func adjustFrameToCenter() { + + guard let unwrappedZoomView = zoomView else { + return + } + + var frameToCenter = unwrappedZoomView.frame + + // center horizontally + if frameToCenter.size.width < bounds.width { + frameToCenter.origin.x = (bounds.width - frameToCenter.size.width) / 2 + } else { + frameToCenter.origin.x = 0 + } + + // center vertically + if frameToCenter.size.height < bounds.height { + frameToCenter.origin.y = (bounds.height - frameToCenter.size.height) / 2 + } else { + frameToCenter.origin.y = 0 + } + + unwrappedZoomView.frame = frameToCenter + } + + private func prepareToResize() { + let boundsCenter = CGPoint(x: bounds.midX, y: bounds.midY) + pointToCenterAfterResize = convert(boundsCenter, to: zoomView) + + scaleToRestoreAfterResize = zoomScale + + // If we're at the minimum zoom scale, preserve that by returning 0, which will be converted to the minimum + // allowable scale when the scale is restored. + if scaleToRestoreAfterResize <= minimumZoomScale + CGFloat(Float.ulpOfOne) { + scaleToRestoreAfterResize = 0 + } + } + + private func recoverFromResizing() { + setMaxMinZoomScalesForCurrentBounds() + + // restore zoom scale, first making sure it is within the allowable range. + let maxZoomScale = max(minimumZoomScale, scaleToRestoreAfterResize) + zoomScale = min(maximumZoomScale, maxZoomScale) + + // restore center point, first making sure it is within the allowable range. + + // convert our desired center point back to our own coordinate space + let boundsCenter = convert(pointToCenterAfterResize, to: zoomView) + + // calculate the content offset that would yield that center point + var offset = CGPoint(x: boundsCenter.x - bounds.size.width/2.0, y: boundsCenter.y - bounds.size.height/2.0) + + // restore offset, adjusted to be within the allowable range + let maxOffset = maximumContentOffset() + let minOffset = minimumContentOffset() + + var realMaxOffset = min(maxOffset.x, offset.x) + offset.x = max(minOffset.x, realMaxOffset) + + realMaxOffset = min(maxOffset.y, offset.y) + offset.y = max(minOffset.y, realMaxOffset) + + contentOffset = offset + } + + private func maximumContentOffset() -> CGPoint { + return CGPoint(x: contentSize.width - bounds.width,y:contentSize.height - bounds.height) + } + + private func minimumContentOffset() -> CGPoint { + return CGPoint.zero + } + + // MARK: - Set up + + open func setup() { + var topSupperView = superview + + while topSupperView?.superview != nil { + topSupperView = topSupperView?.superview + } + + // Make sure views have already layout with precise frame + topSupperView?.layoutIfNeeded() + } + + // MARK: - Display image + + @objc open func display(image: UIImage) { + + if let zoomView = zoomView { + zoomView.removeFromSuperview() + } + + zoomView = UIImageView(image: image) + zoomView!.isUserInteractionEnabled = true + addSubview(zoomView!) + + let tapGesture = UITapGestureRecognizer(target: self, action: #selector(doubleTapGestureRecognizer(_:))) + tapGesture.numberOfTapsRequired = 2 + zoomView!.addGestureRecognizer(tapGesture) + + let downSwipeGesture = UISwipeGestureRecognizer(target: self, action: #selector(swipeUpGestureRecognizer(_:))) + downSwipeGesture.direction = .down + zoomView!.addGestureRecognizer(downSwipeGesture) + + let upSwipeGesture = UISwipeGestureRecognizer(target: self, action: #selector(swipeDownGestureRecognizer(_:))) + upSwipeGesture.direction = .up + zoomView!.addGestureRecognizer(upSwipeGesture) + + configureImageForSize(image.size) + adjustFrameToCenter() + } + + private func configureImageForSize(_ size: CGSize) { + imageSize = size + contentSize = imageSize + setMaxMinZoomScalesForCurrentBounds() + zoomScale = minimumZoomScale + + switch initialOffset { + case .begining: + contentOffset = CGPoint.zero + case .center: + let xOffset = contentSize.width < bounds.width ? 0 : (contentSize.width - bounds.width)/2 + let yOffset = contentSize.height < bounds.height ? 0 : (contentSize.height - bounds.height)/2 + + switch imageContentMode { + case .aspectFit: + contentOffset = CGPoint.zero + case .aspectFill: + contentOffset = CGPoint(x: xOffset, y: yOffset) + case .heightFill: + contentOffset = CGPoint(x: xOffset, y: 0) + case .widthFill: + contentOffset = CGPoint(x: 0, y: yOffset) + } + } + } + + private func setMaxMinZoomScalesForCurrentBounds() { + // calculate min/max zoomscale + let xScale = bounds.width / imageSize.width // the scale needed to perfectly fit the image width-wise + let yScale = bounds.height / imageSize.height // the scale needed to perfectly fit the image height-wise + + var minScale: CGFloat = 1 + + switch imageContentMode { + case .aspectFill: + minScale = max(xScale, yScale) + case .aspectFit: + minScale = min(xScale, yScale) + case .widthFill: + minScale = xScale + case .heightFill: + minScale = yScale + } + + + let maxScale = maxScaleFromMinScale*minScale + + // don't let minScale exceed maxScale. (If the image is smaller than the screen, we don't want to force it to be zoomed.) + if minScale > maxScale { + minScale = maxScale + } + + maximumZoomScale = maxScale + minimumZoomScale = minScale // * 0.999 // the multiply factor to prevent user cannot scroll page while they use this control in UIPageViewController + } + + // MARK: - Gesture + + @objc func doubleTapGestureRecognizer(_ gestureRecognizer: UIGestureRecognizer) { + // zoom out if it bigger than middle scale point. Else, zoom in + if zoomScale >= maximumZoomScale / 2.0 { + setZoomScale(minimumZoomScale, animated: true) + } else { + let center = gestureRecognizer.location(in: gestureRecognizer.view) + let zoomRect = zoomRectForScale(ImageScrollView.kZoomInFactorFromMinWhenDoubleTap * minimumZoomScale, center: center) + zoom(to: zoomRect, animated: true) + } + } + + @objc func swipeUpGestureRecognizer(_ gestureRecognizer: UIGestureRecognizer) { + if gestureRecognizer.state == .ended { + imageScrollViewDelegate?.imageScrollViewDidGestureSwipeUp(imageScrollView: self) + } + } + + @objc func swipeDownGestureRecognizer(_ gestureRecognizer: UIGestureRecognizer) { + if gestureRecognizer.state == .ended { + imageScrollViewDelegate?.imageScrollViewDidGestureSwipeDown(imageScrollView: self) + } + } + + private func zoomRectForScale(_ scale: CGFloat, center: CGPoint) -> CGRect { + var zoomRect = CGRect.zero + + // the zoom rect is in the content view's coordinates. + // at a zoom scale of 1.0, it would be the size of the imageScrollView's bounds. + // as the zoom scale decreases, so more content is visible, the size of the rect grows. + zoomRect.size.height = frame.size.height / scale + zoomRect.size.width = frame.size.width / scale + + // choose an origin so as to get the right center. + zoomRect.origin.x = center.x - (zoomRect.size.width / 2.0) + zoomRect.origin.y = center.y - (zoomRect.size.height / 2.0) + + return zoomRect + } + + open func refresh() { + if let image = zoomView?.image { + display(image: image) + } + } + + open func resize() { + self.configureImageForSize(self.imageSize) + } +} + +extension ImageScrollView: UIScrollViewDelegate { + + public func scrollViewDidScroll(_ scrollView: UIScrollView) { + imageScrollViewDelegate?.scrollViewDidScroll?(scrollView) + } + + public func scrollViewWillBeginDragging(_ scrollView: UIScrollView) { + imageScrollViewDelegate?.scrollViewWillBeginDragging?(scrollView) + } + + public func scrollViewWillEndDragging(_ scrollView: UIScrollView, withVelocity velocity: CGPoint, targetContentOffset: UnsafeMutablePointer) { + imageScrollViewDelegate?.scrollViewWillEndDragging?(scrollView, withVelocity: velocity, targetContentOffset: targetContentOffset) + } + + public func scrollViewDidEndDragging(_ scrollView: UIScrollView, willDecelerate decelerate: Bool) { + imageScrollViewDelegate?.scrollViewDidEndDragging?(scrollView, willDecelerate: decelerate) + } + + public func scrollViewWillBeginDecelerating(_ scrollView: UIScrollView) { + imageScrollViewDelegate?.scrollViewWillBeginDecelerating?(scrollView) + } + + public func scrollViewDidEndDecelerating(_ scrollView: UIScrollView) { + imageScrollViewDelegate?.scrollViewDidEndDecelerating?(scrollView) + } + + public func scrollViewDidEndScrollingAnimation(_ scrollView: UIScrollView) { + imageScrollViewDelegate?.scrollViewDidEndScrollingAnimation?(scrollView) + } + + public func scrollViewWillBeginZooming(_ scrollView: UIScrollView, with view: UIView?) { + imageScrollViewDelegate?.scrollViewWillBeginZooming?(scrollView, with: view) + } + + public func scrollViewDidEndZooming(_ scrollView: UIScrollView, with view: UIView?, atScale scale: CGFloat) { + imageScrollViewDelegate?.scrollViewDidEndZooming?(scrollView, with: view, atScale: scale) + } + + public func scrollViewShouldScrollToTop(_ scrollView: UIScrollView) -> Bool { + return false + } + + @available(iOS 11.0, *) + public func scrollViewDidChangeAdjustedContentInset(_ scrollView: UIScrollView) { + imageScrollViewDelegate?.scrollViewDidChangeAdjustedContentInset?(scrollView) + } + + public func viewForZooming(in scrollView: UIScrollView) -> UIView? { + return zoomView + } + + public func scrollViewDidZoom(_ scrollView: UIScrollView) { + adjustFrameToCenter() + imageScrollViewDelegate?.scrollViewDidZoom?(scrollView) + } + +} diff --git a/Multiplatform/iOS/Article/ImageTransition.swift b/Multiplatform/iOS/Article/ImageTransition.swift new file mode 100644 index 000000000..01f460348 --- /dev/null +++ b/Multiplatform/iOS/Article/ImageTransition.swift @@ -0,0 +1,110 @@ +// +// ImageTransition.swift +// Multiplatform iOS +// +// Created by Maurice Parker on 7/6/20. +// Copyright © 2020 Ranchero Software. All rights reserved. +// + +import UIKit + +class ImageTransition: NSObject, UIViewControllerAnimatedTransitioning { + + private weak var webViewController: WebViewController? + private let duration = 0.4 + var presenting = true + var originFrame: CGRect! + var maskFrame: CGRect! + var originImage: UIImage! + + init(controller: WebViewController) { + self.webViewController = controller + } + + func transitionDuration(using transitionContext: UIViewControllerContextTransitioning?) -> TimeInterval { + return duration + } + + func animateTransition(using transitionContext: UIViewControllerContextTransitioning) { + if presenting { + animateTransitionPresenting(using: transitionContext) + } else { + animateTransitionReturning(using: transitionContext) + } + } + + private func animateTransitionPresenting(using transitionContext: UIViewControllerContextTransitioning) { + + let imageView = UIImageView(image: originImage) + imageView.frame = originFrame + + let fromView = transitionContext.view(forKey: .from)! + fromView.removeFromSuperview() + + transitionContext.containerView.backgroundColor = .systemBackground + transitionContext.containerView.addSubview(imageView) + + webViewController?.hideClickedImage() + + UIView.animate( + withDuration: duration, + delay:0.0, + usingSpringWithDamping: 0.8, + initialSpringVelocity: 0.2, + animations: { + let imageController = transitionContext.viewController(forKey: .to) as! ImageViewController + imageView.frame = imageController.zoomedFrame + }, completion: { _ in + imageView.removeFromSuperview() + let toView = transitionContext.view(forKey: .to)! + transitionContext.containerView.addSubview(toView) + transitionContext.completeTransition(true) + }) + } + + private func animateTransitionReturning(using transitionContext: UIViewControllerContextTransitioning) { + let imageController = transitionContext.viewController(forKey: .from) as! ImageViewController + let imageView = UIImageView(image: originImage) + imageView.frame = imageController.zoomedFrame + + let fromView = transitionContext.view(forKey: .from)! + let windowFrame = fromView.window!.frame + fromView.removeFromSuperview() + + let toView = transitionContext.view(forKey: .to)! + transitionContext.containerView.addSubview(toView) + + let maskingView = UIView() + + let fullMaskFrame = CGRect(x: windowFrame.minX, y: maskFrame.minY, width: windowFrame.width, height: maskFrame.height) + let path = UIBezierPath(rect: fullMaskFrame) + let maskLayer = CAShapeLayer() + maskLayer.path = path.cgPath + maskingView.layer.mask = maskLayer + + maskingView.addSubview(imageView) + transitionContext.containerView.addSubview(maskingView) + + UIView.animate( + withDuration: duration, + delay:0.0, + usingSpringWithDamping: 0.8, + initialSpringVelocity: 0.2, + animations: { + imageView.frame = self.originFrame + }, completion: { _ in + if let controller = self.webViewController { + controller.showClickedImage() { + DispatchQueue.main.asyncAfter(deadline: .now() + 0.1) { + imageView.removeFromSuperview() + transitionContext.completeTransition(true) + } + } + } else { + imageView.removeFromSuperview() + transitionContext.completeTransition(true) + } + }) + } + +} diff --git a/Multiplatform/iOS/Article/ImageViewController.swift b/Multiplatform/iOS/Article/ImageViewController.swift new file mode 100644 index 000000000..4e9ae9466 --- /dev/null +++ b/Multiplatform/iOS/Article/ImageViewController.swift @@ -0,0 +1,92 @@ +// +// ImageViewController.swift +// Multiplatform iOS +// +// Created by Maurice Parker on 7/6/20. +// Copyright © 2020 Ranchero Software. All rights reserved. +// + +import UIKit + +class ImageViewController: UIViewController { + + @IBOutlet weak var closeButton: UIButton! + @IBOutlet weak var shareButton: UIButton! + @IBOutlet weak var imageScrollView: ImageScrollView! + @IBOutlet weak var titleLabel: UILabel! + @IBOutlet weak var titleBackground: UIVisualEffectView! + @IBOutlet weak var titleLeading: NSLayoutConstraint! + @IBOutlet weak var titleTrailing: NSLayoutConstraint! + + var image: UIImage! + var imageTitle: String? + var zoomedFrame: CGRect { + return imageScrollView.zoomedFrame + } + + override func viewDidLoad() { + super.viewDidLoad() + + closeButton.imageView?.contentMode = .scaleAspectFit + closeButton.accessibilityLabel = NSLocalizedString("Close", comment: "Close") + shareButton.accessibilityLabel = NSLocalizedString("Share", comment: "Share") + + imageScrollView.setup() + imageScrollView.imageScrollViewDelegate = self + imageScrollView.imageContentMode = .aspectFit + imageScrollView.initialOffset = .center + imageScrollView.display(image: image) + + titleLabel.text = imageTitle ?? "" + layoutTitleLabel() + + guard imageTitle != "" else { + titleBackground.removeFromSuperview() + return + } + titleBackground.layer.cornerRadius = 6 + } + + override func viewWillTransition(to size: CGSize, with coordinator: UIViewControllerTransitionCoordinator) { + super.viewWillTransition(to: size, with: coordinator) + coordinator.animate(alongsideTransition: { [weak self] context in + self?.imageScrollView.resize() + }) + } + + @IBAction func share(_ sender: Any) { + guard let image = image else { return } + let activityViewController = UIActivityViewController(activityItems: [image], applicationActivities: nil) + activityViewController.popoverPresentationController?.sourceView = shareButton + activityViewController.popoverPresentationController?.sourceRect = shareButton.bounds + present(activityViewController, animated: true) + } + + @IBAction func done(_ sender: Any) { + dismiss(animated: true) + } + + private func layoutTitleLabel(){ + let width = view.frame.width + let multiplier = UIDevice.current.userInterfaceIdiom == .pad ? CGFloat(0.1) : CGFloat(0.04) + titleLeading.constant += width * multiplier + titleTrailing.constant -= width * multiplier + titleLabel.layoutIfNeeded() + } +} + +// MARK: ImageScrollViewDelegate + +extension ImageViewController: ImageScrollViewDelegate { + + func imageScrollViewDidGestureSwipeUp(imageScrollView: ImageScrollView) { + dismiss(animated: true) + } + + func imageScrollViewDidGestureSwipeDown(imageScrollView: ImageScrollView) { + dismiss(animated: true) + } + + +} + diff --git a/Multiplatform/iOS/Article/PreloadedWebView.swift b/Multiplatform/iOS/Article/PreloadedWebView.swift new file mode 100644 index 000000000..944e1d578 --- /dev/null +++ b/Multiplatform/iOS/Article/PreloadedWebView.swift @@ -0,0 +1,75 @@ +// +// PreloadedWebView.swift +// Multiplatform iOS +// +// Created by Maurice Parker on 7/6/20. +// Copyright © 2020 Ranchero Software. All rights reserved. +// + +import Foundation +import WebKit + +class PreloadedWebView: WKWebView { + + private var isReady: Bool = false + private var readyCompletion: ((PreloadedWebView) -> Void)? + + init(articleIconSchemeHandler: ArticleIconSchemeHandler) { + let preferences = WKPreferences() + preferences.javaScriptCanOpenWindowsAutomatically = false + preferences.javaScriptEnabled = true + + let configuration = WKWebViewConfiguration() + configuration.preferences = preferences + configuration.setValue(true, forKey: "allowUniversalAccessFromFileURLs") + configuration.allowsInlineMediaPlayback = true + configuration.mediaTypesRequiringUserActionForPlayback = .audio + configuration.setURLSchemeHandler(articleIconSchemeHandler, forURLScheme: ArticleRenderer.imageIconScheme) + + super.init(frame: .zero, configuration: configuration) + } + + required init?(coder: NSCoder) { + super.init(coder: coder) + } + + func preload() { + navigationDelegate = self + loadFileURL(ArticleRenderer.blank.url, allowingReadAccessTo: ArticleRenderer.blank.baseURL) + } + + func ready(completion: @escaping (PreloadedWebView) -> Void) { + if isReady { + completeRequest(completion: completion) + } else { + readyCompletion = completion + } + } + +} + +// MARK: WKScriptMessageHandler + +extension PreloadedWebView: WKNavigationDelegate { + + func webView(_ webView: WKWebView, didFinish navigation: WKNavigation!) { + isReady = true + if let completion = readyCompletion { + completeRequest(completion: completion) + readyCompletion = nil + } + } + +} + +// MARK: Private + +private extension PreloadedWebView { + + func completeRequest(completion: @escaping (PreloadedWebView) -> Void) { + isReady = false + navigationDelegate = nil + completion(self) + } + +} diff --git a/Multiplatform/iOS/Article/WebViewController.swift b/Multiplatform/iOS/Article/WebViewController.swift new file mode 100644 index 000000000..794b07418 --- /dev/null +++ b/Multiplatform/iOS/Article/WebViewController.swift @@ -0,0 +1,772 @@ +// +// WebViewController.swift +// Multiplatform iOS +// +// Created by Maurice Parker on 7/6/20. +// Copyright © 2020 Ranchero Software. All rights reserved. +// + +import UIKit +import WebKit +import RSCore +import Account +import Articles +import SafariServices +import MessageUI + +protocol WebViewControllerDelegate: class { + func webViewController(_: WebViewController, articleExtractorButtonStateDidUpdate: ArticleExtractorButtonState) +} + +class WebViewController: UIViewController { + + private struct MessageName { + static let imageWasClicked = "imageWasClicked" + static let imageWasShown = "imageWasShown" + static let showFeedInspector = "showFeedInspector" + } + + private var topShowBarsView: UIView! + private var bottomShowBarsView: UIView! + private var topShowBarsViewConstraint: NSLayoutConstraint! + private var bottomShowBarsViewConstraint: NSLayoutConstraint! + + private var webView: PreloadedWebView? { + guard view.subviews.count > 0 else { return nil } + return view.subviews[0] as? PreloadedWebView + } + +// private lazy var contextMenuInteraction = UIContextMenuInteraction(delegate: self) + private var isFullScreenAvailable: Bool { + return AppDefaults.shared.articleFullscreenAvailable && traitCollection.userInterfaceIdiom == .phone // && coordinator.isRootSplitCollapsed + } + private lazy var transition = ImageTransition(controller: self) + private var clickedImageCompletion: (() -> Void)? + + private var articleExtractor: ArticleExtractor? = nil + var extractedArticle: ExtractedArticle? { + didSet { + windowScrollY = 0 + } + } + var isShowingExtractedArticle = false + + var articleExtractorButtonState: ArticleExtractorButtonState = .off { + didSet { + delegate?.webViewController(self, articleExtractorButtonStateDidUpdate: articleExtractorButtonState) + } + } + + var articleModel: ArticleModel? + weak var delegate: WebViewControllerDelegate? + + private(set) var article: Article? + + let scrollPositionQueue = CoalescingQueue(name: "Article Scroll Position", interval: 0.3, maxInterval: 0.3) + var windowScrollY = 0 + + override func viewDidLoad() { + super.viewDidLoad() + + NotificationCenter.default.addObserver(self, selector: #selector(webFeedIconDidBecomeAvailable(_:)), name: .WebFeedIconDidBecomeAvailable, object: nil) + NotificationCenter.default.addObserver(self, selector: #selector(avatarDidBecomeAvailable(_:)), name: .AvatarDidBecomeAvailable, object: nil) + NotificationCenter.default.addObserver(self, selector: #selector(faviconDidBecomeAvailable(_:)), name: .FaviconDidBecomeAvailable, object: nil) + + // Configure the tap zones +// configureTopShowBarsView() +// configureBottomShowBarsView() + + loadWebView() + + } + + // MARK: Notifications + + @objc func webFeedIconDidBecomeAvailable(_ note: Notification) { + reloadArticleImage() + } + + @objc func avatarDidBecomeAvailable(_ note: Notification) { + reloadArticleImage() + } + + @objc func faviconDidBecomeAvailable(_ note: Notification) { + reloadArticleImage() + } + + // MARK: Actions + +// @objc func showBars(_ sender: Any) { +// showBars() +// } + + // MARK: API + + func setArticle(_ article: Article?, updateView: Bool = true) { + stopArticleExtractor() + + if article != self.article { + self.article = article + if updateView { + if article?.webFeed?.isArticleExtractorAlwaysOn ?? false { + startArticleExtractor() + } + windowScrollY = 0 + loadWebView() + } + } + + } + + func focus() { + webView?.becomeFirstResponder() + } + + func canScrollDown() -> Bool { + guard let webView = webView else { return false } + return webView.scrollView.contentOffset.y < finalScrollPosition() + } + + func scrollPageDown() { + guard let webView = webView else { return } + + let overlap = 2 * UIFont.systemFont(ofSize: UIFont.systemFontSize).lineHeight * UIScreen.main.scale + let scrollToY: CGFloat = { + let fullScroll = webView.scrollView.contentOffset.y + webView.scrollView.layoutMarginsGuide.layoutFrame.height - overlap + let final = finalScrollPosition() + return fullScroll < final ? fullScroll : final + }() + + let convertedPoint = self.view.convert(CGPoint(x: 0, y: 0), to: webView.scrollView) + let scrollToPoint = CGPoint(x: convertedPoint.x, y: scrollToY) + webView.scrollView.setContentOffset(scrollToPoint, animated: true) + } + + func hideClickedImage() { + webView?.evaluateJavaScript("hideClickedImage();") + } + + func showClickedImage(completion: @escaping () -> Void) { + clickedImageCompletion = completion + webView?.evaluateJavaScript("showClickedImage();") + } + + func fullReload() { + loadWebView(replaceExistingWebView: true) + } + +// func showBars() { +// AppDefaults.shared.articleFullscreenEnabled = false +// coordinator.showStatusBar() +// topShowBarsViewConstraint?.constant = 0 +// bottomShowBarsViewConstraint?.constant = 0 +// navigationController?.setNavigationBarHidden(false, animated: true) +// navigationController?.setToolbarHidden(false, animated: true) +// configureContextMenuInteraction() +// } +// +// func hideBars() { +// if isFullScreenAvailable { +// AppDefaults.shared.articleFullscreenEnabled = true +// coordinator.hideStatusBar() +// topShowBarsViewConstraint?.constant = -44.0 +// bottomShowBarsViewConstraint?.constant = 44.0 +// navigationController?.setNavigationBarHidden(true, animated: true) +// navigationController?.setToolbarHidden(true, animated: true) +// configureContextMenuInteraction() +// } +// } + + func toggleArticleExtractor() { + + guard let article = article else { + return + } + + guard articleExtractor?.state != .processing else { + stopArticleExtractor() + loadWebView() + return + } + + guard !isShowingExtractedArticle else { + isShowingExtractedArticle = false + loadWebView() + articleExtractorButtonState = .off + return + } + + if let articleExtractor = articleExtractor { + if article.preferredLink == articleExtractor.articleLink { + isShowingExtractedArticle = true + loadWebView() + articleExtractorButtonState = .on + } + } else { + startArticleExtractor() + } + + } + + func stopArticleExtractorIfProcessing() { + if articleExtractor?.state == .processing { + stopArticleExtractor() + } + } + + func stopWebViewActivity() { + if let webView = webView { + stopMediaPlayback(webView) + cancelImageLoad(webView) + } + } + +} + +// MARK: ArticleExtractorDelegate + +extension WebViewController: ArticleExtractorDelegate { + + func articleExtractionDidFail(with: Error) { + stopArticleExtractor() + articleExtractorButtonState = .error + loadWebView() + } + + func articleExtractionDidComplete(extractedArticle: ExtractedArticle) { + if articleExtractor?.state != .cancelled { + self.extractedArticle = extractedArticle + isShowingExtractedArticle = true + loadWebView() + articleExtractorButtonState = .on + } + } + +} + +// MARK: UIContextMenuInteractionDelegate + +//extension WebViewController: UIContextMenuInteractionDelegate { +// func contextMenuInteraction(_ interaction: UIContextMenuInteraction, configurationForMenuAtLocation location: CGPoint) -> UIContextMenuConfiguration? { +// +// return UIContextMenuConfiguration(identifier: nil, previewProvider: contextMenuPreviewProvider) { [weak self] suggestedActions in +// guard let self = self else { return nil } +// var actions = [UIAction]() +// +// if let action = self.prevArticleAction() { +// actions.append(action) +// } +// if let action = self.nextArticleAction() { +// actions.append(action) +// } +// if let action = self.toggleReadAction() { +// actions.append(action) +// } +// actions.append(self.toggleStarredAction()) +// if let action = self.nextUnreadArticleAction() { +// actions.append(action) +// } +// actions.append(self.toggleArticleExtractorAction()) +// actions.append(self.shareAction()) +// +// return UIMenu(title: "", children: actions) +// } +// } +// +// func contextMenuInteraction(_ interaction: UIContextMenuInteraction, willPerformPreviewActionForMenuWith configuration: UIContextMenuConfiguration, animator: UIContextMenuInteractionCommitAnimating) { +// coordinator.showBrowserForCurrentArticle() +// } +// +//} + +// MARK: WKNavigationDelegate + +extension WebViewController: WKNavigationDelegate { + + func webView(_ webView: WKWebView, didFinish navigation: WKNavigation!) { + for (index, view) in view.subviews.enumerated() { + if index != 0, let oldWebView = view as? PreloadedWebView { + oldWebView.removeFromSuperview() + } + } + } + + func webView(_ webView: WKWebView, decidePolicyFor navigationAction: WKNavigationAction, decisionHandler: @escaping (WKNavigationActionPolicy) -> Void) { + + if navigationAction.navigationType == .linkActivated { + guard let url = navigationAction.request.url else { + decisionHandler(.allow) + return + } + + let components = URLComponents(url: url, resolvingAgainstBaseURL: false) + if components?.scheme == "http" || components?.scheme == "https" { + decisionHandler(.cancel) + + // If the resource cannot be opened with an installed app, present the web view. + UIApplication.shared.open(url, options: [.universalLinksOnly: true]) { didOpen in + assert(Thread.isMainThread) + guard didOpen == false else { + return + } + let vc = SFSafariViewController(url: url) + self.present(vc, animated: true) + } + } else if components?.scheme == "mailto" { + decisionHandler(.cancel) + + guard let emailAddress = url.emailAddress else { + return + } + + if MFMailComposeViewController.canSendMail() { + let mailComposeViewController = MFMailComposeViewController() + mailComposeViewController.setToRecipients([emailAddress]) + mailComposeViewController.mailComposeDelegate = self + self.present(mailComposeViewController, animated: true, completion: {}) + } else { + let alert = UIAlertController(title: NSLocalizedString("Error", comment: "Error"), message: NSLocalizedString("This device cannot send emails.", comment: "This device cannot send emails."), preferredStyle: .alert) + alert.addAction(.init(title: NSLocalizedString("Dismiss", comment: "Dismiss"), style: .cancel, handler: nil)) + self.present(alert, animated: true, completion: nil) + } + } else if components?.scheme == "tel" { + decisionHandler(.cancel) + + if UIApplication.shared.canOpenURL(url) { + UIApplication.shared.open(url, options: [.universalLinksOnly : false], completionHandler: nil) + } + + } else { + decisionHandler(.allow) + } + } else { + decisionHandler(.allow) + } + } + + func webViewWebContentProcessDidTerminate(_ webView: WKWebView) { + fullReload() + } + +} + +// MARK: WKUIDelegate + +extension WebViewController: WKUIDelegate { + func webView(_ webView: WKWebView, contextMenuForElement elementInfo: WKContextMenuElementInfo, willCommitWithAnimator animator: UIContextMenuInteractionCommitAnimating) { + // We need to have at least an unimplemented WKUIDelegate assigned to the WKWebView. This makes the + // link preview launch Safari when the link preview is tapped. In theory, you shoud be able to get + // the link from the elementInfo above and transition to SFSafariViewController instead of launching + // Safari. As the time of this writing, the link in elementInfo is always nil. ¯\_(ツ)_/¯ + } +} + +// MARK: WKScriptMessageHandler + +extension WebViewController: WKScriptMessageHandler { + + func userContentController(_ userContentController: WKUserContentController, didReceive message: WKScriptMessage) { + switch message.name { + case MessageName.imageWasShown: + clickedImageCompletion?() + case MessageName.imageWasClicked: + imageWasClicked(body: message.body as? String) + case MessageName.showFeedInspector: + if let webFeed = article?.webFeed { +// coordinator.showFeedInspector(for: webFeed) + } + default: + return + } + } + +} + +// MARK: UIViewControllerTransitioningDelegate + +extension WebViewController: UIViewControllerTransitioningDelegate { + + func animationController(forPresented presented: UIViewController, presenting: UIViewController, source: UIViewController) -> UIViewControllerAnimatedTransitioning? { + transition.presenting = true + return transition + } + + func animationController(forDismissed dismissed: UIViewController) -> UIViewControllerAnimatedTransitioning? { + transition.presenting = false + return transition + } +} + +// MARK: + +extension WebViewController: UIScrollViewDelegate { + + func scrollViewDidScroll(_ scrollView: UIScrollView) { + scrollPositionQueue.add(self, #selector(scrollPositionDidChange)) + } + + @objc func scrollPositionDidChange() { + webView?.evaluateJavaScript("window.scrollY") { (scrollY, error) in + guard error == nil else { return } + let javascriptScrollY = scrollY as? Int ?? 0 + // I don't know why this value gets returned sometimes, but it is in error + guard javascriptScrollY != 33554432 else { return } + self.windowScrollY = javascriptScrollY + } + } + +} + +// MARK: MFMailComposeViewControllerDelegate +extension WebViewController: MFMailComposeViewControllerDelegate { + + func mailComposeController(_ controller: MFMailComposeViewController, didFinishWith result: MFMailComposeResult, error: Error?) { + self.dismiss(animated: true, completion: nil) + } + +} + +// MARK: JSON + +private struct ImageClickMessage: Codable { + let x: Float + let y: Float + let width: Float + let height: Float + let imageTitle: String? + let imageURL: String +} + +// MARK: Private + +private extension WebViewController { + + func loadWebView(replaceExistingWebView: Bool = false) { + guard isViewLoaded else { return } + + if !replaceExistingWebView, let webView = webView { + self.renderPage(webView) + return + } + + articleModel?.webViewProvider?.dequeueWebView() { webView in + + // Add the webview + webView.translatesAutoresizingMaskIntoConstraints = false + self.view.insertSubview(webView, at: 0) + NSLayoutConstraint.activate([ + self.view.leadingAnchor.constraint(equalTo: webView.leadingAnchor), + self.view.trailingAnchor.constraint(equalTo: webView.trailingAnchor), + self.view.topAnchor.constraint(equalTo: webView.topAnchor), + self.view.bottomAnchor.constraint(equalTo: webView.bottomAnchor) + ]) + + // UISplitViewController reports the wrong size to WKWebView which can cause horizontal + // rubberbanding on the iPad. This interferes with our UIPageViewController preventing + // us from easily swiping between WKWebViews. This hack fixes that. + webView.scrollView.contentInset = UIEdgeInsets(top: 0, left: -1, bottom: 0, right: 0) + + webView.scrollView.setZoomScale(1.0, animated: false) + + self.view.setNeedsLayout() + self.view.layoutIfNeeded() + + // Configure the webview + webView.navigationDelegate = self + webView.uiDelegate = self + webView.scrollView.delegate = self +// self.configureContextMenuInteraction() + + webView.configuration.userContentController.add(WrapperScriptMessageHandler(self), name: MessageName.imageWasClicked) + webView.configuration.userContentController.add(WrapperScriptMessageHandler(self), name: MessageName.imageWasShown) + webView.configuration.userContentController.add(WrapperScriptMessageHandler(self), name: MessageName.showFeedInspector) + + self.renderPage(webView) + + } + + } + + func renderPage(_ webView: PreloadedWebView?) { + guard let webView = webView else { return } + + let style = ArticleStylesManager.shared.currentStyle + let rendering: ArticleRenderer.Rendering + + if let articleExtractor = articleExtractor, articleExtractor.state == .processing { + rendering = ArticleRenderer.loadingHTML(style: style) + } else if let articleExtractor = articleExtractor, articleExtractor.state == .failedToParse, let article = article { + rendering = ArticleRenderer.articleHTML(article: article, style: style) + } else if let article = article, let extractedArticle = extractedArticle { + if isShowingExtractedArticle { + rendering = ArticleRenderer.articleHTML(article: article, extractedArticle: extractedArticle, style: style) + } else { + rendering = ArticleRenderer.articleHTML(article: article, style: style) + } + } else if let article = article { + rendering = ArticleRenderer.articleHTML(article: article, style: style) + } else { + rendering = ArticleRenderer.noSelectionHTML(style: style) + } + + let substitutions = [ + "title": rendering.title, + "baseURL": rendering.baseURL, + "style": rendering.style, + "body": rendering.html, + "windowScrollY": String(windowScrollY) + ] + + let html = try! MacroProcessor.renderedText(withTemplate: ArticleRenderer.page.html, substitutions: substitutions) + webView.loadHTMLString(html, baseURL: ArticleRenderer.page.baseURL) + + } + + func finalScrollPosition() -> CGFloat { + guard let webView = webView else { return 0 } + return webView.scrollView.contentSize.height - webView.scrollView.bounds.height + webView.scrollView.safeAreaInsets.bottom + } + + func startArticleExtractor() { + if let link = article?.preferredLink, let extractor = ArticleExtractor(link) { + extractor.delegate = self + extractor.process() + articleExtractor = extractor + articleExtractorButtonState = .animated + } + } + + func stopArticleExtractor() { + articleExtractor?.cancel() + articleExtractor = nil + isShowingExtractedArticle = false + articleExtractorButtonState = .off + } + + func reloadArticleImage() { + guard let article = article else { return } + + var components = URLComponents() + components.scheme = ArticleRenderer.imageIconScheme + components.path = article.articleID + + if let imageSrc = components.string { + webView?.evaluateJavaScript("reloadArticleImage(\"\(imageSrc)\")") + } + } + + func imageWasClicked(body: String?) { + guard let webView = webView, + let body = body, + let data = body.data(using: .utf8), + let clickMessage = try? JSONDecoder().decode(ImageClickMessage.self, from: data), + let range = clickMessage.imageURL.range(of: ";base64,") + else { return } + + let base64Image = String(clickMessage.imageURL.suffix(from: range.upperBound)) + if let imageData = Data(base64Encoded: base64Image), let image = UIImage(data: imageData) { + + let y = CGFloat(clickMessage.y) + webView.safeAreaInsets.top + let rect = CGRect(x: CGFloat(clickMessage.x), y: y, width: CGFloat(clickMessage.width), height: CGFloat(clickMessage.height)) + transition.originFrame = webView.convert(rect, to: nil) + + if navigationController?.navigationBar.isHidden ?? false { + transition.maskFrame = webView.convert(webView.frame, to: nil) + } else { + transition.maskFrame = webView.convert(webView.safeAreaLayoutGuide.layoutFrame, to: nil) + } + + transition.originImage = image + +// coordinator.showFullScreenImage(image: image, imageTitle: clickMessage.imageTitle, transitioningDelegate: self) + } + } + + func stopMediaPlayback(_ webView: WKWebView) { + webView.evaluateJavaScript("stopMediaPlayback();") + } + + func cancelImageLoad(_ webView: WKWebView) { + webView.evaluateJavaScript("cancelImageLoad();") + } + +// func configureTopShowBarsView() { +// topShowBarsView = UIView() +// topShowBarsView.backgroundColor = .clear +// topShowBarsView.translatesAutoresizingMaskIntoConstraints = false +// view.addSubview(topShowBarsView) +// +// if AppDefaults.shared.articleFullscreenEnabled { +// topShowBarsViewConstraint = view.topAnchor.constraint(equalTo: topShowBarsView.bottomAnchor, constant: -44.0) +// } else { +// topShowBarsViewConstraint = view.topAnchor.constraint(equalTo: topShowBarsView.bottomAnchor, constant: 0.0) +// } +// +// NSLayoutConstraint.activate([ +// topShowBarsViewConstraint, +// view.leadingAnchor.constraint(equalTo: topShowBarsView.leadingAnchor), +// view.trailingAnchor.constraint(equalTo: topShowBarsView.trailingAnchor), +// topShowBarsView.heightAnchor.constraint(equalToConstant: 44.0) +// ]) +// topShowBarsView.addGestureRecognizer(UITapGestureRecognizer(target: self, action: #selector(showBars(_:)))) +// } +// +// func configureBottomShowBarsView() { +// bottomShowBarsView = UIView() +// topShowBarsView.backgroundColor = .clear +// bottomShowBarsView.translatesAutoresizingMaskIntoConstraints = false +// view.addSubview(bottomShowBarsView) +// if AppDefaults.shared.articleFullscreenEnabled { +// bottomShowBarsViewConstraint = view.bottomAnchor.constraint(equalTo: bottomShowBarsView.topAnchor, constant: 44.0) +// } else { +// bottomShowBarsViewConstraint = view.bottomAnchor.constraint(equalTo: bottomShowBarsView.topAnchor, constant: 0.0) +// } +// NSLayoutConstraint.activate([ +// bottomShowBarsViewConstraint, +// view.leadingAnchor.constraint(equalTo: bottomShowBarsView.leadingAnchor), +// view.trailingAnchor.constraint(equalTo: bottomShowBarsView.trailingAnchor), +// bottomShowBarsView.heightAnchor.constraint(equalToConstant: 44.0) +// ]) +// bottomShowBarsView.addGestureRecognizer(UITapGestureRecognizer(target: self, action: #selector(showBars(_:)))) +// } + +// func configureContextMenuInteraction() { +// if isFullScreenAvailable { +// if navigationController?.isNavigationBarHidden ?? false { +// webView?.addInteraction(contextMenuInteraction) +// } else { +// webView?.removeInteraction(contextMenuInteraction) +// } +// } +// } +// +// func contextMenuPreviewProvider() -> UIViewController { +// let previewProvider = UIStoryboard.main.instantiateController(ofType: ContextMenuPreviewViewController.self) +// previewProvider.article = article +// return previewProvider +// } +// +// func prevArticleAction() -> UIAction? { +// guard coordinator.isPrevArticleAvailable else { return nil } +// let title = NSLocalizedString("Previous Article", comment: "Previous Article") +// return UIAction(title: title, image: AppAssets.prevArticleImage) { [weak self] action in +// self?.coordinator.selectPrevArticle() +// } +// } +// +// func nextArticleAction() -> UIAction? { +// guard coordinator.isNextArticleAvailable else { return nil } +// let title = NSLocalizedString("Next Article", comment: "Next Article") +// return UIAction(title: title, image: AppAssets.nextArticleImage) { [weak self] action in +// self?.coordinator.selectNextArticle() +// } +// } +// +// func toggleReadAction() -> UIAction? { +// guard let article = article, !article.status.read || article.isAvailableToMarkUnread else { return nil } +// +// let title = article.status.read ? NSLocalizedString("Mark as Unread", comment: "Mark as Unread") : NSLocalizedString("Mark as Read", comment: "Mark as Read") +// let readImage = article.status.read ? AppAssets.circleClosedImage : AppAssets.circleOpenImage +// return UIAction(title: title, image: readImage) { [weak self] action in +// self?.coordinator.toggleReadForCurrentArticle() +// } +// } +// +// func toggleStarredAction() -> UIAction { +// let starred = article?.status.starred ?? false +// let title = starred ? NSLocalizedString("Mark as Unstarred", comment: "Mark as Unstarred") : NSLocalizedString("Mark as Starred", comment: "Mark as Starred") +// let starredImage = starred ? AppAssets.starOpenImage : AppAssets.starClosedImage +// return UIAction(title: title, image: starredImage) { [weak self] action in +// self?.coordinator.toggleStarredForCurrentArticle() +// } +// } +// +// func nextUnreadArticleAction() -> UIAction? { +// guard coordinator.isAnyUnreadAvailable else { return nil } +// let title = NSLocalizedString("Next Unread Article", comment: "Next Unread Article") +// return UIAction(title: title, image: AppAssets.nextUnreadArticleImage) { [weak self] action in +// self?.coordinator.selectNextUnread() +// } +// } +// +// func toggleArticleExtractorAction() -> UIAction { +// let extracted = articleExtractorButtonState == .on +// let title = extracted ? NSLocalizedString("Show Feed Article", comment: "Show Feed Article") : NSLocalizedString("Show Reader View", comment: "Show Reader View") +// let extractorImage = extracted ? AppAssets.articleExtractorOffSF : AppAssets.articleExtractorOnSF +// return UIAction(title: title, image: extractorImage) { [weak self] action in +// self?.toggleArticleExtractor() +// } +// } +// +// func shareAction() -> UIAction { +// let title = NSLocalizedString("Share", comment: "Share") +// return UIAction(title: title, image: AppAssets.shareImage) { [weak self] action in +// self?.showActivityDialog() +// } +// } + +} + +// MARK: Find in Article + +private struct FindInArticleOptions: Codable { + var text: String + var caseSensitive = false + var regex = false +} + +internal struct FindInArticleState: Codable { + struct WebViewClientRect: Codable { + let x: Double + let y: Double + let width: Double + let height: Double + } + + struct FindInArticleResult: Codable { + let rects: [WebViewClientRect] + let bounds: WebViewClientRect + let index: UInt + let matchGroups: [String] + } + + let index: UInt? + let results: [FindInArticleResult] + let count: UInt +} + +extension WebViewController { + + func searchText(_ searchText: String, completionHandler: @escaping (FindInArticleState) -> Void) { + guard let json = try? JSONEncoder().encode(FindInArticleOptions(text: searchText)) else { + return + } + let encoded = json.base64EncodedString() + + webView?.evaluateJavaScript("updateFind(\"\(encoded)\")") { + (result, error) in + guard error == nil, + let b64 = result as? String, + let rawData = Data(base64Encoded: b64), + let findState = try? JSONDecoder().decode(FindInArticleState.self, from: rawData) else { + return + } + + completionHandler(findState) + } + } + + func endSearch() { + webView?.evaluateJavaScript("endFind()") + } + + func selectNextSearchResult() { + webView?.evaluateJavaScript("selectNextResult()") + } + + func selectPreviousSearchResult() { + webView?.evaluateJavaScript("selectPreviousResult()") + } + +} + diff --git a/Multiplatform/iOS/Article/WebViewProvider.swift b/Multiplatform/iOS/Article/WebViewProvider.swift new file mode 100644 index 000000000..dea43b7ff --- /dev/null +++ b/Multiplatform/iOS/Article/WebViewProvider.swift @@ -0,0 +1,133 @@ +// +// WebViewProvider.swift +// Multiplatform iOS +// +// Created by Maurice Parker on 7/6/20. +// Copyright © 2020 Ranchero Software. All rights reserved. +// + +import UIKit +import RSCore +import WebKit + +/// 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. +class WebViewProvider: NSObject { + + private let articleIconSchemeHandler: ArticleIconSchemeHandler + private let operationQueue = MainThreadOperationQueue() + private var queue = NSMutableArray() + + init(sceneModel: SceneModel) { + articleIconSchemeHandler = ArticleIconSchemeHandler(sceneModel: sceneModel) + super.init() + replenishQueueIfNeeded() + } + + func flushQueue() { + operationQueue.add(WebViewProviderFlushQueueOperation(queue: queue)) + } + + func replenishQueueIfNeeded() { + operationQueue.add(WebViewProviderReplenishQueueOperation(queue: queue, articleIconSchemeHandler: articleIconSchemeHandler)) + } + + 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: NSMutableArray + + init(queue: NSMutableArray) { + self.queue = queue + } + + func run() { + queue.removeAllObjects() + 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: NSMutableArray + private var articleIconSchemeHandler: ArticleIconSchemeHandler + + init(queue: NSMutableArray, articleIconSchemeHandler: ArticleIconSchemeHandler) { + self.queue = queue + self.articleIconSchemeHandler = articleIconSchemeHandler + } + + func run() { + while queue.count < minimumQueueDepth { + let webView = PreloadedWebView(articleIconSchemeHandler: articleIconSchemeHandler) + queue.insert(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: NSMutableArray + private var articleIconSchemeHandler: ArticleIconSchemeHandler + private var completion: (PreloadedWebView) -> () + + init(queue: NSMutableArray, articleIconSchemeHandler: ArticleIconSchemeHandler, completion: @escaping (PreloadedWebView) -> ()) { + self.queue = queue + self.articleIconSchemeHandler = articleIconSchemeHandler + self.completion = completion + } + + func run() { + if let webView = queue.lastObject as? PreloadedWebView { + queue.removeLastObject() + webView.ready { preloadedWebView in + self.completion(preloadedWebView) + self.operationDelegate?.operationDidComplete(self) + } + return + } + + assertionFailure("Creating PreloadedWebView in \(#function); queue has run dry.") + + let webView = PreloadedWebView(articleIconSchemeHandler: articleIconSchemeHandler) + webView.preload() + webView.ready { preloadedWebView in + self.completion(preloadedWebView) + self.operationDelegate?.operationDidComplete(self) + } + } + +} diff --git a/Multiplatform/iOS/Article/WrapperScriptMessageHandler.swift b/Multiplatform/iOS/Article/WrapperScriptMessageHandler.swift new file mode 100644 index 000000000..e3a58e19d --- /dev/null +++ b/Multiplatform/iOS/Article/WrapperScriptMessageHandler.swift @@ -0,0 +1,25 @@ +// +// WrapperScriptMessageHandler.swift +// Multiplatform iOS +// +// Created by Maurice Parker on 7/6/20. +// Copyright © 2020 Ranchero Software. All rights reserved. +// + +import Foundation +import WebKit + +class WrapperScriptMessageHandler: NSObject, WKScriptMessageHandler { + + // We need to wrap a message handler to prevent a circlular reference + private weak var handler: WKScriptMessageHandler? + + init(_ handler: WKScriptMessageHandler) { + self.handler = handler + } + + func userContentController(_ userContentController: WKUserContentController, didReceive message: WKScriptMessage) { + handler?.userContentController(userContentController, didReceive: message) + } + +} diff --git a/Multiplatform/macOS/Article/ArticleView.swift b/Multiplatform/macOS/Article/ArticleView.swift index 309551225..dbf33d14c 100644 --- a/Multiplatform/macOS/Article/ArticleView.swift +++ b/Multiplatform/macOS/Article/ArticleView.swift @@ -7,15 +7,15 @@ // import SwiftUI +import Articles struct ArticleView: View { - var body: some View { + + var sceneModel: SceneModel + var articleModel: ArticleModel + var article: Article + + var body: some View { Text(/*@START_MENU_TOKEN@*/"Hello, World!"/*@END_MENU_TOKEN@*/) } } - -struct ArticleView_Previews: PreviewProvider { - static var previews: some View { - ArticleView() - } -} diff --git a/NetNewsWire.xcodeproj/project.pbxproj b/NetNewsWire.xcodeproj/project.pbxproj index ffc5f39e2..a8be3d0c1 100644 --- a/NetNewsWire.xcodeproj/project.pbxproj +++ b/NetNewsWire.xcodeproj/project.pbxproj @@ -219,6 +219,16 @@ 5177470924B2F87600EB0F74 /* SidebarListStyleModifier.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5177470824B2F87600EB0F74 /* SidebarListStyleModifier.swift */; }; 5177470A24B2F87600EB0F74 /* SidebarListStyleModifier.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5177470824B2F87600EB0F74 /* SidebarListStyleModifier.swift */; }; 5177470E24B2FF6F00EB0F74 /* ArticleView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5177470D24B2FF6F00EB0F74 /* ArticleView.swift */; }; + 5177471024B3029400EB0F74 /* ArticleViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5177470F24B3029400EB0F74 /* ArticleViewController.swift */; }; + 5177471224B37C5400EB0F74 /* WebViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5177471124B37C5400EB0F74 /* WebViewController.swift */; }; + 5177471424B37D4000EB0F74 /* PreloadedWebView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5177471324B37D4000EB0F74 /* PreloadedWebView.swift */; }; + 5177471624B37D9700EB0F74 /* ArticleIconSchemeHandler.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5177471524B37D9700EB0F74 /* ArticleIconSchemeHandler.swift */; }; + 5177471824B3812200EB0F74 /* IconView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5177471724B3812200EB0F74 /* IconView.swift */; }; + 5177471A24B3863000EB0F74 /* WebViewProvider.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5177471924B3863000EB0F74 /* WebViewProvider.swift */; }; + 5177471C24B387AC00EB0F74 /* ImageScrollView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5177471B24B387AC00EB0F74 /* ImageScrollView.swift */; }; + 5177471E24B387E100EB0F74 /* ImageTransition.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5177471D24B387E100EB0F74 /* ImageTransition.swift */; }; + 5177472024B3882600EB0F74 /* ImageViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5177471F24B3882600EB0F74 /* ImageViewController.swift */; }; + 5177472224B38CAE00EB0F74 /* ArticleExtractorButtonState.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5177472124B38CAE00EB0F74 /* ArticleExtractorButtonState.swift */; }; 5177475C24B39AD500EB0F74 /* Credits.rtf in Resources */ = {isa = PBXBuildFile; fileRef = 5177475824B39AD400EB0F74 /* Credits.rtf */; }; 5177475D24B39AD500EB0F74 /* Dedication.rtf in Resources */ = {isa = PBXBuildFile; fileRef = 5177475924B39AD400EB0F74 /* Dedication.rtf */; }; 5177475E24B39AD500EB0F74 /* Thanks.rtf in Resources */ = {isa = PBXBuildFile; fileRef = 5177475A24B39AD500EB0F74 /* Thanks.rtf */; }; @@ -233,6 +243,17 @@ 517A757A24451C0700B553B9 /* OAuthSwift.framework in Embed Frameworks */ = {isa = PBXBuildFile; fileRef = 517A755524451BD500B553B9 /* OAuthSwift.framework */; settings = {ATTRIBUTES = (CodeSignOnCopy, RemoveHeadersOnCopy, ); }; }; 517A757B24451C1500B553B9 /* OAuthSwift.framework in Frameworks */ = {isa = PBXBuildFile; fileRef = 517A755324451BD500B553B9 /* OAuthSwift.framework */; }; 517A757C24451C1500B553B9 /* OAuthSwift.framework in Embed Frameworks */ = {isa = PBXBuildFile; fileRef = 517A755324451BD500B553B9 /* OAuthSwift.framework */; settings = {ATTRIBUTES = (CodeSignOnCopy, RemoveHeadersOnCopy, ); }; }; + 517B2EBC24B3E62A001AC46C /* WrapperScriptMessageHandler.swift in Sources */ = {isa = PBXBuildFile; fileRef = 517B2EBB24B3E62A001AC46C /* WrapperScriptMessageHandler.swift */; }; + 517B2EE224B3E8FE001AC46C /* page.html in Resources */ = {isa = PBXBuildFile; fileRef = 517B2EDE24B3E8FE001AC46C /* page.html */; }; + 517B2EE324B3E8FE001AC46C /* page.html in Resources */ = {isa = PBXBuildFile; fileRef = 517B2EDE24B3E8FE001AC46C /* page.html */; }; + 517B2EE424B3E8FE001AC46C /* blank.html in Resources */ = {isa = PBXBuildFile; fileRef = 517B2EDF24B3E8FE001AC46C /* blank.html */; }; + 517B2EE524B3E8FE001AC46C /* blank.html in Resources */ = {isa = PBXBuildFile; fileRef = 517B2EDF24B3E8FE001AC46C /* blank.html */; }; + 517B2EE624B3E8FE001AC46C /* styleSheet.css in Resources */ = {isa = PBXBuildFile; fileRef = 517B2EE024B3E8FE001AC46C /* styleSheet.css */; }; + 517B2EE724B3E8FE001AC46C /* styleSheet.css in Resources */ = {isa = PBXBuildFile; fileRef = 517B2EE024B3E8FE001AC46C /* styleSheet.css */; }; + 517B2EE824B3E8FE001AC46C /* main_multiplatform.js in Resources */ = {isa = PBXBuildFile; fileRef = 517B2EE124B3E8FE001AC46C /* main_multiplatform.js */; }; + 517B2EE924B3E8FE001AC46C /* main_multiplatform.js in Resources */ = {isa = PBXBuildFile; fileRef = 517B2EE124B3E8FE001AC46C /* main_multiplatform.js */; }; + 517B2EEB24B40E09001AC46C /* TimelineTitleModifier.swift in Sources */ = {isa = PBXBuildFile; fileRef = 517B2EEA24B40E09001AC46C /* TimelineTitleModifier.swift */; }; + 517B2EEC24B40E09001AC46C /* TimelineTitleModifier.swift in Sources */ = {isa = PBXBuildFile; fileRef = 517B2EEA24B40E09001AC46C /* TimelineTitleModifier.swift */; }; 5181C5AD24AF89B1002E0F70 /* PreferredColorSchemeModifier.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5181C5AC24AF89B1002E0F70 /* PreferredColorSchemeModifier.swift */; }; 5181C5AE24AF89B1002E0F70 /* PreferredColorSchemeModifier.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5181C5AC24AF89B1002E0F70 /* PreferredColorSchemeModifier.swift */; }; 5181C66224B0C326002E0F70 /* SettingsModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5181C66124B0C326002E0F70 /* SettingsModel.swift */; }; @@ -1872,6 +1893,16 @@ 5177470524B2910300EB0F74 /* ArticleToolbarModifier.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ArticleToolbarModifier.swift; sourceTree = ""; }; 5177470824B2F87600EB0F74 /* SidebarListStyleModifier.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SidebarListStyleModifier.swift; sourceTree = ""; }; 5177470D24B2FF6F00EB0F74 /* ArticleView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ArticleView.swift; sourceTree = ""; }; + 5177470F24B3029400EB0F74 /* ArticleViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ArticleViewController.swift; sourceTree = ""; }; + 5177471124B37C5400EB0F74 /* WebViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = WebViewController.swift; sourceTree = ""; }; + 5177471324B37D4000EB0F74 /* PreloadedWebView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PreloadedWebView.swift; sourceTree = ""; }; + 5177471524B37D9700EB0F74 /* ArticleIconSchemeHandler.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ArticleIconSchemeHandler.swift; sourceTree = ""; }; + 5177471724B3812200EB0F74 /* IconView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = IconView.swift; sourceTree = ""; }; + 5177471924B3863000EB0F74 /* WebViewProvider.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = WebViewProvider.swift; sourceTree = ""; }; + 5177471B24B387AC00EB0F74 /* ImageScrollView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ImageScrollView.swift; sourceTree = ""; }; + 5177471D24B387E100EB0F74 /* ImageTransition.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ImageTransition.swift; sourceTree = ""; }; + 5177471F24B3882600EB0F74 /* ImageViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ImageViewController.swift; sourceTree = ""; }; + 5177472124B38CAE00EB0F74 /* ArticleExtractorButtonState.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ArticleExtractorButtonState.swift; sourceTree = ""; }; 5177475824B39AD400EB0F74 /* Credits.rtf */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.rtf; path = Credits.rtf; sourceTree = ""; }; 5177475924B39AD400EB0F74 /* Dedication.rtf */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.rtf; path = Dedication.rtf; sourceTree = ""; }; 5177475A24B39AD500EB0F74 /* Thanks.rtf */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.rtf; path = Thanks.rtf; sourceTree = ""; }; @@ -1881,6 +1912,12 @@ 5177476624B3BE3400EB0F74 /* SettingsAboutModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SettingsAboutModel.swift; sourceTree = ""; }; 517A745A2443665000B553B9 /* UIPageViewController-Extensions.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UIPageViewController-Extensions.swift"; sourceTree = ""; }; 517A754424451BD500B553B9 /* OAuthSwift.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = OAuthSwift.xcodeproj; path = submodules/OAuthSwift/OAuthSwift.xcodeproj; sourceTree = ""; }; + 517B2EBB24B3E62A001AC46C /* WrapperScriptMessageHandler.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = WrapperScriptMessageHandler.swift; sourceTree = ""; }; + 517B2EDE24B3E8FE001AC46C /* page.html */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.html; path = page.html; sourceTree = ""; }; + 517B2EDF24B3E8FE001AC46C /* blank.html */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.html; path = blank.html; sourceTree = ""; }; + 517B2EE024B3E8FE001AC46C /* styleSheet.css */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.css; path = styleSheet.css; sourceTree = ""; }; + 517B2EE124B3E8FE001AC46C /* main_multiplatform.js */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.javascript; path = main_multiplatform.js; sourceTree = ""; }; + 517B2EEA24B40E09001AC46C /* TimelineTitleModifier.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = TimelineTitleModifier.swift; sourceTree = ""; }; 5181C5AC24AF89B1002E0F70 /* PreferredColorSchemeModifier.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PreferredColorSchemeModifier.swift; sourceTree = ""; }; 5181C66124B0C326002E0F70 /* SettingsModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SettingsModel.swift; sourceTree = ""; }; 5183CCCF226E1E880010922C /* NonIntrinsicLabel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = NonIntrinsicLabel.swift; sourceTree = ""; }; @@ -2424,9 +2461,7 @@ 172199EB24AB228E00A31D04 /* Settings */ = { isa = PBXGroup; children = ( - 65CBAD5924AE03C20006DD91 /* ColorPaletteContainerView.swift */, 65C2E40024B05D8A000AFDF6 /* FeedsSettingsModel.swift */, - 17B223DB24AC24D2001E4592 /* TimelineLayoutView.swift */, 65CBAD5924AE03C20006DD91 /* ColorPaletteContainerView.swift */, 5181C66124B0C326002E0F70 /* SettingsModel.swift */, 172199C824AB228900A31D04 /* SettingsView.swift */, @@ -2663,7 +2698,18 @@ 5177470B24B2FF2C00EB0F74 /* Article */ = { isa = PBXGroup; children = ( + 5177472124B38CAE00EB0F74 /* ArticleExtractorButtonState.swift */, + 5177471524B37D9700EB0F74 /* ArticleIconSchemeHandler.swift */, 5177470D24B2FF6F00EB0F74 /* ArticleView.swift */, + 5177470F24B3029400EB0F74 /* ArticleViewController.swift */, + 5177471724B3812200EB0F74 /* IconView.swift */, + 5177471B24B387AC00EB0F74 /* ImageScrollView.swift */, + 5177471D24B387E100EB0F74 /* ImageTransition.swift */, + 5177471F24B3882600EB0F74 /* ImageViewController.swift */, + 5177471324B37D4000EB0F74 /* PreloadedWebView.swift */, + 5177471124B37C5400EB0F74 /* WebViewController.swift */, + 5177471924B3863000EB0F74 /* WebViewProvider.swift */, + 517B2EBB24B3E62A001AC46C /* WrapperScriptMessageHandler.swift */, ); path = Article; sourceTree = ""; @@ -2762,6 +2808,7 @@ 514E6C0124AD29A300AC6F6E /* TimelineItemStatusView.swift */, 514E6BD924ACEA0400AC6F6E /* TimelineItemView.swift */, 51919FF024AB864A00541E64 /* TimelineModel.swift */, + 517B2EEA24B40E09001AC46C /* TimelineTitleModifier.swift */, 51919FF624AB8B7700541E64 /* TimelineView.swift */, ); path = Timeline; @@ -2779,6 +2826,10 @@ 51A576B924AE617B00078888 /* Article */ = { isa = PBXGroup; children = ( + 517B2EE024B3E8FE001AC46C /* styleSheet.css */, + 517B2EDF24B3E8FE001AC46C /* blank.html */, + 517B2EDE24B3E8FE001AC46C /* page.html */, + 517B2EE124B3E8FE001AC46C /* main_multiplatform.js */, 51A5769524AE617200078888 /* ArticleContainerView.swift */, 51A576BA24AE621800078888 /* ArticleModel.swift */, 5177470524B2910300EB0F74 /* ArticleToolbarModifier.swift */, @@ -4534,14 +4585,18 @@ buildActionMask = 2147483647; files = ( 5177475E24B39AD500EB0F74 /* Thanks.rtf in Resources */, + 517B2EE224B3E8FE001AC46C /* page.html in Resources */, 51E4995F24A875F300B667CB /* shared.css in Resources */, + 517B2EE824B3E8FE001AC46C /* main_multiplatform.js in Resources */, 51E4997324A8784300B667CB /* DefaultFeeds.opml in Resources */, 51C0516224A77DF800194D5E /* Assets.xcassets in Resources */, 5177475F24B39AD500EB0F74 /* About.rtf in Resources */, 51E4996024A875F300B667CB /* template.html in Resources */, 5177475D24B39AD500EB0F74 /* Dedication.rtf in Resources */, 51E4995E24A875F300B667CB /* newsfoot.js in Resources */, + 517B2EE424B3E8FE001AC46C /* blank.html in Resources */, 5177475C24B39AD500EB0F74 /* Credits.rtf in Resources */, + 517B2EE624B3E8FE001AC46C /* styleSheet.css in Resources */, 51E4995D24A875F300B667CB /* main.js in Resources */, ); runOnlyForDeploymentPostprocessing = 0; @@ -4550,12 +4605,16 @@ isa = PBXResourcesBuildPhase; buildActionMask = 2147483647; files = ( + 517B2EE324B3E8FE001AC46C /* page.html in Resources */, 51E4996424A875F400B667CB /* shared.css in Resources */, 51E4997524A8784400B667CB /* DefaultFeeds.opml in Resources */, 51C0516324A77DF800194D5E /* Assets.xcassets in Resources */, 51E4996524A875F400B667CB /* template.html in Resources */, + 517B2EE924B3E8FE001AC46C /* main_multiplatform.js in Resources */, + 517B2EE524B3E8FE001AC46C /* blank.html in Resources */, 51E4996324A875F400B667CB /* newsfoot.js in Resources */, 51E4996224A875F400B667CB /* main.js in Resources */, + 517B2EE724B3E8FE001AC46C /* styleSheet.css in Resources */, ); runOnlyForDeploymentPostprocessing = 0; }; @@ -4880,6 +4939,7 @@ 51E4995924A873F900B667CB /* ErrorHandler.swift in Sources */, 51392D1B24AC19A000BE0D35 /* SidebarExpandedContainers.swift in Sources */, 51E4992F24A8676400B667CB /* ArticleArray.swift in Sources */, + 5177471C24B387AC00EB0F74 /* ImageScrollView.swift in Sources */, 51E498F824A8085D00B667CB /* UnreadFeed.swift in Sources */, FF64D0E724AF53EE0084080A /* RefreshProgressModel.swift in Sources */, 51E4996A24A8762D00B667CB /* ExtractedArticle.swift in Sources */, @@ -4893,11 +4953,13 @@ 5177470624B2910300EB0F74 /* ArticleToolbarModifier.swift in Sources */, 51919FAF24AA8EFA00541E64 /* SidebarItemView.swift in Sources */, 514E6BDA24ACEA0400AC6F6E /* TimelineItemView.swift in Sources */, + 5177471624B37D9700EB0F74 /* ArticleIconSchemeHandler.swift in Sources */, FA80C11724B0728000974098 /* AddFolderView.swift in Sources */, 51E4990D24A808C500B667CB /* RSHTMLMetadata+Extension.swift in Sources */, 51919FF424AB869C00541E64 /* TimelineItem.swift in Sources */, 514E6C0224AD29A300AC6F6E /* TimelineItemStatusView.swift in Sources */, 51E49A0024A91FC100B667CB /* SidebarContainerView.swift in Sources */, + 5177471824B3812200EB0F74 /* IconView.swift in Sources */, 51E4995C24A875F300B667CB /* ArticleRenderer.swift in Sources */, 51E4992324A8095700B667CB /* URL-Extensions.swift in Sources */, 51E4993624A867E800B667CB /* UserInfoKey.swift in Sources */, @@ -4920,6 +4982,7 @@ 51E4991B24A8091000B667CB /* IconImage.swift in Sources */, 51E4995424A8734D00B667CB /* ExtensionPointIdentifer.swift in Sources */, 51E4996924A8760C00B667CB /* ArticleStylesManager.swift in Sources */, + 5177471E24B387E100EB0F74 /* ImageTransition.swift in Sources */, 51E498F324A8085D00B667CB /* PseudoFeed.swift in Sources */, 51A5769624AE617200078888 /* ArticleContainerView.swift in Sources */, 5181C5AD24AF89B1002E0F70 /* PreferredColorSchemeModifier.swift in Sources */, @@ -4929,8 +4992,10 @@ 51E4991D24A8092100B667CB /* NSAttributedString+NetNewsWire.swift in Sources */, FF64D0E924AF53EE0084080A /* RefreshProgressView.swift in Sources */, 51E499FD24A9137600B667CB /* SidebarModel.swift in Sources */, + 517B2EEB24B40E09001AC46C /* TimelineTitleModifier.swift in Sources */, 5181C66224B0C326002E0F70 /* SettingsModel.swift in Sources */, 51E4995324A8734D00B667CB /* RedditFeedProvider-Extensions.swift in Sources */, + 5177471024B3029400EB0F74 /* ArticleViewController.swift in Sources */, 172199C924AB228900A31D04 /* SettingsView.swift in Sources */, 17D232A824AFF10A0005F075 /* AddWebFeedModel.swift in Sources */, 51E4994224A8713C00B667CB /* ArticleStatusSyncTimer.swift in Sources */, @@ -4939,6 +5004,7 @@ 51919FB624AABCA100541E64 /* IconImageView.swift in Sources */, 51919FA624AA64B000541E64 /* SidebarView.swift in Sources */, 51E4997024A8764C00B667CB /* ActivityManager.swift in Sources */, + 5177471224B37C5400EB0F74 /* WebViewController.swift in Sources */, 51E4990F24A808CC00B667CB /* HTMLMetadataDownloader.swift in Sources */, 5177476524B3BDAE00EB0F74 /* AttributedStringView.swift in Sources */, 51E4993124A8676400B667CB /* FetchRequestOperation.swift in Sources */, @@ -4952,7 +5018,10 @@ 51E4991E24A8094300B667CB /* RSImage-AppIcons.swift in Sources */, 51E499D824A912C200B667CB /* SceneModel.swift in Sources */, 5177470E24B2FF6F00EB0F74 /* ArticleView.swift in Sources */, + 5177471424B37D4000EB0F74 /* PreloadedWebView.swift in Sources */, + 517B2EBC24B3E62A001AC46C /* WrapperScriptMessageHandler.swift in Sources */, 51919FB324AAB97900541E64 /* FeedIconImageLoader.swift in Sources */, + 5177472024B3882600EB0F74 /* ImageViewController.swift in Sources */, 51919FB324AAB97900541E64 /* FeedIconImageLoader.swift in Sources */, 51E4991324A808FB00B667CB /* AddWebFeedDefaultContainer.swift in Sources */, 51E4993C24A8709900B667CB /* AppDelegate.swift in Sources */, @@ -4961,6 +5030,8 @@ 17930ED424AF10EE00A9BA52 /* AddWebFeedView.swift in Sources */, 51E4995124A8734D00B667CB /* ExtensionPointManager.swift in Sources */, 51E4990C24A808C500B667CB /* AuthorAvatarDownloader.swift in Sources */, + 5177472224B38CAE00EB0F74 /* ArticleExtractorButtonState.swift in Sources */, + 5177471A24B3863000EB0F74 /* WebViewProvider.swift in Sources */, 51E4992124A8095000B667CB /* RSImage-Extensions.swift in Sources */, 51E4990324A808BB00B667CB /* FaviconDownloader.swift in Sources */, 172199ED24AB2E0100A31D04 /* SafariView.swift in Sources */, @@ -5030,6 +5101,7 @@ 51E4995B24A875D500B667CB /* ArticlePasteboardWriter.swift in Sources */, 51E4993424A867E700B667CB /* UserInfoKey.swift in Sources */, 1776E88F24AC5F8A00E78166 /* AppDefaults.swift in Sources */, + 517B2EEC24B40E09001AC46C /* TimelineTitleModifier.swift in Sources */, 1729529724AA1CD000D65E66 /* MacPreferencesView.swift in Sources */, 51E4994C24A8734C00B667CB /* RedditFeedProvider-Extensions.swift in Sources */, 1729529324AA1CAA00D65E66 /* AccountsPreferencesView.swift in Sources */, diff --git a/iOS/Article/WebViewProvider.swift b/iOS/Article/WebViewProvider.swift index 41c0e550c..0a00d667f 100644 --- a/iOS/Article/WebViewProvider.swift +++ b/iOS/Article/WebViewProvider.swift @@ -12,7 +12,7 @@ import WebKit /// 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. -class WebViewProvider: NSObject { +class : NSObject { private let articleIconSchemeHandler: ArticleIconSchemeHandler private let operationQueue = MainThreadOperationQueue()