* Setup simple share extension * Switch the app url scheme to be the full bundle identifier * Setup a share extension that show a SwiftUI view, uses rust tracing and redirects to the hosting aplication * Move media as json through the custom scheme into the main app and deep link into the media upload preview screen * Fix message forwarding and global search screen room summary provider filtering. * Tweak the message forwarding and global search screen designs. * Add a room selection screen to use after receiving a share request from the share extension * Fix share extension entitlements * Share the temporary directory between the main app and the extensions; rename the caches one. * Remove the no longer needed notification avatar flipping fix. * Extract the placeholder avatar image generator from the NSE * Nest `AvatarSize` within the new `Avatars` enum * Donate an `INSendMessageIntent` to the system every time we send a message so they appear as share suggestions * Support suggestions in the share extension itself * Improve sharing animations and fix presentation when room already on the stack * Clear all routes when sharing without a preselected room. * Fix broken unit tests * Various initial tweaks following code review. * Correctly clean up and dismiss the share extension for all paths. * Move the share extension path to a constants enum * Rename UserSessionFlowCoordinator specific share extension states and events * Add UserSession and Room flow coordinator share route tests * Tweak the share extension logic.
65 lines
1.9 KiB
Swift
65 lines
1.9 KiB
Swift
//
|
|
// Copyright 2022-2024 New Vector Ltd.
|
|
//
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
// Please see LICENSE in the repository root for full details.
|
|
//
|
|
|
|
import Foundation
|
|
|
|
enum FileManagerError: Error {
|
|
case invalidFileSize
|
|
}
|
|
|
|
extension FileManager {
|
|
func directoryExists(at url: URL) -> Bool {
|
|
var isDirectory: ObjCBool = false
|
|
guard fileExists(atPath: url.path(percentEncoded: false), isDirectory: &isDirectory) else {
|
|
return false
|
|
}
|
|
return isDirectory.boolValue
|
|
}
|
|
|
|
func createDirectoryIfNeeded(at url: URL, withIntermediateDirectories: Bool = true) throws {
|
|
guard !directoryExists(at: url) else {
|
|
return
|
|
}
|
|
try createDirectory(at: url, withIntermediateDirectories: withIntermediateDirectories)
|
|
}
|
|
|
|
func copyFileToTemporaryDirectory(file url: URL, with filename: String? = nil) throws -> URL {
|
|
let newURL = URL.temporaryDirectory.appendingPathComponent(filename ?? url.lastPathComponent)
|
|
|
|
try? removeItem(at: newURL)
|
|
try copyItem(at: url, to: newURL)
|
|
|
|
return newURL
|
|
}
|
|
|
|
@discardableResult
|
|
func writeDataToTemporaryDirectory(data: Data, fileName: String) throws -> URL {
|
|
let newURL = URL.appGroupTemporaryDirectory.appendingPathComponent(fileName)
|
|
|
|
try data.write(to: newURL)
|
|
|
|
return newURL
|
|
}
|
|
|
|
/// Retrieve a file's disk size
|
|
/// - Parameter url: the file URL
|
|
/// - Returns: the size in bytes
|
|
func sizeForItem(at url: URL) throws -> Double {
|
|
let attributes = try attributesOfItem(atPath: url.path())
|
|
|
|
guard let size = attributes[FileAttributeKey.size] as? Double else {
|
|
throw FileManagerError.invalidFileSize
|
|
}
|
|
|
|
return size
|
|
}
|
|
|
|
func numberOfItems(at url: URL) throws -> Int {
|
|
try contentsOfDirectory(at: url, includingPropertiesForKeys: nil).count
|
|
}
|
|
}
|