mirror of
https://gitlab.opencode.de/bwi/bundesmessenger/clients/bundesmessenger-ios.git
synced 2026-04-29 20:56:57 +02:00
b298dedc22
Merge commit 'f823ab9aae70e8d15ed7cc079210dd9bbbb6c8e1' into feature/foss_update_1_11_19 * commit 'f823ab9aae70e8d15ed7cc079210dd9bbbb6c8e1': finish version++ version++ comments update submodule remove obsolete tests removed unused code update submodule fix Libolm removal update license macro update license Prepare for new sprint # Conflicts: # Config/AppVersion.xcconfig # IDETemplateMacros.plist # LICENSE # README.md # Riot/Categories/MXSession+Riot.m # Riot/Managers/EncryptionKeyManager/EncryptionKeyManager.swift # Riot/Managers/KeyValueStorage/Extensions/Keychain.swift # Riot/Managers/KeyValueStorage/KeyValueStore.swift # Riot/Managers/KeyValueStorage/KeychainStore.swift # Riot/Managers/KeyValueStorage/MemoryStore.swift # Riot/Managers/PushNotification/PushNotificationService.m # Riot/Managers/Settings/RiotSettings.swift # Riot/Managers/Settings/Shared/RiotSharedSettings.swift # Riot/Modules/Analytics/AnalyticsUIElement.swift # Riot/Modules/Application/AppCoordinator.swift # Riot/Modules/Application/LegacyAppDelegate.h # Riot/Modules/Application/LegacyAppDelegate.m # Riot/Modules/Authentication/Legacy/AuthenticationViewController.h # Riot/Modules/Authentication/Legacy/AuthenticationViewController.m # Riot/Modules/Authentication/Legacy/Views/AuthInputsView.h # Riot/Modules/Authentication/Legacy/Views/AuthInputsView.m # Riot/Modules/Common/Recents/DataSources/RecentsDataSource.m # Riot/Modules/Common/Recents/RecentsViewController.m # Riot/Modules/Common/WebViewController/WebViewViewController.m # Riot/Modules/Contacts/Details/ContactDetailsViewController.m # Riot/Modules/Contacts/Views/ContactTableViewCell.m # Riot/Modules/Favorites/FavouritesViewController.h # Riot/Modules/Favorites/FavouritesViewController.m # Riot/Modules/GlobalSearch/UnifiedSearchViewController.m # Riot/Modules/People/PeopleViewController.h # Riot/Modules/People/PeopleViewController.m # Riot/Modules/Room/ContextualMenu/ReactionsMenu/ReactionsMenuViewModel.swift # Riot/Modules/Room/DataSources/RoomDataSource.m # Riot/Modules/Room/Files/RoomFilesViewController.m # Riot/Modules/Room/Members/Detail/RoomMemberDetailsViewController.m # Riot/Modules/Room/Members/RoomParticipantsViewController.m # Riot/Modules/Room/RoomViewController.m # Riot/Modules/Room/Settings/RoomSettingsViewController.m # Riot/Modules/Room/TimelineCells/RoomCreationIntro/RoomCreationIntroCell.swift # Riot/Modules/Room/TimelineCells/RoomCreationIntro/RoomCreationIntroCellContentView.swift # Riot/Modules/Room/TimelineCells/RoomCreationIntro/RoomCreationIntroViewData.swift # Riot/Modules/Room/TimelineCells/RoomTimelineCellIdentifier.h # Riot/Modules/Rooms/RoomsViewController.h # Riot/Modules/Rooms/ShowDirectory/Cells/Network/DirectoryNetworkTableHeaderFooterView.swift # Riot/Modules/Rooms/ShowDirectory/Cells/Room/DirectoryRoomTableViewCell.swift # Riot/Modules/Rooms/ShowDirectory/PublicRoomsDirectoryViewModel.swift # Riot/Modules/Secrets/Recover/RecoverWithKey/SecretsRecoveryWithKeyCoordinator.swift # Riot/Modules/Secrets/Recover/RecoverWithKey/SecretsRecoveryWithKeyViewController.swift # Riot/Modules/Secrets/Recover/RecoverWithPassphrase/SecretsRecoveryWithPassphraseCoordinator.swift # Riot/Modules/Secrets/Recover/RecoverWithPassphrase/SecretsRecoveryWithPassphraseViewController.swift # Riot/Modules/Secrets/Recover/SecretsRecoveryCoordinator.swift # Riot/Modules/SecureBackup/Setup/Intro/SecureBackupSetupIntroViewController.swift # Riot/Modules/SecureBackup/Setup/Intro/SecureBackupSetupIntroViewModel.swift # Riot/Modules/SecureBackup/Setup/Intro/SecureBackupSetupIntroViewModelType.swift # Riot/Modules/SetPinCode/PinCodePreferences.swift # Riot/Modules/SetPinCode/SetupBiometrics/BiometricsAuthenticationPresenter.swift # Riot/Modules/Settings/Security/ManageSession/ManageSessionViewController.m # Riot/Modules/Settings/Security/SecurityViewController.m # Riot/Modules/Settings/SettingsViewController.m # Riot/Modules/SplitView/SplitViewCoordinator.swift # Riot/Modules/SplitView/SplitViewCoordinatorType.swift # Riot/Modules/StartChat/StartChatViewController.m # Riot/Modules/TabBar/MasterTabBarController.h # Riot/Modules/TabBar/MasterTabBarController.m # Riot/Utils/EventFormatter.m # Riot/Utils/HTMLFormatter.swift # Riot/Utils/Tools.m # RiotNSE/NotificationService.swift
241 lines
11 KiB
Swift
241 lines
11 KiB
Swift
//
|
|
// Copyright 2020-2024 New Vector Ltd.
|
|
//
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
// Please see LICENSE in the repository root for full details.
|
|
//
|
|
|
|
import UIKit
|
|
import Reusable
|
|
import ReadMoreTextView
|
|
|
|
class RoomInfoBasicView: UIView {
|
|
|
|
private enum TopicTextViewConstants {
|
|
static let font = UIFont.systemFont(ofSize: 15)
|
|
static let defaultNumberOfLines = 4
|
|
static let moreLessTextPadding = UIEdgeInsets(top: 16, left: 16, bottom: 16, right: 16)
|
|
}
|
|
|
|
@IBOutlet private weak var mainStackView: UIStackView!
|
|
@IBOutlet private weak var avatarContainerView: UIView!
|
|
@IBOutlet private weak var avatarImageView: MXKImageView!
|
|
@IBOutlet private weak var badgeImageView: UIImageView!
|
|
@IBOutlet private weak var presenceIndicatorView: PresenceIndicatorView! {
|
|
didSet {
|
|
presenceIndicatorView.delegate = self
|
|
}
|
|
}
|
|
@IBOutlet private weak var roomNameStackView: UIStackView!
|
|
@IBOutlet private weak var roomNameLabel: UILabel!
|
|
@IBOutlet private weak var roomAddressLabel: UILabel!
|
|
@IBOutlet private weak var topicContainerView: UIView!
|
|
@IBOutlet private weak var topicTitleLabel: UILabel! {
|
|
didSet {
|
|
topicTitleLabel.text = VectorL10n.roomDetailsTopic
|
|
}
|
|
}
|
|
@IBOutlet private weak var roomTopicTextView: ReadMoreTextView! {
|
|
didSet {
|
|
roomTopicTextView.contentInset = .zero
|
|
roomTopicTextView.textContainerInset = .zero
|
|
roomTopicTextView.textContainer.lineFragmentPadding = 0
|
|
roomTopicTextView.readMoreTextPadding = TopicTextViewConstants.moreLessTextPadding
|
|
roomTopicTextView.readLessTextPadding = TopicTextViewConstants.moreLessTextPadding
|
|
roomTopicTextView.shouldTrim = true
|
|
roomTopicTextView.maximumNumberOfLines = TopicTextViewConstants.defaultNumberOfLines
|
|
roomTopicTextView.onSizeChange = { _ in
|
|
self.roomTopicTextView.textAlignment = .left
|
|
self.onTopicSizeChange?(self)
|
|
}
|
|
}
|
|
}
|
|
@IBOutlet private weak var securityContainerView: UIView!
|
|
@IBOutlet private weak var securityTitleLabel: UILabel!
|
|
@IBOutlet private weak var securityInformationLabel: UILabel!
|
|
|
|
private var pillImageView: UIImageView?
|
|
private var roomDisplayName: String?
|
|
private var isRoomFederated: Bool = false
|
|
|
|
/// Block to be invoked when topic text view changes its content size.
|
|
var onTopicSizeChange: ((RoomInfoBasicView) -> Void)?
|
|
|
|
/// Force to update topic text view trimming.
|
|
func updateTrimmingOnTopic() {
|
|
roomTopicTextView.setNeedsUpdateTrim()
|
|
let currentValue = roomTopicTextView.shouldTrim
|
|
roomTopicTextView.shouldTrim = !currentValue
|
|
roomTopicTextView.shouldTrim = currentValue
|
|
roomTopicTextView.textAlignment = .left
|
|
}
|
|
|
|
func configure(withViewData viewData: RoomInfoBasicViewData) {
|
|
let avatarImage = AvatarGenerator.generateAvatar(forMatrixItem: viewData.roomId, withDisplayName: viewData.roomDisplayName)
|
|
|
|
avatarImageView.contentMode = .scaleAspectFill
|
|
|
|
// bwi: update room avatar
|
|
if let avatarUrl = viewData.avatarUrl {
|
|
if !avatarUrl.isEmpty {
|
|
avatarImageView.enableInMemoryCache = true
|
|
|
|
avatarImageView.setImageURI(avatarUrl,
|
|
withType: nil,
|
|
andImageOrientation: .up,
|
|
toFitViewSize: avatarImageView.frame.size,
|
|
with: MXThumbnailingMethodScale,
|
|
previewImage: avatarImage,
|
|
mediaManager: viewData.mediaManager)
|
|
} else {
|
|
avatarImageView.image = avatarImage
|
|
}
|
|
} else {
|
|
avatarImageView.image = avatarImage
|
|
}
|
|
|
|
if BWIBuildSettings.shared.bwiUseCustomPersonalNotesAvatar {
|
|
if let session = AppDelegate.theDelegate().mxSessions.first as? MXSession {
|
|
let service = PersonalNotesDefaultService(mxSession: session)
|
|
if let personalNotesRoomId = service.personalNotesRoomId(), personalNotesRoomId == viewData.roomId {
|
|
avatarImageView.image = UIImage(named: service.avatarImageUrl())
|
|
}
|
|
}
|
|
}
|
|
|
|
// bwi: 5216 - federation
|
|
if BWIBuildSettings.shared.isFederationEnabled && viewData.isFederated {
|
|
isRoomFederated = true
|
|
self.roomDisplayName = viewData.roomDisplayName
|
|
// add imageview for the pill if it doesn't exist
|
|
if pillImageView == nil {
|
|
let imageView = FederationIconHelper.shared.getFederatedIconImageView(avatarFrame: avatarImageView.frame)
|
|
pillImageView = imageView
|
|
pillImageView?.contentMode = .scaleAspectFit
|
|
avatarContainerView.addSubview(imageView)
|
|
}
|
|
|
|
pillImageView?.image = FederationIconHelper.shared.federationBadgeImage()
|
|
roomNameLabel.attributedText = FederationIconHelper.shared.roomNameWithFederationPill(roomDisplayName: viewData.roomDisplayName, font: roomNameLabel.font)
|
|
} else {
|
|
roomNameLabel.text = viewData.roomDisplayName
|
|
isRoomFederated = false
|
|
pillImageView?.image = nil
|
|
}
|
|
|
|
roomNameLabel.numberOfLines = 0
|
|
|
|
badgeImageView.image = nil
|
|
if BWIBuildSettings.shared.showMatrixIDinRoomInfoScreen {
|
|
roomAddressLabel.text = viewData.mainRoomAlias
|
|
roomAddressLabel.isHidden = roomAddressLabel.text?.isEmpty ?? true
|
|
} else {
|
|
roomAddressLabel.text = nil
|
|
roomAddressLabel.isHidden = true
|
|
}
|
|
roomTopicTextView.text = viewData.roomTopic
|
|
topicContainerView.isHidden = roomTopicTextView.text?.isEmpty ?? true
|
|
securityTitleLabel.text = VectorL10n.securitySettingsTitle
|
|
securityInformationLabel.text = viewData.isDirect ?
|
|
VectorL10n.roomParticipantsSecurityInformationRoomEncryptedForDm :
|
|
BWIL10n.roomParticipantsSecurityInformationRoomEncrypted
|
|
securityContainerView.isHidden = !viewData.isEncrypted
|
|
if let directUserId = viewData.directUserId {
|
|
presenceIndicatorView.configure(userId: directUserId, presence: viewData.directUserPresence)
|
|
} else {
|
|
presenceIndicatorView.stopListeningPresenceUpdates()
|
|
}
|
|
updateRoomNameLabelPosition()
|
|
updateBadgeImageViewPosition(isPresenceDisplayed: viewData.directUserPresence != .unknown)
|
|
}
|
|
|
|
|
|
// bwi
|
|
private func updateRoomNameLabelPosition() {
|
|
let margins = self.layoutMarginsGuide
|
|
roomNameLabel.leadingAnchor.constraint(equalTo: margins.leadingAnchor).isActive = true
|
|
roomNameLabel.trailingAnchor.constraint(equalTo: margins.trailingAnchor).isActive = true
|
|
}
|
|
|
|
private func updateBadgeImageViewPosition(isPresenceDisplayed: Bool) {
|
|
guard badgeImageView.image != nil else {
|
|
badgeImageView.isHidden = true
|
|
return
|
|
}
|
|
|
|
badgeImageView.isHidden = false
|
|
// Update badge position if it doesn't match expectation.
|
|
// If presence is displayed, badge should be in the name stack.
|
|
let isBadgeInRoomNameStackView = roomNameStackView.arrangedSubviews.contains(badgeImageView)
|
|
switch (isPresenceDisplayed, isBadgeInRoomNameStackView) {
|
|
case (true, false):
|
|
badgeImageView.removeFromSuperview()
|
|
roomNameStackView.insertArrangedSubview(badgeImageView, at: 0)
|
|
case (false, true):
|
|
roomNameStackView.removeArrangedSubview(badgeImageView)
|
|
avatarContainerView.addSubview(badgeImageView)
|
|
badgeImageView.trailingAnchor.constraint(equalTo: avatarContainerView.trailingAnchor).isActive = true
|
|
badgeImageView.bottomAnchor.constraint(equalTo: avatarContainerView.bottomAnchor).isActive = true
|
|
case (_, _):
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
extension RoomInfoBasicView: NibLoadable {}
|
|
|
|
extension RoomInfoBasicView: Themable {
|
|
|
|
func update(theme: Theme) {
|
|
backgroundColor = theme.headerBackgroundColor
|
|
roomNameLabel.textColor = theme.textPrimaryColor
|
|
roomAddressLabel.textColor = theme.textSecondaryColor
|
|
topicTitleLabel.textColor = theme.textSecondaryColor
|
|
roomTopicTextView.textColor = theme.textPrimaryColor
|
|
roomTopicTextView.linkTextAttributes = [
|
|
NSAttributedString.Key.font: TopicTextViewConstants.font,
|
|
NSAttributedString.Key.foregroundColor: theme.tintColor
|
|
]
|
|
let mutableReadMore = NSMutableAttributedString(string: "… ", attributes: [
|
|
NSAttributedString.Key.font: TopicTextViewConstants.font,
|
|
NSAttributedString.Key.foregroundColor: theme.textPrimaryColor
|
|
])
|
|
let attributedMore = NSAttributedString(string: VectorL10n.more, attributes: [
|
|
NSAttributedString.Key.font: TopicTextViewConstants.font,
|
|
NSAttributedString.Key.foregroundColor: theme.tintColor
|
|
])
|
|
mutableReadMore.append(attributedMore)
|
|
roomTopicTextView.attributedReadMoreText = mutableReadMore
|
|
|
|
let mutableReadLess = NSMutableAttributedString(string: " ")
|
|
let attributedLess = NSAttributedString(string: VectorL10n.less, attributes: [
|
|
NSAttributedString.Key.font: TopicTextViewConstants.font,
|
|
NSAttributedString.Key.foregroundColor: theme.tintColor
|
|
])
|
|
mutableReadLess.append(attributedLess)
|
|
roomTopicTextView.attributedReadLessText = mutableReadLess
|
|
|
|
securityTitleLabel.textColor = theme.textSecondaryColor
|
|
securityInformationLabel.textColor = theme.textPrimaryColor
|
|
presenceIndicatorView.borderColor = theme.headerBackgroundColor
|
|
|
|
// bwi: 5216 - federation
|
|
if BWIBuildSettings.shared.isFederationEnabled && isRoomFederated {
|
|
if pillImageView != nil {
|
|
pillImageView?.image = FederationIconHelper.shared.federationBadgeImage()
|
|
}
|
|
if let roomDisplayName = self.roomDisplayName {
|
|
roomNameLabel.attributedText = FederationIconHelper.shared.roomNameWithFederationPill(roomDisplayName: roomDisplayName, font: roomNameLabel.font)
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
extension RoomInfoBasicView: PresenceIndicatorViewDelegate {
|
|
func presenceIndicatorViewDidUpdateVisibility(_ presenceIndicatorView: PresenceIndicatorView, isHidden: Bool) {
|
|
updateRoomNameLabelPosition()
|
|
updateBadgeImageViewPosition(isPresenceDisplayed: !isHidden)
|
|
}
|
|
}
|