mirror of
https://gitlab.opencode.de/bwi/bundesmessenger/clients/bundesmessenger-ios.git
synced 2026-04-22 01:22:46 +02:00
UITests and Mock Screens
Adds the test targets for Unit and UI tests Adds mock screen data and utilities to render each mock screen for previews/ui tests. Changes Published property in the service to CurrentValueSubject. we don't need the synthesized aspect of Published and property wrappers cannot be included in protocols.
This commit is contained in:
+3
-6
@@ -27,8 +27,6 @@ class TemplateUserProfileService: TemplateUserProfileServiceProtocol {
|
||||
private let session: MXSession
|
||||
private var listenerReference: Any?
|
||||
|
||||
@Published private var presence: TemplateUserProfilePresence = .offline
|
||||
|
||||
// MARK: Public
|
||||
|
||||
var userId: String {
|
||||
@@ -43,14 +41,13 @@ class TemplateUserProfileService: TemplateUserProfileServiceProtocol {
|
||||
session.myUser.avatarUrl
|
||||
}
|
||||
|
||||
var presencePublisher: AnyPublisher<TemplateUserProfilePresence, Never> {
|
||||
$presence.eraseToAnyPublisher()
|
||||
}
|
||||
private(set) var presenceSubject: CurrentValueSubject<TemplateUserProfilePresence, Never>
|
||||
|
||||
// MARK: - Setup
|
||||
|
||||
init(session: MXSession) {
|
||||
self.session = session
|
||||
self.presenceSubject = CurrentValueSubject(TemplateUserProfilePresence(mxPresence: session.myUser.presence))
|
||||
self.listenerReference = setupPresenceListener()
|
||||
}
|
||||
|
||||
@@ -65,7 +62,7 @@ class TemplateUserProfileService: TemplateUserProfileServiceProtocol {
|
||||
let event = event,
|
||||
case .presence = MXEventType(identifier: event.eventId)
|
||||
else { return }
|
||||
self.presence = TemplateUserProfilePresence(mxPresence: self.session.myUser.presence)
|
||||
self.presenceSubject.send(TemplateUserProfilePresence(mxPresence: self.session.myUser.presence))
|
||||
}
|
||||
// TODO: Add log back when abstract logger added to RiotSwiftUI
|
||||
// if reference == nil {
|
||||
|
||||
+45
@@ -0,0 +1,45 @@
|
||||
//
|
||||
// Copyright 2021 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 Foundation
|
||||
import SwiftUI
|
||||
|
||||
/**
|
||||
Using an enum for the screen allows you define the different state cases with
|
||||
the relevant associated data for each case.
|
||||
*/
|
||||
@available(iOS 14.0, *)
|
||||
enum MockTemplateProfileUserScreenStates: MockScreen {
|
||||
|
||||
case mockPresenceStates(TemplateUserProfilePresence)
|
||||
case mockLongDisplayName(String)
|
||||
|
||||
static var screenStates: [MockTemplateProfileUserScreenStates] = TemplateUserProfilePresence.allCases.map(MockTemplateProfileUserScreenStates.mockPresenceStates)
|
||||
+ [.mockLongDisplayName("Somebody with a super long name we would like to test")]
|
||||
|
||||
static func screen(for state: MockTemplateProfileUserScreenStates) -> some View {
|
||||
let service: MockTemplateUserProfileService
|
||||
switch state {
|
||||
case .mockPresenceStates(let presence):
|
||||
service = MockTemplateUserProfileService(presence: presence)
|
||||
case .mockLongDisplayName(let displayName):
|
||||
service = MockTemplateUserProfileService(displayName: displayName)
|
||||
}
|
||||
let viewModel = TemplateUserProfileViewModel(userService: service)
|
||||
return TemplateUserProfile(viewModel: viewModel)
|
||||
.addDependency(MockAvatarService.example)
|
||||
}
|
||||
}
|
||||
+16
-12
@@ -19,20 +19,24 @@ import Combine
|
||||
|
||||
@available(iOS 14.0, *)
|
||||
class MockTemplateUserProfileService: TemplateUserProfileServiceProtocol {
|
||||
|
||||
static let example = MockTemplateUserProfileService()
|
||||
static let initialPresenceState: TemplateUserProfilePresence = .offline
|
||||
@Published var presence: TemplateUserProfilePresence = initialPresenceState
|
||||
var presencePublisher: AnyPublisher<TemplateUserProfilePresence, Never> {
|
||||
$presence.eraseToAnyPublisher()
|
||||
var presenceSubject: CurrentValueSubject<TemplateUserProfilePresence, Never>
|
||||
|
||||
let userId: String
|
||||
var displayName: String?
|
||||
let avatarUrl: String?
|
||||
init(
|
||||
userId: String = "123",
|
||||
displayName: String? = "Alice",
|
||||
avatarUrl: String? = "mx123@matrix.com",
|
||||
presence: TemplateUserProfilePresence = .offline
|
||||
) {
|
||||
self.userId = userId
|
||||
self.displayName = displayName
|
||||
self.avatarUrl = avatarUrl
|
||||
self.presenceSubject = CurrentValueSubject<TemplateUserProfilePresence, Never>(presence)
|
||||
}
|
||||
let userId: String = "123"
|
||||
let displayName: String? = "Alice"
|
||||
let avatarUrl: String? = "mx123@matrix.com"
|
||||
let currentlyActive: Bool = true
|
||||
let lastActive: UInt = 1630596918513
|
||||
|
||||
func simulateUpdate(presence: TemplateUserProfilePresence) {
|
||||
self.presence = presence
|
||||
self.presenceSubject.send(presence)
|
||||
}
|
||||
}
|
||||
|
||||
+1
-1
@@ -22,7 +22,7 @@ protocol TemplateUserProfileServiceProtocol: Avatarable {
|
||||
var userId: String { get }
|
||||
var displayName: String? { get }
|
||||
var avatarUrl: String? { get }
|
||||
var presencePublisher: AnyPublisher<TemplateUserProfilePresence, Never> { get }
|
||||
var presenceSubject: CurrentValueSubject<TemplateUserProfilePresence, Never> { get }
|
||||
}
|
||||
|
||||
@available(iOS 14.0, *)
|
||||
|
||||
+47
@@ -0,0 +1,47 @@
|
||||
//
|
||||
// Copyright 2021 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 RiotSwiftUI
|
||||
|
||||
@available(iOS 14.0, *)
|
||||
class TestUserProfileUITests: XCTestCase {
|
||||
|
||||
let app = XCUIApplication()
|
||||
|
||||
override func setUpWithError() throws {
|
||||
// Put setup code here. This method is called before the invocation of each test method in the class.
|
||||
|
||||
// In UI tests it is usually best to stop immediately when a failure occurs.
|
||||
continueAfterFailure = false
|
||||
|
||||
// UI tests must launch the application that they test. Doing this in setup will make sure it happens for each test method.
|
||||
app.launch()
|
||||
|
||||
// In UI tests it’s important to set the initial state - such as interface orientation - required for your tests before they run. The setUp method is a good place to do this.
|
||||
}
|
||||
|
||||
override func tearDownWithError() throws {
|
||||
// Put teardown code here. This method is called after the invocation of each test method in the class.
|
||||
}
|
||||
|
||||
func testUserContentTextDisplayed() throws {
|
||||
let userContentText = app.staticTexts["More great user content!"]
|
||||
XCTAssert(userContentText.exists)
|
||||
}
|
||||
|
||||
}
|
||||
+10
-9
@@ -17,27 +17,30 @@
|
||||
import XCTest
|
||||
import Combine
|
||||
|
||||
@testable import Riot
|
||||
@testable import RiotSwiftUI
|
||||
|
||||
@available(iOS 14.0, *)
|
||||
class TemplateUserProfileViewModelTests: XCTestCase {
|
||||
private enum Constants {
|
||||
static let presenceInitialValue: TemplateUserProfilePresence = .offline
|
||||
static let displayName = "Alice"
|
||||
}
|
||||
var service: MockTemplateUserProfileService!
|
||||
var viewModel: TemplateUserProfileViewModel!
|
||||
var cancellables = Set<AnyCancellable>()
|
||||
override func setUpWithError() throws {
|
||||
service = MockTemplateUserProfileService()
|
||||
service = MockTemplateUserProfileService(displayName: Constants.displayName, presence: Constants.presenceInitialValue)
|
||||
viewModel = TemplateUserProfileViewModel(userService: service)
|
||||
}
|
||||
|
||||
func testInitialState() {
|
||||
XCTAssertEqual(viewModel.viewState.displayName, MockTemplateUserProfileService.example.displayName)
|
||||
XCTAssertEqual(viewModel.viewState.avatar?.mxContentUri, MockTemplateUserProfileService.example.avatarUrl)
|
||||
XCTAssertEqual(viewModel.viewState.presence, MockTemplateUserProfileService.initialPresenceState)
|
||||
XCTAssertEqual(viewModel.viewState.displayName, Constants.displayName)
|
||||
XCTAssertEqual(viewModel.viewState.presence, Constants.presenceInitialValue)
|
||||
}
|
||||
|
||||
func testFirstPresenceRecieved() throws {
|
||||
let presencePublisher = viewModel.$viewState.map(\.presence).removeDuplicates().collect(1).first()
|
||||
XCTAssertEqual(try xcAwait(presencePublisher), [MockTemplateUserProfileService.initialPresenceState])
|
||||
XCTAssertEqual(try xcAwait(presencePublisher), [Constants.presenceInitialValue])
|
||||
}
|
||||
|
||||
func testPresenceUpdatesRecieved() throws {
|
||||
@@ -46,8 +49,6 @@ class TemplateUserProfileViewModelTests: XCTestCase {
|
||||
let newPresenceValue2: TemplateUserProfilePresence = .idle
|
||||
service.simulateUpdate(presence: newPresenceValue1)
|
||||
service.simulateUpdate(presence: newPresenceValue2)
|
||||
XCTAssertEqual(try xcAwait(presencePublisher), [MockTemplateUserProfileService.initialPresenceState, newPresenceValue1, newPresenceValue2])
|
||||
XCTAssertEqual(try xcAwait(presencePublisher), [Constants.presenceInitialValue, newPresenceValue1, newPresenceValue2])
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
+2
-2
@@ -45,6 +45,7 @@ struct TemplateUserProfile: View {
|
||||
}
|
||||
.frame(maxHeight: .infinity)
|
||||
}
|
||||
.background(theme.colors.background)
|
||||
.frame(maxHeight: .infinity)
|
||||
.navigationTitle(viewModel.viewState.displayName ?? "")
|
||||
.navigationBarItems(leading: leftButton, trailing: rightButton)
|
||||
@@ -70,7 +71,6 @@ struct TemplateUserProfile: View {
|
||||
@available(iOS 14.0, *)
|
||||
struct TemplateUserProfile_Previews: PreviewProvider {
|
||||
static var previews: some View {
|
||||
TemplateUserProfile(viewModel: TemplateUserProfileViewModel(userService: MockTemplateUserProfileService.example))
|
||||
.addDependency(MockAvatarService.example)
|
||||
MockTemplateProfileUserScreenStates.screenGroup()
|
||||
}
|
||||
}
|
||||
|
||||
+2
-2
@@ -36,7 +36,7 @@ class TemplateUserProfileViewModel: ObservableObject, TemplateUserProfileViewMod
|
||||
self.userService = userService
|
||||
self.viewState = initialState ?? Self.defaultState(userService: userService)
|
||||
|
||||
userService.presencePublisher
|
||||
userService.presenceSubject
|
||||
.map(TemplateUserProfileStateAction.updatePresence)
|
||||
.receive(on: DispatchQueue.main)
|
||||
.sink(receiveValue: { [weak self] action in
|
||||
@@ -46,7 +46,7 @@ class TemplateUserProfileViewModel: ObservableObject, TemplateUserProfileViewMod
|
||||
}
|
||||
|
||||
private static func defaultState(userService: TemplateUserProfileServiceProtocol) -> TemplateUserProfileViewState {
|
||||
return TemplateUserProfileViewState(avatar: userService.avatarData, displayName: userService.displayName, presence: .offline)
|
||||
return TemplateUserProfileViewState(avatar: userService.avatarData, displayName: userService.displayName, presence: userService.presenceSubject.value)
|
||||
}
|
||||
|
||||
// MARK: - Public
|
||||
|
||||
Reference in New Issue
Block a user