mirror of
https://gitlab.opencode.de/bwi/bundesmessenger/clients/bundesmessenger-ios.git
synced 2026-04-17 15:09:31 +02:00
Merge branch 'feature/5304_us14_add_federation_info_view_for_old_rooms' into 'develop'
MESSENGER-5304 add room federation decision sheet See merge request bwmessenger/bundesmessenger/bundesmessenger-ios!295
This commit is contained in:
@@ -59,8 +59,8 @@
|
||||
"kind" : "remoteSourceControl",
|
||||
"location" : "https://github.com/matrix-org/matrix-wysiwyg-composer-swift",
|
||||
"state" : {
|
||||
"revision" : "dfb74c89bf54b41ea000d564d6435ac6444ba6b4",
|
||||
"version" : "2.18.0"
|
||||
"revision" : "0aa1308c43451fd077e332f72d6a32135f258834",
|
||||
"version" : "2.19.0"
|
||||
}
|
||||
},
|
||||
{
|
||||
|
||||
@@ -673,7 +673,7 @@
|
||||
"bwi_room_settings_federation_alert_title" = "Achtung!";
|
||||
"bwi_room_settings_federation_alert_message" = "Durch das Aufheben der Föderation werden alle Mitglieder der anderen Organisation unwiderruflich entfernt.\n\nFöderation trotzdem aufheben?";
|
||||
"bwi_room_settings_federation_alert_withdraw_button" = "Ja, aufheben";
|
||||
"room_details_failed_to_update_room_server_acl_rule" = "Aktualisierung der Föderations Einstellung fehlgeschlagen";
|
||||
"room_details_failed_to_update_room_server_acl_rule" = "Aktualisierung der Föderations-Einstellung fehlgeschlagen";
|
||||
"room_details_failed_to_change_federation_for_room_error_title" = "Föderation aktiv";
|
||||
"room_details_failed_to_change_federation_for_room_error_text" = "Die Föderation konnte nicht geändert werden, bitte versuche es später erneut.";
|
||||
"room_details_failed_to_change_federation_alert_dismiss_button" = "Ok";
|
||||
@@ -684,6 +684,15 @@
|
||||
"create_room_failed_to_deactivate_federation_for_room_error_text" = "Die Föderation konnte nicht deaktiviert werden, bitte versuche es später erneut.";
|
||||
"create_room_failed_to_deactivate_federation_alert_dismiss_button" = "Ok";
|
||||
|
||||
"room_admin_federation_decision_sheet_title" = "\"%@\" für eine Föderation zulassen?";
|
||||
"room_admin_federation_decision_sheet_text" = "Hierdurch kann der Raum von externen Organisationen mitgenutzt werden. Dies kann nachträglich in den Einstellungen geändert werden.";
|
||||
"room_admin_federation_decision_sheet_remind_later_button" = "Später erinnern";
|
||||
"room_admin_federation_decision_sheet_activate_federation_button" = "Raum föderieren";
|
||||
"room_admin_federation_decision_sheet_deactivate_federation_button" = "Raum intern behalten";
|
||||
|
||||
"room_admin_federation_decision_set_federation_error_alert_title" = "Aktualisierung fehlgeschlagen";
|
||||
"room_admin_federation_decision_set_federation_error_alert_text" = "Aktualisierung der Föderations-Einstellung fehlgeschlagen, bitte versuche es später erneut.";
|
||||
|
||||
"room_participants_invite_prompt_federation_for_room_not_allowed_text" = "Du kannst diese Person nicht einladen, da die Föderation für diesen Raum durch den Admin nicht gewünscht ist.";
|
||||
"room_participants_invite_prompt_server_acl_for_room_not_configured_text" = "Du kannst noch keine Personen aus einer föderierten Organisation einladen, da die Freigabe hierfür noch nicht erteilt wurde. Gib dem Admin Bescheid, dass die Einstellung getroffen werden muss.";
|
||||
|
||||
|
||||
@@ -593,6 +593,15 @@
|
||||
"create_room_failed_to_deactivate_federation_for_room_error_text" = "The federation could not be disabled, please try again later.";
|
||||
"create_room_failed_to_deactivate_federation_alert_dismiss_button" = "Ok";
|
||||
|
||||
"room_admin_federation_decision_sheet_title" = "Approve \"%@\" for federation?";
|
||||
"room_admin_federation_decision_sheet_text" = "This will enable external organizations to access the room. It can be reversed in the room settings afterwards.";
|
||||
"room_admin_federation_decision_sheet_remind_later_button" = "Remind later";
|
||||
"room_admin_federation_decision_sheet_activate_federation_button" = "Federrate room";
|
||||
"room_admin_federation_decision_sheet_deactivate_federation_button" = "Keep room internal";
|
||||
|
||||
"room_admin_federation_decision_set_federation_error_alert_title" = "Fail to update";
|
||||
"room_admin_federation_decision_set_federation_error_alert_text" = "Fail to update the federation settings, please try again later.";
|
||||
|
||||
"room_participants_invite_prompt_federation_for_room_not_allowed_text" = "You cannot invite this user because the federation for this room is not desired by the admin";
|
||||
"room_participants_invite_prompt_server_acl_for_room_not_configured_text" = "You cannot yet invite people from a federated organization, as this has not yet been approved. Let the admin know that the setting needs to be made.";
|
||||
|
||||
|
||||
@@ -1127,6 +1127,34 @@ public class BWIL10n: NSObject {
|
||||
public static var retry: String {
|
||||
return BWIL10n.tr("Bwi", "retry")
|
||||
}
|
||||
/// Aktualisierung der Föderations-Einstellung fehlgeschlagen, bitte versuche es später erneut.
|
||||
public static var roomAdminFederationDecisionSetFederationErrorAlertText: String {
|
||||
return BWIL10n.tr("Bwi", "room_admin_federation_decision_set_federation_error_alert_text")
|
||||
}
|
||||
/// Aktualisierung fehlgeschlagen
|
||||
public static var roomAdminFederationDecisionSetFederationErrorAlertTitle: String {
|
||||
return BWIL10n.tr("Bwi", "room_admin_federation_decision_set_federation_error_alert_title")
|
||||
}
|
||||
/// Raum föderieren
|
||||
public static var roomAdminFederationDecisionSheetActivateFederationButton: String {
|
||||
return BWIL10n.tr("Bwi", "room_admin_federation_decision_sheet_activate_federation_button")
|
||||
}
|
||||
/// Raum intern behalten
|
||||
public static var roomAdminFederationDecisionSheetDeactivateFederationButton: String {
|
||||
return BWIL10n.tr("Bwi", "room_admin_federation_decision_sheet_deactivate_federation_button")
|
||||
}
|
||||
/// Später erinnern
|
||||
public static var roomAdminFederationDecisionSheetRemindLaterButton: String {
|
||||
return BWIL10n.tr("Bwi", "room_admin_federation_decision_sheet_remind_later_button")
|
||||
}
|
||||
/// Hierdurch kann der Raum von externen Organisationen mitgenutzt werden. Dies kann nachträglich in den Einstellungen geändert werden.
|
||||
public static var roomAdminFederationDecisionSheetText: String {
|
||||
return BWIL10n.tr("Bwi", "room_admin_federation_decision_sheet_text")
|
||||
}
|
||||
/// "%@" für eine Föderation zulassen?
|
||||
public static func roomAdminFederationDecisionSheetTitle(_ p1: String) -> String {
|
||||
return BWIL10n.tr("Bwi", "room_admin_federation_decision_sheet_title", p1)
|
||||
}
|
||||
/// Raumbild ändern
|
||||
public static var roomAvatarViewAccessibilityHint: String {
|
||||
return BWIL10n.tr("Bwi", "room_avatar_view_accessibility_hint")
|
||||
@@ -1187,7 +1215,7 @@ public class BWIL10n: NSObject {
|
||||
public static var roomDetailsFailedToChangeFederationForRoomErrorTitle: String {
|
||||
return BWIL10n.tr("Bwi", "room_details_failed_to_change_federation_for_room_error_title")
|
||||
}
|
||||
/// Aktualisierung der Föderations Einstellung fehlgeschlagen
|
||||
/// Aktualisierung der Föderations-Einstellung fehlgeschlagen
|
||||
public static var roomDetailsFailedToUpdateRoomServerAclRule: String {
|
||||
return BWIL10n.tr("Bwi", "room_details_failed_to_update_room_server_acl_rule")
|
||||
}
|
||||
|
||||
@@ -242,6 +242,9 @@ static CGSize kThreadListBarButtonItemImageSize;
|
||||
// Check if we should wait for other participants
|
||||
@property (nonatomic, readonly) BOOL shouldWaitForOtherParticipants;
|
||||
|
||||
// bwi: #5304 show federation decision sheet
|
||||
@property (nonatomic) BOOL wasFederationDecisionSheetShownBefore;
|
||||
|
||||
@end
|
||||
|
||||
@implementation RoomViewController
|
||||
@@ -434,6 +437,9 @@ static CGSize kThreadListBarButtonItemImageSize;
|
||||
[self setupCompletionSuggestionViewIfNeeded];
|
||||
|
||||
[self.topBannersStackView vc_removeAllSubviews];
|
||||
|
||||
// bwi: #5304 show federation decision sheet
|
||||
self.wasFederationDecisionSheetShownBefore = false;
|
||||
}
|
||||
|
||||
- (void)userInterfaceThemeDidChange
|
||||
@@ -717,6 +723,10 @@ static CGSize kThreadListBarButtonItemImageSize;
|
||||
}
|
||||
|
||||
[self setMaximisedToolbarIsHiddenIfNeeded: NO];
|
||||
|
||||
|
||||
// bwi: #5304 show federation decision sheet only for admins and only in rooms / not in DMs
|
||||
[self showFederationDecisionSheet];
|
||||
}
|
||||
|
||||
- (void)viewDidDisappear:(BOOL)animated
|
||||
@@ -8231,6 +8241,57 @@ static CGSize kThreadListBarButtonItemImageSize;
|
||||
}
|
||||
}
|
||||
|
||||
#pragma mark - BWI Federation
|
||||
|
||||
// bwi: #5304 show federation decision sheet only for admins and only in rooms / not in DMs
|
||||
- (void)showFederationDecisionSheet {
|
||||
if (BWIBuildSettings.shared.isFederationEnabled)
|
||||
{
|
||||
// Show sheet only when opening room
|
||||
if (!self.wasFederationDecisionSheetShownBefore)
|
||||
{
|
||||
// Do not show sheet if room DM
|
||||
if (!self.roomDataSource.room.isDirect)
|
||||
{
|
||||
// Do not show sheet if room is personal notes room
|
||||
if (!self.roomDataSource.room.isPersonalNotesRoom)
|
||||
{
|
||||
// Do not show sheet if isFederated room flag is false (default == true)
|
||||
if (self.roomDataSource.room.isRoomFederated)
|
||||
{
|
||||
// Do not show sheet if users power level is lower than admin
|
||||
MXRoomPowerLevels *powerLevels = self.roomDataSource.roomState.powerLevels;
|
||||
if ([powerLevels powerLevelOfUserWithUserID:self.mainSession.myUser.userId] >= RoomPowerLevelAdmin)
|
||||
{
|
||||
// Show sheet if no serverACL have been configured
|
||||
[self.roomDataSource.room getCurrentRoomServerACLSettingsWithCompletion:^(NSString *serverACL)
|
||||
{
|
||||
if (serverACL == nil) {
|
||||
self.wasFederationDecisionSheetShownBefore = true;
|
||||
RoomFederationDecisionSheet *federationDecisionView = [[RoomFederationDecisionSheet alloc] init];
|
||||
UIImage *roomAvatarImage;
|
||||
MXKImageView *roomAvatarImageView = ((RoomTitleView*)self.titleView).pictureView;
|
||||
if (roomAvatarImageView && roomAvatarImageView.image)
|
||||
{
|
||||
roomAvatarImage = roomAvatarImageView.image;
|
||||
}
|
||||
else
|
||||
{
|
||||
roomAvatarImage = [AvatarGenerator generateAvatarForMatrixItem:self.roomDataSource.roomId withDisplayName:self.roomDataSource.room.summary.displayName];
|
||||
}
|
||||
UIViewController *sheetViewController = [federationDecisionView makeViewControllerWithRoom:self.roomDataSource.room roomAvatarImage: roomAvatarImage];
|
||||
sheetViewController.modalPresentationStyle = UIModalPresentationFormSheet;
|
||||
[self presentViewController:sheetViewController animated:YES completion:nil];
|
||||
}
|
||||
}];
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#pragma mark - UserSuggestionCoordinatorBridgeDelegate
|
||||
|
||||
- (void)completionSuggestionCoordinatorBridge:(CompletionSuggestionCoordinatorBridge *)coordinator
|
||||
|
||||
200
bwi/Federation/UI/RoomFederationDecisionView.swift
Normal file
200
bwi/Federation/UI/RoomFederationDecisionView.swift
Normal file
@@ -0,0 +1,200 @@
|
||||
//
|
||||
/*
|
||||
* Copyright (c) 2024 BWI GmbH
|
||||
*
|
||||
* 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 SwiftUI
|
||||
|
||||
/// Helper class for making our SwiftUI view available to ObjectiveC
|
||||
@objcMembers class RoomFederationDecisionSheet: NSObject {
|
||||
private var decisionView: RoomFederationDecisionView?
|
||||
@Published var theme: Theme
|
||||
|
||||
override init() {
|
||||
theme = ThemeService.shared().theme
|
||||
}
|
||||
|
||||
func makeViewController(room: MXRoom, roomAvatarImage: UIImage) -> UIViewController {
|
||||
registerThemeServiceDidChangeThemeNotification()
|
||||
self.decisionView = RoomFederationDecisionView(theme: Binding(get: {
|
||||
return self.theme
|
||||
}, set: { newTheme in
|
||||
self.theme = newTheme
|
||||
}), room: room, roomAvatarImage: roomAvatarImage)
|
||||
return UIHostingController(rootView: decisionView)
|
||||
}
|
||||
|
||||
private func registerThemeServiceDidChangeThemeNotification() {
|
||||
NotificationCenter.default.addObserver(self, selector: #selector(themeDidChange), name: .themeServiceDidChangeTheme, object: nil)
|
||||
}
|
||||
|
||||
@objc private func themeDidChange() {
|
||||
self.theme = ThemeService.shared().theme
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
// bwi: #5304
|
||||
struct RoomFederationDecisionView: View {
|
||||
@Environment(\.dismiss) var dismissView
|
||||
@State var isUpdatingServerACLs: Bool = false
|
||||
@Binding var theme: Theme
|
||||
@State var showSetServerACLErrorAlert: Bool = false
|
||||
|
||||
var room: MXRoom
|
||||
@State var pendingRequest: MXHTTPOperation?
|
||||
|
||||
var roomAvatarImage: UIImage
|
||||
let imageStackScale = 0.30
|
||||
let imageStackShift = 5.0
|
||||
let lineWith = 7.0
|
||||
let spacing = 16.0
|
||||
|
||||
|
||||
var body: some View {
|
||||
GeometryReader { geo in
|
||||
ScrollView(.vertical) {
|
||||
VStack(spacing: spacing) {
|
||||
VStack(spacing: spacing) {
|
||||
Spacer()
|
||||
// Federation image with room avatar
|
||||
ZStack(alignment: .center) {
|
||||
Circle()
|
||||
.fill(Color(theme.colors.quinaryContent))
|
||||
.overlay(Circle()
|
||||
.stroke(Color(theme.colors.background), lineWidth: lineWith))
|
||||
.offset(x:((getImageSize(width: geo.size.width, height: geo.size.height) * imageStackScale) / imageStackShift))
|
||||
Image(uiImage: roomAvatarImage)
|
||||
.resizable()
|
||||
.clipShape(Circle())
|
||||
.overlay(Circle()
|
||||
.stroke(Color(theme.colors.background), lineWidth: lineWith))
|
||||
.offset(x:-((getImageSize(width: geo.size.width, height: geo.size.height) * imageStackScale) / imageStackShift))
|
||||
}
|
||||
.frame(width: getImageSize(width: geo.size.width, height: geo.size.height) * imageStackScale, height: getImageSize(width: geo.size.width, height: geo.size.height) * imageStackScale)
|
||||
|
||||
// Federation info text
|
||||
textStack
|
||||
}
|
||||
|
||||
if isUpdatingServerACLs {
|
||||
ProgressView()
|
||||
}
|
||||
|
||||
// Federation actions
|
||||
buttonStack
|
||||
}
|
||||
.frame(minHeight: geo.size.height)
|
||||
}
|
||||
.interactiveDismissDisabled()
|
||||
.frame(width: geo.size.width, height: geo.size.height)
|
||||
.background(Color(theme.colors.background))
|
||||
.alert(isPresented: $showSetServerACLErrorAlert) {
|
||||
Alert(title: Text(BWIL10n.roomAdminFederationDecisionSetFederationErrorAlertTitle), message: Text(BWIL10n.roomAdminFederationDecisionSetFederationErrorAlertText), dismissButton: .default(Text("Ok")))
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func getImageSize(width: CGFloat, height: CGFloat) -> CGFloat {
|
||||
return (width > height ? height : width)
|
||||
}
|
||||
|
||||
|
||||
var textStack: some View {
|
||||
VStack(spacing: spacing) {
|
||||
Text(BWIL10n.roomAdminFederationDecisionSheetTitle(room.displayName ?? ""))
|
||||
.font(.system(size: 24).bold())
|
||||
.multilineTextAlignment(.center)
|
||||
.lineLimit(nil)
|
||||
.foregroundColor(Color(theme.colors.primaryContent))
|
||||
Text(BWIL10n.roomAdminFederationDecisionSheetText)
|
||||
.font(.system(size: 12))
|
||||
.multilineTextAlignment(.center)
|
||||
.lineLimit(nil)
|
||||
.foregroundColor(Color(theme.colors.secondaryContent))
|
||||
}
|
||||
.padding(spacing)
|
||||
}
|
||||
|
||||
|
||||
var buttonStack: some View {
|
||||
VStack(spacing: spacing) {
|
||||
Spacer()
|
||||
Button {
|
||||
closeView()
|
||||
} label: {
|
||||
Text(BWIL10n.roomAdminFederationDecisionSheetRemindLaterButton)
|
||||
}
|
||||
.buttonStyle(SecondaryActionButtonStyle())
|
||||
.accessibilityIdentifier("federationDecisionSheetRemindLaterButton")
|
||||
Button {
|
||||
setServerACL(isFederated: true)
|
||||
} label: {
|
||||
Text(BWIL10n.roomAdminFederationDecisionSheetActivateFederationButton)
|
||||
}
|
||||
.buttonStyle(SecondaryActionButtonStyle())
|
||||
.accessibilityIdentifier("federationDecisionSheetActivateFederationButton")
|
||||
.disabled(isUpdatingServerACLs)
|
||||
Button {
|
||||
setServerACL(isFederated: false)
|
||||
} label: {
|
||||
Text(BWIL10n.roomAdminFederationDecisionSheetDeactivateFederationButton)
|
||||
}
|
||||
.buttonStyle(PrimaryActionButtonStyle())
|
||||
.accessibilityIdentifier("federationDecisionSheetDeactivateFederationButton")
|
||||
.disabled(isUpdatingServerACLs)
|
||||
}
|
||||
.padding(spacing)
|
||||
}
|
||||
|
||||
|
||||
func setServerACL(isFederated: Bool) {
|
||||
var serverACL: [String] = [String]()
|
||||
if isFederated {
|
||||
serverACL.append("*")
|
||||
} else {
|
||||
if let myUserIDComponents = room.mxSession.myUserId?.components(separatedBy: ":")
|
||||
{
|
||||
if myUserIDComponents.count == 2 {
|
||||
serverACL.append(myUserIDComponents[1])
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
var content: [String:Any] = [String:Any]()
|
||||
content.updateValue(serverACL, forKey: "allow")
|
||||
content.updateValue(false, forKey: "allowIPLiterals")
|
||||
isUpdatingServerACLs = true
|
||||
pendingRequest = room.sendStateEvent(MXEventType.roomServerACL, content: content, stateKey: "") { response in
|
||||
isUpdatingServerACLs = false
|
||||
if response.isSuccess {
|
||||
closeView()
|
||||
}
|
||||
else {
|
||||
showSetServerACLErrorAlert = true
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
func closeView() {
|
||||
if let pendingRequest = pendingRequest {
|
||||
pendingRequest.cancel()
|
||||
}
|
||||
dismissView()
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user