element-ios/Riot/Modules/Spaces/SpaceDetail/SpaceDetailViewController.s...

283 lines
11 KiB
Swift

//
// Copyright 2021-2024 New Vector Ltd.
//
// SPDX-License-Identifier: AGPL-3.0-only
// Please see LICENSE in the repository root for full details.
//
import Foundation
class SpaceDetailViewController: UIViewController {
// MARK: - Constants
private enum Constants {
static let popoverWidth: CGFloat = 320
static let topicMaxHeight: CGFloat = 105
}
// MARK: Private
private var theme: Theme!
private var mediaManager: MXMediaManager!
private var viewModel: SpaceDetailViewModelType!
private var errorPresenter: MXKErrorPresentation!
private var activityPresenter: ActivityIndicatorPresenter!
private var isJoined: Bool = false
private var showCancel: Bool = true
// MARK: Outlets
@IBOutlet private weak var inviterPanelHeight: NSLayoutConstraint!
@IBOutlet private weak var inviterAvatarView: RoomAvatarView!
@IBOutlet private weak var inviterTitleLabel: UILabel!
@IBOutlet private weak var inviterIdLabel: UILabel!
@IBOutlet private weak var inviterSeparatorView: UIView!
@IBOutlet private weak var avatarView: SpaceAvatarView!
@IBOutlet private weak var titleLabel: UILabel!
@IBOutlet private weak var closeButton: UIButton!
@IBOutlet private weak var spaceTypeIconView: UIImageView!
@IBOutlet private weak var spaceTypeLabel: UILabel!
@IBOutlet private weak var topicLabel: UILabel!
@IBOutlet private weak var topicScrollView: UIScrollView!
@IBOutlet private weak var joinButtonTopMargin: NSLayoutConstraint!
@IBOutlet private weak var joinButtonBottomMargin: NSLayoutConstraint!
@IBOutlet private weak var joinButton: UIButton!
@IBOutlet private weak var declineButton: UIButton!
@IBOutlet private weak var acceptButton: UIButton!
@IBOutlet private weak var inviteActionPanel: UIView!
// MARK: - Setup
class func instantiate(mediaManager: MXMediaManager, viewModel: SpaceDetailViewModelType!, showCancel: Bool) -> SpaceDetailViewController {
let viewController = StoryboardScene.SpaceDetailViewController.initialScene.instantiate()
viewController.mediaManager = mediaManager
viewController.viewModel = viewModel
viewController.theme = ThemeService.shared().theme
viewController.showCancel = showCancel
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 var preferredStatusBarStyle: UIStatusBarStyle {
return self.theme.statusBarStyle
}
override var preferredContentSize: CGSize {
get {
return CGSize(width: Constants.popoverWidth, height: self.intrisicHeight(with: Constants.popoverWidth))
}
set {
super.preferredContentSize = newValue
}
}
override func viewWillDisappear(_ animated: Bool) {
super.viewWillDisappear(animated)
self.viewModel.process(viewAction: .dismissed)
}
// MARK: - IBActions
@IBAction private func closeAction(sender: UIButton) {
self.viewModel.process(viewAction: .dismiss)
}
@IBAction private func joinAction(sender: UIButton) {
if isJoined {
self.viewModel.process(viewAction: .open)
} else {
self.viewModel.process(viewAction: .join)
}
}
@IBAction private func leaveAction(sender: UIButton) {
self.viewModel.process(viewAction: .leave)
}
// MARK: - Private
private func update(theme: Theme) {
self.theme = theme
self.view.backgroundColor = theme.colors.background
self.inviterAvatarView.update(theme: theme)
self.inviterTitleLabel.textColor = theme.colors.secondaryContent
self.inviterTitleLabel.font = theme.fonts.calloutSB
self.inviterIdLabel.textColor = theme.colors.secondaryContent
self.inviterIdLabel.font = theme.fonts.footnote
self.inviterSeparatorView.backgroundColor = theme.colors.navigation
self.titleLabel.textColor = theme.colors.primaryContent
self.titleLabel.font = theme.fonts.title3SB
self.closeButton.backgroundColor = theme.roomInputTextBorder
self.closeButton.tintColor = theme.noticeSecondaryColor
self.avatarView.update(theme: theme)
self.spaceTypeIconView.tintColor = theme.colors.tertiaryContent
self.spaceTypeLabel.font = theme.fonts.callout
self.spaceTypeLabel.textColor = theme.colors.tertiaryContent
self.topicLabel.font = theme.fonts.caption1
self.topicLabel.textColor = theme.colors.tertiaryContent
apply(theme: theme, on: self.joinButton)
apply(theme: theme, on: self.acceptButton)
self.declineButton.layer.borderColor = theme.colors.alert.cgColor
self.declineButton.tintColor = theme.colors.alert
self.declineButton.setTitleColor(theme.colors.alert, for: .normal)
self.declineButton.titleLabel?.font = theme.fonts.body
}
private func apply(theme: Theme, on button: UIButton) {
button.backgroundColor = theme.colors.accent
button.tintColor = theme.colors.background
button.setTitleColor(theme.colors.background, for: .normal)
button.titleLabel?.font = theme.fonts.bodySB
}
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() {
self.closeButton.layer.masksToBounds = true
self.closeButton.layer.cornerRadius = self.closeButton.bounds.height / 2
self.closeButton.isHidden = !self.showCancel
self.setup(button: self.joinButton, withTitle: VectorL10n.join)
self.setup(button: self.acceptButton, withTitle: VectorL10n.accept)
self.setup(button: self.declineButton, withTitle: VectorL10n.decline)
self.declineButton.layer.borderWidth = 1.0
}
private func setup(button: UIButton, withTitle title: String) {
button.layer.masksToBounds = true
button.layer.cornerRadius = 8.0
button.setTitle(title, for: .normal)
}
private func render(viewState: SpaceDetailViewState) {
switch viewState {
case .loading:
self.renderLoading()
case .loaded(let parameters):
self.renderLoaded(parameters: parameters)
case .error(let error):
self.render(error: error)
}
}
private func renderLoading() {
self.activityPresenter.presentActivityIndicator(on: self.view, animated: true)
}
private func renderLoaded(parameters: SpaceDetailLoadedParameters) {
self.activityPresenter.removeCurrentActivityIndicator(animated: true)
switch parameters.membership {
case .invite:
self.title = VectorL10n.spaceInviteNavTitle
self.joinButton.isHidden = true
self.inviteActionPanel.isHidden = false
case .join:
self.title = VectorL10n.spaceDetailNavTitle
self.inviterPanelHeight.constant = 0
self.joinButton.setTitle(VectorL10n.open, for: .normal)
self.isJoined = true
default:
self.title = VectorL10n.spaceDetailNavTitle
self.inviterPanelHeight.constant = 0
}
let avatarViewData = AvatarViewData(matrixItemId: parameters.spaceId,
displayName: parameters.displayName,
avatarUrl: parameters.avatarUrl,
mediaManager: self.mediaManager,
fallbackImage: .matrixItem(parameters.spaceId, parameters.displayName))
self.titleLabel.text = parameters.displayName
self.avatarView.fill(with: avatarViewData)
self.topicLabel.text = parameters.topic
let joinRuleString = parameters.joinRule == .public ? VectorL10n.spacePublicJoinRule : VectorL10n.spacePrivateJoinRule
let membersCount = parameters.membersCount
let membersString = membersCount == 1 ? VectorL10n.roomTitleOneMember : VectorL10n.roomTitleMembers("\(membersCount)")
self.spaceTypeLabel.text = "\(joinRuleString) · \(membersString)"
let joinRuleIcon = parameters.joinRule == .public ? Asset.Images.spaceTypeIcon : Asset.Images.spacePrivateIcon
self.spaceTypeIconView.image = joinRuleIcon.image
self.inviterIdLabel.text = parameters.inviterId
if let inviterId = parameters.inviterId {
self.inviterTitleLabel.text = "\(parameters.inviter?.displayname ?? inviterId) invited you"
if let inviter = parameters.inviter {
let avatarViewData = AvatarViewData(matrixItemId: inviter.userId, displayName: inviter.displayname, avatarUrl: inviter.avatarUrl, mediaManager: self.mediaManager, fallbackImage: .matrixItem(inviter.userId, inviter.displayname))
self.inviterAvatarView.fill(with: avatarViewData)
}
}
view.layoutIfNeeded()
}
private func render(error: Error) {
self.activityPresenter.removeCurrentActivityIndicator(animated: true)
self.errorPresenter.presentError(from: self, forError: error, animated: true, handler: nil)
}
private func intrisicHeight(with width: CGFloat) -> CGFloat {
let topicHeight = min(self.topicLabel.sizeThatFits(CGSize(width: width - self.topicScrollView.frame.minX * 2, height: 0)).height, Constants.topicMaxHeight)
return self.topicScrollView.frame.minY + topicHeight + self.joinButton.frame.height + self.joinButtonTopMargin.constant + self.joinButtonBottomMargin.constant
}
}
// MARK: - SlidingModalPresentable
extension SpaceDetailViewController: SlidingModalPresentable {
func allowsDismissOnBackgroundTap() -> Bool {
return true
}
func layoutHeightFittingWidth(_ width: CGFloat) -> CGFloat {
return self.intrisicHeight(with: width) + self.joinButtonTopMargin.constant + self.joinButtonBottomMargin.constant
}
}
// MARK: - SpaceDetailViewModelViewDelegate
extension SpaceDetailViewController: SpaceDetailViewModelViewDelegate {
func spaceDetailViewModel(_ viewModel: SpaceDetailViewModelType, didUpdateViewState viewSate: SpaceDetailViewState) {
self.render(viewState: viewSate)
}
}