import 'dart:io'; import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; import 'package:collection/collection.dart'; import 'package:desktop_notifications/desktop_notifications.dart'; import 'package:flutter_local_notifications/flutter_local_notifications.dart'; import 'package:image/image.dart'; import 'package:matrix/matrix.dart'; import 'package:universal_html/html.dart' as html; import 'package:hermes/config/setting_keys.dart'; import 'package:hermes/l10n/l10n.dart'; import 'package:hermes/utils/push_helper.dart'; import 'package:hermes/widgets/hermes_app.dart'; import 'package:hermes/config/app_config.dart'; import 'package:hermes/utils/client_download_content_extension.dart'; import 'package:hermes/utils/matrix_sdk_extensions/matrix_locals.dart'; import 'package:hermes/utils/platform_infos.dart'; import 'package:hermes/widgets/matrix.dart'; extension LocalNotificationsExtension on MatrixState { static final html.AudioElement _audioPlayer = html.AudioElement() ..src = 'assets/assets/sounds/notification.ogg' ..load(); void showLocalNotification(Event event) async { Logs().v( '[Notifications] event received for ${event.room.id} (${event.type})', ); final roomId = event.room.id; if (activeRoomId == roomId) { if (kIsWeb && webHasFocus) return; if ((PlatformInfos.isLinux || PlatformInfos.isWindows) && WidgetsBinding.instance.lifecycleState == AppLifecycleState.resumed) { return; } } final title = event.room.getLocalizedDisplayname(MatrixLocals(L10n.of(context))); final body = await event.calcLocalizedBody( MatrixLocals(L10n.of(context)), withSenderNamePrefix: !event.room.isDirectChat || event.room.lastEvent?.senderId == client.userID, plaintextBody: true, hideReply: true, hideEdit: true, removeMarkdown: true, ); if (kIsWeb) { final avatarUrl = event.senderFromMemoryOrFallback.avatarUrl; Uri? thumbnailUri; if (avatarUrl != null) { const size = 128; const thumbnailMethod = ThumbnailMethod.crop; // Pre-cache so that we can later just set the thumbnail uri as icon: await client.downloadMxcCached( avatarUrl, width: size, height: size, thumbnailMethod: thumbnailMethod, isThumbnail: true, rounded: true, ); thumbnailUri = await event.senderFromMemoryOrFallback.avatarUrl?.getThumbnailUri( client, width: size, height: size, method: thumbnailMethod, ); } _audioPlayer.play(); html.Notification( title, body: body, icon: thumbnailUri?.toString(), tag: event.room.id, ); } else if (Platform.isMacOS) { Logs().v('[Notifications] preparing macOS notification for $roomId'); final plugin = backgroundPush?.notificationsPlugin; if (plugin == null) { Logs().w('Local notification requested on macOS but plugin missing'); return; } await plugin.show( roomId.hashCode, title, body, const NotificationDetails( macOS: DarwinNotificationDetails( presentAlert: true, presentSound: true, presentBadge: true, interruptionLevel: InterruptionLevel.timeSensitive, ), ), payload: roomId, ); Logs().v( '[Notifications] macOS notification displayed via plugin for $roomId', ); } else if (Platform.isLinux) { final avatarUrl = event.room.avatar; final hints = [NotificationHint.soundName('message-new-instant')]; if (avatarUrl != null) { const size = notificationAvatarDimension; const thumbnailMethod = ThumbnailMethod.crop; // Pre-cache so that we can later just set the thumbnail uri as icon: final data = await client.downloadMxcCached( avatarUrl, width: size, height: size, thumbnailMethod: thumbnailMethod, isThumbnail: true, rounded: true, ); final image = decodeImage(data); if (image != null) { final realData = image.getBytes(order: ChannelOrder.rgba); hints.add( NotificationHint.imageData( image.width, image.height, realData, hasAlpha: true, channels: 4, ), ); } } final notification = await linuxNotifications!.notify( title, body: body, replacesId: linuxNotificationIds[roomId] ?? 0, appName: AppSettings.applicationName.value, appIcon: 'hermes', actions: [ NotificationAction( DesktopNotificationActions.openChat.name, L10n.of(context).openChat, ), NotificationAction( DesktopNotificationActions.seen.name, L10n.of(context).markAsRead, ), ], hints: hints, ); notification.action.then((actionStr) { var action = DesktopNotificationActions.values .singleWhereOrNull((a) => a.name == actionStr); if (action == null && actionStr == "default") { action = DesktopNotificationActions.openChat; } switch (action!) { case DesktopNotificationActions.seen: event.room.setReadMarker( event.eventId, mRead: event.eventId, public: AppSettings.sendPublicReadReceipts.value, ); break; case DesktopNotificationActions.openChat: HermesApp.router.go('/rooms/${event.room.id}'); break; } }); linuxNotificationIds[roomId] = notification.id; } } } enum DesktopNotificationActions { seen, openChat }