chore: Follow up manage room aliases

This commit is contained in:
krille-chan 2024-04-15 08:49:25 +02:00
parent 0684bb63ac
commit 4dbe85b278
No known key found for this signature in database
3 changed files with 234 additions and 130 deletions

View file

@ -2509,6 +2509,8 @@
"@passwordIsWrong": {}, "@passwordIsWrong": {},
"publicLink": "Public link", "publicLink": "Public link",
"@publicLink": {}, "@publicLink": {},
"publicLinks": "Public links",
"createNewLink": "Create new link",
"joinSpace": "Join space", "joinSpace": "Join space",
"@joinSpace": {}, "@joinSpace": {},
"publicSpaces": "Public spaces", "publicSpaces": "Public spaces",

View file

@ -152,31 +152,76 @@ class ChatAccessSettingsController extends State<ChatAccessSettings> {
); );
} }
void setCanonicalAlias() async { Future<void> addAlias() async {
final domain = room.client.userID?.domain;
if (domain == null) {
throw Exception('userID or domain is null! This should never happen.');
}
final input = await showTextInputDialog( final input = await showTextInputDialog(
context: context, context: context,
title: L10n.of(context)!.editRoomAliases, title: L10n.of(context)!.editRoomAliases,
cancelLabel: L10n.of(context)!.cancel,
okLabel: L10n.of(context)!.ok,
textFields: [ textFields: [
DialogTextField( DialogTextField(
prefixText: '#', prefixText: '#',
suffixText: room.client.userID!.domain!, suffixText: domain,
initialText: room.canonicalAlias.localpart, hintText: L10n.of(context)!.alias,
), ),
], ],
); );
final newAliasLocalpart = input?.singleOrNull?.trim(); final aliasLocalpart = input?.singleOrNull?.trim();
if (newAliasLocalpart == null || newAliasLocalpart.isEmpty) return; if (aliasLocalpart == null || aliasLocalpart.isEmpty) return;
final alias = '#$aliasLocalpart:$domain';
final result = await showFutureLoadingDialog(
context: context,
future: () => room.client.setRoomAlias(alias, room.id),
);
if (result.error != null) return;
final canonicalAliasConsent = await showOkCancelAlertDialog(
context: context,
title: L10n.of(context)!.setAsCanonicalAlias,
message: alias,
okLabel: L10n.of(context)!.yes,
cancelLabel: L10n.of(context)!.no,
);
final altAliases = room
.getState(EventTypes.RoomCanonicalAlias)
?.content
.tryGetList<String>('alt_aliases')
?.toSet() ??
{};
if (room.canonicalAlias.isNotEmpty) altAliases.add(room.canonicalAlias);
altAliases.add(alias);
if (canonicalAliasConsent == OkCancelResult.ok) {
altAliases.remove(alias);
} else {
altAliases.remove(room.canonicalAlias);
}
await showFutureLoadingDialog( await showFutureLoadingDialog(
context: context, context: context,
future: () => room.setCanonicalAlias( future: () => room.client.setRoomStateWithKey(
'#$newAliasLocalpart:${room.client.userID!.domain!}', room.id,
EventTypes.RoomCanonicalAlias,
'',
{
'alias': canonicalAliasConsent == OkCancelResult.ok
? alias
: room.canonicalAlias,
if (altAliases.isNotEmpty) 'alt_aliases': altAliases.toList(),
},
), ),
); );
} }
void deleteAlias(String alias) => showFutureLoadingDialog(
context: context,
future: () => room.client.deleteRoomAlias(alias),
);
void setChatVisibilityOnDirectory(bool? visibility) async { void setChatVisibilityOnDirectory(bool? visibility) async {
if (visibility == null) return; if (visibility == null) return;
setState(() { setState(() {

View file

@ -23,157 +23,214 @@ class ChatAccessSettingsPageView extends StatelessWidget {
body: MaxWidthBody( body: MaxWidthBody(
child: StreamBuilder<Object>( child: StreamBuilder<Object>(
stream: room.onUpdate.stream, stream: room.onUpdate.stream,
builder: (context, snapshot) => Column( builder: (context, snapshot) {
mainAxisSize: MainAxisSize.min, final canonicalAlias = room.canonicalAlias;
children: [ final altAliases = room
ListTile( .getState(EventTypes.RoomCanonicalAlias)
title: Text( ?.content
L10n.of(context)!.visibilityOfTheChatHistory, .tryGetList<String>('alt_aliases') ??
style: TextStyle( [];
color: Theme.of(context).colorScheme.secondary, return Column(
fontWeight: FontWeight.bold, mainAxisSize: MainAxisSize.min,
), children: [
),
),
for (final historyVisibility in HistoryVisibility.values)
RadioListTile<HistoryVisibility>.adaptive(
title: Text(
historyVisibility
.getLocalizedString(MatrixLocals(L10n.of(context)!)),
),
value: historyVisibility,
groupValue: room.historyVisibility,
onChanged: controller.historyVisibilityLoading ||
!room.canChangeHistoryVisibility
? null
: controller.setHistoryVisibility,
),
Divider(color: Theme.of(context).dividerColor),
ListTile(
title: Text(
L10n.of(context)!.whoIsAllowedToJoinThisGroup,
style: TextStyle(
color: Theme.of(context).colorScheme.secondary,
fontWeight: FontWeight.bold,
),
),
),
for (final joinRule
in JoinRules.values..remove(JoinRules.private))
RadioListTile<JoinRules>.adaptive(
title: Text(
joinRule.localizedString(L10n.of(context)!),
),
value: joinRule,
groupValue: room.joinRules,
onChanged:
controller.joinRulesLoading || !room.canChangeJoinRules
? null
: controller.setJoinRule,
),
Divider(color: Theme.of(context).dividerColor),
if ({JoinRules.public, JoinRules.knock}
.contains(room.joinRules)) ...[
ListTile( ListTile(
title: Text( title: Text(
L10n.of(context)!.areGuestsAllowedToJoin, L10n.of(context)!.visibilityOfTheChatHistory,
style: TextStyle( style: TextStyle(
color: Theme.of(context).colorScheme.secondary, color: Theme.of(context).colorScheme.secondary,
fontWeight: FontWeight.bold, fontWeight: FontWeight.bold,
), ),
), ),
), ),
for (final guestAccess in GuestAccess.values) for (final historyVisibility in HistoryVisibility.values)
RadioListTile<GuestAccess>.adaptive( RadioListTile<HistoryVisibility>.adaptive(
title: Text( title: Text(
guestAccess historyVisibility
.getLocalizedString(MatrixLocals(L10n.of(context)!)), .getLocalizedString(MatrixLocals(L10n.of(context)!)),
), ),
value: guestAccess, value: historyVisibility,
groupValue: room.guestAccess, groupValue: room.historyVisibility,
onChanged: controller.guestAccessLoading || onChanged: controller.historyVisibilityLoading ||
!room.canChangeGuestAccess !room.canChangeHistoryVisibility
? null ? null
: controller.setGuestAccess, : controller.setHistoryVisibility,
), ),
Divider(color: Theme.of(context).dividerColor), Divider(color: Theme.of(context).dividerColor),
FutureBuilder( ListTile(
future: room.client.getRoomVisibilityOnDirectory(room.id), title: Text(
builder: (context, snapshot) => SwitchListTile.adaptive( L10n.of(context)!.whoIsAllowedToJoinThisGroup,
value: snapshot.data == Visibility.public, style: TextStyle(
color: Theme.of(context).colorScheme.secondary,
fontWeight: FontWeight.bold,
),
),
),
for (final joinRule in JoinRules.values)
if (joinRule != JoinRules.private)
RadioListTile<JoinRules>.adaptive(
title: Text(
joinRule.localizedString(L10n.of(context)!),
),
value: joinRule,
groupValue: room.joinRules,
onChanged: controller.joinRulesLoading ||
!room.canChangeJoinRules
? null
: controller.setJoinRule,
),
Divider(color: Theme.of(context).dividerColor),
if ({JoinRules.public, JoinRules.knock}
.contains(room.joinRules)) ...[
ListTile(
title: Text( title: Text(
L10n.of(context)!.chatCanBeDiscoveredViaSearchOnServer( L10n.of(context)!.areGuestsAllowedToJoin,
room.client.userID!.domain!, style: TextStyle(
color: Theme.of(context).colorScheme.secondary,
fontWeight: FontWeight.bold,
), ),
), ),
onChanged: controller.setChatVisibilityOnDirectory, ),
for (final guestAccess in GuestAccess.values)
RadioListTile<GuestAccess>.adaptive(
title: Text(
guestAccess.getLocalizedString(
MatrixLocals(L10n.of(context)!),
),
),
value: guestAccess,
groupValue: room.guestAccess,
onChanged: controller.guestAccessLoading ||
!room.canChangeGuestAccess
? null
: controller.setGuestAccess,
),
Divider(color: Theme.of(context).dividerColor),
ListTile(
title: Text(
L10n.of(context)!.publicLinks,
style: TextStyle(
color: Theme.of(context).colorScheme.secondary,
fontWeight: FontWeight.bold,
),
),
trailing: IconButton(
icon: const Icon(Icons.add_outlined),
tooltip: L10n.of(context)!.createNewLink,
onPressed: controller.addAlias,
),
),
if (canonicalAlias.isNotEmpty)
_AliasListTile(
alias: canonicalAlias,
onDelete: room.canChangeStateEvent(
EventTypes.RoomCanonicalAlias,
)
? () => controller.deleteAlias(canonicalAlias)
: null,
isCanonicalAlias: true,
),
for (final alias in altAliases)
_AliasListTile(
alias: alias,
onDelete: room.canChangeStateEvent(
EventTypes.RoomCanonicalAlias,
)
? () => controller.deleteAlias(alias)
: null,
),
Divider(color: Theme.of(context).dividerColor),
FutureBuilder(
future: room.client.getRoomVisibilityOnDirectory(room.id),
builder: (context, snapshot) => SwitchListTile.adaptive(
value: snapshot.data == Visibility.public,
title: Text(
L10n.of(context)!.chatCanBeDiscoveredViaSearchOnServer(
room.client.userID!.domain!,
),
),
onChanged: controller.setChatVisibilityOnDirectory,
),
),
],
ListTile(
title: Text(L10n.of(context)!.globalChatId),
subtitle: SelectableText(room.id),
trailing: IconButton(
icon: const Icon(Icons.copy_outlined),
onPressed: () => FluffyShare.share(room.id, context),
), ),
), ),
ListTile( ListTile(
title: Text(L10n.of(context)!.publicLink), title: Text(L10n.of(context)!.roomVersion),
subtitle: room.canonicalAlias.isEmpty subtitle: SelectableText(
? Text( room
L10n.of(context)!.noPublicLinkHasBeenCreatedYet, .getState(EventTypes.RoomCreate)!
style: const TextStyle( .content
fontStyle: FontStyle.italic, .tryGet<String>('room_version') ??
), 'Unknown',
),
trailing: room.canSendEvent(EventTypes.RoomTombstone)
? IconButton(
icon: const Icon(Icons.upgrade_outlined),
onPressed: controller.updateRoomAction,
) )
: Text(
'https://matrix.to/#/${room.canonicalAlias}',
style: TextStyle(
decoration: TextDecoration.underline,
color: Theme.of(context).colorScheme.primary,
),
),
onTap: room.canChangeStateEvent(EventTypes.RoomCanonicalAlias)
? controller.setCanonicalAlias
: null, : null,
trailing: room.canonicalAlias.isEmpty
? const Padding(
padding: EdgeInsets.symmetric(horizontal: 8.0),
child: Icon(Icons.add),
)
: IconButton(
icon: Icon(Icons.adaptive.share_outlined),
onPressed: () => FluffyShare.share(
'https://matrix.to/#/${room.canonicalAlias}',
context,
),
),
), ),
], ],
ListTile( );
title: Text(L10n.of(context)!.globalChatId), },
subtitle: SelectableText(room.id),
trailing: IconButton(
icon: const Icon(Icons.copy_outlined),
onPressed: () => FluffyShare.share(room.id, context),
),
),
ListTile(
title: Text(L10n.of(context)!.roomVersion),
subtitle: SelectableText(
room
.getState(EventTypes.RoomCreate)!
.content
.tryGet<String>('room_version') ??
'Unknown',
),
trailing: room.canSendEvent(EventTypes.RoomTombstone)
? IconButton(
icon: const Icon(Icons.upgrade_outlined),
onPressed: controller.updateRoomAction,
)
: null,
),
],
),
), ),
), ),
); );
} }
} }
class _AliasListTile extends StatelessWidget {
const _AliasListTile({
required this.alias,
required this.onDelete,
this.isCanonicalAlias = false,
});
final String alias;
final void Function()? onDelete;
final bool isCanonicalAlias;
@override
Widget build(BuildContext context) {
return ListTile(
title: Row(
children: [
TextButton.icon(
onPressed: () => FluffyShare.share(
'https://matrix.to/#/$alias',
context,
),
icon: isCanonicalAlias
? const Icon(Icons.star)
: const Icon(Icons.link_outlined),
label: SelectableText(
'https://matrix.to/#/$alias',
style: TextStyle(
decoration: TextDecoration.underline,
decorationColor: Theme.of(context).colorScheme.primary,
color: Theme.of(context).colorScheme.primary,
),
),
),
const Spacer(),
],
),
trailing: onDelete != null
? IconButton(
icon: const Icon(Icons.delete_outlined),
onPressed: onDelete,
)
: null,
);
}
}
extension JoinRulesDisplayString on JoinRules { extension JoinRulesDisplayString on JoinRules {
String localizedString(L10n l10n) { String localizedString(L10n l10n) {
switch (this) { switch (this) {