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.
		
		
		
		
		
			
		
			
				
	
	
		
			195 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			Dart
		
	
			
		
		
	
	
			195 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			Dart
		
	
| import 'package:flutter/material.dart';
 | |
| 
 | |
| import 'package:adaptive_dialog/adaptive_dialog.dart';
 | |
| import 'package:collection/collection.dart' show IterableExtension;
 | |
| import 'package:flutter_gen/gen_l10n/l10n.dart';
 | |
| import 'package:future_loading_dialog/future_loading_dialog.dart';
 | |
| import 'package:matrix/matrix.dart';
 | |
| import 'package:punycode/punycode.dart';
 | |
| import 'package:url_launcher/url_launcher.dart';
 | |
| import 'package:vrouter/vrouter.dart';
 | |
| 
 | |
| import 'package:fluffychat/config/app_config.dart';
 | |
| import 'package:fluffychat/widgets/matrix.dart';
 | |
| import 'package:fluffychat/widgets/profile_bottom_sheet.dart';
 | |
| import 'package:fluffychat/widgets/public_room_bottom_sheet.dart';
 | |
| import 'platform_infos.dart';
 | |
| 
 | |
| class UrlLauncher {
 | |
|   final String? url;
 | |
|   final BuildContext context;
 | |
| 
 | |
|   const UrlLauncher(this.context, this.url);
 | |
| 
 | |
|   void launchUrl() {
 | |
|     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 (!{'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}';
 | |
|             launch('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
 | |
|             launch(
 | |
|                 'https://www.openstreetmap.org/?mlat=${latlong.first}&mlon=${latlong.last}#map=16/${latlong.first}/${latlong.last}');
 | |
|           }
 | |
|           return;
 | |
|         }
 | |
|       }
 | |
|       launch(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('.');
 | |
|     launch(uri.replace(host: newHost).toString());
 | |
|   }
 | |
| 
 | |
|   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);
 | |
|       if (room != null) {
 | |
|         if (room.isSpace) {
 | |
|           // TODO: Implement navigate to space
 | |
|           VRouter.of(context).toSegments(['rooms']);
 | |
|           return;
 | |
|         }
 | |
|         // we have the room, so....just open it
 | |
|         if (event != null) {
 | |
|           VRouter.of(context).toSegments(['rooms', room.id],
 | |
|               queryParameters: {'event': event});
 | |
|         } else {
 | |
|           VRouter.of(context).toSegments(['rooms', room.id]);
 | |
|         }
 | |
|         return;
 | |
|       } else {
 | |
|         await showModalBottomSheet(
 | |
|           context: context,
 | |
|           builder: (c) => PublicRoomBottomSheet(
 | |
|             roomAlias: identityParts.primaryIdentifier,
 | |
|             outerContext: context,
 | |
|           ),
 | |
|         );
 | |
|       }
 | |
|       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,
 | |
|             ),
 | |
|           );
 | |
|           if (response.error != null) return;
 | |
|           // 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) {
 | |
|             VRouter.of(context).toSegments(['rooms', response.result!],
 | |
|                 queryParameters: {'event': event});
 | |
|           } else {
 | |
|             VRouter.of(context).toSegments(['rooms', response.result!]);
 | |
|           }
 | |
|         }
 | |
|       }
 | |
|     } else if (identityParts.primaryIdentifier.sigil == '@') {
 | |
|       await showModalBottomSheet(
 | |
|         context: context,
 | |
|         builder: (c) => ProfileBottomSheet(
 | |
|           userId: identityParts.primaryIdentifier,
 | |
|           outerContext: context,
 | |
|         ),
 | |
|       );
 | |
|     }
 | |
|   }
 | |
| }
 |