From 23719d4e2f1a6505371f0e7ece3917e482c44650 Mon Sep 17 00:00:00 2001 From: Mauro Romito Date: Wed, 24 Jul 2024 18:49:18 +0200 Subject: [PATCH 1/6] implemented the pin/unpin logic --- .../en.lproj/Localizable.strings | 1 + ElementX/Sources/Generated/Strings.swift | 2 + .../Mocks/Generated/GeneratedMocks.swift | 227 ++++++++++++++++++ .../Mocks/Generated/SDKGeneratedMocks.swift | 225 +++++++++++++++++ .../RoomScreenInteractionHandler.swift | 7 +- .../Screens/RoomScreen/RoomScreenModels.swift | 59 +++-- .../RoomScreen/RoomScreenViewModel.swift | 16 +- .../ItemMenu/TimelineItemMenuAction.swift | 3 + .../TimelineItemMenuActionProvider.swift | 6 +- .../Screens/RoomScreen/View/RoomScreen.swift | 9 +- .../Style/TimelineItemBubbledStylerView.swift | 1 + .../Sources/Services/Room/RoomProxy.swift | 18 +- .../Services/Room/RoomProxyProtocol.swift | 2 + .../MockRoomTimelineController.swift | 4 + .../RoomTimelineController.swift | 30 +++ .../RoomTimelineControllerProtocol.swift | 4 + .../Services/Timeline/TimelineProxy.swift | 18 ++ .../Timeline/TimelineProxyProtocol.swift | 4 + 18 files changed, 605 insertions(+), 31 deletions(-) diff --git a/ElementX/Resources/Localizations/en.lproj/Localizable.strings b/ElementX/Resources/Localizations/en.lproj/Localizable.strings index 74caa9af29..25405f9d6a 100644 --- a/ElementX/Resources/Localizations/en.lproj/Localizable.strings +++ b/ElementX/Resources/Localizations/en.lproj/Localizable.strings @@ -103,6 +103,7 @@ "action_yes" = "Yes"; "action.load_more" = "Load more"; "action.pin" = "Pin"; +"action.unpin" = "Unpin"; "common_about" = "About"; "common_acceptable_use_policy" = "Acceptable use policy"; "common_advanced_settings" = "Advanced settings"; diff --git a/ElementX/Sources/Generated/Strings.swift b/ElementX/Sources/Generated/Strings.swift index 670faabe46..a51c3937f2 100644 --- a/ElementX/Sources/Generated/Strings.swift +++ b/ElementX/Sources/Generated/Strings.swift @@ -2231,6 +2231,8 @@ internal enum L10n { internal static var loadMore: String { return L10n.tr("Localizable", "action.load_more") } /// Pin internal static var pin: String { return L10n.tr("Localizable", "action.pin") } + /// Unpin + internal static var unpin: String { return L10n.tr("Localizable", "action.unpin") } } internal enum Common { diff --git a/ElementX/Sources/Mocks/Generated/GeneratedMocks.swift b/ElementX/Sources/Mocks/Generated/GeneratedMocks.swift index f86fcb2999..c99f7168bd 100644 --- a/ElementX/Sources/Mocks/Generated/GeneratedMocks.swift +++ b/ElementX/Sources/Mocks/Generated/GeneratedMocks.swift @@ -8251,6 +8251,23 @@ class RoomProxyMock: RoomProxyProtocol { } var underlyingIsFavourite: Bool! var isFavouriteClosure: (() async -> Bool)? + var pinnedEventsCallsCount = 0 + var pinnedEventsCalled: Bool { + return pinnedEventsCallsCount > 0 + } + + var pinnedEvents: [String] { + get async { + pinnedEventsCallsCount += 1 + if let pinnedEventsClosure = pinnedEventsClosure { + return await pinnedEventsClosure() + } else { + return underlyingPinnedEvents + } + } + } + var underlyingPinnedEvents: [String]! + var pinnedEventsClosure: (() async -> [String])? var membership: Membership { get { return underlyingMembership } set(value) { underlyingMembership = value } @@ -10406,6 +10423,76 @@ class RoomProxyMock: RoomProxyProtocol { return canUserTriggerRoomNotificationUserIDReturnValue } } + //MARK: - canUserPinOrUnpin + + var canUserPinOrUnpinUserIDUnderlyingCallsCount = 0 + var canUserPinOrUnpinUserIDCallsCount: Int { + get { + if Thread.isMainThread { + return canUserPinOrUnpinUserIDUnderlyingCallsCount + } else { + var returnValue: Int? = nil + DispatchQueue.main.sync { + returnValue = canUserPinOrUnpinUserIDUnderlyingCallsCount + } + + return returnValue! + } + } + set { + if Thread.isMainThread { + canUserPinOrUnpinUserIDUnderlyingCallsCount = newValue + } else { + DispatchQueue.main.sync { + canUserPinOrUnpinUserIDUnderlyingCallsCount = newValue + } + } + } + } + var canUserPinOrUnpinUserIDCalled: Bool { + return canUserPinOrUnpinUserIDCallsCount > 0 + } + var canUserPinOrUnpinUserIDReceivedUserID: String? + var canUserPinOrUnpinUserIDReceivedInvocations: [String] = [] + + var canUserPinOrUnpinUserIDUnderlyingReturnValue: Result! + var canUserPinOrUnpinUserIDReturnValue: Result! { + get { + if Thread.isMainThread { + return canUserPinOrUnpinUserIDUnderlyingReturnValue + } else { + var returnValue: Result? = nil + DispatchQueue.main.sync { + returnValue = canUserPinOrUnpinUserIDUnderlyingReturnValue + } + + return returnValue! + } + } + set { + if Thread.isMainThread { + canUserPinOrUnpinUserIDUnderlyingReturnValue = newValue + } else { + DispatchQueue.main.sync { + canUserPinOrUnpinUserIDUnderlyingReturnValue = newValue + } + } + } + } + var canUserPinOrUnpinUserIDClosure: ((String) async -> Result)? + + func canUserPinOrUnpin(userID: String) async -> Result { + canUserPinOrUnpinUserIDCallsCount += 1 + canUserPinOrUnpinUserIDReceivedUserID = userID + DispatchQueue.main.async { + self.canUserPinOrUnpinUserIDReceivedInvocations.append(userID) + } + if let canUserPinOrUnpinUserIDClosure = canUserPinOrUnpinUserIDClosure { + return await canUserPinOrUnpinUserIDClosure(userID) + } else { + return canUserPinOrUnpinUserIDReturnValue + } + } //MARK: - kickUser var kickUserUnderlyingCallsCount = 0 @@ -12527,6 +12614,146 @@ class TimelineProxyMock: TimelineProxyProtocol { return redactReasonReturnValue } } + //MARK: - pin + + var pinEventIDUnderlyingCallsCount = 0 + var pinEventIDCallsCount: Int { + get { + if Thread.isMainThread { + return pinEventIDUnderlyingCallsCount + } else { + var returnValue: Int? = nil + DispatchQueue.main.sync { + returnValue = pinEventIDUnderlyingCallsCount + } + + return returnValue! + } + } + set { + if Thread.isMainThread { + pinEventIDUnderlyingCallsCount = newValue + } else { + DispatchQueue.main.sync { + pinEventIDUnderlyingCallsCount = newValue + } + } + } + } + var pinEventIDCalled: Bool { + return pinEventIDCallsCount > 0 + } + var pinEventIDReceivedEventID: String? + var pinEventIDReceivedInvocations: [String] = [] + + var pinEventIDUnderlyingReturnValue: Result! + var pinEventIDReturnValue: Result! { + get { + if Thread.isMainThread { + return pinEventIDUnderlyingReturnValue + } else { + var returnValue: Result? = nil + DispatchQueue.main.sync { + returnValue = pinEventIDUnderlyingReturnValue + } + + return returnValue! + } + } + set { + if Thread.isMainThread { + pinEventIDUnderlyingReturnValue = newValue + } else { + DispatchQueue.main.sync { + pinEventIDUnderlyingReturnValue = newValue + } + } + } + } + var pinEventIDClosure: ((String) async -> Result)? + + func pin(eventID: String) async -> Result { + pinEventIDCallsCount += 1 + pinEventIDReceivedEventID = eventID + DispatchQueue.main.async { + self.pinEventIDReceivedInvocations.append(eventID) + } + if let pinEventIDClosure = pinEventIDClosure { + return await pinEventIDClosure(eventID) + } else { + return pinEventIDReturnValue + } + } + //MARK: - unpin + + var unpinEventIDUnderlyingCallsCount = 0 + var unpinEventIDCallsCount: Int { + get { + if Thread.isMainThread { + return unpinEventIDUnderlyingCallsCount + } else { + var returnValue: Int? = nil + DispatchQueue.main.sync { + returnValue = unpinEventIDUnderlyingCallsCount + } + + return returnValue! + } + } + set { + if Thread.isMainThread { + unpinEventIDUnderlyingCallsCount = newValue + } else { + DispatchQueue.main.sync { + unpinEventIDUnderlyingCallsCount = newValue + } + } + } + } + var unpinEventIDCalled: Bool { + return unpinEventIDCallsCount > 0 + } + var unpinEventIDReceivedEventID: String? + var unpinEventIDReceivedInvocations: [String] = [] + + var unpinEventIDUnderlyingReturnValue: Result! + var unpinEventIDReturnValue: Result! { + get { + if Thread.isMainThread { + return unpinEventIDUnderlyingReturnValue + } else { + var returnValue: Result? = nil + DispatchQueue.main.sync { + returnValue = unpinEventIDUnderlyingReturnValue + } + + return returnValue! + } + } + set { + if Thread.isMainThread { + unpinEventIDUnderlyingReturnValue = newValue + } else { + DispatchQueue.main.sync { + unpinEventIDUnderlyingReturnValue = newValue + } + } + } + } + var unpinEventIDClosure: ((String) async -> Result)? + + func unpin(eventID: String) async -> Result { + unpinEventIDCallsCount += 1 + unpinEventIDReceivedEventID = eventID + DispatchQueue.main.async { + self.unpinEventIDReceivedInvocations.append(eventID) + } + if let unpinEventIDClosure = unpinEventIDClosure { + return await unpinEventIDClosure(eventID) + } else { + return unpinEventIDReturnValue + } + } //MARK: - sendAudio var sendAudioUrlAudioInfoProgressSubjectRequestHandleUnderlyingCallsCount = 0 diff --git a/ElementX/Sources/Mocks/Generated/SDKGeneratedMocks.swift b/ElementX/Sources/Mocks/Generated/SDKGeneratedMocks.swift index c96019a8bd..0ab0d021dc 100644 --- a/ElementX/Sources/Mocks/Generated/SDKGeneratedMocks.swift +++ b/ElementX/Sources/Mocks/Generated/SDKGeneratedMocks.swift @@ -10243,6 +10243,81 @@ open class RoomSDKMock: MatrixRustSDK.Room { } } + //MARK: - canUserPinUnpin + + open var canUserPinUnpinUserIdThrowableError: Error? + var canUserPinUnpinUserIdUnderlyingCallsCount = 0 + open var canUserPinUnpinUserIdCallsCount: Int { + get { + if Thread.isMainThread { + return canUserPinUnpinUserIdUnderlyingCallsCount + } else { + var returnValue: Int? = nil + DispatchQueue.main.sync { + returnValue = canUserPinUnpinUserIdUnderlyingCallsCount + } + + return returnValue! + } + } + set { + if Thread.isMainThread { + canUserPinUnpinUserIdUnderlyingCallsCount = newValue + } else { + DispatchQueue.main.sync { + canUserPinUnpinUserIdUnderlyingCallsCount = newValue + } + } + } + } + open var canUserPinUnpinUserIdCalled: Bool { + return canUserPinUnpinUserIdCallsCount > 0 + } + open var canUserPinUnpinUserIdReceivedUserId: String? + open var canUserPinUnpinUserIdReceivedInvocations: [String] = [] + + var canUserPinUnpinUserIdUnderlyingReturnValue: Bool! + open var canUserPinUnpinUserIdReturnValue: Bool! { + get { + if Thread.isMainThread { + return canUserPinUnpinUserIdUnderlyingReturnValue + } else { + var returnValue: Bool? = nil + DispatchQueue.main.sync { + returnValue = canUserPinUnpinUserIdUnderlyingReturnValue + } + + return returnValue! + } + } + set { + if Thread.isMainThread { + canUserPinUnpinUserIdUnderlyingReturnValue = newValue + } else { + DispatchQueue.main.sync { + canUserPinUnpinUserIdUnderlyingReturnValue = newValue + } + } + } + } + open var canUserPinUnpinUserIdClosure: ((String) async throws -> Bool)? + + open override func canUserPinUnpin(userId: String) async throws -> Bool { + if let error = canUserPinUnpinUserIdThrowableError { + throw error + } + canUserPinUnpinUserIdCallsCount += 1 + canUserPinUnpinUserIdReceivedUserId = userId + DispatchQueue.main.async { + self.canUserPinUnpinUserIdReceivedInvocations.append(userId) + } + if let canUserPinUnpinUserIdClosure = canUserPinUnpinUserIdClosure { + return try await canUserPinUnpinUserIdClosure(userId) + } else { + return canUserPinUnpinUserIdReturnValue + } + } + //MARK: - canUserRedactOther open var canUserRedactOtherUserIdThrowableError: Error? @@ -18491,6 +18566,81 @@ open class TimelineSDKMock: MatrixRustSDK.Timeline { } } + //MARK: - pinEvent + + open var pinEventEventIdStrThrowableError: Error? + var pinEventEventIdStrUnderlyingCallsCount = 0 + open var pinEventEventIdStrCallsCount: Int { + get { + if Thread.isMainThread { + return pinEventEventIdStrUnderlyingCallsCount + } else { + var returnValue: Int? = nil + DispatchQueue.main.sync { + returnValue = pinEventEventIdStrUnderlyingCallsCount + } + + return returnValue! + } + } + set { + if Thread.isMainThread { + pinEventEventIdStrUnderlyingCallsCount = newValue + } else { + DispatchQueue.main.sync { + pinEventEventIdStrUnderlyingCallsCount = newValue + } + } + } + } + open var pinEventEventIdStrCalled: Bool { + return pinEventEventIdStrCallsCount > 0 + } + open var pinEventEventIdStrReceivedEventIdStr: String? + open var pinEventEventIdStrReceivedInvocations: [String] = [] + + var pinEventEventIdStrUnderlyingReturnValue: Bool! + open var pinEventEventIdStrReturnValue: Bool! { + get { + if Thread.isMainThread { + return pinEventEventIdStrUnderlyingReturnValue + } else { + var returnValue: Bool? = nil + DispatchQueue.main.sync { + returnValue = pinEventEventIdStrUnderlyingReturnValue + } + + return returnValue! + } + } + set { + if Thread.isMainThread { + pinEventEventIdStrUnderlyingReturnValue = newValue + } else { + DispatchQueue.main.sync { + pinEventEventIdStrUnderlyingReturnValue = newValue + } + } + } + } + open var pinEventEventIdStrClosure: ((String) async throws -> Bool)? + + open override func pinEvent(eventIdStr: String) async throws -> Bool { + if let error = pinEventEventIdStrThrowableError { + throw error + } + pinEventEventIdStrCallsCount += 1 + pinEventEventIdStrReceivedEventIdStr = eventIdStr + DispatchQueue.main.async { + self.pinEventEventIdStrReceivedInvocations.append(eventIdStr) + } + if let pinEventEventIdStrClosure = pinEventEventIdStrClosure { + return try await pinEventEventIdStrClosure(eventIdStr) + } else { + return pinEventEventIdStrReturnValue + } + } + //MARK: - redactEvent open var redactEventItemReasonThrowableError: Error? @@ -19338,6 +19488,81 @@ open class TimelineSDKMock: MatrixRustSDK.Timeline { } try await toggleReactionEventIdKeyClosure?(eventId, key) } + + //MARK: - unpinEvent + + open var unpinEventEventIdStrThrowableError: Error? + var unpinEventEventIdStrUnderlyingCallsCount = 0 + open var unpinEventEventIdStrCallsCount: Int { + get { + if Thread.isMainThread { + return unpinEventEventIdStrUnderlyingCallsCount + } else { + var returnValue: Int? = nil + DispatchQueue.main.sync { + returnValue = unpinEventEventIdStrUnderlyingCallsCount + } + + return returnValue! + } + } + set { + if Thread.isMainThread { + unpinEventEventIdStrUnderlyingCallsCount = newValue + } else { + DispatchQueue.main.sync { + unpinEventEventIdStrUnderlyingCallsCount = newValue + } + } + } + } + open var unpinEventEventIdStrCalled: Bool { + return unpinEventEventIdStrCallsCount > 0 + } + open var unpinEventEventIdStrReceivedEventIdStr: String? + open var unpinEventEventIdStrReceivedInvocations: [String] = [] + + var unpinEventEventIdStrUnderlyingReturnValue: Bool! + open var unpinEventEventIdStrReturnValue: Bool! { + get { + if Thread.isMainThread { + return unpinEventEventIdStrUnderlyingReturnValue + } else { + var returnValue: Bool? = nil + DispatchQueue.main.sync { + returnValue = unpinEventEventIdStrUnderlyingReturnValue + } + + return returnValue! + } + } + set { + if Thread.isMainThread { + unpinEventEventIdStrUnderlyingReturnValue = newValue + } else { + DispatchQueue.main.sync { + unpinEventEventIdStrUnderlyingReturnValue = newValue + } + } + } + } + open var unpinEventEventIdStrClosure: ((String) async throws -> Bool)? + + open override func unpinEvent(eventIdStr: String) async throws -> Bool { + if let error = unpinEventEventIdStrThrowableError { + throw error + } + unpinEventEventIdStrCallsCount += 1 + unpinEventEventIdStrReceivedEventIdStr = eventIdStr + DispatchQueue.main.async { + self.unpinEventEventIdStrReceivedInvocations.append(eventIdStr) + } + if let unpinEventEventIdStrClosure = unpinEventEventIdStrClosure { + return try await unpinEventEventIdStrClosure(eventIdStr) + } else { + return unpinEventEventIdStrReturnValue + } + } } open class TimelineDiffSDKMock: MatrixRustSDK.TimelineDiff { init() { diff --git a/ElementX/Sources/Screens/RoomScreen/RoomScreenInteractionHandler.swift b/ElementX/Sources/Screens/RoomScreen/RoomScreenInteractionHandler.swift index 1bc317e3ff..452407077c 100644 --- a/ElementX/Sources/Screens/RoomScreen/RoomScreenInteractionHandler.swift +++ b/ElementX/Sources/Screens/RoomScreen/RoomScreenInteractionHandler.swift @@ -172,8 +172,11 @@ class RoomScreenInteractionHandler { case .endPoll(let pollStartID): endPoll(pollStartID: pollStartID) case .pin: - // TODO: Implement the pin action - break + guard let eventID = itemID.eventID else { return } + Task { await timelineController.pin(eventID: eventID) } + case .unpin: + guard let eventID = itemID.eventID else { return } + Task { await timelineController.unpin(eventID: eventID) } } if action.switchToDefaultComposer { diff --git a/ElementX/Sources/Screens/RoomScreen/RoomScreenModels.swift b/ElementX/Sources/Screens/RoomScreen/RoomScreenModels.swift index d29010455a..a65e510bbd 100644 --- a/ElementX/Sources/Screens/RoomScreen/RoomScreenModels.swift +++ b/ElementX/Sources/Screens/RoomScreen/RoomScreenModels.swift @@ -15,9 +15,8 @@ // import Combine -import SwiftUI - import OrderedCollections +import SwiftUI enum RoomScreenViewModelAction { case displayRoomDetails @@ -140,7 +139,7 @@ enum RoomScreenViewAction { case hasSwitchedTimeline case hasScrolled(direction: ScrollDirection) - case nextPin + case tappedPinBanner case viewAllPins } @@ -172,20 +171,11 @@ struct RoomScreenViewState: BindableState { var isPinningEnabled = false var lastScrollDirection: ScrollDirection? - // These are just mocked items used for testing, their types might change - let pinnedItems = [ - "Hello 1", - "How are you 2", - "I am fine 3", - "Thank you 4" - ] - var currentPinIndex = 0 - var shouldShowPinBanner: Bool { - isPinningEnabled && !pinnedItems.isEmpty && lastScrollDirection != .top - } - var selectedPinContent: AttributedString { - .init(pinnedItems[currentPinIndex]) + var pinnedEventsState = PinnedEventsState() + + var shouldShowPinBanner: Bool { + isPinningEnabled && !pinnedEventsState.pinnedEvents.isEmpty && lastScrollDirection != .top } var canJoinCall = false @@ -304,3 +294,40 @@ enum ScrollDirection: Equatable { case top case bottom } + +struct PinnedEventsState: Equatable { + // For now these will only contain and show the event IDs, but in the future they will also contain the content + var pinnedEvents: OrderedSet = [] { + didSet { + if selectedPin == nil, !pinnedEvents.isEmpty { + selectedPin = pinnedEvents.first + } else if pinnedEvents.isEmpty { + selectedPin = nil + } else if let selectedPin, !pinnedEvents.contains(selectedPin) { + self.selectedPin = pinnedEvents.first + } + } + } + + var selectedPin: String? + + var selectedPinIndex: Int { + guard let selectedPin else { + return 0 + } + return pinnedEvents.firstIndex(of: selectedPin) ?? 0 + } + + var selectedPinContent: AttributedString { + .init(selectedPin ?? "") + } + + mutating func nextPin() { + guard !pinnedEvents.isEmpty else { + return + } + let currentIndex = selectedPinIndex + let nextIndex = (currentIndex + 1) % pinnedEvents.count + selectedPin = pinnedEvents[nextIndex] + } +} diff --git a/ElementX/Sources/Screens/RoomScreen/RoomScreenViewModel.swift b/ElementX/Sources/Screens/RoomScreen/RoomScreenViewModel.swift index 945ccc997b..69266d1071 100644 --- a/ElementX/Sources/Screens/RoomScreen/RoomScreenViewModel.swift +++ b/ElementX/Sources/Screens/RoomScreen/RoomScreenViewModel.swift @@ -196,8 +196,11 @@ class RoomScreenViewModel: RoomScreenViewModelType, RoomScreenViewModelProtocol Task { state.timelineViewState.isSwitchingTimelines = false } case let .hasScrolled(direction): state.lastScrollDirection = direction - case .nextPin: - state.currentPinIndex = (state.currentPinIndex + 1) % state.pinnedItems.count + case .tappedPinBanner: + if let eventID = state.pinnedEventsState.selectedPin { + Task { await focusOnEvent(eventID: eventID) } + } + state.pinnedEventsState.nextPin() case .viewAllPins: // TODO: Implement break @@ -368,7 +371,7 @@ class RoomScreenViewModel: RoomScreenViewModelType, RoomScreenViewModelProtocol } if state.isPinningEnabled, - case let .success(value) = await roomProxy.canUser(userID: roomProxy.ownUserID, sendStateEvent: .roomPinnedEvents) { + case let .success(value) = await roomProxy.canUserPinOrUnpin(userID: roomProxy.ownUserID) { state.canCurrentUserPin = value } else { state.canCurrentUserPin = false @@ -404,12 +407,17 @@ class RoomScreenViewModel: RoomScreenViewModelType, RoomScreenViewModelProtocol roomProxy .actionsPublisher .filter { $0 == .roomInfoUpdate } - .throttle(for: .seconds(1), scheduler: DispatchQueue.main, latest: true) + .receive(on: DispatchQueue.main) .sink { [weak self] _ in guard let self else { return } state.roomTitle = roomProxy.roomTitle state.roomAvatar = roomProxy.avatar state.hasOngoingCall = roomProxy.hasOngoingCall + Task { [weak self] in + guard let self else { return } + // TODO: For now we are using the order coming from the room info but we should instead have a ordered timeline of these events + await state.pinnedEventsState.pinnedEvents = .init(roomProxy.pinnedEvents) + } } .store(in: &cancellables) diff --git a/ElementX/Sources/Screens/RoomScreen/View/ItemMenu/TimelineItemMenuAction.swift b/ElementX/Sources/Screens/RoomScreen/View/ItemMenu/TimelineItemMenuAction.swift index cf4c9ed9ab..a387e7c100 100644 --- a/ElementX/Sources/Screens/RoomScreen/View/ItemMenu/TimelineItemMenuAction.swift +++ b/ElementX/Sources/Screens/RoomScreen/View/ItemMenu/TimelineItemMenuAction.swift @@ -62,6 +62,7 @@ enum TimelineItemMenuAction: Identifiable, Hashable { case toggleReaction(key: String) case endPoll(pollStartID: String) case pin + case unpin var id: Self { self } @@ -136,6 +137,8 @@ enum TimelineItemMenuAction: Identifiable, Hashable { Label(L10n.actionEndPoll, icon: \.pollsEnd) case .pin: Label(L10n.Action.pin, icon: \.pin) + case .unpin: + Label(L10n.Action.unpin, icon: \.unpin) } } } diff --git a/ElementX/Sources/Screens/RoomScreen/View/ItemMenu/TimelineItemMenuActionProvider.swift b/ElementX/Sources/Screens/RoomScreen/View/ItemMenu/TimelineItemMenuActionProvider.swift index 09ebd6d76c..df91674e35 100644 --- a/ElementX/Sources/Screens/RoomScreen/View/ItemMenu/TimelineItemMenuActionProvider.swift +++ b/ElementX/Sources/Screens/RoomScreen/View/ItemMenu/TimelineItemMenuActionProvider.swift @@ -21,6 +21,7 @@ struct TimelineItemMenuActionProvider { let canCurrentUserRedactSelf: Bool let canCurrentUserRedactOthers: Bool let canCurrentUserPin: Bool + let pinnedEvents: Set let isDM: Bool let isViewSourceEnabled: Bool @@ -66,9 +67,8 @@ struct TimelineItemMenuActionProvider { actions.append(.forward(itemID: item.id)) } - if canCurrentUserPin { - // TODO: If the event is already pinned use the unpinned action - actions.append(.pin) + if canCurrentUserPin, let eventID = item.id.eventID { + actions.append(pinnedEvents.contains(eventID) ? .unpin : .pin) } if item.isEditable { diff --git a/ElementX/Sources/Screens/RoomScreen/View/RoomScreen.swift b/ElementX/Sources/Screens/RoomScreen/View/RoomScreen.swift index c62571c1c3..8e96150002 100644 --- a/ElementX/Sources/Screens/RoomScreen/View/RoomScreen.swift +++ b/ElementX/Sources/Screens/RoomScreen/View/RoomScreen.swift @@ -69,6 +69,7 @@ struct RoomScreen: View { canCurrentUserRedactSelf: context.viewState.canCurrentUserRedactSelf, canCurrentUserRedactOthers: context.viewState.canCurrentUserRedactOthers, canCurrentUserPin: context.viewState.canCurrentUserPin, + pinnedEvents: context.viewState.pinnedEventsState.pinnedEvents.set, isDM: context.viewState.isEncryptedOneToOneRoom, isViewSourceEnabled: context.viewState.isViewSourceEnabled).makeActions() if let actions { @@ -109,10 +110,10 @@ struct RoomScreen: View { } private var pinnedItemsBanner: some View { - PinnedItemsBannerView(pinIndex: context.viewState.currentPinIndex, - pinsCount: context.viewState.pinnedItems.count, - pinContent: context.viewState.selectedPinContent, - onMainButtonTap: { context.send(viewAction: .nextPin) }, + PinnedItemsBannerView(pinIndex: context.viewState.pinnedEventsState.selectedPinIndex, + pinsCount: context.viewState.pinnedEventsState.pinnedEvents.count, + pinContent: context.viewState.pinnedEventsState.selectedPinContent, + onMainButtonTap: { context.send(viewAction: .tappedPinBanner) }, onViewAllButtonTap: { context.send(viewAction: .viewAllPins) }) .transition(.move(edge: .top)) } diff --git a/ElementX/Sources/Screens/RoomScreen/View/Style/TimelineItemBubbledStylerView.swift b/ElementX/Sources/Screens/RoomScreen/View/Style/TimelineItemBubbledStylerView.swift index 84c5cad0ac..f907ca6c1d 100644 --- a/ElementX/Sources/Screens/RoomScreen/View/Style/TimelineItemBubbledStylerView.swift +++ b/ElementX/Sources/Screens/RoomScreen/View/Style/TimelineItemBubbledStylerView.swift @@ -147,6 +147,7 @@ struct TimelineItemBubbledStylerView: View { canCurrentUserRedactSelf: context.viewState.canCurrentUserRedactSelf, canCurrentUserRedactOthers: context.viewState.canCurrentUserRedactOthers, canCurrentUserPin: context.viewState.canCurrentUserPin, + pinnedEvents: context.viewState.pinnedEventsState.pinnedEvents.set, isDM: context.viewState.isEncryptedOneToOneRoom, isViewSourceEnabled: context.viewState.isViewSourceEnabled) TimelineItemMacContextMenu(item: timelineItem, actionProvider: provider) { action in diff --git a/ElementX/Sources/Services/Room/RoomProxy.swift b/ElementX/Sources/Services/Room/RoomProxy.swift index 9a32bc181f..70457a0c61 100644 --- a/ElementX/Sources/Services/Room/RoomProxy.swift +++ b/ElementX/Sources/Services/Room/RoomProxy.swift @@ -16,9 +16,8 @@ import Combine import Foundation -import UIKit - import MatrixRustSDK +import UIKit class RoomProxy: RoomProxyProtocol { private let roomListItem: RoomListItemProtocol @@ -87,6 +86,12 @@ class RoomProxy: RoomProxyProtocol { } } + var pinnedEvents: [String] { + get async { + await (try? room.roomInfo().pinnedEventIds) ?? [] + } + } + var hasOngoingCall: Bool { room.hasActiveRoomCall() } @@ -493,6 +498,15 @@ class RoomProxy: RoomProxyProtocol { } } + func canUserPinOrUnpin(userID: String) async -> Result { + do { + return try await .success(room.canUserPinUnpin(userId: userID)) + } catch { + MXLog.error("Failed checking if the user can pin or unnpin: \(error)") + return .failure(.sdkError(error)) + } + } + // MARK: - Moderation func kickUser(_ userID: String) async -> Result { diff --git a/ElementX/Sources/Services/Room/RoomProxyProtocol.swift b/ElementX/Sources/Services/Room/RoomProxyProtocol.swift index 6916f63cd0..fa6f0f0778 100644 --- a/ElementX/Sources/Services/Room/RoomProxyProtocol.swift +++ b/ElementX/Sources/Services/Room/RoomProxyProtocol.swift @@ -38,6 +38,7 @@ protocol RoomProxyProtocol { var isSpace: Bool { get } var isEncrypted: Bool { get } var isFavourite: Bool { get async } + var pinnedEvents: [String] { get async } var membership: Membership { get } var hasOngoingCall: Bool { get } var canonicalAlias: String? { get } @@ -121,6 +122,7 @@ protocol RoomProxyProtocol { func canUserKick(userID: String) async -> Result func canUserBan(userID: String) async -> Result func canUserTriggerRoomNotification(userID: String) async -> Result + func canUserPinOrUnpin(userID: String) async -> Result // MARK: - Moderation diff --git a/ElementX/Sources/Services/Timeline/TimelineController/MockRoomTimelineController.swift b/ElementX/Sources/Services/Timeline/TimelineController/MockRoomTimelineController.swift index 2eeaff01ee..1d452481fe 100644 --- a/ElementX/Sources/Services/Timeline/TimelineController/MockRoomTimelineController.swift +++ b/ElementX/Sources/Services/Timeline/TimelineController/MockRoomTimelineController.swift @@ -100,6 +100,10 @@ class MockRoomTimelineController: RoomTimelineControllerProtocol { func redact(_ itemID: TimelineItemIdentifier) async { } + func pin(eventID: String) async { } + + func unpin(eventID: String) async { } + func messageEventContent(for itemID: TimelineItemIdentifier) -> RoomMessageEventContentWithoutRelation? { .init(noPointer: .init()) } diff --git a/ElementX/Sources/Services/Timeline/TimelineController/RoomTimelineController.swift b/ElementX/Sources/Services/Timeline/TimelineController/RoomTimelineController.swift index d63345a24d..60083697f2 100644 --- a/ElementX/Sources/Services/Timeline/TimelineController/RoomTimelineController.swift +++ b/ElementX/Sources/Services/Timeline/TimelineController/RoomTimelineController.swift @@ -237,6 +237,36 @@ class RoomTimelineController: RoomTimelineControllerProtocol { } } + func pin(eventID: String) async { + MXLog.info("Pinning event \(eventID) in \(roomID)") + + switch await activeTimeline.pin(eventID: eventID) { + case .success(let value): + if value { + MXLog.info("Finished pinning event \(eventID)") + } else { + MXLog.info("Failed pinning event \(eventID) because is already pinned") + } + case .failure(let error): + MXLog.info("Failed pinning event \(eventID) with error: \(error)") + } + } + + func unpin(eventID: String) async { + MXLog.info("Unpinning event \(eventID) in \(roomID)") + + switch await activeTimeline.unpin(eventID: eventID) { + case .success(let value): + if value { + MXLog.info("Finished unpinning event \(eventID)") + } else { + MXLog.info("Failed unpinning event \(eventID) because is not pinned") + } + case .failure(let error): + MXLog.info("Failed unpinning event \(eventID) with error: \(error)") + } + } + func messageEventContent(for timelineItemID: TimelineItemIdentifier) async -> RoomMessageEventContentWithoutRelation? { await activeTimeline.messageEventContent(for: timelineItemID) } diff --git a/ElementX/Sources/Services/Timeline/TimelineController/RoomTimelineControllerProtocol.swift b/ElementX/Sources/Services/Timeline/TimelineController/RoomTimelineControllerProtocol.swift index 9e21465968..832b7e539e 100644 --- a/ElementX/Sources/Services/Timeline/TimelineController/RoomTimelineControllerProtocol.swift +++ b/ElementX/Sources/Services/Timeline/TimelineController/RoomTimelineControllerProtocol.swift @@ -68,6 +68,10 @@ protocol RoomTimelineControllerProtocol { func redact(_ itemID: TimelineItemIdentifier) async + func pin(eventID: String) async + + func unpin(eventID: String) async + func messageEventContent(for itemID: TimelineItemIdentifier) async -> RoomMessageEventContentWithoutRelation? func debugInfo(for itemID: TimelineItemIdentifier) -> TimelineItemDebugInfo diff --git a/ElementX/Sources/Services/Timeline/TimelineProxy.swift b/ElementX/Sources/Services/Timeline/TimelineProxy.swift index eda53d0f5b..db1e0f7a24 100644 --- a/ElementX/Sources/Services/Timeline/TimelineProxy.swift +++ b/ElementX/Sources/Services/Timeline/TimelineProxy.swift @@ -202,6 +202,24 @@ final class TimelineProxy: TimelineProxyProtocol { } } + func pin(eventID: String) async -> Result { + do { + return try await .success(timeline.pinEvent(eventIdStr: eventID)) + } catch { + MXLog.error("Failed to pin the event \(eventID) with error: \(error)") + return .failure(.sdkError(error)) + } + } + + func unpin(eventID: String) async -> Result { + do { + return try await .success(timeline.unpinEvent(eventIdStr: eventID)) + } catch { + MXLog.error("Failed to unpin the event \(eventID) with error: \(error)") + return .failure(.sdkError(error)) + } + } + // MARK: - Sending func sendAudio(url: URL, diff --git a/ElementX/Sources/Services/Timeline/TimelineProxyProtocol.swift b/ElementX/Sources/Services/Timeline/TimelineProxyProtocol.swift index 646024c198..511c494099 100644 --- a/ElementX/Sources/Services/Timeline/TimelineProxyProtocol.swift +++ b/ElementX/Sources/Services/Timeline/TimelineProxyProtocol.swift @@ -46,6 +46,10 @@ protocol TimelineProxyProtocol { func redact(_ timelineItemID: TimelineItemIdentifier, reason: String?) async -> Result + func pin(eventID: String) async -> Result + + func unpin(eventID: String) async -> Result + // MARK: - Sending func sendAudio(url: URL, From cec68ae0e652e381af95bcf1b9a8d61de9fdf2a7 Mon Sep 17 00:00:00 2001 From: Mauro Romito Date: Wed, 24 Jul 2024 18:54:47 +0200 Subject: [PATCH 2/6] error logging --- .../TimelineController/RoomTimelineController.swift | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/ElementX/Sources/Services/Timeline/TimelineController/RoomTimelineController.swift b/ElementX/Sources/Services/Timeline/TimelineController/RoomTimelineController.swift index 60083697f2..8db6d0c3bd 100644 --- a/ElementX/Sources/Services/Timeline/TimelineController/RoomTimelineController.swift +++ b/ElementX/Sources/Services/Timeline/TimelineController/RoomTimelineController.swift @@ -245,10 +245,10 @@ class RoomTimelineController: RoomTimelineControllerProtocol { if value { MXLog.info("Finished pinning event \(eventID)") } else { - MXLog.info("Failed pinning event \(eventID) because is already pinned") + MXLog.error("Failed pinning event \(eventID) because is already pinned") } case .failure(let error): - MXLog.info("Failed pinning event \(eventID) with error: \(error)") + MXLog.error("Failed pinning event \(eventID) with error: \(error)") } } @@ -260,10 +260,10 @@ class RoomTimelineController: RoomTimelineControllerProtocol { if value { MXLog.info("Finished unpinning event \(eventID)") } else { - MXLog.info("Failed unpinning event \(eventID) because is not pinned") + MXLog.error("Failed unpinning event \(eventID) because is not pinned") } case .failure(let error): - MXLog.info("Failed unpinning event \(eventID) with error: \(error)") + MXLog.error("Failed unpinning event \(eventID) with error: \(error)") } } From 86ba82ef12bbf4a578539774d881a5fdcc15e9a1 Mon Sep 17 00:00:00 2001 From: Mauro Romito Date: Thu, 25 Jul 2024 11:39:39 +0200 Subject: [PATCH 3/6] pr comments --- .../en.lproj/Localizable.strings | 10 +++--- ElementX/Sources/Generated/Strings.swift | 33 ++++++++----------- .../Mocks/Generated/GeneratedMocks.swift | 20 +++++------ .../Screens/RoomScreen/RoomScreenModels.swift | 31 ++++++++--------- .../RoomScreen/RoomScreenViewModel.swift | 14 +++++--- .../ItemMenu/TimelineItemMenuAction.swift | 4 +-- .../TimelineItemMenuActionProvider.swift | 4 +-- .../PinnedItemsBannerView.swift | 20 +++++------ .../Screens/RoomScreen/View/RoomScreen.swift | 6 ++-- .../Style/TimelineItemBubbledStylerView.swift | 2 +- .../Sources/Services/Room/RoomProxy.swift | 2 +- .../Services/Room/RoomProxyProtocol.swift | 2 +- 12 files changed, 72 insertions(+), 76 deletions(-) diff --git a/ElementX/Resources/Localizations/en.lproj/Localizable.strings b/ElementX/Resources/Localizations/en.lproj/Localizable.strings index 25405f9d6a..1cdb9970d1 100644 --- a/ElementX/Resources/Localizations/en.lproj/Localizable.strings +++ b/ElementX/Resources/Localizations/en.lproj/Localizable.strings @@ -70,6 +70,7 @@ "action_ok" = "OK"; "action_open_settings" = "Settings"; "action_open_with" = "Open with"; +"action_pin" = "Pin"; "action_quick_reply" = "Quick reply"; "action_quote" = "Quote"; "action_react" = "React"; @@ -99,11 +100,10 @@ "action_take_photo" = "Take photo"; "action_tap_for_options" = "Tap for options"; "action_try_again" = "Try again"; +"action_unpin" = "Unpin"; "action_view_source" = "View source"; "action_yes" = "Yes"; "action.load_more" = "Load more"; -"action.pin" = "Pin"; -"action.unpin" = "Unpin"; "common_about" = "About"; "common_acceptable_use_policy" = "Acceptable use policy"; "common_advanced_settings" = "Advanced settings"; @@ -314,9 +314,9 @@ "screen_advanced_settings_element_call_base_url_description" = "Set a custom base URL for Element Call."; "screen_advanced_settings_element_call_base_url_validation_error" = "Invalid URL, please make sure you include the protocol (http/https) and the correct address."; "screen_room_mentions_at_room_subtitle" = "Notify the whole room"; -"screen.room.pinned_banner_indicator" = "%1$@ of %2$@"; -"screen.room.pinned_banner_indicator_description" = "%1$@ Pinned messages"; -"screen.room.pinned_banner_view_all_button_title" = "View All"; +"screen_room_pinned_banner_indicator" = "%1$@ of %2$@"; +"screen_room_pinned_banner_indicator_description" = "%1$@ Pinned messages"; +"screen_room_pinned_banner_view_all_button_title" = "View All"; "screen_account_provider_change" = "Change account provider"; "screen_account_provider_form_hint" = "Homeserver address"; "screen_account_provider_form_notice" = "Enter a search term or a domain address."; diff --git a/ElementX/Sources/Generated/Strings.swift b/ElementX/Sources/Generated/Strings.swift index a51c3937f2..708ea61ac5 100644 --- a/ElementX/Sources/Generated/Strings.swift +++ b/ElementX/Sources/Generated/Strings.swift @@ -174,6 +174,8 @@ internal enum L10n { internal static var actionOpenSettings: String { return L10n.tr("Localizable", "action_open_settings") } /// Open with internal static var actionOpenWith: String { return L10n.tr("Localizable", "action_open_with") } + /// Pin + internal static var actionPin: String { return L10n.tr("Localizable", "action_pin") } /// Quick reply internal static var actionQuickReply: String { return L10n.tr("Localizable", "action_quick_reply") } /// Quote @@ -232,6 +234,8 @@ internal enum L10n { internal static var actionTapForOptions: String { return L10n.tr("Localizable", "action_tap_for_options") } /// Try again internal static var actionTryAgain: String { return L10n.tr("Localizable", "action_try_again") } + /// Unpin + internal static var actionUnpin: String { return L10n.tr("Localizable", "action_unpin") } /// View source internal static var actionViewSource: String { return L10n.tr("Localizable", "action_view_source") } /// Yes @@ -1629,6 +1633,16 @@ internal enum L10n { internal static var screenRoomNotificationSettingsModeMentionsAndKeywords: String { return L10n.tr("Localizable", "screen_room_notification_settings_mode_mentions_and_keywords") } /// In this room, notify me for internal static var screenRoomNotificationSettingsRoomCustomSettingsTitle: String { return L10n.tr("Localizable", "screen_room_notification_settings_room_custom_settings_title") } + /// %1$@ of %2$@ + internal static func screenRoomPinnedBannerIndicator(_ p1: Any, _ p2: Any) -> String { + return L10n.tr("Localizable", "screen_room_pinned_banner_indicator", String(describing: p1), String(describing: p2)) + } + /// %1$@ Pinned messages + internal static func screenRoomPinnedBannerIndicatorDescription(_ p1: Any) -> String { + return L10n.tr("Localizable", "screen_room_pinned_banner_indicator_description", String(describing: p1)) + } + /// View All + internal static var screenRoomPinnedBannerViewAllButtonTitle: String { return L10n.tr("Localizable", "screen_room_pinned_banner_view_all_button_title") } /// Send again internal static var screenRoomRetrySendMenuSendAgainAction: String { return L10n.tr("Localizable", "screen_room_retry_send_menu_send_again_action") } /// Your message failed to send @@ -2229,10 +2243,6 @@ internal enum L10n { internal enum Action { /// Load more internal static var loadMore: String { return L10n.tr("Localizable", "action.load_more") } - /// Pin - internal static var pin: String { return L10n.tr("Localizable", "action.pin") } - /// Unpin - internal static var unpin: String { return L10n.tr("Localizable", "action.unpin") } } internal enum Common { @@ -2243,21 +2253,6 @@ internal enum L10n { /// Send to internal static var sendTo: String { return L10n.tr("Localizable", "common.send_to") } } - - internal enum Screen { - internal enum Room { - /// %1$@ of %2$@ - internal static func pinnedBannerIndicator(_ p1: Any, _ p2: Any) -> String { - return L10n.tr("Localizable", "screen.room.pinned_banner_indicator", String(describing: p1), String(describing: p2)) - } - /// %1$@ Pinned messages - internal static func pinnedBannerIndicatorDescription(_ p1: Any) -> String { - return L10n.tr("Localizable", "screen.room.pinned_banner_indicator_description", String(describing: p1)) - } - /// View All - internal static var pinnedBannerViewAllButtonTitle: String { return L10n.tr("Localizable", "screen.room.pinned_banner_view_all_button_title") } - } - } } // swiftlint:enable explicit_type_interface function_parameter_count identifier_name line_length // swiftlint:enable nesting type_body_length type_name vertical_whitespace_opening_braces diff --git a/ElementX/Sources/Mocks/Generated/GeneratedMocks.swift b/ElementX/Sources/Mocks/Generated/GeneratedMocks.swift index c99f7168bd..f989f22f70 100644 --- a/ElementX/Sources/Mocks/Generated/GeneratedMocks.swift +++ b/ElementX/Sources/Mocks/Generated/GeneratedMocks.swift @@ -8251,23 +8251,23 @@ class RoomProxyMock: RoomProxyProtocol { } var underlyingIsFavourite: Bool! var isFavouriteClosure: (() async -> Bool)? - var pinnedEventsCallsCount = 0 - var pinnedEventsCalled: Bool { - return pinnedEventsCallsCount > 0 + var pinnedEventIDsCallsCount = 0 + var pinnedEventIDsCalled: Bool { + return pinnedEventIDsCallsCount > 0 } - var pinnedEvents: [String] { + var pinnedEventIDs: [String] { get async { - pinnedEventsCallsCount += 1 - if let pinnedEventsClosure = pinnedEventsClosure { - return await pinnedEventsClosure() + pinnedEventIDsCallsCount += 1 + if let pinnedEventIDsClosure = pinnedEventIDsClosure { + return await pinnedEventIDsClosure() } else { - return underlyingPinnedEvents + return underlyingPinnedEventIDs } } } - var underlyingPinnedEvents: [String]! - var pinnedEventsClosure: (() async -> [String])? + var underlyingPinnedEventIDs: [String]! + var pinnedEventIDsClosure: (() async -> [String])? var membership: Membership { get { return underlyingMembership } set(value) { underlyingMembership = value } diff --git a/ElementX/Sources/Screens/RoomScreen/RoomScreenModels.swift b/ElementX/Sources/Screens/RoomScreen/RoomScreenModels.swift index a65e510bbd..1897d8f8d5 100644 --- a/ElementX/Sources/Screens/RoomScreen/RoomScreenModels.swift +++ b/ElementX/Sources/Screens/RoomScreen/RoomScreenModels.swift @@ -175,7 +175,7 @@ struct RoomScreenViewState: BindableState { var pinnedEventsState = PinnedEventsState() var shouldShowPinBanner: Bool { - isPinningEnabled && !pinnedEventsState.pinnedEvents.isEmpty && lastScrollDirection != .top + isPinningEnabled && !pinnedEventsState.pinnedEventIDs.isEmpty && lastScrollDirection != .top } var canJoinCall = false @@ -297,37 +297,38 @@ enum ScrollDirection: Equatable { struct PinnedEventsState: Equatable { // For now these will only contain and show the event IDs, but in the future they will also contain the content - var pinnedEvents: OrderedSet = [] { + var pinnedEventIDs: OrderedSet = [] { didSet { - if selectedPin == nil, !pinnedEvents.isEmpty { - selectedPin = pinnedEvents.first - } else if pinnedEvents.isEmpty { - selectedPin = nil - } else if let selectedPin, !pinnedEvents.contains(selectedPin) { - self.selectedPin = pinnedEvents.first + if selectedPinEventID == nil, !pinnedEventIDs.isEmpty { + selectedPinEventID = pinnedEventIDs.first + } else if pinnedEventIDs.isEmpty { + selectedPinEventID = nil + } else if let selectedPinEventID, !pinnedEventIDs.contains(selectedPinEventID) { + self.selectedPinEventID = pinnedEventIDs.first } } } - var selectedPin: String? + var selectedPinEventID: String? var selectedPinIndex: Int { - guard let selectedPin else { + guard let selectedPinEventID else { return 0 } - return pinnedEvents.firstIndex(of: selectedPin) ?? 0 + return pinnedEventIDs.firstIndex(of: selectedPinEventID) ?? 0 } + // For now we show the event ID as the content, but is just until we have a way to get the real content var selectedPinContent: AttributedString { - .init(selectedPin ?? "") + .init(selectedPinEventID ?? "") } mutating func nextPin() { - guard !pinnedEvents.isEmpty else { + guard !pinnedEventIDs.isEmpty else { return } let currentIndex = selectedPinIndex - let nextIndex = (currentIndex + 1) % pinnedEvents.count - selectedPin = pinnedEvents[nextIndex] + let nextIndex = (currentIndex + 1) % pinnedEventIDs.count + selectedPinEventID = pinnedEventIDs[nextIndex] } } diff --git a/ElementX/Sources/Screens/RoomScreen/RoomScreenViewModel.swift b/ElementX/Sources/Screens/RoomScreen/RoomScreenViewModel.swift index 69266d1071..4a12b5ce4d 100644 --- a/ElementX/Sources/Screens/RoomScreen/RoomScreenViewModel.swift +++ b/ElementX/Sources/Screens/RoomScreen/RoomScreenViewModel.swift @@ -197,7 +197,7 @@ class RoomScreenViewModel: RoomScreenViewModelType, RoomScreenViewModelProtocol case let .hasScrolled(direction): state.lastScrollDirection = direction case .tappedPinBanner: - if let eventID = state.pinnedEventsState.selectedPin { + if let eventID = state.pinnedEventsState.selectedPinEventID { Task { await focusOnEvent(eventID: eventID) } } state.pinnedEventsState.nextPin() @@ -404,19 +404,25 @@ class RoomScreenViewModel: RoomScreenViewModelType, RoomScreenViewModelProtocol } .store(in: &cancellables) - roomProxy + let roomInfoSubscription = roomProxy .actionsPublisher .filter { $0 == .roomInfoUpdate } - .receive(on: DispatchQueue.main) + roomInfoSubscription + .throttle(for: .seconds(1), scheduler: DispatchQueue.main, latest: true) .sink { [weak self] _ in guard let self else { return } state.roomTitle = roomProxy.roomTitle state.roomAvatar = roomProxy.avatar state.hasOngoingCall = roomProxy.hasOngoingCall + } + .store(in: &cancellables) + roomInfoSubscription + .receive(on: DispatchQueue.main) + .sink { [weak self] _ in Task { [weak self] in guard let self else { return } // TODO: For now we are using the order coming from the room info but we should instead have a ordered timeline of these events - await state.pinnedEventsState.pinnedEvents = .init(roomProxy.pinnedEvents) + await state.pinnedEventsState.pinnedEventIDs = .init(roomProxy.pinnedEventIDs) } } .store(in: &cancellables) diff --git a/ElementX/Sources/Screens/RoomScreen/View/ItemMenu/TimelineItemMenuAction.swift b/ElementX/Sources/Screens/RoomScreen/View/ItemMenu/TimelineItemMenuAction.swift index a387e7c100..4fa340a75a 100644 --- a/ElementX/Sources/Screens/RoomScreen/View/ItemMenu/TimelineItemMenuAction.swift +++ b/ElementX/Sources/Screens/RoomScreen/View/ItemMenu/TimelineItemMenuAction.swift @@ -136,9 +136,9 @@ enum TimelineItemMenuAction: Identifiable, Hashable { case .endPoll: Label(L10n.actionEndPoll, icon: \.pollsEnd) case .pin: - Label(L10n.Action.pin, icon: \.pin) + Label(L10n.actionPin, icon: \.pin) case .unpin: - Label(L10n.Action.unpin, icon: \.unpin) + Label(L10n.actionUnpin, icon: \.unpin) } } } diff --git a/ElementX/Sources/Screens/RoomScreen/View/ItemMenu/TimelineItemMenuActionProvider.swift b/ElementX/Sources/Screens/RoomScreen/View/ItemMenu/TimelineItemMenuActionProvider.swift index df91674e35..45f5028a45 100644 --- a/ElementX/Sources/Screens/RoomScreen/View/ItemMenu/TimelineItemMenuActionProvider.swift +++ b/ElementX/Sources/Screens/RoomScreen/View/ItemMenu/TimelineItemMenuActionProvider.swift @@ -21,7 +21,7 @@ struct TimelineItemMenuActionProvider { let canCurrentUserRedactSelf: Bool let canCurrentUserRedactOthers: Bool let canCurrentUserPin: Bool - let pinnedEvents: Set + let pinnedEventIDs: Set let isDM: Bool let isViewSourceEnabled: Bool @@ -68,7 +68,7 @@ struct TimelineItemMenuActionProvider { } if canCurrentUserPin, let eventID = item.id.eventID { - actions.append(pinnedEvents.contains(eventID) ? .unpin : .pin) + actions.append(pinnedEventIDs.contains(eventID) ? .unpin : .pin) } if item.isEditable { diff --git a/ElementX/Sources/Screens/RoomScreen/View/PinnedItemsBanner/PinnedItemsBannerView.swift b/ElementX/Sources/Screens/RoomScreen/View/PinnedItemsBanner/PinnedItemsBannerView.swift index 80bcb915e5..506ebdd9a3 100644 --- a/ElementX/Sources/Screens/RoomScreen/View/PinnedItemsBanner/PinnedItemsBannerView.swift +++ b/ElementX/Sources/Screens/RoomScreen/View/PinnedItemsBanner/PinnedItemsBannerView.swift @@ -18,18 +18,16 @@ import Compound import SwiftUI struct PinnedItemsBannerView: View { - let pinIndex: Int - let pinsCount: Int - let pinContent: AttributedString + let pinnedEventsState: PinnedEventsState let onMainButtonTap: () -> Void let onViewAllButtonTap: () -> Void private var bannerIndicatorDescription: AttributedString { - let index = pinIndex + 1 + let index = pinnedEventsState.selectedPinIndex + 1 let boldPlaceholder = "{bold}" - var finalString = AttributedString(L10n.Screen.Room.pinnedBannerIndicatorDescription(boldPlaceholder)) - var boldString = AttributedString(L10n.Screen.Room.pinnedBannerIndicator(index, pinsCount)) + var finalString = AttributedString(L10n.screenRoomPinnedBannerIndicatorDescription(boldPlaceholder)) + var boldString = AttributedString(L10n.screenRoomPinnedBannerIndicator(index, pinnedEventsState.pinnedEventIDs.count)) boldString.bold() finalString.replace(boldPlaceholder, with: boldString) return finalString @@ -50,7 +48,7 @@ struct PinnedItemsBannerView: View { Button { onMainButtonTap() } label: { HStack(spacing: 0) { HStack(spacing: 10) { - PinnedItemsIndicatorView(pinIndex: pinIndex, pinsCount: pinsCount) + PinnedItemsIndicatorView(pinIndex: pinnedEventsState.selectedPinIndex, pinsCount: pinnedEventsState.pinnedEventIDs.count) .accessibilityHidden(true) CompoundIcon(\.pinSolid, size: .small, relativeTo: .compound.bodyMD) .foregroundColor(Color.compound.iconSecondaryAlpha) @@ -65,7 +63,7 @@ struct PinnedItemsBannerView: View { private var viewAllButton: some View { Button { onViewAllButtonTap() } label: { - Text(L10n.Screen.Room.pinnedBannerViewAllButtonTitle) + Text(L10n.screenRoomPinnedBannerViewAllButtonTitle) .font(.compound.bodyMDSemibold) .foregroundStyle(Color.compound.textPrimary) .padding(.horizontal, 16) @@ -79,7 +77,7 @@ struct PinnedItemsBannerView: View { .font(.compound.bodySM) .foregroundColor(.compound.textActionAccent) .lineLimit(1) - Text(pinContent) + Text(pinnedEventsState.selectedPinContent) .font(.compound.bodyMD) .foregroundColor(.compound.textPrimary) .lineLimit(1) @@ -89,9 +87,7 @@ struct PinnedItemsBannerView: View { struct PinnedItemsBannerView_Previews: PreviewProvider, TestablePreview { static var previews: some View { - PinnedItemsBannerView(pinIndex: 0, - pinsCount: 3, - pinContent: .init(stringLiteral: "Content"), + PinnedItemsBannerView(pinnedEventsState: .init(pinnedEventIDs: ["Content", "NotShown1", "NotShown2"], selectedPinEventID: "Content"), onMainButtonTap: { }, onViewAllButtonTap: { }) } diff --git a/ElementX/Sources/Screens/RoomScreen/View/RoomScreen.swift b/ElementX/Sources/Screens/RoomScreen/View/RoomScreen.swift index 8e96150002..45ece1897a 100644 --- a/ElementX/Sources/Screens/RoomScreen/View/RoomScreen.swift +++ b/ElementX/Sources/Screens/RoomScreen/View/RoomScreen.swift @@ -69,7 +69,7 @@ struct RoomScreen: View { canCurrentUserRedactSelf: context.viewState.canCurrentUserRedactSelf, canCurrentUserRedactOthers: context.viewState.canCurrentUserRedactOthers, canCurrentUserPin: context.viewState.canCurrentUserPin, - pinnedEvents: context.viewState.pinnedEventsState.pinnedEvents.set, + pinnedEventIDs: context.viewState.pinnedEventsState.pinnedEventIDs.set, isDM: context.viewState.isEncryptedOneToOneRoom, isViewSourceEnabled: context.viewState.isViewSourceEnabled).makeActions() if let actions { @@ -110,9 +110,7 @@ struct RoomScreen: View { } private var pinnedItemsBanner: some View { - PinnedItemsBannerView(pinIndex: context.viewState.pinnedEventsState.selectedPinIndex, - pinsCount: context.viewState.pinnedEventsState.pinnedEvents.count, - pinContent: context.viewState.pinnedEventsState.selectedPinContent, + PinnedItemsBannerView(pinnedEventsState: context.viewState.pinnedEventsState, onMainButtonTap: { context.send(viewAction: .tappedPinBanner) }, onViewAllButtonTap: { context.send(viewAction: .viewAllPins) }) .transition(.move(edge: .top)) diff --git a/ElementX/Sources/Screens/RoomScreen/View/Style/TimelineItemBubbledStylerView.swift b/ElementX/Sources/Screens/RoomScreen/View/Style/TimelineItemBubbledStylerView.swift index f907ca6c1d..fff5b30f53 100644 --- a/ElementX/Sources/Screens/RoomScreen/View/Style/TimelineItemBubbledStylerView.swift +++ b/ElementX/Sources/Screens/RoomScreen/View/Style/TimelineItemBubbledStylerView.swift @@ -147,7 +147,7 @@ struct TimelineItemBubbledStylerView: View { canCurrentUserRedactSelf: context.viewState.canCurrentUserRedactSelf, canCurrentUserRedactOthers: context.viewState.canCurrentUserRedactOthers, canCurrentUserPin: context.viewState.canCurrentUserPin, - pinnedEvents: context.viewState.pinnedEventsState.pinnedEvents.set, + pinnedEventIDs: context.viewState.pinnedEventsState.pinnedEventIDs.set, isDM: context.viewState.isEncryptedOneToOneRoom, isViewSourceEnabled: context.viewState.isViewSourceEnabled) TimelineItemMacContextMenu(item: timelineItem, actionProvider: provider) { action in diff --git a/ElementX/Sources/Services/Room/RoomProxy.swift b/ElementX/Sources/Services/Room/RoomProxy.swift index 70457a0c61..eb274047fe 100644 --- a/ElementX/Sources/Services/Room/RoomProxy.swift +++ b/ElementX/Sources/Services/Room/RoomProxy.swift @@ -86,7 +86,7 @@ class RoomProxy: RoomProxyProtocol { } } - var pinnedEvents: [String] { + var pinnedEventIDs: [String] { get async { await (try? room.roomInfo().pinnedEventIds) ?? [] } diff --git a/ElementX/Sources/Services/Room/RoomProxyProtocol.swift b/ElementX/Sources/Services/Room/RoomProxyProtocol.swift index fa6f0f0778..1aa02c185d 100644 --- a/ElementX/Sources/Services/Room/RoomProxyProtocol.swift +++ b/ElementX/Sources/Services/Room/RoomProxyProtocol.swift @@ -38,7 +38,7 @@ protocol RoomProxyProtocol { var isSpace: Bool { get } var isEncrypted: Bool { get } var isFavourite: Bool { get async } - var pinnedEvents: [String] { get async } + var pinnedEventIDs: [String] { get async } var membership: Membership { get } var hasOngoingCall: Bool { get } var canonicalAlias: String? { get } From 1218e94cc5754ece83b88addc0b3b47c6fa9bfe6 Mon Sep 17 00:00:00 2001 From: Mauro Romito Date: Thu, 25 Jul 2024 16:18:02 +0200 Subject: [PATCH 4/6] storable Task to handle the update --- ElementX/Sources/Other/Extensions/Task.swift | 11 ++++++++++ .../RoomScreen/RoomScreenViewModel.swift | 22 ++++++++++++------- 2 files changed, 25 insertions(+), 8 deletions(-) diff --git a/ElementX/Sources/Other/Extensions/Task.swift b/ElementX/Sources/Other/Extensions/Task.swift index 99f73bbe21..a58731bccc 100644 --- a/ElementX/Sources/Other/Extensions/Task.swift +++ b/ElementX/Sources/Other/Extensions/Task.swift @@ -14,6 +14,7 @@ // limitations under the License. // +import Combine import Foundation public extension Task where Success == Never, Failure == Never { @@ -61,3 +62,13 @@ public extension Task where Success == Never, Failure == Never { } } } + +extension Task { + func store(in cancellables: inout Set) { + asCancellable().store(in: &cancellables) + } + + func asCancellable() -> AnyCancellable { + AnyCancellable(cancel) + } +} diff --git a/ElementX/Sources/Screens/RoomScreen/RoomScreenViewModel.swift b/ElementX/Sources/Screens/RoomScreen/RoomScreenViewModel.swift index 4a12b5ce4d..4d32e659a7 100644 --- a/ElementX/Sources/Screens/RoomScreen/RoomScreenViewModel.swift +++ b/ElementX/Sources/Screens/RoomScreen/RoomScreenViewModel.swift @@ -407,6 +407,7 @@ class RoomScreenViewModel: RoomScreenViewModelType, RoomScreenViewModelProtocol let roomInfoSubscription = roomProxy .actionsPublisher .filter { $0 == .roomInfoUpdate } + roomInfoSubscription .throttle(for: .seconds(1), scheduler: DispatchQueue.main, latest: true) .sink { [weak self] _ in @@ -416,16 +417,21 @@ class RoomScreenViewModel: RoomScreenViewModelType, RoomScreenViewModelProtocol state.hasOngoingCall = roomProxy.hasOngoingCall } .store(in: &cancellables) - roomInfoSubscription - .receive(on: DispatchQueue.main) - .sink { [weak self] _ in - Task { [weak self] in - guard let self else { return } - // TODO: For now we are using the order coming from the room info but we should instead have a ordered timeline of these events - await state.pinnedEventsState.pinnedEventIDs = .init(roomProxy.pinnedEventIDs) + + Task { [weak self] in + guard let self else { + return + } + // If the subscription has sent a value before the Task has started it might be lost, so before entering the loop we always do an update. + await state.pinnedEventsState.pinnedEventIDs = .init(roomProxy.pinnedEventIDs) + for await _ in roomInfoSubscription.receive(on: DispatchQueue.main).values { + guard !Task.isCancelled else { + return } + await state.pinnedEventsState.pinnedEventIDs = .init(roomProxy.pinnedEventIDs) } - .store(in: &cancellables) + } + .store(in: &cancellables) appSettings.$sharePresence .weakAssign(to: \.state.showReadReceipts, on: self) From e9204321f809b6dc8d458ff08bf57ed06f703076 Mon Sep 17 00:00:00 2001 From: Mauro Romito Date: Fri, 26 Jul 2024 16:14:07 +0200 Subject: [PATCH 5/6] updated SDK --- ElementX.xcodeproj/project.pbxproj | 2 +- .../xcshareddata/swiftpm/Package.resolved | 4 +- .../Mocks/Generated/SDKGeneratedMocks.swift | 104 +++++++++--------- .../Services/Timeline/TimelineProxy.swift | 4 +- project.yml | 2 +- 5 files changed, 58 insertions(+), 58 deletions(-) diff --git a/ElementX.xcodeproj/project.pbxproj b/ElementX.xcodeproj/project.pbxproj index be57beaa94..ea63ed8776 100644 --- a/ElementX.xcodeproj/project.pbxproj +++ b/ElementX.xcodeproj/project.pbxproj @@ -7501,7 +7501,7 @@ repositoryURL = "https://github.com/element-hq/matrix-rust-components-swift"; requirement = { kind = exactVersion; - version = 1.0.28; + version = 1.0.29; }; }; 701C7BEF8F70F7A83E852DCC /* XCRemoteSwiftPackageReference "GZIP" */ = { diff --git a/ElementX.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved b/ElementX.xcodeproj/project.xcworkspace/xcshareddata/swiftpm/Package.resolved index af9059fec7..b452f42d23 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" : "1a1cbc9d9d43a188d9b07fe00a141d02f7c43c7c", - "version" : "1.0.28" + "revision" : "d0226f669526e908d45bf9b5682f372d84cf9ffe", + "version" : "1.0.29" } }, { diff --git a/ElementX/Sources/Mocks/Generated/SDKGeneratedMocks.swift b/ElementX/Sources/Mocks/Generated/SDKGeneratedMocks.swift index 0ab0d021dc..a37d07a20a 100644 --- a/ElementX/Sources/Mocks/Generated/SDKGeneratedMocks.swift +++ b/ElementX/Sources/Mocks/Generated/SDKGeneratedMocks.swift @@ -18568,16 +18568,16 @@ open class TimelineSDKMock: MatrixRustSDK.Timeline { //MARK: - pinEvent - open var pinEventEventIdStrThrowableError: Error? - var pinEventEventIdStrUnderlyingCallsCount = 0 - open var pinEventEventIdStrCallsCount: Int { + open var pinEventEventIdThrowableError: Error? + var pinEventEventIdUnderlyingCallsCount = 0 + open var pinEventEventIdCallsCount: Int { get { if Thread.isMainThread { - return pinEventEventIdStrUnderlyingCallsCount + return pinEventEventIdUnderlyingCallsCount } else { var returnValue: Int? = nil DispatchQueue.main.sync { - returnValue = pinEventEventIdStrUnderlyingCallsCount + returnValue = pinEventEventIdUnderlyingCallsCount } return returnValue! @@ -18585,29 +18585,29 @@ open class TimelineSDKMock: MatrixRustSDK.Timeline { } set { if Thread.isMainThread { - pinEventEventIdStrUnderlyingCallsCount = newValue + pinEventEventIdUnderlyingCallsCount = newValue } else { DispatchQueue.main.sync { - pinEventEventIdStrUnderlyingCallsCount = newValue + pinEventEventIdUnderlyingCallsCount = newValue } } } } - open var pinEventEventIdStrCalled: Bool { - return pinEventEventIdStrCallsCount > 0 + open var pinEventEventIdCalled: Bool { + return pinEventEventIdCallsCount > 0 } - open var pinEventEventIdStrReceivedEventIdStr: String? - open var pinEventEventIdStrReceivedInvocations: [String] = [] + open var pinEventEventIdReceivedEventId: String? + open var pinEventEventIdReceivedInvocations: [String] = [] - var pinEventEventIdStrUnderlyingReturnValue: Bool! - open var pinEventEventIdStrReturnValue: Bool! { + var pinEventEventIdUnderlyingReturnValue: Bool! + open var pinEventEventIdReturnValue: Bool! { get { if Thread.isMainThread { - return pinEventEventIdStrUnderlyingReturnValue + return pinEventEventIdUnderlyingReturnValue } else { var returnValue: Bool? = nil DispatchQueue.main.sync { - returnValue = pinEventEventIdStrUnderlyingReturnValue + returnValue = pinEventEventIdUnderlyingReturnValue } return returnValue! @@ -18615,29 +18615,29 @@ open class TimelineSDKMock: MatrixRustSDK.Timeline { } set { if Thread.isMainThread { - pinEventEventIdStrUnderlyingReturnValue = newValue + pinEventEventIdUnderlyingReturnValue = newValue } else { DispatchQueue.main.sync { - pinEventEventIdStrUnderlyingReturnValue = newValue + pinEventEventIdUnderlyingReturnValue = newValue } } } } - open var pinEventEventIdStrClosure: ((String) async throws -> Bool)? + open var pinEventEventIdClosure: ((String) async throws -> Bool)? - open override func pinEvent(eventIdStr: String) async throws -> Bool { - if let error = pinEventEventIdStrThrowableError { + open override func pinEvent(eventId: String) async throws -> Bool { + if let error = pinEventEventIdThrowableError { throw error } - pinEventEventIdStrCallsCount += 1 - pinEventEventIdStrReceivedEventIdStr = eventIdStr + pinEventEventIdCallsCount += 1 + pinEventEventIdReceivedEventId = eventId DispatchQueue.main.async { - self.pinEventEventIdStrReceivedInvocations.append(eventIdStr) + self.pinEventEventIdReceivedInvocations.append(eventId) } - if let pinEventEventIdStrClosure = pinEventEventIdStrClosure { - return try await pinEventEventIdStrClosure(eventIdStr) + if let pinEventEventIdClosure = pinEventEventIdClosure { + return try await pinEventEventIdClosure(eventId) } else { - return pinEventEventIdStrReturnValue + return pinEventEventIdReturnValue } } @@ -19491,16 +19491,16 @@ open class TimelineSDKMock: MatrixRustSDK.Timeline { //MARK: - unpinEvent - open var unpinEventEventIdStrThrowableError: Error? - var unpinEventEventIdStrUnderlyingCallsCount = 0 - open var unpinEventEventIdStrCallsCount: Int { + open var unpinEventEventIdThrowableError: Error? + var unpinEventEventIdUnderlyingCallsCount = 0 + open var unpinEventEventIdCallsCount: Int { get { if Thread.isMainThread { - return unpinEventEventIdStrUnderlyingCallsCount + return unpinEventEventIdUnderlyingCallsCount } else { var returnValue: Int? = nil DispatchQueue.main.sync { - returnValue = unpinEventEventIdStrUnderlyingCallsCount + returnValue = unpinEventEventIdUnderlyingCallsCount } return returnValue! @@ -19508,29 +19508,29 @@ open class TimelineSDKMock: MatrixRustSDK.Timeline { } set { if Thread.isMainThread { - unpinEventEventIdStrUnderlyingCallsCount = newValue + unpinEventEventIdUnderlyingCallsCount = newValue } else { DispatchQueue.main.sync { - unpinEventEventIdStrUnderlyingCallsCount = newValue + unpinEventEventIdUnderlyingCallsCount = newValue } } } } - open var unpinEventEventIdStrCalled: Bool { - return unpinEventEventIdStrCallsCount > 0 + open var unpinEventEventIdCalled: Bool { + return unpinEventEventIdCallsCount > 0 } - open var unpinEventEventIdStrReceivedEventIdStr: String? - open var unpinEventEventIdStrReceivedInvocations: [String] = [] + open var unpinEventEventIdReceivedEventId: String? + open var unpinEventEventIdReceivedInvocations: [String] = [] - var unpinEventEventIdStrUnderlyingReturnValue: Bool! - open var unpinEventEventIdStrReturnValue: Bool! { + var unpinEventEventIdUnderlyingReturnValue: Bool! + open var unpinEventEventIdReturnValue: Bool! { get { if Thread.isMainThread { - return unpinEventEventIdStrUnderlyingReturnValue + return unpinEventEventIdUnderlyingReturnValue } else { var returnValue: Bool? = nil DispatchQueue.main.sync { - returnValue = unpinEventEventIdStrUnderlyingReturnValue + returnValue = unpinEventEventIdUnderlyingReturnValue } return returnValue! @@ -19538,29 +19538,29 @@ open class TimelineSDKMock: MatrixRustSDK.Timeline { } set { if Thread.isMainThread { - unpinEventEventIdStrUnderlyingReturnValue = newValue + unpinEventEventIdUnderlyingReturnValue = newValue } else { DispatchQueue.main.sync { - unpinEventEventIdStrUnderlyingReturnValue = newValue + unpinEventEventIdUnderlyingReturnValue = newValue } } } } - open var unpinEventEventIdStrClosure: ((String) async throws -> Bool)? + open var unpinEventEventIdClosure: ((String) async throws -> Bool)? - open override func unpinEvent(eventIdStr: String) async throws -> Bool { - if let error = unpinEventEventIdStrThrowableError { + open override func unpinEvent(eventId: String) async throws -> Bool { + if let error = unpinEventEventIdThrowableError { throw error } - unpinEventEventIdStrCallsCount += 1 - unpinEventEventIdStrReceivedEventIdStr = eventIdStr + unpinEventEventIdCallsCount += 1 + unpinEventEventIdReceivedEventId = eventId DispatchQueue.main.async { - self.unpinEventEventIdStrReceivedInvocations.append(eventIdStr) + self.unpinEventEventIdReceivedInvocations.append(eventId) } - if let unpinEventEventIdStrClosure = unpinEventEventIdStrClosure { - return try await unpinEventEventIdStrClosure(eventIdStr) + if let unpinEventEventIdClosure = unpinEventEventIdClosure { + return try await unpinEventEventIdClosure(eventId) } else { - return unpinEventEventIdStrReturnValue + return unpinEventEventIdReturnValue } } } diff --git a/ElementX/Sources/Services/Timeline/TimelineProxy.swift b/ElementX/Sources/Services/Timeline/TimelineProxy.swift index db1e0f7a24..dce0dd1e08 100644 --- a/ElementX/Sources/Services/Timeline/TimelineProxy.swift +++ b/ElementX/Sources/Services/Timeline/TimelineProxy.swift @@ -204,7 +204,7 @@ final class TimelineProxy: TimelineProxyProtocol { func pin(eventID: String) async -> Result { do { - return try await .success(timeline.pinEvent(eventIdStr: eventID)) + return try await .success(timeline.pinEvent(eventId: eventID)) } catch { MXLog.error("Failed to pin the event \(eventID) with error: \(error)") return .failure(.sdkError(error)) @@ -213,7 +213,7 @@ final class TimelineProxy: TimelineProxyProtocol { func unpin(eventID: String) async -> Result { do { - return try await .success(timeline.unpinEvent(eventIdStr: eventID)) + return try await .success(timeline.unpinEvent(eventId: eventID)) } catch { MXLog.error("Failed to unpin the event \(eventID) with error: \(error)") return .failure(.sdkError(error)) diff --git a/project.yml b/project.yml index 0044a0a868..419bae1a80 100644 --- a/project.yml +++ b/project.yml @@ -60,7 +60,7 @@ packages: # Element/Matrix dependencies MatrixRustSDK: url: https://github.com/element-hq/matrix-rust-components-swift - exactVersion: 1.0.28 + exactVersion: 1.0.29 # path: ../matrix-rust-sdk Compound: url: https://github.com/element-hq/compound-ios From 9278aa615f2c7bf50bae5ea1faf9bf09bf87f174 Mon Sep 17 00:00:00 2001 From: Mauro Romito Date: Fri, 26 Jul 2024 16:46:33 +0200 Subject: [PATCH 6/6] added a mock for the pinned event ids --- ElementX/Sources/Mocks/RoomProxyMock.swift | 3 +++ 1 file changed, 3 insertions(+) diff --git a/ElementX/Sources/Mocks/RoomProxyMock.swift b/ElementX/Sources/Mocks/RoomProxyMock.swift index 30e755def5..85fc38e886 100644 --- a/ElementX/Sources/Mocks/RoomProxyMock.swift +++ b/ElementX/Sources/Mocks/RoomProxyMock.swift @@ -29,6 +29,7 @@ struct RoomProxyMockConfiguration { var isEncrypted = true var hasOngoingCall = true var canonicalAlias: String? + var pinnedEventIDs: [String] = [] var timelineStartReached = false @@ -63,6 +64,8 @@ extension RoomProxyMock { hasOngoingCall = configuration.hasOngoingCall canonicalAlias = configuration.canonicalAlias + underlyingPinnedEventIDs = configuration.pinnedEventIDs + let timeline = TimelineProxyMock() timeline.sendMessageEventContentReturnValue = .success(()) timeline.paginateBackwardsRequestSizeReturnValue = .success(())