From 3023ba6e8e82ee01cef1e522a7f0fee287197465 Mon Sep 17 00:00:00 2001 From: uttiya10 <56562649+uttiya10@users.noreply.github.com> Date: Wed, 30 Jun 2021 22:04:10 -0400 Subject: [PATCH 1/6] Refactor RootViewController to Allow Opening AnyLinks from the currently displayed navigation controller in TabBarController --- ...nimeScheduleCollectionViewController.swift | 2 +- .../AnimeInformationTableViewController.swift | 5 +- .../ContentListViewController.swift | 2 +- .../LibrarySceneController+Tips.swift | 2 +- .../LibrarySceneController.swift | 2 +- ...ibrarySubscriptionCategoryController.swift | 2 +- .../Controllers/RootViewController.swift | 72 ++++++++++++------- .../Search Scene/SearchViewController.swift | 4 +- 8 files changed, 55 insertions(+), 36 deletions(-) diff --git a/NineAnimator/Controllers/Anime Discovery Scenes/AnimeScheduleCollectionViewController.swift b/NineAnimator/Controllers/Anime Discovery Scenes/AnimeScheduleCollectionViewController.swift index 07d0ad86..92bec1e8 100644 --- a/NineAnimator/Controllers/Anime Discovery Scenes/AnimeScheduleCollectionViewController.swift +++ b/NineAnimator/Controllers/Anime Discovery Scenes/AnimeScheduleCollectionViewController.swift @@ -139,7 +139,7 @@ class AnimeScheduleCollectionViewController: MinFilledCollectionViewController, override func collectionView(_ collectionView: UICollectionView, didSelectItemAt indexPath: IndexPath) { if let view = collectionView.cellForItem(at: indexPath) as? CalendarAnimeCell, let item = view.representingScheduledAnime { - RootViewController.shared?.open(immedietly: item.link, in: self) + RootViewController.shared?.open(immedietly: item.link, method: .inController(self)) } } } diff --git a/NineAnimator/Controllers/Anime Information Scene/AnimeInformationTableViewController.swift b/NineAnimator/Controllers/Anime Information Scene/AnimeInformationTableViewController.swift index 93d10c0a..9ef93abd 100644 --- a/NineAnimator/Controllers/Anime Information Scene/AnimeInformationTableViewController.swift +++ b/NineAnimator/Controllers/Anime Information Scene/AnimeInformationTableViewController.swift @@ -319,7 +319,8 @@ class AnimeInformationTableViewController: UITableViewController, DontBotherView let cell = tableView.dequeueReusableCell(withIdentifier: "anime.related", for: indexPath) as! InformationSceneRelatedTableViewCell cell.initialize(_relatedReferences!) { [weak self] reference in - RootViewController.shared?.open(immedietly: .listingReference(reference), in: self) + guard let self = self else { return } + RootViewController.shared?.open(immedietly: .listingReference(reference), method: .inController(self)) } return cell case .airingSchedule: @@ -658,7 +659,7 @@ extension AnimeInformationTableViewController { /// Open the match directly private func onPerfectMatch(_ animeLink: AnimeLink) { if presentingAnimeInformation == nil { navigationController?.popViewController(animated: true) } - RootViewController.shared?.open(immedietly: .anime(animeLink), in: self) + RootViewController.shared?.open(immedietly: .anime(animeLink), method: .inController(self)) } /// Present options to the user for multiple match diff --git a/NineAnimator/Controllers/Content List Scene/ContentListViewController.swift b/NineAnimator/Controllers/Content List Scene/ContentListViewController.swift index 62f37d7f..3d385893 100644 --- a/NineAnimator/Controllers/Content List Scene/ContentListViewController.swift +++ b/NineAnimator/Controllers/Content List Scene/ContentListViewController.swift @@ -153,7 +153,7 @@ extension ContentListViewController { // Open the link if it exists if let cell = cell as? ListingEntryTableViewCell, let link = cell.link { - RootViewController.shared?.open(immedietly: link, in: self) + RootViewController.shared?.open(immedietly: link, method: .inController(self)) } } } diff --git a/NineAnimator/Controllers/Library Scene/LibrarySceneController+Tips.swift b/NineAnimator/Controllers/Library Scene/LibrarySceneController+Tips.swift index 07a92482..0ebce7dd 100644 --- a/NineAnimator/Controllers/Library Scene/LibrarySceneController+Tips.swift +++ b/NineAnimator/Controllers/Library Scene/LibrarySceneController+Tips.swift @@ -42,7 +42,7 @@ extension LibrarySceneController { let anime = updatedAnimeLinks.first { RootViewController.shared?.open( immedietly: .anime(anime), - in: parent + method: .inController(parent) ) } else if let category = parent.category(withIdentifier: "library.category.subscribed") { // Else present the subscription category diff --git a/NineAnimator/Controllers/Library Scene/LibrarySceneController.swift b/NineAnimator/Controllers/Library Scene/LibrarySceneController.swift index a243bedc..dc843789 100644 --- a/NineAnimator/Controllers/Library Scene/LibrarySceneController.swift +++ b/NineAnimator/Controllers/Library Scene/LibrarySceneController.swift @@ -434,7 +434,7 @@ extension LibrarySceneController { let anime = cachedRecentlyWatchedList[indexPath.item] RootViewController.shared?.open( immedietly: .anime(anime), - in: self + method: .inController(self) ) case .collection: let collection = self.collection(atPath: indexPath) diff --git a/NineAnimator/Controllers/Library Scene/LibrarySubscriptionCategoryController.swift b/NineAnimator/Controllers/Library Scene/LibrarySubscriptionCategoryController.swift index c4c19f7e..57248f8a 100644 --- a/NineAnimator/Controllers/Library Scene/LibrarySubscriptionCategoryController.swift +++ b/NineAnimator/Controllers/Library Scene/LibrarySubscriptionCategoryController.swift @@ -99,7 +99,7 @@ extension LibrarySubscriptionCategoryController { override func collectionView(_ collectionView: UICollectionView, didSelectItemAt indexPath: IndexPath) { let link = cachedWatchedAnimeItems[indexPath.item] - RootViewController.shared?.open(immedietly: link, in: self) + RootViewController.shared?.open(immedietly: link, method: .inController(self)) } override func collectionView(_ collectionView: UICollectionView, canMoveItemAt indexPath: IndexPath) -> Bool { diff --git a/NineAnimator/Controllers/RootViewController.swift b/NineAnimator/Controllers/RootViewController.swift index 56b78de7..3c7afcd6 100644 --- a/NineAnimator/Controllers/RootViewController.swift +++ b/NineAnimator/Controllers/RootViewController.swift @@ -59,7 +59,7 @@ class RootViewController: UITabBarController, Themable { // Open the pending if there is any if let pendingOpeningLink = RootViewController._pendingOpeningLink { RootViewController._pendingOpeningLink = nil - open(immedietly: pendingOpeningLink) + open(immedietly: pendingOpeningLink, method: .inWatchTab) } // Restore config if there is any @@ -144,7 +144,7 @@ extension RootViewController { /// Open an `AnyLink` struct static func open(whenReady link: AnyLink) { if let sharedRootViewController = shared { - sharedRootViewController.open(immedietly: link) + sharedRootViewController.open(immedietly: link, method: .inWatchTab) } else { _pendingOpeningLink = link } } @@ -167,7 +167,46 @@ extension RootViewController { extension RootViewController { fileprivate static var _pendingOpeningLink: AnyLink? - func open(immedietly link: AnyLink, in viewController: UIViewController? = nil) { + enum LinkOpeningMethod { + /// Opens the link in the controller's nearest navigation controller or presents it modally if no navigation controller is found + case inController(UIViewController) + /// Opens the link in the navigation controller that is currently selected in the UITabBarController + case inCurrentlyDisplayedTab + /// Opens the link in the watch next tab + /// - Note: Any UIViewControllers in the watch next tab will be popped off the navigation stack before opening the link + case inWatchTab + } + + func open(immedietly link: AnyLink, method: LinkOpeningMethod) { + guard let controllerForLink = retrieveViewController(forLink: link) else { + Log.error("[RootViewController] Failed to retrieve view controller for anylink: %@", link) + return + } + + switch method { + case let .inController(controller): + if let navigationController = controller.navigationController { + navigationController.pushViewController(controllerForLink, animated: true) + } else { controller.present(controllerForLink, animated: true) } + case .inWatchTab: + guard let navController = viewControllers?[NineAnimatorRootScene.toWatch.rawValue] as? ApplicationNavigationController else { + Log.error("[RootViewController] The watch next tab controller is not ApplicationNavigationController.") + return + } + self.navigate(toScene: .toWatch) + navController.popToRootViewController(animated: true) + navController.pushViewController(controllerForLink, animated: true) + case .inCurrentlyDisplayedTab: + guard let navController = viewControllers?[selectedIndex] as? ApplicationNavigationController else { + Log.error("[RootViewController] The currently selected tab controller is not ApplicationNavigationController.") + return + } + navController.pushViewController(controllerForLink, animated: true) + } + } + + /// Initializes view controller responsible for the corresponding AnyLink + private func retrieveViewController(forLink link: AnyLink) -> UIViewController? { let targetViewController: UIViewController // Determine if the link is supported @@ -178,7 +217,7 @@ extension RootViewController { // Instantiate the view controller from storyboard guard let animeViewController = storyboard.instantiateInitialViewController() as? AnimeViewController else { Log.error("The view controller instantiated from AnimePlayer.storyboard is not AnimeViewController.") - return + return nil } // Initialize the AnimeViewController with the link @@ -190,35 +229,14 @@ extension RootViewController { // Instantiate the view controller from storyboard guard let animeInformationController = storyboard.instantiateInitialViewController() as? AnimeInformationTableViewController else { Log.error("The view controller instantiated from AnimeInformation.storyboard is not AnimeInformationTableViewController.") - return + return nil } // Initialize the AnimeInformationTableViewController with the link animeInformationController.setPresenting(link) targetViewController = animeInformationController } - - // If a view controller is provided - if let viewController = viewController { - // If the provided view controller has a navigation controller, - // open the link in the navigation controller. Else present it - // directly from the provided view controller. - if let navigationController = viewController.navigationController { - navigationController.pushViewController(targetViewController, animated: true) - } else { viewController.present(targetViewController, animated: true) } - } else { // If no view controller is provided, present the link in the featured tab - // Jump to Featured tab - selectedIndex = 0 - - guard let navigationController = viewControllers?.first as? ApplicationNavigationController else { - Log.error("The first view controller is not ApplicationNavigationController.") - return - } - - // Pop to root view controller - navigationController.popToRootViewController(animated: true) - navigationController.pushViewController(targetViewController, animated: true) - } + return targetViewController } } diff --git a/NineAnimator/Controllers/Search Scene/SearchViewController.swift b/NineAnimator/Controllers/Search Scene/SearchViewController.swift index e1061533..477988d9 100644 --- a/NineAnimator/Controllers/Search Scene/SearchViewController.swift +++ b/NineAnimator/Controllers/Search Scene/SearchViewController.swift @@ -205,12 +205,12 @@ extension SearchViewController { if case let .anime(animeLink) = link { // Open the anime link directly if the current source is the link's source if animeLink.source.name == source.name { - RootViewController.shared?.open(immedietly: link, in: self) + RootViewController.shared?.open(immedietly: link, method: .inController(self)) } else { performSearch(keywords: item.keywords) } } else { - RootViewController.shared?.open(immedietly: link, in: self) + RootViewController.shared?.open(immedietly: link, method: .inController(self)) } } else { // For keywords, just search directly performSearch(keywords: item.keywords) From b50389a76990a534c6131dbc1bf1e2def4e02803 Mon Sep 17 00:00:00 2001 From: uttiya10 <56562649+uttiya10@users.noreply.github.com> Date: Wed, 30 Jun 2021 22:07:49 -0400 Subject: [PATCH 2/6] Add Initial Support for Reverse Image Search --- .../ListServices/TraceMoe/TraceMoe.swift | 159 ++++++++++ NineAnimator.xcodeproj/project.pbxproj | 24 ++ .../xcshareddata/swiftpm/Package.resolved | 10 +- .../Search Scene/SearchViewController.swift | 23 +- .../ImageSearchResultsView+AVPlayer.swift | 106 +++++++ .../ImageSearchResultsView.swift | 215 ++++++++++++++ .../ImageSearchSelectorView+PhotoPicker.swift | 86 ++++++ .../ImageSearchSelectorView.swift | 279 ++++++++++++++++++ 8 files changed, 895 insertions(+), 7 deletions(-) create mode 100644 Modules/Sources/NineAnimatorNativeListServices/ListServices/TraceMoe/TraceMoe.swift create mode 100644 NineAnimator/Views/Image Search Scene/ImageSearchResultsView+AVPlayer.swift create mode 100644 NineAnimator/Views/Image Search Scene/ImageSearchResultsView.swift create mode 100644 NineAnimator/Views/Image Search Scene/ImageSearchSelectorView+PhotoPicker.swift create mode 100644 NineAnimator/Views/Image Search Scene/ImageSearchSelectorView.swift diff --git a/Modules/Sources/NineAnimatorNativeListServices/ListServices/TraceMoe/TraceMoe.swift b/Modules/Sources/NineAnimatorNativeListServices/ListServices/TraceMoe/TraceMoe.swift new file mode 100644 index 00000000..933f27b6 --- /dev/null +++ b/Modules/Sources/NineAnimatorNativeListServices/ListServices/TraceMoe/TraceMoe.swift @@ -0,0 +1,159 @@ +// +// This file is part of the NineAnimator project. +// +// Copyright © 2018-2020 Marcus Zhou. All rights reserved. +// +// NineAnimator is free software: you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation, either version 3 of the License, or +// (at your option) any later version. +// +// NineAnimator is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with NineAnimator. If not, see . +// + +import Alamofire +import NineAnimatorCommon +import UIKit + +public class TraceMoe { + var apiURL: URL { URL(string: "https://api.trace.moe")! } + + let requestManager = NAEndpointRelativeRequestManager() + + public init() {} + + public func search(with image: UIImage) -> NineAnimatorPromise<[TraceMoeSearchResult]> { + NineAnimatorPromise.firstly { + guard let jpegData = image.jpegData(compressionQuality: 0.8) else { + throw NineAnimatorError.unknownError("Could not convert image to jpeg.") + } + return jpegData + } .thenPromise { + jpegData -> NineAnimatorPromise in + let jsonDecoder = JSONDecoder() + jsonDecoder.keyDecodingStrategy = .convertFromSnakeCase + + let request = self.requestManager.request { session in + return self.requestManager.session.upload( + multipartFormData: { form in + form.append(jpegData, withName: "image", fileName: "image", mimeType: "image/jpeg") + }, + to: self.apiURL.appendingPathComponent("/search").absoluteString + "?anilistInfo" + ) + } + + return request + .onReceivingResponse { try self.validateResponse($0) } + .responseDecodable(type: TraceMoeSearchResponse.self, decoder: jsonDecoder) + } .then { + responseObject in + let filteredResponses = responseObject.result.filter { $0.similarity >= 0.6 } + guard !filteredResponses.isEmpty else { throw NineAnimatorError.searchError("No Results Found") } + return filteredResponses + } + } + + public func search(with imageURL: URL) -> NineAnimatorPromise<[TraceMoeSearchResult]> { + let jsonDecoder = JSONDecoder() + jsonDecoder.keyDecodingStrategy = .convertFromSnakeCase + return self.requestManager.request( + url: self.apiURL.appendingPathComponent("/search"), + method: .post, + query: [ + "url": imageURL.absoluteString, + "anilistInfo": "" + ] + ) + .onReceivingResponse { try self.validateResponse($0) } + .responseDecodable(type: TraceMoeSearchResponse.self, decoder: jsonDecoder) + .then { + responseObject in + let filteredResponses = responseObject.result.filter { $0.similarity >= 0.6 } + guard !filteredResponses.isEmpty else { throw NineAnimatorError.searchError("No Results Found") } + return filteredResponses + } + } + + private func validateResponse(_ response: NARequestManager.Response) throws { + guard let HTTPresponse = response.response else { return } + switch HTTPresponse.statusCode { + case 400: + throw NineAnimatorError.searchError("Image Was Not Provided") + case 413: + throw NineAnimatorError.searchError("Your image was above 10MB") + case 429: + throw NineAnimatorError.searchError("You are hitting the rate limiter. Calm down!") + case 500..<600: + if let data = response.data, let errorString = String(data: data, encoding: .utf8) { + throw NineAnimatorError.searchError("Trace.moe experienced a backend error: \(errorString)") + } + throw NineAnimatorError.searchError("Trace.moe experienced a backend error") + default: break + } + } +} + +// MARK: - Data Types +public extension TraceMoe { + struct TraceMoeSearchResponse: Codable { + public let result: [TraceMoeSearchResult] + } + + struct TraceMoeSearchResult: Codable { + public let from, to: Double + public let anilist: TraceMoeAnilistInfo + public let filename: String + public let video: URL + public let image: URL + public let episode: TraceMoeEpisode? + public let similarity: Double + } + + struct TraceMoeAnilistInfo: Codable { + public let id, idMal: Int + public let title: TraceMoeAnilistTitle + public let synonyms: [String] + public let isAdult: Bool + } + + struct TraceMoeAnilistTitle: Codable { + public let romaji: String? + public let english: String? + public let native: String? + public let userPreferred: String? + } + + /// Abstracts Trace.moe episode (Int or String) to a string value type + struct TraceMoeEpisode: Codable { + public let value: String + + public init(_ value: String) { + self.value = value + } + + public init(from decoder: Decoder) throws { + let container = try decoder.singleValueContainer() + // Attempt to decode from String, Int, and Double + if let str = try? container.decode(String.self) { + value = str + } else if let int = try? container.decode(Int.self) { + value = String(int) + } else if let double = try? container.decode(Double.self) { + value = String(double) + } else { + throw DecodingError.typeMismatch(String.self, .init(codingPath: decoder.codingPath, debugDescription: "Could not convert trace.moe episode into String")) + } + } + + public func encode(to encoder: Encoder) throws { + var container = encoder.singleValueContainer() + try container.encode(value) + } + } +} diff --git a/NineAnimator.xcodeproj/project.pbxproj b/NineAnimator.xcodeproj/project.pbxproj index fce420ad..1afb53c4 100644 --- a/NineAnimator.xcodeproj/project.pbxproj +++ b/NineAnimator.xcodeproj/project.pbxproj @@ -216,7 +216,11 @@ 52CADD6721BD86F40077BEB1 /* UITableView+DeselectRows.swift in Sources */ = {isa = PBXBuildFile; fileRef = 52CADD6621BD86F40077BEB1 /* UITableView+DeselectRows.swift */; }; BF6820E326817C93004DCDBD /* NineAnimatorCommon in Frameworks */ = {isa = PBXBuildFile; productRef = BF6820E226817C93004DCDBD /* NineAnimatorCommon */; }; BF6820E426817C93004DCDBD /* NineAnimatorCommon in Embed Frameworks */ = {isa = PBXBuildFile; productRef = BF6820E226817C93004DCDBD /* NineAnimatorCommon */; settings = {ATTRIBUTES = (CodeSignOnCopy, ); }; }; + BF8CC475268562AB00A0A485 /* ImageSearchSelectorView.swift in Sources */ = {isa = PBXBuildFile; fileRef = BF8CC474268562AB00A0A485 /* ImageSearchSelectorView.swift */; }; + BF8CC4772685646100A0A485 /* ImageSearchSelectorView+PhotoPicker.swift in Sources */ = {isa = PBXBuildFile; fileRef = BF8CC4762685646100A0A485 /* ImageSearchSelectorView+PhotoPicker.swift */; }; BF9203DC25E07E6E00CB2D91 /* AudioBackgroundController.swift in Sources */ = {isa = PBXBuildFile; fileRef = BF9203DB25E07E6E00CB2D91 /* AudioBackgroundController.swift */; }; + BF99DE63268698240019BFC6 /* ImageSearchResultsView+AVPlayer.swift in Sources */ = {isa = PBXBuildFile; fileRef = BF99DE62268698240019BFC6 /* ImageSearchResultsView+AVPlayer.swift */; }; + BF99DE65268698920019BFC6 /* ImageSearchResultsView.swift in Sources */ = {isa = PBXBuildFile; fileRef = BF99DE64268698920019BFC6 /* ImageSearchResultsView.swift */; }; BFF95F0A257C2DA8001F8484 /* SettingsSceneController+DocumentPicker.swift in Sources */ = {isa = PBXBuildFile; fileRef = BFF95F09257C2DA8001F8484 /* SettingsSceneController+DocumentPicker.swift */; }; /* End PBXBuildFile section */ @@ -540,7 +544,11 @@ 2CF8912121DB39A1006D4799 /* icon_rendered.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = icon_rendered.png; sourceTree = ""; }; 52CADD6621BD86F40077BEB1 /* UITableView+DeselectRows.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UITableView+DeselectRows.swift"; sourceTree = ""; }; 52F6A4F221DABDB00005D78D /* .swiftlint.yml */ = {isa = PBXFileReference; lastKnownFileType = text; path = .swiftlint.yml; sourceTree = SOURCE_ROOT; }; + BF8CC474268562AB00A0A485 /* ImageSearchSelectorView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ImageSearchSelectorView.swift; sourceTree = ""; }; + BF8CC4762685646100A0A485 /* ImageSearchSelectorView+PhotoPicker.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "ImageSearchSelectorView+PhotoPicker.swift"; sourceTree = ""; }; BF9203DB25E07E6E00CB2D91 /* AudioBackgroundController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AudioBackgroundController.swift; sourceTree = ""; }; + BF99DE62268698240019BFC6 /* ImageSearchResultsView+AVPlayer.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "ImageSearchResultsView+AVPlayer.swift"; sourceTree = ""; }; + BF99DE64268698920019BFC6 /* ImageSearchResultsView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ImageSearchResultsView.swift; sourceTree = ""; }; BFF95F09257C2DA8001F8484 /* SettingsSceneController+DocumentPicker.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "SettingsSceneController+DocumentPicker.swift"; sourceTree = ""; }; /* End PBXFileReference section */ @@ -869,6 +877,7 @@ 2C4ABC7721B59C95009B4D47 /* Views */ = { isa = PBXGroup; children = ( + BF8CC4732685621F00A0A485 /* Image Search Scene */, 2C0DA2E3238B2784002C6447 /* Settings Scene */, 2C665570235CE70300A6126A /* Library Scene */, 2C71DEC1226CF748004F3AC3 /* Server Selection Scene */, @@ -1261,6 +1270,17 @@ path = "User Notification"; sourceTree = ""; }; + BF8CC4732685621F00A0A485 /* Image Search Scene */ = { + isa = PBXGroup; + children = ( + BF8CC474268562AB00A0A485 /* ImageSearchSelectorView.swift */, + BF8CC4762685646100A0A485 /* ImageSearchSelectorView+PhotoPicker.swift */, + BF99DE62268698240019BFC6 /* ImageSearchResultsView+AVPlayer.swift */, + BF99DE64268698920019BFC6 /* ImageSearchResultsView.swift */, + ); + path = "Image Search Scene"; + sourceTree = ""; + }; /* End PBXGroup section */ /* Begin PBXNativeTarget section */ @@ -1608,12 +1628,14 @@ 2C71DEC5226E0610004F3AC3 /* ServerSelectionView.swift in Sources */, 2CA1DD5C21E2B404009241CB /* HomeController.swift in Sources */, 2C90C6A6236F98C600C18620 /* LibraryTrackingReferenceCell.swift in Sources */, + BF8CC4772685646100A0A485 /* ImageSearchSelectorView+PhotoPicker.swift in Sources */, 2C11187125B624F500BD49D2 /* CoreExportsLogging.swift in Sources */, 2C90C6A4236F988100C18620 /* LibraryUntrackedReferenceCell.swift in Sources */, 2CB004252200AEBC00351B7E /* OfflineContent.swift in Sources */, 2C96DAE424562A2F007A29B0 /* InformationSceneAiringEpisodeCell.swift in Sources */, 2CAA70D321CFF42A00F0D082 /* NativePlayerController.swift in Sources */, 2C368A0D2592F0580069DE91 /* CoreExportsNineAnimator+Fetch.swift in Sources */, + BF8CC475268562AB00A0A485 /* ImageSearchSelectorView.swift in Sources */, 2C9383CD21FBF442008C0D01 /* UIView+MakeThemable.swift in Sources */, 2C1C717322186EC900760A69 /* TrackingServiceTableViewController.swift in Sources */, 2C93BD43221E1FE8000411CB /* InformationSceneStatisticsTableViewCell.swift in Sources */, @@ -1642,6 +1664,7 @@ 2C85367E222EEC7400A5CFB9 /* RecentsSceneCollectionCollectionViewCell.swift in Sources */, 2CB9919421FEAC5D0055B7A7 /* DetailedEpisodeTableViewCell.swift in Sources */, 2C86C1EB225D6F9F00950ABF /* SetupWelcomeViewController.swift in Sources */, + BF99DE65268698920019BFC6 /* ImageSearchResultsView.swift in Sources */, 2C66779821BC2213000E5ACC /* SearchViewController.swift in Sources */, 2CB4AF96239AD1B700EE4EDA /* CachableAVAssetLoaderDelegate.swift in Sources */, 2C9C44FE21CD2B33004C8F0C /* SettingsSceneController.swift in Sources */, @@ -1656,6 +1679,7 @@ 52CADD6721BD86F40077BEB1 /* UITableView+DeselectRows.swift in Sources */, 2C12EDDE21BDE6B20064D2BD /* HalfFillTransitionDelegate.swift in Sources */, 2CE18B0C2374839100771A16 /* LibrarySubscriptionCell.swift in Sources */, + BF99DE63268698240019BFC6 /* ImageSearchResultsView+AVPlayer.swift in Sources */, 2C9383D421FCCC43008C0D01 /* UILabel+StyleAttributes.swift in Sources */, 2C35C968227F223700D469E6 /* CalendarHeaderView.swift in Sources */, 2C12EDDB21BDE6B20064D2BD /* HalfFillViewControllerProtocol.swift in Sources */, diff --git a/NineAnimator.xcworkspace/xcshareddata/swiftpm/Package.resolved b/NineAnimator.xcworkspace/xcshareddata/swiftpm/Package.resolved index 74e77786..d7ca4f93 100644 --- a/NineAnimator.xcworkspace/xcshareddata/swiftpm/Package.resolved +++ b/NineAnimator.xcworkspace/xcshareddata/swiftpm/Package.resolved @@ -15,8 +15,8 @@ "repositoryURL": "https://github.com/microsoft/appcenter-sdk-apple.git", "state": { "branch": null, - "revision": "013bffe628b49433d7bc334146a4ed8c507c0e32", - "version": "4.1.1" + "revision": "6d60d2361445e959b4fbb3a9d0dbae0e07eead1a", + "version": "4.2.0" } }, { @@ -42,7 +42,7 @@ "repositoryURL": "https://github.com/SuperMarcus/NineAnimatorCommon.git", "state": { "branch": "master", - "revision": "8c3c5fd14f7e04fffbe2584e3f05a1e81047e2a6", + "revision": "be4004a86b0e041cfe182894e9ca4c83d9d3e8d1", "version": null } }, @@ -60,8 +60,8 @@ "repositoryURL": "https://github.com/microsoft/PLCrashReporter.git", "state": { "branch": null, - "revision": "de6b8f9db4b2a0aa859a5507550a70548e4da936", - "version": "1.8.1" + "revision": "d747ab5de269cd44022bbe96ff9609d8626694ab", + "version": "1.9.0" } }, { diff --git a/NineAnimator/Controllers/Search Scene/SearchViewController.swift b/NineAnimator/Controllers/Search Scene/SearchViewController.swift index 477988d9..4efd8b87 100644 --- a/NineAnimator/Controllers/Search Scene/SearchViewController.swift +++ b/NineAnimator/Controllers/Search Scene/SearchViewController.swift @@ -18,8 +18,7 @@ // import NineAnimatorCommon -import NineAnimatorNativeParsers -import NineAnimatorNativeSources +import SwiftUI import UIKit class SearchViewController: UITableViewController, UISearchResultsUpdating, UISearchControllerDelegate, UISearchBarDelegate { @@ -68,6 +67,17 @@ class SearchViewController: UITableViewController, UISearchResultsUpdating, UISe // Hide table cell separators at empty state tableView.tableFooterView = UIView() + // Add Reverse Image Search Button + if #available(iOS 14.0, *) { + navigationItem.rightBarButtonItems? + .insert(UIBarButtonItem( + barButtonSystemItem: .camera, + target: self, + action: #selector(onReverseImageSearchButtonTapped)), + at: 0 + ) + } + // Themes tableView.makeThemable() searchController.searchBar.makeThemable() @@ -285,6 +295,15 @@ extension SearchViewController { alertView.addAction(UIAlertAction(title: "Cancel", style: .cancel)) present(alertView, animated: true) } + + @available(iOS 14.0, *) + @objc func onReverseImageSearchButtonTapped() { + let hostingView = UIHostingController(rootView: ImageSearchSelectorView()) + // Manually setting the nav title here to fix a SwiftUI issue + // where the nav title pops in without an animation + hostingView.navigationItem.title = "Image Search" + navigationController?.pushViewController(hostingView, animated: true) + } } // MARK: - Defs & Helpers diff --git a/NineAnimator/Views/Image Search Scene/ImageSearchResultsView+AVPlayer.swift b/NineAnimator/Views/Image Search Scene/ImageSearchResultsView+AVPlayer.swift new file mode 100644 index 00000000..76dcec74 --- /dev/null +++ b/NineAnimator/Views/Image Search Scene/ImageSearchResultsView+AVPlayer.swift @@ -0,0 +1,106 @@ +// +// This file is part of the NineAnimator project. +// +// Copyright © 2018-2020 Marcus Zhou. All rights reserved. +// +// NineAnimator is free software: you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation, either version 3 of the License, or +// (at your option) any later version. +// +// NineAnimator is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with NineAnimator. If not, see . +// + +import AVKit +import SwiftUI + +@available(iOS 14.0, *) +extension ImageSearchResultsView { + struct LoopingVideoPlayer: UIViewRepresentable { + let videoURL: URL + + func updateUIView(_ uiView: UIView, context: UIViewRepresentableContext) { + } + + func makeUIView(context: Context) -> UIView { + LoopingPlayerUIView(frame: .zero, url: videoURL) + } + } + + private class LoopingPlayerUIView: UIView { + private let playerLayer = AVPlayerLayer() + private let loadingIndicator = UIActivityIndicatorView() + private var playerLooper: AVPlayerLooper? + private var player = AVQueuePlayer() + + @available(*, unavailable) + required init?(coder: NSCoder) { + fatalError("init(coder:) has not been implemented") + } + + init(frame: CGRect, url: URL) { + super.init(frame: frame) + + // Show progress indicator until video has loaded + addSubview(loadingIndicator) + loadingIndicator.translatesAutoresizingMaskIntoConstraints = false + loadingIndicator.centerXAnchor.constraint(equalTo: centerXAnchor).isActive = true + loadingIndicator.centerYAnchor.constraint(equalTo: centerYAnchor).isActive = true + loadingIndicator.startAnimating() + + // Load the resource + let asset = AVAsset(url: url) + let item = AVPlayerItem(asset: asset) + + // Setup the player + player.isMuted = true + player.allowsExternalPlayback = false + playerLayer.player = player + playerLayer.videoGravity = .resizeAspect + layer.addSublayer(playerLayer) + + // Manually pause video playback when app goes to background + // This fixes an issue where the playerLayer will become blank + let notificationCenter = NotificationCenter.default + notificationCenter.addObserver( + self, + selector: #selector(appMovedToBackground), + name: UIApplication.willResignActiveNotification, + object: nil + ) + notificationCenter.addObserver( + self, + selector: #selector(appMovedToForeground), + name: UIApplication.didBecomeActiveNotification, + object: nil + ) + + // Create a new player looper with the queue player and template item + self.playerLooper = AVPlayerLooper(player: self.player, templateItem: item) + + // Start the video + self.player.play() + } + + override func layoutSubviews() { + super.layoutSubviews() + playerLayer.frame = bounds + loadingIndicator.frame = bounds + } + + @objc func appMovedToBackground() { + player.pause() + } + + @objc func appMovedToForeground() { + player.seek(to: .zero) + player.play() + } + } +} diff --git a/NineAnimator/Views/Image Search Scene/ImageSearchResultsView.swift b/NineAnimator/Views/Image Search Scene/ImageSearchResultsView.swift new file mode 100644 index 00000000..62aa76e6 --- /dev/null +++ b/NineAnimator/Views/Image Search Scene/ImageSearchResultsView.swift @@ -0,0 +1,215 @@ +// +// This file is part of the NineAnimator project. +// +// Copyright © 2018-2020 Marcus Zhou. All rights reserved. +// +// NineAnimator is free software: you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation, either version 3 of the License, or +// (at your option) any later version. +// +// NineAnimator is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with NineAnimator. If not, see . +// + +import Kingfisher +import NineAnimatorCommon +import NineAnimatorNativeListServices +import SwiftUI + +@available(iOS 14.0, *) +struct ImageSearchResultsView: View { + @Binding var searchResults: [TraceMoe.TraceMoeSearchResult] + + var body: some View { + ScrollView { + AdaptiveStack(horizontalAlignment: .leading, verticalAlignment: .top, verticalSpacing: 15) { + TopResultView(result: searchResults[0]) + // Assigning manual id to ensure smooth transitions when + // adaptive stack switches between HStack and VStack + .id("FIRST RESULT") + if searchResults.count > 1 { + VStack(alignment: .leading) { + Text("Similar Results") + .font(.title2) + .fontWeight(.bold) + ForEach(searchResults[1...], id: \.video) { searchResult in + SimilarAnimeView(result: searchResult) + } + } + // Assigning manual id to ensure smooth transitions when + // adaptive stack switches between HStack and VStack + .id("SIMILAR RESULTS") + } + } + .padding([.bottom, .horizontal]) + } + .navigationTitle("Search Results") + // Fixes SwiftUI Bug With Navigation bar and ScrollView Interactions + // https://stackoverflow.com/a/64281045 + .padding(.top, 1) + } +} + +@available(iOS 14.0, *) +private extension ImageSearchResultsView { + struct TopResultView: View { + let result: TraceMoe.TraceMoeSearchResult + @State private var imageRequestTask: NineAnimatorAsyncTask? + @State private var imageURL: URL? + @State private var animeSynopsis: String? + + var body: some View { + VStack(alignment: .leading) { + LoopingVideoPlayer(videoURL: result.video) + .cornerRadius(8) + .aspectRatio(16/9, contentMode: .fit) + .frame(maxWidth: .infinity) + Button { openAnilistView() } label: { + HStack(alignment: .top) { + if let coverURL = imageURL { + KFImage(coverURL) + .fade(duration: 0.5) + .resizable() + .frame(width: 100, height: 150) + .cornerRadius(8) + } else { + ProgressView() + .frame(width: 100, height: 150, alignment: .center) + .foregroundColor(.clear) + } + VStack(alignment: .leading) { + AnimeTitlesView(result: result) + AnimeDetailedInfoView(result: result, synopsis: $animeSynopsis) + } + } + } + // Prevent button from changing text colour to accent colour + .foregroundColor(.primary) + } + .onLoad { + retrieveAnilistInfo() + } + } + + func retrieveAnilistInfo() { + let anilist = NineAnimator.default.service(type: Anilist.self) + let listingReference = ListingAnimeReference( + anilist, + name: result.anilist.title.romaji ?? "Unknown Title", + identifier: String(result.anilist.id) + ) + imageRequestTask = anilist.listingAnime(from: listingReference) + .error { _ in imageURL = NineAnimator.placeholderArtworkUrl } + .finally { + animeInfo in + imageURL = animeInfo.artwork + animeSynopsis = animeInfo.description + } + } + + func openAnilistView() { + let anilist = NineAnimator.default.service(type: Anilist.self) + let listingReference = ListingAnimeReference( + anilist, + name: result.anilist.title.romaji ?? "Unknown Title", + identifier: String(result.anilist.id), + artwork: imageURL + ) + + let link = AnyLink.listingReference(listingReference) + RootViewController.shared?.open(immedietly: link, method: .inCurrentlyDisplayedTab) + } + } + + struct AnimeDetailedInfoView: View { + let result: TraceMoe.TraceMoeSearchResult + @Binding var synopsis: String? + + var body: some View { + VStack(alignment: .leading) { + if let ep = result.episode?.value, + !ep.isEmpty { + Text("Episode: \(ep)") + .font(.subheadline) + .fontWeight(.light) + } + if let synopsis = synopsis { + Text(synopsis) + .font(.subheadline) + .fontWeight(.light) + .lineLimit(2) + } + Text("Similarity: \(Int(result.similarity * 100))%") + .font(.subheadline) + .fontWeight(.medium) + } + } + } + + struct SimilarAnimeView: View { + let result: TraceMoe.TraceMoeSearchResult + + var body: some View { + Button { openAnilistView() } label: { + HStack { + VStack(alignment: .leading) { + Text(result.anilist.title.english ?? "No English Title") + .font(.title3) + .lineLimit(2) + AnimeDetailedInfoView(result: result, synopsis: .constant(nil)) + } + .frame(maxWidth: .infinity, alignment: .leading) + LoopingVideoPlayer(videoURL: result.video) + .frame(maxWidth: .infinity, alignment: .trailing) + } + } + // Prevent button from changing text colour to accent colour + .foregroundColor(.primary) + } + + func openAnilistView() { + let anilist = NineAnimator.default.service(type: Anilist.self) + let listingReference = ListingAnimeReference( + anilist, + name: result.anilist.title.romaji ?? "Unknown Title", + identifier: String(result.anilist.id) + ) + + let link = AnyLink.listingReference(listingReference) + RootViewController.shared?.open(immedietly: link, method: .inCurrentlyDisplayedTab) + } + } + + struct AnimeTitlesView: View { + let result: TraceMoe.TraceMoeSearchResult + + var body: some View { + VStack(alignment: .leading) { + Text(result.anilist.title.english ?? "No English Title") + .font(.title) + .lineLimit(2) + .padding(0) + if let nativeTitle = result.anilist.title.native { + Text(nativeTitle) + .font(.subheadline) + .fontWeight(.light) + .lineLimit(1) + .foregroundColor(.gray) + } + if let romaji = result.anilist.title.romaji { + Text(romaji) + .font(.subheadline) + .fontWeight(.light) + .lineLimit(1) + .foregroundColor(.gray) + } + } + } + } +} diff --git a/NineAnimator/Views/Image Search Scene/ImageSearchSelectorView+PhotoPicker.swift b/NineAnimator/Views/Image Search Scene/ImageSearchSelectorView+PhotoPicker.swift new file mode 100644 index 00000000..e15c46d7 --- /dev/null +++ b/NineAnimator/Views/Image Search Scene/ImageSearchSelectorView+PhotoPicker.swift @@ -0,0 +1,86 @@ +// +// This file is part of the NineAnimator project. +// +// Copyright © 2018-2020 Marcus Zhou. All rights reserved. +// +// NineAnimator is free software: you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation, either version 3 of the License, or +// (at your option) any later version. +// +// NineAnimator is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with NineAnimator. If not, see . +// + +import NineAnimatorCommon +import PhotosUI +import SwiftUI + +@available(iOS 14.0, *) +extension ImageSearchSelectorView { + struct PhotoPicker: UIViewControllerRepresentable { + @Binding var isPresented: Bool + @Binding var selectedImageState: ImageSearchSelectorView.ImageUploadState + @Binding var selectedImage: ImageSearchSelectorView.SelectedImage + + func makeUIViewController(context: Context) -> PHPickerViewController { + var configuration = PHPickerConfiguration() + configuration.filter = .images + configuration.preferredAssetRepresentationMode = .compatible + let controller = PHPickerViewController(configuration: configuration) + controller.delegate = context.coordinator + return controller + } + + func updateUIViewController(_ uiViewController: PHPickerViewController, context: Context) { } + + func makeCoordinator() -> Coordinator { + Coordinator(self) + } + + class Coordinator: PHPickerViewControllerDelegate { + private let parent: PhotoPicker + + init(_ parent: PhotoPicker) { + self.parent = parent + } + + func picker(_ picker: PHPickerViewController, didFinishPicking results: [PHPickerResult]) { + self.parent.isPresented = false + + guard let item = results.first?.itemProvider else { return } + + guard item.canLoadObject(ofClass: UIImage.self) else { + Log.error("[PhotoPickerController] Cannot load imported image: %@", item) + return self.parent.selectedImageState = .errored( + error: NineAnimatorError.unknownError("Unable to import this image: \(item)") + ) + } + + self.parent.selectedImageState = .downloading + item.loadObject(ofClass: UIImage.self) { + image, error in + + guard error == nil else { + Log.error(error) + return self.parent.selectedImageState = .errored(error: error!) + } + guard let image = image as? UIImage else { + Log.error("[PhotoPickerController] Could not convert item to image") + return self.parent.selectedImageState = .errored( + error: NineAnimatorError.unknownError("Could not convert item to image") + ) + } + + self.parent.selectedImage = .localImage(image) + self.parent.selectedImageState = .completed + } + } + } + } +} diff --git a/NineAnimator/Views/Image Search Scene/ImageSearchSelectorView.swift b/NineAnimator/Views/Image Search Scene/ImageSearchSelectorView.swift new file mode 100644 index 00000000..88d0d5bc --- /dev/null +++ b/NineAnimator/Views/Image Search Scene/ImageSearchSelectorView.swift @@ -0,0 +1,279 @@ +// +// This file is part of the NineAnimator project. +// +// Copyright © 2018-2020 Marcus Zhou. All rights reserved. +// +// NineAnimator is free software: you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation, either version 3 of the License, or +// (at your option) any later version. +// +// NineAnimator is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +// GNU General Public License for more details. +// +// You should have received a copy of the GNU General Public License +// along with NineAnimator. If not, see . +// + +import Kingfisher +import NineAnimatorCommon +import NineAnimatorNativeListServices +import SwiftUI + +@available(iOS 14.0, *) +struct ImageSearchSelectorView: View { + private let traceMoeEngine = TraceMoe() + @State private var selectedImage: SelectedImage = .none + @State private var imageUploadState: ImageUploadState = .completed + + @State private var inputTextURL = "" + + @State private var shouldDisplayError = false + @State private var shouldDisplayPhotoPicker = false + + @State private var searchResults: [TraceMoe.TraceMoeSearchResult] = [] + @State private var shouldDisplayResultsView: Bool = false + + var body: some View { + // To-Do: Remove VStack if i plan not to use it + VStack { + Form { + ImagePreview( + selectedImage: $selectedImage, + imageState: $imageUploadState, + shouldDisplayError: $shouldDisplayError) + .scaledToFit() + .listRowInsets(EdgeInsets()) + .padding() + .frame(minWidth: 0, maxWidth: .infinity, minHeight: 300, maxHeight: 300, alignment: .center) + Section(header: Text("ENTER IMAGE LINK")) { + HStack { + TextField("Enter URL", text: $inputTextURL, onEditingChanged: { _ in }, onCommit: { loadInputURL() }) + .textFieldStyle(PlainTextFieldStyle()) + .textContentType(.URL) + .keyboardType(.URL) + .disableAutocorrection(true) + .autocapitalization(.none) + Button("Load URL") { loadInputURL() } + .buttonStyle(PlainButtonStyle()) + .foregroundColor(Color.accentColor) + } + } + Section(header: Text("UPLOAD IMAGE")) { + Button("Select Image From Library") { + shouldDisplayPhotoPicker = true + } + } + Button(action: { uploadImage() }, label: { + if case .uploading = imageUploadState { + ProgressView() + .frame(maxWidth: .infinity) + } else { + Text("UPLOAD") + .frame(maxWidth: .infinity) + } + }) + .disabled(isUploadButtonDisabled) + } + NavigationLink(destination: ImageSearchResultsView(searchResults: $searchResults), isActive: $shouldDisplayResultsView) { EmptyView() } + } + .sheet( + isPresented: $shouldDisplayPhotoPicker, + onDismiss: { + // Displacing error alert after + // photo picker is dismissed to + // avoid weird animation + if case .errored = imageUploadState { + shouldDisplayError = true + } + }, content: { + PhotoPicker( + isPresented: $shouldDisplayPhotoPicker, + selectedImageState: $imageUploadState, + selectedImage: $selectedImage + ) + } + ) + .alert(isPresented: $shouldDisplayError) { + generateErrorAlert() + } + .navigationTitle("Image Search") + } + + private var isUploadButtonDisabled: Bool { + if selectedImage != .none, + case .completed = imageUploadState { + return false + } else { return true } + } + + private var uploadButtonColor: Color { + isUploadButtonDisabled ? .gray : .accentColor + } +} + +// View methods +@available(iOS 14.0, *) +private extension ImageSearchSelectorView { + func loadInputURL() { + // Dismiss keyboard + UIApplication.shared.sendAction(#selector(UIResponder.resignFirstResponder), to: nil, from: nil, for: nil) + guard !inputTextURL.isEmpty else { return } + // Add https:// prefix for the lazy humans + var selectedText = inputTextURL + if !selectedText.hasPrefix("https://") { + selectedText = "https://\(selectedText)" + } + guard let URL = URL(string: selectedText) else { + imageUploadState = .errored(error: NineAnimatorError.urlError) + selectedImage = .none + return + } + imageUploadState = .downloading + selectedImage = .remoteURL(URL) + } + + func uploadImage() { + func handleError(_ error: Error) { + Log.error("[ImageSearchSelectorView] Failed To Upload Image: %@", error) + imageUploadState = .errored(error: error) + shouldDisplayError = true + } + + func handleResult(_ results: [TraceMoe.TraceMoeSearchResult]) { + imageUploadState = .completed + searchResults = results + shouldDisplayResultsView = true + } + + switch selectedImage { + case .none: + Log.error("[ImageSearchSelectorView] Tried Uploading Without A Selected Image") + case let .localImage(localImage): + imageUploadState = .uploading(task: traceMoeEngine.search(with: localImage) + .error { handleError($0) } + .finally { handleResult($0) }) + case let .remoteURL(url): + imageUploadState = .uploading(task: traceMoeEngine.search(with: url) + .error { handleError($0) } + .finally { handleResult($0) }) + } + } + + func generateErrorAlert() -> Alert { + guard case let .errored(error) = imageUploadState else { + return Alert( + title: Text("Unknown Error"), + message: Text("Unknown Error"), + dismissButton: .default(Text("OK")) { + selectedImage = .none + imageUploadState = .completed + } + ) + } + + var errorMessage: String + if let nineError = error as? NineAnimatorError { + errorMessage = nineError.description + } else if let kingFisherError = error as? KingfisherError { + errorMessage = kingFisherError.shortenedDescription + } else { + errorMessage = error.localizedDescription + } + + return Alert( + title: Text("Error"), + message: Text(errorMessage), + dismissButton: .default(Text("OK")) { + selectedImage = .none + imageUploadState = .completed + } + ) + } +} + +// Internal State +@available(iOS 14.0, *) +extension ImageSearchSelectorView { + enum ImageUploadState { + /// The selected image has been downloaded/uploaded + case completed + + /// The selected image is downloading for previewing + case downloading + + /// The selected image is being uploaded to retrieve search results + case uploading(task: NineAnimatorAsyncTask) + + /// The selected image has failed to download/upload + case errored(error: Error) + } + + enum SelectedImage: Hashable { + /// Represents an image stored on the user's device + case localImage(UIImage) + + /// Represents a URL pointing to an image stored remotely + case remoteURL(URL) + + /// No image has been selected + case none + } +} + +@available(iOS 14.0, *) +private extension ImageSearchSelectorView { + struct ImagePreview: View { + @Binding fileprivate var selectedImage: SelectedImage + @Binding fileprivate var imageState: ImageUploadState + @Binding fileprivate var shouldDisplayError: Bool + + var body: some View { + switch selectedImage { + case .none: + // Display Default Placeholder Image + Image("NineAnimator Lists Tip") + .resizable() + case let .localImage(image): + Image(uiImage: image) + .resizable() + case let .remoteURL(url): + KFImage(url) + .fade(duration: 0.5) + .resizable() + .placeholder { + ProgressView() + .scaleEffect(2) + } + .onSuccess { _ in + // SwiftUI will force kingfisher to reload many times 🤦‍♂️ + // This guard statement will ensure internal state does not get + // updated unless image was actually being requested to download. + guard case .downloading = imageState else { return } + imageState = .completed + } + .onFailure { error in + guard case .downloading = imageState else { return } + Log.error(error) + selectedImage = .none + imageState = .errored(error: error) + shouldDisplayError = true + } + // Loading image immediately to fix bug + // https://github.com/onevcat/Kingfisher/issues/1660 + .loadImmediately() + } + } + } +} + +@available(iOS 14.0, *) +struct ImageSelector_Previews: PreviewProvider { + static var previews: some View { + NavigationView { + ImageSearchSelectorView().preferredColorScheme(.light) + } + } +} From d35ad088c2b1c59aa97a8a9a0c2ac52c3ee25311 Mon Sep 17 00:00:00 2001 From: uttiya10 <56562649+uttiya10@users.noreply.github.com> Date: Thu, 29 Jul 2021 21:48:39 -0400 Subject: [PATCH 3/6] Fix Issue where selecting a result in reverse image search would not display anime cover --- .../Anime Information Scene/InformationSceneHeadingView.swift | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/NineAnimator/Views/Anime Information Scene/InformationSceneHeadingView.swift b/NineAnimator/Views/Anime Information Scene/InformationSceneHeadingView.swift index c34d9fbe..6b05a9d1 100644 --- a/NineAnimator/Views/Anime Information Scene/InformationSceneHeadingView.swift +++ b/NineAnimator/Views/Anime Information Scene/InformationSceneHeadingView.swift @@ -70,7 +70,7 @@ class InformationSceneHeadingView: UIView, Themable { optionsButton.isEnabled = false animeArtworkImageView.alpha = 0.0 - animeArtworkImageView.kf.setImage(with: reference.artwork, progressBlock: nil) { + animeArtworkImageView.kf.setImage(with: reference.artwork) { [weak animeArtworkImageView] _ in UIView.animate(withDuration: 0.1) { animeArtworkImageView?.alpha = 1.0 @@ -87,6 +87,7 @@ class InformationSceneHeadingView: UIView, Themable { } } } + animeArtworkImageView.kf.setImage(with: animeInformation.artwork, options: [.transition(.fade(0.2))]) UIView.animate(withDuration: 0.2) { [weak self] in From d86c03ea0a4bb5baf265b108610b1aafd04976a2 Mon Sep 17 00:00:00 2001 From: uttiya10 <56562649+uttiya10@users.noreply.github.com> Date: Thu, 29 Jul 2021 21:49:37 -0400 Subject: [PATCH 4/6] Fix Issue With Trace.Moe custom Episode Type not Decoding Properly --- .../ListServices/TraceMoe/TraceMoe.swift | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/Modules/Sources/NineAnimatorNativeListServices/ListServices/TraceMoe/TraceMoe.swift b/Modules/Sources/NineAnimatorNativeListServices/ListServices/TraceMoe/TraceMoe.swift index 933f27b6..e6bb7646 100644 --- a/Modules/Sources/NineAnimatorNativeListServices/ListServices/TraceMoe/TraceMoe.swift +++ b/Modules/Sources/NineAnimatorNativeListServices/ListServices/TraceMoe/TraceMoe.swift @@ -129,7 +129,7 @@ public extension TraceMoe { public let userPreferred: String? } - /// Abstracts Trace.moe episode (Int or String) to a string value type + /// Abstracts Trace.moe's custom episode type to a string value type struct TraceMoeEpisode: Codable { public let value: String @@ -139,13 +139,14 @@ public extension TraceMoe { public init(from decoder: Decoder) throws { let container = try decoder.singleValueContainer() - // Attempt to decode from String, Int, and Double if let str = try? container.decode(String.self) { value = str } else if let int = try? container.decode(Int.self) { value = String(int) } else if let double = try? container.decode(Double.self) { value = String(double) + } else if let array = try? container.decode([Int].self) { + value = array.description } else { throw DecodingError.typeMismatch(String.self, .init(codingPath: decoder.codingPath, debugDescription: "Could not convert trace.moe episode into String")) } From b213b1bac52e5f02b328cd398e0e655cc3c9720d Mon Sep 17 00:00:00 2001 From: uttiya10 <56562649+uttiya10@users.noreply.github.com> Date: Thu, 29 Jul 2021 21:51:08 -0400 Subject: [PATCH 5/6] Basic Code Cleanup --- NineAnimator.xcodeproj/project.pbxproj | 16 ++-- .../Search Scene/SearchViewController.swift | 2 +- ...ew.swift => ImageSearchResultsScene.swift} | 85 +++++++++---------- .../ImageSearchResultsView+AVPlayer.swift | 4 +- ...w.swift => ImageSearchSelectorScene.swift} | 25 +++--- .../ImageSearchSelectorView+PhotoPicker.swift | 11 +-- 6 files changed, 70 insertions(+), 73 deletions(-) rename NineAnimator/Views/Image Search Scene/{ImageSearchResultsView.swift => ImageSearchResultsScene.swift} (83%) rename NineAnimator/Views/Image Search Scene/{ImageSearchSelectorView.swift => ImageSearchSelectorScene.swift} (92%) diff --git a/NineAnimator.xcodeproj/project.pbxproj b/NineAnimator.xcodeproj/project.pbxproj index 1afb53c4..ec55c0be 100644 --- a/NineAnimator.xcodeproj/project.pbxproj +++ b/NineAnimator.xcodeproj/project.pbxproj @@ -216,11 +216,11 @@ 52CADD6721BD86F40077BEB1 /* UITableView+DeselectRows.swift in Sources */ = {isa = PBXBuildFile; fileRef = 52CADD6621BD86F40077BEB1 /* UITableView+DeselectRows.swift */; }; BF6820E326817C93004DCDBD /* NineAnimatorCommon in Frameworks */ = {isa = PBXBuildFile; productRef = BF6820E226817C93004DCDBD /* NineAnimatorCommon */; }; BF6820E426817C93004DCDBD /* NineAnimatorCommon in Embed Frameworks */ = {isa = PBXBuildFile; productRef = BF6820E226817C93004DCDBD /* NineAnimatorCommon */; settings = {ATTRIBUTES = (CodeSignOnCopy, ); }; }; - BF8CC475268562AB00A0A485 /* ImageSearchSelectorView.swift in Sources */ = {isa = PBXBuildFile; fileRef = BF8CC474268562AB00A0A485 /* ImageSearchSelectorView.swift */; }; + BF8CC475268562AB00A0A485 /* ImageSearchSelectorScene.swift in Sources */ = {isa = PBXBuildFile; fileRef = BF8CC474268562AB00A0A485 /* ImageSearchSelectorScene.swift */; }; BF8CC4772685646100A0A485 /* ImageSearchSelectorView+PhotoPicker.swift in Sources */ = {isa = PBXBuildFile; fileRef = BF8CC4762685646100A0A485 /* ImageSearchSelectorView+PhotoPicker.swift */; }; BF9203DC25E07E6E00CB2D91 /* AudioBackgroundController.swift in Sources */ = {isa = PBXBuildFile; fileRef = BF9203DB25E07E6E00CB2D91 /* AudioBackgroundController.swift */; }; BF99DE63268698240019BFC6 /* ImageSearchResultsView+AVPlayer.swift in Sources */ = {isa = PBXBuildFile; fileRef = BF99DE62268698240019BFC6 /* ImageSearchResultsView+AVPlayer.swift */; }; - BF99DE65268698920019BFC6 /* ImageSearchResultsView.swift in Sources */ = {isa = PBXBuildFile; fileRef = BF99DE64268698920019BFC6 /* ImageSearchResultsView.swift */; }; + BF99DE65268698920019BFC6 /* ImageSearchResultsScene.swift in Sources */ = {isa = PBXBuildFile; fileRef = BF99DE64268698920019BFC6 /* ImageSearchResultsScene.swift */; }; BFF95F0A257C2DA8001F8484 /* SettingsSceneController+DocumentPicker.swift in Sources */ = {isa = PBXBuildFile; fileRef = BFF95F09257C2DA8001F8484 /* SettingsSceneController+DocumentPicker.swift */; }; /* End PBXBuildFile section */ @@ -544,11 +544,11 @@ 2CF8912121DB39A1006D4799 /* icon_rendered.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; path = icon_rendered.png; sourceTree = ""; }; 52CADD6621BD86F40077BEB1 /* UITableView+DeselectRows.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UITableView+DeselectRows.swift"; sourceTree = ""; }; 52F6A4F221DABDB00005D78D /* .swiftlint.yml */ = {isa = PBXFileReference; lastKnownFileType = text; path = .swiftlint.yml; sourceTree = SOURCE_ROOT; }; - BF8CC474268562AB00A0A485 /* ImageSearchSelectorView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ImageSearchSelectorView.swift; sourceTree = ""; }; + BF8CC474268562AB00A0A485 /* ImageSearchSelectorScene.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ImageSearchSelectorScene.swift; sourceTree = ""; }; BF8CC4762685646100A0A485 /* ImageSearchSelectorView+PhotoPicker.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "ImageSearchSelectorView+PhotoPicker.swift"; sourceTree = ""; }; BF9203DB25E07E6E00CB2D91 /* AudioBackgroundController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AudioBackgroundController.swift; sourceTree = ""; }; BF99DE62268698240019BFC6 /* ImageSearchResultsView+AVPlayer.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "ImageSearchResultsView+AVPlayer.swift"; sourceTree = ""; }; - BF99DE64268698920019BFC6 /* ImageSearchResultsView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ImageSearchResultsView.swift; sourceTree = ""; }; + BF99DE64268698920019BFC6 /* ImageSearchResultsScene.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ImageSearchResultsScene.swift; sourceTree = ""; }; BFF95F09257C2DA8001F8484 /* SettingsSceneController+DocumentPicker.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "SettingsSceneController+DocumentPicker.swift"; sourceTree = ""; }; /* End PBXFileReference section */ @@ -1273,10 +1273,10 @@ BF8CC4732685621F00A0A485 /* Image Search Scene */ = { isa = PBXGroup; children = ( - BF8CC474268562AB00A0A485 /* ImageSearchSelectorView.swift */, + BF8CC474268562AB00A0A485 /* ImageSearchSelectorScene.swift */, BF8CC4762685646100A0A485 /* ImageSearchSelectorView+PhotoPicker.swift */, BF99DE62268698240019BFC6 /* ImageSearchResultsView+AVPlayer.swift */, - BF99DE64268698920019BFC6 /* ImageSearchResultsView.swift */, + BF99DE64268698920019BFC6 /* ImageSearchResultsScene.swift */, ); path = "Image Search Scene"; sourceTree = ""; @@ -1635,7 +1635,7 @@ 2C96DAE424562A2F007A29B0 /* InformationSceneAiringEpisodeCell.swift in Sources */, 2CAA70D321CFF42A00F0D082 /* NativePlayerController.swift in Sources */, 2C368A0D2592F0580069DE91 /* CoreExportsNineAnimator+Fetch.swift in Sources */, - BF8CC475268562AB00A0A485 /* ImageSearchSelectorView.swift in Sources */, + BF8CC475268562AB00A0A485 /* ImageSearchSelectorScene.swift in Sources */, 2C9383CD21FBF442008C0D01 /* UIView+MakeThemable.swift in Sources */, 2C1C717322186EC900760A69 /* TrackingServiceTableViewController.swift in Sources */, 2C93BD43221E1FE8000411CB /* InformationSceneStatisticsTableViewCell.swift in Sources */, @@ -1664,7 +1664,7 @@ 2C85367E222EEC7400A5CFB9 /* RecentsSceneCollectionCollectionViewCell.swift in Sources */, 2CB9919421FEAC5D0055B7A7 /* DetailedEpisodeTableViewCell.swift in Sources */, 2C86C1EB225D6F9F00950ABF /* SetupWelcomeViewController.swift in Sources */, - BF99DE65268698920019BFC6 /* ImageSearchResultsView.swift in Sources */, + BF99DE65268698920019BFC6 /* ImageSearchResultsScene.swift in Sources */, 2C66779821BC2213000E5ACC /* SearchViewController.swift in Sources */, 2CB4AF96239AD1B700EE4EDA /* CachableAVAssetLoaderDelegate.swift in Sources */, 2C9C44FE21CD2B33004C8F0C /* SettingsSceneController.swift in Sources */, diff --git a/NineAnimator/Controllers/Search Scene/SearchViewController.swift b/NineAnimator/Controllers/Search Scene/SearchViewController.swift index e9754605..f847558e 100644 --- a/NineAnimator/Controllers/Search Scene/SearchViewController.swift +++ b/NineAnimator/Controllers/Search Scene/SearchViewController.swift @@ -294,7 +294,7 @@ extension SearchViewController { @available(iOS 14.0, *) @objc func onReverseImageSearchButtonTapped() { - let hostingView = UIHostingController(rootView: ImageSearchSelectorView()) + let hostingView = UIHostingController(rootView: ImageSearchSelectorScene()) // Manually setting the nav title here to fix a SwiftUI issue // where the nav title pops in without an animation hostingView.navigationItem.title = "Image Search" diff --git a/NineAnimator/Views/Image Search Scene/ImageSearchResultsView.swift b/NineAnimator/Views/Image Search Scene/ImageSearchResultsScene.swift similarity index 83% rename from NineAnimator/Views/Image Search Scene/ImageSearchResultsView.swift rename to NineAnimator/Views/Image Search Scene/ImageSearchResultsScene.swift index 62aa76e6..240d63db 100644 --- a/NineAnimator/Views/Image Search Scene/ImageSearchResultsView.swift +++ b/NineAnimator/Views/Image Search Scene/ImageSearchResultsScene.swift @@ -17,34 +17,30 @@ // along with NineAnimator. If not, see . // +import AVKit import Kingfisher import NineAnimatorCommon import NineAnimatorNativeListServices import SwiftUI @available(iOS 14.0, *) -struct ImageSearchResultsView: View { +struct ImageSearchResultsScene: View { @Binding var searchResults: [TraceMoe.TraceMoeSearchResult] var body: some View { ScrollView { AdaptiveStack(horizontalAlignment: .leading, verticalAlignment: .top, verticalSpacing: 15) { TopResultView(result: searchResults[0]) - // Assigning manual id to ensure smooth transitions when - // adaptive stack switches between HStack and VStack - .id("FIRST RESULT") if searchResults.count > 1 { VStack(alignment: .leading) { Text("Similar Results") .font(.title2) .fontWeight(.bold) - ForEach(searchResults[1...], id: \.video) { searchResult in + ForEach(searchResults[1...], id: \.video) { + searchResult in SimilarAnimeView(result: searchResult) } } - // Assigning manual id to ensure smooth transitions when - // adaptive stack switches between HStack and VStack - .id("SIMILAR RESULTS") } } .padding([.bottom, .horizontal]) @@ -57,12 +53,12 @@ struct ImageSearchResultsView: View { } @available(iOS 14.0, *) -private extension ImageSearchResultsView { +private extension ImageSearchResultsScene { struct TopResultView: View { let result: TraceMoe.TraceMoeSearchResult @State private var imageRequestTask: NineAnimatorAsyncTask? @State private var imageURL: URL? - @State private var animeSynopsis: String? + @ScaledMetric private var imageHeight: CGFloat = 150 var body: some View { VStack(alignment: .leading) { @@ -76,16 +72,18 @@ private extension ImageSearchResultsView { KFImage(coverURL) .fade(duration: 0.5) .resizable() - .frame(width: 100, height: 150) + .aspectRatio(2/3, contentMode: .fit) + .frame(height: imageHeight) .cornerRadius(8) } else { ProgressView() - .frame(width: 100, height: 150, alignment: .center) + .aspectRatio(2/3, contentMode: .fit) + .frame(height: imageHeight) .foregroundColor(.clear) } VStack(alignment: .leading) { AnimeTitlesView(result: result) - AnimeDetailedInfoView(result: result, synopsis: $animeSynopsis) + AnimeDetailedInfoView(result: result) } } } @@ -109,7 +107,6 @@ private extension ImageSearchResultsView { .finally { animeInfo in imageURL = animeInfo.artwork - animeSynopsis = animeInfo.description } } @@ -123,32 +120,10 @@ private extension ImageSearchResultsView { ) let link = AnyLink.listingReference(listingReference) - RootViewController.shared?.open(immedietly: link, method: .inCurrentlyDisplayedTab) - } - } - - struct AnimeDetailedInfoView: View { - let result: TraceMoe.TraceMoeSearchResult - @Binding var synopsis: String? - - var body: some View { - VStack(alignment: .leading) { - if let ep = result.episode?.value, - !ep.isEmpty { - Text("Episode: \(ep)") - .font(.subheadline) - .fontWeight(.light) - } - if let synopsis = synopsis { - Text(synopsis) - .font(.subheadline) - .fontWeight(.light) - .lineLimit(2) - } - Text("Similarity: \(Int(result.similarity * 100))%") - .font(.subheadline) - .fontWeight(.medium) - } + RootViewController.shared?.open( + immedietly: link, + method: .inCurrentlyDisplayedTab + ) } } @@ -159,10 +134,13 @@ private extension ImageSearchResultsView { Button { openAnilistView() } label: { HStack { VStack(alignment: .leading) { - Text(result.anilist.title.english ?? "No English Title") + Text( + result.anilist.title.english ?? + result.anilist.title.native ?? + "No Title") .font(.title3) .lineLimit(2) - AnimeDetailedInfoView(result: result, synopsis: .constant(nil)) + AnimeDetailedInfoView(result: result) } .frame(maxWidth: .infinity, alignment: .leading) LoopingVideoPlayer(videoURL: result.video) @@ -182,7 +160,10 @@ private extension ImageSearchResultsView { ) let link = AnyLink.listingReference(listingReference) - RootViewController.shared?.open(immedietly: link, method: .inCurrentlyDisplayedTab) + RootViewController.shared?.open( + immedietly: link, + method: .inCurrentlyDisplayedTab + ) } } @@ -212,4 +193,22 @@ private extension ImageSearchResultsView { } } } + + struct AnimeDetailedInfoView: View { + let result: TraceMoe.TraceMoeSearchResult + + var body: some View { + VStack(alignment: .leading) { + if let ep = result.episode?.value, + !ep.isEmpty { + Text("Episode: \(ep)") + .font(.subheadline) + .fontWeight(.light) + } + Text("Similarity: \(Int(result.similarity * 100))%") + .font(.subheadline) + .fontWeight(.medium) + } + } + } } diff --git a/NineAnimator/Views/Image Search Scene/ImageSearchResultsView+AVPlayer.swift b/NineAnimator/Views/Image Search Scene/ImageSearchResultsView+AVPlayer.swift index 76dcec74..9b02da4c 100644 --- a/NineAnimator/Views/Image Search Scene/ImageSearchResultsView+AVPlayer.swift +++ b/NineAnimator/Views/Image Search Scene/ImageSearchResultsView+AVPlayer.swift @@ -21,10 +21,10 @@ import AVKit import SwiftUI @available(iOS 14.0, *) -extension ImageSearchResultsView { +extension ImageSearchResultsScene { struct LoopingVideoPlayer: UIViewRepresentable { let videoURL: URL - + func updateUIView(_ uiView: UIView, context: UIViewRepresentableContext) { } diff --git a/NineAnimator/Views/Image Search Scene/ImageSearchSelectorView.swift b/NineAnimator/Views/Image Search Scene/ImageSearchSelectorScene.swift similarity index 92% rename from NineAnimator/Views/Image Search Scene/ImageSearchSelectorView.swift rename to NineAnimator/Views/Image Search Scene/ImageSearchSelectorScene.swift index 88d0d5bc..658b5c3f 100644 --- a/NineAnimator/Views/Image Search Scene/ImageSearchSelectorView.swift +++ b/NineAnimator/Views/Image Search Scene/ImageSearchSelectorScene.swift @@ -23,9 +23,9 @@ import NineAnimatorNativeListServices import SwiftUI @available(iOS 14.0, *) -struct ImageSearchSelectorView: View { +struct ImageSearchSelectorScene: View { private let traceMoeEngine = TraceMoe() - @State private var selectedImage: SelectedImage = .none + @State private var selectedImage: SelectedImage? @State private var imageUploadState: ImageUploadState = .completed @State private var inputTextURL = "" @@ -37,7 +37,6 @@ struct ImageSearchSelectorView: View { @State private var shouldDisplayResultsView: Bool = false var body: some View { - // To-Do: Remove VStack if i plan not to use it VStack { Form { ImagePreview( @@ -77,7 +76,7 @@ struct ImageSearchSelectorView: View { }) .disabled(isUploadButtonDisabled) } - NavigationLink(destination: ImageSearchResultsView(searchResults: $searchResults), isActive: $shouldDisplayResultsView) { EmptyView() } + NavigationLink(destination: ImageSearchResultsScene(searchResults: $searchResults), isActive: $shouldDisplayResultsView) { EmptyView() } } .sheet( isPresented: $shouldDisplayPhotoPicker, @@ -116,7 +115,7 @@ struct ImageSearchSelectorView: View { // View methods @available(iOS 14.0, *) -private extension ImageSearchSelectorView { +private extension ImageSearchSelectorScene { func loadInputURL() { // Dismiss keyboard UIApplication.shared.sendAction(#selector(UIResponder.resignFirstResponder), to: nil, from: nil, for: nil) @@ -137,7 +136,7 @@ private extension ImageSearchSelectorView { func uploadImage() { func handleError(_ error: Error) { - Log.error("[ImageSearchSelectorView] Failed To Upload Image: %@", error) + Log.error("[ImageSearchSelectorScene] Failed To Upload Image: %@", error) imageUploadState = .errored(error: error) shouldDisplayError = true } @@ -150,7 +149,7 @@ private extension ImageSearchSelectorView { switch selectedImage { case .none: - Log.error("[ImageSearchSelectorView] Tried Uploading Without A Selected Image") + Log.error("[ImageSearchSelectorScene] Tried Uploading Without A Selected Image") case let .localImage(localImage): imageUploadState = .uploading(task: traceMoeEngine.search(with: localImage) .error { handleError($0) } @@ -164,6 +163,7 @@ private extension ImageSearchSelectorView { func generateErrorAlert() -> Alert { guard case let .errored(error) = imageUploadState else { + assertionFailure("Alert displayed without any error!") return Alert( title: Text("Unknown Error"), message: Text("Unknown Error"), @@ -196,7 +196,7 @@ private extension ImageSearchSelectorView { // Internal State @available(iOS 14.0, *) -extension ImageSearchSelectorView { +extension ImageSearchSelectorScene { enum ImageUploadState { /// The selected image has been downloaded/uploaded case completed @@ -217,16 +217,13 @@ extension ImageSearchSelectorView { /// Represents a URL pointing to an image stored remotely case remoteURL(URL) - - /// No image has been selected - case none } } @available(iOS 14.0, *) -private extension ImageSearchSelectorView { +private extension ImageSearchSelectorScene { struct ImagePreview: View { - @Binding fileprivate var selectedImage: SelectedImage + @Binding fileprivate var selectedImage: SelectedImage? @Binding fileprivate var imageState: ImageUploadState @Binding fileprivate var shouldDisplayError: Bool @@ -273,7 +270,7 @@ private extension ImageSearchSelectorView { struct ImageSelector_Previews: PreviewProvider { static var previews: some View { NavigationView { - ImageSearchSelectorView().preferredColorScheme(.light) + ImageSearchSelectorScene().preferredColorScheme(.light) } } } diff --git a/NineAnimator/Views/Image Search Scene/ImageSearchSelectorView+PhotoPicker.swift b/NineAnimator/Views/Image Search Scene/ImageSearchSelectorView+PhotoPicker.swift index e15c46d7..0131fc9f 100644 --- a/NineAnimator/Views/Image Search Scene/ImageSearchSelectorView+PhotoPicker.swift +++ b/NineAnimator/Views/Image Search Scene/ImageSearchSelectorView+PhotoPicker.swift @@ -22,11 +22,11 @@ import PhotosUI import SwiftUI @available(iOS 14.0, *) -extension ImageSearchSelectorView { +extension ImageSearchSelectorScene { struct PhotoPicker: UIViewControllerRepresentable { @Binding var isPresented: Bool - @Binding var selectedImageState: ImageSearchSelectorView.ImageUploadState - @Binding var selectedImage: ImageSearchSelectorView.SelectedImage + @Binding var selectedImageState: ImageSearchSelectorScene.ImageUploadState + @Binding var selectedImage: ImageSearchSelectorScene.SelectedImage? func makeUIViewController(context: Context) -> PHPickerViewController { var configuration = PHPickerConfiguration() @@ -66,10 +66,11 @@ extension ImageSearchSelectorView { item.loadObject(ofClass: UIImage.self) { image, error in - guard error == nil else { + if let error = error { Log.error(error) - return self.parent.selectedImageState = .errored(error: error!) + return self.parent.selectedImageState = .errored(error: error) } + guard let image = image as? UIImage else { Log.error("[PhotoPickerController] Could not convert item to image") return self.parent.selectedImageState = .errored( From 53c975a10b40fbf0ccadc749f14340c8cddc8e2c Mon Sep 17 00:00:00 2001 From: uttiya10 <56562649+uttiya10@users.noreply.github.com> Date: Fri, 6 Aug 2021 20:17:46 -0400 Subject: [PATCH 6/6] Pause Video Previews when not Visible - Also has basic code refactoring --- .../Controllers/RootViewController.swift | 2 +- .../ImageSearchResultsScene.swift | 13 +++++++++---- .../ImageSearchResultsView+AVPlayer.swift | 16 +++++++++++++--- .../ImageSearchSelectorScene.swift | 12 +++++++++--- 4 files changed, 32 insertions(+), 11 deletions(-) diff --git a/NineAnimator/Controllers/RootViewController.swift b/NineAnimator/Controllers/RootViewController.swift index 3c7afcd6..bad55aaa 100644 --- a/NineAnimator/Controllers/RootViewController.swift +++ b/NineAnimator/Controllers/RootViewController.swift @@ -179,7 +179,7 @@ extension RootViewController { func open(immedietly link: AnyLink, method: LinkOpeningMethod) { guard let controllerForLink = retrieveViewController(forLink: link) else { - Log.error("[RootViewController] Failed to retrieve view controller for anylink: %@", link) + Log.error("[RootViewController] Failed to retrieve view controller to handle anylink: %@", link) return } diff --git a/NineAnimator/Views/Image Search Scene/ImageSearchResultsScene.swift b/NineAnimator/Views/Image Search Scene/ImageSearchResultsScene.swift index 240d63db..5bb3841a 100644 --- a/NineAnimator/Views/Image Search Scene/ImageSearchResultsScene.swift +++ b/NineAnimator/Views/Image Search Scene/ImageSearchResultsScene.swift @@ -58,11 +58,12 @@ private extension ImageSearchResultsScene { let result: TraceMoe.TraceMoeSearchResult @State private var imageRequestTask: NineAnimatorAsyncTask? @State private var imageURL: URL? + @State var shouldPlayPreview = true @ScaledMetric private var imageHeight: CGFloat = 150 var body: some View { VStack(alignment: .leading) { - LoopingVideoPlayer(videoURL: result.video) + LoopingVideoPlayer(videoURL: result.video, isPlaying: $shouldPlayPreview) .cornerRadius(8) .aspectRatio(16/9, contentMode: .fit) .frame(maxWidth: .infinity) @@ -93,6 +94,8 @@ private extension ImageSearchResultsScene { .onLoad { retrieveAnilistInfo() } + .onDisappear { shouldPlayPreview = false } + .onAppear { shouldPlayPreview = true } } func retrieveAnilistInfo() { @@ -119,9 +122,8 @@ private extension ImageSearchResultsScene { artwork: imageURL ) - let link = AnyLink.listingReference(listingReference) RootViewController.shared?.open( - immedietly: link, + immedietly: .listingReference(listingReference), method: .inCurrentlyDisplayedTab ) } @@ -129,6 +131,7 @@ private extension ImageSearchResultsScene { struct SimilarAnimeView: View { let result: TraceMoe.TraceMoeSearchResult + @State var shouldPlayPreview = true var body: some View { Button { openAnilistView() } label: { @@ -143,12 +146,14 @@ private extension ImageSearchResultsScene { AnimeDetailedInfoView(result: result) } .frame(maxWidth: .infinity, alignment: .leading) - LoopingVideoPlayer(videoURL: result.video) + LoopingVideoPlayer(videoURL: result.video, isPlaying: $shouldPlayPreview) .frame(maxWidth: .infinity, alignment: .trailing) } } // Prevent button from changing text colour to accent colour .foregroundColor(.primary) + .onDisappear { shouldPlayPreview = false } + .onAppear { shouldPlayPreview = true } } func openAnilistView() { diff --git a/NineAnimator/Views/Image Search Scene/ImageSearchResultsView+AVPlayer.swift b/NineAnimator/Views/Image Search Scene/ImageSearchResultsView+AVPlayer.swift index 9b02da4c..1fbc401d 100644 --- a/NineAnimator/Views/Image Search Scene/ImageSearchResultsView+AVPlayer.swift +++ b/NineAnimator/Views/Image Search Scene/ImageSearchResultsView+AVPlayer.swift @@ -24,16 +24,18 @@ import SwiftUI extension ImageSearchResultsScene { struct LoopingVideoPlayer: UIViewRepresentable { let videoURL: URL + @Binding var isPlaying: Bool - func updateUIView(_ uiView: UIView, context: UIViewRepresentableContext) { + func updateUIView(_ uiView: LoopingPlayerUIView, context: UIViewRepresentableContext) { + uiView.isPlaying = isPlaying } - func makeUIView(context: Context) -> UIView { + func makeUIView(context: Context) -> LoopingPlayerUIView { LoopingPlayerUIView(frame: .zero, url: videoURL) } } - private class LoopingPlayerUIView: UIView { + class LoopingPlayerUIView: UIView { private let playerLayer = AVPlayerLayer() private let loadingIndicator = UIActivityIndicatorView() private var playerLooper: AVPlayerLooper? @@ -87,6 +89,14 @@ extension ImageSearchResultsScene { // Start the video self.player.play() } + + var isPlaying: Bool { + get { player.rate == 1 } + set { + let playRate: Float = newValue ? 1 : 0 + player.rate = playRate + } + } override func layoutSubviews() { super.layoutSubviews() diff --git a/NineAnimator/Views/Image Search Scene/ImageSearchSelectorScene.swift b/NineAnimator/Views/Image Search Scene/ImageSearchSelectorScene.swift index 658b5c3f..5f061ba9 100644 --- a/NineAnimator/Views/Image Search Scene/ImageSearchSelectorScene.swift +++ b/NineAnimator/Views/Image Search Scene/ImageSearchSelectorScene.swift @@ -49,7 +49,11 @@ struct ImageSearchSelectorScene: View { .frame(minWidth: 0, maxWidth: .infinity, minHeight: 300, maxHeight: 300, alignment: .center) Section(header: Text("ENTER IMAGE LINK")) { HStack { - TextField("Enter URL", text: $inputTextURL, onEditingChanged: { _ in }, onCommit: { loadInputURL() }) + TextField( + "Enter URL", + text: $inputTextURL, + onEditingChanged: { _ in }, + onCommit: { loadInputURL() }) .textFieldStyle(PlainTextFieldStyle()) .textContentType(.URL) .keyboardType(.URL) @@ -57,7 +61,7 @@ struct ImageSearchSelectorScene: View { .autocapitalization(.none) Button("Load URL") { loadInputURL() } .buttonStyle(PlainButtonStyle()) - .foregroundColor(Color.accentColor) + .foregroundColor(.accentColor) } } Section(header: Text("UPLOAD IMAGE")) { @@ -76,7 +80,9 @@ struct ImageSearchSelectorScene: View { }) .disabled(isUploadButtonDisabled) } - NavigationLink(destination: ImageSearchResultsScene(searchResults: $searchResults), isActive: $shouldDisplayResultsView) { EmptyView() } + NavigationLink( + destination: ImageSearchResultsScene(searchResults: $searchResults), + isActive: $shouldDisplayResultsView) { EmptyView() } } .sheet( isPresented: $shouldDisplayPhotoPicker,