element-ios/Riot/Modules/SideMenu/SideMenuViewController.swift

225 lines
7.8 KiB
Swift

// File created from ScreenTemplate
// $ createScreen.sh SideMenu SideMenu
/*
Copyright 2020 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 UIKit
final class SideMenuViewController: UIViewController {
// MARK: - Constants
private enum Constants {
static let sideMenuActionViewHeight: CGFloat = 44.0
}
// MARK: - Properties
// MARK: Outlets
@IBOutlet weak var spaceListContainerView: UIView!
// User info
@IBOutlet private weak var userAvatarView: UserAvatarView!
@IBOutlet private weak var userDisplayNameLabel: UILabel!
@IBOutlet private weak var userIdLabel: UILabel!
// Bottom menu items
@IBOutlet private weak var menuItemsStackView: UIStackView!
// MARK: Private
private var viewModel: SideMenuViewModelType!
private var theme: Theme!
private var keyboardAvoider: KeyboardAvoider?
private var errorPresenter: MXKErrorPresentation!
private var activityPresenter: ActivityIndicatorPresenter!
private var screenTracker = AnalyticsScreenTracker(screen: .sidebar)
private var sideMenuActionViews: [SideMenuActionView] = []
private weak var sideMenuVersionView: SideMenuVersionView?
// MARK: - Setup
class func instantiate(with viewModel: SideMenuViewModelType) -> SideMenuViewController {
let viewController = StoryboardScene.SideMenuViewController.initialScene.instantiate()
viewController.viewModel = viewModel
viewController.theme = ThemeService.shared().theme
return viewController
}
// MARK: - Life cycle
override func viewDidLoad() {
super.viewDidLoad()
// Do any additional setup after loading the view.
self.setupViews()
self.activityPresenter = ActivityIndicatorPresenter()
self.errorPresenter = MXKErrorAlertPresentation()
self.registerThemeServiceDidChangeThemeNotification()
self.update(theme: self.theme)
self.viewModel.viewDelegate = self
self.viewModel.process(viewAction: .loadData)
}
override func viewWillAppear(_ animated: Bool) {
super.viewWillAppear(animated)
navigationController?.setNavigationBarHidden(true, animated: animated)
screenTracker.trackScreen()
}
override var preferredStatusBarStyle: UIStatusBarStyle {
return self.theme.statusBarStyle
}
// MARK: - Private
private func update(theme: Theme) {
self.theme = theme
if let navigationBar = self.navigationController?.navigationBar {
theme.applyStyle(onNavigationBar: navigationBar)
}
self.view.backgroundColor = theme.colors.background
self.userAvatarView.update(theme: theme)
self.userDisplayNameLabel.textColor = theme.textPrimaryColor
self.userDisplayNameLabel.font = theme.fonts.title3SB
self.userIdLabel.textColor = theme.textSecondaryColor
for sideMenuActionView in self.sideMenuActionViews {
sideMenuActionView.update(theme: theme)
}
self.sideMenuVersionView?.update(theme: theme)
}
private func registerThemeServiceDidChangeThemeNotification() {
NotificationCenter.default.addObserver(self, selector: #selector(themeDidChange), name: .themeServiceDidChangeTheme, object: nil)
}
@objc private func themeDidChange() {
self.update(theme: ThemeService.shared().theme)
}
private func setupViews() {
}
private func render(viewState: SideMenuViewState) {
switch viewState {
case .loading:
self.renderLoading()
case .loaded(let viewData):
self.renderLoaded(viewData: viewData)
case .error(let error):
self.render(error: error)
}
}
private func renderLoading() {
self.activityPresenter.presentActivityIndicator(on: self.view, animated: true)
}
private func renderLoaded(viewData: SideMenuViewData) {
self.activityPresenter.removeCurrentActivityIndicator(animated: true)
self.updateUserInformation(with: viewData.userAvatarViewData)
self.updateBottomMenuItems(with: viewData)
}
private func updateUserInformation(with userAvatarViewData: UserAvatarViewData) {
self.userIdLabel.text = userAvatarViewData.userId
self.userDisplayNameLabel.text = userAvatarViewData.displayName
self.userDisplayNameLabel.isHidden = userAvatarViewData.displayName.isEmptyOrNil
self.userAvatarView.fill(with: userAvatarViewData)
}
private func updateBottomMenuItems(with viewData: SideMenuViewData) {
self.menuItemsStackView.vc_removeAllSubviews()
self.sideMenuActionViews = []
for sideMenuItem in viewData.sideMenuItems {
let sideMenuActionView = SideMenuActionView.instantiate()
sideMenuActionView.translatesAutoresizingMaskIntoConstraints = false
let heightConstraint = sideMenuActionView.heightAnchor.constraint(equalToConstant: 0)
heightConstraint.priority = .defaultLow
heightConstraint.isActive = true
sideMenuActionView.update(theme: self.theme)
sideMenuActionView.fill(with: sideMenuItem)
sideMenuActionView.delegate = self
self.menuItemsStackView.addArrangedSubview(sideMenuActionView)
sideMenuActionView.widthAnchor.constraint(equalTo: menuItemsStackView.widthAnchor).isActive = true
self.sideMenuActionViews.append(sideMenuActionView)
}
if let appVersion = viewData.appVersion {
let sideMenuVersionView = SideMenuVersionView.instantiate()
sideMenuVersionView.translatesAutoresizingMaskIntoConstraints = false
sideMenuVersionView.update(theme: self.theme)
sideMenuVersionView.fill(with: appVersion)
self.menuItemsStackView.addArrangedSubview(sideMenuVersionView)
sideMenuVersionView.widthAnchor.constraint(equalTo: menuItemsStackView.widthAnchor).isActive = true
self.sideMenuVersionView = sideMenuVersionView
}
}
private func render(error: Error) {
self.activityPresenter.removeCurrentActivityIndicator(animated: true)
self.errorPresenter.presentError(from: self, forError: error, animated: true, handler: nil)
}
// MARK: - Actions
@IBAction func headerTapAction(sender: UIView) {
self.viewModel.process(viewAction: .tapHeader(sourceView: sender))
}
}
// MARK: - SideMenuViewModelViewDelegate
extension SideMenuViewController: SideMenuViewModelViewDelegate {
func sideMenuViewModel(_ viewModel: SideMenuViewModelType, didUpdateViewState viewSate: SideMenuViewState) {
self.render(viewState: viewSate)
}
}
// MARK: - SideMenuActionViewDelegate
extension SideMenuViewController: SideMenuActionViewDelegate {
func sideMenuActionView(_ actionView: SideMenuActionView, didTapMenuItem sideMenuItem: SideMenuItem?) {
guard let sideMenuItem = sideMenuItem else {
return
}
self.viewModel.process(viewAction: .tap(menuItem: sideMenuItem, sourceView: actionView))
}
}