diff --git a/Configuration/Version.xcconfig b/Configuration/Version.xcconfig index bd92fe0db4..657865d520 100644 --- a/Configuration/Version.xcconfig +++ b/Configuration/Version.xcconfig @@ -1 +1 @@ -MARKETING_VERSION = 7.126.0 +MARKETING_VERSION = 7.127.0 diff --git a/Core/AppPrivacyConfigurationDataProvider.swift b/Core/AppPrivacyConfigurationDataProvider.swift index 4961a858f8..5f5446df62 100644 --- a/Core/AppPrivacyConfigurationDataProvider.swift +++ b/Core/AppPrivacyConfigurationDataProvider.swift @@ -23,8 +23,8 @@ import BrowserServicesKit final public class AppPrivacyConfigurationDataProvider: EmbeddedDataProvider { public struct Constants { - public static let embeddedDataETag = "\"6e864c9ba3042d9fd90f2e4f9713ef06\"" - public static let embeddedDataSHA = "59fc3aa71802c502fbe5042d03da8c1007f4f68014405386fc4c2fa267729319" + public static let embeddedDataETag = "\"8d80c53fc5696854e2ef43bcb28089a1\"" + public static let embeddedDataSHA = "cdc42bdffa7ab5478ca80febf325cf689148cd19a68153e0a2cafed888b1d1ce" } public var embeddedDataEtag: String { diff --git a/Core/BookmarksStateRepair.swift b/Core/BookmarksStateRepair.swift new file mode 100644 index 0000000000..0c5e09e96e --- /dev/null +++ b/Core/BookmarksStateRepair.swift @@ -0,0 +1,83 @@ +// +// BookmarksStateRepair.swift +// DuckDuckGo +// +// Copyright © 2024 DuckDuckGo. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +import Foundation +import CoreData +import Bookmarks +import Persistence + +public class BookmarksStateRepair { + + enum Constants { + static let pendingDeletionRepaired = "stateRepair_pendingDeletionRepaired" + } + + public enum RepairStatus: Equatable { + case alreadyPerformed + case noBrokenData + case dataRepaired + case repairError(Error) + + public static func == (lhs: BookmarksStateRepair.RepairStatus, rhs: BookmarksStateRepair.RepairStatus) -> Bool { + switch (lhs, rhs) { + case (.alreadyPerformed, .alreadyPerformed), (.noBrokenData, .noBrokenData), (.dataRepaired, .dataRepaired), (.repairError, .repairError): + return true + default: + return false + } + } + } + + let keyValueStore: KeyValueStoring + + public init(keyValueStore: KeyValueStoring) { + self.keyValueStore = keyValueStore + } + + public func validateAndRepairPendingDeletionState(in context: NSManagedObjectContext) -> RepairStatus { + + guard keyValueStore.object(forKey: Constants.pendingDeletionRepaired) == nil else { + return .alreadyPerformed + } + + do { + let fr = BookmarkEntity.fetchRequest() + fr.predicate = NSPredicate(format: "%K == nil", #keyPath(BookmarkEntity.isPendingDeletion)) + + let result = try context.fetch(fr) + + if !result.isEmpty { + for obj in result { + obj.setValue(false, forKey: #keyPath(BookmarkEntity.isPendingDeletion)) + } + + try context.save() + + keyValueStore.set(true, forKey: Constants.pendingDeletionRepaired) + return .dataRepaired + } else { + keyValueStore.set(true, forKey: Constants.pendingDeletionRepaired) + return .noBrokenData + } + } catch { + return .repairError(error) + } + } + +} diff --git a/Core/PixelEvent.swift b/Core/PixelEvent.swift index 450d4f39bd..d8a87f3132 100644 --- a/Core/PixelEvent.swift +++ b/Core/PixelEvent.swift @@ -313,6 +313,7 @@ extension Pixel { case networkProtectionControllerStartFailure case networkProtectionTunnelStartAttempt + case networkProtectionTunnelStartAttemptOnDemandWithoutAccessToken case networkProtectionTunnelStartSuccess case networkProtectionTunnelStartFailure @@ -338,7 +339,7 @@ extension Pixel { case networkProtectionTunnelFailureDetected case networkProtectionTunnelFailureRecovered - + case networkProtectionLatency(quality: NetworkProtectionLatencyMonitor.ConnectionQuality) case networkProtectionLatencyError @@ -415,6 +416,9 @@ extension Pixel { case networkProtectionDNSUpdateCustom case networkProtectionDNSUpdateDefault + case networkProtectionVPNConfigurationRemoved + case networkProtectionVPNConfigurationRemovalFailed + // MARK: remote messaging pixels case remoteMessageShown @@ -504,7 +508,10 @@ extension Pixel { case debugBookmarksStructureLost case debugBookmarksInvalidRoots case debugBookmarksValidationFailed - + + case debugBookmarksPendingDeletionFixed + case debugBookmarksPendingDeletionRepairError + case debugCannotClearObservationsDatabase case debugWebsiteDataStoresNotClearedMultiple case debugWebsiteDataStoresNotClearedOne @@ -643,7 +650,6 @@ extension Pixel { case privacyProRestoreAfterPurchaseAttempt case privacyProSubscriptionActivated case privacyProWelcomeAddDevice - case privacyProSettingsAddDevice case privacyProAddDeviceEnterEmail case privacyProWelcomeVPN case privacyProWelcomePersonalInformationRemoval @@ -1020,6 +1026,7 @@ extension Pixel.Event { case .networkProtectionControllerStartSuccess: return "m_netp_controller_start_success" case .networkProtectionControllerStartFailure: return "m_netp_controller_start_failure" case .networkProtectionTunnelStartAttempt: return "m_netp_tunnel_start_attempt" + case .networkProtectionTunnelStartAttemptOnDemandWithoutAccessToken: return "m_netp_tunnel_start_attempt_on_demand_without_access_token" case .networkProtectionTunnelStartSuccess: return "m_netp_tunnel_start_success" case .networkProtectionTunnelStartFailure: return "m_netp_tunnel_start_failure" case .networkProtectionTunnelStopAttempt: return "m_netp_tunnel_stop_attempt" @@ -1098,6 +1105,9 @@ extension Pixel.Event { case .networkProtectionDNSUpdateCustom: return "m_netp_ev_update_dns_custom" case .networkProtectionDNSUpdateDefault: return "m_netp_ev_update_dns_default" + case .networkProtectionVPNConfigurationRemoved: return "m_netp_vpn_configuration_removed" + case .networkProtectionVPNConfigurationRemovalFailed: return "m_netp_vpn_configuration_removal_failed" + // MARK: remote messaging pixels case .remoteMessageShown: return "m_remote_message_shown" @@ -1183,6 +1193,9 @@ extension Pixel.Event { case .debugBookmarksInvalidRoots: return "m_d_bookmarks_invalid_roots" case .debugBookmarksValidationFailed: return "m_d_bookmarks_validation_failed" + case .debugBookmarksPendingDeletionFixed: return "m_debug_bookmarks_pending_deletion_fixed" + case .debugBookmarksPendingDeletionRepairError: return "m_debug_bookmarks_pending_deletion_repair_error" + case .debugCannotClearObservationsDatabase: return "m_d_cannot_clear_observations_database" case .debugWebsiteDataStoresNotClearedMultiple: return "m_d_wkwebsitedatastoresnotcleared_multiple" case .debugWebsiteDataStoresNotClearedOne: return "m_d_wkwebsitedatastoresnotcleared_one" @@ -1340,7 +1353,6 @@ extension Pixel.Event { case .privacyProRestoreAfterPurchaseAttempt: return "m_privacy-pro_app_subscription-restore-after-purchase-attempt_success" case .privacyProSubscriptionActivated: return "m_privacy-pro_app_subscription_activated_u" case .privacyProWelcomeAddDevice: return "m_privacy-pro_welcome_add-device_click_u" - case .privacyProSettingsAddDevice: return "m_privacy-pro_settings_add-device_click" case .privacyProAddDeviceEnterEmail: return "m_privacy-pro_add-device_enter-email_click" case .privacyProWelcomeVPN: return "m_privacy-pro_welcome_vpn_click_u" case .privacyProWelcomePersonalInformationRemoval: return "m_privacy-pro_welcome_personal-information-removal_click_u" diff --git a/Core/ios-config.json b/Core/ios-config.json index 95bd0f251d..5d30590562 100644 --- a/Core/ios-config.json +++ b/Core/ios-config.json @@ -1,6 +1,6 @@ { "readme": "https://github.com/duckduckgo/privacy-configuration", - "version": 1719225510541, + "version": 1719853254811, "features": { "adClickAttribution": { "readme": "https://help.duckduckgo.com/duckduckgo-help-pages/privacy/web-tracking-protections/#3rd-party-tracker-loading-protection", @@ -300,6 +300,9 @@ { "domain": "malmostadsteater.se" }, + { + "domain": "hertz.co.uk" + }, { "domain": "hertz.com" }, @@ -353,7 +356,7 @@ } } }, - "hash": "63f99d701b33267410e99d637e3f7778" + "hash": "c47027bc9d235b909d868f5ffa97d3c9" }, "autofill": { "exceptions": [ @@ -1010,6 +1013,72 @@ { "domain": "pocketbook.digital" }, + { + "domain": "vinted.at" + }, + { + "domain": "vinted.be" + }, + { + "domain": "vinted.com" + }, + { + "domain": "vinted.co.uk" + }, + { + "domain": "vinted.cz" + }, + { + "domain": "vinted.de" + }, + { + "domain": "vinted.dk" + }, + { + "domain": "vinted.es" + }, + { + "domain": "vinted.fi" + }, + { + "domain": "vinted.fr" + }, + { + "domain": "vinted.gr" + }, + { + "domain": "vinted.hr" + }, + { + "domain": "vinted.hu" + }, + { + "domain": "vinted.it" + }, + { + "domain": "vinted.lt" + }, + { + "domain": "vinted.lu" + }, + { + "domain": "vinted.nl" + }, + { + "domain": "vinted.pl" + }, + { + "domain": "vinted.pt" + }, + { + "domain": "vinted.ro" + }, + { + "domain": "vinted.se" + }, + { + "domain": "vinted.sk" + }, { "domain": "marvel.com" }, @@ -1035,7 +1104,7 @@ } }, "state": "disabled", - "hash": "4104bd45b436c084ff8fd0297883d40e" + "hash": "1d9a8170b92ba9902d28fad82713a403" }, "clickToPlay": { "exceptions": [ @@ -1290,11 +1359,6 @@ "state": "disabled", "hash": "314df3f87ca501a86eb1d04db8bf7bbc" }, - "dummyWebMessageListener": { - "exceptions": [], - "state": "disabled", - "hash": "728493ef7a1488e4781656d3f9db84aa" - }, "elementHiding": { "exceptions": [ { @@ -2455,6 +2519,15 @@ } ] }, + { + "domain": "eporner.com", + "rules": [ + { + "selector": "#admobiletop", + "type": "hide" + } + ] + }, { "domain": "essentiallysports.com", "rules": [ @@ -4486,7 +4559,7 @@ ] }, "state": "enabled", - "hash": "85bebc8a4c01285c2cfb97032a0610bb" + "hash": "573f3bc0a0f15e5dc7d561034e7a082d" }, "exceptionHandler": { "exceptions": [ @@ -5239,6 +5312,11 @@ "state": "disabled", "hash": "3c850040d1c9b3e06841b1491c5d2940" }, + "remoteMessaging": { + "state": "enabled", + "exceptions": [], + "hash": "697382e31649d84b01166f1dc6f790d6" + }, "requestFilterer": { "state": "disabled", "exceptions": [ @@ -5313,6 +5391,18 @@ "minSupportedVersion": "7.104.0", "hash": "d7dca6ee484eadebb5133e3f15fd9f41" }, + "toggleReports": { + "state": "enabled", + "exceptions": [], + "settings": { + "dismissLogicEnabled": true, + "dismissInterval": 172800, + "promptLimitLogicEnabled": true, + "promptInterval": 172800, + "maxPromptCount": 3 + }, + "hash": "a05b4413a3745762c46ec89aa5037693" + }, "trackerAllowlist": { "state": "enabled", "settings": { @@ -8746,6 +8836,16 @@ }, "hash": "25d935f0276cd0d81fc6f25811f7cb36" }, + "webViewBlobDownload": { + "exceptions": [], + "state": "disabled", + "hash": "728493ef7a1488e4781656d3f9db84aa" + }, + "windowsExternalPreviewReleases": { + "exceptions": [], + "state": "disabled", + "hash": "728493ef7a1488e4781656d3f9db84aa" + }, "windowsPermissionUsage": { "exceptions": [], "state": "disabled", diff --git a/DuckDuckGo.xcodeproj/project.pbxproj b/DuckDuckGo.xcodeproj/project.pbxproj index 4cac42c404..530e6d59b6 100644 --- a/DuckDuckGo.xcodeproj/project.pbxproj +++ b/DuckDuckGo.xcodeproj/project.pbxproj @@ -172,6 +172,9 @@ 31CB4251273AF50700FA0F3F /* SpeechRecognizerProtocol.swift in Sources */ = {isa = PBXBuildFile; fileRef = 31CB4250273AF50700FA0F3F /* SpeechRecognizerProtocol.swift */; }; 31CC224928369B38001654A4 /* AutofillLoginSettingsListViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 31CC224828369B38001654A4 /* AutofillLoginSettingsListViewController.swift */; }; 31DD208427395A5A008FB313 /* VoiceSearchHelper.swift in Sources */ = {isa = PBXBuildFile; fileRef = 31DD208327395A5A008FB313 /* VoiceSearchHelper.swift */; }; + 31DE43C22C2C480D00F8C51F /* DuckPlayerFeaturePresentationView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 31DE43C12C2C480D00F8C51F /* DuckPlayerFeaturePresentationView.swift */; }; + 31DE43C42C2C60E800F8C51F /* DuckPlayerModalPresenter.swift in Sources */ = {isa = PBXBuildFile; fileRef = 31DE43C32C2C60E800F8C51F /* DuckPlayerModalPresenter.swift */; }; + 31DE43C62C2DA70A00F8C51F /* DuckPlayer-ModalAnimation.json in Resources */ = {isa = PBXBuildFile; fileRef = 31DE43C52C2DA70A00F8C51F /* DuckPlayer-ModalAnimation.json */; }; 31E69A63280F4CB600478327 /* DuckUI in Frameworks */ = {isa = PBXBuildFile; productRef = 31E69A62280F4CB600478327 /* DuckUI */; }; 31EF52E1281B3BDC0034796E /* AutofillLoginListItemViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = 31EF52E0281B3BDC0034796E /* AutofillLoginListItemViewModel.swift */; }; 3712091E2C21E390003ADF3D /* RemoteMessagingStoreErrorHandling.swift in Sources */ = {isa = PBXBuildFile; fileRef = 3712091D2C21E390003ADF3D /* RemoteMessagingStoreErrorHandling.swift */; }; @@ -247,10 +250,16 @@ 56D8556C2BEA91C4009F9698 /* SyncAlertsPresenting.swift in Sources */ = {isa = PBXBuildFile; fileRef = 56D8556B2BEA91C4009F9698 /* SyncAlertsPresenting.swift */; }; 6AC6DAB328804F97002723C0 /* BarsAnimator.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6AC6DAB228804F97002723C0 /* BarsAnimator.swift */; }; 6AC98419288055C1005FA9CA /* BarsAnimatorTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6AC98418288055C1005FA9CA /* BarsAnimatorTests.swift */; }; + 6F5CC0812C2AFFE400AFC840 /* ToggleExpandButtonView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6F5CC0802C2AFFE400AFC840 /* ToggleExpandButtonView.swift */; }; 6F655BE22BAB289E00AC3597 /* DefaultTheme.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6F655BE12BAB289E00AC3597 /* DefaultTheme.swift */; }; 6F8496412BC3D8EE00ADA54E /* OnboardingButtonsView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6F8496402BC3D8EE00ADA54E /* OnboardingButtonsView.swift */; }; + 6F96FF102C2B128500162692 /* NewTabPageCustomizeButtonView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6F96FF0F2C2B128500162692 /* NewTabPageCustomizeButtonView.swift */; }; 6FB1FE9E2C24D41D0075B68B /* NewTabPageSectionsDebugView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FB1FE9D2C24D41D0075B68B /* NewTabPageSectionsDebugView.swift */; }; 6FB1FEA22C256ACD0075B68B /* NewTabPageManager.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FB1FEA12C256ACD0075B68B /* NewTabPageManager.swift */; }; + 6FB2A67A2C2C5BAE004D20C8 /* FavoriteEmptyStateItem.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FB2A6792C2C5BAE004D20C8 /* FavoriteEmptyStateItem.swift */; }; + 6FB2A67C2C2D9DF0004D20C8 /* FavoritesEmptyStateView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FB2A67B2C2D9DF0004D20C8 /* FavoritesEmptyStateView.swift */; }; + 6FB2A67E2C2DAFB4004D20C8 /* NewTabPageGridView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FB2A67D2C2DAFB4004D20C8 /* NewTabPageGridView.swift */; }; + 6FB2A6802C2EA950004D20C8 /* FavoritesModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FB2A67F2C2EA950004D20C8 /* FavoritesModel.swift */; }; 6FBF0F8B2BD7C0A900136CF0 /* AllProtectedCell.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FBF0F8A2BD7C0A900136CF0 /* AllProtectedCell.swift */; }; 6FD1BAE42B87A107000C475C /* AdAttributionPixelReporter.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FD1BAE12B87A107000C475C /* AdAttributionPixelReporter.swift */; }; 6FD1BAE52B87A107000C475C /* AdAttributionReporterStorage.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FD1BAE22B87A107000C475C /* AdAttributionReporterStorage.swift */; }; @@ -258,10 +267,15 @@ 6FD3AEE32B8F4EEB0060FCCC /* AdAttributionPixelReporterTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FD3AEE12B8DFBB80060FCCC /* AdAttributionPixelReporterTests.swift */; }; 6FDA1FB32B59584400AC962A /* AddressDisplayHelper.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FDA1FB22B59584400AC962A /* AddressDisplayHelper.swift */; }; 6FDB3F192BD11A4400F7A307 /* AutocompleteSuggestionsModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FDB3F182BD11A4400F7A307 /* AutocompleteSuggestionsModel.swift */; }; + 6FE018402C25CB3F001F680D /* FavoritesSectionHeader.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FE0183F2C25CB3F001F680D /* FavoritesSectionHeader.swift */; }; 6FE095D82BD90AFB00490FF8 /* UniversalOmniBarState.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FE095D72BD90AFB00490FF8 /* UniversalOmniBarState.swift */; }; 6FE127382C20492500EB5724 /* NewTabPage.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FE127372C20492500EB5724 /* NewTabPage.swift */; }; 6FE1273A2C204BD000EB5724 /* NewTabPageView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FE127392C204BD000EB5724 /* NewTabPageView.swift */; }; + 6FE1273D2C204C2500EB5724 /* FavoritesView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FE1273C2C204C2500EB5724 /* FavoritesView.swift */; }; + 6FE127402C204D9B00EB5724 /* ShortcutsView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FE1273F2C204D9B00EB5724 /* ShortcutsView.swift */; }; + 6FE127432C204DF700EB5724 /* FavoriteItemView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FE127422C204DF700EB5724 /* FavoriteItemView.swift */; }; 6FE127462C2054A900EB5724 /* NewTabPageViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FE127452C2054A900EB5724 /* NewTabPageViewController.swift */; }; + 6FE1274B2C20943500EB5724 /* ShortcutItemView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FE1274A2C20943500EB5724 /* ShortcutItemView.swift */; }; 6FF915822B88E07A0042AC87 /* AdAttributionFetcherTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6FF915802B88E0750042AC87 /* AdAttributionFetcherTests.swift */; }; 7BC571202BDBB877003B0CCE /* VPNActivationDateStore.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7BC5711F2BDBB877003B0CCE /* VPNActivationDateStore.swift */; }; 7BC571212BDBB977003B0CCE /* VPNActivationDateStore.swift in Sources */ = {isa = PBXBuildFile; fileRef = 7BC5711F2BDBB877003B0CCE /* VPNActivationDateStore.swift */; }; @@ -499,6 +513,8 @@ 9833913727AC400800DAF119 /* AppTrackerDataSetProvider.swift in Sources */ = {isa = PBXBuildFile; fileRef = 9833913627AC400800DAF119 /* AppTrackerDataSetProvider.swift */; }; 9838059F2228208E00385F1A /* PositiveFeedbackViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 9838059E2228208E00385F1A /* PositiveFeedbackViewController.swift */; }; 983BD6B52B34760600AAC78E /* MockPrivacyConfiguration.swift in Sources */ = {isa = PBXBuildFile; fileRef = C1B0F6412AB08BE9001EAF05 /* MockPrivacyConfiguration.swift */; }; + 983C52E42C2C050B007B5747 /* BookmarksStateRepair.swift in Sources */ = {isa = PBXBuildFile; fileRef = 983C52E32C2C050B007B5747 /* BookmarksStateRepair.swift */; }; + 983C52E72C2C0ACB007B5747 /* BookmarkStateRepairTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = 983C52E52C2C0ABA007B5747 /* BookmarkStateRepairTests.swift */; }; 983D71B12A286E810072E26D /* SyncDebugViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = 983D71B02A286E810072E26D /* SyncDebugViewController.swift */; }; 983EABB8236198F6003948D1 /* DatabaseMigration.swift in Sources */ = {isa = PBXBuildFile; fileRef = 983EABB7236198F6003948D1 /* DatabaseMigration.swift */; }; 984147A824F0259000362052 /* Onboarding.storyboard in Resources */ = {isa = PBXBuildFile; fileRef = 984147AA24F0259000362052 /* Onboarding.storyboard */; }; @@ -1272,6 +1288,9 @@ 31CB4250273AF50700FA0F3F /* SpeechRecognizerProtocol.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SpeechRecognizerProtocol.swift; sourceTree = ""; }; 31CC224828369B38001654A4 /* AutofillLoginSettingsListViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AutofillLoginSettingsListViewController.swift; sourceTree = ""; }; 31DD208327395A5A008FB313 /* VoiceSearchHelper.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = VoiceSearchHelper.swift; sourceTree = ""; }; + 31DE43C12C2C480D00F8C51F /* DuckPlayerFeaturePresentationView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = DuckPlayerFeaturePresentationView.swift; sourceTree = ""; }; + 31DE43C32C2C60E800F8C51F /* DuckPlayerModalPresenter.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = DuckPlayerModalPresenter.swift; sourceTree = ""; }; + 31DE43C52C2DA70A00F8C51F /* DuckPlayer-ModalAnimation.json */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.json; path = "DuckPlayer-ModalAnimation.json"; sourceTree = ""; }; 31EF52E0281B3BDC0034796E /* AutofillLoginListItemViewModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AutofillLoginListItemViewModel.swift; sourceTree = ""; }; 3712091D2C21E390003ADF3D /* RemoteMessagingStoreErrorHandling.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = RemoteMessagingStoreErrorHandling.swift; sourceTree = ""; }; 372A0FEF2B2389590033BF7F /* SyncMetricsEventsHandler.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SyncMetricsEventsHandler.swift; sourceTree = ""; }; @@ -1335,11 +1354,17 @@ 56D8556B2BEA91C4009F9698 /* SyncAlertsPresenting.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SyncAlertsPresenting.swift; sourceTree = ""; }; 6AC6DAB228804F97002723C0 /* BarsAnimator.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = BarsAnimator.swift; sourceTree = ""; }; 6AC98418288055C1005FA9CA /* BarsAnimatorTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = BarsAnimatorTests.swift; sourceTree = ""; }; + 6F5CC0802C2AFFE400AFC840 /* ToggleExpandButtonView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ToggleExpandButtonView.swift; sourceTree = ""; }; 6F655BE12BAB289E00AC3597 /* DefaultTheme.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = DefaultTheme.swift; sourceTree = ""; }; 6F8496402BC3D8EE00ADA54E /* OnboardingButtonsView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = OnboardingButtonsView.swift; sourceTree = ""; }; + 6F96FF0F2C2B128500162692 /* NewTabPageCustomizeButtonView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = NewTabPageCustomizeButtonView.swift; sourceTree = ""; }; 6FB030C7234331B400A10DB9 /* Configuration.xcconfig */ = {isa = PBXFileReference; lastKnownFileType = text.xcconfig; name = Configuration.xcconfig; path = Configuration/Configuration.xcconfig; sourceTree = ""; }; 6FB1FE9D2C24D41D0075B68B /* NewTabPageSectionsDebugView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = NewTabPageSectionsDebugView.swift; sourceTree = ""; }; 6FB1FEA12C256ACD0075B68B /* NewTabPageManager.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = NewTabPageManager.swift; sourceTree = ""; }; + 6FB2A6792C2C5BAE004D20C8 /* FavoriteEmptyStateItem.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = FavoriteEmptyStateItem.swift; sourceTree = ""; }; + 6FB2A67B2C2D9DF0004D20C8 /* FavoritesEmptyStateView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = FavoritesEmptyStateView.swift; sourceTree = ""; }; + 6FB2A67D2C2DAFB4004D20C8 /* NewTabPageGridView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = NewTabPageGridView.swift; sourceTree = ""; }; + 6FB2A67F2C2EA950004D20C8 /* FavoritesModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = FavoritesModel.swift; sourceTree = ""; }; 6FBF0F8A2BD7C0A900136CF0 /* AllProtectedCell.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AllProtectedCell.swift; sourceTree = ""; }; 6FD1BAE12B87A107000C475C /* AdAttributionPixelReporter.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = AdAttributionPixelReporter.swift; path = AdAttribution/AdAttributionPixelReporter.swift; sourceTree = ""; }; 6FD1BAE22B87A107000C475C /* AdAttributionReporterStorage.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; name = AdAttributionReporterStorage.swift; path = AdAttribution/AdAttributionReporterStorage.swift; sourceTree = ""; }; @@ -1347,10 +1372,15 @@ 6FD3AEE12B8DFBB80060FCCC /* AdAttributionPixelReporterTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AdAttributionPixelReporterTests.swift; sourceTree = ""; }; 6FDA1FB22B59584400AC962A /* AddressDisplayHelper.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AddressDisplayHelper.swift; sourceTree = ""; }; 6FDB3F182BD11A4400F7A307 /* AutocompleteSuggestionsModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AutocompleteSuggestionsModel.swift; sourceTree = ""; }; + 6FE0183F2C25CB3F001F680D /* FavoritesSectionHeader.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = FavoritesSectionHeader.swift; sourceTree = ""; }; 6FE095D72BD90AFB00490FF8 /* UniversalOmniBarState.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UniversalOmniBarState.swift; sourceTree = ""; }; 6FE127372C20492500EB5724 /* NewTabPage.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = NewTabPage.swift; sourceTree = ""; }; 6FE127392C204BD000EB5724 /* NewTabPageView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = NewTabPageView.swift; sourceTree = ""; }; + 6FE1273C2C204C2500EB5724 /* FavoritesView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = FavoritesView.swift; sourceTree = ""; }; + 6FE1273F2C204D9B00EB5724 /* ShortcutsView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ShortcutsView.swift; sourceTree = ""; }; + 6FE127422C204DF700EB5724 /* FavoriteItemView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = FavoriteItemView.swift; sourceTree = ""; }; 6FE127452C2054A900EB5724 /* NewTabPageViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = NewTabPageViewController.swift; sourceTree = ""; }; + 6FE1274A2C20943500EB5724 /* ShortcutItemView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ShortcutItemView.swift; sourceTree = ""; }; 6FF915802B88E0750042AC87 /* AdAttributionFetcherTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = AdAttributionFetcherTests.swift; sourceTree = ""; }; 7BC5711F2BDBB877003B0CCE /* VPNActivationDateStore.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = VPNActivationDateStore.swift; sourceTree = ""; }; 83004E7F2193BB8200DA013C /* WKNavigationExtension.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = WKNavigationExtension.swift; sourceTree = ""; }; @@ -1648,6 +1678,8 @@ 983A4B8F251EABEA00F3EDF1 /* et */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = et; path = et.lproj/InfoPlist.strings; sourceTree = ""; }; 983A4B90251EABEA00F3EDF1 /* et */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = et; path = et.lproj/Localizable.strings; sourceTree = ""; }; 983A4B91251EABEA00F3EDF1 /* et */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = et; path = et.lproj/InfoPlist.strings; sourceTree = ""; }; + 983C52E32C2C050B007B5747 /* BookmarksStateRepair.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = BookmarksStateRepair.swift; sourceTree = ""; }; + 983C52E52C2C0ABA007B5747 /* BookmarkStateRepairTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = BookmarkStateRepairTests.swift; sourceTree = ""; }; 983D71B02A286E810072E26D /* SyncDebugViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SyncDebugViewController.swift; sourceTree = ""; }; 983E1349251EABF200149BD9 /* fi */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = fi; path = fi.lproj/InfoPlist.strings; sourceTree = ""; }; 983E134A251EABF200149BD9 /* fi */ = {isa = PBXFileReference; lastKnownFileType = text.plist.strings; name = fi; path = fi.lproj/InfoPlist.strings; sourceTree = ""; }; @@ -3197,6 +3229,24 @@ name = Helper; sourceTree = ""; }; + 31DE43C72C2DAA7F00F8C51F /* Resources */ = { + isa = PBXGroup; + children = ( + 31BC5F402C2B0B540004DF37 /* DuckPlayer.xcassets */, + 31DE43C52C2DA70A00F8C51F /* DuckPlayer-ModalAnimation.json */, + ); + path = Resources; + sourceTree = ""; + }; + 31DE43C82C2DAA8F00F8C51F /* Modal */ = { + isa = PBXGroup; + children = ( + 31DE43C12C2C480D00F8C51F /* DuckPlayerFeaturePresentationView.swift */, + 31DE43C32C2C60E800F8C51F /* DuckPlayerModalPresenter.swift */, + ); + path = Modal; + sourceTree = ""; + }; 31E69A60280F4BAD00478327 /* LocalPackages */ = { isa = PBXGroup; children = ( @@ -3335,6 +3385,16 @@ name = NewTabPageSectionsDebugView; sourceTree = ""; }; + 6FB2A6782C2C5B9E004D20C8 /* EmptyState */ = { + isa = PBXGroup; + children = ( + 6FE0183F2C25CB3F001F680D /* FavoritesSectionHeader.swift */, + 6FB2A6792C2C5BAE004D20C8 /* FavoriteEmptyStateItem.swift */, + 6FB2A67B2C2D9DF0004D20C8 /* FavoritesEmptyStateView.swift */, + ); + name = EmptyState; + sourceTree = ""; + }; 6FD1BAE02B87A0E8000C475C /* AdAttribution */ = { isa = PBXGroup; children = ( @@ -3352,10 +3412,43 @@ 6FE127392C204BD000EB5724 /* NewTabPageView.swift */, 6FE127452C2054A900EB5724 /* NewTabPageViewController.swift */, 6FB1FEA12C256ACD0075B68B /* NewTabPageManager.swift */, + 6FE1273B2C204C0D00EB5724 /* Views */, ); name = HomeRedesign; sourceTree = ""; }; + 6FE1273B2C204C0D00EB5724 /* Views */ = { + isa = PBXGroup; + children = ( + 6FE127472C20941A00EB5724 /* Shortcuts */, + 6FE127412C204DE900EB5724 /* Favorites */, + 6F5CC0802C2AFFE400AFC840 /* ToggleExpandButtonView.swift */, + 6F96FF0F2C2B128500162692 /* NewTabPageCustomizeButtonView.swift */, + 6FB2A67D2C2DAFB4004D20C8 /* NewTabPageGridView.swift */, + ); + name = Views; + sourceTree = ""; + }; + 6FE127412C204DE900EB5724 /* Favorites */ = { + isa = PBXGroup; + children = ( + 6FB2A6782C2C5B9E004D20C8 /* EmptyState */, + 6FE1273C2C204C2500EB5724 /* FavoritesView.swift */, + 6FE127422C204DF700EB5724 /* FavoriteItemView.swift */, + 6FB2A67F2C2EA950004D20C8 /* FavoritesModel.swift */, + ); + name = Favorites; + sourceTree = ""; + }; + 6FE127472C20941A00EB5724 /* Shortcuts */ = { + isa = PBXGroup; + children = ( + 6FE1273F2C204D9B00EB5724 /* ShortcutsView.swift */, + 6FE1274A2C20943500EB5724 /* ShortcutItemView.swift */, + ); + name = Shortcuts; + sourceTree = ""; + }; 6FF9157F2B88E04F0042AC87 /* AdAttribution */ = { isa = PBXGroup; children = ( @@ -4239,11 +4332,12 @@ C14882DD27F20D7300D59F0C /* Bookmarks */ = { isa = PBXGroup; children = ( + C14882DE27F20D7E00D59F0C /* ImportExport */, 987130BD294AAB8200AB05E0 /* BSK */, 98AAF8E3292EB46000DBDF06 /* BookmarksMigrationTests.swift */, 98983095255B5019003339A2 /* BookmarksCachingSearchTests.swift */, 98629D322C21BDEB001E6031 /* BookmarksStateValidationTests.swift */, - C14882DE27F20D7E00D59F0C /* ImportExport */, + 983C52E52C2C0ABA007B5747 /* BookmarkStateRepairTests.swift */, ); name = Bookmarks; sourceTree = ""; @@ -4431,13 +4525,14 @@ D63FF8922C1B67D1006DE24D /* DuckPlayer */ = { isa = PBXGroup; children = ( + 31DE43C82C2DAA8F00F8C51F /* Modal */, + 31DE43C72C2DAA7F00F8C51F /* Resources */, D63FF8972C1B6A45006DE24D /* DuckPlayer.swift */, D62EC3C12C248AF800FC9D04 /* DuckNavigationHandling.swift */, D63FF8892C1B21C2006DE24D /* YouTubePlayerNavigationHandler.swift */, D63FF88B2C1B21ED006DE24D /* DuckPlayerURLExtension.swift */, D63FF8942C1B67E8006DE24D /* YoutubeOverlayUserScript.swift */, D63FF8932C1B67E8006DE24D /* YoutubePlayerUserScript.swift */, - 31BC5F402C2B0B540004DF37 /* DuckPlayer.xcassets */, ); path = DuckPlayer; sourceTree = ""; @@ -5289,6 +5384,7 @@ 9856A1982933D2EB00ACB44F /* BookmarksModelsErrorHandling.swift */, 379E877329E97C8D001C8BB0 /* BookmarksCleanupErrorHandling.swift */, 98629D302C21765A001E6031 /* BookmarksStateValidation.swift */, + 983C52E32C2C050B007B5747 /* BookmarksStateRepair.swift */, C14882D627F2010700D59F0C /* ImportExport */, F1CE42A81ECA0A660074A8DF /* LegacyStore */, ); @@ -6050,6 +6146,7 @@ 8524AAAC2A3888FE00EEC6D2 /* Waitlist.xcassets in Resources */, 982686B92600C0960011A8D6 /* ActionMessageView.xib in Resources */, F4F7F10A25813FE200045D62 /* 01_Fire_really_small.json in Resources */, + 31DE43C62C2DA70A00F8C51F /* DuckPlayer-ModalAnimation.json in Resources */, 1E0A75EA27A2FBD000A2BFB6 /* Downloads.storyboard in Resources */, 8517D98B221783A0006A8DD0 /* FindInPage.xcassets in Resources */, 984147C924F02E9E00362052 /* DaxOnboarding.storyboard in Resources */, @@ -6349,6 +6446,7 @@ C17B59592A03AAD30055F2D1 /* PasswordGenerationPromptViewModel.swift in Sources */, D65625A12C232F5E006EF297 /* SettingsDuckPlayerView.swift in Sources */, D6FEB8B52B74994000C3615F /* HeadlessWebViewCoordinator.swift in Sources */, + 6FE1273D2C204C2500EB5724 /* FavoritesView.swift in Sources */, 8528AE81212F15D600D0BD74 /* AppRatingPrompt.xcdatamodeld in Sources */, 1E24295E293F57FA00584836 /* LottieView.swift in Sources */, 8577A1C5255D2C0D00D43FCD /* HitTestingToolbar.swift in Sources */, @@ -6381,9 +6479,11 @@ C1CDA3162AFB9C7F006D1476 /* AutofillNeverPromptWebsitesManager.swift in Sources */, D668D9272B6937D2008E2FF2 /* SubscriptionITPViewModel.swift in Sources */, F1CA3C371F045878005FADB3 /* PrivacyStore.swift in Sources */, + 31DE43C42C2C60E800F8C51F /* DuckPlayerModalPresenter.swift in Sources */, 37FCAAC029930E26000E420A /* FailedAssertionView.swift in Sources */, 4BBBBA922B03291700D965DA /* VPNWaitlistUserText.swift in Sources */, F4E1936625AF722F001D2666 /* HighlightCutOutView.swift in Sources */, + 6FB2A67C2C2D9DF0004D20C8 /* FavoritesEmptyStateView.swift in Sources */, 1E162605296840D80004127F /* Triangle.swift in Sources */, B609D5522862EAFF0088CAC2 /* InlineWKDownloadDelegate.swift in Sources */, BDFF03222BA3D8E200F324C9 /* NetworkProtectionFeatureVisibility.swift in Sources */, @@ -6453,6 +6553,7 @@ CBEFB9142AE0844700DEDE7B /* CriticalAlerts.swift in Sources */, 986B16C425E92DF0007D23E8 /* BrowsingMenuViewController.swift in Sources */, 988AC355257E47C100793C64 /* RequeryLogic.swift in Sources */, + 6FB2A67A2C2C5BAE004D20C8 /* FavoriteEmptyStateItem.swift in Sources */, 6FBF0F8B2BD7C0A900136CF0 /* AllProtectedCell.swift in Sources */, 1E4F4A5A297193DE00625985 /* MainViewController+CookiesManaged.swift in Sources */, 8586A10D24CBA7070049720E /* FindInPageActivity.swift in Sources */, @@ -6507,6 +6608,7 @@ 9FA5E44B2BF1AF3400BDEF02 /* SubscriptionContainerViewFactory.swift in Sources */, C1F341C72A6924100032057B /* EmailAddressPromptViewModel.swift in Sources */, F47E53D9250A97330037C686 /* OnboardingDefaultBroswerViewController.swift in Sources */, + 6FE127432C204DF700EB5724 /* FavoriteItemView.swift in Sources */, F13B4BD51F183B3600814661 /* TabsModelPersistenceExtension.swift in Sources */, 980891A52237D4F500313A70 /* FeedbackNavigator.swift in Sources */, 1E8AD1C927BFAD1500ABA377 /* DirectoryMonitor.swift in Sources */, @@ -6553,7 +6655,9 @@ 319A37172829C8AD0079FBCE /* UITableViewExtension.swift in Sources */, 85EE7F59224673C5000FE757 /* WebContainerNavigationController.swift in Sources */, 6FD1BAE52B87A107000C475C /* AdAttributionReporterStorage.swift in Sources */, + 6F5CC0812C2AFFE400AFC840 /* ToggleExpandButtonView.swift in Sources */, D68A21462B7EC16200BB372E /* SubscriptionExternalLinkViewModel.swift in Sources */, + 31DE43C22C2C480D00F8C51F /* DuckPlayerFeaturePresentationView.swift in Sources */, F4C9FBF528340DDA002281CC /* AutofillInterfaceEmailTruncator.swift in Sources */, 1E016AB42949FEB500F21625 /* OmniBarNotificationViewModel.swift in Sources */, 6AC6DAB328804F97002723C0 /* BarsAnimator.swift in Sources */, @@ -6648,6 +6752,7 @@ C17B595A2A03AAD30055F2D1 /* PasswordGenerationPromptViewController.swift in Sources */, 8531A08E1F9950E6000484F0 /* UnprotectedSitesViewController.swift in Sources */, CBD4F13C279EBF4A00B20FD7 /* HomeMessage.swift in Sources */, + 6FB2A67E2C2DAFB4004D20C8 /* NewTabPageGridView.swift in Sources */, 1DEAADEC2BA45B4500E25A97 /* SettingsAccessibilityView.swift in Sources */, 85C8E61D2B0E47380029A6BD /* BookmarksDatabaseSetup.swift in Sources */, 3132FA2C27A07A1B00DD7A12 /* FilePreview.swift in Sources */, @@ -6673,6 +6778,7 @@ D6FEB8B12B7498A300C3615F /* HeadlessWebView.swift in Sources */, F1FDC9352BF51E41006B1435 /* VPNSettings+Environment.swift in Sources */, 850ABD012AC3961100A733DF /* MainViewController+Segues.swift in Sources */, + 6FB2A6802C2EA950004D20C8 /* FavoritesModel.swift in Sources */, 9817C9C321EF594700884F65 /* AutoClear.swift in Sources */, 9821234E2B6D0A6300F08C57 /* UserAuthenticator.swift in Sources */, 310C4B47281B60E300BA79A9 /* AutofillLoginDetailsViewModel.swift in Sources */, @@ -6688,6 +6794,7 @@ 854A012B2A54412600FCC628 /* ActivityViewController.swift in Sources */, F1CA3C391F045885005FADB3 /* PrivacyUserDefaults.swift in Sources */, 6F655BE22BAB289E00AC3597 /* DefaultTheme.swift in Sources */, + 6FE1274B2C20943500EB5724 /* ShortcutItemView.swift in Sources */, BD862E072B30F5E30073E2EE /* VPNFeedbackSender.swift in Sources */, AA4D6A6A23DB87B1007E8790 /* AppIconManager.swift in Sources */, 8563A03C1F9288D600F04442 /* BrowserChromeManager.swift in Sources */, @@ -6698,6 +6805,7 @@ 85058368219C49E000ED4EDB /* HomeViewSectionRenderers.swift in Sources */, 1DEAADEE2BA45DFE00E25A97 /* SettingsDataClearingView.swift in Sources */, D65625902C22D307006EF297 /* DuckPlayerURLExtension.swift in Sources */, + 6F96FF102C2B128500162692 /* NewTabPageCustomizeButtonView.swift in Sources */, EE01EB432AFC1E0A0096AAC9 /* NetworkProtectionVPNLocationView.swift in Sources */, 7BC571202BDBB877003B0CCE /* VPNActivationDateStore.swift in Sources */, 9820EAF522613CD30089094D /* WebProgressWorker.swift in Sources */, @@ -6738,6 +6846,7 @@ 1E7A71172934EB6400B7EA19 /* OmniBarNotificationAnimator.swift in Sources */, 85C2971A248162CA0063A335 /* DaxOnboardingPadViewController.swift in Sources */, F4F6DFB826EA9AA600ED7E12 /* BookmarksTextFieldCell.swift in Sources */, + 6FE127402C204D9B00EB5724 /* ShortcutsView.swift in Sources */, 85F98F92296F32BD00742F4A /* SyncSettingsViewController.swift in Sources */, 84E341961E2F7EFB00BDBA6F /* AppDelegate.swift in Sources */, 310D091D2799F57200DC0060 /* Download.swift in Sources */, @@ -6806,6 +6915,7 @@ D6E0C1892B7A2E0D00D5E1E9 /* DesktopDownloadViewModel.swift in Sources */, 8524CC9A246DA81700E59D45 /* FullscreenDaxDialogViewController.swift in Sources */, CBBB9A192BED441400BEAC71 /* PixelExperimentForBrokenSites.swift in Sources */, + 6FE018402C25CB3F001F680D /* FavoritesSectionHeader.swift in Sources */, F17669D71E43401C003D3222 /* MainViewController.swift in Sources */, 6FE127462C2054A900EB5724 /* NewTabPageViewController.swift in Sources */, 984D60B2222A1284003B9E3B /* FeedbackFormViewController.swift in Sources */, @@ -6844,6 +6954,7 @@ 9FA5E44E2BF1B16400BDEF02 /* SubscriptionContainerViewModelTests.swift in Sources */, 85BA58581F34F72F00C6E8CA /* AppUserDefaultsTests.swift in Sources */, F1134EBC1F40D45700B73467 /* MockStatisticsStore.swift in Sources */, + 983C52E72C2C0ACB007B5747 /* BookmarkStateRepairTests.swift in Sources */, 31C138AC27A403CB00FFD4B2 /* DownloadManagerTests.swift in Sources */, EEFE9C732A603CE9005B0A26 /* NetworkProtectionStatusViewModelTests.swift in Sources */, F13B4BF91F18CA0600814661 /* TabsModelTests.swift in Sources */, @@ -7138,6 +7249,7 @@ 1E4DCF4C27B6A4CB00961E25 /* URLFileExtension.swift in Sources */, EE50053029C3BA0800AE0773 /* InternalUserStore.swift in Sources */, F1D477CB1F2149C40031ED49 /* Type.swift in Sources */, + 983C52E42C2C050B007B5747 /* BookmarksStateRepair.swift in Sources */, 1E05D1D629C46EBB00BF9A1F /* DailyPixel.swift in Sources */, 1CB7B82123CEA1F800AA24EA /* DateExtension.swift in Sources */, 379E877429E97C8D001C8BB0 /* BookmarksCleanupErrorHandling.swift in Sources */, @@ -8017,7 +8129,7 @@ CODE_SIGN_ENTITLEMENTS = PacketTunnelProvider/PacketTunnelProvider.entitlements; CODE_SIGN_IDENTITY = "Apple Development"; CODE_SIGN_STYLE = Automatic; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 0; DEVELOPMENT_TEAM = HKE973VLUW; GCC_C_LANGUAGE_STANDARD = gnu11; GENERATE_INFOPLIST_FILE = YES; @@ -8054,7 +8166,7 @@ CODE_SIGN_IDENTITY = "iPhone Distribution"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Distribution"; CODE_SIGN_STYLE = Manual; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 0; DEVELOPMENT_TEAM = ""; "DEVELOPMENT_TEAM[sdk=iphoneos*]" = HKE973VLUW; GCC_C_LANGUAGE_STANDARD = gnu11; @@ -8144,7 +8256,7 @@ CODE_SIGN_IDENTITY = "iPhone Developer"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; CODE_SIGN_STYLE = Automatic; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 0; GCC_C_LANGUAGE_STANDARD = gnu11; INFOPLIST_FILE = ShareExtension/Info.plist; LD_RUNPATH_SEARCH_PATHS = ( @@ -8171,7 +8283,7 @@ CODE_SIGN_IDENTITY = "iPhone Developer"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Distribution"; CODE_SIGN_STYLE = Manual; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 0; DEVELOPMENT_TEAM = ""; "DEVELOPMENT_TEAM[sdk=iphoneos*]" = HKE973VLUW; GCC_C_LANGUAGE_STANDARD = gnu11; @@ -8320,7 +8432,7 @@ CODE_SIGN_ENTITLEMENTS = DuckDuckGo/DuckDuckGo.entitlements; CODE_SIGN_IDENTITY = "iPhone Distribution"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 0; DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; DEVELOPMENT_ASSET_PATHS = ""; "DEVELOPMENT_TEAM[sdk=iphoneos*]" = HKE973VLUW; @@ -8345,7 +8457,7 @@ CODE_SIGN_ENTITLEMENTS = DuckDuckGo/DuckDuckGo.entitlements; CODE_SIGN_IDENTITY = "iPhone Distribution"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Distribution"; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 0; "DEVELOPMENT_TEAM[sdk=iphoneos*]" = HKE973VLUW; INFOPLIST_FILE = DuckDuckGo/Info.plist; LD_RUNPATH_SEARCH_PATHS = ( @@ -8414,7 +8526,7 @@ CODE_SIGN_IDENTITY = "iPhone Developer"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; CODE_SIGN_STYLE = Automatic; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 0; DEAD_CODE_STRIPPING = NO; GCC_C_LANGUAGE_STANDARD = gnu11; INFOPLIST_FILE = Widgets/Info.plist; @@ -8448,7 +8560,7 @@ CODE_SIGN_IDENTITY = "iPhone Developer"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Distribution"; CODE_SIGN_STYLE = Manual; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 0; DEAD_CODE_STRIPPING = NO; DEVELOPMENT_TEAM = ""; "DEVELOPMENT_TEAM[sdk=iphoneos*]" = HKE973VLUW; @@ -8481,7 +8593,7 @@ CODE_SIGN_IDENTITY = "iPhone Developer"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; CODE_SIGN_STYLE = Automatic; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 0; GCC_C_LANGUAGE_STANDARD = gnu11; INFOPLIST_FILE = OpenAction/Info.plist; LD_RUNPATH_SEARCH_PATHS = ( @@ -8511,7 +8623,7 @@ CODE_SIGN_IDENTITY = "iPhone Developer"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Distribution"; CODE_SIGN_STYLE = Manual; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 0; DEVELOPMENT_TEAM = ""; "DEVELOPMENT_TEAM[sdk=iphoneos*]" = HKE973VLUW; GCC_C_LANGUAGE_STANDARD = gnu11; @@ -8821,7 +8933,7 @@ CODE_SIGN_ENTITLEMENTS = DuckDuckGo/DuckDuckGoAlpha.entitlements; CODE_SIGN_IDENTITY = "iPhone Distribution"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 0; DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; DEVELOPMENT_ASSET_PATHS = ""; "DEVELOPMENT_TEAM[sdk=iphoneos*]" = HKE973VLUW; @@ -8852,7 +8964,7 @@ CODE_SIGN_IDENTITY = "iPhone Developer"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; CODE_SIGN_STYLE = Automatic; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 0; GCC_C_LANGUAGE_STANDARD = gnu11; INFOPLIST_FILE = ShareExtension/Info.plist; LD_RUNPATH_SEARCH_PATHS = ( @@ -8880,7 +8992,7 @@ CODE_SIGN_IDENTITY = "iPhone Developer"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; CODE_SIGN_STYLE = Automatic; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 0; GCC_C_LANGUAGE_STANDARD = gnu11; INFOPLIST_FILE = OpenAction/Info.plist; LD_RUNPATH_SEARCH_PATHS = ( @@ -8913,7 +9025,7 @@ CODE_SIGN_IDENTITY = "iPhone Developer"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Developer"; CODE_SIGN_STYLE = Automatic; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 0; DEAD_CODE_STRIPPING = NO; GCC_C_LANGUAGE_STANDARD = gnu11; INFOPLIST_FILE = Widgets/Info.plist; @@ -8943,7 +9055,7 @@ CODE_SIGN_ENTITLEMENTS = PacketTunnelProvider/PacketTunnelProviderAlpha.entitlements; CODE_SIGN_IDENTITY = "Apple Development"; CODE_SIGN_STYLE = Automatic; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 0; DEVELOPMENT_TEAM = HKE973VLUW; GCC_C_LANGUAGE_STANDARD = gnu11; GENERATE_INFOPLIST_FILE = YES; @@ -8976,11 +9088,11 @@ CLANG_ENABLE_MODULES = YES; CODE_SIGN_IDENTITY = ""; CODE_SIGN_STYLE = Automatic; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 0; DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; DEFINES_MODULE = YES; DYLIB_COMPATIBILITY_VERSION = 1; - DYLIB_CURRENT_VERSION = 1; + DYLIB_CURRENT_VERSION = 0; DYLIB_INSTALL_NAME_BASE = "@rpath"; INFOPLIST_FILE = Core/Info.plist; INSTALL_PATH = "$(LOCAL_LIBRARY_DIR)/Frameworks"; @@ -9213,7 +9325,7 @@ CODE_SIGN_ENTITLEMENTS = DuckDuckGo/DuckDuckGoAlpha.entitlements; CODE_SIGN_IDENTITY = "iPhone Distribution"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Distribution"; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 0; DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; DEVELOPMENT_ASSET_PATHS = ""; "DEVELOPMENT_TEAM[sdk=iphoneos*]" = HKE973VLUW; @@ -9240,7 +9352,7 @@ CODE_SIGN_IDENTITY = "Apple Development"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Distribution"; CODE_SIGN_STYLE = Manual; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 0; DEVELOPMENT_TEAM = ""; "DEVELOPMENT_TEAM[sdk=iphoneos*]" = HKE973VLUW; GCC_C_LANGUAGE_STANDARD = gnu11; @@ -9272,7 +9384,7 @@ CODE_SIGN_IDENTITY = "iPhone Developer"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Distribution"; CODE_SIGN_STYLE = Manual; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 0; DEVELOPMENT_TEAM = ""; "DEVELOPMENT_TEAM[sdk=iphoneos*]" = HKE973VLUW; GCC_C_LANGUAGE_STANDARD = gnu11; @@ -9309,7 +9421,7 @@ CODE_SIGN_IDENTITY = "iPhone Developer"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Distribution"; CODE_SIGN_STYLE = Manual; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 0; DEAD_CODE_STRIPPING = NO; DEVELOPMENT_TEAM = ""; "DEVELOPMENT_TEAM[sdk=iphoneos*]" = HKE973VLUW; @@ -9344,7 +9456,7 @@ CODE_SIGN_IDENTITY = "Apple Development"; "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "iPhone Distribution"; CODE_SIGN_STYLE = Manual; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 0; DEVELOPMENT_TEAM = ""; "DEVELOPMENT_TEAM[sdk=iphoneos*]" = HKE973VLUW; GCC_C_LANGUAGE_STANDARD = gnu11; @@ -9379,11 +9491,11 @@ CLANG_ENABLE_MODULES = YES; CODE_SIGN_IDENTITY = ""; CODE_SIGN_STYLE = Automatic; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 0; DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; DEFINES_MODULE = YES; DYLIB_COMPATIBILITY_VERSION = 1; - DYLIB_CURRENT_VERSION = 1; + DYLIB_CURRENT_VERSION = 0; DYLIB_INSTALL_NAME_BASE = "@rpath"; INFOPLIST_FILE = Core/Info.plist; INSTALL_PATH = "$(LOCAL_LIBRARY_DIR)/Frameworks"; @@ -9556,11 +9668,11 @@ CLANG_ENABLE_MODULES = YES; CODE_SIGN_IDENTITY = ""; CODE_SIGN_STYLE = Automatic; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 0; DEBUG_INFORMATION_FORMAT = "dwarf-with-dsym"; DEFINES_MODULE = YES; DYLIB_COMPATIBILITY_VERSION = 1; - DYLIB_CURRENT_VERSION = 1; + DYLIB_CURRENT_VERSION = 0; DYLIB_INSTALL_NAME_BASE = "@rpath"; INFOPLIST_FILE = Core/Info.plist; INSTALL_PATH = "$(LOCAL_LIBRARY_DIR)/Frameworks"; @@ -9589,10 +9701,10 @@ CLANG_ENABLE_MODULES = YES; CODE_SIGN_IDENTITY = ""; CODE_SIGN_STYLE = Automatic; - CURRENT_PROJECT_VERSION = 1; + CURRENT_PROJECT_VERSION = 0; DEFINES_MODULE = YES; DYLIB_COMPATIBILITY_VERSION = 1; - DYLIB_CURRENT_VERSION = 1; + DYLIB_CURRENT_VERSION = 0; DYLIB_INSTALL_NAME_BASE = "@rpath"; INFOPLIST_FILE = Core/Info.plist; INSTALL_PATH = "$(LOCAL_LIBRARY_DIR)/Frameworks"; @@ -9794,7 +9906,7 @@ repositoryURL = "https://github.com/DuckDuckGo/BrowserServicesKit"; requirement = { kind = exactVersion; - version = 161.1.2; + version = 164.2.0; }; }; 9F8FE9472BAE50E50071E372 /* XCRemoteSwiftPackageReference "lottie-spm" */ = { diff --git a/DuckDuckGo.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved b/DuckDuckGo.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved index d8ca804f52..17f922a2ea 100644 --- a/DuckDuckGo.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved +++ b/DuckDuckGo.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved @@ -32,8 +32,8 @@ "kind" : "remoteSourceControl", "location" : "https://github.com/DuckDuckGo/BrowserServicesKit", "state" : { - "revision" : "cde956dec771bb7196ab3f4b07e00254e433d58d", - "version" : "161.1.2" + "revision" : "912001a8676345e96a8be360d5a6e3dca6d8e0ec", + "version" : "164.2.0" } }, { @@ -41,8 +41,8 @@ "kind" : "remoteSourceControl", "location" : "https://github.com/duckduckgo/content-scope-scripts", "state" : { - "revision" : "4689746e42b24c40c18896d697ea02b854e90d35", - "version" : "5.21.0" + "revision" : "7ac68ae3bc052fa59adbc1ba8fd5cb5849a6bc99", + "version" : "5.25.0" } }, { diff --git a/DuckDuckGo/AddressDisplayHelper.swift b/DuckDuckGo/AddressDisplayHelper.swift index 0016af4702..a65a4e7a09 100644 --- a/DuckDuckGo/AddressDisplayHelper.swift +++ b/DuckDuckGo/AddressDisplayHelper.swift @@ -24,18 +24,24 @@ extension OmniBar { struct AddressDisplayHelper { static func addressForDisplay(url: URL, showsFullURL: Bool) -> NSAttributedString { - + + if url.isDuckPlayer, + let playerURL = getDuckPlayerURL(url: url, showsFullURL: showsFullURL) { + return playerURL + } + if !showsFullURL, let shortAddress = shortURLString(url) { return NSAttributedString( string: shortAddress, attributes: [.foregroundColor: ThemeManager.shared.currentTheme.searchBarTextColor]) + } else { return deemphasisePath(forUrl: url) } } static func deemphasisePath(forUrl url: URL) -> NSAttributedString { - + let s = url.absoluteString let attributedString = NSMutableAttributedString(string: s) guard let c = URLComponents(url: url, resolvingAgainstBaseURL: true) else { @@ -72,5 +78,20 @@ extension OmniBar { return url.host?.droppingWwwPrefix() } + + private static func getDuckPlayerURL(url: URL, showsFullURL: Bool) -> NSAttributedString? { + if !showsFullURL { + return NSAttributedString( + string: UserText.duckPlayerFeatureName, + attributes: [.foregroundColor: ThemeManager.shared.currentTheme.searchBarTextColor]) + } else { + if let (videoID, _) = url.youtubeVideoParams { + return NSAttributedString( + string: URL.duckPlayer(videoID).absoluteString, + attributes: [.foregroundColor: ThemeManager.shared.currentTheme.searchBarTextColor]) + } + } + return nil + } } } diff --git a/DuckDuckGo/AppDelegate.swift b/DuckDuckGo/AppDelegate.swift index 81b50dd257..ef4149ff99 100644 --- a/DuckDuckGo/AppDelegate.swift +++ b/DuckDuckGo/AppDelegate.swift @@ -344,6 +344,8 @@ import WebKit AppDependencyProvider.shared.userBehaviorMonitor.handleAction(.reopenApp) + AppDependencyProvider.shared.subscriptionManager.loadInitialData() + setUpAutofillPixelReporter() return true @@ -460,6 +462,7 @@ import WebKit } } + // swiftlint:disable:next function_body_length func applicationDidBecomeActive(_ application: UIApplication) { guard !testing else { return } @@ -509,8 +512,6 @@ import WebKit #if NETWORK_PROTECTION widgetRefreshModel.refreshVPNWidget() - stopTunnelAndShowThankYouMessagingIfNeeded() - if tunnelDefaults.showEntitlementAlert { presentExpiredEntitlementAlert() } @@ -518,52 +519,30 @@ import WebKit presentExpiredEntitlementNotificationIfNeeded() Task { + await stopAndRemoveVPNIfNotAuthenticated() await refreshShortcuts() await vpnWorkaround.installRedditSessionWorkaround() } #endif - updateSubscriptionStatus() + AppDependencyProvider.shared.subscriptionManager.refreshCachedSubscriptionAndEntitlements { isSubscriptionActive in + if isSubscriptionActive { + DailyPixel.fire(pixel: .privacyProSubscriptionActive) + } + } let importPasswordsStatusHandler = ImportPasswordsStatusHandler(syncService: syncService) importPasswordsStatusHandler.checkSyncSuccessStatus() } - private func stopTunnelAndShowThankYouMessagingIfNeeded() { - if accountManager.isUserAuthenticated { - return - } - - if AppDependencyProvider.shared.vpnFeatureVisibility.isPrivacyProLaunched() && !accountManager.isUserAuthenticated { - Task { - await self.stopAndRemoveVPN(with: "subscription-check") - } - } - } - - private func stopAndRemoveVPN(with reason: String) async { - guard await AppDependencyProvider.shared.networkProtectionTunnelController.isInstalled else { + private func stopAndRemoveVPNIfNotAuthenticated() async { + // Only remove the VPN if the user is not authenticated, and it's installed: + guard !accountManager.isUserAuthenticated, await AppDependencyProvider.shared.networkProtectionTunnelController.isInstalled else { return } await AppDependencyProvider.shared.networkProtectionTunnelController.stop() - await AppDependencyProvider.shared.networkProtectionTunnelController.removeVPN() - } - - func updateSubscriptionStatus() { - Task { - guard let token = accountManager.accessToken else { return } - var subscriptionService: SubscriptionEndpointService { - AppDependencyProvider.shared.subscriptionManager.subscriptionEndpointService - } - if case .success(let subscription) = await subscriptionService.getSubscription(accessToken: token, - cachePolicy: .reloadIgnoringLocalCacheData) { - if subscription.isActive { - DailyPixel.fire(pixel: .privacyProSubscriptionActive) - } - } - await accountManager.fetchEntitlements(cachePolicy: .reloadIgnoringLocalCacheData) - } + await AppDependencyProvider.shared.networkProtectionTunnelController.removeVPN(reason: .didBecomeActiveCheck) } func applicationWillResignActive(_ application: UIApplication) { diff --git a/DuckDuckGo/AppUserDefaults.swift b/DuckDuckGo/AppUserDefaults.swift index 971b7d3090..51d060f184 100644 --- a/DuckDuckGo/AppUserDefaults.swift +++ b/DuckDuckGo/AppUserDefaults.swift @@ -39,6 +39,7 @@ public class AppUserDefaults: AppSettings { public static let addressBarPositionChanged = Notification.Name("com.duckduckgo.app.AddressBarPositionChanged") public static let showsFullURLAddressSettingChanged = Notification.Name("com.duckduckgo.app.ShowsFullURLAddressSettingChanged") public static let autofillDebugScriptToggled = Notification.Name("com.duckduckgo.app.DidToggleAutofillDebugScript") + public static let duckPlayerModeChanged = Notification.Name("com.duckduckgo.app.DuckPlayerModeChanged") } private let groupName: String @@ -389,9 +390,11 @@ public class AppUserDefaults: AppSettings { return .alwaysAsk } set { - // Here we set both the DuckPlayer mode and the overlayInteracte + // Here we set both the DuckPlayer mode and the duckPlayerAskModeOverlayHidden userDefaults?.set(newValue.stringValue, forKey: Keys.duckPlayerMode) userDefaults?.set(false, forKey: Keys.duckPlayerAskModeOverlayHidden) + NotificationCenter.default.post(name: AppUserDefaults.Notifications.duckPlayerModeChanged, + object: duckPlayerMode) } } } diff --git a/DuckDuckGo/BookmarksDatabaseSetup.swift b/DuckDuckGo/BookmarksDatabaseSetup.swift index a05de2ce9d..93364b97ba 100644 --- a/DuckDuckGo/BookmarksDatabaseSetup.swift +++ b/DuckDuckGo/BookmarksDatabaseSetup.swift @@ -72,6 +72,7 @@ struct BookmarksDatabaseSetup { let contextForValidation = bookmarksDatabase.makeContext(concurrencyType: .privateQueueConcurrencyType) contextForValidation.performAndWait { validator.validateBookmarksStructure(context: contextForValidation) + repairDeletedFlag(context: contextForValidation) } if migrationHappened { @@ -84,7 +85,24 @@ struct BookmarksDatabaseSetup { return migrationHappened } - + + private func repairDeletedFlag(context: NSManagedObjectContext) { + let stateRepair = BookmarksStateRepair(keyValueStore: UserDefaults.app) + let status = stateRepair.validateAndRepairPendingDeletionState(in: context) + switch status { + case .alreadyPerformed, .noBrokenData: + break + case .dataRepaired: + Pixel.fire(pixel: .debugBookmarksPendingDeletionFixed) + case .repairError(let underlyingError): + let processedErrors = CoreDataErrorsParser.parse(error: underlyingError as NSError) + + DailyPixel.fireDailyAndCount(pixel: .debugBookmarksPendingDeletionRepairError, + withAdditionalParameters: processedErrors.errorPixelParameters, + includedParameters: [.appVersion]) + } + } + private func migrateToFormFactorSpecificFavorites(_ context: NSManagedObjectContext, _ oldFavoritesOrder: [String]?) -> Bool { do { BookmarkFormFactorFavoritesMigration.migrateToFormFactorSpecificFavorites(byCopyingExistingTo: .mobile, diff --git a/DuckDuckGo/DuckPlayer/DuckNavigationHandling.swift b/DuckDuckGo/DuckPlayer/DuckNavigationHandling.swift index a854668e41..809889a797 100644 --- a/DuckDuckGo/DuckPlayer/DuckNavigationHandling.swift +++ b/DuckDuckGo/DuckPlayer/DuckNavigationHandling.swift @@ -23,11 +23,12 @@ protocol DuckNavigationHandling { func handleNavigation(_ navigationAction: WKNavigationAction, webView: WKWebView, completion: @escaping (WKNavigationActionPolicy) -> Void) - func handleRedirect(url: URL?, webView: WKWebView) - func handleRedirect(_ navigationAction: WKNavigationAction, - completion: @escaping (WKNavigationActionPolicy) -> Void, - webView: WKWebView) - func goBack(webView: WKWebView) + func handleURLChange(url: URL?, webView: WKWebView) + func handleDecidePolicyFor(_ navigationAction: WKNavigationAction, + completion: @escaping (WKNavigationActionPolicy) -> Void, + webView: WKWebView) + func handleGoBack(webView: WKWebView) + func handleReload(webView: WKWebView) } extension WKWebView { diff --git a/DuckDuckGo/DuckPlayer/DuckPlayer.swift b/DuckDuckGo/DuckPlayer/DuckPlayer.swift index bbec3ff700..8dde9d1fbd 100644 --- a/DuckDuckGo/DuckPlayer/DuckPlayer.swift +++ b/DuckDuckGo/DuckPlayer/DuckPlayer.swift @@ -30,7 +30,7 @@ enum DuckPlayerMode: Equatable, Codable, CustomStringConvertible, CaseIterable { private static let enabledString = "enabled" private static let alwaysAskString = "alwaysAsk" - private static let neverString = "alwaysAsk" + private static let neverString = "disabled" var description: String { switch self { @@ -99,7 +99,7 @@ public struct UserValues: Codable { final class DuckPlayerSettings { - private var appSettings: AppSettings + var appSettings: AppSettings init(appSettings: AppSettings = AppDependencyProvider.shared.appSettings) { self.appSettings = appSettings @@ -132,9 +132,12 @@ final class DuckPlayer { private var settings: DuckPlayerSettings - - init(settings: DuckPlayerSettings = DuckPlayerSettings()) { + @Published var userValues: UserValues + + init(settings: DuckPlayerSettings = DuckPlayerSettings(), userValues: UserValues? = nil) { self.settings = settings + self.userValues = userValues ?? UserValues(duckPlayerMode: settings.mode, askModeOverlayHidden: settings.askModeOverlayHidden) + registerForNotificationChanges() } // MARK: - Common Message Handlers @@ -144,8 +147,10 @@ final class DuckPlayer { assertionFailure("DuckPlayer: expected JSON representation of UserValues") return nil } + settings.mode = userValues.duckPlayerMode settings.askModeOverlayHidden = userValues.askModeOverlayHidden + return userValues } @@ -181,5 +186,21 @@ final class DuckPlayer { return InitialSetupSettings(userValues: userValues, settings: playerSettings) } + + private func registerForNotificationChanges() { + NotificationCenter.default.addObserver(self, + selector: #selector(updatePlayerMode), + name: AppUserDefaults.Notifications.duckPlayerModeChanged, + object: nil) + } + + + @objc private func updatePlayerMode(_ notification: Notification) { + userValues = encodeUserValues() + } + + deinit { + NotificationCenter.default.removeObserver(self) + } } diff --git a/DuckDuckGo/Settings.xcassets/Images/SettingsDuckPlayer.imageset/Contents.json b/DuckDuckGo/DuckPlayer/DuckPlayer.xcassets/DuckPlayerIcon.imageset/Contents.json similarity index 100% rename from DuckDuckGo/Settings.xcassets/Images/SettingsDuckPlayer.imageset/Contents.json rename to DuckDuckGo/DuckPlayer/DuckPlayer.xcassets/DuckPlayerIcon.imageset/Contents.json diff --git a/DuckDuckGo/Settings.xcassets/Images/SettingsDuckPlayer.imageset/Video-Player-Color-24.svg b/DuckDuckGo/DuckPlayer/DuckPlayer.xcassets/DuckPlayerIcon.imageset/Video-Player-Color-24.svg similarity index 100% rename from DuckDuckGo/Settings.xcassets/Images/SettingsDuckPlayer.imageset/Video-Player-Color-24.svg rename to DuckDuckGo/DuckPlayer/DuckPlayer.xcassets/DuckPlayerIcon.imageset/Video-Player-Color-24.svg diff --git a/DuckDuckGo/DuckPlayer/DuckPlayerURLExtension.swift b/DuckDuckGo/DuckPlayer/DuckPlayerURLExtension.swift index 2d13cac720..96f1b0eaeb 100644 --- a/DuckDuckGo/DuckPlayer/DuckPlayerURLExtension.swift +++ b/DuckDuckGo/DuckPlayer/DuckPlayerURLExtension.swift @@ -49,12 +49,12 @@ extension URL { var isDuckURLScheme: Bool { navigationalScheme == .duck } - + private var isYoutubeWatch: Bool { guard let host else { return false } return host.contains("youtube.com") && path == "/watch" } - + private var isYoutubeNoCookie: Bool { host == "www.youtube-nocookie.com" && pathComponents.count == 3 && pathComponents[safe: 1] == "embed" } diff --git a/DuckDuckGo/DuckPlayer/Modal/DuckPlayerFeaturePresentationView.swift b/DuckDuckGo/DuckPlayer/Modal/DuckPlayerFeaturePresentationView.swift new file mode 100644 index 0000000000..d72cef2ec5 --- /dev/null +++ b/DuckDuckGo/DuckPlayer/Modal/DuckPlayerFeaturePresentationView.swift @@ -0,0 +1,130 @@ +// +// DuckPlayerFeaturePresentationView.swift +// DuckDuckGo +// +// Copyright © 2024 DuckDuckGo. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +import SwiftUI +import DesignResourcesKit +import DuckUI +import Lottie + +struct DuckPlayerFeaturePresentationView: View { + @Environment(\.horizontalSizeClass) var horizontalSizeClass + @Environment(\.verticalSizeClass) var verticalSizeClass + @State private var isAnimating: Bool = false + var dismisPresentation: (() -> Void)? + + var body: some View { + ZStack { + + VStack(alignment: .center, spacing: stackVerticalSpacing) { + animation + + Text(UserText.duckPlayerPresentationModalTitle) + .daxTitle2() + .multilineTextAlignment(.center) + .foregroundColor(Color(designSystemColor: .textPrimary)) + .minimumScaleFactor(Constants.textMinimumScaleFactor) + + Text(UserText.duckPlayerPresentationModalBody) + .daxBodyRegular() + .multilineTextAlignment(.center) + .foregroundColor(Color(designSystemColor: .textSecondary)) + .minimumScaleFactor(Constants.textMinimumScaleFactor) + + Spacer() + + Button(UserText.duckPlayerPresentationModalDismissButton, action: dismissButtonTapped) + .buttonStyle(PrimaryButtonStyle()) + .frame(maxWidth: Constants.buttonCTAMaxWidth) + } + .padding(.horizontal, Constants.horizontalPadding) + .padding(.top, contentVerticalPadding) + .padding(.bottom) + + VStack { + HStack { + Spacer() + Button(action: dismissButtonTapped) { + Image(systemName: Constants.closeButtonSystemImage) + .foregroundColor(Color(designSystemColor: .textPrimary)) + .daxBodyRegular() + .frame(width: Constants.closeButtonSize.width, height: Constants.closeButtonSize.height) + } + .padding(8) + } + Spacer() + } + } + .background(Color(designSystemColor: .backgroundSheets)) + } + + private func dismissButtonTapped() { + dismisPresentation?() + } + + @ViewBuilder + private var animation: some View { + LottieView(lottieFile: "DuckPlayer-ModalAnimation", + isAnimating: $isAnimating) + .frame(width: Constants.heroImageSize.width, height: Constants.heroImageSize.height) + .cornerRadius(8) + .onAppear { + DispatchQueue.main.asyncAfter(deadline: .now() + Constants.animationDelay) { + isAnimating = true + } + } + } +} + +extension DuckPlayerFeaturePresentationView { + + enum Constants { + static let heroImageSize: CGSize = .init(width: 302, height: 180) + static let closeButtonSystemImage = "xmark" + static let closeButtonSize: CGSize = .init(width: 30, height: 30) + static let buttonCTAMaxWidth: CGFloat = 310 + static let textMinimumScaleFactor: CGFloat = 0.8 + static let horizontalPadding: CGFloat = 30 + static let animationDelay: Double = 2 + + } + + private var isSpaceConstrained: Bool { + verticalSizeClass == .compact + } + + private var stackVerticalSpacing: CGFloat { + if isSpaceConstrained { + return 5 + } else { + return 22 + } + } + + private var contentVerticalPadding: CGFloat { + if isSpaceConstrained { + return 4 + } else { + return 40 + } + } +} + +#Preview { + DuckPlayerFeaturePresentationView() +} diff --git a/DuckDuckGo/DuckPlayer/Modal/DuckPlayerModalPresenter.swift b/DuckDuckGo/DuckPlayer/Modal/DuckPlayerModalPresenter.swift new file mode 100644 index 0000000000..503e6af4e5 --- /dev/null +++ b/DuckDuckGo/DuckPlayer/Modal/DuckPlayerModalPresenter.swift @@ -0,0 +1,77 @@ +// +// DuckPlayerModalPresenter.swift +// DuckDuckGo +// +// Copyright © 2024 DuckDuckGo. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +import Foundation +import UIKit +import SwiftUI + +struct DuckPlayerModalPresenter { + + func presentDuckPlayerFeatureModal(on viewController: UIViewController) { + let hostingController = createHostingController() + configurePresentationStyle(for: hostingController, on: viewController) + viewController.present(hostingController, animated: true, completion: nil) + + hostingController.rootView.dismisPresentation = { + viewController.dismiss(animated: true) + } + } + + private func createHostingController() -> UIHostingController { + let duckPlayerFeaturePresentationView = DuckPlayerFeaturePresentationView() + let hostingController = UIHostingController(rootView: duckPlayerFeaturePresentationView) + hostingController.modalPresentationStyle = .pageSheet + hostingController.modalTransitionStyle = .coverVertical + return hostingController + } + + private func configurePresentationStyle(for hostingController: UIHostingController, on viewController: UIViewController) { + if #available(iOS 15.0, *) { + if let sheet = hostingController.presentationController as? UISheetPresentationController { + + if #available(iOS 16.0, *) { + let targetSize = getTargetSizeForPresentationView(on: viewController) + sheet.detents = [.custom { _ in targetSize.height }] + } else { + sheet.detents = [.large()] + + } + } + } + } + + @available(iOS 16.0, *) + private func getTargetSizeForPresentationView(on viewController: UIViewController) -> CGSize { + let duckPlayerFeaturePresentationView = DuckPlayerFeaturePresentationView() + let sizeHostingController = UIHostingController(rootView: duckPlayerFeaturePresentationView) + sizeHostingController.view.translatesAutoresizingMaskIntoConstraints = false + + viewController.view.addSubview(sizeHostingController.view) + NSLayoutConstraint.activate([ + sizeHostingController.view.widthAnchor.constraint(equalToConstant: viewController.view.frame.width) + ]) + + sizeHostingController.view.layoutIfNeeded() + + let targetSize = sizeHostingController.view.systemLayoutSizeFitting(UIView.layoutFittingCompressedSize) + sizeHostingController.view.removeFromSuperview() + + return targetSize + } +} diff --git a/DuckDuckGo/DuckPlayer/Resources/DuckPlayer-ModalAnimation.json b/DuckDuckGo/DuckPlayer/Resources/DuckPlayer-ModalAnimation.json new file mode 100644 index 0000000000..a4ab771677 --- /dev/null +++ b/DuckDuckGo/DuckPlayer/Resources/DuckPlayer-ModalAnimation.json @@ -0,0 +1 @@ +{"v":"4.8.0","meta":{"g":"LottieFiles AE 3.5.7","a":"","k":"","d":"","tc":""},"fr":29.9700012207031,"ip":7.00000028511585,"op":368.000014988947,"w":320,"h":180,"nm":"Comp 1","ddd":0,"assets":[{"id":"image_0","w":20,"h":20,"u":"","p":"","e":1},{"id":"image_1","w":42,"h":9,"u":"","p":"","e":1},{"id":"image_2","w":67,"h":17,"u":"","p":"","e":1},{"id":"image_3","w":141,"h":49,"u":"","p":"","e":1},{"id":"image_4","w":29,"h":4,"u":"","p":"","e":1},{"id":"image_5","w":101,"h":4,"u":"","p":"","e":1},{"id":"image_6","w":141,"h":87,"u":"","p":"","e":1},{"id":"image_7","w":141,"h":18,"u":"","p":"","e":1},{"id":"image_8","w":405,"h":708,"u":"","p":"","e":1}],"layers":[{"ddd":0,"ind":1,"ty":2,"nm":"Channel.pdf","cl":"pdf","refId":"image_0","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":47,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":64,"s":[0]},{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.333],"y":[0]},"t":349,"s":[0]},{"t":360.000014663101,"s":[100]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":47,"s":[220.75,47.75,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.667,"y":0.667},"o":{"x":0.333,"y":0.333},"t":64,"s":[262.25,47.75,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":349,"s":[262.25,47.75,0],"to":[0,0,0],"ti":[0,0,0]},{"t":360.000014663101,"s":[220.75,47.75,0]}],"ix":2},"a":{"a":0,"k":[10,10,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ip":0,"op":900.000036657751,"st":0,"bm":0},{"ddd":0,"ind":2,"ty":2,"nm":"Text Ad.pdf","cl":"pdf","refId":"image_1","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":47,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":64,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":349,"s":[0]},{"t":360.000014663101,"s":[100]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":47,"s":[120.75,43.75,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.667,"y":0.667},"o":{"x":0.333,"y":0.333},"t":64,"s":[59.25,43.75,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":349,"s":[59.25,43.75,0],"to":[0,0,0],"ti":[0,0,0]},{"t":360.000014663101,"s":[120.75,43.75,0]}],"ix":2},"a":{"a":0,"k":[21,4.5,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ip":0,"op":900.000036657751,"st":0,"bm":0},{"ddd":0,"ind":3,"ty":2,"nm":"Ad.pdf","cl":"pdf","refId":"image_2","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":47,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":64,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":349,"s":[0]},{"t":360.000014663101,"s":[100]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":47,"s":[130.5,95.875,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.667,"y":0.667},"o":{"x":0.333,"y":0.333},"t":64,"s":[48,95.875,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":349,"s":[48,95.875,0],"to":[0,0,0],"ti":[0,0,0]},{"t":360.000014663101,"s":[130.5,95.875,0]}],"ix":2},"a":{"a":0,"k":[33.5,8.5,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ip":0,"op":900.000036657751,"st":0,"bm":0},{"ddd":0,"ind":4,"ty":2,"nm":"Description.pdf","cl":"pdf","refId":"image_3","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":47,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":64,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":349,"s":[0]},{"t":360.000014663101,"s":[100]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":47,"s":[161.417,143,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.667,"y":0.667},"o":{"x":0.167,"y":0.167},"t":64,"s":[162.917,204,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0},"t":349,"s":[162.917,204,0],"to":[0,0,0],"ti":[0,0,0]},{"t":360.000014663101,"s":[161.417,143,0]}],"ix":2},"a":{"a":0,"k":[70.5,24.5,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ip":0,"op":900.000036657751,"st":0,"bm":0},{"ddd":0,"ind":5,"ty":2,"nm":"Line 9.pdf","cl":"pdf","parent":7,"refId":"image_4","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":64,"s":[20.208,79.125,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0.333},"t":76,"s":[20.208,80.125,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0.167},"t":92,"s":[20.208,80.125,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":341,"s":[20.208,80.125,0],"to":[0,0,0],"ti":[0,0,0]},{"t":355.000014459446,"s":[20.208,79.125,0]}],"ix":2},"a":{"a":0,"k":[14.5,2,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ef":[{"ty":20,"nm":"Tint","np":6,"mn":"ADBE Tint","ix":1,"en":1,"ef":[{"ty":2,"nm":"Map Black To","mn":"ADBE Tint-0001","ix":1,"v":{"a":0,"k":[1,0.800000071526,0.200000017881,1],"ix":1}},{"ty":2,"nm":"Map White To","mn":"ADBE Tint-0002","ix":2,"v":{"a":0,"k":[0.93412989378,0.700597524643,0,1],"ix":2}},{"ty":0,"nm":"Amount to Tint","mn":"ADBE Tint-0003","ix":3,"v":{"a":1,"k":[{"i":{"x":[0.142],"y":[1]},"o":{"x":[0.951],"y":[0.71]},"t":47,"s":[100]},{"i":{"x":[0.142],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":59,"s":[0]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":341,"s":[0]},{"t":355.000014459446,"s":[100]}],"ix":3}},{"ty":6,"nm":"","mn":"ADBE Tint-0004","ix":4,"v":0}]}],"ip":0,"op":900.000036657751,"st":0,"bm":0},{"ddd":0,"ind":6,"ty":2,"nm":"Line 10.pdf","cl":"pdf","parent":7,"refId":"image_5","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":64,"s":[85.708,79.125,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0.333},"t":76,"s":[85.708,80.125,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.667,"y":0.667},"o":{"x":0.167,"y":0.167},"t":92,"s":[85.708,80.125,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":341,"s":[85.708,80.125,0],"to":[0,0,0],"ti":[0,0,0]},{"t":355.000014459446,"s":[85.708,79.125,0]}],"ix":2},"a":{"a":0,"k":[50.5,2,0],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0.833,0.833,0.833],"y":[0.833,0.833,0.833]},"o":{"x":[0.167,0.167,0.167],"y":[0.167,0.167,0.167]},"t":64,"s":[101.485,100,100]},{"t":92.0000037472368,"s":[100,100,100]}],"ix":6}},"ao":0,"ip":0,"op":900.000036657751,"st":0,"bm":0},{"ddd":0,"ind":7,"ty":2,"nm":"Duck PLayer.pdf","cl":"pdf","refId":"image_6","sr":1,"ks":{"o":{"a":0,"k":100,"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.333,"y":0},"t":47,"s":[162.917,75.75,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0.333},"t":64,"s":[162.917,91.25,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.667,"y":0.667},"o":{"x":0.167,"y":0.167},"t":92,"s":[162.917,91.25,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.333,"y":0},"t":341,"s":[162.917,91.25,0],"to":[0,0,0],"ti":[0,0,0]},{"t":355.000014459446,"s":[162.917,75.75,0]}],"ix":2},"a":{"a":0,"k":[70.5,43.5,0],"ix":1},"s":{"a":1,"k":[{"i":{"x":[0,0,0.667],"y":[0.978,0.978,1]},"o":{"x":[0.055,0.055,0.333],"y":[0.409,0.409,0]},"t":99,"s":[100,100,100]},{"i":{"x":[0.348,0.348,0.667],"y":[1,1,1]},"o":{"x":[0.167,0.167,0.167],"y":[0,0,0]},"t":123,"s":[167.816,167.816,100]},{"i":{"x":[0.858,0.858,0.667],"y":[1.003,1.003,1]},"o":{"x":[0.527,0.527,0.167],"y":[0.386,0.386,0]},"t":341,"s":[167.816,167.816,100]},{"t":355.000014459446,"s":[100,100,100]}],"ix":6}},"ao":0,"hasMask":true,"masksProperties":[{"inv":false,"mode":"a","pt":{"a":0,"k":{"i":[[0,0],[0,0],[0,0],[0,0]],"o":[[0,0],[0,0],[0,0],[0,0]],"v":[[0,0],[0,87],[141,87],[141,0]],"c":true},"ix":1},"o":{"a":0,"k":100,"ix":3},"x":{"a":0,"k":0,"ix":4},"nm":"Mask 1"}],"ip":0,"op":900.000036657751,"st":0,"bm":0},{"ddd":0,"ind":8,"ty":2,"nm":"Top Bar.pdf","cl":"pdf","refId":"image_7","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0.833],"y":[0.833]},"o":{"x":[0.167],"y":[0.167]},"t":47,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":64,"s":[3]},{"i":{"x":[0.667],"y":[1]},"o":{"x":[0.333],"y":[0]},"t":346,"s":[3]},{"t":360.000014663101,"s":[100]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":1,"k":[{"i":{"x":0.667,"y":1},"o":{"x":0.167,"y":0.167},"t":47,"s":[163.042,23.75,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.667,"y":0.667},"o":{"x":0.333,"y":0.333},"t":64,"s":[162.917,-14.75,0],"to":[0,0,0],"ti":[0,0,0]},{"i":{"x":0.833,"y":0.833},"o":{"x":0.167,"y":0},"t":346,"s":[162.917,-14.75,0],"to":[0,0,0],"ti":[0,0,0]},{"t":360.000014663101,"s":[163.042,23.75,0]}],"ix":2},"a":{"a":0,"k":[70.5,9,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ip":0,"op":900.000036657751,"st":0,"bm":0},{"ddd":0,"ind":9,"ty":1,"nm":"Black Solid 1","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0],"y":[1.016]},"o":{"x":[0.077],"y":[0.68]},"t":99,"s":[9]},{"i":{"x":[0],"y":[8.191]},"o":{"x":[0.167],"y":[0]},"t":123,"s":[59]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":341,"s":[59]},{"t":355.000014459446,"s":[9]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[160.5,88,0],"ix":2},"a":{"a":0,"k":[160,90,0],"ix":1},"s":{"a":0,"k":[105,108.889,100],"ix":6}},"ao":0,"sw":320,"sh":180,"sc":"#000000","ip":0,"op":900.000036657751,"st":0,"bm":0},{"ddd":0,"ind":10,"ty":2,"nm":"Rectangle.png","cl":"png","refId":"image_8","sr":1,"ks":{"o":{"a":1,"k":[{"i":{"x":[0],"y":[1.038]},"o":{"x":[0.027],"y":[1.042]},"t":99,"s":[0]},{"i":{"x":[0],"y":[35.719]},"o":{"x":[0.167],"y":[0]},"t":123,"s":[100]},{"i":{"x":[0.833],"y":[1]},"o":{"x":[0.167],"y":[0]},"t":341,"s":[100]},{"t":355.000014459446,"s":[0]}],"ix":11},"r":{"a":0,"k":0,"ix":10},"p":{"a":0,"k":[132.5,-168.5,0],"ix":2},"a":{"a":0,"k":[202.5,354,0],"ix":1},"s":{"a":0,"k":[100,100,100],"ix":6}},"ao":0,"ip":55.0000022401959,"op":955.000038897947,"st":55.0000022401959,"bm":0}],"markers":[]} \ No newline at end of file diff --git a/DuckDuckGo/DuckPlayer/DuckPlayer.xcassets/Contents.json b/DuckDuckGo/DuckPlayer/Resources/DuckPlayer.xcassets/Contents.json similarity index 100% rename from DuckDuckGo/DuckPlayer/DuckPlayer.xcassets/Contents.json rename to DuckDuckGo/DuckPlayer/Resources/DuckPlayer.xcassets/Contents.json diff --git a/DuckDuckGo/DuckPlayer/DuckPlayer.xcassets/SettingsDuckPlayerHero.imageset/Contents.json b/DuckDuckGo/DuckPlayer/Resources/DuckPlayer.xcassets/SettingsDuckPlayerHero.imageset/Contents.json similarity index 100% rename from DuckDuckGo/DuckPlayer/DuckPlayer.xcassets/SettingsDuckPlayerHero.imageset/Contents.json rename to DuckDuckGo/DuckPlayer/Resources/DuckPlayer.xcassets/SettingsDuckPlayerHero.imageset/Contents.json diff --git a/DuckDuckGo/DuckPlayer/DuckPlayer.xcassets/SettingsDuckPlayerHero.imageset/SettingsDuckPlayerHero.pdf b/DuckDuckGo/DuckPlayer/Resources/DuckPlayer.xcassets/SettingsDuckPlayerHero.imageset/SettingsDuckPlayerHero.pdf similarity index 100% rename from DuckDuckGo/DuckPlayer/DuckPlayer.xcassets/SettingsDuckPlayerHero.imageset/SettingsDuckPlayerHero.pdf rename to DuckDuckGo/DuckPlayer/Resources/DuckPlayer.xcassets/SettingsDuckPlayerHero.imageset/SettingsDuckPlayerHero.pdf diff --git a/DuckDuckGo/DuckPlayer/YouTubePlayerNavigationHandler.swift b/DuckDuckGo/DuckPlayer/YouTubePlayerNavigationHandler.swift index 520067b19a..50938264d4 100644 --- a/DuckDuckGo/DuckPlayer/YouTubePlayerNavigationHandler.swift +++ b/DuckDuckGo/DuckPlayer/YouTubePlayerNavigationHandler.swift @@ -21,7 +21,14 @@ import Foundation import ContentScopeScripts import WebKit -struct YoutubePlayerNavigationHandler { +final class YoutubePlayerNavigationHandler { + + var duckPlayerMode: DuckPlayerMode + + init(duckPlayerMode: DuckPlayerMode = AppDependencyProvider.shared.appSettings.duckPlayerMode) { + self.duckPlayerMode = duckPlayerMode + registerForNotificationChanges() + } private static let templateDirectory = "pages/duckplayer" private static let templateName = "index" @@ -69,14 +76,40 @@ struct YoutubePlayerNavigationHandler { let duckPlayerRequest = Self.makeDuckPlayerRequest(from: request) performNavigation(duckPlayerRequest, responseHTML: html, webView: webView) } + + private func registerForNotificationChanges() { + NotificationCenter.default.addObserver(self, + selector: #selector(updatePlayerMode), + name: AppUserDefaults.Notifications.duckPlayerModeChanged, + object: nil) + } + + + @objc private func updatePlayerMode(_ notification: Notification) { + if let mode = notification.object as? DuckPlayerMode { + self.duckPlayerMode = mode + } + } + + deinit { + NotificationCenter.default.removeObserver(self) + } + } extension YoutubePlayerNavigationHandler: DuckNavigationHandling { - + + // Handle rendering the simulated request if the URL is duck:// + // and DuckPlayer is either enabled or alwaysAsk + @MainActor func handleNavigation(_ navigationAction: WKNavigationAction, webView: WKWebView, completion: @escaping (WKNavigationActionPolicy) -> Void) { - if let url = navigationAction.request.url, url.isDuckPlayer { + + // If DuckPlayer is Enabled or in ask mode, render the video + if let url = navigationAction.request.url, + url.isDuckURLScheme, + duckPlayerMode == .enabled || duckPlayerMode == .alwaysAsk { let html = Self.makeHTMLFromTemplate() let newRequest = Self.makeDuckPlayerRequest(from: URLRequest(url: url)) if #available(iOS 15.0, *) { @@ -85,10 +118,22 @@ extension YoutubePlayerNavigationHandler: DuckNavigationHandling { return } } - completion(.cancel) + + // DuckPlayer is disabled, so we redirect to the video in YouTube + if let url = navigationAction.request.url, let (videoID, timestamp) = url.youtubeVideoParams, duckPlayerMode == .disabled { + webView.load(URLRequest(url: URL.youtube(videoID, timestamp: timestamp))) + completion(.allow) + return + } + + completion(.allow) + } - func handleRedirect(url: URL?, webView: WKWebView) { + // Handle URL changes not triggered via Omnibar + // such as changes triggered via JS + @MainActor + func handleURLChange(url: URL?, webView: WKWebView) { if let url = url, url.isYoutubeVideo, !url.isDuckPlayer, let (videoID, timestamp) = url.youtubeVideoParams { webView.stopLoading() let newURL = URL.duckPlayer(videoID, timestamp: timestamp) @@ -96,9 +141,12 @@ extension YoutubePlayerNavigationHandler: DuckNavigationHandling { } } - func handleRedirect(_ navigationAction: WKNavigationAction, - completion: @escaping (WKNavigationActionPolicy) -> Void, - webView: WKWebView) { + // DecidePolicyFor handler to redirect relevant requests + // to duck://player + @MainActor + func handleDecidePolicyFor(_ navigationAction: WKNavigationAction, + completion: @escaping (WKNavigationActionPolicy) -> Void, + webView: WKWebView) { if let url = navigationAction.request.url, url.isYoutubeVideo, !url.isDuckPlayer, let (videoID, timestamp) = url.youtubeVideoParams { webView.load(URLRequest(url: .duckPlayer(videoID, timestamp: timestamp))) completion(.allow) @@ -107,13 +155,27 @@ extension YoutubePlayerNavigationHandler: DuckNavigationHandling { completion(.cancel) } - func goBack(webView: WKWebView) { + // Handle Webview BackButton on DuckPlayer videos + @MainActor + func handleGoBack(webView: WKWebView) { guard let backURL = webView.backForwardList.backItem?.url, backURL.isYoutubeVideo, - backURL.youtubeVideoParams?.videoID == webView.url?.youtubeVideoParams?.videoID else { + backURL.youtubeVideoParams?.videoID == webView.url?.youtubeVideoParams?.videoID, + duckPlayerMode == .enabled else { webView.goBack() return } webView.goBack(skippingHistoryItems: 2) } + + + // Handle Reload for DuckPlayer Videos + @MainActor + func handleReload(webView: WKWebView) { + if let url = webView.url, url.isDuckPlayer, !url.isDuckURLScheme, let (videoID, timestamp) = url.youtubeVideoParams { + webView.load(URLRequest(url: .duckPlayer(videoID, timestamp: timestamp))) + } else { + webView.reload() + } + } } diff --git a/DuckDuckGo/DuckPlayer/YoutubeOverlayUserScript.swift b/DuckDuckGo/DuckPlayer/YoutubeOverlayUserScript.swift index ee09c20949..5e02a00298 100644 --- a/DuckDuckGo/DuckPlayer/YoutubeOverlayUserScript.swift +++ b/DuckDuckGo/DuckPlayer/YoutubeOverlayUserScript.swift @@ -21,10 +21,12 @@ import Foundation import WebKit import Common import UserScript +import Combine final class YoutubeOverlayUserScript: NSObject, Subfeature { private var duckPlayer: DuckPlayer + private var userValuesCancellable = Set() struct Constants { static let featureName = "duckPlayer" @@ -32,6 +34,18 @@ final class YoutubeOverlayUserScript: NSObject, Subfeature { init(duckPlayer: DuckPlayer) { self.duckPlayer = duckPlayer + super.init() + subscribeToDuckPlayerMode() + } + + // Listen to DuckPlayer Settings changed + private func subscribeToDuckPlayerMode() { + duckPlayer.$userValues + .dropFirst() + .sink { [weak self] updatedValues in + self?.userValuesUpdated(userValues: updatedValues) + } + .store(in: &userValuesCancellable) } enum MessageOrigin { @@ -96,10 +110,9 @@ final class YoutubeOverlayUserScript: NSObject, Subfeature { } public func userValuesUpdated(userValues: UserValues) { - guard let webView = webView else { - return assertionFailure("Could not access webView") + if let webView { + broker?.push(method: "onUserValuesChanged", params: userValues, for: self, into: webView) } - broker?.push(method: "onUserValuesChanged", params: userValues, for: self, into: webView) } // MARK: - Private Methods @@ -123,6 +136,10 @@ final class YoutubeOverlayUserScript: NSObject, Subfeature { struct UserValuesNotification: Encodable { let userValuesNotification: UserValues } + + deinit { + userValuesCancellable.removeAll() + } } extension YoutubeOverlayUserScript { @@ -131,8 +148,8 @@ extension YoutubeOverlayUserScript { guard let body = message.messageBody as? [String: Any], let parameters = body["params"] as? [String: Any] else { return nil } - let pixelName = parameters["pixelName"] as? String - + // let pixelName = parameters["pixelName"] as? String + // To be implemented at a later point return nil } diff --git a/DuckDuckGo/DuckPlayer/YoutubePlayerUserScript.swift b/DuckDuckGo/DuckPlayer/YoutubePlayerUserScript.swift index b451c46e07..fcbea69a99 100644 --- a/DuckDuckGo/DuckPlayer/YoutubePlayerUserScript.swift +++ b/DuckDuckGo/DuckPlayer/YoutubePlayerUserScript.swift @@ -20,6 +20,7 @@ import WebKit import Common import UserScript +import Combine final class YoutubePlayerUserScript: NSObject, Subfeature { @@ -35,8 +36,22 @@ final class YoutubePlayerUserScript: NSObject, Subfeature { static let initialSetup = "initialSetup" } + private var userValuesCancellable = Set() + init(duckPlayer: DuckPlayer) { self.duckPlayer = duckPlayer + super.init() + subscribeToDuckPlayerMode() + } + + // Listen to DuckPlayer Settings changed + private func subscribeToDuckPlayerMode() { + duckPlayer.$userValues + .dropFirst() + .sink { [weak self] updatedValues in + self?.userValuesUpdated(userValues: updatedValues) + } + .store(in: &userValuesCancellable) } weak var broker: UserScriptMessageBroker? @@ -66,9 +81,13 @@ final class YoutubePlayerUserScript: NSObject, Subfeature { } } - func userValuesUpdated(userValues: UserValues) { - if let webView = webView { + public func userValuesUpdated(userValues: UserValues) { + if let webView { broker?.push(method: "onUserValuesChanged", params: userValues, for: self, into: webView) } } + + deinit { + userValuesCancellable.removeAll() + } } diff --git a/DuckDuckGo/FavoriteEmptyStateItem.swift b/DuckDuckGo/FavoriteEmptyStateItem.swift new file mode 100644 index 0000000000..e5b69afd8f --- /dev/null +++ b/DuckDuckGo/FavoriteEmptyStateItem.swift @@ -0,0 +1,33 @@ +// +// FavoriteEmptyStateItem.swift +// DuckDuckGo +// +// Copyright © 2024 DuckDuckGo. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +import SwiftUI + +struct FavoriteEmptyStateItem: View { + var body: some View { + RoundedRectangle(cornerRadius: 8, style: .continuous) + .stroke(Color(designSystemColor: .lines), + style: StrokeStyle(lineWidth: 1.5, dash: [4, 2])) + .padding(1) // So the stroke is not clipped on the edges + } +} + +#Preview { + FavoriteEmptyStateItem() +} diff --git a/DuckDuckGo/FavoriteItemView.swift b/DuckDuckGo/FavoriteItemView.swift new file mode 100644 index 0000000000..0780d761f4 --- /dev/null +++ b/DuckDuckGo/FavoriteItemView.swift @@ -0,0 +1,61 @@ +// +// FavoriteItemView.swift +// DuckDuckGo +// +// Copyright © 2024 DuckDuckGo. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +import DesignResourcesKit +import SwiftUI + +struct FavoriteItemView: View { + let favicon: Image? + let name: String + + var body: some View { + VStack(spacing: 6) { + ZStack { + RoundedRectangle(cornerRadius: 8) + .fill(Color(designSystemColor: .surface)) + .shadow(color: .shade(0.12), radius: 0.5, y: 1) + .aspectRatio(1, contentMode: .fit) + + FavoriteIconView(favicon: favicon) + + } + + Text(name) + .daxCaption() + .foregroundColor(Color(designSystemColor: .textPrimary)) + .frame(alignment: .center) + } + } +} + +private struct FavoriteIconView: View { + let favicon: Image? + + var body: some View { + if let favicon { + favicon + .resizable() + .aspectRatio(1.0, contentMode: .fit) + } + } +} + +#Preview { + FavoriteItemView(favicon: nil, name: "Text").frame(width: 64, height: 64) +} diff --git a/DuckDuckGo/FavoritesEmptyStateView.swift b/DuckDuckGo/FavoritesEmptyStateView.swift new file mode 100644 index 0000000000..90af87372c --- /dev/null +++ b/DuckDuckGo/FavoritesEmptyStateView.swift @@ -0,0 +1,63 @@ +// +// FavoritesEmptyStateView.swift +// DuckDuckGo +// +// Copyright © 2024 DuckDuckGo. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +import SwiftUI + +struct FavoritesEmptyStateView: View { + @Environment(\.horizontalSizeClass) var horizontalSizeClass + + @State var headerPadding: CGFloat = 10 + + var body: some View { + VStack(spacing: 16) { + FavoritesSectionHeader() + .padding(.horizontal, headerPadding) + + NewTabPageGridView { placeholdersCount in + let placeholders = Array(0.. CGFloat) { + value = nextValue() + } + static var defaultValue: CGFloat = .zero +} diff --git a/DuckDuckGo/FavoritesModel.swift b/DuckDuckGo/FavoritesModel.swift new file mode 100644 index 0000000000..b1e8a494ca --- /dev/null +++ b/DuckDuckGo/FavoritesModel.swift @@ -0,0 +1,44 @@ +// +// FavoritesModel.swift +// DuckDuckGo +// +// Copyright © 2024 DuckDuckGo. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +import Foundation + +struct Favorite: Identifiable, Equatable { + let id: Int +} + +final class FavoritesModel: ObservableObject { + + @Published private(set) var allFavorites: [Favorite] + var isEmpty: Bool { + allFavorites.isEmpty + } + + init() { + self.allFavorites = [] + } + + func toggleFavoritesPresence() { + if isEmpty { + allFavorites = (1...50).map { Favorite(id: $0) } + } else { + allFavorites = [] + } + } +} diff --git a/DuckDuckGo/FavoritesSectionHeader.swift b/DuckDuckGo/FavoritesSectionHeader.swift new file mode 100644 index 0000000000..e190400aec --- /dev/null +++ b/DuckDuckGo/FavoritesSectionHeader.swift @@ -0,0 +1,42 @@ +// +// FavoritesSectionHeader.swift +// DuckDuckGo +// +// Copyright © 2024 DuckDuckGo. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +import SwiftUI +import DesignResourcesKit + +struct FavoritesSectionHeader: View { + var body: some View { + HStack(spacing: 16, content: { + Text("Favorites") + .font(.system(size: 15, weight: .semibold)) + .foregroundColor(Color(designSystemColor: .textPrimary)) + .frame(alignment: .leading) + + Spacer() + + Button(action: {}, label: { + Image(.info12) + }).tintIfAvailable(Color(designSystemColor: .textPrimary)) + }) + } +} + +#Preview { + FavoritesSectionHeader() +} diff --git a/DuckDuckGo/FavoritesView.swift b/DuckDuckGo/FavoritesView.swift new file mode 100644 index 0000000000..cdb66ae363 --- /dev/null +++ b/DuckDuckGo/FavoritesView.swift @@ -0,0 +1,58 @@ +// +// FavoritesView.swift +// DuckDuckGo +// +// Copyright © 2024 DuckDuckGo. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +import Common +import DesignResourcesKit +import DuckUI +import SwiftUI + +struct FavoritesView: View { + @Environment(\.horizontalSizeClass) var horizontalSizeClass + @ObservedObject var model: FavoritesModel + + @State var isCollapsed: Bool = true + + var body: some View { + VStack(alignment: .center) { + + let collapsedMaxItemsCount = NewTabPageGrid.columnsCount(for: horizontalSizeClass) * 2 + + let data = isCollapsed ? Array(model.allFavorites.prefix(collapsedMaxItemsCount)) : model.allFavorites + + NewTabPageGridView { _ in + ForEach(data) { item in + FavoriteItemView(favicon: nil, name: "\(item.id)") + .frame(width: NewTabPageGrid.Item.edgeSize) + } + } + + if model.allFavorites.count > collapsedMaxItemsCount { + Button(action: { + isCollapsed.toggle() + }, label: { + ToggleExpandButtonView(isIndicatingExpand: isCollapsed).padding() + }) + } + } + } +} + +#Preview { + FavoritesView(model: FavoritesModel()) +} diff --git a/DuckDuckGo/Feedback/VPNFeedbackFormView.swift b/DuckDuckGo/Feedback/VPNFeedbackFormView.swift index fd116f15e6..b74eca8b17 100644 --- a/DuckDuckGo/Feedback/VPNFeedbackFormView.swift +++ b/DuckDuckGo/Feedback/VPNFeedbackFormView.swift @@ -25,8 +25,7 @@ import NetworkProtection @available(iOS 15.0, *) struct VPNFeedbackFormCategoryView: View { @Environment(\.dismiss) private var dismiss - let collector = DefaultVPNMetadataCollector(statusObserver: AppDependencyProvider.shared.connectionObserver, - tokenStore: AppDependencyProvider.shared.networkProtectionKeychainTokenStore) + let collector = DefaultVPNMetadataCollector(statusObserver: AppDependencyProvider.shared.connectionObserver) var body: some View { VStack { diff --git a/DuckDuckGo/Feedback/VPNMetadataCollector.swift b/DuckDuckGo/Feedback/VPNMetadataCollector.swift index b439d4c0f9..8345ec44b5 100644 --- a/DuckDuckGo/Feedback/VPNMetadataCollector.swift +++ b/DuckDuckGo/Feedback/VPNMetadataCollector.swift @@ -65,8 +65,8 @@ struct VPNMetadata: Encodable { } struct PrivacyProInfo: Encodable { - let hasToken: Bool - let subscriptionActive: Bool + let hasPrivacyProAccount: Bool + let hasVPNEntitlement: Bool } struct LastDisconnectError: Encodable { @@ -114,18 +114,18 @@ protocol VPNMetadataCollector { final class DefaultVPNMetadataCollector: VPNMetadataCollector { private let statusObserver: ConnectionStatusObserver private let serverInfoObserver: ConnectionServerInfoObserver - private let tokenStore: NetworkProtectionTokenStore + private let accountManager: AccountManager private let settings: VPNSettings private let defaults: UserDefaults init(statusObserver: ConnectionStatusObserver, serverInfoObserver: ConnectionServerInfoObserver = ConnectionServerInfoObserverThroughSession(), - tokenStore: NetworkProtectionTokenStore, + accountManager: AccountManager = AppDependencyProvider.shared.subscriptionManager.accountManager, settings: VPNSettings = .init(defaults: .networkProtectionGroupDefaults), defaults: UserDefaults = .networkProtectionGroupDefaults) { self.statusObserver = statusObserver self.serverInfoObserver = serverInfoObserver - self.tokenStore = tokenStore + self.accountManager = accountManager self.settings = settings self.defaults = defaults } @@ -136,7 +136,7 @@ final class DefaultVPNMetadataCollector: VPNMetadataCollector { let networkInfoMetadata = await collectNetworkInformation() let vpnState = await collectVPNState() let vpnSettingsState = collectVPNSettingsState() - let privacyProInfo = collectPrivacyProInfo() + let privacyProInfo = await collectPrivacyProInfo() return VPNMetadata( appInfo: appInfoMetadata, @@ -253,20 +253,14 @@ final class DefaultVPNMetadataCollector: VPNMetadataCollector { ) } - func collectPrivacyProInfo() -> VPNMetadata.PrivacyProInfo { - var hasToken: Bool { - guard let token = try? tokenStore.fetchToken(), - !token.hasPrefix(NetworkProtectionKeychainTokenStore.authTokenPrefix) else { - return false - } - return true - } - + func collectPrivacyProInfo() async -> VPNMetadata.PrivacyProInfo { + let hasVPNEntitlement = (try? await accountManager.hasEntitlement(forProductName: .networkProtection).get()) ?? false return .init( - hasToken: hasToken, - subscriptionActive: AppDependencyProvider.shared.subscriptionManager.accountManager.isUserAuthenticated + hasPrivacyProAccount: accountManager.isUserAuthenticated, + hasVPNEntitlement: hasVPNEntitlement ) } + } private extension NSError { diff --git a/DuckDuckGo/MainViewController.swift b/DuckDuckGo/MainViewController.swift index e04428bd65..5b2a294cdc 100644 --- a/DuckDuckGo/MainViewController.swift +++ b/DuckDuckGo/MainViewController.swift @@ -748,7 +748,7 @@ class MainViewController: UIViewController { } if homeTabManager.isNewTabPageSectionsEnabled { - let controller = NewTabPageViewController(rootView: NewTabPageView()) + let controller = NewTabPageViewController(rootView: NewTabPageView(favoritesModel: FavoritesModel())) newTabPageViewController = controller addToContentContainer(controller: controller) viewCoordinator.logoContainer.isHidden = true @@ -1503,7 +1503,7 @@ class MainViewController: UIViewController { } await networkProtectionTunnelController.stop() - await networkProtectionTunnelController.removeVPN() + await networkProtectionTunnelController.removeVPN(reason: .entitlementCheck) } } @@ -1511,7 +1511,7 @@ class MainViewController: UIViewController { private func onNetworkProtectionAccountSignOut(_ notification: Notification) { Task { await networkProtectionTunnelController.stop() - await networkProtectionTunnelController.removeVPN() + await networkProtectionTunnelController.removeVPN(reason: .signedOut) } } #endif diff --git a/DuckDuckGo/NetworkProtectionDebugViewController.swift b/DuckDuckGo/NetworkProtectionDebugViewController.swift index 27cdd8d7ac..c37fa4c53a 100644 --- a/DuckDuckGo/NetworkProtectionDebugViewController.swift +++ b/DuckDuckGo/NetworkProtectionDebugViewController.swift @@ -654,8 +654,7 @@ shouldShowVPNShortcut: \(vpnVisibility.shouldShowVPNShortcut() ? "YES" : "NO") @MainActor private func refreshMetadata() async { - let collector = DefaultVPNMetadataCollector(statusObserver: AppDependencyProvider.shared.connectionObserver, - tokenStore: AppDependencyProvider.shared.networkProtectionKeychainTokenStore) + let collector = DefaultVPNMetadataCollector(statusObserver: AppDependencyProvider.shared.connectionObserver) self.vpnMetadata = await collector.collectMetadata() self.tableView.reloadData() } @@ -701,7 +700,7 @@ shouldShowVPNShortcut: \(vpnVisibility.shouldShowVPNShortcut() ? "YES" : "NO") private func deleteVPNConfiguration() { Task { await AppDependencyProvider.shared.networkProtectionTunnelController.stop() - await AppDependencyProvider.shared.networkProtectionTunnelController.removeVPN() + await AppDependencyProvider.shared.networkProtectionTunnelController.removeVPN(reason: .debugMenu) } } } diff --git a/DuckDuckGo/NetworkProtectionTunnelController.swift b/DuckDuckGo/NetworkProtectionTunnelController.swift index 80ee7defa2..02f5953de7 100644 --- a/DuckDuckGo/NetworkProtectionTunnelController.swift +++ b/DuckDuckGo/NetworkProtectionTunnelController.swift @@ -26,6 +26,13 @@ import NetworkExtension import NetworkProtection import Subscription +enum VPNConfigurationRemovalReason: String { + case didBecomeActiveCheck + case entitlementCheck + case signedOut + case debugMenu +} + final class NetworkProtectionTunnelController: TunnelController { static var shouldSimulateFailure: Bool = false @@ -114,8 +121,18 @@ final class NetworkProtectionTunnelController: TunnelController { tunnelManager.connection.stopVPNTunnel() } - func removeVPN() async { - try? await tunnelManager?.removeFromPreferences() + func removeVPN(reason: VPNConfigurationRemovalReason) async { + do { + try await tunnelManager?.removeFromPreferences() + + DailyPixel.fireDailyAndCount(pixel: .networkProtectionVPNConfigurationRemoved, withAdditionalParameters: [ + PixelParameters.reason: reason.rawValue + ]) + } catch { + DailyPixel.fireDailyAndCount(pixel: .networkProtectionVPNConfigurationRemovalFailed, error: error, withAdditionalParameters: [ + PixelParameters.reason: reason.rawValue + ]) + } } // MARK: - Connection Status Querying diff --git a/DuckDuckGo/NewTabPageCustomizeButtonView.swift b/DuckDuckGo/NewTabPageCustomizeButtonView.swift new file mode 100644 index 0000000000..2d74483a27 --- /dev/null +++ b/DuckDuckGo/NewTabPageCustomizeButtonView.swift @@ -0,0 +1,34 @@ +// +// NewTabPageCustomizeButtonView.swift +// DuckDuckGo +// +// Copyright © 2024 DuckDuckGo. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +import SwiftUI +import DuckUI + +struct NewTabPageCustomizeButtonView: View { + var body: some View { + HStack { + Image(.options16) + Text("Customize") + } + } +} + +#Preview { + NewTabPageCustomizeButtonView() +} diff --git a/DuckDuckGo/NewTabPageGridView.swift b/DuckDuckGo/NewTabPageGridView.swift new file mode 100644 index 0000000000..6a01cf2393 --- /dev/null +++ b/DuckDuckGo/NewTabPageGridView.swift @@ -0,0 +1,56 @@ +// +// NewTabPageGridView.swift +// DuckDuckGo +// +// Copyright © 2024 DuckDuckGo. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +import SwiftUI + +struct NewTabPageGridView: View { + @Environment(\.horizontalSizeClass) var horizontalSizeClass + + @ViewBuilder var content: (_ columnsCount: Int) -> Content + + var body: some View { + let columnsCount = NewTabPageGrid.columnsCount(for: horizontalSizeClass) + + LazyVGrid(columns: flexibleColumns(columnsCount), content: { + content(columnsCount) + }) + .padding(0) + .offset(.zero) + .clipped() + } + + private func flexibleColumns(_ count: Int) -> [GridItem] { + Array(repeating: GridItem(.flexible(minimum: NewTabPageGrid.Item.edgeSize)), count: count) + } +} + +enum NewTabPageGrid { + enum ColumnCount { + static let compact = 4 + static let regular = 6 + } + + enum Item { + static let edgeSize = 64.0 + } + + static func columnsCount(for sizeClass: UserInterfaceSizeClass?) -> Int { + sizeClass == .regular ? ColumnCount.regular : ColumnCount.compact + } +} diff --git a/DuckDuckGo/NewTabPageView.swift b/DuckDuckGo/NewTabPageView.swift index 719eb8bbd2..ebe2d7155c 100644 --- a/DuckDuckGo/NewTabPageView.swift +++ b/DuckDuckGo/NewTabPageView.swift @@ -18,13 +18,42 @@ // import SwiftUI +import DuckUI struct NewTabPageView: View { + @ObservedObject var favoritesModel: FavoritesModel + var body: some View { - Text("Empty") + ScrollView { + VStack { + if favoritesModel.isEmpty { + FavoritesEmptyStateView() + .padding(Constant.sectionPadding) + } else { + FavoritesView(model: favoritesModel) + .padding(Constant.sectionPadding) + } + + ShortcutsView() + .padding(Constant.sectionPadding) + + Button(action: { + // Temporary action for testing purposes + favoritesModel.toggleFavoritesPresence() + }, label: { + NewTabPageCustomizeButtonView() + }).buttonStyle(SecondaryFillButtonStyle(compact: true, fullWidth: false)) + .padding(EdgeInsets(top: 88, leading: 0, bottom: 16, trailing: 0)) + } + } + .background(Color(designSystemColor: .background)) + } + + private struct Constant { + static let sectionPadding = EdgeInsets(top: 16, leading: 24, bottom: 16, trailing: 24) } } #Preview { - NewTabPageView() + NewTabPageView(favoritesModel: FavoritesModel()) } diff --git a/DuckDuckGo/OmniBar.swift b/DuckDuckGo/OmniBar.swift index 7932bbf32a..0117db51df 100644 --- a/DuckDuckGo/OmniBar.swift +++ b/DuckDuckGo/OmniBar.swift @@ -25,10 +25,14 @@ import DesignResourcesKit extension OmniBar: NibLoading {} +public enum OmniBarIcon: String { + case duckPlayer = "DuckPlayerIcon" +} + // swiftlint:disable file_length // swiftlint:disable type_body_length class OmniBar: UIView { - + public static let didLayoutNotification = Notification.Name("com.duckduckgo.app.OmniBarDidLayout") @IBOutlet weak var searchLoupe: UIView! @@ -69,6 +73,9 @@ class OmniBar: UIView { private var privacyIconAndTrackersAnimator = PrivacyIconAndTrackersAnimator() private var notificationAnimator = OmniBarNotificationAnimator() + + // Set up a view to add a custom icon to the Omnibar + private var customIconView: UIImageView = UIImageView(frame: CGRect(x: 4, y: 8, width: 26, height: 26)) static func loadFromXib() -> OmniBar { return OmniBar.load(nibName: "OmniBar") @@ -263,12 +270,25 @@ class OmniBar: UIView { !privacyIconAndTrackersAnimator.isAnimatingForDaxDialog else { return } + if privacyInfo.url.isDuckPlayer { + showCustomIcon(icon: .duckPlayer) + return + } + + customIconView.isHidden = true privacyInfoContainer.privacyIcon.isHidden = false - let icon = PrivacyIconLogic.privacyIcon(for: privacyInfo) privacyInfoContainer.privacyIcon.updateIcon(icon) } + // Support static custom icons, for things like internal pages, for example + func showCustomIcon(icon: OmniBarIcon) { + privacyInfoContainer.privacyIcon.isHidden = true + customIconView.image = UIImage(named: icon.rawValue) + privacyInfoContainer.addSubview(customIconView) + customIconView.isHidden = false + } + public func startTrackersAnimation(_ privacyInfo: PrivacyInfo, forDaxDialog: Bool) { guard state.allowsTrackersAnimation, !privacyInfoContainer.isAnimationPlaying else { return } diff --git a/DuckDuckGo/Settings.bundle/Root.plist b/DuckDuckGo/Settings.bundle/Root.plist index 866e1853fe..64d4cdd2c2 100644 --- a/DuckDuckGo/Settings.bundle/Root.plist +++ b/DuckDuckGo/Settings.bundle/Root.plist @@ -6,7 +6,7 @@ DefaultValue - 7.126.0 + 7.127.0 Key version Title diff --git a/DuckDuckGo/SettingsDuckPlayerView.swift b/DuckDuckGo/SettingsDuckPlayerView.swift index 978d86cee7..3280da7f7e 100644 --- a/DuckDuckGo/SettingsDuckPlayerView.swift +++ b/DuckDuckGo/SettingsDuckPlayerView.swift @@ -31,7 +31,7 @@ struct SettingsDuckPlayerView: View { Image("SettingsDuckPlayerHero") .padding(.top, -20) // Adjust for the image padding - Text(UserText.settingsDuckPlayerTitle) + Text(UserText.duckPlayerFeatureName) .daxTitle3() Text(UserText.settingsDuckPlayerInfoText) @@ -57,7 +57,7 @@ struct SettingsDuckPlayerView: View { .multilineTextAlignment(.center) } } - .applySettingsListModifiers(title: UserText.settingsDuckPlayerTitle, + .applySettingsListModifiers(title: UserText.duckPlayerFeatureName, displayMode: .inline, viewModel: viewModel) } diff --git a/DuckDuckGo/SettingsMainSettingsView.swift b/DuckDuckGo/SettingsMainSettingsView.swift index ef185dc038..7268ddc014 100644 --- a/DuckDuckGo/SettingsMainSettingsView.swift +++ b/DuckDuckGo/SettingsMainSettingsView.swift @@ -71,8 +71,8 @@ struct SettingsMainSettingsView: View { // Duck Player if viewModel.isInternalUser { NavigationLink(destination: SettingsDuckPlayerView().environmentObject(viewModel)) { - SettingsCellView(label: "Duck Player", - image: Image("SettingsDuckPlayer")) + SettingsCellView(label: UserText.duckPlayerFeatureName, + image: Image("DuckPlayerIcon")) } } } diff --git a/DuckDuckGo/SettingsRootView.swift b/DuckDuckGo/SettingsRootView.swift index cc3e346a67..32d4599313 100644 --- a/DuckDuckGo/SettingsRootView.swift +++ b/DuckDuckGo/SettingsRootView.swift @@ -117,9 +117,6 @@ struct SettingsRootView: View { SubscriptionContainerViewFactory.makeSubscribeFlow(origin: origin, navigationCoordinator: subscriptionNavigationCoordinator, subscriptionManager: AppDependencyProvider.shared.subscriptionManager) - case .subscriptionRestoreFlow: - SubscriptionContainerViewFactory.makeRestoreFlow(navigationCoordinator: subscriptionNavigationCoordinator, - subscriptionManager: AppDependencyProvider.shared.subscriptionManager) default: EmptyView() } diff --git a/DuckDuckGo/SettingsViewModel.swift b/DuckDuckGo/SettingsViewModel.swift index 228140d87b..5bedc53e6f 100644 --- a/DuckDuckGo/SettingsViewModel.swift +++ b/DuckDuckGo/SettingsViewModel.swift @@ -251,7 +251,9 @@ final class SettingsViewModel: ObservableObject { var duckPlayerModeBinding: Binding { Binding( - get: { self.state.duckPlayerMode ?? .alwaysAsk }, + get: { + return self.state.duckPlayerMode ?? .alwaysAsk + }, set: { self.appSettings.duckPlayerMode = $0 self.state.duckPlayerMode = $0 @@ -609,7 +611,6 @@ extension SettingsViewModel { case dbp case itr case subscriptionFlow(origin: String? = nil) - case subscriptionRestoreFlow // Add other cases as needed var id: String { @@ -618,7 +619,6 @@ extension SettingsViewModel { case .dbp: return "dbp" case .itr: return "itr" case .subscriptionFlow: return "subscriptionFlow" - case .subscriptionRestoreFlow: return "subscriptionRestoreFlow" // Ensure all cases are covered } } diff --git a/DuckDuckGo/ShortcutItemView.swift b/DuckDuckGo/ShortcutItemView.swift new file mode 100644 index 0000000000..914ba7b202 --- /dev/null +++ b/DuckDuckGo/ShortcutItemView.swift @@ -0,0 +1,49 @@ +// +// ShortcutItemView.swift +// DuckDuckGo +// +// Copyright © 2024 DuckDuckGo. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +import DesignResourcesKit +import SwiftUI + +struct ShortcutItemView: View { + let name: String + + var body: some View { + VStack(spacing: 6) { + ZStack { + RoundedRectangle(cornerRadius: 8) + .fill(Color(designSystemColor: .surface)) + .shadow(color: .shade(0.12), radius: 0.5, y: 1) + .aspectRatio(1, contentMode: .fit) + .frame(width: NewTabPageGrid.Item.edgeSize) + Image("Login-32-Color") + .resizable() + .aspectRatio(1.0, contentMode: .fit) + .frame(width: NewTabPageGrid.Item.edgeSize * 0.5) + } + Text(name) + .daxCaption() + .foregroundColor(Color(designSystemColor: .textPrimary)) + .frame(alignment: .center) + } + } +} + +#Preview { + ShortcutItemView(name: "Shortcut") +} diff --git a/DuckDuckGo/ShortcutsView.swift b/DuckDuckGo/ShortcutsView.swift new file mode 100644 index 0000000000..cb3bb026d0 --- /dev/null +++ b/DuckDuckGo/ShortcutsView.swift @@ -0,0 +1,56 @@ +// +// ShortcutsView.swift +// DuckDuckGo +// +// Copyright © 2024 DuckDuckGo. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +import SwiftUI + +enum Shortcut: Int, CaseIterable, Equatable, Identifiable { + var id: Int { rawValue } + + case bookmarks, aiChat, vpn, passwords + + var name: String { + switch self { + case .bookmarks: + UserText.homeTabShortcutBookmarks + case .aiChat: + UserText.homeTabShortcutAIChat + case .vpn: + UserText.homeTabShortcutVPN + case .passwords: + UserText.homeTabShortcutPasswords + } + } +} + +struct ShortcutsView: View { + @Environment(\.horizontalSizeClass) var horizontalSizeClass + @State var enabledShortcuts: [Shortcut] = Array(Shortcut.allCases.prefix(upTo: 3)) + + var body: some View { + NewTabPageGridView { _ in + ForEach(enabledShortcuts) { shortcut in + ShortcutItemView(name: shortcut.name) + } + } + } +} + +#Preview { + ShortcutsView() +} diff --git a/DuckDuckGo/Subscription/SubscriptionManageriOS14.swift b/DuckDuckGo/Subscription/SubscriptionManageriOS14.swift index b909855ae6..684ea20a41 100644 --- a/DuckDuckGo/Subscription/SubscriptionManageriOS14.swift +++ b/DuckDuckGo/Subscription/SubscriptionManageriOS14.swift @@ -32,7 +32,7 @@ class SubscriptionManageriOS14: SubscriptionManager { var currentEnvironment: SubscriptionEnvironment = SubscriptionEnvironment.default var canPurchase: Bool = false func loadInitialData() {} - func updateSubscriptionStatus(completion: @escaping (Bool) -> Void) {} + func refreshCachedSubscriptionAndEntitlements(completion: @escaping (Bool) -> Void) {} func url(for type: SubscriptionURL) -> URL { URL(string: "https://duckduckgo.com")! diff --git a/DuckDuckGo/Subscription/UserScripts/SubscriptionPagesUseSubscriptionFeature.swift b/DuckDuckGo/Subscription/UserScripts/SubscriptionPagesUseSubscriptionFeature.swift index 699720a670..e257a026be 100644 --- a/DuckDuckGo/Subscription/UserScripts/SubscriptionPagesUseSubscriptionFeature.swift +++ b/DuckDuckGo/Subscription/UserScripts/SubscriptionPagesUseSubscriptionFeature.swift @@ -95,15 +95,19 @@ final class SubscriptionPagesUseSubscriptionFeature: Subfeature, ObservableObjec private let subscriptionManager: SubscriptionManager private var accountManager: AccountManager { subscriptionManager.accountManager } private let appStorePurchaseFlow: AppStorePurchaseFlow + private let appStoreRestoreFlow: AppStoreRestoreFlow + private let appStoreAccountManagementFlow: AppStoreAccountManagementFlow init(subscriptionManager: SubscriptionManager, subscriptionAttributionOrigin: String?, appStorePurchaseFlow: AppStorePurchaseFlow, - appStoreRestoreFlow: AppStoreRestoreFlow) { + appStoreRestoreFlow: AppStoreRestoreFlow, + appStoreAccountManagementFlow: AppStoreAccountManagementFlow) { self.subscriptionManager = subscriptionManager self.appStorePurchaseFlow = appStorePurchaseFlow self.appStoreRestoreFlow = appStoreRestoreFlow + self.appStoreAccountManagementFlow = appStoreAccountManagementFlow self.subscriptionAttributionOrigin = subscriptionAttributionOrigin } @@ -187,7 +191,9 @@ final class SubscriptionPagesUseSubscriptionFeature: Subfeature, ObservableObjec // MARK: Broker Methods (Called from WebView via UserScripts) func getSubscription(params: Any, original: WKScriptMessage) async -> Encodable? { + await appStoreAccountManagementFlow.refreshAuthTokenIfNeeded() let authToken = accountManager.authToken ?? Constants.empty + return [Constants.token: authToken] } @@ -399,7 +405,6 @@ final class SubscriptionPagesUseSubscriptionFeature: Subfeature, ObservableObjec // MARK: Native methods - Called from ViewModels func restoreAccountFromAppStorePurchase() async throws { setTransactionStatus(.restoring) -// let appStoreRestoreFlow = AppStoreRestoreFlow(subscriptionManager: subscriptionManager) let result = await appStoreRestoreFlow.restoreAccountFromPastPurchase() switch result { case .success: diff --git a/DuckDuckGo/Subscription/ViewModel/SubscriptionContainerViewModel.swift b/DuckDuckGo/Subscription/ViewModel/SubscriptionContainerViewModel.swift index 0bfbc9410a..17274f5c48 100644 --- a/DuckDuckGo/Subscription/ViewModel/SubscriptionContainerViewModel.swift +++ b/DuckDuckGo/Subscription/ViewModel/SubscriptionContainerViewModel.swift @@ -38,15 +38,13 @@ final class SubscriptionContainerViewModel: ObservableObject { self.userScript = userScript subFeature.cleanup() self.subFeature = subFeature - let appStoreAccountManagementFlow = DefaultAppStoreAccountManagementFlow(subscriptionManager: subscriptionManager) self.flow = SubscriptionFlowViewModel(origin: origin, userScript: userScript, subFeature: subFeature, subscriptionManager: subscriptionManager) self.restore = SubscriptionRestoreViewModel(userScript: userScript, subFeature: subFeature, - subscriptionManager: subscriptionManager, - appStoreAccountManagementFlow: appStoreAccountManagementFlow) + subscriptionManager: subscriptionManager) self.email = SubscriptionEmailViewModel(userScript: userScript, subFeature: subFeature, subscriptionManager: subscriptionManager) diff --git a/DuckDuckGo/Subscription/ViewModel/SubscriptionEmailViewModel.swift b/DuckDuckGo/Subscription/ViewModel/SubscriptionEmailViewModel.swift index 8fc0ea302e..b185c9d512 100644 --- a/DuckDuckGo/Subscription/ViewModel/SubscriptionEmailViewModel.swift +++ b/DuckDuckGo/Subscription/ViewModel/SubscriptionEmailViewModel.swift @@ -79,6 +79,11 @@ final class SubscriptionEmailViewModel: ObservableObject { webViewModel.url?.forComparison() == subscriptionPurchaseURL.forComparison() } + private var isVerifySubscriptionPage: Bool { + let confirmSubscriptionURL = subscriptionManager.url(for: .baseURL).appendingPathComponent("confirm") + return webViewModel.url?.forComparison() == confirmSubscriptionURL.forComparison() + } + init(userScript: SubscriptionPagesUserScript, subFeature: SubscriptionPagesUseSubscriptionFeature, subscriptionManager: SubscriptionManager) { @@ -132,7 +137,7 @@ final class SubscriptionEmailViewModel: ObservableObject { let addEmailToSubscriptionURL = subscriptionManager.url(for: .addEmail) let manageSubscriptionEmailURL = subscriptionManager.url(for: .manageEmail) emailURL = accountManager.email == nil ? addEmailToSubscriptionURL : manageSubscriptionEmailURL - state.viewTitle = accountManager.email == nil ? UserText.subscriptionRestoreAddEmailTitle : UserText.subscriptionManageEmailTitle + state.viewTitle = accountManager.email == nil ? UserText.subscriptionRestoreAddEmailTitle : UserText.subscriptionEditEmailTitle // Also we assume subscription requires managing, and not activation state.managingSubscriptionEmail = true @@ -224,15 +229,13 @@ final class SubscriptionEmailViewModel: ObservableObject { private func updateBackButton(canNavigateBack: Bool) { // If the view is not Activation Success, or Welcome page, allow WebView Back Navigation - if !isWelcomePageOrSuccessPage { + if !isWelcomePageOrSuccessPage && !isVerifySubscriptionPage { self.state.canNavigateBack = canNavigateBack self.state.backButtonTitle = UserText.backButtonTitle } else { self.state.canNavigateBack = false self.state.backButtonTitle = UserText.settingsTitle } - - } // MARK: - diff --git a/DuckDuckGo/Subscription/ViewModel/SubscriptionRestoreViewModel.swift b/DuckDuckGo/Subscription/ViewModel/SubscriptionRestoreViewModel.swift index f6fb943634..0c178c8257 100644 --- a/DuckDuckGo/Subscription/ViewModel/SubscriptionRestoreViewModel.swift +++ b/DuckDuckGo/Subscription/ViewModel/SubscriptionRestoreViewModel.swift @@ -30,7 +30,6 @@ final class SubscriptionRestoreViewModel: ObservableObject { let subFeature: SubscriptionPagesUseSubscriptionFeature let subscriptionManager: SubscriptionManager var accountManager: AccountManager { subscriptionManager.accountManager } - let appStoreAccountManagementFlow: AppStoreAccountManagementFlow private var cancellables = Set() @@ -39,7 +38,6 @@ final class SubscriptionRestoreViewModel: ObservableObject { } struct State { - var isAddingDevice: Bool = false var transactionStatus: SubscriptionTransactionStatus = .idle var activationResult: SubscriptionActivationResult = .unknown var subscriptionEmail: String? @@ -60,68 +58,28 @@ final class SubscriptionRestoreViewModel: ObservableObject { init(userScript: SubscriptionPagesUserScript, subFeature: SubscriptionPagesUseSubscriptionFeature, subscriptionManager: SubscriptionManager, - appStoreAccountManagementFlow: AppStoreAccountManagementFlow, isAddingDevice: Bool = false) { self.userScript = userScript self.subFeature = subFeature self.subscriptionManager = subscriptionManager - self.appStoreAccountManagementFlow = appStoreAccountManagementFlow - self.state.isAddingDevice = false } func onAppear() { DispatchQueue.main.async { self.resetState() } - Task { await setupContent() } } func onFirstAppear() async { - Pixel.fire(pixel: .privacyProSettingsAddDevice) await setupTransactionObserver() - await refreshToken() } private func cleanUp() { cancellables.removeAll() } - private func refreshToken() async { - if state.isAddingDevice { - await appStoreAccountManagementFlow.refreshAuthTokenIfNeeded() - } - } - - private func setupContent() async { - if state.isAddingDevice { - DispatchQueue.main.async { - self.state.isLoading = true - } - - guard let token = accountManager.accessToken else { return } - switch await accountManager.fetchAccountDetails(with: token) { - case .success(let details): - DispatchQueue.main.async { - self.state.subscriptionEmail = details.email - self.state.isLoading = false - self.state.viewTitle = UserText.subscriptionAddDeviceTitle - } - default: - DispatchQueue.main.async { - self.state.viewTitle = UserText.subscriptionActivate - self.state.isLoading = false - } - } - } - } - @MainActor private func resetState() { - state.isAddingDevice = false - if accountManager.isUserAuthenticated { - state.isAddingDevice = true - } - state.isShowingActivationFlow = false state.shouldShowPlans = false state.isShowingWelcomePage = false diff --git a/DuckDuckGo/Subscription/ViewModel/SubscriptionSettingsViewModel.swift b/DuckDuckGo/Subscription/ViewModel/SubscriptionSettingsViewModel.swift index bb67efca93..61afacec98 100644 --- a/DuckDuckGo/Subscription/ViewModel/SubscriptionSettingsViewModel.swift +++ b/DuckDuckGo/Subscription/ViewModel/SubscriptionSettingsViewModel.swift @@ -27,21 +27,22 @@ import Core final class SubscriptionSettingsViewModel: ObservableObject { private let subscriptionManager: SubscriptionManager - private var subscriptionUpdateTimer: Timer? private var signOutObserver: Any? private var externalAllowedDomains = ["stripe.com"] struct State { var subscriptionDetails: String = "" - var subscriptionType: String = "" + var subscriptionEmail: String? var isShowingRemovalNotice: Bool = false var shouldDismissView: Bool = false var isShowingGoogleView: Bool = false var isShowingFAQView: Bool = false + var isShowingLearnMoreView: Bool = false var subscriptionInfo: Subscription? var isLoadingSubscriptionInfo: Bool = false - + var isLoadingEmailInfo: Bool = false + // Used to display stripe WebUI var stripeViewModel: SubscriptionExternalLinkViewModel? var isShowingStripeView: Bool = false @@ -51,9 +52,11 @@ final class SubscriptionSettingsViewModel: ObservableObject { // Used to display the FAQ WebUI var faqViewModel: SubscriptionExternalLinkViewModel + var learnMoreViewModel: SubscriptionExternalLinkViewModel - init(faqURL: URL) { + init(faqURL: URL, learnMoreURL: URL) { self.faqViewModel = SubscriptionExternalLinkViewModel(url: faqURL) + self.learnMoreViewModel = SubscriptionExternalLinkViewModel(url: learnMoreURL) } } @@ -67,56 +70,111 @@ final class SubscriptionSettingsViewModel: ObservableObject { init(subscriptionManager: SubscriptionManager = AppDependencyProvider.shared.subscriptionManager) { self.subscriptionManager = subscriptionManager let subscriptionFAQURL = subscriptionManager.url(for: .faq) - self.state = State(faqURL: subscriptionFAQURL) + let learnMoreURL = subscriptionFAQURL.appendingPathComponent("adding-email") + self.state = State(faqURL: subscriptionFAQURL, learnMoreURL: learnMoreURL) - setupSubscriptionUpdater() setupNotificationObservers() } private var dateFormatter: DateFormatter = { let formatter = DateFormatter() - formatter.dateFormat = "MMM dd, yyyy" + formatter.dateFormat = "MMMM dd, yyyy" return formatter }() func onFirstAppear() { - self.fetchAndUpdateSubscriptionDetails(cachePolicy: .returnCacheDataElseLoad) - } - - private func fetchAndUpdateSubscriptionDetails(cachePolicy: APICachePolicy = .returnCacheDataElseLoad, - loadingIndicator: Bool = true) { Task { - if loadingIndicator { displayLoader(true) } - guard let token = self.subscriptionManager.accountManager.accessToken else { return } - let subscriptionResult = await self.subscriptionManager.subscriptionEndpointService.getSubscription(accessToken: token, - cachePolicy: cachePolicy) - switch subscriptionResult { - case .success(let subscription): - DispatchQueue.main.async { - self.state.subscriptionInfo = subscription - if loadingIndicator { self.displayLoader(false) } - } - await updateSubscriptionsStatusMessage(status: subscription.status, - date: subscription.expiresOrRenewsAt, - product: subscription.productId, - billingPeriod: subscription.billingPeriod) - default: - DispatchQueue.main.async { - if loadingIndicator { self.displayLoader(true) } - self.showConnectionError(true) - } - - subscriptionUpdateTimer?.invalidate() + // Load initial state from the cache + async let loadedEmailFromCache = await self.fetchAndUpdateAccountEmail(cachePolicy: .returnCacheDataDontLoad, + loadingIndicator: false) + async let loadedSubscriptionFromCache = await self.fetchAndUpdateSubscriptionDetails(cachePolicy: .returnCacheDataDontLoad, + loadingIndicator: false) + let (hasLoadedEmailFromCache, hasLoadedSubscriptionFromCache) = await (loadedEmailFromCache, loadedSubscriptionFromCache) + + // Reload remote subscription and email state + async let reloadedEmail = await self.fetchAndUpdateAccountEmail(cachePolicy: .reloadIgnoringLocalCacheData, + loadingIndicator: !hasLoadedEmailFromCache) + async let reloadedSubscription = await self.fetchAndUpdateSubscriptionDetails(cachePolicy: .reloadIgnoringLocalCacheData, + loadingIndicator: !hasLoadedSubscriptionFromCache) + let (hasReloadedEmail, hasReloadedSubscription) = await (reloadedEmail, reloadedSubscription) + + // In case any fetch fails show an error + if !hasReloadedEmail || !hasReloadedSubscription { + self.showConnectionError(true) } } } - - private func displayLoader(_ show: Bool) { + + private func fetchAndUpdateSubscriptionDetails(cachePolicy: APICachePolicy, loadingIndicator: Bool) async -> Bool { + guard let token = self.subscriptionManager.accountManager.accessToken else { return false } + + if loadingIndicator { displaySubscriptionLoader(true) } + let subscriptionResult = await self.subscriptionManager.subscriptionEndpointService.getSubscription(accessToken: token, + cachePolicy: cachePolicy) + switch subscriptionResult { + case .success(let subscription): + DispatchQueue.main.async { + self.state.subscriptionInfo = subscription + if loadingIndicator { self.displaySubscriptionLoader(false) } + } + await updateSubscriptionsStatusMessage(status: subscription.status, + date: subscription.expiresOrRenewsAt, + product: subscription.productId, + billingPeriod: subscription.billingPeriod) + return true + default: + DispatchQueue.main.async { + if loadingIndicator { self.displaySubscriptionLoader(true) } + } + return false + } + } + + func fetchAndUpdateAccountEmail(cachePolicy: APICachePolicy = .returnCacheDataElseLoad, loadingIndicator: Bool) async -> Bool { + guard let token = self.subscriptionManager.accountManager.accessToken else { return false } + + switch cachePolicy { + case .returnCacheDataDontLoad, .returnCacheDataElseLoad: + self.state.subscriptionEmail = self.subscriptionManager.accountManager.email + return true + case .reloadIgnoringLocalCacheData: + break + } + + if loadingIndicator { displayEmailLoader(true) } + switch await self.subscriptionManager.accountManager.fetchAccountDetails(with: token) { + case .success(let details): + DispatchQueue.main.async { + self.state.subscriptionEmail = details.email + if loadingIndicator { self.displayEmailLoader(false) } + } + + // If fetched email is different then update accountManager + if details.email != subscriptionManager.accountManager.email { + let externalID = subscriptionManager.accountManager.externalID + subscriptionManager.accountManager.storeAccount(token: token, email: details.email, externalID: externalID) + } + return true + default: + DispatchQueue.main.async { + if loadingIndicator { self.displayEmailLoader(true) } + } + return false + } + } + + private func displaySubscriptionLoader(_ show: Bool) { DispatchQueue.main.async { self.state.isLoadingSubscriptionInfo = show } } - + + private func displayEmailLoader(_ show: Bool) { + DispatchQueue.main.async { + self.state.isLoadingEmailInfo = show + } + } + func manageSubscription() { switch state.subscriptionInfo?.platform { case .apple: @@ -140,25 +198,18 @@ final class SubscriptionSettingsViewModel: ObservableObject { } } - // Re-fetch subscription from server ignoring cache - // This ensure that if the user re-subscribed or changed plan on the Apple view, state is updated - private func setupSubscriptionUpdater() { - subscriptionUpdateTimer = Timer.scheduledTimer(withTimeInterval: 8.0, repeats: true) { [weak self] _ in - guard let strongSelf = self else { return } - strongSelf.fetchAndUpdateSubscriptionDetails(cachePolicy: .reloadIgnoringLocalCacheData, loadingIndicator: false) - } - } - @MainActor private func updateSubscriptionsStatusMessage(status: Subscription.Status, date: Date, product: String, billingPeriod: Subscription.BillingPeriod) { + let billingPeriod = billingPeriod == .monthly ? UserText.subscriptionMonthlyBillingPeriod : UserText.subscriptionAnnualBillingPeriod let date = dateFormatter.string(from: date) - let expiredStates: [Subscription.Status] = [.expired, .inactive] - if expiredStates.contains(status) { + + switch status { + case .autoRenewable: + state.subscriptionDetails = UserText.renewingSubscriptionInfo(billingPeriod: billingPeriod, renewalDate: date) + case .expired, .inactive: state.subscriptionDetails = UserText.expiredSubscriptionInfo(expiration: date) - } else { - let statusString = (status == .autoRenewable) ? UserText.subscriptionRenews : UserText.subscriptionExpires - state.subscriptionDetails = UserText.subscriptionInfo(status: statusString, expiration: date) - state.subscriptionType = billingPeriod == .monthly ? UserText.subscriptionMonthly : UserText.subscriptionAnnual + default: + state.subscriptionDetails = UserText.expiringSubscriptionInfo(billingPeriod: billingPeriod, expiryDate: date) } } @@ -192,7 +243,13 @@ final class SubscriptionSettingsViewModel: ObservableObject { state.isShowingFAQView = value } } - + + func displayLearnMoreView(_ value: Bool) { + if value != state.isShowingLearnMoreView { + state.isShowingLearnMoreView = value + } + } + func showConnectionError(_ value: Bool) { if value != state.isShowingConnectionError { DispatchQueue.main.async { @@ -248,7 +305,6 @@ final class SubscriptionSettingsViewModel: ObservableObject { } deinit { - subscriptionUpdateTimer?.invalidate() signOutObserver = nil } } diff --git a/DuckDuckGo/Subscription/Views/SubscriptionContainerView.swift b/DuckDuckGo/Subscription/Views/SubscriptionContainerView.swift index 5f56f8d7d6..61b38b3292 100644 --- a/DuckDuckGo/Subscription/Views/SubscriptionContainerView.swift +++ b/DuckDuckGo/Subscription/Views/SubscriptionContainerView.swift @@ -24,7 +24,7 @@ import SwiftUI struct SubscriptionContainerView: View { enum CurrentView { - case subscribe, restore + case subscribe, restore, email } @Environment(\.dismiss) var dismiss @@ -54,6 +54,8 @@ struct SubscriptionContainerView: View { SubscriptionRestoreView(viewModel: restoreViewModel, emailViewModel: emailViewModel, currentView: $currentViewState).environmentObject(subscriptionNavigationCoordinator) + case .email: + SubscriptionEmailView(viewModel: emailViewModel) } } } diff --git a/DuckDuckGo/Subscription/Views/SubscriptionContainerViewFactory.swift b/DuckDuckGo/Subscription/Views/SubscriptionContainerViewFactory.swift index d3fc8369ad..32c1423204 100644 --- a/DuckDuckGo/Subscription/Views/SubscriptionContainerViewFactory.swift +++ b/DuckDuckGo/Subscription/Views/SubscriptionContainerViewFactory.swift @@ -27,6 +27,7 @@ enum SubscriptionContainerViewFactory { let appStoreRestoreFlow = DefaultAppStoreRestoreFlow(subscriptionManager: subscriptionManager) let appStorePurchaseFlow = DefaultAppStorePurchaseFlow(subscriptionManager: subscriptionManager, appStoreRestoreFlow: appStoreRestoreFlow) + let appStoreAccountManagementFlow = DefaultAppStoreAccountManagementFlow(subscriptionManager: subscriptionManager) let viewModel = SubscriptionContainerViewModel( subscriptionManager: subscriptionManager, @@ -35,7 +36,8 @@ enum SubscriptionContainerViewFactory { subFeature: SubscriptionPagesUseSubscriptionFeature(subscriptionManager: subscriptionManager, subscriptionAttributionOrigin: origin, appStorePurchaseFlow: appStorePurchaseFlow, - appStoreRestoreFlow: appStoreRestoreFlow) + appStoreRestoreFlow: appStoreRestoreFlow, + appStoreAccountManagementFlow: appStoreAccountManagementFlow) ) return SubscriptionContainerView(currentView: .subscribe, viewModel: viewModel) .environmentObject(navigationCoordinator) @@ -45,6 +47,7 @@ enum SubscriptionContainerViewFactory { let appStoreRestoreFlow = DefaultAppStoreRestoreFlow(subscriptionManager: subscriptionManager) let appStorePurchaseFlow = DefaultAppStorePurchaseFlow(subscriptionManager: subscriptionManager, appStoreRestoreFlow: appStoreRestoreFlow) + let appStoreAccountManagementFlow = DefaultAppStoreAccountManagementFlow(subscriptionManager: subscriptionManager) let viewModel = SubscriptionContainerViewModel( subscriptionManager: subscriptionManager, @@ -53,10 +56,34 @@ enum SubscriptionContainerViewFactory { subFeature: SubscriptionPagesUseSubscriptionFeature(subscriptionManager: subscriptionManager, subscriptionAttributionOrigin: nil, appStorePurchaseFlow: appStorePurchaseFlow, - appStoreRestoreFlow: appStoreRestoreFlow) + appStoreRestoreFlow: appStoreRestoreFlow, + appStoreAccountManagementFlow: appStoreAccountManagementFlow) ) return SubscriptionContainerView(currentView: .restore, viewModel: viewModel) .environmentObject(navigationCoordinator) } + static func makeEmailFlow(navigationCoordinator: SubscriptionNavigationCoordinator, + subscriptionManager: SubscriptionManager, + onDisappear: @escaping () -> Void) -> some View { + let appStoreRestoreFlow = DefaultAppStoreRestoreFlow(subscriptionManager: subscriptionManager) + let appStorePurchaseFlow = DefaultAppStorePurchaseFlow(subscriptionManager: subscriptionManager, + appStoreRestoreFlow: appStoreRestoreFlow) + let appStoreAccountManagementFlow = DefaultAppStoreAccountManagementFlow(subscriptionManager: subscriptionManager) + + let viewModel = SubscriptionContainerViewModel( + subscriptionManager: subscriptionManager, + origin: nil, + userScript: SubscriptionPagesUserScript(), + subFeature: SubscriptionPagesUseSubscriptionFeature(subscriptionManager: subscriptionManager, + subscriptionAttributionOrigin: nil, + appStorePurchaseFlow: appStorePurchaseFlow, + appStoreRestoreFlow: appStoreRestoreFlow, + appStoreAccountManagementFlow: appStoreAccountManagementFlow) + ) + return SubscriptionContainerView(currentView: .email, viewModel: viewModel) + .environmentObject(navigationCoordinator) + .onDisappear(perform: { onDisappear() }) + } + } diff --git a/DuckDuckGo/Subscription/Views/SubscriptionRestoreView.swift b/DuckDuckGo/Subscription/Views/SubscriptionRestoreView.swift index 648cc3b925..3368a036b0 100644 --- a/DuckDuckGo/Subscription/Views/SubscriptionRestoreView.swift +++ b/DuckDuckGo/Subscription/Views/SubscriptionRestoreView.swift @@ -23,7 +23,6 @@ import DesignResourcesKit import Core @available(iOS 15.0, *) -// swiftlint:disable type_body_length struct SubscriptionRestoreView: View { @Environment(\.dismiss) var dismiss @@ -65,18 +64,11 @@ struct SubscriptionRestoreView: View { } var body: some View { - if viewModel.state.isAddingDevice { - ZStack { - baseView - } - } else { - ZStack { - baseView - - if viewModel.state.transactionStatus != .idle { - PurchaseInProgressView(status: getTransactionStatus()) - } - + ZStack { + baseView + + if viewModel.state.transactionStatus != .idle { + PurchaseInProgressView(status: getTransactionStatus()) } } } @@ -153,8 +145,6 @@ struct SubscriptionRestoreView: View { .onAppear { viewModel.onAppear() } - - } // MARK: - @@ -182,38 +172,15 @@ struct SubscriptionRestoreView: View { if !viewModel.state.isLoading { VStack(alignment: .leading) { - if !viewModel.state.isAddingDevice { - Text(UserText.subscriptionActivateEmailDescription) - .daxSubheadRegular() - .foregroundColor(Color(designSystemColor: .textSecondary)) - getCellButton(buttonText: UserText.subscriptionActivateEmailButton, - action: { - DailyPixel.fireDailyAndCount(pixel: .privacyProRestorePurchaseEmailStart) - DailyPixel.fire(pixel: .privacyProWelcomeAddDevice) - viewModel.showActivationFlow(true) - }) - } else if viewModel.state.subscriptionEmail == nil { - Text(UserText.subscriptionAddDeviceEmailDescription) - .daxSubheadRegular() - .foregroundColor(Color(designSystemColor: .textSecondary)) - getCellButton(buttonText: UserText.subscriptionRestoreAddEmailButton, - action: { - Pixel.fire(pixel: .privacyProAddDeviceEnterEmail, debounce: 1) - viewModel.showActivationFlow(true) - }) - } else { - Text(viewModel.state.subscriptionEmail ?? "").daxSubheadSemibold() - Text(UserText.subscriptionManageEmailDescription) - .daxSubheadRegular() - .foregroundColor(Color(designSystemColor: .textSecondary)) - HStack { - getCellButton(buttonText: UserText.subscriptionManageEmailButton, - action: { - Pixel.fire(pixel: .privacyProSubscriptionManagementEmail, debounce: 1) - viewModel.showActivationFlow(true) - }) - } - } + Text(UserText.subscriptionActivateEmailDescription) + .daxSubheadRegular() + .foregroundColor(Color(designSystemColor: .textSecondary)) + getCellButton(buttonText: UserText.subscriptionActivateEmailButton, + action: { + DailyPixel.fireDailyAndCount(pixel: .privacyProRestorePurchaseEmailStart) + DailyPixel.fire(pixel: .privacyProWelcomeAddDevice) + viewModel.showActivationFlow(true) + }) } } else { SwiftUI.ProgressView() @@ -258,11 +225,11 @@ struct SubscriptionRestoreView: View { private var headerView: some View { VStack(spacing: Constants.headerLineSpacing) { Image(Constants.heroImage) - Text(viewModel.state.isAddingDevice ? UserText.subscriptionAddDeviceHeaderTitle : UserText.subscriptionActivateTitle) + Text(UserText.subscriptionActivateTitle) .daxHeadline() .multilineTextAlignment(.center) .foregroundColor(Color(designSystemColor: .textPrimary)) - Text(viewModel.state.isAddingDevice ? UserText.subscriptionAddDeviceDescription : UserText.subscriptionActivateHeaderDescription) + Text(UserText.subscriptionActivateHeaderDescription) .daxFootnoteRegular() .foregroundColor(Color(designSystemColor: .textSecondary)) .multilineTextAlignment(.center) @@ -272,22 +239,20 @@ struct SubscriptionRestoreView: View { @ViewBuilder private var footerView: some View { - if !viewModel.state.isAddingDevice { - VStack(alignment: .leading, spacing: Constants.footerLineSpacing) { - Text(UserText.subscriptionActivateDescription) - .daxFootnoteRegular() - .foregroundColor(Color(designSystemColor: .textSecondary)) - Button(action: { - viewModel.restoreAppstoreTransaction() - }, label: { - Text(UserText.subscriptionRestoreAppleID) - .daxFootnoteSemibold() - .foregroundColor(Color(designSystemColor: .accent)) - }) - } + VStack(alignment: .leading, spacing: Constants.footerLineSpacing) { + Text(UserText.subscriptionActivateDescription) + .daxFootnoteRegular() + .foregroundColor(Color(designSystemColor: .textSecondary)) + Button(action: { + viewModel.restoreAppstoreTransaction() + }, label: { + Text(UserText.subscriptionRestoreAppleID) + .daxFootnoteSemibold() + .foregroundColor(Color(designSystemColor: .accent)) + }) } } - + private func getAlert() -> Alert { switch viewModel.state.activationResult { case .activated: @@ -336,4 +301,3 @@ struct SubscriptionRestoreView: View { } } -// swiftlint:enable type_body_length diff --git a/DuckDuckGo/Subscription/Views/SubscriptionSettingsView.swift b/DuckDuckGo/Subscription/Views/SubscriptionSettingsView.swift index 679bae8284..76dcf7f7d7 100644 --- a/DuckDuckGo/Subscription/Views/SubscriptionSettingsView.swift +++ b/DuckDuckGo/Subscription/Views/SubscriptionSettingsView.swift @@ -34,10 +34,10 @@ struct SubscriptionSettingsView: View { @State var isShowingGoogleView = false @State var isShowingRemovalNotice = false @State var isShowingFAQView = false - @State var isShowingRestoreView = false + @State var isShowingLearnMoreView = false + @State var isShowingEmailView = false @State var isShowingConnectionError = false - @State var isLoading = false - + enum Constants { static let alertIcon = "Exclamation-Color-16" } @@ -54,35 +54,66 @@ struct SubscriptionSettingsView: View { private var headerSection: some View { Section { - let active = viewModel.state.subscriptionInfo?.isActive ?? false + let isExpired = !(viewModel.state.subscriptionInfo?.isActive ?? false) VStack(alignment: .center, spacing: 7) { Image("Privacy-Pro-96x96") Text(UserText.subscriptionTitle).daxTitle2() - if !viewModel.state.isLoadingSubscriptionInfo { - if active { - Text(viewModel.state.subscriptionType).daxHeadline() - } + + if isExpired { HStack { - if !active { Image(Constants.alertIcon) } + Image(Constants.alertIcon) Text(viewModel.state.subscriptionDetails) .daxSubheadRegular() .foregroundColor(Color(designSystemColor: .textSecondary)) } - } else { - SwiftUI.ProgressView() } } } .listRowBackground(Color.clear) .frame(maxWidth: .infinity, alignment: .center) - } - + + private var devicesSection: some View { + Section(header: Text(UserText.subscriptionDevicesSectionHeader), + footer: devicesSectionFooter) { + + if !viewModel.state.isLoadingEmailInfo { + NavigationLink(destination: SubscriptionContainerViewFactory.makeEmailFlow( + navigationCoordinator: subscriptionNavigationCoordinator, + subscriptionManager: AppDependencyProvider.shared.subscriptionManager, + onDisappear: { + Task { await viewModel.fetchAndUpdateAccountEmail(cachePolicy: .reloadIgnoringLocalCacheData, loadingIndicator: false) } + }), + isActive: $isShowingEmailView) { + if let email = viewModel.state.subscriptionEmail { + SettingsCellView(label: UserText.subscriptionEditEmailButton, + subtitle: email) + } else { + SettingsCellView(label: UserText.subscriptionAddEmailButton) + } + }.isDetailLink(false) + } else { + SwiftUI.ProgressView() + } + } + } + + private var devicesSectionFooter: some View { + let hasEmail = !(viewModel.state.subscriptionEmail ?? "").isEmpty + let footerText = hasEmail ? UserText.subscriptionDevicesSectionWithEmailFooter : UserText.subscriptionDevicesSectionNoEmailFooter + return Text(.init("\(footerText)")) // required to parse markdown formatting + .environment(\.openURL, OpenURLAction { _ in + viewModel.displayLearnMoreView(true) + return .handled + }) + } + private var manageSection: some View { - Section(header: Text(UserText.subscriptionManageTitle)) { + Section(header: Text(UserText.subscriptionManageTitle), + footer: manageSectionFooter) { let active = viewModel.state.subscriptionInfo?.isActive ?? false SettingsCustomCell(content: { - + if !viewModel.state.isLoadingSubscriptionInfo { if active { Text(UserText.subscriptionChangePlan) @@ -115,21 +146,6 @@ struct SubscriptionSettingsView: View { SubscriptionExternalLinkView(viewModel: stripeViewModel, title: UserText.subscriptionManagePlan) } } - } - } - - private var devicesSection: some View { - Section(header: Text(UserText.subscriptionManageDevices)) { - - NavigationLink(destination: SubscriptionContainerViewFactory.makeRestoreFlow( - navigationCoordinator: subscriptionNavigationCoordinator, - subscriptionManager: AppDependencyProvider.shared.subscriptionManager), - isActive: $isShowingRestoreView) { - SettingsCustomCell(content: { - Text(UserText.subscriptionAddDeviceButton) - .daxBodyRegular() - }) - }.isDetailLink(false) SettingsCustomCell(content: { Text(UserText.subscriptionRemoveFromDevice) @@ -137,10 +153,20 @@ struct SubscriptionSettingsView: View { .foregroundColor(Color.init(designSystemColor: .accent))}, action: { viewModel.displayRemovalNotice(true) }, isButton: true) - } } - + + private var manageSectionFooter: some View { + let isExpired = !(viewModel.state.subscriptionInfo?.isActive ?? false) + return Group { + if isExpired { + EmptyView() + } else { + Text(viewModel.state.subscriptionDetails) + } + } + } + @ViewBuilder var helpSection: some View { Section(header: Text(UserText.subscriptionHelpAndSupport), footer: Text(UserText.subscriptionFAQFooter)) { @@ -167,7 +193,6 @@ struct SubscriptionSettingsView: View { List { headerSection - manageSection devicesSection .alert(isPresented: $isShowingRemovalNotice) { Alert( @@ -182,6 +207,7 @@ struct SubscriptionSettingsView: View { } ) } + manageSection helpSection } @@ -218,14 +244,22 @@ struct SubscriptionSettingsView: View { viewModel.displayRemovalNotice(value) } - // Removal Notice + // FAQ .onChange(of: viewModel.state.isShowingFAQView) { value in isShowingFAQView = value } .onChange(of: isShowingFAQView) { value in viewModel.displayFAQView(value) } - + + // Learn More + .onChange(of: viewModel.state.isShowingLearnMoreView) { value in + isShowingLearnMoreView = value + } + .onChange(of: isShowingLearnMoreView) { value in + viewModel.displayLearnMoreView(value) + } + // Connection Error .onChange(of: viewModel.state.isShowingConnectionError) { value in isShowingConnectionError = value @@ -233,11 +267,20 @@ struct SubscriptionSettingsView: View { .onChange(of: isShowingConnectionError) { value in viewModel.showConnectionError(value) } - + + .onChange(of: isShowingEmailView) { value in + if value { + if let email = viewModel.state.subscriptionEmail, !email.isEmpty { + Pixel.fire(pixel: .privacyProSubscriptionManagementEmail, debounce: 1) + } else { + Pixel.fire(pixel: .privacyProAddDeviceEnterEmail, debounce: 1) + } + } + } .onReceive(subscriptionNavigationCoordinator.$shouldPopToSubscriptionSettings) { shouldDismiss in if shouldDismiss { - isShowingRestoreView = false + isShowingEmailView = false } } @@ -254,7 +297,11 @@ struct SubscriptionSettingsView: View { .sheet(isPresented: $isShowingFAQView, content: { SubscriptionExternalLinkView(viewModel: viewModel.state.faqViewModel, title: UserText.subscriptionFAQ) }) - + + .sheet(isPresented: $isShowingLearnMoreView, content: { + SubscriptionExternalLinkView(viewModel: viewModel.state.learnMoreViewModel, title: UserText.subscriptionFAQ) + }) + .onFirstAppear { viewModel.onFirstAppear() } diff --git a/DuckDuckGo/TabViewController.swift b/DuckDuckGo/TabViewController.swift index 2c5e3fb563..f20df4c932 100644 --- a/DuckDuckGo/TabViewController.swift +++ b/DuckDuckGo/TabViewController.swift @@ -348,9 +348,8 @@ class TabViewController: UIViewController { decorate() addTextSizeObserver() subscribeToEmailProtectionSignOutNotification() - registerForDownloadsNotifications() - + if #available(iOS 16.4, *) { registerForInspectableWebViewNotifications() } @@ -359,7 +358,13 @@ class TabViewController: UIViewController { observeNetPConnectionStatusChanges() #endif } - + + private func configureDuckPlayerUserScripts() { + userScripts?.youtubeOverlayScript?.webView = webView + userScripts?.youtubePlayerUserScript?.webView = webView + } + + @available(iOS 16.4, *) private func registerForInspectableWebViewNotifications() { NotificationCenter.default.addObserver(self, @@ -660,7 +665,7 @@ class TabViewController: UIViewController { let url, url.isYoutubeVideo, appSettings.duckPlayerMode == .enabled { - handler.handleRedirect(url: url, webView: webView) + handler.handleURLChange(url: url, webView: webView) } } } @@ -713,7 +718,11 @@ class TabViewController: UIViewController { public func reload() { updateContentMode() cachedRuntimeConfigurationForDomain = [:] - webView.reload() + if let url = webView.url, url.isDuckPlayer, let handler = youtubeNavigationHandler { + handler.handleReload(webView: webView) + } else { + webView.reload() + } privacyDashboard?.dismiss(animated: true) } @@ -725,7 +734,7 @@ class TabViewController: UIViewController { dismissJSAlertIfNeeded() if let url = url, url.isDuckPlayer, let handler = youtubeNavigationHandler { - handler.goBack(webView: webView) + handler.handleGoBack(webView: webView) chromeDelegate?.omniBar.resignFirstResponder() return } @@ -1647,7 +1656,7 @@ extension TabViewController: WKNavigationDelegate { if let handler = youtubeNavigationHandler, url.isYoutubeVideo, appSettings.duckPlayerMode == .enabled { - handler.handleRedirect(navigationAction, completion: completion, webView: webView) + handler.handleDecidePolicyFor(navigationAction, completion: completion, webView: webView) return } @@ -2316,12 +2325,14 @@ extension TabViewController: UserContentControllerDelegate { userScripts.textSizeUserScript.textSizeAdjustmentInPercents = appSettings.textSize userScripts.loginFormDetectionScript?.delegate = self userScripts.autoconsentUserScript.delegate = self - + userScripts.youtubeOverlayScript?.webView = webView + userScripts.youtubePlayerUserScript?.webView = webView + performanceMetrics = PerformanceMetricsSubfeature(targetWebview: webView) userScripts.contentScopeUserScriptIsolated.registerSubfeature(delegate: performanceMetrics!) adClickAttributionLogic.onRulesChanged(latestRules: ContentBlocking.shared.contentBlockingManager.currentRules) - + let tdsKey = DefaultContentBlockerRulesListsSource.Constants.trackerDataSetRulesListName let notificationsTriggeringReload = [ PreserveLogins.Notifications.loginDetectionStateChanged, diff --git a/DuckDuckGo/TabViewControllerBrowsingMenuExtension.swift b/DuckDuckGo/TabViewControllerBrowsingMenuExtension.swift index 6ae62b1a5e..c9cb80dd49 100644 --- a/DuckDuckGo/TabViewControllerBrowsingMenuExtension.swift +++ b/DuckDuckGo/TabViewControllerBrowsingMenuExtension.swift @@ -419,8 +419,8 @@ extension TabViewController { } private func onToggleProtectionAction(forDomain domain: String, isProtected: Bool) { - let config = ContentBlocking.shared.privacyConfigurationManager.privacyConfig - if isProtected && ToggleReportsFeature(privacyConfiguration: config).isEnabled { + let manager = ToggleReportsManager(feature: ToggleReportsFeature(manager: ContentBlocking.shared.privacyConfigurationManager)) + if isProtected && manager.shouldShowToggleReport { delegate?.tab(self, didRequestToggleReportWithCompletionHandler: { [weak self] didSendReport in self?.togglePrivacyProtection(domain: domain, didSendReport: didSendReport) }) diff --git a/DuckDuckGo/ToggleExpandButtonView.swift b/DuckDuckGo/ToggleExpandButtonView.swift new file mode 100644 index 0000000000..376a1e52b1 --- /dev/null +++ b/DuckDuckGo/ToggleExpandButtonView.swift @@ -0,0 +1,35 @@ +// +// ToggleExpandButtonView.swift +// DuckDuckGo +// +// Copyright © 2024 DuckDuckGo. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +import SwiftUI +import DuckUI + +struct ToggleExpandButtonView: View { + + let isIndicatingExpand: Bool + + var body: some View { + Text(isIndicatingExpand ? "Show more" : "Show less") + .daxCaption() + } +} + +#Preview { + return ToggleExpandButtonView(isIndicatingExpand: true) +} diff --git a/DuckDuckGo/UserScripts.swift b/DuckDuckGo/UserScripts.swift index a8d6e2697a..3a5f3ae18b 100644 --- a/DuckDuckGo/UserScripts.swift +++ b/DuckDuckGo/UserScripts.swift @@ -108,5 +108,5 @@ final class UserScripts: UserScriptsProvider { return wkUserScripts } } - + } diff --git a/DuckDuckGo/UserText.swift b/DuckDuckGo/UserText.swift index 17849eb004..c9f1a16264 100644 --- a/DuckDuckGo/UserText.swift +++ b/DuckDuckGo/UserText.swift @@ -1030,13 +1030,20 @@ But if you *do* want a peek under the hood, you can find more information about public static let subscriptionTitle = NSLocalizedString("subscription.title", value: "Privacy Pro", comment: "Navigation bar Title for subscriptions") public static let subscriptionCloseButton = NSLocalizedString("subscription.close", value: "Close", comment: "Navigation Button for closing subscription view") - static func subscriptionInfo(status: String, expiration: String) -> String { - let localized = NSLocalizedString("subscription.subscription.active.caption", - value: "Your subscription %@ on %@", - comment: "Subscription Expiration Data. This reads as 'Your subscription (renews or expires) on (date)'") - return String(format: localized, status, expiration) + static func renewingSubscriptionInfo(billingPeriod: String, renewalDate: String) -> String { + let localized = NSLocalizedString("subscription.subscription.renewing.caption", + value: "Your %@ subscription renews on %@.", + comment: "Subscription renewal info. This reads as 'Your (monthly or annual) subscription renews on (date)'") + return String(format: localized, billingPeriod, renewalDate) } - + + static func expiringSubscriptionInfo(billingPeriod: String, expiryDate: String) -> String { + let localized = NSLocalizedString("subscription.subscription.expiring.caption", + value: "Your %@ subscription expires on %@.", + comment: "Subscription expiration info. This reads as 'Your (monthly or annual) subscription expires on (date)'") + return String(format: localized, billingPeriod, expiryDate) + } + static func expiredSubscriptionInfo(expiration: String) -> String { let localized = NSLocalizedString("subscription.subscription.expired.caption", value: "Your subscription expired on %@", @@ -1044,20 +1051,19 @@ But if you *do* want a peek under the hood, you can find more information about return String(format: localized, expiration) } - public static let subscriptionRenews = NSLocalizedString("subscription.renews", value: "renews", comment: "text for renewal string") - public static let subscriptionExpires = NSLocalizedString("subscription.expires", value: "expires", comment: "text for expiration string") - public static let subscriptionMonthly = NSLocalizedString("subscription.monthly", value: "Monthly Subscription", comment: "Subscription type") - public static let subscriptionAnnual = NSLocalizedString("subscription.annual", value: "Annual Subscription", comment: "Subscription type") - - public static let subscriptionManageDevices = NSLocalizedString("subscription.manage.devices", value: "Manage Devices", comment: "Header for the device management section") - public static let subscriptionAddDeviceButton = NSLocalizedString("subscription.add.device.button", value: "Add to Another Device", comment: "Add to another device button") + public static let subscriptionMonthlyBillingPeriod = NSLocalizedString("subscription.billing.period.monthly", value: "monthly", comment: "Subscription monthly billing period type") + public static let subscriptionAnnualBillingPeriod = NSLocalizedString("subscription.billing.period.annual", value: "annual", comment: "Subscription annual billing period type") + + public static let subscriptionDevicesSectionHeader = NSLocalizedString("subscription.devices.header", value: "Activate on Other Devices", comment: "Header for section for activating subscription on other devices") + public static let subscriptionDevicesSectionNoEmailFooter = NSLocalizedString("subscription.devices.no.email.footer", value: "Add an optional email to your subscription or use your Apple ID to access Privacy Pro on other devices. **[Learn more](https://duckduckgo.com/duckduckgo-help-pages/privacy-pro/adding-email/)**", comment: "Footer for section for activating subscription on other devices when email was not yet added") + public static let subscriptionDevicesSectionWithEmailFooter = NSLocalizedString("subscription.devices.with.email.footer", value: "Use this email to activate your subscription in Settings > Privacy Pro in the DuckDuckGo app on your other devices. **[Learn more](https://duckduckgo.com/duckduckgo-help-pages/privacy-pro/adding-email/)**", comment: "Footer for section for activating subscription on other devices when email is added") public static let subscriptionRemoveFromDevice = NSLocalizedString("subscription.remove.from.device.button", value: "Remove From This Device", comment: "Remove from this device button") public static let subscriptionManageTitle = NSLocalizedString("subscription.manage.title", value: "Subscription", comment: "Header for the subscription section") public static let subscriptionManagePlan = NSLocalizedString("subscription.manage.plan", value: "Manage Plan", comment: "Manage Plan header") - public static let subscriptionChangePlan = NSLocalizedString("subscription.change.plan", value: "Change Plan or Billing", comment: "Change plan or billing title") + public static let subscriptionChangePlan = NSLocalizedString("subscription.change.plan", value: "Update Plan or Cancel", comment: "Change plan or cancel title") public static let subscriptionHelpAndSupport = NSLocalizedString("subscription.help", value: "Help and support", comment: "Help and support Section header") - public static let subscriptionFAQ = NSLocalizedString("subscription.faq", value: "Privacy Pro FAQ", comment: "FAQ Button") - public static let subscriptionFAQFooter = NSLocalizedString("subscription.faq.description", value: "Get answers to frequently asked questions about Privacy Pro in our help pages.", comment: "FAQ Description") + public static let subscriptionFAQ = NSLocalizedString("subscription.faq", value: "FAQs and Support", comment: "FAQ Button") + public static let subscriptionFAQFooter = NSLocalizedString("subscription.faq.description", value: "Get answers to frequently asked questions or contact Privacy Pro support from our help pages.", comment: "FAQ Description") // Remove subscription confirmation public static let subscriptionRemoveFromDeviceConfirmTitle = NSLocalizedString("subscription.remove.from.device.title", value: "Remove from this device?", comment: "Remove from device confirmation dialog title") @@ -1067,7 +1073,6 @@ But if you *do* want a peek under the hood, you can find more information about public static let subscriptionRemovalConfirmation = NSLocalizedString("subscription.cancel.message", value: "Your subscription has been removed from this device.", comment: "Subscription Removal confirmation message") // Subscription Restore - public static let subscriptionActivate = NSLocalizedString("subscription.activate", value: "Activate Subscription", comment: "Subscription Activation Window Title") public static let subscriptionActivateTitle = NSLocalizedString("subscription.activate.title", value: "Activate your subscription on this device", comment: "Subscription Activation Title") public static let subscriptionActivateDescription = NSLocalizedString("subscription.activate.description", value: "Your subscription is automatically available in DuckDuckGo on any device signed in to your Apple ID.", comment: "Subscription Activation Info") public static let subscriptionActivateHeaderDescription = NSLocalizedString("subscription.activate..header.description", value: "Access your Privacy Pro subscription on this device via Apple ID or an email address.", comment: "Subscription Activation Info") @@ -1078,14 +1083,11 @@ But if you *do* want a peek under the hood, you can find more information about public static let subscriptionActivateEmail = NSLocalizedString("subscription.activate.email", value: "Email", comment: "Email option for activation") public static let subscriptionActivateEmailTitle = NSLocalizedString("subscription.activate.email.title", value: "Activate Subscription", comment: "Activate subscription title") public static let subscriptionActivateEmailDescription = NSLocalizedString("subscription.activate.email.description", value: "Use your email to activate your subscription on this device.", comment: "Description for Email activation") - public static let subscriptionAddDeviceEmailDescription = NSLocalizedString("subscription.addDevice.email.description", value: "Add an email address to access your subscription in DuckDuckGo on other devices. We’ll only use this address to verify your subscription.", comment: "Description for Email adding") - public static let subscriptionAddEmailButton = NSLocalizedString("subscription.activate.add.email.button", value: "Add Email", comment: "Restore button title for Email") + public static let subscriptionAddEmailButton = NSLocalizedString("subscription.activate.add.email.button", value: "Add Email", comment: "Button for adding email address to subscription") + public static let subscriptionEditEmailButton = NSLocalizedString("subscription.activate.edit.email.button", value: "Edit Email", comment: "Button for editing email address added to subscription") public static let subscriptionActivateEmailButton = NSLocalizedString("subscription.activate.email.button", value: "Enter Email", comment: "Restore button title for Email") // Add to other devices (AppleID / Email) - public static let subscriptionAddDeviceTitle = NSLocalizedString("subscription.add.device.title", value: "Add Device", comment: "Add to another device view title") - public static let subscriptionAddDeviceHeaderTitle = NSLocalizedString("subscription.add.device.header.title", value: "Use your subscription on other devices", comment: "Add subscription to other device title ") - public static let subscriptionAddDeviceDescription = NSLocalizedString("subscription.add.device.description", value: "Access your Privacy Pro subscription via an email address.", comment: "Subscription Add device Info") public static let subscriptionAvailableInApple = NSLocalizedString("subscription.available.apple", value: "Privacy Pro is available on any device signed in to the same Apple ID.", comment: "Subscription availability message on Apple devices") public static let subscriptionManageEmailResendInstructions = NSLocalizedString("subscription.add.device.resend.instructions", value: "Resend Instructions", comment: "Resend activation instructions button") @@ -1094,13 +1096,10 @@ But if you *do* want a peek under the hood, you can find more information about // Add Email To subscription public static let subscriptionAddEmail = NSLocalizedString("subscription.add.email", value: "Add an email address to activate your subscription on your other devices. We’ll only use this address to verify your subscription.", comment: "Add email to an existing subscription") - public static let subscriptionRestoreAddEmailButton = NSLocalizedString("subscription.add.email.button", value: "Add Email", comment: "Button title for adding email to subscription") public static let subscriptionRestoreAddEmailTitle = NSLocalizedString("subscription.add.email.title", value: "Add Email", comment: "View title for adding email to subscription") // Manage Subscription Email - public static let subscriptionManageEmailDescription = NSLocalizedString("subscription.manage.email.description", value: "Use this email to activate your subscription from browser settings in the DuckDuckGo app on other devices..", comment: "Description for Email Management options") - public static let subscriptionManageEmailButton = NSLocalizedString("subscription.activate.manage.email.button", value: "Manage", comment: "Restore button title for Managing Email") - public static let subscriptionManageEmailTitle = NSLocalizedString("subscription.activate.manage.email.title", value: "Manage Email", comment: "View Title for managing your email account") + public static let subscriptionEditEmailTitle = NSLocalizedString("subscription.activate.edit.email.title", value: "Edit Email", comment: "View Title for editing your email account") public static let subscriptionManageEmailCancelButton = NSLocalizedString("subscription.activate.manage.email.cancel", value: "Cancel", comment: "Button title for cancelling email deletion") public static let subscriptionManageEmailOKButton = NSLocalizedString("subscription.activate.manage.email.OK", value: "OK", comment: "Button title for confirming email deletion") @@ -1163,14 +1162,24 @@ But if you *do* want a peek under the hood, you can find more information about // Duck Player public static let duckPlayerAlwaysEnabledLabel = NSLocalizedString("duckPlayer.alwaysEnabled.label", value: "Always", comment: "Text displayed when DuckPlayer is always enabled") + public static let duckPlayerAskLabel = NSLocalizedString("duckPlayer.ask.label", value: "Ask every time", comment: "Text displayed when DuckPlayer is in 'Ask' mode.") public static let duckPlayerDisabledLabel = NSLocalizedString("duckPlayer.never.label", value: "Never", comment: "Text displayed when DuckPlayer is in off.") public static let settingsOpenVideosInDuckPlayerLabel = NSLocalizedString("duckplayer.settings.open-videos-in", value: "Open Videos in Duck Player", comment: "Settings screen cell text for DuckPlayer settings") - public static let settingsDuckPlayerTitle = NSLocalizedString("duckplayer.settings.title", value: "Duck Player", comment: "Settings screen cell text for DuckPlayer settings") + public static let duckPlayerFeatureName = NSLocalizedString("duckplayer.settings.title", value: "Duck Player", comment: "Settings screen cell text for DuckPlayer settings") public static let settingsOpenVideosInDuckPlayerTitle = NSLocalizedString("duckplayer.settings.title", value: "Duck Player", comment: "Settings screen cell text for DuckPlayer settings") public static let settingsDuckPlayerFooter = NSLocalizedString("duckplayer.settings.footer", value: "DuckDuckGo provides all the privacy essentials you need to protect yourself as you browse the web.", comment: "Footer label in the settings screen for Duck Player") public static let settingsDuckPlayerLearnMore = NSLocalizedString("duckplayer.settings.learn-more", value: "Learn More", comment: "Button that takes the user to learn more about Duck Player.") public static let settingsDuckPlayerInfoText = NSLocalizedString("duckplayer.settings.info-text", value: "Duck Player provides a clean viewing experience without personalized ads and prevents viewing activity from influencing your YouTube recommendations.", comment: "Text explaining what Duck Player is in the settings screen.") + public static let duckPlayerPresentationModalTitle = NSLocalizedString("duckplayer.presentation.modal.title", value: "Drowning in ads on YouTube? Try Duck Player!", comment: "Two line title (separated by \n) for the feature explanation") + public static let duckPlayerPresentationModalBody = NSLocalizedString("duckplayer.presentation.modal.body", value: "Duck Player lets you watch YouTube without targeted ads in a theater-like experience in DuckDuckGo and what you watch won’t influence your recommendations.", comment: "Body text for the modal feature explanation") + public static let duckPlayerPresentationModalDismissButton = NSLocalizedString("duckplayer.presentation.modal.dismiss-button", value: "Got it!", comment: "Button that will dismiss the modal") + + // Home Tab Shortcuts + public static let homeTabShortcutBookmarks = NSLocalizedString("home.tab.shortcut.bookmarks", value: "Bookmarks", comment: "Shortcut title leading to Bookmarks") + public static let homeTabShortcutAIChat = NSLocalizedString("home.tab.shortcut.ai.chat", value: "AI Chat", comment: "Shortcut title leading to AI Chat") + public static let homeTabShortcutVPN = NSLocalizedString("home.tab.shortcut.vpn", value: "VPN", comment: "Shortcut title leading to VPN") + public static let homeTabShortcutPasswords = NSLocalizedString("home.tab.shortcut.passwords", value: "Passwords", comment: "Shortcut title leading to Passwords") } diff --git a/DuckDuckGo/en.lproj/Localizable.strings b/DuckDuckGo/en.lproj/Localizable.strings index 26c4f07dc6..dde3b956c3 100644 --- a/DuckDuckGo/en.lproj/Localizable.strings +++ b/DuckDuckGo/en.lproj/Localizable.strings @@ -877,6 +877,15 @@ /* Text displayed when DuckPlayer is in off. */ "duckPlayer.never.label" = "Never"; +/* Body text for the modal feature explanation */ +"duckplayer.presentation.modal.body" = "Duck Player lets you watch YouTube without targeted ads in a theater-like experience in DuckDuckGo and what you watch won’t influence your recommendations."; + +/* Button that will dismiss the modal */ +"duckplayer.presentation.modal.dismiss-button" = "Got it!"; + +/* Two line title (separated by \n) for the feature explanation */ +"duckplayer.presentation.modal.title" = "Drowning in ads on YouTube? Try Duck Player!"; + /* Footer label in the settings screen for Duck Player */ "duckplayer.settings.footer" = "DuckDuckGo provides all the privacy essentials you need to protect yourself as you browse the web."; @@ -1234,6 +1243,18 @@ /* Home is this context is the bottom home row (dock) */ "home.row.reminder.title" = "Take DuckDuckGo home"; +/* Shortcut title leading to AI Chat */ +"home.tab.shortcut.ai.chat" = "AI Chat"; + +/* Shortcut title leading to Bookmarks */ +"home.tab.shortcut.bookmarks" = "Bookmarks"; + +/* Shortcut title leading to Passwords */ +"home.tab.shortcut.passwords" = "Passwords"; + +/* Shortcut title leading to VPN */ +"home.tab.shortcut.vpn" = "VPN"; + /* This describes empty tab */ "homeTab.searchAndFavorites" = "Search or enter address"; @@ -1981,13 +2002,10 @@ But if you *do* want a peek under the hood, you can find more information about /* No comment provided by engineer. */ "siteFeedback.urlPlaceholder" = "Which website is broken?"; -/* Subscription Activation Window Title */ -"subscription.activate" = "Activate Subscription"; - /* Subscription Activation Info */ "subscription.activate..header.description" = "Access your Privacy Pro subscription on this device via Apple ID or an email address."; -/* Restore button title for Email */ +/* Button for adding email address to subscription */ "subscription.activate.add.email.button" = "Add Email"; /* Apple ID option for activation */ @@ -2002,6 +2020,12 @@ But if you *do* want a peek under the hood, you can find more information about /* Subscription Activation Info */ "subscription.activate.description" = "Your subscription is automatically available in DuckDuckGo on any device signed in to your Apple ID."; +/* Button for editing email address added to subscription */ +"subscription.activate.edit.email.button" = "Edit Email"; + +/* View Title for editing your email account */ +"subscription.activate.edit.email.title" = "Edit Email"; + /* Email option for activation */ "subscription.activate.email" = "Email"; @@ -2014,57 +2038,30 @@ But if you *do* want a peek under the hood, you can find more information about /* Activate subscription title */ "subscription.activate.email.title" = "Activate Subscription"; -/* Restore button title for Managing Email */ -"subscription.activate.manage.email.button" = "Manage"; - /* Button title for cancelling email deletion */ "subscription.activate.manage.email.cancel" = "Cancel"; /* Button title for confirming email deletion */ "subscription.activate.manage.email.OK" = "OK"; -/* View Title for managing your email account */ -"subscription.activate.manage.email.title" = "Manage Email"; - /* Restore button title for AppleID */ "subscription.activate.restore.apple" = "Restore Purchase"; /* Subscription Activation Title */ "subscription.activate.title" = "Activate your subscription on this device"; -/* Add to another device button */ -"subscription.add.device.button" = "Add to Another Device"; - -/* Subscription Add device Info */ -"subscription.add.device.description" = "Access your Privacy Pro subscription via an email address."; - -/* Add subscription to other device title */ -"subscription.add.device.header.title" = "Use your subscription on other devices"; - /* Resend activation instructions button */ "subscription.add.device.resend.instructions" = "Resend Instructions"; -/* Add to another device view title */ -"subscription.add.device.title" = "Add Device"; - /* Add email to an existing subscription */ "subscription.add.email" = "Add an email address to activate your subscription on your other devices. We’ll only use this address to verify your subscription."; -/* Button title for adding email to subscription */ -"subscription.add.email.button" = "Add Email"; - /* View title for adding email to subscription */ "subscription.add.email.title" = "Add Email"; -/* Description for Email adding */ -"subscription.addDevice.email.description" = "Add an email address to access your subscription in DuckDuckGo on other devices. We’ll only use this address to verify your subscription."; - /* Title for Alert messages */ "subscription.alert.title" = "subscription.alert.title"; -/* Subscription type */ -"subscription.annual" = "Annual Subscription"; - /* Subscription availability message on Apple devices */ "subscription.available.apple" = "Privacy Pro is available on any device signed in to the same Apple ID."; @@ -2074,11 +2071,17 @@ But if you *do* want a peek under the hood, you can find more information about /* Title for the manage billing page */ "subscription.billing.google.title" = "Subscription Plans"; +/* Subscription annual billing period type */ +"subscription.billing.period.annual" = "annual"; + +/* Subscription monthly billing period type */ +"subscription.billing.period.monthly" = "monthly"; + /* Subscription Removal confirmation message */ "subscription.cancel.message" = "Your subscription has been removed from this device."; -/* Change plan or billing title */ -"subscription.change.plan" = "Change Plan or Billing"; +/* Change plan or cancel title */ +"subscription.change.plan" = "Update Plan or Cancel"; /* Navigation Button for closing subscription view */ "subscription.close" = "Close"; @@ -2086,6 +2089,15 @@ But if you *do* want a peek under the hood, you can find more information about /* Title for Confirm messages */ "subscription.confirm.title" = "Are you sure?"; +/* Header for section for activating subscription on other devices */ +"subscription.devices.header" = "Activate on Other Devices"; + +/* Footer for section for activating subscription on other devices when email was not yet added */ +"subscription.devices.no.email.footer" = "Add an optional email to your subscription or use your Apple ID to access Privacy Pro on other devices. **[Learn more](https://duckduckgo.com/duckduckgo-help-pages/privacy-pro/adding-email/)**"; + +/* Footer for section for activating subscription on other devices when email is added */ +"subscription.devices.with.email.footer" = "Use this email to activate your subscription in Settings > Privacy Pro in the DuckDuckGo app on your other devices. **[Learn more](https://duckduckgo.com/duckduckgo-help-pages/privacy-pro/adding-email/)**"; + /* Alert content for not found subscription */ "subscription.email.inactive.alert.message" = "The subscription associated with this email is no longer active."; @@ -2095,14 +2107,11 @@ But if you *do* want a peek under the hood, you can find more information about /* Alert content for not found subscription */ "subscription.expired.alert.message" = "The subscription associated with this Apple ID is no longer active."; -/* text for expiration string */ -"subscription.expires" = "expires"; - /* FAQ Button */ -"subscription.faq" = "Privacy Pro FAQ"; +"subscription.faq" = "FAQs and Support"; /* FAQ Description */ -"subscription.faq.description" = "Get answers to frequently asked questions about Privacy Pro in our help pages."; +"subscription.faq.description" = "Get answers to frequently asked questions or contact Privacy Pro support from our help pages."; /* Cancel action for the existing subscription dialog */ "subscription.found.cancel" = "Cancel"; @@ -2119,21 +2128,12 @@ But if you *do* want a peek under the hood, you can find more information about /* Help and support Section header */ "subscription.help" = "Help and support"; -/* Header for the device management section */ -"subscription.manage.devices" = "Manage Devices"; - -/* Description for Email Management options */ -"subscription.manage.email.description" = "Use this email to activate your subscription from browser settings in the DuckDuckGo app on other devices.."; - /* Manage Plan header */ "subscription.manage.plan" = "Manage Plan"; /* Header for the subscription section */ "subscription.manage.title" = "Subscription"; -/* Subscription type */ -"subscription.monthly" = "Monthly Subscription"; - /* Alert content for not found subscription */ "subscription.notFound.alert.message" = "There is no subscription associated with this Apple ID."; @@ -2185,9 +2185,6 @@ But if you *do* want a peek under the hood, you can find more information about /* Remove subscription cancel button text */ "subscription.remove.subscription.cancel" = "Cancel"; -/* text for renewal string */ -"subscription.renews" = "renews"; - /* Button text for general error message */ "subscription.restore.backend.error.button" = "Back to Settings"; @@ -2212,12 +2209,15 @@ But if you *do* want a peek under the hood, you can find more information about /* Alert title for restored purchase */ "subscription.restore.success.alert.title" = "You’re all set."; -/* Subscription Expiration Data. This reads as 'Your subscription (renews or expires) on (date)' */ -"subscription.subscription.active.caption" = "Your subscription %1$@ on %2$@"; - /* Subscription Expired Data. This reads as 'Your subscription expired on (date)' */ "subscription.subscription.expired.caption" = "Your subscription expired on %@"; +/* Subscription expiration info. This reads as 'Your (monthly or annual) subscription expires on (date)' */ +"subscription.subscription.expiring.caption" = "Your %1$@ subscription expires on %2$@."; + +/* Subscription renewal info. This reads as 'Your (monthly or annual) subscription renews on (date)' */ +"subscription.subscription.renewing.caption" = "Your %1$@ subscription renews on %2$@."; + /* Navigation bar Title for subscriptions */ "subscription.title" = "Privacy Pro"; diff --git a/DuckDuckGoTests/BookmarkStateRepairTests.swift b/DuckDuckGoTests/BookmarkStateRepairTests.swift new file mode 100644 index 0000000000..421f4a3f7a --- /dev/null +++ b/DuckDuckGoTests/BookmarkStateRepairTests.swift @@ -0,0 +1,118 @@ +// +// BookmarkStateRepairTests.swift +// DuckDuckGo +// +// Copyright © 2024 DuckDuckGo. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +import XCTest +import CoreData +import Bookmarks +import TestUtils +@testable import Core +@testable import DuckDuckGo + +class BookmarkStateRepairTests: XCTestCase { + + let dbStack = MockBookmarksDatabase.make(prepareFolderStructure: false) + + let mockKeyValueStore = MockKeyValueStore() + + override func setUp() async throws { + try await super.setUp() + + let containerLocation = MockBookmarksDatabase.tempDBDir() + try FileManager.default.createDirectory(at: containerLocation, withIntermediateDirectories: true) + } + + override func tearDown() async throws { + try await super.tearDown() + + try dbStack.tearDown(deleteStores: true) + } + + func testWhenThereIsNoIssueThenThereAreNoChanges() { + prepareValidStructure() + + let testContext = dbStack.makeContext(concurrencyType: .privateQueueConcurrencyType) + testContext.performAndWait { + + let repair = BookmarksStateRepair(keyValueStore: mockKeyValueStore) + + XCTAssertEqual(repair.validateAndRepairPendingDeletionState(in: testContext), .noBrokenData) + XCTAssertEqual(repair.validateAndRepairPendingDeletionState(in: testContext), .alreadyPerformed) + } + } + + func testWhenPendingDeletionIsNilThenItIsFixed() { + prepareBrokenStructure() + + let testContext = dbStack.makeContext(concurrencyType: .privateQueueConcurrencyType) + testContext.performAndWait { + + let repair = BookmarksStateRepair(keyValueStore: mockKeyValueStore) + + XCTAssertEqual(repair.validateAndRepairPendingDeletionState(in: testContext), .dataRepaired) + + mockKeyValueStore.removeObject(forKey: BookmarksStateRepair.Constants.pendingDeletionRepaired) + XCTAssertEqual(repair.validateAndRepairPendingDeletionState(in: testContext), .noBrokenData) + XCTAssertEqual(repair.validateAndRepairPendingDeletionState(in: testContext), .alreadyPerformed) + } + } + + private func prepareStructure(_ block: (NSManagedObjectContext) -> Void) { + let context = dbStack.makeContext(concurrencyType: .privateQueueConcurrencyType) + + context.performAndWait { + BookmarkUtils.prepareFoldersStructure(in: context) + block(context) + do { + try context.save() + } catch { + XCTFail("Could not save context") + } + } + } + + private func prepareValidStructure() { + prepareStructure { context in + guard let root = BookmarkUtils.fetchRootFolder(context) else { + XCTFail("Root missing") + return + } + + let bookmarkA = BookmarkEntity.makeBookmark(title: "A", url: "A", parent: root, context: context) + let bookmarkB = BookmarkEntity.makeBookmark(title: "B", url: "B", parent: root, context: context) + let bookmarkC = BookmarkEntity.makeBookmark(title: "C", url: "C", parent: root, context: context) + } + } + + private func prepareBrokenStructure() { + prepareStructure { context in + guard let root = BookmarkUtils.fetchRootFolder(context) else { + XCTFail("Root missing") + return + } + + let bookmarkA = BookmarkEntity.makeBookmark(title: "A", url: "A", parent: root, context: context) + let bookmarkB = BookmarkEntity.makeBookmark(title: "B", url: "B", parent: root, context: context) + let bookmarkC = BookmarkEntity.makeBookmark(title: "C", url: "C", parent: root, context: context) + + bookmarkA.setValue(nil, forKey: #keyPath(BookmarkEntity.isPendingDeletion)) + bookmarkB.setValue(nil, forKey: #keyPath(BookmarkEntity.isPendingDeletion)) + } + } + +} diff --git a/DuckDuckGoTests/SubscriptionContainerViewModelTests.swift b/DuckDuckGoTests/SubscriptionContainerViewModelTests.swift index 747a509329..922c406ba6 100644 --- a/DuckDuckGoTests/SubscriptionContainerViewModelTests.swift +++ b/DuckDuckGoTests/SubscriptionContainerViewModelTests.swift @@ -35,6 +35,7 @@ final class SubscriptionContainerViewModelTests: XCTestCase { let appStoreRestoreFlow = DefaultAppStoreRestoreFlow(subscriptionManager: mockDependencyProvider.subscriptionManager) let appStorePurchaseFlow = DefaultAppStorePurchaseFlow(subscriptionManager: mockDependencyProvider.subscriptionManager, appStoreRestoreFlow: appStoreRestoreFlow) + let appStoreAccountManagementFlow = DefaultAppStoreAccountManagementFlow(subscriptionManager: mockDependencyProvider.subscriptionManager) // WHEN sut = .init(subscriptionManager: mockDependencyProvider.subscriptionManager, @@ -43,7 +44,8 @@ final class SubscriptionContainerViewModelTests: XCTestCase { subFeature: .init(subscriptionManager: mockDependencyProvider.subscriptionManager, subscriptionAttributionOrigin: nil, appStorePurchaseFlow: appStorePurchaseFlow, - appStoreRestoreFlow: appStoreRestoreFlow)) + appStoreRestoreFlow: appStoreRestoreFlow, + appStoreAccountManagementFlow: appStoreAccountManagementFlow)) // THEN XCTAssertEqual(sut.flow.purchaseURL, expectedURL) @@ -53,6 +55,8 @@ final class SubscriptionContainerViewModelTests: XCTestCase { let appStoreRestoreFlow = DefaultAppStoreRestoreFlow(subscriptionManager: mockDependencyProvider.subscriptionManager) let appStorePurchaseFlow = DefaultAppStorePurchaseFlow(subscriptionManager: mockDependencyProvider.subscriptionManager, appStoreRestoreFlow: appStoreRestoreFlow) + let appStoreAccountManagementFlow = DefaultAppStoreAccountManagementFlow(subscriptionManager: mockDependencyProvider.subscriptionManager) + // WHEN sut = .init(subscriptionManager: mockDependencyProvider.subscriptionManager, origin: nil, @@ -60,7 +64,8 @@ final class SubscriptionContainerViewModelTests: XCTestCase { subFeature: .init(subscriptionManager: mockDependencyProvider.subscriptionManager, subscriptionAttributionOrigin: nil, appStorePurchaseFlow: appStorePurchaseFlow, - appStoreRestoreFlow: appStoreRestoreFlow)) + appStoreRestoreFlow: appStoreRestoreFlow, + appStoreAccountManagementFlow: appStoreAccountManagementFlow)) // THEN XCTAssertEqual(sut.flow.purchaseURL, SubscriptionURL.purchase.subscriptionURL(environment: .production)) diff --git a/DuckDuckGoTests/SubscriptionFlowViewModelTests.swift b/DuckDuckGoTests/SubscriptionFlowViewModelTests.swift index 48a0a3ba34..6d375a2a87 100644 --- a/DuckDuckGoTests/SubscriptionFlowViewModelTests.swift +++ b/DuckDuckGoTests/SubscriptionFlowViewModelTests.swift @@ -36,12 +36,14 @@ final class SubscriptionFlowViewModelTests: XCTestCase { let appStoreRestoreFlow = DefaultAppStoreRestoreFlow(subscriptionManager: mockDependencyProvider.subscriptionManager) let appStorePurchaseFlow = DefaultAppStorePurchaseFlow(subscriptionManager: mockDependencyProvider.subscriptionManager, appStoreRestoreFlow: appStoreRestoreFlow) - + let appStoreAccountManagementFlow = DefaultAppStoreAccountManagementFlow(subscriptionManager: mockDependencyProvider.subscriptionManager) + // WHEN sut = .init(origin: origin, userScript: .init(), subFeature: .init(subscriptionManager: mockDependencyProvider.subscriptionManager, subscriptionAttributionOrigin: nil, appStorePurchaseFlow: appStorePurchaseFlow, - appStoreRestoreFlow: appStoreRestoreFlow), + appStoreRestoreFlow: appStoreRestoreFlow, + appStoreAccountManagementFlow: appStoreAccountManagementFlow), subscriptionManager: mockDependencyProvider.subscriptionManager) // THEN @@ -52,12 +54,14 @@ final class SubscriptionFlowViewModelTests: XCTestCase { let appStoreRestoreFlow = DefaultAppStoreRestoreFlow(subscriptionManager: mockDependencyProvider.subscriptionManager) let appStorePurchaseFlow = DefaultAppStorePurchaseFlow(subscriptionManager: mockDependencyProvider.subscriptionManager, appStoreRestoreFlow: appStoreRestoreFlow) - + let appStoreAccountManagementFlow = DefaultAppStoreAccountManagementFlow(subscriptionManager: mockDependencyProvider.subscriptionManager) + // WHEN sut = .init(origin: nil, userScript: .init(), subFeature: .init(subscriptionManager: mockDependencyProvider.subscriptionManager, subscriptionAttributionOrigin: nil, appStorePurchaseFlow: appStorePurchaseFlow, - appStoreRestoreFlow: appStoreRestoreFlow), + appStoreRestoreFlow: appStoreRestoreFlow, + appStoreAccountManagementFlow: appStoreAccountManagementFlow), subscriptionManager: mockDependencyProvider.subscriptionManager) // THEN diff --git a/DuckDuckGoTests/YoutublePlayerNavigationHandlerTests.swift b/DuckDuckGoTests/YoutublePlayerNavigationHandlerTests.swift index 06e3378db6..a92804fe8e 100644 --- a/DuckDuckGoTests/YoutublePlayerNavigationHandlerTests.swift +++ b/DuckDuckGoTests/YoutublePlayerNavigationHandlerTests.swift @@ -40,11 +40,23 @@ class MockWKNavigationDelegate: NSObject, WKNavigationDelegate { decidePolicyForNavigationAction?(webView, navigationAction, decisionHandler) ?? decisionHandler(.allow) } } + class MockWebView: WKWebView { var didStopLoadingCalled = false var lastLoadedRequest: URLRequest? var lastResponseHTML: String? var goToCalledWith: WKBackForwardListItem? + var canGoBackMock = false + var currentURL: URL? + + private var _url: URL? + override var url: URL? { + return currentURL + } + + func setCurrentURL(_ url: URL) { + self.currentURL = url + } override func stopLoading() { didStopLoadingCalled = true @@ -55,13 +67,11 @@ class MockWebView: WKWebView { return nil } - override func go(to item: WKBackForwardListItem) -> WKNavigation? { - goToCalledWith = item + override func reload() -> WKNavigation? { return nil } } - class MockNavigationAction: WKNavigationAction { private let _request: URLRequest @@ -76,12 +86,12 @@ class MockNavigationAction: WKNavigationAction { class YoutubePlayerNavigationHandlerTests: XCTestCase { - + var handler: YoutubePlayerNavigationHandler! var webView: WKWebView! var mockWebView: MockWebView! var mockNavigationDelegate: MockWKNavigationDelegate! - + override func setUp() { super.setUp() handler = YoutubePlayerNavigationHandler() @@ -90,7 +100,7 @@ class YoutubePlayerNavigationHandlerTests: XCTestCase { mockNavigationDelegate = MockWKNavigationDelegate() webView.navigationDelegate = mockNavigationDelegate } - + override func tearDown() { webView.navigationDelegate = nil webView = nil @@ -111,16 +121,16 @@ class YoutubePlayerNavigationHandlerTests: XCTestCase { // Test for makeDuckPlayerRequest(from:) func testMakeDuckPlayerRequestFromOriginalRequest() { let originalRequest = URLRequest(url: URL(string: "https://www.youtube.com/watch?v=abc123&t=10s")!) - + let duckPlayerRequest = YoutubePlayerNavigationHandler.makeDuckPlayerRequest(from: originalRequest) - + XCTAssertEqual(duckPlayerRequest.url?.host, "www.youtube-nocookie.com") XCTAssertEqual(duckPlayerRequest.url?.path, "/embed/abc123") XCTAssertEqual(duckPlayerRequest.url?.query?.contains("t=10s"), true) XCTAssertEqual(duckPlayerRequest.value(forHTTPHeaderField: "Referer"), "http://localhost/") XCTAssertEqual(duckPlayerRequest.httpMethod, "GET") } - + // Test for makeDuckPlayerRequest(for:timestamp:) func testMakeDuckPlayerRequestForVideoID() { let videoID = "abc123" @@ -134,7 +144,7 @@ class YoutubePlayerNavigationHandlerTests: XCTestCase { XCTAssertEqual(duckPlayerRequest.value(forHTTPHeaderField: "Referer"), "http://localhost/") XCTAssertEqual(duckPlayerRequest.httpMethod, "GET") } - + // Test for makeHTMLFromTemplate func testMakeHTMLFromTemplate() { let expectedHtml = try? String(contentsOfFile: YoutubePlayerNavigationHandler.htmlTemplatePath) @@ -142,11 +152,11 @@ class YoutubePlayerNavigationHandlerTests: XCTestCase { XCTAssertEqual(html, expectedHtml) } - // Validate redirects are properly triggered - func testHandleRedirect() { - + // Test for handleURLChange + @MainActor + func testHandleURLChange() { let youtubeURL = URL(string: "https://www.youtube.com/watch?v=abc123&t=10s")! - handler.handleRedirect(url: youtubeURL, webView: mockWebView) + handler.handleURLChange(url: youtubeURL, webView: mockWebView) XCTAssertTrue(mockWebView.didStopLoadingCalled, "Expected stopLoading to be called") XCTAssertNotNil(mockWebView.lastLoadedRequest, "Expected a new request to be loaded") @@ -159,29 +169,29 @@ class YoutubePlayerNavigationHandlerTests: XCTestCase { } } - func testHandleRedirectForNonYouTubeVideo() { - - let youtubeURL = URL(string: "https://www.google.com.com/watch?v=abc123&t=10s")! - handler.handleRedirect(url: youtubeURL, webView: mockWebView) + @MainActor + func testHandleURLChangeForNonYouTubeVideo() { + let nonYouTubeURL = URL(string: "https://www.google.com")! + handler.handleURLChange(url: nonYouTubeURL, webView: mockWebView) XCTAssertFalse(mockWebView.didStopLoadingCalled, "Expected stopLoading not to be called") XCTAssertNil(mockWebView.lastLoadedRequest, "Expected a new request not to be loaded") - } - func testHandleRedirectWithNavigationAction() { - + // Test for handleDecidePolicyFor + @MainActor + func testHandleDecidePolicyFor() { let youtubeURL = URL(string: "https://www.youtube.com/watch?v=abc123&t=10s")! let navigationAction = MockNavigationAction(request: URLRequest(url: youtubeURL)) let expectation = self.expectation(description: "Completion handler called") var navigationPolicy: WKNavigationActionPolicy? - handler.handleRedirect(navigationAction, completion: { policy in + handler.handleDecidePolicyFor(navigationAction, completion: { policy in navigationPolicy = policy expectation.fulfill() }, webView: mockWebView) - + waitForExpectations(timeout: 1, handler: nil) XCTAssertEqual(navigationPolicy, .allow, "Expected navigation policy to be .allow") @@ -195,75 +205,50 @@ class YoutubePlayerNavigationHandlerTests: XCTestCase { } } - func testHandleRedirectWithNavigationActionForNonYouTubeVideo() { - - let youtubeURL = URL(string: "https://www.google.com.com/watch?v=abc123&t=10s")! - let navigationAction = MockNavigationAction(request: URLRequest(url: youtubeURL)) + @MainActor + func testHandleDecidePolicyForNonYouTubeVideo() { + let nonYouTubeURL = URL(string: "https://www.google.com")! + let navigationAction = MockNavigationAction(request: URLRequest(url: nonYouTubeURL)) let expectation = self.expectation(description: "Completion handler called") var navigationPolicy: WKNavigationActionPolicy? - handler.handleRedirect(navigationAction, completion: { policy in + handler.handleDecidePolicyFor(navigationAction, completion: { policy in navigationPolicy = policy expectation.fulfill() }, webView: mockWebView) - + waitForExpectations(timeout: 1, handler: nil) XCTAssertEqual(navigationPolicy, .cancel, "Expected navigation policy to be .cancel") XCTAssertNil(mockWebView.lastLoadedRequest, "Expected a new request not to be loaded") } - // Test for handleNavigation for duck:// links - func testHandleNavigationWithDuckPlayerURL() { - let handler = YoutubePlayerNavigationHandler() - let mockWebView = MockWebView() - let duckPlayerURL = URL(string: "duck://player/abc123&t=30")! - let navigationAction = MockNavigationAction(request: URLRequest(url: duckPlayerURL)) - let expectation = self.expectation(description: "Completion handler called") - - var navigationPolicy: WKNavigationActionPolicy? - - handler.handleNavigation(navigationAction, webView: mockWebView) { policy in - navigationPolicy = policy - expectation.fulfill() - } - - waitForExpectations(timeout: 1, handler: nil) - - XCTAssertEqual(navigationPolicy, .allow, "Expected navigation policy to be .allow") + @MainActor + func testHandleReloadForDuckPlayerVideo() { + let duckPlayerURL = URL(string: "https://www.youtube-nocookie.com/embed/abc123?t=10s")! - if let responseHTML = mockWebView.lastResponseHTML { - let expectedHtml = try? String(contentsOfFile: YoutubePlayerNavigationHandler.htmlTemplatePath) - XCTAssertEqual(responseHTML, expectedHtml) - } - } - - // Test for handleNavigation for non duck:// links - func testHandleNavigationWithNonDuckPlayerURL() { - let handler = YoutubePlayerNavigationHandler() - let mockWebView = MockWebView() - let duckPlayerURL = URL(string: "https://www.youtube.com/watch?v=abc123")! - let navigationAction = MockNavigationAction(request: URLRequest(url: duckPlayerURL)) - let expectation = self.expectation(description: "Completion handler called") + mockWebView.setCurrentURL(duckPlayerURL) + handler.handleReload(webView: mockWebView) - var navigationPolicy: WKNavigationActionPolicy? + XCTAssertNotNil(mockWebView.lastLoadedRequest, "Expected a new request to be loaded") - handler.handleNavigation(navigationAction, webView: mockWebView) { policy in - navigationPolicy = policy - expectation.fulfill() + if let loadedRequest = mockWebView.lastLoadedRequest { + XCTAssertEqual(loadedRequest.url?.scheme, "duck") + XCTAssertEqual(loadedRequest.url?.host, "player") + XCTAssertEqual(loadedRequest.url?.path, "/abc123") + XCTAssertEqual(loadedRequest.url?.query?.contains("t=10s"), true) } + } + + @MainActor + func testHandleReloadForNonDuckPlayerVideo() { + let nonDuckPlayerURL = URL(string: "https://www.google.com")! - waitForExpectations(timeout: 1, handler: nil) - - XCTAssertEqual(navigationPolicy, .cancel, "Expected navigation policy to be .cancel") + // Simulate the current URL + mockWebView.setCurrentURL(nonDuckPlayerURL) + handler.handleReload(webView: mockWebView) XCTAssertNil(mockWebView.lastLoadedRequest, "Expected a new request not to be loaded") - XCTAssertNil(mockWebView.lastResponseHTML, "Expected the response HTML not to be loaded") - - if let responseHTML = mockWebView.lastResponseHTML { - let expectedHtml = try? String(contentsOfFile: YoutubePlayerNavigationHandler.htmlTemplatePath) - XCTAssertEqual(responseHTML, expectedHtml) - } } - + } diff --git a/PacketTunnelProvider/NetworkProtection/NetworkProtectionPacketTunnelProvider.swift b/PacketTunnelProvider/NetworkProtection/NetworkProtectionPacketTunnelProvider.swift index 54ae6ecda7..7b66f1efa8 100644 --- a/PacketTunnelProvider/NetworkProtection/NetworkProtectionPacketTunnelProvider.swift +++ b/PacketTunnelProvider/NetworkProtection/NetworkProtectionPacketTunnelProvider.swift @@ -144,6 +144,8 @@ final class NetworkProtectionPacketTunnelProvider: PacketTunnelProvider { case .success: DailyPixel.fireDailyAndCount(pixel: .networkProtectionServerMigrationAttemptSuccess) } + case .tunnelStartOnDemandWithoutAccessToken: + DailyPixel.fireDailyAndCount(pixel: .networkProtectionTunnelStartAttemptOnDemandWithoutAccessToken) } } diff --git a/fastlane/metadata/default/release_notes.txt b/fastlane/metadata/default/release_notes.txt index 947a33a7e1..b4dad65e3a 100644 --- a/fastlane/metadata/default/release_notes.txt +++ b/fastlane/metadata/default/release_notes.txt @@ -1,2 +1,5 @@ -Bug fixes and other improvements. +- Bug fixes and other improvements. + +For Privacy Pro subscribers: +- You can now specify a custom DNS server under VPN Settings to route your DNS queries through while the VPN is active.