带有推送通知的深层链接 - FCM - Android

Posted

技术标签:

【中文标题】带有推送通知的深层链接 - FCM - Android【英文标题】:Deep Link with Push Notification - FCM - Android 【发布时间】:2018-12-11 09:13:36 【问题描述】:

我想要什么:我想向用户发送推送通知。当用户点击该通知时,用户应导航到特定活动。

我做了什么:我在 Firebase 控制台中创建了一个深层链接。我也实现了 FirebaseInstanceIdServiceFirebaseMessagingService。我能够捕获从 Firebase 控制台发送的 Firebase 消息。

问题是什么:我无法捕捉到我在 Firebase 控制台中创建的动态链接。

我的代码如下。

MyFirebaseInstanceIDService.java

    public class MyFirebaseInstanceIDService extends FirebaseInstanceIdService 

    private final String TAG = "MyFirebaseInstanceID";

    @Override
    public void onTokenRefresh() 

        String refreshedToken = FirebaseInstanceId.getInstance().getToken();

        Log.e(TAG, "Refreshed token: " + refreshedToken);
    

MyFirebaseMessagingService.java

    public class MyFirebaseMessagingService extends FirebaseMessagingService 

    private final String TAG = "MyFbaseMessagingService";

    @Override
    public void onMessageReceived(RemoteMessage remoteMessage) 

        String message = remoteMessage.getNotification().getBody();

        Log.e(TAG, "\nmessage: " + message);

        sendNotification(message);
    

    private void sendNotification(String message) 

        Intent intent = new Intent(this, TestDeepLinkActivity.class);
        intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);

        Uri defaultSoundUri = RingtoneManager.getDefaultUri(RingtoneManager.TYPE_NOTIFICATION);

        PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, intent, PendingIntent.FLAG_UPDATE_CURRENT);

        NotificationCompat.Builder builder = new NotificationCompat.Builder(this)
                .setAutoCancel(true)
                .setContentTitle("FCM Test")
                .setContentText(message)
                .setSound(defaultSoundUri)
                .setSmallIcon(R.drawable.common_google_signin_btn_icon_dark)
                .setContentIntent(pendingIntent);

        NotificationManager manager = (NotificationManager) getSystemService(NOTIFICATION_SERVICE);

        manager.notify(0, builder.build());
    

Firebase 控制台图片

【问题讨论】:

【参考方案1】:

解决方案:

我必须在我想去的清单文件中的一个活动中添加意图过滤器,点击推送通知。此通知将包含一些在 android 术语中称为深度链接的 url。您可以参考下面的链接了解有关深度链接的更多信息。

https://developer.android.com/training/app-links/deep-linking

我将这两个链接用作深层链接:“www.somedomain.com/about”和“www.somedomain.com/app”。

请不要在意图过滤器中添加 httphttps,它们不受支持。 Chekout this 对话以获得更多说明。我也放了那个聊天的图片,以防万一将来链接过期。

请参阅下面的代码了解我如何将深层链接传递给 NotificationManager。意图过滤器自动拦截并启动该特定活动。

MyFirebaseMessagingService.java

public class MyFirebaseMessagingService extends FirebaseMessagingService 

    @Override
    public void onMessageReceived(RemoteMessage remoteMessage) 

        Map<String, String> data = remoteMessage.getData();

        String title = data.get("title");
        String message = data.get("message");
        String deepLink = data.get("deepLink");

        Notification notification = new Notification();
        notification.setTitle(title);
        notification.setMessage(message);
        notification.setDeepLink(deepLink);

        sendNotification(this, title, message, deepLink);
    

    public static void sendNotification(Context context, String title, String message, String deepLink) 

        NotificationManager notificationManager = (NotificationManager) context.getSystemService(Context.NOTIFICATION_SERVICE);

        if (Build.VERSION.SDK_INT >= 26) 
            NotificationChannel notificationChannel = new NotificationChannel("any_default_id", "any_channel_name",
                    NotificationManager.IMPORTANCE_HIGH);
            notificationChannel.setDescription("Any description can be given!");
            notificationManager.createNotificationChannel(notificationChannel);
        

        Uri defaultSoundUri = RingtoneManager.getDefaultUri(RingtoneManager.TYPE_NOTIFICATION);

        NotificationCompat.Builder notificationBuilder = new NotificationCompat.Builder(context)
                .setAutoCancel(true)
                .setSound(defaultSoundUri)
                .setSmallIcon(R.mipmap.ic_launcher)
                .setPriority(android.app.Notification.PRIORITY_MAX)
                .setDefaults(android.app.Notification.DEFAULT_ALL)
                .setLargeIcon(BitmapFactory.decodeResource(context.getResources(), R.mipmap.ic_launcher));

        Intent intent = new Intent();

        intent.setAction(Intent.ACTION_VIEW);
        intent.setData(Uri.parse(deepLink));
        intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP);

        PendingIntent pendingIntent = PendingIntent.getActivity(context, 0, intent, PendingIntent.FLAG_ONE_SHOT);

        notificationBuilder
                .setContentTitle(title)
                .setContentText(message)
                .setContentIntent(pendingIntent);

        notificationManager.notify(0, notificationBuilder.build());
    

AndroidManifest.xml

<activity
    android:name=".mvp.view.activity.ActivityName"
    android:label="@string/title_activity_name"
    android:theme="@style/AppTheme.NoActionBar">

    <intent-filter>
        <action android:name="android.intent.action.VIEW" />

        <category android:name="android.intent.category.DEFAULT" />
        <category android:name="android.intent.category.BROWSABLE" />

        <data
            android:host="www.somedomain.com"
            android:path="/about"
            android:scheme="app" />
    </intent-filter>

    <intent-filter>
        <action android:name="android.intent.action.VIEW" />

        <category android:name="android.intent.category.DEFAULT" />
        <category android:name="android.intent.category.BROWSABLE" />

        <data
            android:host="www.somedomain.com"
            android:path="/contact"
            android:scheme="app" />
    </intent-filter>
</activity>

额外:

如果您想在该活动中接收更多数据(即 userId 或 loanId),您可以在从服务器(即后端或基于 Web 的仪表板)发送推送通知时将其传递给。你可以像下面那样做。


 "data": 
 "userId": "65431214564651251456",
 "deepLink": "www.somedomain.com/app",
 "title": "This is title!",
 "message": "This is message!"
 ,
"to": "FCM token here"

重要提示:以下 JSON 无效,仅供参考。这在文档中也没有提到。所以请好好照顾它。上面是正确的 JSON。


    "to": "FCM Token here",
        "notification": 
            "Body": "This week’s edition is now available.",
            "title": "NewsMagazine.com",
            "icon": "new"
        ,
        "data": 
            "title": "This is title!",
            "message": "This is message!"
    

您可以在 MyFirebaseMessagingService 类的方法 onMessageReceived 中接收额外的数据(即 userId 或 LoanId),如下所示。
String userId = data.get("userId");
intent.putExtra(Intent.EXTRA_TEXT, userId);
在该活动中,您可以在 onCreate 方法中编写如下代码。
Intent intent = getIntent();
if (intent != null) 
    String intentStringExtra = intent.getStringExtra(Intent.EXTRA_TEXT);
    if (intentStringExtra != null) 
        userId = intentStringExtra;
    


【讨论】:

很好的解释:在你的回答中可能会在 this&lt;data android:host="www.somedomain.com" android:path="/contact" android:scheme="app" /&gt; 中产生冲突,我们使用 android:scheme="app" 来打开特定的活动,而 android:scheme="https" 用于特定的 url。请参阅您附加的链接。 onMessageReceived() 仅在应用程序打开时执行。所以在其他情况下(我认为大约 90% 的时间!)推送通知不适用于深度链接。 @PayamRoozbahaniFard 这就是您发送“通知”消息或“通知 + 数据”消息的时候。如果'发送数据消息(由你处理),则触发onMessageReceived(无论前台还是后台) @s_o_m_m_y_e_e 你的意思是如果服务器发送数据那么 onMessageReceived 每次都会被调用,不管你的应用是在前台还是在后台? @MaulikDodia 是的,如果您的服务器仅在 fcm 有效负载中发送“数据”(而不是“通知”),那么每次都会调用 onMessageReceived。参考here

以上是关于带有推送通知的深层链接 - FCM - Android的主要内容,如果未能解决你的问题,请参考以下文章

带有 FCM 推送通知的 Safari [重复]

IOS推送通知深度链接?

带有 Firebase (FCM) 推送通知的 Node js

如何构造 FCM 推送通知以在 Flutter(Android 和 iOS)的默认系统 Web 浏览器应用中打开指定的 URL

带有 FCM 推送通知的 Angular 服务工作者

推送通知在带有 release-apk 的 Android 7.0 (FCM) 上不起作用