mirror of
https://gitlab.opencode.de/bwi/bundesmessenger/clients/bundesmessenger-ios.git
synced 2026-04-20 00:24:43 +02:00
Add ViewActions and ViewModelActions for Coordinator
- also add Marks
This commit is contained in:
+15
-26
@@ -20,56 +20,45 @@ import SwiftUI
|
||||
|
||||
final class TemplateUserProfileCoordinator: Coordinator {
|
||||
|
||||
typealias Completion = () -> Void
|
||||
// MARK: - Properties
|
||||
|
||||
// MARK: Private
|
||||
|
||||
private let session: MXSession
|
||||
private let templateUserProfileViewController: UIViewController
|
||||
private var templateUserProfileViewModel: TemplateUserProfileViewModelProtocol
|
||||
|
||||
// MARK: Public
|
||||
|
||||
// Must be used only internally
|
||||
var childCoordinators: [Coordinator] = []
|
||||
var completion: Completion?
|
||||
var completion: (() -> Void)?
|
||||
|
||||
// MARK: - Setup
|
||||
|
||||
@available(iOS 14.0, *)
|
||||
init(session: MXSession) {
|
||||
self.session = session
|
||||
let hostViewController = VectorHostingController()
|
||||
templateUserProfileViewController = UINavigationController(rootViewController: hostViewController)
|
||||
let rootView = TemplateUserProfile.instantiate(session: session, completion: self.userProfileCompletion(result:))
|
||||
hostViewController.setRoot(view: rootView)
|
||||
}
|
||||
|
||||
@available(iOS 14.0, *)
|
||||
func userProfileCompletion(result: TemplateUserProfile.Result) {
|
||||
switch result {
|
||||
case .cancel, .done:
|
||||
completion?()
|
||||
break
|
||||
}
|
||||
let viewModel = TemplateUserProfileViewModel(userService: MXTemplateUserService(session: session))
|
||||
let view = TemplateUserProfile(viewModel: viewModel)
|
||||
.addDependency(MXAvatarService.instantiate(mediaManager: session.mediaManager))
|
||||
templateUserProfileViewModel = viewModel
|
||||
templateUserProfileViewController = VectorHostingController(rootView: view)
|
||||
}
|
||||
|
||||
// MARK: - Public methods
|
||||
|
||||
func start() {
|
||||
|
||||
templateUserProfileViewModel.completion = { [weak self] result in
|
||||
guard let self = self else { return }
|
||||
switch result {
|
||||
case .cancel, .done:
|
||||
self.completion?()
|
||||
break
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func toPresentable() -> UIViewController {
|
||||
return self.templateUserProfileViewController
|
||||
}
|
||||
}
|
||||
|
||||
@available(iOS 14.0, *)
|
||||
extension TemplateUserProfile {
|
||||
static func instantiate(session: MXSession, completion: @escaping TemplateUserProfile.Completion) -> some View {
|
||||
let templateUserProfileViewModel = TemplateUserProfileViewModel(userService: MXTemplateUserService(session: session))
|
||||
let templateUserProfile = TemplateUserProfile(viewModel: templateUserProfileViewModel, completion: completion)
|
||||
return templateUserProfile.addDependency(MXAvatarService.instantiate(mediaManager: session.mediaManager))
|
||||
}
|
||||
}
|
||||
|
||||
+22
@@ -0,0 +1,22 @@
|
||||
//
|
||||
// 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
|
||||
|
||||
enum TemplateProfileViewAction {
|
||||
case cancel
|
||||
case done
|
||||
}
|
||||
+22
@@ -0,0 +1,22 @@
|
||||
//
|
||||
// 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
|
||||
|
||||
enum TemplateUserProfileViewModelResult {
|
||||
case cancel
|
||||
case done
|
||||
}
|
||||
+24
-17
@@ -18,24 +18,17 @@ import Foundation
|
||||
import Combine
|
||||
|
||||
@available(iOS 14.0, *)
|
||||
class MXTemplateUserService: TemplateUserServiceType {
|
||||
class MXTemplateUserService: TemplateUserServiceProtocol {
|
||||
|
||||
let session: MXSession
|
||||
var listenerReference: Any!
|
||||
@Published var presence: TemplatePresence = .offline
|
||||
// MARK: - Properties
|
||||
|
||||
init(session: MXSession) {
|
||||
self.session = session
|
||||
|
||||
let listenerReference = session.myUser.listen { [weak self] event in
|
||||
guard let self = self,
|
||||
let event = event,
|
||||
case .presence = MXEventType(identifier: event.eventId)
|
||||
else { return }
|
||||
self.presence = TemplatePresence(mxPresence: self.session.myUser.presence)
|
||||
}
|
||||
self.listenerReference = listenerReference
|
||||
}
|
||||
// MARK: Private
|
||||
|
||||
private let session: MXSession
|
||||
private var listenerReference: Any!
|
||||
@Published private var presence: TemplatePresence = .offline
|
||||
|
||||
// MARK: Public
|
||||
|
||||
var userId: String {
|
||||
return session.myUser.userId
|
||||
@@ -53,6 +46,21 @@ class MXTemplateUserService: TemplateUserServiceType {
|
||||
$presence.eraseToAnyPublisher()
|
||||
}
|
||||
|
||||
// MARK: - Setup
|
||||
|
||||
init(session: MXSession) {
|
||||
self.session = session
|
||||
|
||||
let listenerReference = session.myUser.listen { [weak self] event in
|
||||
guard let self = self,
|
||||
let event = event,
|
||||
case .presence = MXEventType(identifier: event.eventId)
|
||||
else { return }
|
||||
self.presence = TemplatePresence(mxPresence: self.session.myUser.presence)
|
||||
}
|
||||
self.listenerReference = listenerReference
|
||||
}
|
||||
|
||||
deinit {
|
||||
session.myUser.removeListener(listenerReference)
|
||||
}
|
||||
@@ -61,7 +69,6 @@ class MXTemplateUserService: TemplateUserServiceType {
|
||||
fileprivate extension TemplatePresence {
|
||||
|
||||
init(mxPresence: MXPresence) {
|
||||
|
||||
switch mxPresence {
|
||||
case MXPresenceOnline:
|
||||
self = .online
|
||||
|
||||
+1
-1
@@ -18,7 +18,7 @@ import Foundation
|
||||
import Combine
|
||||
|
||||
@available(iOS 14.0, *)
|
||||
class MockTemplateUserService: TemplateUserServiceType, ObservableObject {
|
||||
class MockTemplateUserService: TemplateUserServiceProtocol, ObservableObject {
|
||||
|
||||
static let example = MockTemplateUserService()
|
||||
@Published var presence: TemplatePresence = .online
|
||||
|
||||
+2
-2
@@ -18,7 +18,7 @@ import Foundation
|
||||
import Combine
|
||||
|
||||
@available(iOS 14.0, *)
|
||||
protocol TemplateUserServiceType: Avatarable {
|
||||
protocol TemplateUserServiceProtocol: Avatarable {
|
||||
var userId: String { get }
|
||||
var displayName: String? { get }
|
||||
var avatarUrl: String? { get }
|
||||
@@ -26,7 +26,7 @@ protocol TemplateUserServiceType: Avatarable {
|
||||
}
|
||||
|
||||
@available(iOS 14.0, *)
|
||||
extension TemplateUserServiceType {
|
||||
extension TemplateUserServiceProtocol {
|
||||
var mxContentUri: String? {
|
||||
avatarUrl
|
||||
}
|
||||
@@ -18,28 +18,19 @@ import SwiftUI
|
||||
|
||||
@available(iOS 14.0, *)
|
||||
struct TemplateUserProfile: View {
|
||||
|
||||
enum Result {
|
||||
case cancel
|
||||
case done
|
||||
}
|
||||
|
||||
typealias Completion = (Result) -> Void
|
||||
|
||||
|
||||
@Environment(\.theme) var theme: ThemeSwiftUI
|
||||
@ObservedObject var viewModel: TemplateUserProfileViewModel
|
||||
|
||||
var completion: Completion
|
||||
|
||||
|
||||
var leftButton: some View {
|
||||
Button(VectorL10n.cancel) {
|
||||
completion(.cancel)
|
||||
viewModel.proccess(viewAction: .cancel)
|
||||
}
|
||||
}
|
||||
|
||||
var rightButton: some View {
|
||||
Button(VectorL10n.done) {
|
||||
completion(.done)
|
||||
viewModel.proccess(viewAction: .cancel)
|
||||
}
|
||||
}
|
||||
|
||||
@@ -71,9 +62,7 @@ struct TemplateUserProfile: View {
|
||||
@available(iOS 14.0, *)
|
||||
struct TemplateUserProfile_Previews: PreviewProvider {
|
||||
static var previews: some View {
|
||||
TemplateUserProfile(viewModel: TemplateUserProfileViewModel(userService: MockTemplateUserService.example)) { _ in
|
||||
|
||||
}
|
||||
TemplateUserProfile(viewModel: TemplateUserProfileViewModel(userService: MockTemplateUserService.example))
|
||||
.addDependency(MockAvatarService.example)
|
||||
}
|
||||
}
|
||||
|
||||
+32
-6
@@ -16,20 +16,27 @@
|
||||
|
||||
import SwiftUI
|
||||
import Combine
|
||||
|
||||
|
||||
@available(iOS 14.0, *)
|
||||
class TemplateUserProfileViewModel: ObservableObject {
|
||||
class TemplateUserProfileViewModel: ObservableObject, TemplateUserProfileViewModelProtocol {
|
||||
|
||||
private let userService: TemplateUserServiceType
|
||||
private var cancellables = Set<AnyCancellable>()
|
||||
// MARK: - Properties
|
||||
|
||||
// MARK: Private
|
||||
private let userService: TemplateUserServiceProtocol
|
||||
private var cancellables = Set<AnyCancellable>()
|
||||
|
||||
// MARK: Public
|
||||
@Published private(set) var viewState: TemplateUserProfileViewState
|
||||
|
||||
private static func defaultState(userService: TemplateUserServiceType) -> TemplateUserProfileViewState {
|
||||
var completion: ((TemplateUserProfileViewModelResult) -> Void)?
|
||||
|
||||
private static func defaultState(userService: TemplateUserServiceProtocol) -> TemplateUserProfileViewState {
|
||||
return TemplateUserProfileViewState(avatar: userService.avatarData, displayName: userService.displayName)
|
||||
}
|
||||
|
||||
init(userService: TemplateUserServiceType, initialState: TemplateUserProfileViewState? = nil) {
|
||||
// MARK: - Setup
|
||||
init(userService: TemplateUserServiceProtocol, initialState: TemplateUserProfileViewState? = nil) {
|
||||
self.userService = userService
|
||||
self.viewState = initialState ?? Self.defaultState(userService: userService)
|
||||
|
||||
@@ -40,6 +47,17 @@ class TemplateUserProfileViewModel: ObservableObject {
|
||||
.store(in: &cancellables)
|
||||
}
|
||||
|
||||
// MARK: - Public methods
|
||||
func proccess(viewAction: TemplateProfileViewAction) {
|
||||
switch viewAction {
|
||||
case .cancel:
|
||||
self.cancel()
|
||||
case .done:
|
||||
self.done()
|
||||
}
|
||||
}
|
||||
|
||||
// MARK: - Private methods
|
||||
/**
|
||||
Send state actions to mutate the state.
|
||||
*/
|
||||
@@ -58,4 +76,12 @@ class TemplateUserProfileViewModel: ObservableObject {
|
||||
state.presence = presence
|
||||
}
|
||||
}
|
||||
|
||||
private func done() {
|
||||
completion?(.done)
|
||||
}
|
||||
|
||||
private func cancel() {
|
||||
completion?(.cancel)
|
||||
}
|
||||
}
|
||||
|
||||
+21
@@ -0,0 +1,21 @@
|
||||
//
|
||||
// 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
|
||||
|
||||
protocol TemplateUserProfileViewModelProtocol {
|
||||
var completion: ((TemplateUserProfileViewModelResult) -> Void)? { get set }
|
||||
}
|
||||
Reference in New Issue
Block a user