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.
824 lines
34 KiB
Swift
824 lines
34 KiB
Swift
//
|
|
// Copyright (c) 2018 Open Whisper Systems. All rights reserved.
|
|
//
|
|
|
|
import UIKit
|
|
|
|
import SignalMessaging
|
|
import PureLayout
|
|
import SignalServiceKit
|
|
import PromiseKit
|
|
|
|
@objc
|
|
public class ShareViewController: UIViewController, ShareViewDelegate, SAEFailedViewDelegate {
|
|
|
|
enum ShareViewControllerError: Error {
|
|
case assertionError(description: String)
|
|
case unsupportedMedia
|
|
case notRegistered()
|
|
}
|
|
|
|
private var hasInitialRootViewController = false
|
|
private var isReadyForAppExtensions = false
|
|
private var areVersionMigrationsComplete = false
|
|
|
|
private var progressPoller: ProgressPoller?
|
|
var loadViewController: SAELoadViewController?
|
|
|
|
let shareViewNavigationController: UINavigationController = UINavigationController()
|
|
|
|
override open func loadView() {
|
|
super.loadView()
|
|
Logger.debug("\(self.logTag) \(#function)")
|
|
|
|
// This should be the first thing we do.
|
|
let appContext = ShareAppExtensionContext(rootViewController:self)
|
|
SetCurrentAppContext(appContext)
|
|
|
|
DebugLogger.shared().enableTTYLogging()
|
|
if _isDebugAssertConfiguration() {
|
|
DebugLogger.shared().enableFileLogging()
|
|
} else if OWSPreferences.isLoggingEnabled() {
|
|
DebugLogger.shared().enableFileLogging()
|
|
}
|
|
|
|
_ = AppVersion()
|
|
|
|
startupLogging()
|
|
|
|
SetRandFunctionSeed()
|
|
|
|
// We don't need to use DeviceSleepManager in the SAE.
|
|
|
|
// TODO: Do we need to applySignalAppearence in the SAE?
|
|
UIUtil.applySignalAppearence()
|
|
|
|
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 TSSStorageManager,
|
|
// 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 {
|
|
// If we don't have TSSStorageManager, we can't consult TSAccountManager
|
|
// for isRegistered, so we use OWSPreferences which is usually-accurate
|
|
// copy of that state.
|
|
if OWSPreferences.isRegistered() {
|
|
showNotReadyView()
|
|
} else {
|
|
showNotRegisteredView()
|
|
}
|
|
return
|
|
}
|
|
|
|
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).then { () -> Void in
|
|
guard self.presentedViewController == nil else {
|
|
Logger.debug("\(self.logTag) setup completed quickly, no need to present load view controller.")
|
|
return
|
|
}
|
|
|
|
Logger.debug("\(self.logTag) setup is slow - showing loading screen")
|
|
self.showPrimaryViewController(loadViewController)
|
|
}.retainUntilComplete()
|
|
|
|
// We shouldn't set up our environment until after we've consulted isReadyForAppExtensions.
|
|
AppSetup.setupEnvironment({
|
|
return NoopCallMessageHandler()
|
|
}) {
|
|
return NoopNotificationsManager()
|
|
}
|
|
|
|
// performUpdateCheck must be invoked after Environment has been initialized because
|
|
// upgrade process may depend on Environment.
|
|
VersionMigrations.performUpdateCheck(completion: {
|
|
AssertIsOnMainThread()
|
|
|
|
self.versionMigrationsDidComplete()
|
|
})
|
|
|
|
// We don't need to use "screen protection" in the SAE.
|
|
|
|
// Ensure OWSContactsSyncing is instantiated.
|
|
OWSContactsSyncing.sharedManager()
|
|
|
|
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)
|
|
|
|
Logger.info("\(self.logTag) application: didFinishLaunchingWithOptions completed.")
|
|
|
|
OWSAnalytics.appLaunchDidBegin()
|
|
}
|
|
|
|
deinit {
|
|
Logger.info("\(self.logTag) dealloc")
|
|
NotificationCenter.default.removeObserver(self)
|
|
}
|
|
|
|
private func activate() {
|
|
Logger.debug("\(self.logTag) \(#function)")
|
|
|
|
// 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 strongSelf = self else { return }
|
|
Logger.info("\(strongSelf.logTag) 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("\(self.logTag) 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 strongSelf = self else { return }
|
|
Logger.info("\(strongSelf.logTag) running post launch block for registered user: \(TSAccountManager.localNumber)")
|
|
|
|
// We don't need to use the TSSocketManager in the SAE.
|
|
|
|
Environment.current().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("\(self.logTag) \(#function)")
|
|
|
|
areVersionMigrationsComplete = true
|
|
|
|
checkIsAppReady()
|
|
}
|
|
|
|
@objc
|
|
func storageIsReady() {
|
|
AssertIsOnMainThread()
|
|
|
|
Logger.debug("\(self.logTag) \(#function)")
|
|
|
|
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("\(self.logTag) \(#function)")
|
|
|
|
// 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("\(self.logTag) 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.instance().saeLaunchDidComplete()
|
|
|
|
Environment.current().contactsManager.loadSignalAccountsFromCache()
|
|
|
|
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 OWSOrphanedDataCleaner in the SAE.
|
|
|
|
// We don't need to fetch the local profile in the SAE
|
|
|
|
OWSReadReceiptManager.shared().prepareCachedValues()
|
|
}
|
|
|
|
@objc
|
|
func registrationStateDidChange() {
|
|
AssertIsOnMainThread()
|
|
|
|
Logger.debug("\(self.logTag) \(#function)")
|
|
|
|
if TSAccountManager.isRegistered() {
|
|
Logger.info("\(self.logTag) 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() {
|
|
Logger.debug("\(self.logTag) \(#function)")
|
|
|
|
guard AppReadiness.isAppReady() else {
|
|
return
|
|
}
|
|
guard !hasInitialRootViewController else {
|
|
return
|
|
}
|
|
hasInitialRootViewController = true
|
|
|
|
Logger.info("Presenting initial root view controller")
|
|
|
|
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.presentConversationPicker()
|
|
}
|
|
}
|
|
|
|
// 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 {
|
|
owsFail("Locale Identifier: Unknown")
|
|
}
|
|
if let countryCode = locale.object(forKey:NSLocale.Key.countryCode) as? String,
|
|
countryCode.count > 0 {
|
|
Logger.info("Country Code: \(countryCode)")
|
|
} else {
|
|
owsFail("Country Code: Unknown")
|
|
}
|
|
if let languageCode = locale.object(forKey:NSLocale.Key.languageCode) as? String,
|
|
languageCode.count > 0 {
|
|
Logger.info("Language Code: \(languageCode)")
|
|
} else {
|
|
owsFail("Language Code: Unknown")
|
|
}
|
|
}
|
|
|
|
// MARK: Error Views
|
|
|
|
private func showNotReadyView() {
|
|
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() {
|
|
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) {
|
|
let viewController = SAEFailedViewController(delegate:self, title:title, message:message)
|
|
self.showPrimaryViewController(viewController)
|
|
}
|
|
|
|
// MARK: View Lifecycle
|
|
|
|
override open func viewDidLoad() {
|
|
super.viewDidLoad()
|
|
|
|
Logger.debug("\(self.logTag) \(#function)")
|
|
|
|
if isReadyForAppExtensions {
|
|
AppReadiness.runNowOrWhenAppIsReady {
|
|
self.activate()
|
|
}
|
|
}
|
|
}
|
|
|
|
override open func viewWillAppear(_ animated: Bool) {
|
|
Logger.debug("\(self.logTag) \(#function)")
|
|
|
|
super.viewWillAppear(animated)
|
|
}
|
|
|
|
override open func viewDidAppear(_ animated: Bool) {
|
|
Logger.debug("\(self.logTag) \(#function)")
|
|
|
|
super.viewDidAppear(animated)
|
|
}
|
|
|
|
override open func viewWillDisappear(_ animated: Bool) {
|
|
Logger.debug("\(self.logTag) \(#function)")
|
|
|
|
super.viewWillDisappear(animated)
|
|
|
|
Logger.flush()
|
|
}
|
|
|
|
override open func viewDidDisappear(_ animated: Bool) {
|
|
Logger.debug("\(self.logTag) \(#function)")
|
|
|
|
super.viewDidDisappear(animated)
|
|
|
|
Logger.flush()
|
|
}
|
|
|
|
@objc
|
|
func owsApplicationWillEnterForeground() throws {
|
|
AssertIsOnMainThread()
|
|
|
|
Logger.debug("\(self.logTag) \(#function)")
|
|
|
|
// 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 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 shareViewWasCompleted() {
|
|
Logger.info("\(self.logTag) \(#function)")
|
|
|
|
self.dismiss(animated: true) {
|
|
self.extensionContext!.completeRequest(returningItems: [], completionHandler: nil)
|
|
}
|
|
}
|
|
|
|
public func shareViewWasCancelled() {
|
|
Logger.info("\(self.logTag) \(#function)")
|
|
|
|
self.dismiss(animated: true) {
|
|
self.extensionContext!.completeRequest(returningItems: [], completionHandler: nil)
|
|
}
|
|
}
|
|
|
|
public func shareViewFailed(error: Error) {
|
|
Logger.info("\(self.logTag) \(#function)")
|
|
|
|
self.dismiss(animated: true) {
|
|
self.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) {
|
|
shareViewNavigationController.setViewControllers([viewController], animated: false)
|
|
if self.presentedViewController == nil {
|
|
Logger.debug("\(self.logTag) presenting modally: \(viewController)")
|
|
self.present(shareViewNavigationController, animated: true)
|
|
} else {
|
|
Logger.debug("\(self.logTag) modal already presented. swapping modal content for: \(viewController)")
|
|
assert(self.presentedViewController == shareViewNavigationController)
|
|
}
|
|
}
|
|
|
|
private func presentConversationPicker() {
|
|
self.buildAttachment().then { attachment -> Void in
|
|
let conversationPicker = SharingThreadPickerViewController(shareViewDelegate: self)
|
|
conversationPicker.attachment = attachment
|
|
self.progressPoller = nil
|
|
self.loadViewController = nil
|
|
self.showPrimaryViewController(conversationPicker)
|
|
Logger.info("showing picker with attachment: \(attachment)")
|
|
}.catch { error in
|
|
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(withTitle: alertTitle,
|
|
message: error.localizedDescription,
|
|
buttonTitle: CommonStrings.cancelButton) { _ in
|
|
self.shareViewWasCancelled()
|
|
}
|
|
owsFail("\(self.logTag) building attachment failed with error: \(error)")
|
|
}.retainUntilComplete()
|
|
}
|
|
|
|
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 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 utiTypeForItem(itemProvider: NSItemProvider) -> String? {
|
|
Logger.info("\(self.logTag) 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)
|
|
} else {
|
|
guard let dataSource = DataSourcePath.dataSource(with: url) else {
|
|
return nil
|
|
}
|
|
|
|
if let customFileName = customFileName {
|
|
dataSource.sourceFilename = customFileName
|
|
} else {
|
|
// Ignore the filename for URLs.
|
|
dataSource.sourceFilename = url.lastPathComponent
|
|
}
|
|
return dataSource
|
|
}
|
|
}
|
|
|
|
private func buildAttachment() -> Promise<SignalAttachment> {
|
|
guard let inputItem: NSExtensionItem = self.extensionContext?.inputItems.first as? NSExtensionItem else {
|
|
let error = ShareViewControllerError.assertionError(description: "no input item")
|
|
return Promise(error: error)
|
|
}
|
|
|
|
// TODO Multiple attachments. In that case I'm unclear if we'll
|
|
// be given multiple inputItems or a single inputItem with multiple attachments.
|
|
guard let itemProvider: NSItemProvider = inputItem.attachments?.first as? NSItemProvider else {
|
|
let error = ShareViewControllerError.assertionError(description: "No item provider in input item attachments")
|
|
return Promise(error: error)
|
|
}
|
|
Logger.info("\(self.logTag) 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.utiTypeForItem(itemProvider: itemProvider) else {
|
|
let error = ShareViewControllerError.unsupportedMedia
|
|
return Promise(error: error)
|
|
}
|
|
Logger.debug("\(logTag) matched utiType: \(srcUtiType)")
|
|
|
|
let (promise, fulfill, reject) = Promise<(itemUrl: URL, utiType: String)>.pending()
|
|
|
|
var customFileName: String?
|
|
var isConvertibleToTextMessage = false
|
|
|
|
let loadCompletion: NSItemProvider.CompletionHandler = {
|
|
(value, error) in
|
|
|
|
guard error == nil else {
|
|
reject(error!)
|
|
return
|
|
}
|
|
|
|
guard let value = value else {
|
|
let missingProviderError = ShareViewControllerError.assertionError(description: "missing item provider")
|
|
reject(missingProviderError)
|
|
return
|
|
}
|
|
|
|
Logger.info("\(self.logTag) value type: \(type(of:value))")
|
|
|
|
if let data = value as? Data {
|
|
// 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) {
|
|
customFileName = "Contact.vcf"
|
|
}
|
|
|
|
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))")
|
|
reject(writeError)
|
|
return
|
|
}
|
|
let fileUrl = URL(fileURLWithPath:tempFilePath)
|
|
fulfill((itemUrl: fileUrl, utiType: srcUtiType))
|
|
} else if let string = value as? String {
|
|
Logger.debug("\(self.logTag) string provider: \(string)")
|
|
guard let data = string.data(using: String.Encoding.utf8) else {
|
|
let writeError = ShareViewControllerError.assertionError(description: "Error writing item data: \(String(describing: error))")
|
|
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))")
|
|
reject(writeError)
|
|
return
|
|
}
|
|
|
|
let fileUrl = URL(fileURLWithPath:tempFilePath)
|
|
|
|
isConvertibleToTextMessage = !itemProvider.registeredTypeIdentifiers.contains(kUTTypeFileURL as String)
|
|
|
|
if UTTypeConformsTo(srcUtiType as CFString, kUTTypeText) {
|
|
fulfill((itemUrl: fileUrl, utiType: srcUtiType))
|
|
} else {
|
|
fulfill((itemUrl: fileUrl, utiType: kUTTypeText as String))
|
|
}
|
|
} 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.
|
|
isConvertibleToTextMessage = (itemProvider.registeredTypeIdentifiers.contains(kUTTypeURL as String) &&
|
|
!itemProvider.registeredTypeIdentifiers.contains(kUTTypeFileURL as String))
|
|
if isConvertibleToTextMessage {
|
|
fulfill((itemUrl: url, utiType: kUTTypeURL as String))
|
|
} else {
|
|
fulfill((itemUrl: url, utiType: srcUtiType))
|
|
}
|
|
} else if let image = value as? UIImage {
|
|
if let data = UIImagePNGRepresentation(image) {
|
|
let tempFilePath = OWSFileSystem.temporaryFilePath(withFileExtension:"png")
|
|
do {
|
|
let url = NSURL.fileURL(withPath:tempFilePath)
|
|
try data.write(to: url)
|
|
fulfill((url, srcUtiType))
|
|
} catch {
|
|
reject(ShareViewControllerError.assertionError(description: "couldn't write UIImage: \(String(describing: error))"))
|
|
}
|
|
} else {
|
|
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.
|
|
//
|
|
// See comments on NSItemProvider+OWS.h.
|
|
let unexpectedTypeError = ShareViewControllerError.assertionError(description: "unexpected value: \(String(describing: value))")
|
|
reject(unexpectedTypeError)
|
|
}
|
|
}
|
|
|
|
// See comments on NSItemProvider+OWS.h.
|
|
itemProvider.loadData(forTypeIdentifier: srcUtiType, options: nil, completionHandler: loadCompletion)
|
|
|
|
return promise.then { (itemUrl: URL, utiType: String) -> Promise<SignalAttachment> in
|
|
|
|
let url: URL = try {
|
|
if self.isVideoNeedingRelocation(itemProvider: itemProvider, itemUrl: itemUrl) {
|
|
return try SignalAttachment.copyToVideoTempDir(url: itemUrl)
|
|
} else {
|
|
return itemUrl
|
|
}
|
|
}()
|
|
|
|
Logger.debug("\(self.logTag) building DataSource with url: \(url), utiType: \(utiType)")
|
|
|
|
guard let dataSource = ShareViewController.createDataSource(utiType: utiType, url: url, customFileName: customFileName) else {
|
|
throw ShareViewControllerError.assertionError(description: "Unable to read attachment data")
|
|
}
|
|
|
|
// 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("\(self.logTag) 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 {
|
|
owsFail("load view controller was unexpectedly nil")
|
|
return promise
|
|
}
|
|
|
|
loadViewController.progress = progressPoller.progress
|
|
}
|
|
|
|
return promise
|
|
}
|
|
|
|
let attachment = SignalAttachment.attachment(dataSource: dataSource, dataUTI: specificUTIType, imageQuality: .medium)
|
|
if isConvertibleToTextMessage {
|
|
Logger.info("\(self.logTag) isConvertibleToTextMessage")
|
|
attachment.isConvertibleToTextMessage = isConvertibleToTextMessage
|
|
}
|
|
return Promise(value: attachment)
|
|
}
|
|
}
|
|
|
|
// 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 {
|
|
Logger.info("\(self.logTag) isVideoNeedingRelocation: \(itemProvider.registeredTypeIdentifiers), itemUrl: \(itemUrl)")
|
|
|
|
let pathExtension = itemUrl.pathExtension
|
|
guard pathExtension.count > 0 else {
|
|
Logger.verbose("\(self.logTag) in \(#function): item URL has no file extension: \(itemUrl).")
|
|
return false
|
|
}
|
|
guard let utiTypeForURL = MIMETypeUtil.utiType(forFileExtension: pathExtension) else {
|
|
Logger.verbose("\(self.logTag) in \(#function): item has unknown UTI type: \(itemUrl).")
|
|
return false
|
|
}
|
|
Logger.verbose("\(self.logTag) 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)
|
|
}
|
|
}
|
|
|
|
// Exposes a Progress object, whose progress is updated by polling the return of a given block
|
|
private class ProgressPoller {
|
|
|
|
let TAG = "[ProgressPoller]"
|
|
|
|
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 {
|
|
owsFail("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("\(strongSelf.TAG) progress complete")
|
|
timer.invalidate()
|
|
}
|
|
}
|
|
}
|
|
}
|