如何使用 Firebase 在 Xamarin 中实现推送通知和使用 C# 后端的 Apple 推送通知

Posted

技术标签:

【中文标题】如何使用 Firebase 在 Xamarin 中实现推送通知和使用 C# 后端的 Apple 推送通知【英文标题】:How to implement Push Notification in Xamarin with Firebase and Apple Push Notification with C# backend 【发布时间】:2020-08-28 04:51:38 【问题描述】:

在我从 Native androidios 迁移到 Xamarin.Forms 期间,我决定使用 AppCenter Push 来发送通知,因为它是免费且易于使用的(当然,我花了很多时间让它工作,因为这是相对较新的东西,在线指导较少)。你可以在How to implement AppCenter Push API?找到我的原始分享。

在 Microsoft 宣布停用 AppCenter Push (https://devblogs.microsoft.com/appcenter/app-center-mbaas-retirement/) 并鼓励用户迁移到 Azure(这是一项付费服务​​)之前,我对此感到满意。我决定重新使用本机 FCM 和 APN 进行推送通知。

问题是没有关于如何做整个事情的直接教程。到处都有问题和解决方案,例如 iOS .P8 仅适用于 HTTP/2,而我的项目在不受支持的 .Net Framework 上运行。只有 .Net Core 支持 HTTP/2 协议。

【问题讨论】:

虽然可以回答您自己的问题,但请考虑re-wording 您的问题,以免它读起来像blog。除了标题之外,我不确定您是否真的在这里提问。 【参考方案1】:

我当前的项目在作为后端的 ASP.NET C# 上运行,它使用 Xamarin.Forms 向 Xamarin.Android 和 Xamarin.iOS 发送通知。如果您像我一样,请在下面找到我的答案,我将在下面分享完整的 C# 后端和 Xamarin.Forms 解决方案。让更多用户可以享受免费服务,而不是被推送到 Azure 付费服务。

第 1 部分 C# 后端 - C# ASP.NET 后端。对于 FCM 和 APN,它将分为两部分。

1.1) Firebase (FCM)

    要设置 FCM,您需要注册一个帐户。网上有大量的指南,这是https://xmonkeys360.com/2019/12/08/xamarin-forms-fcm-setup-configuration-part-i/ 的好指南之一。请记住获取 服务器密钥 并将 google-services.json 文件下载到您的 Xamarin.Android 项目。右键单击并将构建操作设置为“GoogleServiceJson”(Where can i add google-services.json in xamarin app)。

    下面是我的 Firebase

    using Newtonsoft.Json.Linq;
    using System;
    using System.Collections.Generic;
    using System.IO;
    using System.Net;
    using System.Web.Script.Serialization;
    
    namespace PushNotificationLibrary
    
        public class FirebaseCloudMessagingPush
        
            private const string WEB_ADDRESS = "https://fcm.googleapis.com/fcm/send";
    
            private const string SENDER_ID = "YOUR SENDER ID";
            private const string SERVER_KEY = "YOUR SERVER KEY";
    
            public string SendNotification(string deviceToken, string title, string message, string priority = "high", int badge = 0, List<Tuple<string, string>> parameters = null)
            
                var result = "-1";
                var httpWebRequest = (HttpWebRequest)WebRequest.Create(WEB_ADDRESS);
    
                parameters = parameters ?? new List<Tuple<string, string>>();
    
                httpWebRequest.ContentType = "application/json";
                httpWebRequest.Headers.Add(string.Format("Authorization: key=0", SERVER_KEY));
                httpWebRequest.Headers.Add(string.Format("Sender: id=0", SENDER_ID));
                httpWebRequest.Method = "POST";
    
                if (title.Length > 100)
                    title = title.Substring(0, 95) + "...";
    
                //Message cannot exceed 100
                if (message.Length > 100)
                    message = message.Substring(0, 95) + "...";
    
                JObject jObject = new JObject();
                jObject.Add("to", deviceToken);
                jObject.Add("priority", priority);
                jObject.Add("content_available", true);
    
                JObject jObjNotification = new JObject();
                jObjNotification.Add("body", message);
                jObjNotification.Add("title", title);            
    
                jObject.Add("notification", jObjNotification);
    
                JObject jObjData = new JObject();
    
                jObjData.Add("badge", badge);
                jObjData.Add("body", message);
                jObjData.Add("title", title);
    
                foreach (Tuple<string, string> parameter in parameters)
                
                    jObjData.Add(parameter.Item1, parameter.Item2);
                
    
                jObject.Add("data", jObjData);
    
                var serializer = new javascriptSerializer();
                using (var streamWriter = new StreamWriter(httpWebRequest.GetRequestStream()))
                
                    string json = jObject.ToString();
                    streamWriter.Write(json);
                    streamWriter.Flush();
                
    
                var httpResponse = (HttpWebResponse)httpWebRequest.GetResponse();
                using (var streamReader = new StreamReader(httpResponse.GetResponseStream()))
                
                    result = streamReader.ReadToEnd();
                
    
                return result;
            
        
    
    

1.2) iOS (APN)

    对于 APN,有两种处理方式。一种是使用.P12 常规方式。另一种方法是使用 .P8 的 Apple Auth Keys。我更喜欢使用 .P8,因为 .P12 证书每年到期,需要每年更新。使用 .P8 的问题在于,它使用的是 .Net Framework 不支持的 HTTP/2,但幸运的是,我设法克服了这个问题。请参考How to implement apple token based push notifications (using p8 file) in C#?,在该帖子中找到我的Answer,了解如何为 .Net Framework 实现整个过程。 *如果您已经在使用 .Net Core,答案将与我的相似,但您不必使用自定义 WinHTTPHandler。只需普通的 HTTPClient 即可。

第 2 部分 Xamarin.Forms - 接下来,您必须在 Xamarin.Forms 项目中支持通知。

    对于这部分来说,一点也不难。您只需参考https://github.com/CrossGeeks/PushNotificationPlugin,下载 Nuget 并按照链接中的说明设置您的 Xamarin 项目(Forms、Android、iOS)。

    我想强调的唯一一件事是如何以及在何处获取您的设备令牌。最初我试图在下面的代码(OnTokenRefresh)中获取设备令牌。但是你很快就会注意到这段代码并不总是被调用,我怀疑它只会在令牌被刷新时被调用,而不是每次你调试的时候。为了每次都获得您的设备令牌,只需在您项目的任何地方调用CrossPushNotification.Current.Token。将该设备令牌注册到您的服务器后端。并使用我在上面第 1 部分中的代码使用设备令牌发送通知。

    CrossPushNotification.Current.OnTokenRefresh += (s,p) =>
    
        System.Diagnostics.Debug.WriteLine($"TOKEN : p.Token");
    ;
    

就是这样!这很容易,但在将它们拼凑在一起之前,我花了数周时间尝试和出错。希望它能够为他人节省宝贵的时间。

【讨论】:

一个很好的详细答案,但请考虑重新措辞您的问题,不要像 博客 对于过时的 JavaScriptSerializer,使用 newtonsoft: string json = JsonConvert.SerializeObject(jObject); 连序列化器都没用? 只需使用 FIrebase SDK。手动编写 POST 请求是如此 2000【参考方案2】:

服务器:

在服务器端尝试 FirebaseAdmin。这个包非常简单。

https://github.com/firebase/firebase-admin-dotnet

按照这些设置说明进行操作:

https://firebase.google.com/docs/admin/setup#c

对于 Xamarin 应用:

我决定不想使用 CrossGeeks 插件,它非常简单。

对于 Android:

安装相关的 Xamarin.Firebase 包并在继承包 FirebaseMessagingService 的 Android 项目中创建自己的 Firebase 消息传递类。

[Service]
[IntentFilter(new[]  "com.google.firebase.INSTANCE_ID_EVENT" )]
[IntentFilter(new[]  "com.google.firebase.MESSAGING_EVENT" )]
class PushNotificationFirebaseMessagingService : FirebaseMessagingService

    private static string foregroundChannelId = "9001";

    public override void OnNewToken(string refreshedToken)
    
        base.OnNewToken(refreshedToken);

        SendRegistrationToServer(refreshedToken);
    

    private void SendRegistrationToServer(string token)
    
        //Your code here to register device token on server
    

    public override void OnMessageReceived(RemoteMessage message)
    
        SendNotification(message);

        base.OnMessageReceived(message);
    

    private void SendNotification(RemoteMessage message)
    
        try
        
            var notificationManager = GetSystemService(Context.NotificationService) as NotificationManager;

            var notificationChannel = new NotificationChannel(foregroundChannelId, "messaging_channel", NotificationImportance.High);

            var audioAttributes = new AudioAttributes.Builder()
                .SetContentType(AudioContentType.Sonification)
                .SetUsage(AudioUsageKind.Notification).Build();

            var notificationUri = RingtoneManager.GetDefaultUri(RingtoneType.Notification);

            notificationChannel.EnableLights(true);
            notificationChannel.EnableVibration(true);
            notificationChannel.SetSound(notificationUri, audioAttributes);


            notificationManager.CreateNotificationChannel(notificationChannel);

            var remoteNotification = message.GetNotification();

            var builder = new Notification.Builder(this, foregroundChannelId)
                .SetContentTitle(remoteNotification.Title)
                .SetContentText(remoteNotification.Body)
                .SetSmallIcon(Resource.Mipmap.icon);

            var notification = builder.Build();

            notificationManager.Notify(0, notification);
        
        catch (Exception ex)
        

        
    

在Application标签的AndroidManifest.xml中添加以下内容。

<receiver android:name="com.google.firebase.iid.FirebaseInstanceIdInternalReceiver" android:exported="false" />
<receiver android:name="com.google.firebase.iid.FirebaseInstanceIdReceiver" android:exported="true" android:permission="com.google.android.c2dm.permission.SEND">
    <intent-filter>
        <action android:name="com.google.android.c2dm.intent.RECEIVE" />
        <action android:name="com.google.android.c2dm.intent.REGISTRATION" />
        <category android:name="$applicationId" />
    </intent-filter>
</receiver>

【讨论】:

2021 这已经过时了 嗨,你能扩大一点吗? 您不再需要 并且 'com.google.firebase.iid.FirebaseInstanceIdReceiver' 已被 FirebaseMessagingService 取代 啊,好的,谢谢。所以只需从清单中删除标签?我需要换什么吗?以上取自这里:docs.microsoft.com/en-us/xamarin/android/data-cloud/…

以上是关于如何使用 Firebase 在 Xamarin 中实现推送通知和使用 C# 后端的 Apple 推送通知的主要内容,如果未能解决你的问题,请参考以下文章

在 Xamarin iOS 中使用 Firebase Crashlytics

如何使用 Firebase 在 Xamarin 中实现推送通知和使用 C# 后端的 Apple 推送通知

如何将 Firebase 消息传递到 Xamarin.Android 应用程序?

如何从 Xamarin 中的 Firebase 获取价值?

Xamarin Forms 2.3.3.175 和 Xamarin.Firebase.Messaging 不可能?

使用 Firebase 云消息传递在 Xamarin 跨平台上的聊天应用程序