From a81e732617b4b00c7ddd107e89ddaaf4d9c0b256 Mon Sep 17 00:00:00 2001 From: TheOneWithTheBraid Date: Sat, 5 Feb 2022 00:44:18 +0100 Subject: [PATCH] feat: propose homeserver based on response time - use `package:matrix_homeserver_recommendations` to benchmark homeservers - propose fastest server without anti-features as homeserver - display small button with server information - use Matrix.org / the default configuration as fallback Signed-off-by: TheOneWithTheBraid --- assets/l10n/intl_en.arb | 12 +- .../homeserver_picker/homeserver_picker.dart | 95 +++++++++++++-- .../homeserver_picker_view.dart | 51 +++++++-- .../homeserver_picker/homeserver_tile.dart | 108 ++++++++++++++++++ macos/Flutter/GeneratedPluginRegistrant.swift | 2 + pubspec.lock | 27 ++++- pubspec.yaml | 1 + .../flutter/generated_plugin_registrant.cc | 20 +++- windows/flutter/generated_plugin_registrant.h | 2 + windows/flutter/generated_plugins.cmake | 4 +- 10 files changed, 291 insertions(+), 31 deletions(-) create mode 100644 lib/pages/homeserver_picker/homeserver_tile.dart diff --git a/assets/l10n/intl_en.arb b/assets/l10n/intl_en.arb index 135fcfc4..76a8f1d8 100644 --- a/assets/l10n/intl_en.arb +++ b/assets/l10n/intl_en.arb @@ -2709,5 +2709,15 @@ "@iUnderstand": {}, "openChat": "Open Chat", "markAsRead": "Mark as read", - "reportUser": "Report user" + "reportUser": "Report user", + "showAvailableHomeservers": "Show available homeservers (Advanced users)", + "antiFeatures": "Anti-features", + "noAntiFeaturesRecorded": "No anti features recorded", + "serverRules": "Server rules", + "jurisdiction": "Jurisdiction", + "selectServer": "Select server", + "responseTime": "Response time", + "openServerList": "Visit", + "reportServerListProblem": "Report list issue", + "serverListJoinMatrix": "Server list by joinMatrix.org" } diff --git a/lib/pages/homeserver_picker/homeserver_picker.dart b/lib/pages/homeserver_picker/homeserver_picker.dart index af392c8f..cec008a2 100644 --- a/lib/pages/homeserver_picker/homeserver_picker.dart +++ b/lib/pages/homeserver_picker/homeserver_picker.dart @@ -3,10 +3,12 @@ import 'dart:async'; import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; +import 'package:animations/animations.dart'; import 'package:flutter_gen/gen_l10n/l10n.dart'; import 'package:flutter_inappwebview/flutter_inappwebview.dart'; import 'package:future_loading_dialog/future_loading_dialog.dart'; import 'package:matrix/matrix.dart'; +import 'package:matrix_homeserver_recommendations/matrix_homeserver_recommendations.dart'; import 'package:uni_links/uni_links.dart'; import 'package:url_launcher/url_launcher.dart'; import 'package:vrouter/vrouter.dart'; @@ -19,6 +21,7 @@ import 'package:fluffychat/utils/platform_infos.dart'; import 'package:fluffychat/widgets/matrix.dart'; import '../../main.dart'; import '../../utils/localized_exception_extension.dart'; +import 'homeserver_tile.dart'; class HomeserverPicker extends StatefulWidget { const HomeserverPicker({Key? key}) : super(key: key); @@ -28,14 +31,19 @@ class HomeserverPicker extends StatefulWidget { } class HomeserverPickerController extends State { - bool isLoading = false; - String domain = AppConfig.defaultHomeserver; - final TextEditingController homeserverController = - TextEditingController(text: AppConfig.defaultHomeserver); + bool isLoading = true; + + String? domain; + List? benchmarkResults; + + TextEditingController? homeserverController; + StreamSubscription? _intentDataStreamSubscription; String? error; Timer? _coolDown; + late HomeserverListProvider parser; + void setDomain(String domain) { this.domain = domain; _coolDown?.cancel(); @@ -94,7 +102,7 @@ class HomeserverPickerController extends State { if (token != null) _loginWithToken(token); }); } - checkHomeserverAction(); + benchmarkHomeServers(); } @override @@ -112,10 +120,12 @@ class HomeserverPickerController extends State { Future checkHomeserverAction() async { _coolDown?.cancel(); if (_lastCheckedHomeserver == domain) return; - if (domain.isEmpty) throw L10n.of(context)!.changeTheHomeserver; + if (domain == null || domain!.isEmpty) { + throw L10n.of(context)!.changeTheHomeserver; + } var homeserver = domain; - if (!homeserver.startsWith('https://')) { + if (!homeserver!.startsWith('https://')) { homeserver = 'https://$homeserver'; } @@ -223,7 +233,7 @@ class HomeserverPickerController extends State { void signUpAction() => VRouter.of(context).to( 'signup', - queryParameters: {'domain': domain}, + queryParameters: {'domain': domain!}, ); @override @@ -231,6 +241,75 @@ class HomeserverPickerController extends State { Matrix.of(context).navigatorContext = context; return HomeserverPickerView(this); } + + Future benchmarkHomeServers() async { + try { + parser = JoinmatrixOrgParser(); + final homeserverList = await parser.fetchHomeservers(); + final benchmark = await HomeserverListProvider.benchmarkHomeserver( + homeserverList, + timeout: const Duration(seconds: 10), + // TODO: do not rely on the homeserver list information telling the server supports registration + ); + + if (benchmark.isEmpty) { + throw NullThrownError(); + } + + // trying to use server without anti-features + final goodServers = []; + final badServers = []; + for (final result in benchmark) { + if (result.homeserver.antiFeatures == null) { + goodServers.add(result); + } else { + badServers.add(result); + } + } + + goodServers.sort(); + badServers.sort(); + + benchmarkResults = List.from([...goodServers, ...badServers]); + + domain = benchmarkResults!.first.homeserver.baseUrl.host; + } on Exception catch (e, s) { + Logs().e('Homeserver benchmark failed', e, s); + domain = AppConfig.defaultHomeserver; + } finally { + homeserverController = TextEditingController(text: domain); + checkHomeserverAction(); + } + } + + Future showServerPicker() async { + final selection = await showModal( + context: context, + builder: (context) => SimpleDialog( + title: Text(L10n.of(context)!.changeTheHomeserver), + children: [ + ...benchmarkResults!.map( + (e) => HomeserverTile( + benchmark: e, + onSelect: () { + Navigator.of(context).pop(e.homeserver); + }, + ), + ), + const Divider(), + JoinMatrixAttributionTile(), + ]), + ); + if (selection is Homeserver) { + if (domain != selection.baseUrl.host) { + setState(() { + domain = selection.baseUrl.host; + homeserverController!.text = domain!; + }); + checkHomeserverAction(); + } + } + } } class IdentityProvider { diff --git a/lib/pages/homeserver_picker/homeserver_picker_view.dart b/lib/pages/homeserver_picker/homeserver_picker_view.dart index c79e84e7..b8d49e3c 100644 --- a/lib/pages/homeserver_picker/homeserver_picker_view.dart +++ b/lib/pages/homeserver_picker/homeserver_picker_view.dart @@ -1,6 +1,7 @@ import 'package:flutter/cupertino.dart'; import 'package:flutter/material.dart'; +import 'package:animations/animations.dart'; import 'package:cached_network_image/cached_network_image.dart'; import 'package:flutter_gen/gen_l10n/l10n.dart'; import 'package:matrix/matrix.dart'; @@ -25,18 +26,44 @@ class HomeserverPickerView extends StatelessWidget { child: Scaffold( appBar: AppBar( titleSpacing: 8, - title: DefaultAppBarSearchField( - prefixText: 'https://', - hintText: L10n.of(context)!.enterYourHomeserver, - searchController: controller.homeserverController, - suffix: const Icon(Icons.edit_outlined), - padding: EdgeInsets.zero, - onChanged: controller.setDomain, - readOnly: !AppConfig.allowOtherHomeservers, - onSubmit: (_) => controller.checkHomeserverAction(), - unfocusOnClear: false, - autocorrect: false, - labelText: L10n.of(context)!.homeserver, + title: PageTransitionSwitcher( + transitionBuilder: ( + Widget child, + Animation primaryAnimation, + Animation secondaryAnimation, + ) { + return SharedAxisTransition( + animation: primaryAnimation, + secondaryAnimation: secondaryAnimation, + transitionType: SharedAxisTransitionType.vertical, + fillColor: Colors.transparent, + child: child, + ); + }, + child: controller.homeserverController == null + ? Center( + key: ValueKey(controller.homeserverController), + child: const CircularProgressIndicator(), + ) + : DefaultAppBarSearchField( + key: ValueKey(controller.homeserverController), + prefixIcon: IconButton( + icon: const Icon(Icons.format_list_numbered), + onPressed: controller.showServerPicker, + tooltip: L10n.of(context)!.showAvailableHomeservers, + ), + prefixText: 'https://', + hintText: L10n.of(context)!.enterYourHomeserver, + searchController: controller.homeserverController, + suffix: const Icon(Icons.edit_outlined), + padding: EdgeInsets.zero, + onChanged: controller.setDomain, + readOnly: !AppConfig.allowOtherHomeservers, + onSubmit: (_) => controller.checkHomeserverAction(), + unfocusOnClear: false, + autocorrect: false, + labelText: L10n.of(context)!.homeserver, + ), ), elevation: 0, ), diff --git a/lib/pages/homeserver_picker/homeserver_tile.dart b/lib/pages/homeserver_picker/homeserver_tile.dart new file mode 100644 index 00000000..39cceefa --- /dev/null +++ b/lib/pages/homeserver_picker/homeserver_tile.dart @@ -0,0 +1,108 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_gen/gen_l10n/l10n.dart'; +import 'package:matrix_homeserver_recommendations/matrix_homeserver_recommendations.dart'; +import 'package:url_launcher/link.dart'; + +class HomeserverTile extends StatelessWidget { + final HomeserverBenchmarkResult benchmark; + final VoidCallback onSelect; + + const HomeserverTile( + {Key? key, required this.benchmark, required this.onSelect}) + : super(key: key); + @override + Widget build(BuildContext context) { + return ExpansionTile( + title: Text(benchmark.homeserver.baseUrl.host), + subtitle: benchmark.homeserver.description != null + ? Text(benchmark.homeserver.description!) + : null, + children: [ + benchmark.homeserver.antiFeatures != null && + benchmark.homeserver.antiFeatures!.isNotEmpty + ? ListTile( + leading: const Icon(Icons.thumb_down), + title: Text(benchmark.homeserver.antiFeatures!), + subtitle: Text(L10n.of(context)!.antiFeatures), + ) + : ListTile( + leading: const Icon(Icons.recommend), + title: Text(L10n.of(context)!.noAntiFeaturesRecorded), + ), + if (benchmark.homeserver.jurisdiction != null) + ListTile( + leading: const Icon(Icons.public), + title: Text(benchmark.homeserver.jurisdiction!), + subtitle: Text(L10n.of(context)!.jurisdiction), + ), + ListTile( + leading: const Icon(Icons.speed), + title: Text("${benchmark.responseTime!.inMilliseconds} ms"), + subtitle: Text(L10n.of(context)!.responseTime), + ), + ButtonBar( + /* spacing: 8, + runSpacing: 8, + alignment: WrapAlignment.end, + runAlignment: WrapAlignment.center, + crossAxisAlignment: WrapCrossAlignment.center, */ + children: [ + if (benchmark.homeserver.privacyPolicy != null) + Link( + uri: benchmark.homeserver.privacyPolicy!, + target: LinkTarget.blank, + builder: (context, callback) => TextButton( + onPressed: callback, + child: Text(L10n.of(context)!.privacy), + ), + ), + if (benchmark.homeserver.rules != null) + Link( + uri: benchmark.homeserver.rules!, + target: LinkTarget.blank, + builder: (context, callback) => TextButton( + onPressed: callback, + child: Text(L10n.of(context)!.serverRules), + ), + ), + OutlinedButton( + onPressed: onSelect.call, + child: Text(L10n.of(context)!.selectServer), + ), + ], + ), + ], + ); + } +} + +class JoinMatrixAttributionTile extends StatelessWidget { + final parser = JoinmatrixOrgParser(); + + JoinMatrixAttributionTile({Key? key}) : super(key: key); + @override + Widget build(BuildContext context) { + return ListTile( + title: Text(L10n.of(context)!.serverListJoinMatrix), + subtitle: ButtonBar(children: [ + Link( + uri: parser.externalUri, + target: LinkTarget.blank, + builder: (context, callback) => TextButton( + onPressed: callback, + child: Text(L10n.of(context)!.openServerList), + ), + ), + Link( + uri: parser.errorReportUrl, + target: LinkTarget.blank, + builder: (context, callback) => TextButton( + onPressed: callback, + child: Text(L10n.of(context)!.reportServerListProblem), + ), + ), + ]), + ); + } +} diff --git a/macos/Flutter/GeneratedPluginRegistrant.swift b/macos/Flutter/GeneratedPluginRegistrant.swift index 4368a0a5..d7158b13 100644 --- a/macos/Flutter/GeneratedPluginRegistrant.swift +++ b/macos/Flutter/GeneratedPluginRegistrant.swift @@ -17,6 +17,7 @@ import path_provider_macos import shared_preferences_macos import sqflite import url_launcher_macos +import video_compress import wakelock_macos func RegisterGeneratedPlugins(registry: FlutterPluginRegistry) { @@ -32,5 +33,6 @@ func RegisterGeneratedPlugins(registry: FlutterPluginRegistry) { SharedPreferencesPlugin.register(with: registry.registrar(forPlugin: "SharedPreferencesPlugin")) SqflitePlugin.register(with: registry.registrar(forPlugin: "SqflitePlugin")) UrlLauncherPlugin.register(with: registry.registrar(forPlugin: "UrlLauncherPlugin")) + VideoCompressPlugin.register(with: registry.registrar(forPlugin: "VideoCompressPlugin")) WakelockMacosPlugin.register(with: registry.registrar(forPlugin: "WakelockMacosPlugin")) } diff --git a/pubspec.lock b/pubspec.lock index 557a592b..61b1748b 100644 --- a/pubspec.lock +++ b/pubspec.lock @@ -211,6 +211,13 @@ packages: url: "https://pub.dartlang.org" source: hosted version: "0.17.1" + csv: + dependency: transitive + description: + name: csv + url: "https://pub.dartlang.org" + source: hosted + version: "5.0.1" cupertino_icons: dependency: "direct main" description: @@ -812,6 +819,13 @@ packages: url: "https://pub.dartlang.org" source: hosted version: "0.12.11" + material_color_utilities: + dependency: transitive + description: + name: material_color_utilities + url: "https://pub.dartlang.org" + source: hosted + version: "0.1.3" matrix: dependency: "direct main" description: @@ -826,6 +840,13 @@ packages: url: "https://pub.dartlang.org" source: hosted version: "0.5.3" + matrix_homeserver_recommendations: + dependency: "direct main" + description: + name: matrix_homeserver_recommendations + url: "https://pub.dartlang.org" + source: hosted + version: "0.1.4" matrix_link_text: dependency: "direct main" description: @@ -1411,21 +1432,21 @@ packages: name: test url: "https://pub.dartlang.org" source: hosted - version: "1.17.12" + version: "1.19.5" test_api: dependency: transitive description: name: test_api url: "https://pub.dartlang.org" source: hosted - version: "0.4.3" + version: "0.4.8" test_core: dependency: transitive description: name: test_core url: "https://pub.dartlang.org" source: hosted - version: "0.4.2" + version: "0.4.9" timezone: dependency: transitive description: diff --git a/pubspec.yaml b/pubspec.yaml index 5f0bb77f..b51bf050 100644 --- a/pubspec.yaml +++ b/pubspec.yaml @@ -51,6 +51,7 @@ dependencies: localstorage: ^4.0.0+1 lottie: ^1.2.1 matrix: ^0.8.2 + matrix_homeserver_recommendations: ^0.1.4 matrix_link_text: ^1.0.2 open_noti_settings: ^0.4.0 package_info_plus: ^1.2.1 diff --git a/windows/flutter/generated_plugin_registrant.cc b/windows/flutter/generated_plugin_registrant.cc index 9579d43c..bf1e2199 100644 --- a/windows/flutter/generated_plugin_registrant.cc +++ b/windows/flutter/generated_plugin_registrant.cc @@ -2,14 +2,22 @@ // Generated file. Do not edit. // +// clang-format off + #include "generated_plugin_registrant.h" -#include -#include +#include +#include +#include +#include void RegisterPlugins(flutter::PluginRegistry* registry) { - FileChooserPluginRegisterWithRegistrar( - registry->GetRegistrarForPlugin("FileChooserPlugin")); - UrlLauncherPluginRegisterWithRegistrar( - registry->GetRegistrarForPlugin("UrlLauncherPlugin")); + DesktopDropPluginRegisterWithRegistrar( + registry->GetRegistrarForPlugin("DesktopDropPlugin")); + FileSelectorPluginRegisterWithRegistrar( + registry->GetRegistrarForPlugin("FileSelectorPlugin")); + FlutterSecureStorageWindowsPluginRegisterWithRegistrar( + registry->GetRegistrarForPlugin("FlutterSecureStorageWindowsPlugin")); + UrlLauncherWindowsRegisterWithRegistrar( + registry->GetRegistrarForPlugin("UrlLauncherWindows")); } diff --git a/windows/flutter/generated_plugin_registrant.h b/windows/flutter/generated_plugin_registrant.h index 9846246b..dc139d85 100644 --- a/windows/flutter/generated_plugin_registrant.h +++ b/windows/flutter/generated_plugin_registrant.h @@ -2,6 +2,8 @@ // Generated file. Do not edit. // +// clang-format off + #ifndef GENERATED_PLUGIN_REGISTRANT_ #define GENERATED_PLUGIN_REGISTRANT_ diff --git a/windows/flutter/generated_plugins.cmake b/windows/flutter/generated_plugins.cmake index 5c2bdaf0..acf95a15 100644 --- a/windows/flutter/generated_plugins.cmake +++ b/windows/flutter/generated_plugins.cmake @@ -3,7 +3,9 @@ # list(APPEND FLUTTER_PLUGIN_LIST - file_chooser + desktop_drop + file_selector_windows + flutter_secure_storage_windows url_launcher_windows )