首先,我使用Xamarin Forms作为WP8,iOS和Android应用程序。
目标:
我想在点击吐司时转到特定页面 根据吐司通知的有效载荷信息。
我使用Azure通知中心推送通知所有设置和运行良好。我使用MVVMLight及其依赖注入来专门为每个平台设置推送通知。
由于所需的格式不同,每个有效负载需要稍微不同。每次你都会注意到我想在有效载荷中发送一个SignalId,以便根据常规推送通知在接收设备上执行不同的操作。
的Android
{
"data" : {
"msg" : "message in here",
"signalId" : "id-in-here",
},
}
的iOS
{
"aps" : { "alert" : "message in here" },
"signalId" : "id-in-here"
}
Windows Phone 8
<?xml version="1.0" encoding="utf-8"?>
<wp:Notification xmlns:wp="WPNotification">
<wp:Toast>
<wp:Text1>category</wp:Text1>
<wp:Text2>message in here</wp:Text2>
<wp:Param>?signalId=id-in-here</wp:Param>
</wp:Toast>
</wp:Notification>
问题:
如何在Xamarin Forms应用中获取此信息并重定向到 重新激活应用程序时的相应页面,因为 用户点击了吐司通知?
我想在应用加载时获取有效负载信息,然后说,是,这包含一个SignalId,让我们重定向到此页面。
此时它只是在点击Toast通知时显示应用程序。我必须特定于应用程序,还是有Xamarin Forms方式?
即使您只知道如何在一个平台上进行操作,任何帮助都会受到赞赏,我可以从那里开始绕过其他平台。
答案 0 :(得分:3)
我找到了为所有平台做这件事的方法。 Windows已经过测试,Android和iOS都没有。
如果应用程序在后台,Windows和iOS会处理节目Toast通知,或者如果应用程序位于前台,则让代码处理它。无论应用程序状态如何,Android都会显示toast。
使用Windows Phone 8,我需要转到MainPage.xaml.cs并添加此覆盖。
protected override void OnNavigatedTo(System.Windows.Navigation.NavigationEventArgs e)
{
base.OnNavigatedTo(e);
if (this.NavigationContext.QueryString.ContainsKey("signalId"))
{
var signalId = this.NavigationContext.QueryString["signalId"];
var id = Guid.Empty;
if (signalId != null
&& Guid.TryParse(signalId, out id)
&& id != Guid.Empty)
{
this.NavigationContext.QueryString.Clear();
Deployment.Current.Dispatcher.BeginInvoke(() =>
{
// Do my navigation to a new page
});
}
}
}
对于GcmService中的Android
protected override void OnMessage(Context context, Intent intent)
{
Log.Info(Tag, "GCM Message Received!");
var message = intent.Extras.Get("msg").ToString();
var signalId = Guid.Empty;
if (intent.Extras.ContainsKey("signalId"))
{
signalId = new Guid(intent.Extras.Get("signalId").ToString());
}
// Show notification as usual
CreateNotification("", message, signalId);
}
然后在CreateNotification函数中,在Intent中添加一些额外的信息。
var uiIntent = new Intent(this, typeof(MainActivity));
if (signalId != Guid.Empty)
{
uiIntent.PutExtra("SignalId", signalId.ToString());
}
然后在MainActivity.cs中覆盖此函数
protected override void OnActivityResult(int requestCode, Result resultCode, Intent data)
{
if (data.HasExtra("SignalId"))
{
Guid signalId = new Guid(data.GetStringExtra("SignalId"));
if (signalId != Guid.Empty)
{
data.RemoveExtra("SignalId");
// Do you navigation
}
}
}
在iOS中,您会注意到我已经增强了默认的ProcessNotification()
void ProcessNotification(NSDictionary options, bool fromFinishedLaunching)
{
// Check to see if the dictionary has the aps key. This is the notification payload you would have sent
if (null != options && options.ContainsKey(new NSString("aps")))
{
//Get the aps dictionary
var aps = options.ObjectForKey(new NSString("aps")) as NSDictionary;
var alert = string.Empty;
//Extract the alert text
// NOTE: If you're using the simple alert by just specifying
// " aps:{alert:"alert msg here"} " this will work fine.
// But if you're using a complex alert with Localization keys, etc.,
// your "alert" object from the aps dictionary will be another NSDictionary.
// Basically the json gets dumped right into a NSDictionary,
// so keep that in mind.
if (aps.ContainsKey(new NSString("alert")))
alert = ((NSString) aps[new NSString("alert")]).ToString();
// If this came from the ReceivedRemoteNotification while the app was running,
// we of course need to manually process things like the sound, badge, and alert.
if (!fromFinishedLaunching)
{
//Manually show an alert
if (!string.IsNullOrEmpty(alert))
{
var signalId = new Guid(options.ObjectForKey(new NSString("signalId")) as NSString);
// Show my own toast with the signalId
}
}
}
}
然后在FinishedLaunching函数中检查是否有任何有效载荷
// Check if any payload from the push notification
if (options.ContainsKey("signalId"))
{
var signalId = new Guid(options.ObjectForKey(new NSString("signalId")) as NSString);
// Do the navigation here
}