You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
163 lines
4.9 KiB
Dart
163 lines
4.9 KiB
Dart
import 'package:flutter/material.dart';
|
|
|
|
import 'package:collection/collection.dart' show IterableExtension;
|
|
import 'package:flutter_gen/gen_l10n/l10n.dart';
|
|
import 'package:matrix/encryption/utils/key_verification.dart';
|
|
import 'package:matrix/matrix.dart';
|
|
|
|
import 'package:fluffychat/pages/device_settings/device_settings_view.dart';
|
|
import 'package:fluffychat/pages/key_verification/key_verification_dialog.dart';
|
|
import 'package:fluffychat/widgets/adaptive_dialogs/show_ok_cancel_alert_dialog.dart';
|
|
import 'package:fluffychat/widgets/adaptive_dialogs/show_text_input_dialog.dart';
|
|
import 'package:fluffychat/widgets/future_loading_dialog.dart';
|
|
import '../../widgets/matrix.dart';
|
|
|
|
class DevicesSettings extends StatefulWidget {
|
|
const DevicesSettings({super.key});
|
|
|
|
@override
|
|
DevicesSettingsController createState() => DevicesSettingsController();
|
|
}
|
|
|
|
class DevicesSettingsController extends State<DevicesSettings> {
|
|
List<Device>? devices;
|
|
Future<bool> loadUserDevices(BuildContext context) async {
|
|
if (devices != null) return true;
|
|
devices = await Matrix.of(context).client.getDevices();
|
|
return true;
|
|
}
|
|
|
|
void reload() => setState(() => devices = null);
|
|
|
|
bool? chatBackupEnabled;
|
|
|
|
@override
|
|
void initState() {
|
|
_checkChatBackup();
|
|
super.initState();
|
|
}
|
|
|
|
void _checkChatBackup() async {
|
|
final client = Matrix.of(context).client;
|
|
if (client.encryption?.keyManager.enabled == true) {
|
|
if (await client.encryption?.keyManager.isCached() == false ||
|
|
await client.encryption?.crossSigning.isCached() == false ||
|
|
client.isUnknownSession && !mounted) {
|
|
setState(() {
|
|
chatBackupEnabled = false;
|
|
});
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
void removeDevicesAction(List<Device> devices) async {
|
|
if (await showOkCancelAlertDialog(
|
|
context: context,
|
|
title: L10n.of(context).areYouSure,
|
|
okLabel: L10n.of(context).remove,
|
|
cancelLabel: L10n.of(context).cancel,
|
|
message: L10n.of(context).removeDevicesDescription,
|
|
isDestructive: true,
|
|
) ==
|
|
OkCancelResult.cancel) {
|
|
return;
|
|
}
|
|
final matrix = Matrix.of(context);
|
|
final deviceIds = <String>[];
|
|
for (final userDevice in devices) {
|
|
deviceIds.add(userDevice.deviceId);
|
|
}
|
|
|
|
await showFutureLoadingDialog(
|
|
context: context,
|
|
delay: false,
|
|
future: () => matrix.client.uiaRequestBackground(
|
|
(auth) => matrix.client.deleteDevices(
|
|
deviceIds,
|
|
auth: auth,
|
|
),
|
|
),
|
|
);
|
|
reload();
|
|
}
|
|
|
|
void renameDeviceAction(Device device) async {
|
|
final displayName = await showTextInputDialog(
|
|
context: context,
|
|
title: L10n.of(context).changeDeviceName,
|
|
okLabel: L10n.of(context).ok,
|
|
cancelLabel: L10n.of(context).cancel,
|
|
hintText: device.displayName,
|
|
);
|
|
if (displayName == null) return;
|
|
final success = await showFutureLoadingDialog(
|
|
context: context,
|
|
future: () => Matrix.of(context)
|
|
.client
|
|
.updateDevice(device.deviceId, displayName: displayName),
|
|
);
|
|
if (success.error == null) {
|
|
reload();
|
|
}
|
|
}
|
|
|
|
void verifyDeviceAction(Device device) async {
|
|
final consent = await showOkCancelAlertDialog(
|
|
context: context,
|
|
title: L10n.of(context).verifyOtherDevice,
|
|
message: L10n.of(context).verifyOtherDeviceDescription,
|
|
okLabel: L10n.of(context).ok,
|
|
cancelLabel: L10n.of(context).cancel,
|
|
);
|
|
if (consent != OkCancelResult.ok) return;
|
|
final req = await Matrix.of(context)
|
|
.client
|
|
.userDeviceKeys[Matrix.of(context).client.userID!]!
|
|
.deviceKeys[device.deviceId]!
|
|
.startVerification();
|
|
req.onUpdate = () {
|
|
if ({KeyVerificationState.error, KeyVerificationState.done}
|
|
.contains(req.state)) {
|
|
setState(() {});
|
|
}
|
|
};
|
|
await KeyVerificationDialog(request: req).show(context);
|
|
}
|
|
|
|
void blockDeviceAction(Device device) async {
|
|
final key = Matrix.of(context)
|
|
.client
|
|
.userDeviceKeys[Matrix.of(context).client.userID!]!
|
|
.deviceKeys[device.deviceId]!;
|
|
if (key.directVerified) {
|
|
await key.setVerified(false);
|
|
}
|
|
await key.setBlocked(true);
|
|
setState(() {});
|
|
}
|
|
|
|
void unblockDeviceAction(Device device) async {
|
|
final key = Matrix.of(context)
|
|
.client
|
|
.userDeviceKeys[Matrix.of(context).client.userID!]!
|
|
.deviceKeys[device.deviceId]!;
|
|
await key.setBlocked(false);
|
|
setState(() {});
|
|
}
|
|
|
|
bool _isOwnDevice(Device userDevice) =>
|
|
userDevice.deviceId == Matrix.of(context).client.deviceID;
|
|
|
|
Device? get thisDevice => devices!.firstWhereOrNull(
|
|
_isOwnDevice,
|
|
);
|
|
|
|
List<Device> get notThisDevice => List<Device>.from(devices!)
|
|
..removeWhere(_isOwnDevice)
|
|
..sort((a, b) => (b.lastSeenTs ?? 0).compareTo(a.lastSeenTs ?? 0));
|
|
|
|
@override
|
|
Widget build(BuildContext context) => DevicesSettingsView(this);
|
|
}
|