mirror of
				https://gitlab.com/famedly/fluffychat.git
				synced 2025-10-26 08:37:23 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			198 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
			
		
		
	
	
			198 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
| import 'dart:async';
 | |
| 
 | |
| import 'package:flutter/foundation.dart';
 | |
| import 'package:flutter/material.dart';
 | |
| 
 | |
| import 'package:adaptive_dialog/adaptive_dialog.dart';
 | |
| import 'package:file_picker_cross/file_picker_cross.dart';
 | |
| import 'package:flutter_gen/gen_l10n/l10n.dart';
 | |
| import 'package:future_loading_dialog/future_loading_dialog.dart';
 | |
| import 'package:hive_flutter/hive_flutter.dart';
 | |
| import 'package:matrix/matrix.dart';
 | |
| import 'package:matrix_homeserver_recommendations/matrix_homeserver_recommendations.dart';
 | |
| import 'package:vrouter/vrouter.dart';
 | |
| 
 | |
| import 'package:fluffychat/config/app_config.dart';
 | |
| import 'package:fluffychat/pages/homeserver_picker/homeserver_bottom_sheet.dart';
 | |
| import 'package:fluffychat/pages/homeserver_picker/homeserver_picker_view.dart';
 | |
| import 'package:fluffychat/widgets/matrix.dart';
 | |
| import '../../utils/localized_exception_extension.dart';
 | |
| 
 | |
| import 'package:fluffychat/utils/tor_stub.dart'
 | |
|     if (dart.library.html) 'package:tor_detector_web/tor_detector_web.dart';
 | |
| 
 | |
| class HomeserverPicker extends StatefulWidget {
 | |
|   const HomeserverPicker({Key? key}) : super(key: key);
 | |
| 
 | |
|   @override
 | |
|   HomeserverPickerController createState() => HomeserverPickerController();
 | |
| }
 | |
| 
 | |
| class HomeserverPickerController extends State<HomeserverPicker> {
 | |
|   bool isLoading = false;
 | |
|   final TextEditingController homeserverController = TextEditingController(
 | |
|     text: AppConfig.defaultHomeserver,
 | |
|   );
 | |
|   final FocusNode homeserverFocusNode = FocusNode();
 | |
|   String? error;
 | |
|   List<HomeserverBenchmarkResult>? benchmarkResults;
 | |
|   bool displayServerList = false;
 | |
| 
 | |
|   bool get loadingHomeservers =>
 | |
|       AppConfig.allowOtherHomeservers && benchmarkResults == null;
 | |
|   String searchTerm = '';
 | |
| 
 | |
|   bool isTorBrowser = false;
 | |
| 
 | |
|   Future<void> _checkTorBrowser() async {
 | |
|     if (!kIsWeb) return;
 | |
| 
 | |
|     Hive.openBox('test').then((value) => null).catchError(
 | |
|       (e, s) async {
 | |
|         await showOkAlertDialog(
 | |
|             context: context,
 | |
|             title: L10n.of(context)!.indexedDbErrorTitle,
 | |
|             message: L10n.of(context)!.indexedDbErrorLong,
 | |
|             onWillPop: () async => false);
 | |
|         _checkTorBrowser();
 | |
|       },
 | |
|     );
 | |
| 
 | |
|     final isTor = await TorBrowserDetector.isTorBrowser;
 | |
|     setState(() => isTorBrowser = isTor);
 | |
|   }
 | |
| 
 | |
|   void _updateFocus() {
 | |
|     if (benchmarkResults == null) _loadHomeserverList();
 | |
|     if (homeserverFocusNode.hasFocus) {
 | |
|       setState(() {
 | |
|         displayServerList = true;
 | |
|       });
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   void showServerInfo(HomeserverBenchmarkResult server) => showModalBottomSheet(
 | |
|         context: context,
 | |
|         builder: (_) => HomeserverBottomSheet(
 | |
|           homeserver: server,
 | |
|         ),
 | |
|       );
 | |
| 
 | |
|   void onChanged(String text) => setState(() {
 | |
|         searchTerm = text;
 | |
|       });
 | |
| 
 | |
|   List<HomeserverBenchmarkResult> get filteredHomeservers => benchmarkResults!
 | |
|       .where((element) =>
 | |
|           element.homeserver.baseUrl.host.contains(searchTerm) ||
 | |
|           (element.homeserver.description?.contains(searchTerm) ?? false))
 | |
|       .toList();
 | |
| 
 | |
|   void _loadHomeserverList() async {
 | |
|     try {
 | |
|       final homeserverList =
 | |
|           await const JoinmatrixOrgParser().fetchHomeservers();
 | |
|       final benchmark = await HomeserverListProvider.benchmarkHomeserver(
 | |
|         homeserverList,
 | |
|         timeout: const Duration(seconds: 10),
 | |
|       );
 | |
|       setState(() {
 | |
|         benchmarkResults = benchmark;
 | |
|       });
 | |
|     } catch (e, s) {
 | |
|       Logs().e('Homeserver benchmark failed', e, s);
 | |
|       benchmarkResults = [];
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   void setServer(String server) => setState(() {
 | |
|         homeserverController.text = server;
 | |
|         searchTerm = '';
 | |
|         homeserverFocusNode.unfocus();
 | |
|         displayServerList = false;
 | |
|       });
 | |
| 
 | |
|   /// Starts an analysis of the given homeserver. It uses the current domain and
 | |
|   /// makes sure that it is prefixed with https. Then it searches for the
 | |
|   /// well-known information and forwards to the login page depending on the
 | |
|   /// login type.
 | |
|   Future<void> checkHomeserverAction() async {
 | |
|     setState(() {
 | |
|       homeserverFocusNode.unfocus();
 | |
|       error = null;
 | |
|       isLoading = true;
 | |
|       searchTerm = '';
 | |
|       displayServerList = false;
 | |
|     });
 | |
| 
 | |
|     try {
 | |
|       homeserverController.text =
 | |
|           homeserverController.text.trim().toLowerCase().replaceAll(' ', '-');
 | |
|       var homeserver = Uri.parse(homeserverController.text);
 | |
|       if (homeserver.scheme.isEmpty) {
 | |
|         homeserver = Uri.https(homeserverController.text, '');
 | |
|       }
 | |
|       final matrix = Matrix.of(context);
 | |
|       matrix.loginHomeserverSummary =
 | |
|           await matrix.getLoginClient().checkHomeserver(homeserver);
 | |
|       final ssoSupported = matrix.loginHomeserverSummary!.loginFlows
 | |
|           .any((flow) => flow.type == 'm.login.sso');
 | |
| 
 | |
|       try {
 | |
|         await Matrix.of(context).getLoginClient().register();
 | |
|         matrix.loginRegistrationSupported = true;
 | |
|       } on MatrixException catch (e) {
 | |
|         matrix.loginRegistrationSupported = e.requireAdditionalAuthentication;
 | |
|       }
 | |
| 
 | |
|       if (!ssoSupported && matrix.loginRegistrationSupported == false) {
 | |
|         // Server does not support SSO or registration. We can skip to login page:
 | |
|         VRouter.of(context).to('login');
 | |
|       } else {
 | |
|         VRouter.of(context).to('connect');
 | |
|       }
 | |
|     } catch (e) {
 | |
|       setState(() => error = (e).toLocalizedString(context));
 | |
|     } finally {
 | |
|       if (mounted) {
 | |
|         setState(() => isLoading = false);
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   @override
 | |
|   void dispose() {
 | |
|     homeserverFocusNode.removeListener(_updateFocus);
 | |
|     super.dispose();
 | |
|   }
 | |
| 
 | |
|   @override
 | |
|   void initState() {
 | |
|     homeserverFocusNode.addListener(_updateFocus);
 | |
|     _checkTorBrowser();
 | |
|     super.initState();
 | |
|   }
 | |
| 
 | |
|   @override
 | |
|   Widget build(BuildContext context) {
 | |
|     Matrix.of(context).navigatorContext = context;
 | |
|     return HomeserverPickerView(this);
 | |
|   }
 | |
| 
 | |
|   Future<void> restoreBackup() async {
 | |
|     await showFutureLoadingDialog(
 | |
|         context: context,
 | |
|         future: () async {
 | |
|           try {
 | |
|             final file = await FilePickerCross.importFromStorage(
 | |
|                 fileExtension: '.fluffybackup');
 | |
|             final client = Matrix.of(context).getLoginClient();
 | |
|             await client.importDump(file.toString());
 | |
|             Matrix.of(context).initMatrix();
 | |
|           } catch (e, s) {
 | |
|             Logs().e('Future error:', e, s);
 | |
|           }
 | |
|         });
 | |
|   }
 | |
| }
 | 
