From 3aa7edc508661817b2b8953faedc069eb51a675e Mon Sep 17 00:00:00 2001 From: Doug <6060466+pixlwave@users.noreply.github.com> Date: Thu, 19 Dec 2024 14:15:31 +0000 Subject: [PATCH] =?UTF-8?q?Enable=20the=20media=20browser=20feature=20?= =?UTF-8?q?=F0=9F=96=BC=EF=B8=8F=20(#3642)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * Overlay a progress indicator for downloads instead of using a toast indicator. * Update the SDK. * Remove the feature flag for the media browser. * Remove the media captions feature flag too. * Add unit test cases for download failure and swiping between items. * Snapshots (with the media browser visible in the screen). --- ElementX.xcodeproj/project.pbxproj | 2 +- .../xcshareddata/swiftpm/Package.resolved | 4 +- .../en.lproj/Localizable.strings | 4 ++ .../Sources/Application/AppSettings.swift | 8 --- ElementX/Sources/Generated/Strings.swift | 8 +++ .../TimelineMediaPreviewModels.swift | 1 + .../TimelineMediaPreviewViewModel.swift | 41 +++----------- .../View/TimelineMediaPreviewScreen.swift | 53 ++++++++++++++++++- .../View/PinnedEventsTimelineScreen.swift | 1 - .../RoomDetailsScreenModels.swift | 2 - .../RoomDetailsScreenViewModel.swift | 4 -- .../View/RoomDetailsScreen.swift | 10 ++-- .../Screens/RoomScreen/View/RoomScreen.swift | 1 - .../DeveloperOptionsScreenModels.swift | 2 - .../View/DeveloperOptionsScreen.swift | 8 --- .../Screens/Timeline/TimelineModels.swift | 1 - .../Screens/Timeline/TimelineViewModel.swift | 5 -- .../View/ItemMenu/TimelineItemMenu.swift | 1 - .../TimelineItemMenuActionProvider.swift | 3 +- .../Style/TimelineItemBubbledStylerView.swift | 1 - ...t_roomDetailsScreen-iPad-en-GB.DM-Room.png | 4 +- ..._roomDetailsScreen-iPad-pseudo.DM-Room.png | 4 +- ...mDetailsScreen-iPhone-16-en-GB.DM-Room.png | 4 +- .../TimelineMediaPreviewViewModelTests.swift | 42 +++++++++++++++ project.yml | 2 +- 25 files changed, 127 insertions(+), 89 deletions(-) diff --git a/ElementX.xcodeproj/project.pbxproj b/ElementX.xcodeproj/project.pbxproj index 99326b9feb..d07d16f1b4 100644 --- a/ElementX.xcodeproj/project.pbxproj +++ b/ElementX.xcodeproj/project.pbxproj @@ -8427,7 +8427,7 @@ repositoryURL = "https://github.com/element-hq/matrix-rust-components-swift"; requirement = { kind = exactVersion; - version = 1.0.82; + version = 1.0.83; }; }; 701C7BEF8F70F7A83E852DCC /* XCRemoteSwiftPackageReference "GZIP" */ = { diff --git a/ElementX.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved b/ElementX.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved index 8f92300dc3..af76e8c09a 100644 --- a/ElementX.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved +++ b/ElementX.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved @@ -149,8 +149,8 @@ "kind" : "remoteSourceControl", "location" : "https://github.com/element-hq/matrix-rust-components-swift", "state" : { - "revision" : "043fcb8c80bbb6257f3f2075f2a72b8c9fdcbed2", - "version" : "1.0.82" + "revision" : "0d20974d1c44225596b24af1ec1f36716dd6e512", + "version" : "1.0.83" } }, { diff --git a/ElementX/Resources/Localizations/en.lproj/Localizable.strings b/ElementX/Resources/Localizations/en.lproj/Localizable.strings index 8fd3c11660..63ab6a7c3b 100644 --- a/ElementX/Resources/Localizations/en.lproj/Localizable.strings +++ b/ElementX/Resources/Localizations/en.lproj/Localizable.strings @@ -145,6 +145,7 @@ "common_developer_options" = "Developer options"; "common_device_id" = "Device ID"; "common_direct_chat" = "Direct chat"; +"common_download_failed" = "Download failed"; "common_downloading" = "Downloading"; "common_edited_suffix" = "(edited)"; "common_editing" = "Editing"; @@ -160,6 +161,8 @@ "common_favourite" = "Favourite"; "common_favourited" = "Favourited"; "common_file" = "File"; +"common_file_deleted" = "File deleted"; +"common_file_saved" = "File saved"; "common_forward_message" = "Forward message"; "common_frequently_used" = "Frequently used"; "common_gif" = "GIF"; @@ -625,6 +628,7 @@ "screen_login_title_with_homeserver" = "Sign in to %1$@"; "screen_media_browser_delete_confirmation_subtitle" = "This file will be removed from the room and members won’t have access to it."; "screen_media_browser_delete_confirmation_title" = "Delete file?"; +"screen_media_browser_download_error_message" = "Check your internet connection and try again."; "screen_media_browser_files_empty_state_subtitle" = "Documents, audio files, and voice messages uploaded to this room will be shown here."; "screen_media_browser_files_empty_state_title" = "No files uploaded yet"; "screen_media_browser_list_loading_files" = "Loading files…"; diff --git a/ElementX/Sources/Application/AppSettings.swift b/ElementX/Sources/Application/AppSettings.swift index 44bcb6b9fa..0b9a4b6b3a 100644 --- a/ElementX/Sources/Application/AppSettings.swift +++ b/ElementX/Sources/Application/AppSettings.swift @@ -49,8 +49,6 @@ final class AppSettings { case fuzzyRoomListSearchEnabled case enableOnlySignedDeviceIsolationMode case knockingEnabled - case createMediaCaptionsEnabled - case mediaBrowserEnabled case eventCacheEnabled } @@ -292,12 +290,6 @@ final class AppSettings { @UserPreference(key: UserDefaultsKeys.knockingEnabled, defaultValue: false, storageType: .userDefaults(store)) var knockingEnabled - @UserPreference(key: UserDefaultsKeys.createMediaCaptionsEnabled, defaultValue: false, storageType: .userDefaults(store)) - var createMediaCaptionsEnabled - - @UserPreference(key: UserDefaultsKeys.mediaBrowserEnabled, defaultValue: false, storageType: .userDefaults(store)) - var mediaBrowserEnabled - #endif // MARK: - Shared diff --git a/ElementX/Sources/Generated/Strings.swift b/ElementX/Sources/Generated/Strings.swift index ef72933439..5fcbeb9425 100644 --- a/ElementX/Sources/Generated/Strings.swift +++ b/ElementX/Sources/Generated/Strings.swift @@ -330,6 +330,8 @@ internal enum L10n { internal static var commonDeviceId: String { return L10n.tr("Localizable", "common_device_id") } /// Direct chat internal static var commonDirectChat: String { return L10n.tr("Localizable", "common_direct_chat") } + /// Download failed + internal static var commonDownloadFailed: String { return L10n.tr("Localizable", "common_download_failed") } /// Downloading internal static var commonDownloading: String { return L10n.tr("Localizable", "common_downloading") } /// (edited) @@ -362,6 +364,10 @@ internal enum L10n { internal static var commonFavourited: String { return L10n.tr("Localizable", "common_favourited") } /// File internal static var commonFile: String { return L10n.tr("Localizable", "common_file") } + /// File deleted + internal static var commonFileDeleted: String { return L10n.tr("Localizable", "common_file_deleted") } + /// File saved + internal static var commonFileSaved: String { return L10n.tr("Localizable", "common_file_saved") } /// Forward message internal static var commonForwardMessage: String { return L10n.tr("Localizable", "common_forward_message") } /// Frequently used @@ -1396,6 +1402,8 @@ internal enum L10n { internal static var screenMediaBrowserDeleteConfirmationSubtitle: String { return L10n.tr("Localizable", "screen_media_browser_delete_confirmation_subtitle") } /// Delete file? internal static var screenMediaBrowserDeleteConfirmationTitle: String { return L10n.tr("Localizable", "screen_media_browser_delete_confirmation_title") } + /// Check your internet connection and try again. + internal static var screenMediaBrowserDownloadErrorMessage: String { return L10n.tr("Localizable", "screen_media_browser_download_error_message") } /// Documents, audio files, and voice messages uploaded to this room will be shown here. internal static var screenMediaBrowserFilesEmptyStateSubtitle: String { return L10n.tr("Localizable", "screen_media_browser_files_empty_state_subtitle") } /// No files uploaded yet diff --git a/ElementX/Sources/Screens/FilePreviewScreen/TimelineMediaPreviewModels.swift b/ElementX/Sources/Screens/FilePreviewScreen/TimelineMediaPreviewModels.swift index fd088f92cf..84b8c40c12 100644 --- a/ElementX/Sources/Screens/FilePreviewScreen/TimelineMediaPreviewModels.swift +++ b/ElementX/Sources/Screens/FilePreviewScreen/TimelineMediaPreviewModels.swift @@ -52,6 +52,7 @@ enum TimelineMediaPreviewAlertType { class TimelineMediaPreviewItem: NSObject, QLPreviewItem, Identifiable { let timelineItem: EventBasedMessageTimelineItemProtocol var fileHandle: MediaFileHandleProxy? + var downloadError: Error? init(timelineItem: EventBasedMessageTimelineItemProtocol) { self.timelineItem = timelineItem diff --git a/ElementX/Sources/Screens/FilePreviewScreen/TimelineMediaPreviewViewModel.swift b/ElementX/Sources/Screens/FilePreviewScreen/TimelineMediaPreviewViewModel.swift index 14c274187e..d944a7cf3e 100644 --- a/ElementX/Sources/Screens/FilePreviewScreen/TimelineMediaPreviewViewModel.swift +++ b/ElementX/Sources/Screens/FilePreviewScreen/TimelineMediaPreviewViewModel.swift @@ -80,21 +80,21 @@ class TimelineMediaPreviewViewModel: TimelineMediaPreviewViewModelType { } private func updateCurrentItem(_ previewItem: TimelineMediaPreviewItem) async { + previewItem.downloadError = nil // Clear any existing error. state.currentItem = previewItem currentItemIDHandler?(previewItem.id) + rebuildCurrentItemActions() if previewItem.fileHandle == nil, let source = previewItem.mediaSource { - showDownloadingIndicator(itemID: previewItem.id) - defer { hideDownloadingIndicator(itemID: previewItem.id) } - switch await mediaProvider.loadFileFromSource(source, filename: previewItem.filename) { case .success(let handle): previewItem.fileHandle = handle state.fileLoadedPublisher.send(previewItem.id) case .failure(let error): MXLog.error("Failed loading media: \(error)") - showDownloadErrorIndicator() + context.objectWillChange.send() // Manually trigger the SwiftUI view update. + previewItem.downloadError = error } } } @@ -108,7 +108,6 @@ class TimelineMediaPreviewViewModel: TimelineMediaPreviewViewModelType { pinnedEventIDs: timelineContext.viewState.pinnedEventIDs, isDM: timelineContext.viewState.isEncryptedOneToOneRoom, isViewSourceEnabled: timelineContext.viewState.isViewSourceEnabled, - isCreateMediaCaptionsEnabled: timelineContext.viewState.isCreateMediaCaptionsEnabled, timelineKind: timelineContext.viewState.timelineKind, emojiProvider: timelineContext.viewState.emojiProvider) state.currentItemActions = provider.makeActions() @@ -156,39 +155,17 @@ class TimelineMediaPreviewViewModel: TimelineMediaPreviewViewModelType { // MARK: - Indicators - private func showDownloadingIndicator(itemID: TimelineItemIdentifier) { - let indicatorID = makeDownloadIndicatorID(itemID: itemID) - userIndicatorController.submitIndicator(UserIndicator(id: indicatorID, - type: .toast(progress: .indeterminate), - title: L10n.commonDownloading, - persistent: true), - delay: .seconds(0.1)) // Don't show the indicator when the SDK loads the file from the store. - } - - private func hideDownloadingIndicator(itemID: TimelineItemIdentifier) { - let indicatorID = makeDownloadIndicatorID(itemID: itemID) - userIndicatorController.retractIndicatorWithId(indicatorID) - } - - // FIXME: Add the strings and correct indicator types - private func showDownloadErrorIndicator() { - userIndicatorController.submitIndicator(UserIndicator(id: downloadErrorIndicatorID, - type: .modal, - title: L10n.errorUnknown, - iconName: "exclamationmark.circle.fill")) - } - private func showRedactedIndicator() { userIndicatorController.submitIndicator(UserIndicator(id: statusIndicatorID, type: .toast, - title: "File deleted", + title: L10n.commonFileDeleted, iconName: "checkmark")) } private func showSavedIndicator() { userIndicatorController.submitIndicator(UserIndicator(id: statusIndicatorID, type: .toast, - title: "File saved", + title: L10n.commonFileSaved, iconName: "checkmark")) } @@ -200,10 +177,4 @@ class TimelineMediaPreviewViewModel: TimelineMediaPreviewViewModelType { } private var statusIndicatorID: String { "\(Self.self)-Status" } - - // Separate indicator IDs for downloads as these can be triggered in the background when swiping between items - private var downloadErrorIndicatorID: String { "\(Self.self)-DownloadError" } - private func makeDownloadIndicatorID(itemID: TimelineItemIdentifier) -> String { - "\(Self.self)-Download-\(itemID.uniqueID.id)" - } } diff --git a/ElementX/Sources/Screens/FilePreviewScreen/View/TimelineMediaPreviewScreen.swift b/ElementX/Sources/Screens/FilePreviewScreen/View/TimelineMediaPreviewScreen.swift index fe7c563b5b..bde0f7ea6c 100644 --- a/ElementX/Sources/Screens/FilePreviewScreen/View/TimelineMediaPreviewScreen.swift +++ b/ElementX/Sources/Screens/FilePreviewScreen/View/TimelineMediaPreviewScreen.swift @@ -47,6 +47,7 @@ struct TimelineMediaPreviewScreen: View { Color.clear // A completely clear view breaks any SwiftUI gestures (such as drag to dismiss). .background { QuickLookView(viewModelContext: context).ignoresSafeArea() } // Not the root view to stop QL hijacking the toolbar. .overlay(alignment: .topTrailing) { fullScreenButton } + .overlay { downloadStatusIndicator } .toolbar { toolbar } .toolbar(toolbarVisibility, for: .navigationBar) .toolbar(toolbarVisibility, for: .bottomBar) @@ -69,6 +70,36 @@ struct TimelineMediaPreviewScreen: View { .padding(.trailing, 14) } + @ViewBuilder + private var downloadStatusIndicator: some View { + if currentItem.downloadError != nil { + VStack(spacing: 24) { + CompoundIcon(\.error, size: .custom(48), relativeTo: .compound.headingLG) + .foregroundStyle(.compound.iconCriticalPrimary) + .padding(.vertical, 24.5) + .padding(.horizontal, 28.5) + + VStack(spacing: 2) { + Text(L10n.commonDownloadFailed) + .font(.compound.headingMDBold) + .foregroundStyle(.compound.textPrimary) + .multilineTextAlignment(.center) + Text(L10n.screenMediaBrowserDownloadErrorMessage) + .font(.compound.bodyMD) + .foregroundStyle(.compound.textPrimary) + .multilineTextAlignment(.center) + } + } + .padding(.horizontal, 24) + .padding(.vertical, 40) + .background(.compound.bgSubtlePrimary, in: RoundedRectangle(cornerRadius: 14)) + } else if currentItem.fileHandle == nil { + ProgressView() + .controlSize(.large) + .tint(.compound.iconPrimary) + } + } + @ViewBuilder private var caption: some View { if let caption = currentItem.caption, !isFullScreen { @@ -220,12 +251,19 @@ struct TimelineMediaPreviewScreen_Previews: PreviewProvider { @Namespace private static var namespace static let viewModel = makeViewModel() + static let downloadingViewModel = makeViewModel(isDownloading: true) + static let downloadErrorViewModel = makeViewModel(isDownloadError: true) static var previews: some View { TimelineMediaPreviewScreen(context: viewModel.context) + .previewDisplayName("Normal") + TimelineMediaPreviewScreen(context: downloadingViewModel.context) + .previewDisplayName("Downloading") + TimelineMediaPreviewScreen(context: downloadErrorViewModel.context) + .previewDisplayName("Download Error") } - static func makeViewModel() -> TimelineMediaPreviewViewModel { + static func makeViewModel(isDownloading: Bool = false, isDownloadError: Bool = false) -> TimelineMediaPreviewViewModel { let item = FileRoomTimelineItem(id: .randomEvent, timestamp: .mock, isOutgoing: false, @@ -243,11 +281,22 @@ struct TimelineMediaPreviewScreen_Previews: PreviewProvider { let timelineController = MockRoomTimelineController(timelineKind: .media(.mediaFilesScreen)) timelineController.timelineItems = [item] + let mediaProvider = MediaProviderMock(configuration: .init()) + + if isDownloading { + mediaProvider.loadFileFromSourceFilenameClosure = { _, _ in + try? await Task.sleep(for: .seconds(3600)) + return .failure(.failedRetrievingFile) + } + } else if isDownloadError { + mediaProvider.loadFileFromSourceFilenameClosure = { _, _ in .failure(.failedRetrievingFile) } + } + return TimelineMediaPreviewViewModel(context: .init(item: item, viewModel: TimelineViewModel.mock(timelineKind: timelineController.timelineKind, timelineController: timelineController), namespace: namespace), - mediaProvider: MediaProviderMock(configuration: .init()), + mediaProvider: mediaProvider, photoLibraryManager: PhotoLibraryManagerMock(.init()), userIndicatorController: UserIndicatorControllerMock(), appMediator: AppMediatorMock()) diff --git a/ElementX/Sources/Screens/PinnedEventsTimelineScreen/View/PinnedEventsTimelineScreen.swift b/ElementX/Sources/Screens/PinnedEventsTimelineScreen/View/PinnedEventsTimelineScreen.swift index 8df6f8e0f6..07d089c84b 100644 --- a/ElementX/Sources/Screens/PinnedEventsTimelineScreen/View/PinnedEventsTimelineScreen.swift +++ b/ElementX/Sources/Screens/PinnedEventsTimelineScreen/View/PinnedEventsTimelineScreen.swift @@ -37,7 +37,6 @@ struct PinnedEventsTimelineScreen: View { pinnedEventIDs: timelineContext.viewState.pinnedEventIDs, isDM: timelineContext.viewState.isEncryptedOneToOneRoom, isViewSourceEnabled: timelineContext.viewState.isViewSourceEnabled, - isCreateMediaCaptionsEnabled: timelineContext.viewState.isCreateMediaCaptionsEnabled, timelineKind: timelineContext.viewState.timelineKind, emojiProvider: timelineContext.viewState.emojiProvider) .makeActions() diff --git a/ElementX/Sources/Screens/RoomDetailsScreen/RoomDetailsScreenModels.swift b/ElementX/Sources/Screens/RoomDetailsScreen/RoomDetailsScreenModels.swift index 25b3e6e59b..e8875dc426 100644 --- a/ElementX/Sources/Screens/RoomDetailsScreen/RoomDetailsScreenModels.swift +++ b/ElementX/Sources/Screens/RoomDetailsScreen/RoomDetailsScreenModels.swift @@ -63,8 +63,6 @@ struct RoomDetailsScreenViewState: BindableState { knockingEnabled && dmRecipient == nil && canEditRolesOrPermissions } - var mediaBrowserEnabled = false - var canEdit: Bool { !isDirect && (canEditRoomName || canEditRoomTopic || canEditRoomAvatar) } diff --git a/ElementX/Sources/Screens/RoomDetailsScreen/RoomDetailsScreenViewModel.swift b/ElementX/Sources/Screens/RoomDetailsScreen/RoomDetailsScreenViewModel.swift index e4fd606a74..2266a2fcec 100644 --- a/ElementX/Sources/Screens/RoomDetailsScreen/RoomDetailsScreenViewModel.swift +++ b/ElementX/Sources/Screens/RoomDetailsScreen/RoomDetailsScreenViewModel.swift @@ -79,10 +79,6 @@ class RoomDetailsScreenViewModel: RoomDetailsScreenViewModelType, RoomDetailsScr .weakAssign(to: \.state.knockingEnabled, on: self) .store(in: &cancellables) - appSettings.$mediaBrowserEnabled - .weakAssign(to: \.state.mediaBrowserEnabled, on: self) - .store(in: &cancellables) - appMediator.networkMonitor.reachabilityPublisher .filter { $0 == .reachable } .receive(on: DispatchQueue.main) diff --git a/ElementX/Sources/Screens/RoomDetailsScreen/View/RoomDetailsScreen.swift b/ElementX/Sources/Screens/RoomDetailsScreen/View/RoomDetailsScreen.swift index 2b02060bb8..d68b6910aa 100644 --- a/ElementX/Sources/Screens/RoomDetailsScreen/View/RoomDetailsScreen.swift +++ b/ElementX/Sources/Screens/RoomDetailsScreen/View/RoomDetailsScreen.swift @@ -170,12 +170,10 @@ struct RoomDetailsScreen: View { }) .accessibilityIdentifier(A11yIdentifiers.roomDetailsScreen.pollsHistory) - if context.viewState.mediaBrowserEnabled { - ListRow(label: .default(title: L10n.screenMediaBrowserTitle, icon: \.image), - kind: .navigationLink { - context.send(viewAction: .processTapMediaEvents) - }) - } + ListRow(label: .default(title: L10n.screenMediaBrowserTitle, icon: \.image), + kind: .navigationLink { + context.send(viewAction: .processTapMediaEvents) + }) } } diff --git a/ElementX/Sources/Screens/RoomScreen/View/RoomScreen.swift b/ElementX/Sources/Screens/RoomScreen/View/RoomScreen.swift index 7d855b524a..932bf73818 100644 --- a/ElementX/Sources/Screens/RoomScreen/View/RoomScreen.swift +++ b/ElementX/Sources/Screens/RoomScreen/View/RoomScreen.swift @@ -75,7 +75,6 @@ struct RoomScreen: View { pinnedEventIDs: timelineContext.viewState.pinnedEventIDs, isDM: timelineContext.viewState.isEncryptedOneToOneRoom, isViewSourceEnabled: timelineContext.viewState.isViewSourceEnabled, - isCreateMediaCaptionsEnabled: timelineContext.viewState.isCreateMediaCaptionsEnabled, timelineKind: timelineContext.viewState.timelineKind, emojiProvider: timelineContext.viewState.emojiProvider) .makeActions() diff --git a/ElementX/Sources/Screens/Settings/DeveloperOptionsScreen/DeveloperOptionsScreenModels.swift b/ElementX/Sources/Screens/Settings/DeveloperOptionsScreen/DeveloperOptionsScreenModels.swift index ed3deb3bc6..ab0d01705d 100644 --- a/ElementX/Sources/Screens/Settings/DeveloperOptionsScreen/DeveloperOptionsScreenModels.swift +++ b/ElementX/Sources/Screens/Settings/DeveloperOptionsScreen/DeveloperOptionsScreenModels.swift @@ -50,8 +50,6 @@ protocol DeveloperOptionsProtocol: AnyObject { var enableOnlySignedDeviceIsolationMode: Bool { get set } var elementCallBaseURLOverride: URL? { get set } var knockingEnabled: Bool { get set } - var createMediaCaptionsEnabled: Bool { get set } - var mediaBrowserEnabled: Bool { get set } var eventCacheEnabled: Bool { get set } } diff --git a/ElementX/Sources/Screens/Settings/DeveloperOptionsScreen/View/DeveloperOptionsScreen.swift b/ElementX/Sources/Screens/Settings/DeveloperOptionsScreen/View/DeveloperOptionsScreen.swift index faa486f9a3..67cca634c0 100644 --- a/ElementX/Sources/Screens/Settings/DeveloperOptionsScreen/View/DeveloperOptionsScreen.swift +++ b/ElementX/Sources/Screens/Settings/DeveloperOptionsScreen/View/DeveloperOptionsScreen.swift @@ -62,14 +62,6 @@ struct DeveloperOptionsScreen: View { Toggle(isOn: $context.hideTimelineMedia) { Text("Hide image & video previews") } - - Toggle(isOn: $context.createMediaCaptionsEnabled) { - Text("Allow creation of media captions") - } - - Toggle(isOn: $context.mediaBrowserEnabled) { - Text("Enable the media browser") - } } Section("Join rules") { diff --git a/ElementX/Sources/Screens/Timeline/TimelineModels.swift b/ElementX/Sources/Screens/Timeline/TimelineModels.swift index e44cb865a1..b27f432aad 100644 --- a/ElementX/Sources/Screens/Timeline/TimelineModels.swift +++ b/ElementX/Sources/Screens/Timeline/TimelineModels.swift @@ -99,7 +99,6 @@ struct TimelineViewState: BindableState { var canCurrentUserRedactSelf = false var canCurrentUserPin = false var isViewSourceEnabled: Bool - var isCreateMediaCaptionsEnabled: Bool var hideTimelineMedia: Bool // The `pinnedEventIDs` are used only to determine if an item is already pinned or not. diff --git a/ElementX/Sources/Screens/Timeline/TimelineViewModel.swift b/ElementX/Sources/Screens/Timeline/TimelineViewModel.swift index 3456c2a45f..8c281b5187 100644 --- a/ElementX/Sources/Screens/Timeline/TimelineViewModel.swift +++ b/ElementX/Sources/Screens/Timeline/TimelineViewModel.swift @@ -81,7 +81,6 @@ class TimelineViewModel: TimelineViewModelType, TimelineViewModelProtocol { timelineState: TimelineState(focussedEvent: focussedEventID.map { .init(eventID: $0, appearance: .immediate) }), ownUserID: roomProxy.ownUserID, isViewSourceEnabled: appSettings.viewSourceEnabled, - isCreateMediaCaptionsEnabled: appSettings.createMediaCaptionsEnabled, hideTimelineMedia: appSettings.hideTimelineMedia, pinnedEventIDs: roomProxy.infoPublisher.value.pinnedEventIDs, bindings: .init(reactionsCollapsed: [:]), @@ -449,10 +448,6 @@ class TimelineViewModel: TimelineViewModelType, TimelineViewModelProtocol { .weakAssign(to: \.state.isViewSourceEnabled, on: self) .store(in: &cancellables) - appSettings.$createMediaCaptionsEnabled - .weakAssign(to: \.state.isCreateMediaCaptionsEnabled, on: self) - .store(in: &cancellables) - appSettings.$hideTimelineMedia .weakAssign(to: \.state.hideTimelineMedia, on: self) .store(in: &cancellables) diff --git a/ElementX/Sources/Screens/Timeline/View/ItemMenu/TimelineItemMenu.swift b/ElementX/Sources/Screens/Timeline/View/ItemMenu/TimelineItemMenu.swift index 7e90df5c7c..6500ac41f6 100644 --- a/ElementX/Sources/Screens/Timeline/View/ItemMenu/TimelineItemMenu.swift +++ b/ElementX/Sources/Screens/Timeline/View/ItemMenu/TimelineItemMenu.swift @@ -344,7 +344,6 @@ struct TimelineItemMenu_Previews: PreviewProvider, TestablePreview { pinnedEventIDs: [], isDM: true, isViewSourceEnabled: true, - isCreateMediaCaptionsEnabled: true, timelineKind: .live, emojiProvider: EmojiProvider(appSettings: ServiceLocator.shared.settings)) guard let actions = provider.makeActions() else { return nil } diff --git a/ElementX/Sources/Screens/Timeline/View/ItemMenu/TimelineItemMenuActionProvider.swift b/ElementX/Sources/Screens/Timeline/View/ItemMenu/TimelineItemMenuActionProvider.swift index 2a075f7155..505d1ae45a 100644 --- a/ElementX/Sources/Screens/Timeline/View/ItemMenu/TimelineItemMenuActionProvider.swift +++ b/ElementX/Sources/Screens/Timeline/View/ItemMenu/TimelineItemMenuActionProvider.swift @@ -16,7 +16,6 @@ struct TimelineItemMenuActionProvider { let pinnedEventIDs: Set let isDM: Bool let isViewSourceEnabled: Bool - let isCreateMediaCaptionsEnabled: Bool let timelineKind: TimelineKind let emojiProvider: EmojiProviderProtocol @@ -63,7 +62,7 @@ struct TimelineItemMenuActionProvider { if item.supportsMediaCaption { if item.hasMediaCaption { actions.append(.editCaption) - } else if isCreateMediaCaptionsEnabled { + } else { actions.append(.addCaption) } } else if item is PollRoomTimelineItem { diff --git a/ElementX/Sources/Screens/Timeline/View/Style/TimelineItemBubbledStylerView.swift b/ElementX/Sources/Screens/Timeline/View/Style/TimelineItemBubbledStylerView.swift index 1a39986359..cfe3626912 100644 --- a/ElementX/Sources/Screens/Timeline/View/Style/TimelineItemBubbledStylerView.swift +++ b/ElementX/Sources/Screens/Timeline/View/Style/TimelineItemBubbledStylerView.swift @@ -149,7 +149,6 @@ struct TimelineItemBubbledStylerView: View { pinnedEventIDs: context.viewState.pinnedEventIDs, isDM: context.viewState.isEncryptedOneToOneRoom, isViewSourceEnabled: context.viewState.isViewSourceEnabled, - isCreateMediaCaptionsEnabled: context.viewState.isCreateMediaCaptionsEnabled, timelineKind: context.viewState.timelineKind, emojiProvider: context.viewState.emojiProvider) TimelineItemMacContextMenu(item: timelineItem, actionProvider: provider) { action in diff --git a/PreviewTests/Sources/__Snapshots__/PreviewTests/test_roomDetailsScreen-iPad-en-GB.DM-Room.png b/PreviewTests/Sources/__Snapshots__/PreviewTests/test_roomDetailsScreen-iPad-en-GB.DM-Room.png index 0f5419362f..2eb2a00b78 100644 --- a/PreviewTests/Sources/__Snapshots__/PreviewTests/test_roomDetailsScreen-iPad-en-GB.DM-Room.png +++ b/PreviewTests/Sources/__Snapshots__/PreviewTests/test_roomDetailsScreen-iPad-en-GB.DM-Room.png @@ -1,3 +1,3 @@ version https://git-lfs.github.com/spec/v1 -oid sha256:9feae2a0bd58c472b8565c718cd24617a042d76d2a36a03a0e61fa0f5f52e7d4 -size 228681 +oid sha256:94c47ee747ccae418bfb137fa8d7c3dfe9ff0a213d750cfa6c4e7267921b9f73 +size 233395 diff --git a/PreviewTests/Sources/__Snapshots__/PreviewTests/test_roomDetailsScreen-iPad-pseudo.DM-Room.png b/PreviewTests/Sources/__Snapshots__/PreviewTests/test_roomDetailsScreen-iPad-pseudo.DM-Room.png index 89e0141729..e14ef85718 100644 --- a/PreviewTests/Sources/__Snapshots__/PreviewTests/test_roomDetailsScreen-iPad-pseudo.DM-Room.png +++ b/PreviewTests/Sources/__Snapshots__/PreviewTests/test_roomDetailsScreen-iPad-pseudo.DM-Room.png @@ -1,3 +1,3 @@ version https://git-lfs.github.com/spec/v1 -oid sha256:a1b7ee09b2669592cd30baba2978bdf6660746dd59c8b583bd7f9258046054d6 -size 233197 +oid sha256:a9a91f791358737b68ab3695f523c6b821ab84387d6857ed6cf0b01c824d68cc +size 238682 diff --git a/PreviewTests/Sources/__Snapshots__/PreviewTests/test_roomDetailsScreen-iPhone-16-en-GB.DM-Room.png b/PreviewTests/Sources/__Snapshots__/PreviewTests/test_roomDetailsScreen-iPhone-16-en-GB.DM-Room.png index 37208e6ec3..af9e87b6d9 100644 --- a/PreviewTests/Sources/__Snapshots__/PreviewTests/test_roomDetailsScreen-iPhone-16-en-GB.DM-Room.png +++ b/PreviewTests/Sources/__Snapshots__/PreviewTests/test_roomDetailsScreen-iPhone-16-en-GB.DM-Room.png @@ -1,3 +1,3 @@ version https://git-lfs.github.com/spec/v1 -oid sha256:2ff0db42a2dbbf0adea05e7d5f65c4031d2def0fbc20466892a296bbd0288182 -size 168684 +oid sha256:dbad14cfc4ef11542ce1fcd25a63c4206326b9db1eb4ef4ecb4c9d07ae96ad3c +size 169529 diff --git a/UnitTests/Sources/TimelineMediaPreviewViewModelTests.swift b/UnitTests/Sources/TimelineMediaPreviewViewModelTests.swift index c130d43c1e..e96bf68927 100644 --- a/UnitTests/Sources/TimelineMediaPreviewViewModelTests.swift +++ b/UnitTests/Sources/TimelineMediaPreviewViewModelTests.swift @@ -36,6 +36,48 @@ class TimelineMediaPreviewViewModelTests: XCTestCase { XCTAssertNotNil(context.viewState.currentItemActions) } + func testLoadingItemFailure() async throws { + // Given a fresh view model. + setupViewModel() + XCTAssertFalse(mediaProvider.loadFileFromSourceFilenameCalled) + XCTAssertEqual(context.viewState.currentItem, context.viewState.previewItems[0]) + XCTAssertNil(context.viewState.currentItem.downloadError) + + // When the preview controller sets an item that fails to load. + mediaProvider.loadFileFromSourceFilenameClosure = { _, _ in .failure(.failedRetrievingFile) } + let failure = deferFailure(viewModel.state.fileLoadedPublisher, timeout: 1) { _ in true } + context.send(viewAction: .updateCurrentItem(context.viewState.previewItems[0])) + try await failure.fulfill() + + // Then the view model should load the item and update its view state. + XCTAssertTrue(mediaProvider.loadFileFromSourceFilenameCalled) + XCTAssertEqual(context.viewState.currentItem, context.viewState.previewItems[0]) + XCTAssertNotNil(context.viewState.currentItem.downloadError) + } + + func testSwipingBetweenItems() async throws { + // Given a view model with a loaded item. + try await testLoadingItem() + + // When swiping to another item. + let deferred = deferFulfillment(viewModel.state.fileLoadedPublisher) { _ in true } + context.send(viewAction: .updateCurrentItem(context.viewState.previewItems[1])) + try await deferred.fulfill() + + // Then the view model should load the item and update its view state. + XCTAssertEqual(mediaProvider.loadFileFromSourceFilenameCallsCount, 2) + XCTAssertEqual(context.viewState.currentItem, context.viewState.previewItems[1]) + + // When swiping back to the first item. + let failure = deferFailure(viewModel.state.fileLoadedPublisher, timeout: 1) { _ in true } + context.send(viewAction: .updateCurrentItem(context.viewState.previewItems[0])) + try await failure.fulfill() + + // Then the view model should not need to load the item, but should still update its view state. + XCTAssertEqual(mediaProvider.loadFileFromSourceFilenameCallsCount, 2) + XCTAssertEqual(context.viewState.currentItem, context.viewState.previewItems[0]) + } + func testViewInRoomTimeline() async throws { // Given a view model with a loaded item. try await testLoadingItem() diff --git a/project.yml b/project.yml index 77ce8d8c84..afd60d79d4 100644 --- a/project.yml +++ b/project.yml @@ -61,7 +61,7 @@ packages: # Element/Matrix dependencies MatrixRustSDK: url: https://github.com/element-hq/matrix-rust-components-swift - exactVersion: 1.0.82 + exactVersion: 1.0.83 # path: ../matrix-rust-sdk Compound: url: https://github.com/element-hq/compound-ios