mirror of https://github.com/oxen-io/session-ios
				
				
				
			
			You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			1095 lines
		
	
	
		
			44 KiB
		
	
	
	
		
			Swift
		
	
			
		
		
	
	
			1095 lines
		
	
	
		
			44 KiB
		
	
	
	
		
			Swift
		
	
| //
 | |
| //  Copyright (c) 2019 Open Whisper Systems. All rights reserved.
 | |
| //
 | |
| 
 | |
| import UIKit
 | |
| 
 | |
| import SignalMessaging
 | |
| import PureLayout
 | |
| import SessionServiceKit
 | |
| import PromiseKit
 | |
| 
 | |
| @objc
 | |
| public class ShareViewController: UIViewController, ShareViewDelegate, SAEFailedViewDelegate, AppModeManagerDelegate {
 | |
| 
 | |
|     // MARK: - Dependencies
 | |
| 
 | |
|     private var tsAccountManager: TSAccountManager {
 | |
|         return TSAccountManager.sharedInstance()
 | |
|     }
 | |
| 
 | |
|     // MARK: -
 | |
| 
 | |
|     enum ShareViewControllerError: Error {
 | |
|         case assertionError(description: String)
 | |
|         case unsupportedMedia
 | |
|         case notRegistered
 | |
|         case obsoleteShare
 | |
|     }
 | |
| 
 | |
|     private var hasInitialRootViewController = false
 | |
|     private var isReadyForAppExtensions = false
 | |
|     private var areVersionMigrationsComplete = false
 | |
| 
 | |
|     private var progressPoller: ProgressPoller?
 | |
|     var loadViewController: SAELoadViewController?
 | |
| 
 | |
|     private var shareViewNavigationController: OWSNavigationController?
 | |
| 
 | |
|     override open func loadView() {
 | |
|         super.loadView()
 | |
| 
 | |
|         // This should be the first thing we do.
 | |
|         let appContext = ShareAppExtensionContext(rootViewController: self)
 | |
|         SetCurrentAppContext(appContext)
 | |
| 
 | |
|         AppModeManager.configure(delegate: self)
 | |
| 
 | |
|         DebugLogger.shared().enableTTYLogging()
 | |
|         if _isDebugAssertConfiguration() {
 | |
|             DebugLogger.shared().enableFileLogging()
 | |
|         } else if OWSPreferences.isLoggingEnabled() {
 | |
|             DebugLogger.shared().enableFileLogging()
 | |
|         }
 | |
| 
 | |
|         Logger.info("")
 | |
| 
 | |
|         _ = AppVersion.sharedInstance()
 | |
| 
 | |
|         startupLogging()
 | |
| 
 | |
|         Cryptography.seedRandom()
 | |
| 
 | |
|         // We don't need to use DeviceSleepManager in the SAE.
 | |
| 
 | |
|         // We don't need to use applySignalAppearence in the SAE.
 | |
| 
 | |
|         if CurrentAppContext().isRunningTests {
 | |
|             // TODO: Do we need to implement isRunningTests in the SAE context?
 | |
|             return
 | |
|         }
 | |
| 
 | |
|         // If we haven't migrated the database file to the shared data
 | |
|         // directory we can't load it, and therefore can't init TSSPrimaryStorage,
 | |
|         // and therefore don't want to setup most of our machinery (Environment,
 | |
|         // most of the singletons, etc.).  We just want to show an error view and
 | |
|         // abort.
 | |
|         isReadyForAppExtensions = OWSPreferences.isReadyForAppExtensions()
 | |
|         guard isReadyForAppExtensions else {
 | |
|             showNotReadyView()
 | |
|             return
 | |
|         }
 | |
| 
 | |
|         // We shouldn't set up our environment until after we've consulted isReadyForAppExtensions.
 | |
|         AppSetup.setupEnvironment(appSpecificSingletonBlock: {
 | |
|             SSKEnvironment.shared.callMessageHandler = NoopCallMessageHandler()
 | |
|             SSKEnvironment.shared.notificationsManager = NoopNotificationsManager()
 | |
|             },
 | |
|             migrationCompletion: { [weak self] in
 | |
|                                     AssertIsOnMainThread()
 | |
| 
 | |
|                                     guard let strongSelf = self else { return }
 | |
| 
 | |
|                                     // performUpdateCheck must be invoked after Environment has been initialized because
 | |
|                                     // upgrade process may depend on Environment.
 | |
|                                     strongSelf.versionMigrationsDidComplete()
 | |
|         })
 | |
| 
 | |
|         let shareViewNavigationController = OWSNavigationController()
 | |
|         self.shareViewNavigationController = shareViewNavigationController
 | |
| 
 | |
|         let loadViewController = SAELoadViewController(delegate: self)
 | |
|         self.loadViewController = loadViewController
 | |
| 
 | |
|         // Don't display load screen immediately, in hopes that we can avoid it altogether.
 | |
|         after(seconds: 0.5).done { [weak self] in
 | |
|             AssertIsOnMainThread()
 | |
| 
 | |
|             guard let strongSelf = self else { return }
 | |
|             guard strongSelf.presentedViewController == nil else {
 | |
|                 Logger.debug("setup completed quickly, no need to present load view controller.")
 | |
|                 return
 | |
|             }
 | |
| 
 | |
|             Logger.debug("setup is slow - showing loading screen")
 | |
|             strongSelf.showPrimaryViewController(loadViewController)
 | |
|             }.retainUntilComplete()
 | |
| 
 | |
|         // We don't need to use "screen protection" in the SAE.
 | |
| 
 | |
|         NotificationCenter.default.addObserver(self,
 | |
|                                                selector: #selector(storageIsReady),
 | |
|                                                name: .StorageIsReady,
 | |
|                                                object: nil)
 | |
|         NotificationCenter.default.addObserver(self,
 | |
|                                                selector: #selector(registrationStateDidChange),
 | |
|                                                name: .RegistrationStateDidChange,
 | |
|                                                object: nil)
 | |
|         NotificationCenter.default.addObserver(self,
 | |
|                                                selector: #selector(owsApplicationWillEnterForeground),
 | |
|                                                name: .OWSApplicationWillEnterForeground,
 | |
|                                                object: nil)
 | |
|         NotificationCenter.default.addObserver(self,
 | |
|                                                selector: #selector(applicationDidEnterBackground),
 | |
|                                                name: .OWSApplicationDidEnterBackground,
 | |
|                                                object: nil)
 | |
| 
 | |
|         Logger.info("completed.")
 | |
| 
 | |
|         OWSAnalytics.appLaunchDidBegin()
 | |
|     }
 | |
| 
 | |
|     deinit {
 | |
|         Logger.info("deinit")
 | |
|         NotificationCenter.default.removeObserver(self)
 | |
| 
 | |
|         // Share extensions reside in a process that may be reused between usages.
 | |
|         // That isn't safe; the codebase is full of statics (e.g. singletons) which
 | |
|         // we can't easily clean up.
 | |
|         ExitShareExtension()
 | |
|     }
 | |
| 
 | |
|     @objc
 | |
|     public func applicationDidEnterBackground() {
 | |
|         AssertIsOnMainThread()
 | |
| 
 | |
|         Logger.info("")
 | |
| 
 | |
|         if OWSScreenLock.shared.isScreenLockEnabled() {
 | |
| 
 | |
|             Logger.info("dismissing.")
 | |
| 
 | |
|             self.dismiss(animated: false) { [weak self] in
 | |
|                 AssertIsOnMainThread()
 | |
|                 guard let strongSelf = self else { return }
 | |
|                 strongSelf.extensionContext!.completeRequest(returningItems: [], completionHandler: nil)
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     private func activate() {
 | |
|         AssertIsOnMainThread()
 | |
| 
 | |
|         Logger.debug("")
 | |
| 
 | |
|         // We don't need to use "screen protection" in the SAE.
 | |
| 
 | |
|         ensureRootViewController()
 | |
| 
 | |
|         // Always check prekeys after app launches, and sometimes check on app activation.
 | |
|         TSPreKeyManager.checkPreKeysIfNecessary()
 | |
| 
 | |
|         // We don't need to use RTCInitializeSSL() in the SAE.
 | |
| 
 | |
|         if tsAccountManager.isRegistered() {
 | |
|             // At this point, potentially lengthy DB locking migrations could be running.
 | |
|             // Avoid blocking app launch by putting all further possible DB access in async block
 | |
|             DispatchQueue.global().async { [weak self] in
 | |
|                 guard let _ = self else { return }
 | |
|                 Logger.info("running post launch block for registered user: \(TSAccountManager.localNumber)")
 | |
| 
 | |
|                 // We don't need to use OWSDisappearingMessagesJob in the SAE.
 | |
| 
 | |
|                 // We don't need to use OWSFailedMessagesJob in the SAE.
 | |
| 
 | |
|                 // We don't need to use OWSFailedAttachmentDownloadsJob in the SAE.
 | |
|             }
 | |
|         } else {
 | |
|             Logger.info("running post launch block for unregistered user.")
 | |
| 
 | |
|             // We don't need to update the app icon badge number in the SAE.
 | |
| 
 | |
|             // We don't need to prod the TSSocketManager in the SAE.
 | |
|         }
 | |
| 
 | |
|         if tsAccountManager.isRegistered() {
 | |
|             DispatchQueue.main.async { [weak self] in
 | |
|                 guard let _ = self else { return }
 | |
|                 Logger.info("running post launch block for registered user: \(TSAccountManager.localNumber)")
 | |
| 
 | |
|                 // We don't need to use the TSSocketManager in the SAE.
 | |
| 
 | |
|                 Environment.shared.contactsManager.fetchSystemContactsOnceIfAlreadyAuthorized()
 | |
| 
 | |
|                 // We don't need to fetch messages in the SAE.
 | |
| 
 | |
|                 // We don't need to use OWSSyncPushTokensJob in the SAE.
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     @objc
 | |
|     func versionMigrationsDidComplete() {
 | |
|         AssertIsOnMainThread()
 | |
| 
 | |
|         Logger.debug("")
 | |
| 
 | |
|         areVersionMigrationsComplete = true
 | |
| 
 | |
|         checkIsAppReady()
 | |
|     }
 | |
| 
 | |
|     @objc
 | |
|     func storageIsReady() {
 | |
|         AssertIsOnMainThread()
 | |
| 
 | |
|         Logger.debug("")
 | |
| 
 | |
|         checkIsAppReady()
 | |
|     }
 | |
| 
 | |
|     @objc
 | |
|     func checkIsAppReady() {
 | |
|         AssertIsOnMainThread()
 | |
| 
 | |
|         // App isn't ready until storage is ready AND all version migrations are complete.
 | |
|         guard areVersionMigrationsComplete else {
 | |
|             return
 | |
|         }
 | |
|         guard OWSStorage.isStorageReady() else {
 | |
|             return
 | |
|         }
 | |
|         guard !AppReadiness.isAppReady() else {
 | |
|             // Only mark the app as ready once.
 | |
|             return
 | |
|         }
 | |
| 
 | |
|         Logger.debug("")
 | |
| 
 | |
|         // TODO: Once "app ready" logic is moved into AppSetup, move this line there.
 | |
|         OWSProfileManager.shared().ensureLocalProfileCached()
 | |
| 
 | |
|         // Note that this does much more than set a flag;
 | |
|         // it will also run all deferred blocks.
 | |
|         AppReadiness.setAppIsReady()
 | |
| 
 | |
|         if tsAccountManager.isRegistered() {
 | |
|             Logger.info("localNumber: \(TSAccountManager.localNumber)")
 | |
| 
 | |
|             // We don't need to use messageFetcherJob in the SAE.
 | |
| 
 | |
|             // We don't need to use SyncPushTokensJob in the SAE.
 | |
|         }
 | |
| 
 | |
|         // We don't need to use DeviceSleepManager in the SAE.
 | |
| 
 | |
|         AppVersion.sharedInstance().saeLaunchDidComplete()
 | |
| 
 | |
|         ensureRootViewController()
 | |
| 
 | |
|         // We don't need to use OWSMessageReceiver in the SAE.
 | |
|         // We don't need to use OWSBatchMessageProcessor in the SAE.
 | |
| 
 | |
|         OWSProfileManager.shared().ensureLocalProfileCached()
 | |
| 
 | |
|         // We don't need to use OWSOrphanDataCleaner in the SAE.
 | |
| 
 | |
|         // We don't need to fetch the local profile in the SAE
 | |
| 
 | |
|         OWSReadReceiptManager.shared().prepareCachedValues()
 | |
|     }
 | |
| 
 | |
|     @objc
 | |
|     func registrationStateDidChange() {
 | |
|         AssertIsOnMainThread()
 | |
| 
 | |
|         Logger.debug("")
 | |
| 
 | |
|         if tsAccountManager.isRegistered() {
 | |
|             Logger.info("localNumber: \(TSAccountManager.localNumber)")
 | |
| 
 | |
|             // We don't need to use ExperienceUpgradeFinder in the SAE.
 | |
| 
 | |
|             // We don't need to use OWSDisappearingMessagesJob in the SAE.
 | |
| 
 | |
|             OWSProfileManager.shared().ensureLocalProfileCached()
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     private func ensureRootViewController() {
 | |
|         AssertIsOnMainThread()
 | |
| 
 | |
|         Logger.debug("")
 | |
| 
 | |
|         guard AppReadiness.isAppReady() else {
 | |
|             return
 | |
|         }
 | |
|         guard !hasInitialRootViewController else {
 | |
|             return
 | |
|         }
 | |
|         hasInitialRootViewController = true
 | |
| 
 | |
|         Logger.info("Presenting initial root view controller")
 | |
| 
 | |
|         if OWSScreenLock.shared.isScreenLockEnabled() {
 | |
|             presentScreenLock()
 | |
|         } else {
 | |
|             presentContentView()
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     private func presentContentView() {
 | |
|         AssertIsOnMainThread()
 | |
| 
 | |
|         Logger.debug("")
 | |
| 
 | |
|         Logger.info("Presenting content view")
 | |
| 
 | |
|         if !tsAccountManager.isRegistered() {
 | |
|             showNotRegisteredView()
 | |
|         } else if !OWSProfileManager.shared().localProfileExists() {
 | |
|             // This is a rare edge case, but we want to ensure that the user
 | |
|             // is has already saved their local profile key in the main app.
 | |
|             showNotReadyView()
 | |
|         } else {
 | |
|             DispatchQueue.main.async { [weak self] in
 | |
|                 guard let strongSelf = self else { return }
 | |
|                 strongSelf.buildAttachmentsAndPresentConversationPicker()
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // We don't use the AppUpdateNag in the SAE.
 | |
|     }
 | |
| 
 | |
|     func startupLogging() {
 | |
|         Logger.info("iOS Version: \(UIDevice.current.systemVersion)}")
 | |
| 
 | |
|         let locale = NSLocale.current as NSLocale
 | |
|         if let localeIdentifier = locale.object(forKey: NSLocale.Key.identifier) as? String,
 | |
|             localeIdentifier.count > 0 {
 | |
|             Logger.info("Locale Identifier: \(localeIdentifier)")
 | |
|         } else {
 | |
|             owsFailDebug("Locale Identifier: Unknown")
 | |
|         }
 | |
|         if let countryCode = locale.object(forKey: NSLocale.Key.countryCode) as? String,
 | |
|             countryCode.count > 0 {
 | |
|             Logger.info("Country Code: \(countryCode)")
 | |
|         } else {
 | |
|             owsFailDebug("Country Code: Unknown")
 | |
|         }
 | |
|         if let languageCode = locale.object(forKey: NSLocale.Key.languageCode) as? String,
 | |
|             languageCode.count > 0 {
 | |
|             Logger.info("Language Code: \(languageCode)")
 | |
|         } else {
 | |
|             owsFailDebug("Language Code: Unknown")
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // MARK: Error Views
 | |
| 
 | |
|     private func showNotReadyView() {
 | |
|         AssertIsOnMainThread()
 | |
| 
 | |
|         let failureTitle = NSLocalizedString("SHARE_EXTENSION_NOT_YET_MIGRATED_TITLE",
 | |
|                                              comment: "Title indicating that the share extension cannot be used until the main app has been launched at least once.")
 | |
|         let failureMessage = NSLocalizedString("SHARE_EXTENSION_NOT_YET_MIGRATED_MESSAGE",
 | |
|                                                comment: "Message indicating that the share extension cannot be used until the main app has been launched at least once.")
 | |
|         showErrorView(title: failureTitle, message: failureMessage)
 | |
|     }
 | |
| 
 | |
|     private func showNotRegisteredView() {
 | |
|         AssertIsOnMainThread()
 | |
| 
 | |
|         let failureTitle = NSLocalizedString("SHARE_EXTENSION_NOT_REGISTERED_TITLE",
 | |
|                                              comment: "Title indicating that the share extension cannot be used until the user has registered in the main app.")
 | |
|         let failureMessage = NSLocalizedString("SHARE_EXTENSION_NOT_REGISTERED_MESSAGE",
 | |
|                                                comment: "Message indicating that the share extension cannot be used until the user has registered in the main app.")
 | |
|         showErrorView(title: failureTitle, message: failureMessage)
 | |
|     }
 | |
| 
 | |
|     private func showErrorView(title: String, message: String) {
 | |
|         AssertIsOnMainThread()
 | |
| 
 | |
|         let viewController = SAEFailedViewController(delegate: self, title: title, message: message)
 | |
|         self.showPrimaryViewController(viewController)
 | |
|     }
 | |
| 
 | |
|     // MARK: View Lifecycle
 | |
| 
 | |
|     override open func viewDidLoad() {
 | |
|         super.viewDidLoad()
 | |
| 
 | |
|         Logger.debug("")
 | |
| 
 | |
|         if isReadyForAppExtensions {
 | |
|             AppReadiness.runNowOrWhenAppDidBecomeReady { [weak self] in
 | |
|                 AssertIsOnMainThread()
 | |
|                 guard let strongSelf = self else { return }
 | |
|                 strongSelf.activate()
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     override open func viewWillAppear(_ animated: Bool) {
 | |
|         Logger.debug("")
 | |
| 
 | |
|         super.viewWillAppear(animated)
 | |
|     }
 | |
| 
 | |
|     override open func viewDidAppear(_ animated: Bool) {
 | |
|         Logger.debug("")
 | |
| 
 | |
|         super.viewDidAppear(animated)
 | |
|     }
 | |
| 
 | |
|     override open func viewWillDisappear(_ animated: Bool) {
 | |
|         Logger.debug("")
 | |
| 
 | |
|         super.viewWillDisappear(animated)
 | |
| 
 | |
|         Logger.flush()
 | |
|     }
 | |
| 
 | |
|     override open func viewDidDisappear(_ animated: Bool) {
 | |
|         Logger.debug("")
 | |
| 
 | |
|         super.viewDidDisappear(animated)
 | |
| 
 | |
|         Logger.flush()
 | |
| 
 | |
|         // Share extensions reside in a process that may be reused between usages.
 | |
|         // That isn't safe; the codebase is full of statics (e.g. singletons) which
 | |
|         // we can't easily clean up.
 | |
|         ExitShareExtension()
 | |
|     }
 | |
| 
 | |
|     @objc
 | |
|     func owsApplicationWillEnterForeground() throws {
 | |
|         AssertIsOnMainThread()
 | |
| 
 | |
|         Logger.debug("")
 | |
| 
 | |
|         // If a user unregisters in the main app, the SAE should shut down
 | |
|         // immediately.
 | |
|         guard !tsAccountManager.isRegistered() else {
 | |
|             // If user is registered, do nothing.
 | |
|             return
 | |
|         }
 | |
|         guard let shareViewNavigationController = shareViewNavigationController else {
 | |
|             owsFailDebug("Missing shareViewNavigationController")
 | |
|             return
 | |
|         }
 | |
|         guard let firstViewController = shareViewNavigationController.viewControllers.first else {
 | |
|             // If no view has been presented yet, do nothing.
 | |
|             return
 | |
|         }
 | |
|         if let _ = firstViewController as? SAEFailedViewController {
 | |
|             // If root view is an error view, do nothing.
 | |
|             return
 | |
|         }
 | |
|         throw ShareViewControllerError.notRegistered
 | |
|     }
 | |
| 
 | |
|     // MARK: ShareViewDelegate, SAEFailedViewDelegate
 | |
| 
 | |
|     public func shareViewWasUnlocked() {
 | |
|         Logger.info("")
 | |
| 
 | |
|         presentContentView()
 | |
|     }
 | |
| 
 | |
|     public func shareViewWasCompleted() {
 | |
|         Logger.info("")
 | |
| 
 | |
|         self.dismiss(animated: true) { [weak self] in
 | |
|             AssertIsOnMainThread()
 | |
|             guard let strongSelf = self else { return }
 | |
|             strongSelf.extensionContext!.completeRequest(returningItems: [], completionHandler: nil)
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     public func shareViewWasCancelled() {
 | |
|         Logger.info("")
 | |
| 
 | |
|         self.dismiss(animated: true) { [weak self] in
 | |
|             AssertIsOnMainThread()
 | |
|             guard let strongSelf = self else { return }
 | |
|             strongSelf.extensionContext!.completeRequest(returningItems: [], completionHandler: nil)
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     public func shareViewFailed(error: Error) {
 | |
|         Logger.info("")
 | |
| 
 | |
|         self.dismiss(animated: true) { [weak self] in
 | |
|             AssertIsOnMainThread()
 | |
|             guard let strongSelf = self else { return }
 | |
|             strongSelf.extensionContext!.cancelRequest(withError: error)
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // MARK: Helpers
 | |
| 
 | |
|     // This view controller is not visible to the user. It exists to intercept touches, set up the
 | |
|     // extensions dependencies, and eventually present a visible view to the user.
 | |
|     // For speed of presentation, we only present a single modal, and if it's already been presented
 | |
|     // we swap out the contents.
 | |
|     // e.g. if loading is taking a while, the user will see the load screen presented with a modal
 | |
|     // animation. Next, when loading completes, the load view will be switched out for the contact
 | |
|     // picker view.
 | |
|     private func showPrimaryViewController(_ viewController: UIViewController) {
 | |
|         AssertIsOnMainThread()
 | |
| 
 | |
|         guard let shareViewNavigationController = shareViewNavigationController else {
 | |
|             owsFailDebug("Missing shareViewNavigationController")
 | |
|             return
 | |
|         }
 | |
|         shareViewNavigationController.setViewControllers([viewController], animated: false)
 | |
|         if self.presentedViewController == nil {
 | |
|             Logger.debug("presenting modally: \(viewController)")
 | |
|             self.present(shareViewNavigationController, animated: true)
 | |
|         } else {
 | |
|             Logger.debug("modal already presented. swapping modal content for: \(viewController)")
 | |
|             assert(self.presentedViewController == shareViewNavigationController)
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     private func buildAttachmentsAndPresentConversationPicker() {
 | |
|         AssertIsOnMainThread()
 | |
| 
 | |
|         self.buildAttachments().map { [weak self] attachments in
 | |
|             AssertIsOnMainThread()
 | |
|             guard let strongSelf = self else { return }
 | |
| 
 | |
|             strongSelf.progressPoller = nil
 | |
|             strongSelf.loadViewController = nil
 | |
| 
 | |
|             let conversationPicker = SharingThreadPickerViewController(shareViewDelegate: strongSelf)
 | |
|             Logger.debug("presentConversationPicker: \(conversationPicker)")
 | |
|             conversationPicker.attachments = attachments
 | |
|             strongSelf.showPrimaryViewController(conversationPicker)
 | |
|             Logger.info("showing picker with attachments: \(attachments)")
 | |
|         }.catch { [weak self] error in
 | |
|             AssertIsOnMainThread()
 | |
|             guard let strongSelf = self else { return }
 | |
| 
 | |
|             let alertTitle = NSLocalizedString("SHARE_EXTENSION_UNABLE_TO_BUILD_ATTACHMENT_ALERT_TITLE",
 | |
|                                                comment: "Shown when trying to share content to a Signal user for the share extension. Followed by failure details.")
 | |
|             OWSAlerts.showAlert(title: alertTitle,
 | |
|                                 message: error.localizedDescription,
 | |
|                                 buttonTitle: CommonStrings.cancelButton) { _ in
 | |
|                                     strongSelf.shareViewWasCancelled()
 | |
|             }
 | |
|             owsFailDebug("building attachment failed with error: \(error)")
 | |
|         }.retainUntilComplete()
 | |
|     }
 | |
| 
 | |
|     private func presentScreenLock() {
 | |
|         AssertIsOnMainThread()
 | |
| 
 | |
|         let screenLockUI = SAEScreenLockViewController(shareViewDelegate: self)
 | |
|         Logger.debug("presentScreenLock: \(screenLockUI)")
 | |
|         showPrimaryViewController(screenLockUI)
 | |
|         Logger.info("showing screen lock")
 | |
|     }
 | |
| 
 | |
|     private class func itemMatchesSpecificUtiType(itemProvider: NSItemProvider, utiType: String) -> Bool {
 | |
|         // URLs, contacts and other special items have to be detected separately.
 | |
|         // Many shares (e.g. pdfs) will register many UTI types and/or conform to kUTTypeData.
 | |
|         guard itemProvider.registeredTypeIdentifiers.count == 1 else {
 | |
|             return false
 | |
|         }
 | |
|         guard let firstUtiType = itemProvider.registeredTypeIdentifiers.first else {
 | |
|             return false
 | |
|         }
 | |
|         return firstUtiType == utiType
 | |
|     }
 | |
| 
 | |
|     private class func isVisualMediaItem(itemProvider: NSItemProvider) -> Bool {
 | |
|         return (itemProvider.hasItemConformingToTypeIdentifier(kUTTypeImage as String) ||
 | |
|             itemProvider.hasItemConformingToTypeIdentifier(kUTTypeMovie as String))
 | |
|     }
 | |
| 
 | |
|     private class func isUrlItem(itemProvider: NSItemProvider) -> Bool {
 | |
|         return itemMatchesSpecificUtiType(itemProvider: itemProvider,
 | |
|                                           utiType: kUTTypeURL as String)
 | |
|     }
 | |
| 
 | |
|     private class func isContactItem(itemProvider: NSItemProvider) -> Bool {
 | |
|         return itemMatchesSpecificUtiType(itemProvider: itemProvider,
 | |
|                                           utiType: kUTTypeContact as String)
 | |
|     }
 | |
| 
 | |
|     private class func utiType(itemProvider: NSItemProvider) -> String? {
 | |
|         Logger.info("utiTypeForItem: \(itemProvider.registeredTypeIdentifiers)")
 | |
| 
 | |
|         if isUrlItem(itemProvider: itemProvider) {
 | |
|             return kUTTypeURL as String
 | |
|         } else if isContactItem(itemProvider: itemProvider) {
 | |
|             return kUTTypeContact as String
 | |
|         }
 | |
| 
 | |
|         // Use the first UTI that conforms to "data".
 | |
|         let matchingUtiType = itemProvider.registeredTypeIdentifiers.first { (utiType: String) -> Bool in
 | |
|             UTTypeConformsTo(utiType as CFString, kUTTypeData)
 | |
|         }
 | |
|         return matchingUtiType
 | |
|     }
 | |
| 
 | |
|     private class func createDataSource(utiType: String, url: URL, customFileName: String?) -> DataSource? {
 | |
|         if utiType == (kUTTypeURL as String) {
 | |
|             // Share URLs as oversize text messages whose text content is the URL.
 | |
|             //
 | |
|             // NOTE: SharingThreadPickerViewController will try to unpack them
 | |
|             //       and send them as normal text messages if possible.
 | |
|             let urlString = url.absoluteString
 | |
|             return DataSourceValue.dataSource(withOversizeText: urlString)
 | |
|         } else if UTTypeConformsTo(utiType as CFString, kUTTypeText) {
 | |
|             // Share text as oversize text messages.
 | |
|             //
 | |
|             // NOTE: SharingThreadPickerViewController will try to unpack them
 | |
|             //       and send them as normal text messages if possible.
 | |
|             return DataSourcePath.dataSource(with: url,
 | |
|                                              shouldDeleteOnDeallocation: false)
 | |
|         } else {
 | |
|             guard let dataSource = DataSourcePath.dataSource(with: url,
 | |
|                                                              shouldDeleteOnDeallocation: false) else {
 | |
|                 return nil
 | |
|             }
 | |
| 
 | |
|             if let customFileName = customFileName {
 | |
|                 dataSource.sourceFilename = customFileName
 | |
|             } else {
 | |
|                 // Ignore the filename for URLs.
 | |
|                 dataSource.sourceFilename = url.lastPathComponent
 | |
|             }
 | |
|             return dataSource
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     private class func preferredItemProviders(inputItem: NSExtensionItem) -> [NSItemProvider]? {
 | |
|         guard let attachments = inputItem.attachments else {
 | |
|             return nil
 | |
|         }
 | |
| 
 | |
|         var visualMediaItemProviders = [NSItemProvider]()
 | |
|         var hasNonVisualMedia = false
 | |
|         for attachment in attachments {
 | |
|             guard let itemProvider = attachment as? NSItemProvider else {
 | |
|                 owsFailDebug("Unexpected attachment type: \(String(describing: attachment))")
 | |
|                 continue
 | |
|             }
 | |
|             if isVisualMediaItem(itemProvider: itemProvider) {
 | |
|                 visualMediaItemProviders.append(itemProvider)
 | |
|             } else {
 | |
|                 hasNonVisualMedia = true
 | |
|             }
 | |
|         }
 | |
|         // Only allow multiple-attachment sends if all attachments
 | |
|         // are visual media.
 | |
|         if visualMediaItemProviders.count > 0 && !hasNonVisualMedia {
 | |
|             return visualMediaItemProviders
 | |
|         }
 | |
| 
 | |
|         // A single inputItem can have multiple attachments, e.g. sharing from Firefox gives
 | |
|         // one url attachment and another text attachment, where the the url would be https://some-news.com/articles/123-cat-stuck-in-tree
 | |
|         // and the text attachment would be something like "Breaking news - cat stuck in tree"
 | |
|         //
 | |
|         // FIXME: For now, we prefer the URL provider and discard the text provider, since it's more useful to share the URL than the caption
 | |
|         // but we *should* include both. This will be a bigger change though since our share extension is currently heavily predicated
 | |
|         // on one itemProvider per share.
 | |
| 
 | |
|         // Prefer a URL provider if available
 | |
|         if let preferredAttachment = attachments.first(where: { (attachment: Any) -> Bool in
 | |
|             guard let itemProvider = attachment as? NSItemProvider else {
 | |
|                 return false
 | |
|             }
 | |
|             return isUrlItem(itemProvider: itemProvider)
 | |
|         }) {
 | |
|             if let itemProvider = preferredAttachment as? NSItemProvider {
 | |
|                 return [itemProvider]
 | |
|             } else {
 | |
|                 owsFailDebug("Unexpected attachment type: \(String(describing: preferredAttachment))")
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // else return whatever is available
 | |
|         if let itemProvider = inputItem.attachments?.first as? NSItemProvider {
 | |
|             return [itemProvider]
 | |
|         } else {
 | |
|             owsFailDebug("Missing attachment.")
 | |
|         }
 | |
|         return []
 | |
|     }
 | |
| 
 | |
|     private func selectItemProviders() -> Promise<[NSItemProvider]> {
 | |
|         guard let inputItems = self.extensionContext?.inputItems else {
 | |
|             let error = ShareViewControllerError.assertionError(description: "no input item")
 | |
|             return Promise(error: error)
 | |
|         }
 | |
| 
 | |
|         for inputItemRaw in inputItems {
 | |
|             guard let inputItem = inputItemRaw as? NSExtensionItem else {
 | |
|                 Logger.error("invalid inputItem \(inputItemRaw)")
 | |
|                 continue
 | |
|             }
 | |
|             if let itemProviders = ShareViewController.preferredItemProviders(inputItem: inputItem) {
 | |
|                 return Promise.value(itemProviders)
 | |
|             }
 | |
|         }
 | |
|         let error = ShareViewControllerError.assertionError(description: "no input item")
 | |
|         return Promise(error: error)
 | |
|     }
 | |
| 
 | |
|     private
 | |
|     struct LoadedItem {
 | |
|         let itemProvider: NSItemProvider
 | |
|         let itemUrl: URL
 | |
|         let utiType: String
 | |
| 
 | |
|         var customFileName: String?
 | |
|         var isConvertibleToTextMessage = false
 | |
|         var isConvertibleToContactShare = false
 | |
| 
 | |
|         init(itemProvider: NSItemProvider,
 | |
|              itemUrl: URL,
 | |
|              utiType: String,
 | |
|              customFileName: String? = nil,
 | |
|              isConvertibleToTextMessage: Bool = false,
 | |
|              isConvertibleToContactShare: Bool = false) {
 | |
|             self.itemProvider = itemProvider
 | |
|             self.itemUrl = itemUrl
 | |
|             self.utiType = utiType
 | |
|             self.customFileName = customFileName
 | |
|             self.isConvertibleToTextMessage = isConvertibleToTextMessage
 | |
|             self.isConvertibleToContactShare = isConvertibleToContactShare
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     private func loadItemProvider(itemProvider: NSItemProvider) -> Promise<LoadedItem> {
 | |
|         Logger.info("attachment: \(itemProvider)")
 | |
| 
 | |
|         // We need to be very careful about which UTI type we use.
 | |
|         //
 | |
|         // * In the case of "textual" shares (e.g. web URLs and text snippets), we want to
 | |
|         //   coerce the UTI type to kUTTypeURL or kUTTypeText.
 | |
|         // * We want to treat shared files as file attachments.  Therefore we do not
 | |
|         //   want to treat file URLs like web URLs.
 | |
|         // * UTIs aren't very descriptive (there are far more MIME types than UTI types)
 | |
|         //   so in the case of file attachments we try to refine the attachment type
 | |
|         //   using the file extension.
 | |
|         guard let srcUtiType = ShareViewController.utiType(itemProvider: itemProvider) else {
 | |
|             let error = ShareViewControllerError.unsupportedMedia
 | |
|             return Promise(error: error)
 | |
|         }
 | |
|         Logger.debug("matched utiType: \(srcUtiType)")
 | |
| 
 | |
|         let (promise, resolver) = Promise<LoadedItem>.pending()
 | |
| 
 | |
|         let loadCompletion: NSItemProvider.CompletionHandler = { [weak self]
 | |
|             (value, error) in
 | |
| 
 | |
|             guard let _ = self else { return }
 | |
|             guard error == nil else {
 | |
|                 resolver.reject(error!)
 | |
|                 return
 | |
|             }
 | |
| 
 | |
|             guard let value = value else {
 | |
|                 let missingProviderError = ShareViewControllerError.assertionError(description: "missing item provider")
 | |
|                 resolver.reject(missingProviderError)
 | |
|                 return
 | |
|             }
 | |
| 
 | |
|             Logger.info("value type: \(type(of: value))")
 | |
| 
 | |
|             if let data = value as? Data {
 | |
|                 let customFileName = "Contact.vcf"
 | |
|                 var isConvertibleToContactShare = false
 | |
| 
 | |
|                 // Although we don't support contacts _yet_, when we do we'll want to make
 | |
|                 // sure they are shared with a reasonable filename.
 | |
|                 if ShareViewController.itemMatchesSpecificUtiType(itemProvider: itemProvider,
 | |
|                                                                   utiType: kUTTypeVCard as String) {
 | |
| 
 | |
|                     if Contact(vCardData: data) != nil {
 | |
|                         isConvertibleToContactShare = true
 | |
|                     } else {
 | |
|                         Logger.error("could not parse vcard.")
 | |
|                         let writeError = ShareViewControllerError.assertionError(description: "Could not parse vcard data.")
 | |
|                         resolver.reject(writeError)
 | |
|                         return
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 let customFileExtension = MIMETypeUtil.fileExtension(forUTIType: srcUtiType)
 | |
|                 guard let tempFilePath = OWSFileSystem.writeData(toTemporaryFile: data, fileExtension: customFileExtension) else {
 | |
|                     let writeError = ShareViewControllerError.assertionError(description: "Error writing item data: \(String(describing: error))")
 | |
|                     resolver.reject(writeError)
 | |
|                     return
 | |
|                 }
 | |
|                 let fileUrl = URL(fileURLWithPath: tempFilePath)
 | |
|                 resolver.fulfill(LoadedItem(itemProvider: itemProvider,
 | |
|                                             itemUrl: fileUrl,
 | |
|                                             utiType: srcUtiType,
 | |
|                                             customFileName: customFileName,
 | |
|                                             isConvertibleToContactShare: isConvertibleToContactShare))
 | |
|             } else if let string = value as? String {
 | |
|                 Logger.debug("string provider: \(string)")
 | |
|                 guard let data = string.filterStringForDisplay().data(using: String.Encoding.utf8) else {
 | |
|                     let writeError = ShareViewControllerError.assertionError(description: "Error writing item data: \(String(describing: error))")
 | |
|                     resolver.reject(writeError)
 | |
|                     return
 | |
|                 }
 | |
|                 guard let tempFilePath = OWSFileSystem.writeData(toTemporaryFile: data, fileExtension: "txt") else {
 | |
|                     let writeError = ShareViewControllerError.assertionError(description: "Error writing item data: \(String(describing: error))")
 | |
|                     resolver.reject(writeError)
 | |
|                     return
 | |
|                 }
 | |
| 
 | |
|                 let fileUrl = URL(fileURLWithPath: tempFilePath)
 | |
| 
 | |
|                 let isConvertibleToTextMessage = !itemProvider.registeredTypeIdentifiers.contains(kUTTypeFileURL as String)
 | |
| 
 | |
|                 if UTTypeConformsTo(srcUtiType as CFString, kUTTypeText) {
 | |
|                     resolver.fulfill(LoadedItem(itemProvider: itemProvider,
 | |
|                                                 itemUrl: fileUrl,
 | |
|                                                 utiType: srcUtiType,
 | |
|                                                 isConvertibleToTextMessage: isConvertibleToTextMessage))
 | |
|                 } else {
 | |
|                     resolver.fulfill(LoadedItem(itemProvider: itemProvider,
 | |
|                                                 itemUrl: fileUrl,
 | |
|                                                 utiType: kUTTypeText as String,
 | |
|                                                 isConvertibleToTextMessage: isConvertibleToTextMessage))
 | |
|                 }
 | |
|             } else if let url = value as? URL {
 | |
|                 // If the share itself is a URL (e.g. a link from Safari), try to send this as a text message.
 | |
|                 let isConvertibleToTextMessage = (itemProvider.registeredTypeIdentifiers.contains(kUTTypeURL as String) &&
 | |
|                     !itemProvider.registeredTypeIdentifiers.contains(kUTTypeFileURL as String))
 | |
|                 if isConvertibleToTextMessage {
 | |
|                     resolver.fulfill(LoadedItem(itemProvider: itemProvider,
 | |
|                                                 itemUrl: url,
 | |
|                                                 utiType: kUTTypeURL as String,
 | |
|                                                 isConvertibleToTextMessage: isConvertibleToTextMessage))
 | |
|                 } else {
 | |
|                     resolver.fulfill(LoadedItem(itemProvider: itemProvider,
 | |
|                                                 itemUrl: url,
 | |
|                                                 utiType: srcUtiType,
 | |
|                                                 isConvertibleToTextMessage: isConvertibleToTextMessage))
 | |
|                 }
 | |
|             } else if let image = value as? UIImage {
 | |
|                 if let data = image.pngData() {
 | |
|                     let tempFilePath = OWSFileSystem.temporaryFilePath(withFileExtension: "png")
 | |
|                     do {
 | |
|                         let url = NSURL.fileURL(withPath: tempFilePath)
 | |
|                         try data.write(to: url)
 | |
|                         resolver.fulfill(LoadedItem(itemProvider: itemProvider, itemUrl: url,
 | |
|                                                     utiType: srcUtiType))
 | |
|                     } catch {
 | |
|                         resolver.reject(ShareViewControllerError.assertionError(description: "couldn't write UIImage: \(String(describing: error))"))
 | |
|                     }
 | |
|                 } else {
 | |
|                     resolver.reject(ShareViewControllerError.assertionError(description: "couldn't convert UIImage to PNG: \(String(describing: error))"))
 | |
|                 }
 | |
|             } else {
 | |
|                 // It's unavoidable that we may sometimes receives data types that we
 | |
|                 // don't know how to handle.
 | |
|                 let unexpectedTypeError = ShareViewControllerError.assertionError(description: "unexpected value: \(String(describing: value))")
 | |
|                 resolver.reject(unexpectedTypeError)
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         itemProvider.loadItem(forTypeIdentifier: srcUtiType, options: nil, completionHandler: loadCompletion)
 | |
| 
 | |
|         return promise
 | |
|     }
 | |
| 
 | |
|     private func buildAttachment(forLoadedItem loadedItem: LoadedItem) -> Promise<SignalAttachment> {
 | |
|         let itemProvider = loadedItem.itemProvider
 | |
|         let itemUrl = loadedItem.itemUrl
 | |
|         let utiType = loadedItem.utiType
 | |
| 
 | |
|         var url = itemUrl
 | |
|         do {
 | |
|             if isVideoNeedingRelocation(itemProvider: itemProvider, itemUrl: itemUrl) {
 | |
|                 url = try SignalAttachment.copyToVideoTempDir(url: itemUrl)
 | |
|             }
 | |
|         } catch {
 | |
|             let error = ShareViewControllerError.assertionError(description: "Could not copy video")
 | |
|             return Promise(error: error)
 | |
|         }
 | |
| 
 | |
|         Logger.debug("building DataSource with url: \(url), utiType: \(utiType)")
 | |
| 
 | |
|         guard let dataSource = ShareViewController.createDataSource(utiType: utiType, url: url, customFileName: loadedItem.customFileName) else {
 | |
|             let error = ShareViewControllerError.assertionError(description: "Unable to read attachment data")
 | |
|             return Promise(error: error)
 | |
|         }
 | |
| 
 | |
|         // start with base utiType, but it might be something generic like "image"
 | |
|         var specificUTIType = utiType
 | |
|         if utiType == (kUTTypeURL as String) {
 | |
|             // Use kUTTypeURL for URLs.
 | |
|         } else if UTTypeConformsTo(utiType as CFString, kUTTypeText) {
 | |
|             // Use kUTTypeText for text.
 | |
|         } else if url.pathExtension.count > 0 {
 | |
|             // Determine a more specific utiType based on file extension
 | |
|             if let typeExtension = MIMETypeUtil.utiType(forFileExtension: url.pathExtension) {
 | |
|                 Logger.debug("utiType based on extension: \(typeExtension)")
 | |
|                 specificUTIType = typeExtension
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         guard !SignalAttachment.isInvalidVideo(dataSource: dataSource, dataUTI: specificUTIType) else {
 | |
|             // This can happen, e.g. when sharing a quicktime-video from iCloud drive.
 | |
| 
 | |
|             let (promise, exportSession) = SignalAttachment.compressVideoAsMp4(dataSource: dataSource, dataUTI: specificUTIType)
 | |
| 
 | |
|             // TODO: How can we move waiting for this export to the end of the share flow rather than having to do it up front?
 | |
|             // Ideally we'd be able to start it here, and not block the UI on conversion unless there's still work to be done
 | |
|             // when the user hits "send".
 | |
|             if let exportSession = exportSession {
 | |
|                 let progressPoller = ProgressPoller(timeInterval: 0.1, ratioCompleteBlock: { return exportSession.progress })
 | |
|                 self.progressPoller = progressPoller
 | |
|                 progressPoller.startPolling()
 | |
| 
 | |
|                 guard let loadViewController = self.loadViewController else {
 | |
|                     owsFailDebug("load view controller was unexpectedly nil")
 | |
|                     return promise
 | |
|                 }
 | |
| 
 | |
|                 DispatchQueue.main.async {
 | |
|                     loadViewController.progress = progressPoller.progress
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             return promise
 | |
|         }
 | |
| 
 | |
|         let attachment = SignalAttachment.attachment(dataSource: dataSource, dataUTI: specificUTIType, imageQuality: .original)
 | |
|         if loadedItem.isConvertibleToContactShare {
 | |
|             Logger.info("isConvertibleToContactShare")
 | |
|             attachment.isConvertibleToContactShare = true
 | |
|         } else if loadedItem.isConvertibleToTextMessage {
 | |
|             Logger.info("isConvertibleToTextMessage")
 | |
|             attachment.isConvertibleToTextMessage = true
 | |
|         }
 | |
|         return Promise.value(attachment)
 | |
|     }
 | |
| 
 | |
|     private func buildAttachments() -> Promise<[SignalAttachment]> {
 | |
|         return selectItemProviders().then {  [weak self] (itemProviders) -> Promise<[SignalAttachment]> in
 | |
|             guard let strongSelf = self else {
 | |
|                 let error = ShareViewControllerError.assertionError(description: "expired")
 | |
|                 return Promise(error: error)
 | |
|             }
 | |
| 
 | |
|             var loadPromises = [Promise<SignalAttachment>]()
 | |
| 
 | |
|             for itemProvider in itemProviders.prefix(SignalAttachment.maxAttachmentsAllowed) {
 | |
|                 let loadPromise = strongSelf.loadItemProvider(itemProvider: itemProvider)
 | |
|                     .then({ (loadedItem) -> Promise<SignalAttachment> in
 | |
|                         return strongSelf.buildAttachment(forLoadedItem: loadedItem)
 | |
|                     })
 | |
| 
 | |
|                 loadPromises.append(loadPromise)
 | |
|             }
 | |
|             return when(fulfilled: loadPromises)
 | |
|         }.map { (signalAttachments) -> [SignalAttachment] in
 | |
|             guard signalAttachments.count > 0 else {
 | |
|                 let error = ShareViewControllerError.assertionError(description: "no valid attachments")
 | |
|                 throw error
 | |
|             }
 | |
|             return signalAttachments
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // Some host apps (e.g. iOS Photos.app) sometimes auto-converts some video formats (e.g. com.apple.quicktime-movie)
 | |
|     // into mp4s as part of the NSItemProvider `loadItem` API. (Some files the Photo's app doesn't auto-convert)
 | |
|     //
 | |
|     // However, when using this url to the converted item, AVFoundation operations such as generating a
 | |
|     // preview image and playing the url in the AVMoviePlayer fails with an unhelpful error: "The operation could not be completed"
 | |
|     //
 | |
|     // We can work around this by first copying the media into our container.
 | |
|     //
 | |
|     // I don't understand why this is, and I haven't found any relevant documentation in the NSItemProvider
 | |
|     // or AVFoundation docs.
 | |
|     //
 | |
|     // Notes:
 | |
|     //
 | |
|     // These operations succeed when sending a video which initially existed on disk as an mp4.
 | |
|     // (e.g. Alice sends a video to Bob through the main app, which ensures it's an mp4. Bob saves it, then re-shares it)
 | |
|     //
 | |
|     // I *did* verify that the size and SHA256 sum of the original url matches that of the copied url. So there
 | |
|     // is no difference between the contents of the file, yet one works one doesn't.
 | |
|     // Perhaps the AVFoundation APIs require some extra file system permssion we don't have in the
 | |
|     // passed through URL.
 | |
|     private func isVideoNeedingRelocation(itemProvider: NSItemProvider, itemUrl: URL) -> Bool {
 | |
|         let pathExtension = itemUrl.pathExtension
 | |
|         guard pathExtension.count > 0 else {
 | |
|             Logger.verbose("item URL has no file extension: \(itemUrl).")
 | |
|             return false
 | |
|         }
 | |
|         guard let utiTypeForURL = MIMETypeUtil.utiType(forFileExtension: pathExtension) else {
 | |
|             Logger.verbose("item has unknown UTI type: \(itemUrl).")
 | |
|             return false
 | |
|         }
 | |
|         Logger.verbose("utiTypeForURL: \(utiTypeForURL)")
 | |
|         guard utiTypeForURL == kUTTypeMPEG4 as String else {
 | |
|             // Either it's not a video or it was a video which was not auto-converted to mp4.
 | |
|             // Not affected by the issue.
 | |
|             return false
 | |
|         }
 | |
| 
 | |
|         // If video file already existed on disk as an mp4, then the host app didn't need to
 | |
|         // apply any conversion, so no need to relocate the app.
 | |
|         return !itemProvider.registeredTypeIdentifiers.contains(kUTTypeMPEG4 as String)
 | |
|     }
 | |
| 
 | |
|     // MARK: App Mode
 | |
| 
 | |
|     public func getCurrentAppMode() -> AppMode {
 | |
|         guard let window = self.view.window else { return .light }
 | |
|         let userInterfaceStyle = window.traitCollection.userInterfaceStyle
 | |
|         let isLightMode = (userInterfaceStyle == .light || userInterfaceStyle == .unspecified)
 | |
|         return isLightMode ? .light : .dark
 | |
|     }
 | |
| 
 | |
|     public func setCurrentAppMode(to appMode: AppMode) {
 | |
|         return // Not applicable to share extensions
 | |
|     }
 | |
| }
 | |
| 
 | |
| // Exposes a Progress object, whose progress is updated by polling the return of a given block
 | |
| private class ProgressPoller: NSObject {
 | |
| 
 | |
|     let progress: Progress
 | |
|     private(set) var timer: Timer?
 | |
| 
 | |
|     // Higher number offers higher ganularity
 | |
|     let progressTotalUnitCount: Int64 = 10000
 | |
|     private let timeInterval: Double
 | |
|     private let ratioCompleteBlock: () -> Float
 | |
| 
 | |
|     init(timeInterval: TimeInterval, ratioCompleteBlock: @escaping () -> Float) {
 | |
|         self.timeInterval = timeInterval
 | |
|         self.ratioCompleteBlock = ratioCompleteBlock
 | |
| 
 | |
|         self.progress = Progress()
 | |
| 
 | |
|         progress.totalUnitCount = progressTotalUnitCount
 | |
|         progress.completedUnitCount = Int64(ratioCompleteBlock() * Float(progressTotalUnitCount))
 | |
|     }
 | |
| 
 | |
|     func startPolling() {
 | |
|         guard self.timer == nil else {
 | |
|             owsFailDebug("already started timer")
 | |
|             return
 | |
|         }
 | |
| 
 | |
|         self.timer = WeakTimer.scheduledTimer(timeInterval: timeInterval, target: self, userInfo: nil, repeats: true) { [weak self] (timer) in
 | |
|             guard let strongSelf = self else {
 | |
|                 return
 | |
|             }
 | |
| 
 | |
|             let completedUnitCount = Int64(strongSelf.ratioCompleteBlock() * Float(strongSelf.progressTotalUnitCount))
 | |
|             strongSelf.progress.completedUnitCount = completedUnitCount
 | |
| 
 | |
|             if completedUnitCount == strongSelf.progressTotalUnitCount {
 | |
|                 Logger.debug("progress complete")
 | |
|                 timer.invalidate()
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 |