Add Simple Template Example

- Add a simple Template example that shows a user profile with avatar, displayName and presence.
- ScreenCoordinator: closure based with less protocols and delegates.
- Reducer: Reducer function that manages all state modifications.
- SwiftUI View: Decomposes UI into appropriate sub components.
- Uses Theme and Dependency Management Infrastructure
This commit is contained in:
David Langley
2021-09-06 17:12:31 +01:00
parent 655d67582a
commit d82ae75a0b
18 changed files with 438 additions and 31 deletions
@@ -15,13 +15,15 @@
//
import SwiftUI
import Combine
@available(iOS 14.0, *)
class TemplateUserProfileViewModel: ObservableObject {
private let userService: TemplateUserServiceType
private var cancellables = Set<AnyCancellable>()
@Published var viewState: TemplateUserProfileViewState
@Published private(set) var viewState: TemplateUserProfileViewState
private static func defaultState(userService: TemplateUserServiceType) -> TemplateUserProfileViewState {
return TemplateUserProfileViewState(avatar: userService.avatarData, displayName: userService.displayName)
@@ -30,5 +32,30 @@ class TemplateUserProfileViewModel: ObservableObject {
init(userService: TemplateUserServiceType, initialState: TemplateUserProfileViewState? = nil) {
self.userService = userService
self.viewState = initialState ?? Self.defaultState(userService: userService)
userService.presencePublisher
.map(TemplateProfileStateAction.updatePresence)
.receive(on: DispatchQueue.main)
.sink(receiveValue: self.dispatch(action:))
.store(in: &cancellables)
}
/**
Send state actions to mutate the state.
*/
private func dispatch(action: TemplateProfileStateAction) {
var newState = self.viewState
reducer(state: &newState, action: action)
self.viewState = newState
}
/**
A redux style reducer, all modifications to state happen here. Recieves a state and a state action and produces a new state.
*/
private func reducer(state: inout TemplateUserProfileViewState, action: TemplateProfileStateAction) {
switch action {
case .updatePresence(let presence):
state.presence = presence
}
}
}
@@ -1,34 +0,0 @@
//
// 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 TemplateUserServiceType: Avatarable {
var userId: String { get }
var displayName: String? { get }
var avatarUrl: String? { get }
var currentlyActive: Bool { get }
var lastActive: UInt { get }
}
extension TemplateUserServiceType {
var mxContentUri: String? {
avatarUrl
}
var matrixItemId: String {
userId
}
}