* Basic implementation completed but it requires some bug fixing and polishing (+4 squashed commits) Squashed commits: [43f0064e] now the content of the room details reflects the expected dm details content when the conditions are met [fe6f2b5e] replaced existing headers with a composable and reusable HeaderView [3c00120a] now the members are fetched every time the room is opened [421d539b] refactored the members fetching to be immediate in case the room could be a direct chat * polished the code and fixed some bugs that prevented the updated state of the members to be displayed in the room details view (+1 squashed commit) Squashed commits: [6bf98473] fix for the button not being destructive in room details screen * removed copy button and updated copy * moved a file and renamed header view * addressed all PR comments * sdk bump * UI tests * unit tests * changelog * updated mocks * removed a duplicated signature in a mock
176 lines
7.7 KiB
Swift
176 lines
7.7 KiB
Swift
//
|
|
// Copyright 2022 New Vector Ltd
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
//
|
|
|
|
import XCTest
|
|
|
|
@testable import ElementX
|
|
|
|
@MainActor
|
|
class RoomDetailsScreenViewModelTests: XCTestCase {
|
|
var viewModel: RoomDetailsViewModelProtocol!
|
|
var roomProxyMock: RoomProxyMock!
|
|
var context: RoomDetailsViewModelType.Context { viewModel.context }
|
|
|
|
override func setUp() {
|
|
roomProxyMock = RoomProxyMock(with: .init(displayName: "Test"))
|
|
viewModel = RoomDetailsViewModel(roomProxy: roomProxyMock, mediaProvider: MockMediaProvider())
|
|
}
|
|
|
|
func testLeaveRoomTappedWhenPublic() async {
|
|
let mockedMembers: [RoomMemberProxyMock] = [.mockBob, .mockAlice]
|
|
roomProxyMock = RoomProxyMock(with: .init(displayName: "Test", isPublic: true, members: mockedMembers))
|
|
viewModel = RoomDetailsViewModel(roomProxy: roomProxyMock, mediaProvider: MockMediaProvider())
|
|
context.send(viewAction: .processTapLeave)
|
|
await Task.yield()
|
|
XCTAssertEqual(context.leaveRoomAlertItem?.state, .public)
|
|
XCTAssertEqual(context.leaveRoomAlertItem?.subtitle, L10n.leaveRoomAlertSubtitle)
|
|
}
|
|
|
|
func testLeavRoomTappedWhenRoomNotPublic() async {
|
|
let mockedMembers: [RoomMemberProxyMock] = [.mockBob, .mockAlice]
|
|
roomProxyMock = RoomProxyMock(with: .init(displayName: "Test", isPublic: false, members: mockedMembers))
|
|
viewModel = RoomDetailsViewModel(roomProxy: roomProxyMock, mediaProvider: MockMediaProvider())
|
|
context.send(viewAction: .processTapLeave)
|
|
await Task.yield()
|
|
XCTAssertEqual(context.leaveRoomAlertItem?.state, .private)
|
|
XCTAssertEqual(context.leaveRoomAlertItem?.subtitle, L10n.leaveRoomAlertPrivateSubtitle)
|
|
}
|
|
|
|
func testLeaveRoomTappedWithLessThanTwoMembers() async {
|
|
context.send(viewAction: .processTapLeave)
|
|
await Task.yield()
|
|
XCTAssertEqual(context.leaveRoomAlertItem?.state, .empty)
|
|
XCTAssertEqual(context.leaveRoomAlertItem?.subtitle, L10n.leaveRoomAlertEmptySubtitle)
|
|
}
|
|
|
|
func testLeaveRoomSuccess() async {
|
|
roomProxyMock.leaveRoomClosure = {
|
|
.success(())
|
|
}
|
|
viewModel.callback = { action in
|
|
switch action {
|
|
case .leftRoom:
|
|
break
|
|
default:
|
|
XCTFail("leftRoom expected")
|
|
}
|
|
}
|
|
context.send(viewAction: .confirmLeave)
|
|
await Task.yield()
|
|
XCTAssertEqual(roomProxyMock.leaveRoomCallsCount, 1)
|
|
}
|
|
|
|
func testLeaveRoomError() async {
|
|
roomProxyMock.leaveRoomClosure = {
|
|
.failure(.failedLeavingRoom)
|
|
}
|
|
context.send(viewAction: .confirmLeave)
|
|
await Task.yield()
|
|
XCTAssertEqual(roomProxyMock.leaveRoomCallsCount, 1)
|
|
XCTAssertNotNil(context.alertInfo)
|
|
}
|
|
|
|
func testInitialDMDetailsState() async {
|
|
let recipient = RoomMemberProxyMock.mockDan
|
|
let mockedMembers: [RoomMemberProxyMock] = [.mockMe, recipient]
|
|
roomProxyMock = RoomProxyMock(with: .init(displayName: "Test", isDirect: true, isEncrypted: true, members: mockedMembers))
|
|
viewModel = RoomDetailsViewModel(roomProxy: roomProxyMock, mediaProvider: MockMediaProvider())
|
|
XCTAssertEqual(context.viewState.dmRecipient, RoomMemberDetails(withProxy: recipient))
|
|
}
|
|
|
|
func testIgnoreSuccess() async throws {
|
|
let recipient = RoomMemberProxyMock.mockDan
|
|
recipient.ignoreUserClosure = {
|
|
try? await Task.sleep(for: .milliseconds(10))
|
|
return .success(())
|
|
}
|
|
let mockedMembers: [RoomMemberProxyMock] = [.mockMe, recipient]
|
|
roomProxyMock = RoomProxyMock(with: .init(displayName: "Test", isDirect: true, isEncrypted: true, members: mockedMembers))
|
|
viewModel = RoomDetailsViewModel(roomProxy: roomProxyMock, mediaProvider: MockMediaProvider())
|
|
XCTAssertEqual(context.viewState.dmRecipient, RoomMemberDetails(withProxy: recipient))
|
|
|
|
context.send(viewAction: .ignoreConfirmed)
|
|
await Task.yield()
|
|
|
|
XCTAssertTrue(context.viewState.isProcessingIgnoreRequest)
|
|
try await Task.sleep(for: .milliseconds(10))
|
|
XCTAssertFalse(context.viewState.isProcessingIgnoreRequest)
|
|
XCTAssert(context.viewState.dmRecipient?.isIgnored == true)
|
|
}
|
|
|
|
func testIgnoreFailure() async throws {
|
|
let recipient = RoomMemberProxyMock.mockDan
|
|
recipient.ignoreUserClosure = {
|
|
try? await Task.sleep(for: .milliseconds(10))
|
|
return .failure(.ignoreUserFailed)
|
|
}
|
|
let mockedMembers: [RoomMemberProxyMock] = [.mockMe, recipient]
|
|
roomProxyMock = RoomProxyMock(with: .init(displayName: "Test", isDirect: true, isEncrypted: true, members: mockedMembers))
|
|
viewModel = RoomDetailsViewModel(roomProxy: roomProxyMock, mediaProvider: MockMediaProvider())
|
|
XCTAssertEqual(context.viewState.dmRecipient, RoomMemberDetails(withProxy: recipient))
|
|
|
|
context.send(viewAction: .ignoreConfirmed)
|
|
await Task.yield()
|
|
|
|
XCTAssertTrue(context.viewState.isProcessingIgnoreRequest)
|
|
try await Task.sleep(for: .milliseconds(10))
|
|
XCTAssertFalse(context.viewState.isProcessingIgnoreRequest)
|
|
XCTAssert(context.viewState.dmRecipient?.isIgnored == false)
|
|
XCTAssertNotNil(context.alertInfo)
|
|
}
|
|
|
|
func testUnignoreSuccess() async throws {
|
|
let recipient = RoomMemberProxyMock.mockIgnored
|
|
recipient.unignoreUserClosure = {
|
|
try? await Task.sleep(for: .milliseconds(10))
|
|
return .success(())
|
|
}
|
|
let mockedMembers: [RoomMemberProxyMock] = [.mockMe, recipient]
|
|
roomProxyMock = RoomProxyMock(with: .init(displayName: "Test", isDirect: true, isEncrypted: true, members: mockedMembers))
|
|
viewModel = RoomDetailsViewModel(roomProxy: roomProxyMock, mediaProvider: MockMediaProvider())
|
|
XCTAssertEqual(context.viewState.dmRecipient, RoomMemberDetails(withProxy: recipient))
|
|
|
|
context.send(viewAction: .unignoreConfirmed)
|
|
await Task.yield()
|
|
|
|
XCTAssertTrue(context.viewState.isProcessingIgnoreRequest)
|
|
try await Task.sleep(for: .milliseconds(10))
|
|
XCTAssertFalse(context.viewState.isProcessingIgnoreRequest)
|
|
XCTAssert(context.viewState.dmRecipient?.isIgnored == false)
|
|
}
|
|
|
|
func testUnignoreFailure() async throws {
|
|
let recipient = RoomMemberProxyMock.mockIgnored
|
|
recipient.unignoreUserClosure = {
|
|
try? await Task.sleep(for: .milliseconds(10))
|
|
return .failure(.unignoreUserFailed)
|
|
}
|
|
let mockedMembers: [RoomMemberProxyMock] = [.mockMe, recipient]
|
|
roomProxyMock = RoomProxyMock(with: .init(displayName: "Test", isDirect: true, isEncrypted: true, members: mockedMembers))
|
|
viewModel = RoomDetailsViewModel(roomProxy: roomProxyMock, mediaProvider: MockMediaProvider())
|
|
XCTAssertEqual(context.viewState.dmRecipient, RoomMemberDetails(withProxy: recipient))
|
|
|
|
context.send(viewAction: .unignoreConfirmed)
|
|
await Task.yield()
|
|
XCTAssertTrue(context.viewState.isProcessingIgnoreRequest)
|
|
|
|
try await Task.sleep(for: .milliseconds(10))
|
|
XCTAssertFalse(context.viewState.isProcessingIgnoreRequest)
|
|
XCTAssert(context.viewState.dmRecipient?.isIgnored == true)
|
|
XCTAssertNotNil(context.alertInfo)
|
|
}
|
|
}
|