* build: Reenable shrink resources and minify in gradle * build: (deps): bump image from 4.6.0 to 4.7.1 Bumps [image](https://github.com/brendan-duncan/image) from 4.6.0 to 4.7.1. - [Changelog](https://github.com/brendan-duncan/image/blob/main/CHANGELOG.md) - [Commits](https://github.com/brendan-duncan/image/commits) --- updated-dependencies: - dependency-name: image dependency-version: 4.7.1 dependency-type: direct:production update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] <support@github.com> * build: (deps): bump file_picker from 10.3.7 to 10.3.8 Bumps [file_picker](https://github.com/miguelpruivo/flutter_file_picker) from 10.3.7 to 10.3.8. - [Release notes](https://github.com/miguelpruivo/flutter_file_picker/releases) - [Changelog](https://github.com/miguelpruivo/flutter_file_picker/blob/master/CHANGELOG.md) - [Commits](https://github.com/miguelpruivo/flutter_file_picker/compare/v10.3.7...v10.3.8) --- updated-dependencies: - dependency-name: file_picker dependency-version: 10.3.8 dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] <support@github.com> * feat: Improved search * build: Use matrix sdk vom pub.dev again * chore: Follow up better search * build: (deps): bump image from 4.7.1 to 4.7.2 Bumps [image](https://github.com/brendan-duncan/image) from 4.7.1 to 4.7.2. - [Changelog](https://github.com/brendan-duncan/image/blob/main/CHANGELOG.md) - [Commits](https://github.com/brendan-duncan/image/commits) --- updated-dependencies: - dependency-name: image dependency-version: 4.7.2 dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] <support@github.com> * chore: Make cross signing self sign mandatory for bootstrap * chore: Update user device keys before creating bootstrap * fix: Better wait for secrets after verification bootstrap * refactor: Remove native imaging and enable web worker * refactor: Remove unused html onfocus streams * build: (deps): bump flutter_foreground_task from 9.1.0 to 9.2.0 Bumps [flutter_foreground_task](https://github.com/Dev-hwang/flutter_foreground_task) from 9.1.0 to 9.2.0. - [Changelog](https://github.com/Dev-hwang/flutter_foreground_task/blob/master/CHANGELOG.md) - [Commits](https://github.com/Dev-hwang/flutter_foreground_task/commits) --- updated-dependencies: - dependency-name: flutter_foreground_task dependency-version: 9.2.0 dependency-type: direct:production update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] <support@github.com> * chore(translations): Translated using Weblate (Uzbek) Currently translated at 99.7% (823 of 825 strings) Translation: FluffyChat/Translations Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/uz/ * chore(translations): Translated using Weblate (Russian) Currently translated at 99.8% (824 of 825 strings) Translation: FluffyChat/Translations Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/ru/ * chore(translations): Translated using Weblate (Norwegian Bokmål) Currently translated at 90.9% (750 of 825 strings) Translation: FluffyChat/Translations Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/nb_NO/ * chore(translations): Translated using Weblate (Galician) Currently translated at 100.0% (825 of 825 strings) Translation: FluffyChat/Translations Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/gl/ * chore(translations): Translated using Weblate (Basque) Currently translated at 99.7% (823 of 825 strings) Translation: FluffyChat/Translations Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/eu/ * chore(translations): Translated using Weblate (Ukrainian) Currently translated at 100.0% (825 of 825 strings) Translation: FluffyChat/Translations Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/uk/ * chore(translations): Translated using Weblate (Estonian) Currently translated at 100.0% (825 of 825 strings) Translation: FluffyChat/Translations Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/et/ * chore(translations): Translated using Weblate (Dutch) Currently translated at 100.0% (825 of 825 strings) Translation: FluffyChat/Translations Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/nl/ * chore(translations): Translated using Weblate (Russian) Currently translated at 100.0% (825 of 825 strings) Translation: FluffyChat/Translations Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/ru/ * chore(translations): Translated using Weblate (Spanish) Currently translated at 95.2% (788 of 827 strings) Translation: FluffyChat/Translations Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/es/ * chore(translations): Translated using Weblate (Spanish) Currently translated at 96.3% (797 of 827 strings) Translation: FluffyChat/Translations Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/es/ * chore(translations): Translated using Weblate (Russian) Currently translated at 100.0% (825 of 825 strings) Translation: FluffyChat/Translations Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/ru/ * chore(translations): Translated using Weblate (Russian) Currently translated at 100.0% (825 of 825 strings) Translation: FluffyChat/Translations Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/ru/ * fix: Broken ruzzian plurals * chore(translations): Translated using Weblate (Norwegian Bokmål) Currently translated at 91.2% (753 of 825 strings) Translation: FluffyChat/Translations Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/nb_NO/ * chore(translations): Translated using Weblate (Bengali) Currently translated at 4.5% (38 of 827 strings) Translation: FluffyChat/Translations Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/bn/ * chore(translations): Translated using Weblate (French) Currently translated at 82.3% (679 of 825 strings) Translation: FluffyChat/Translations Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/fr/ * build: (deps): bump translations_cleaner from 0.0.5 to 0.1.0 Bumps [translations_cleaner](https://github.com/Chinmay-KB/translations_cleaner) from 0.0.5 to 0.1.0. - [Changelog](https://github.com/Chinmay-KB/translations_cleaner/blob/main/CHANGELOG.md) - [Commits](https://github.com/Chinmay-KB/translations_cleaner/commits) --- updated-dependencies: - dependency-name: translations_cleaner dependency-version: 0.1.0 dependency-type: direct:production update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] <support@github.com> * chore(translations): Translated using Weblate (German) Currently translated at 99.2% (821 of 827 strings) Translation: FluffyChat/Translations Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/de/ * chore(translations): Translated using Weblate (Estonian) Currently translated at 100.0% (827 of 827 strings) Translation: FluffyChat/Translations Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/et/ * build: Bump version to 2.4.0 * build: (deps): bump sqflite_common_ffi from 2.3.6 to 2.3.7+1 Bumps [sqflite_common_ffi](https://github.com/tekartik/sqflite) from 2.3.6 to 2.3.7+1. - [Commits](https://github.com/tekartik/sqflite/compare/sqflite_common_ffi_v2.3.6...sqflite_common_ffi/v2.3.7) --- updated-dependencies: - dependency-name: sqflite_common_ffi dependency-version: 2.3.7+1 dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] <support@github.com> * chore(translations): Translated using Weblate (Czech) Currently translated at 66.1% (547 of 827 strings) Translation: FluffyChat/Translations Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/cs/ * chore(translations): Translated using Weblate (Czech) Currently translated at 72.7% (602 of 827 strings) Translation: FluffyChat/Translations Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/cs/ * chore(translations): Translated using Weblate (German) Currently translated at 99.8% (826 of 827 strings) Translation: FluffyChat/Translations Translate-URL: https://hosted.weblate.org/projects/fluffychat/translations/de/ * chore: Add security.md file * fix: Locale unlocalized strings * build: (deps): bump matrix from 4.1.0 to 5.0.0 Bumps [matrix](https://github.com/famedly/matrix-dart-sdk) from 4.1.0 to 5.0.0. - [Release notes](https://github.com/famedly/matrix-dart-sdk/releases) - [Changelog](https://github.com/famedly/matrix-dart-sdk/blob/main/CHANGELOG.md) - [Commits](https://github.com/famedly/matrix-dart-sdk/compare/v4.1.0...v5.0.0) --- updated-dependencies: - dependency-name: matrix dependency-version: 5.0.0 dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] <support@github.com> * fix: Notifications on web correctly managed when tab not focused * chore: Add changelog for android * chore: Remove duplicated localization * fix: Sign in label * chore: Versionize fcm shared isolate * build: Remove unused packag * build: (deps): bump package_info_plus from 8.3.1 to 9.0.0 Bumps [package_info_plus](https://github.com/fluttercommunity/plus_plugins/tree/main/packages/package_info_plus) from 8.3.1 to 9.0.0. - [Release notes](https://github.com/fluttercommunity/plus_plugins/releases) - [Commits](https://github.com/fluttercommunity/plus_plugins/commits/package_info_plus-v9.0.0/packages/package_info_plus) --- updated-dependencies: - dependency-name: package_info_plus dependency-version: 9.0.0 dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] <support@github.com> * feat: Display particle animation on login page * chore: Use fixed version of fcm shared isolate * fix: apk crash on some platforms due new flutter version * chore: Correct kotlin format * fix iOS notifications * fluffychat merge * fluffychat merge * fluffychat merge * fluffychat merge * fluffychat merge * fluffychat merge * add missing type annotations * update matrix version * fluffychat merge * fluffychat merge * fix notification on click actions --------- Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: Christian Kußowski <c.kussowski@famedly.com> Co-authored-by: Krille-chan <christian-kussowski@posteo.de> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> Co-authored-by: BeMeritus <bemerituss@gmail.com> Co-authored-by: Frank Paul Silye <frankps@gmail.com> Co-authored-by: josé m. <correoxm@disroot.org> Co-authored-by: xabirequejo <xabi.rn@gmail.com> Co-authored-by: Ihor Hordiichuk <igor_ck@outlook.com> Co-authored-by: Priit Jõerüüt <jrthwlate@users.noreply.hosted.weblate.org> Co-authored-by: Jelv <post@jelv.nl> Co-authored-by: Дмитрий Михирев <bizdelnick@gmail.com> Co-authored-by: Kimby <kimbyqs@gmail.com> Co-authored-by: Christian <christian-pauly@posteo.de> Co-authored-by: Kom nake <kominak310@svcache.com> Co-authored-by: hugues de keyzer <komputilisto@hugues.info> Co-authored-by: nautilusx <translate@disroot.org> Co-authored-by: Šebestová <ka.sebestova.cz@gmail.com>
247 lines
8.9 KiB
Dart
247 lines
8.9 KiB
Dart
import 'package:flutter/material.dart';
|
|
|
|
import 'package:collection/collection.dart' show IterableExtension;
|
|
import 'package:go_router/go_router.dart';
|
|
import 'package:matrix/matrix.dart';
|
|
import 'package:punycode/punycode.dart';
|
|
import 'package:url_launcher/url_launcher_string.dart';
|
|
|
|
import 'package:fluffychat/config/app_config.dart';
|
|
import 'package:fluffychat/l10n/l10n.dart';
|
|
import 'package:fluffychat/pangea/chat_list/widgets/public_room_bottom_sheet.dart';
|
|
import 'package:fluffychat/widgets/adaptive_dialogs/show_ok_cancel_alert_dialog.dart';
|
|
import 'package:fluffychat/widgets/adaptive_dialogs/user_dialog.dart';
|
|
import 'package:fluffychat/widgets/future_loading_dialog.dart';
|
|
import 'package:fluffychat/widgets/matrix.dart';
|
|
import 'platform_infos.dart';
|
|
|
|
class UrlLauncher {
|
|
/// The url to open.
|
|
final String? url;
|
|
|
|
/// The visible name in the GUI. For example the name of a markdown link
|
|
/// which may differ from the actual url to open.
|
|
final String? name;
|
|
|
|
final BuildContext context;
|
|
|
|
const UrlLauncher(this.context, this.url, [this.name]);
|
|
|
|
void launchUrl() async {
|
|
if (url!.toLowerCase().startsWith(AppConfig.deepLinkPrefix) ||
|
|
url!.toLowerCase().startsWith(AppConfig.inviteLinkPrefix) ||
|
|
{'#', '@', '!', '+', '\$'}.contains(url![0]) ||
|
|
url!.toLowerCase().startsWith(AppConfig.schemePrefix)) {
|
|
return openMatrixToUrl();
|
|
}
|
|
final uri = Uri.tryParse(url!);
|
|
if (uri == null) {
|
|
// we can't open this thing
|
|
ScaffoldMessenger.of(context).showSnackBar(
|
|
SnackBar(content: Text(L10n.of(context).cantOpenUri(url!))),
|
|
);
|
|
return;
|
|
}
|
|
|
|
if (name != null && url != name) {
|
|
// If there is a name which differs from the url, we need to make sure
|
|
// that the user can see the actual url before opening the browser.
|
|
final consent = await showOkCancelAlertDialog(
|
|
context: context,
|
|
title: L10n.of(context).openLinkInBrowser,
|
|
message: url,
|
|
okLabel: L10n.of(context).open,
|
|
cancelLabel: L10n.of(context).cancel,
|
|
);
|
|
if (consent != OkCancelResult.ok) return;
|
|
}
|
|
|
|
if (!{'https', 'http'}.contains(uri.scheme)) {
|
|
// just launch non-https / non-http uris directly
|
|
|
|
// we need to transmute geo URIs on desktop and on iOS
|
|
if ((!PlatformInfos.isMobile || PlatformInfos.isIOS) &&
|
|
uri.scheme == 'geo') {
|
|
final latlong = uri.path
|
|
.split(';')
|
|
.first
|
|
.split(',')
|
|
.map((s) => double.tryParse(s))
|
|
.toList();
|
|
if (latlong.length == 2 &&
|
|
latlong.first != null &&
|
|
latlong.last != null) {
|
|
if (PlatformInfos.isIOS) {
|
|
// iOS is great at not following standards, so we need to transmute the geo URI
|
|
// to an apple maps thingy
|
|
// https://developer.apple.com/library/archive/featuredarticles/iPhoneURLScheme_Reference/MapLinks/MapLinks.html
|
|
final ll = '${latlong.first},${latlong.last}';
|
|
launchUrlString('https://maps.apple.com/?q=$ll&sll=$ll');
|
|
} else {
|
|
// transmute geo URIs on desktop to openstreetmap links, as those usually can't handle
|
|
// geo URIs
|
|
launchUrlString(
|
|
'https://www.openstreetmap.org/?mlat=${latlong.first}&mlon=${latlong.last}#map=16/${latlong.first}/${latlong.last}',
|
|
);
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
launchUrlString(url!);
|
|
return;
|
|
}
|
|
if (uri.host.isEmpty) {
|
|
ScaffoldMessenger.of(context).showSnackBar(
|
|
SnackBar(content: Text(L10n.of(context).cantOpenUri(url!))),
|
|
);
|
|
return;
|
|
}
|
|
// okay, we have either an http or an https URI.
|
|
// As some platforms have issues with opening unicode URLs, we are going to help
|
|
// them out by punycode-encoding them for them ourself.
|
|
final newHost = uri.host
|
|
.split('.')
|
|
.map((hostPartEncoded) {
|
|
final hostPart = Uri.decodeComponent(hostPartEncoded);
|
|
final hostPartPunycode = punycodeEncode(hostPart);
|
|
return hostPartPunycode != '$hostPart-'
|
|
? 'xn--$hostPartPunycode'
|
|
: hostPart;
|
|
})
|
|
.join('.');
|
|
// Force LaunchMode.externalApplication, otherwise url_launcher will default
|
|
// to opening links in a webview on mobile platforms.
|
|
launchUrlString(
|
|
uri.replace(host: newHost).toString(),
|
|
mode: LaunchMode.externalApplication,
|
|
);
|
|
}
|
|
|
|
void openMatrixToUrl() async {
|
|
final matrix = Matrix.of(context);
|
|
final url = this.url!.replaceFirst(
|
|
AppConfig.deepLinkPrefix,
|
|
AppConfig.inviteLinkPrefix,
|
|
);
|
|
|
|
// The identifier might be a matrix.to url and needs escaping. Or, it might have multiple
|
|
// identifiers (room id & event id), or it might also have a query part.
|
|
// All this needs parsing.
|
|
final identityParts =
|
|
url.parseIdentifierIntoParts() ??
|
|
Uri.tryParse(url)?.host.parseIdentifierIntoParts() ??
|
|
Uri.tryParse(url)?.pathSegments
|
|
.lastWhereOrNull((_) => true)
|
|
?.parseIdentifierIntoParts();
|
|
if (identityParts == null) {
|
|
return; // no match, nothing to do
|
|
}
|
|
if (identityParts.primaryIdentifier.sigil == '#' ||
|
|
identityParts.primaryIdentifier.sigil == '!') {
|
|
// we got a room! Let's open that one
|
|
final roomIdOrAlias = identityParts.primaryIdentifier;
|
|
final event = identityParts.secondaryIdentifier;
|
|
var room =
|
|
matrix.client.getRoomByAlias(roomIdOrAlias) ??
|
|
matrix.client.getRoomById(roomIdOrAlias);
|
|
var roomId = room?.id;
|
|
// we make the servers a set and later on convert to a list, so that we can easily
|
|
// deduplicate servers added via alias lookup and query parameter
|
|
final servers = <String>{};
|
|
if (room == null && roomIdOrAlias.sigil == '#') {
|
|
// we were unable to find the room locally...so resolve it
|
|
final response = await showFutureLoadingDialog(
|
|
context: context,
|
|
future: () => matrix.client.getRoomIdByAlias(roomIdOrAlias),
|
|
);
|
|
if (response.error != null) {
|
|
return; // nothing to do, the alias doesn't exist
|
|
}
|
|
roomId = response.result!.roomId;
|
|
servers.addAll(response.result!.servers!);
|
|
room = matrix.client.getRoomById(roomId!);
|
|
}
|
|
servers.addAll(identityParts.via);
|
|
// #Pangea
|
|
if (room != null && room.membership != Membership.leave) {
|
|
// if (room != null) {
|
|
// Pangea#
|
|
if (room.isSpace) {
|
|
// TODO: Implement navigate to space
|
|
context.go('/rooms/${room.id}');
|
|
|
|
return;
|
|
}
|
|
// we have the room, so....just open it
|
|
if (event != null) {
|
|
context.go(
|
|
'/${Uri(pathSegments: ['rooms', room.id], queryParameters: {'event': event})}',
|
|
);
|
|
} else {
|
|
context.go('/rooms/${room.id}');
|
|
}
|
|
return;
|
|
} else {
|
|
// #Pangea
|
|
// await showAdaptiveDialog(
|
|
// context: context,
|
|
// builder: (c) =>
|
|
// PublicRoomDialog(roomAlias: identityParts.primaryIdentifier),
|
|
// );
|
|
await PublicRoomBottomSheet.show(
|
|
context: context,
|
|
roomAlias: identityParts.primaryIdentifier,
|
|
);
|
|
// Pangea#
|
|
}
|
|
if (roomIdOrAlias.sigil == '!') {
|
|
if (await showOkCancelAlertDialog(
|
|
useRootNavigator: false,
|
|
context: context,
|
|
title: 'Join room $roomIdOrAlias',
|
|
) ==
|
|
OkCancelResult.ok) {
|
|
roomId = roomIdOrAlias;
|
|
final response = await showFutureLoadingDialog(
|
|
context: context,
|
|
future: () => matrix.client.joinRoom(
|
|
roomIdOrAlias,
|
|
serverName: servers.isNotEmpty ? servers.toList() : null,
|
|
),
|
|
);
|
|
// wait for two seconds so that it probably came down /sync
|
|
await showFutureLoadingDialog(
|
|
context: context,
|
|
future: () => Future.delayed(const Duration(seconds: 2)),
|
|
);
|
|
if (event != null) {
|
|
context.go(
|
|
Uri(
|
|
pathSegments: ['rooms', response.result!],
|
|
queryParameters: {'event': event},
|
|
).toString(),
|
|
);
|
|
} else {
|
|
context.go('/rooms/${response.result!}');
|
|
}
|
|
}
|
|
}
|
|
} else if (identityParts.primaryIdentifier.sigil == '@') {
|
|
final userId = identityParts.primaryIdentifier;
|
|
var noProfileWarning = false;
|
|
final profileResult = await showFutureLoadingDialog(
|
|
context: context,
|
|
future: () =>
|
|
matrix.client.getProfileFromUserId(userId).catchError((_) {
|
|
noProfileWarning = true;
|
|
return Profile(userId: userId);
|
|
}),
|
|
);
|
|
await UserDialog.show(
|
|
context: context,
|
|
profile: profileResult.result!,
|
|
noProfileWarning: noProfileWarning,
|
|
);
|
|
}
|
|
}
|
|
}
|