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.
		
		
		
		
		
			
		
			
				
	
	
		
			213 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			Dart
		
	
			
		
		
	
	
			213 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			Dart
		
	
import 'dart:async';
 | 
						|
 | 
						|
import 'package:flutter/material.dart';
 | 
						|
 | 
						|
import 'package:collection/collection.dart';
 | 
						|
import 'package:image_picker/image_picker.dart';
 | 
						|
import 'package:matrix/matrix.dart';
 | 
						|
 | 
						|
import 'package:fluffychat/l10n/l10n.dart';
 | 
						|
import 'package:fluffychat/utils/file_selector.dart';
 | 
						|
import 'package:fluffychat/utils/platform_infos.dart';
 | 
						|
import 'package:fluffychat/widgets/adaptive_dialogs/show_modal_action_popup.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';
 | 
						|
import '../bootstrap/bootstrap_dialog.dart';
 | 
						|
import 'settings_view.dart';
 | 
						|
 | 
						|
class Settings extends StatefulWidget {
 | 
						|
  const Settings({super.key});
 | 
						|
 | 
						|
  @override
 | 
						|
  SettingsController createState() => SettingsController();
 | 
						|
}
 | 
						|
 | 
						|
class SettingsController extends State<Settings> {
 | 
						|
  Future<Profile>? profileFuture;
 | 
						|
  bool profileUpdated = false;
 | 
						|
 | 
						|
  void updateProfile() => setState(() {
 | 
						|
        profileUpdated = true;
 | 
						|
        profileFuture = null;
 | 
						|
      });
 | 
						|
 | 
						|
  void setDisplaynameAction() async {
 | 
						|
    final profile = await profileFuture;
 | 
						|
    final input = await showTextInputDialog(
 | 
						|
      useRootNavigator: false,
 | 
						|
      context: context,
 | 
						|
      title: L10n.of(context).editDisplayname,
 | 
						|
      okLabel: L10n.of(context).ok,
 | 
						|
      cancelLabel: L10n.of(context).cancel,
 | 
						|
      initialText:
 | 
						|
          profile?.displayName ?? Matrix.of(context).client.userID!.localpart,
 | 
						|
    );
 | 
						|
    if (input == null) return;
 | 
						|
    final matrix = Matrix.of(context);
 | 
						|
    final success = await showFutureLoadingDialog(
 | 
						|
      context: context,
 | 
						|
      future: () => matrix.client.setDisplayName(matrix.client.userID!, input),
 | 
						|
    );
 | 
						|
    if (success.error == null) {
 | 
						|
      updateProfile();
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  void logoutAction() async {
 | 
						|
    final noBackup = showChatBackupBanner == true;
 | 
						|
    if (await showOkCancelAlertDialog(
 | 
						|
          useRootNavigator: false,
 | 
						|
          context: context,
 | 
						|
          title: L10n.of(context).areYouSureYouWantToLogout,
 | 
						|
          message: L10n.of(context).noBackupWarning,
 | 
						|
          isDestructive: noBackup,
 | 
						|
          okLabel: L10n.of(context).logout,
 | 
						|
          cancelLabel: L10n.of(context).cancel,
 | 
						|
        ) ==
 | 
						|
        OkCancelResult.cancel) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    final matrix = Matrix.of(context);
 | 
						|
    await showFutureLoadingDialog(
 | 
						|
      context: context,
 | 
						|
      future: () => matrix.client.logout(),
 | 
						|
    );
 | 
						|
  }
 | 
						|
 | 
						|
  void setAvatarAction() async {
 | 
						|
    final profile = await profileFuture;
 | 
						|
    final actions = [
 | 
						|
      if (PlatformInfos.isMobile)
 | 
						|
        AdaptiveModalAction(
 | 
						|
          value: AvatarAction.camera,
 | 
						|
          label: L10n.of(context).openCamera,
 | 
						|
          isDefaultAction: true,
 | 
						|
          icon: const Icon(Icons.camera_alt_outlined),
 | 
						|
        ),
 | 
						|
      AdaptiveModalAction(
 | 
						|
        value: AvatarAction.file,
 | 
						|
        label: L10n.of(context).openGallery,
 | 
						|
        icon: const Icon(Icons.photo_outlined),
 | 
						|
      ),
 | 
						|
      if (profile?.avatarUrl != null)
 | 
						|
        AdaptiveModalAction(
 | 
						|
          value: AvatarAction.remove,
 | 
						|
          label: L10n.of(context).removeYourAvatar,
 | 
						|
          isDestructive: true,
 | 
						|
          icon: const Icon(Icons.delete_outlined),
 | 
						|
        ),
 | 
						|
    ];
 | 
						|
    final action = actions.length == 1
 | 
						|
        ? actions.single.value
 | 
						|
        : await showModalActionPopup<AvatarAction>(
 | 
						|
            context: context,
 | 
						|
            title: L10n.of(context).changeYourAvatar,
 | 
						|
            cancelLabel: L10n.of(context).cancel,
 | 
						|
            actions: actions,
 | 
						|
          );
 | 
						|
    if (action == null) return;
 | 
						|
    final matrix = Matrix.of(context);
 | 
						|
    if (action == AvatarAction.remove) {
 | 
						|
      final success = await showFutureLoadingDialog(
 | 
						|
        context: context,
 | 
						|
        future: () => matrix.client.setAvatar(null),
 | 
						|
      );
 | 
						|
      if (success.error == null) {
 | 
						|
        updateProfile();
 | 
						|
      }
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    MatrixFile file;
 | 
						|
    if (PlatformInfos.isMobile) {
 | 
						|
      final result = await ImagePicker().pickImage(
 | 
						|
        source: action == AvatarAction.camera
 | 
						|
            ? ImageSource.camera
 | 
						|
            : ImageSource.gallery,
 | 
						|
        imageQuality: 50,
 | 
						|
      );
 | 
						|
      if (result == null) return;
 | 
						|
      file = MatrixFile(
 | 
						|
        bytes: await result.readAsBytes(),
 | 
						|
        name: result.path,
 | 
						|
      );
 | 
						|
    } else {
 | 
						|
      final result = await selectFiles(
 | 
						|
        context,
 | 
						|
        type: FileSelectorType.images,
 | 
						|
      );
 | 
						|
      final pickedFile = result.firstOrNull;
 | 
						|
      if (pickedFile == null) return;
 | 
						|
      file = MatrixFile(
 | 
						|
        bytes: await pickedFile.readAsBytes(),
 | 
						|
        name: pickedFile.name,
 | 
						|
      );
 | 
						|
    }
 | 
						|
    final success = await showFutureLoadingDialog(
 | 
						|
      context: context,
 | 
						|
      future: () => matrix.client.setAvatar(file),
 | 
						|
    );
 | 
						|
    if (success.error == null) {
 | 
						|
      updateProfile();
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  @override
 | 
						|
  void initState() {
 | 
						|
    WidgetsBinding.instance.addPostFrameCallback((_) => checkBootstrap());
 | 
						|
 | 
						|
    super.initState();
 | 
						|
  }
 | 
						|
 | 
						|
  void checkBootstrap() async {
 | 
						|
    final client = Matrix.of(context).client;
 | 
						|
    if (!client.encryptionEnabled) return;
 | 
						|
    await client.accountDataLoading;
 | 
						|
    await client.userDeviceKeysLoading;
 | 
						|
    if (client.prevBatch == null) {
 | 
						|
      await client.onSync.stream.first;
 | 
						|
    }
 | 
						|
    final crossSigning =
 | 
						|
        await client.encryption?.crossSigning.isCached() ?? false;
 | 
						|
    final needsBootstrap =
 | 
						|
        await client.encryption?.keyManager.isCached() == false ||
 | 
						|
            client.encryption?.crossSigning.enabled == false ||
 | 
						|
            crossSigning == false;
 | 
						|
    final isUnknownSession = client.isUnknownSession;
 | 
						|
    setState(() {
 | 
						|
      showChatBackupBanner = needsBootstrap || isUnknownSession;
 | 
						|
    });
 | 
						|
  }
 | 
						|
 | 
						|
  bool? crossSigningCached;
 | 
						|
  bool? showChatBackupBanner;
 | 
						|
 | 
						|
  void firstRunBootstrapAction([_]) async {
 | 
						|
    if (showChatBackupBanner != true) {
 | 
						|
      showOkAlertDialog(
 | 
						|
        context: context,
 | 
						|
        title: L10n.of(context).chatBackup,
 | 
						|
        message: L10n.of(context).onlineKeyBackupEnabled,
 | 
						|
        okLabel: L10n.of(context).close,
 | 
						|
      );
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    await BootstrapDialog(
 | 
						|
      client: Matrix.of(context).client,
 | 
						|
    ).show(context);
 | 
						|
    checkBootstrap();
 | 
						|
  }
 | 
						|
 | 
						|
  @override
 | 
						|
  Widget build(BuildContext context) {
 | 
						|
    final client = Matrix.of(context).client;
 | 
						|
    profileFuture ??= client.getProfileFromUserId(
 | 
						|
      client.userID!,
 | 
						|
    );
 | 
						|
    return SettingsView(this);
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
enum AvatarAction { camera, file, remove }
 |