Windows Phone开发(44):推送通知第二集——磁贴通知

简介: 原文: Windows Phone开发(44):推送通知第二集——磁贴通知 前面我们说了第一个类型——Toast通知,这玩意儿不知大家是不是觉得很新鲜,以前玩.NET编程应该没接触过吧? 其实这东西绝对不复杂,只是刚接触的时候会有点莫名罢了,Toast通知和今天要说的磁贴通知,都有一个共同点,那就是格式都规定死了D。
原文: Windows Phone开发(44):推送通知第二集——磁贴通知

前面我们说了第一个类型——Toast通知,这玩意儿不知大家是不是觉得很新鲜,以前玩.NET编程应该没接触过吧?

其实这东西绝对不复杂,只是刚接触的时候会有点莫名罢了,Toast通知和今天要说的磁贴通知,都有一个共同点,那就是格式都规定死了D。

本质就是向特定的URI地址POST一个XML文档罢了,相信很多人都会,如果你还不会,真的,要补一补基础课了。

 多说无益,还是快点切入主题,开门见水吧。

首先,我们要知道我们在服务器端要POST什么样的XML文档,来,一起来看看。

<?xml version="1.0" encoding="utf-8" ?>
<wp:Notification xmlns:wp="WPNotification">
  <wp:Tile ID="导航URI">
    <wp:BackgroundImage>正面背景图片</wp:BackgroundImage>
    <wp:Count>计数器</wp:Count>
    <wp:Title>正面标题</wp:Title>
    <wp:BackBackgroundImage>背面背景图片</wp:BackBackgroundImage>
    <wp:BackTitle>背面标题</wp:BackTitle>
    <wp:BackContent>背面内容</wp:BackContent>
  </wp:Tile>
</wp:Notification>


前面关于磁贴的内容,大家有印象吧?

磁帖者,有正面的标题、背景图、计数器;背面有标题、背景图和正文。有印象就好,不用我打水口枪。

 

来吧,我们通过一个现场演练来体会体会吧。

 

先做服务器端,这回我选择用ASP.NET,不要告诉我你不会。

启动VS,建一个ASP.NET网站,然后,把default.aspx改造一下,如果你嫌生成的代码不好看,可以把文件删除,然后新建一个页面。

好了,页面布局嘛,我贴一下HTML就行了。

<%@ Page Language="C#" AutoEventWireup="true" CodeFile="Default.aspx.cs" Inherits="_Default" %>

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">

<html xmlns="http://www.w3.org/1999/xhtml">
<head runat="server">
    <title></title>
</head>
<body>
    <form id="form1" runat="server">
    <div>
        <div>
            目标URI:
            <asp:TextBox ID="txtURI" runat="server" Width="911px"></asp:TextBox>
        </div>
        <div>
            <table border="0">
                <tr>
                    <td>正面背景:</td>
                    <td>
                        <asp:TextBox ID="txtBackImg" runat="server" Width="316px"></asp:TextBox></td>
                </tr>
                <tr>
                    <td>正面标题:</td>
                    <td>
                        <asp:TextBox ID="txtTitle" runat="server" Width="316px"></asp:TextBox>
                    </td>
                </tr>
                <tr>
                    <td>计数:</td>
                    <td>
                        <asp:TextBox ID="txtCount" runat="server" Width="313px"></asp:TextBox>
                    </td>
                </tr>
                <tr>
                    <td>背面背景:</td>
                    <td>
                        <asp:TextBox ID="txtBackBackImg" runat="server" Width="316px"></asp:TextBox>
                    </td>
                </tr>
                <tr>
                    <td>背面标题:</td>
                    <td>
                        <asp:TextBox ID="txtBackTitle" runat="server" Width="321px"></asp:TextBox>
                    </td>
                </tr>
                <tr>
                    <td>背面正文:</td>
                    <td>
                        <asp:TextBox ID="txtBackContent" runat="server" Width="309px"></asp:TextBox>
                    </td>
                </tr>
            </table>
            <div style="margin-left:20px; margin-top:10px;">
                <asp:Button ID="btnSend" runat="server" Text="发送" onclick="btnSend_Click" /></div>
        </div>
        <div style=" margin-top:20px;">
            <asp:TextBox ID="txtRes" runat="server" Height="155px" TextMode="MultiLine" 
                Width="729px"></asp:TextBox>
        </div>
    </div>
    </form>
</body>
</html>


 

还是别少了后台代码。

/*
 <?xml version="1.0" encoding="utf-8" ?>
<wp:Notification xmlns:wp="WPNotification">
  <wp:Tile ID="导航URI">
    <wp:BackgroundImage>正面背景图片</wp:BackgroundImage>
    <wp:Count>计数器</wp:Count>
    <wp:Title>正面标题</wp:Title>
    <wp:BackBackgroundImage>背面背景图片</wp:BackBackgroundImage>
    <wp:BackTitle>背面标题</wp:BackTitle>
    <wp:BackContent>背面内容</wp:BackContent>
  </wp:Tile>
</wp:Notification>

 * 清除磁贴的属性值
 <?xml version="1.0" encoding="utf-8" ?>
<wp:Notification xmlns:wp="WPNotification">
  <wp:Tile ID="导航URI">
    <wp:BackgroundImage></wp:BackgroundImage>
    <wp:Count Action="Clear"></wp:Count>
    <wp:Title Action="Clear"></wp:Title>
    <wp:BackBackgroundImage Action="Clear"></wp:BackBackgroundImage>
    <wp:BackTitle Action="Clear"></wp:BackTitle>
    <wp:BackContent Action="Clear"></wp:BackContent>
  </wp:Tile>
</wp:Notification>

 * HTTP标头
 X-WindowsPhone-Target: token
X-NotificationClass:1
1 立即发送
11 450秒发送
21  900秒发送
 
 */

using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Web.UI;
using System.Web.UI.WebControls;

using System.Net;
using System.Net.Mime;
using System.IO;
using System.Text;


public partial class _Default : System.Web.UI.Page
{
    protected void Page_Load(object sender, EventArgs e)
    {

    }
    protected void btnSend_Click(object sender, EventArgs e)
    {
        HttpWebRequest request = (HttpWebRequest)WebRequest.Create(txtURI.Text);
        request.Method = WebRequestMethods.Http.Post;
        // 加上HTTP标头
        request.Headers.Add("X-WindowsPhone-Target", "token");
        request.Headers.Add("X-NotificationClass", "1");
        // 拼接内容,XML文档
        string Msg = "<?xml version=\"1.0\" encoding=\"utf-8\" ?>" +
                     "<wp:Notification xmlns:wp=\"WPNotification\">" +
                        "<wp:Tile>" +
                            "<wp:BackgroundImage>" + txtBackImg.Text + "</wp:BackgroundImage>" +
                            "<wp:Count>" + txtCount.Text + "</wp:Count>" +
                            "<wp:Title>" + txtTitle.Text + "</wp:Title>" +
                            "<wp:BackBackgroundImage>" + txtBackBackImg.Text + "</wp:BackBackgroundImage>" +
                            "<wp:BackTitle>" + txtBackTitle.Text + "</wp:BackTitle>" +
                            "<wp:BackContent>" + txtBackContent.Text + "</wp:BackContent>" +
                        "</wp:Tile>" +
                      "</wp:Notification>";
        byte[] buffer = Encoding.UTF8.GetBytes(Msg);
        request.ContentType = MediaTypeNames.Text.Xml;
        // POST数据要记得设置内容长度
        request.ContentLength = buffer.Length;
        // 写入流
        using (Stream stream = request.GetRequestStream())
        {
            stream.Write(buffer, 0, buffer.Length);
        }
        // 接收回应
        HttpWebResponse response = (HttpWebResponse)request.GetResponse();
        // 读出响应的HTTP头
        string headers = "";
        foreach (string key in response.Headers.AllKeys)
        {
            headers += key + " : " + response.Headers.Get(key) + "\r\n";
        }
        txtRes.Text = headers;
    }
}


 

补充一下,上面代码中,前面的注释我已经写上了,其实MSDN上都有,我想很多人不看,我说一下,如果你打算清除磁贴某些属性的值,如标题等,这可以用以下的XML文档。

<?xml version="1.0" encoding="utf-8" ?>
<wp:Notification xmlns:wp="WPNotification">
  <wp:Tile ID="导航URI">
    <wp:BackgroundImage></wp:BackgroundImage>
    <wp:Count Action="Clear"></wp:Count>
    <wp:Title Action="Clear"></wp:Title>
    <wp:BackBackgroundImage Action="Clear"></wp:BackBackgroundImage>
    <wp:BackTitle Action="Clear"></wp:BackTitle>
    <wp:BackContent Action="Clear"></wp:BackContent>
  </wp:Tile>
</wp:Notification>

重点就是,Action="Clear",但要注意,磁贴正面的背景图不能清除。

 

好,再来新建一个WP应用,这回要做客户端。

直接新建即可,XAML文档不用改,因为我们不需要界面设计了,直打开后台代码吧。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Net;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Animation;
using System.Windows.Shapes;
using Microsoft.Phone.Controls;

using Microsoft.Phone.Notification;

namespace WPClient
{
    public partial class MainPage : PhoneApplicationPage
    {
        // 构造函数
        public MainPage()
        {
            InitializeComponent();
            HttpNotificationChannel Channel = null;
            // 通道名,随便弄一个,不要与其它应用程序重复
            string Channel_Name = "TileNoftification";
            // 在现有的通道里面找找,看能不能找着?
            Channel = HttpNotificationChannel.Find(Channel_Name);
            if (Channel == null)
            {
                // 找不到,那就新建一个呗
                Channel = new HttpNotificationChannel(Channel_Name);
                // 打开通道前要先注册事件处理,为什么?自己想一下吧
                // 就是因为ChannelUriUpdated事件,如不这样,
                // 当第一次获取URI时你就得不到更新通知
                Channel.ChannelUriUpdated += new EventHandler<NotificationChannelUriEventArgs>(Channel_ChannelUriUpdated);
                // 出事了,总得向上级汇报一下吧?
                Channel.ErrorOccurred += new EventHandler<NotificationChannelErrorEventArgs>(Channel_ErrorOccurred);
                // 登记注册完毕,着手办喜事
                Channel.Open();
                // 办喜事别忘记了发请帖啊,调用BindToShellTile
                Channel.BindToShellTile();
            }
            else
            {
                // 如果找到了通道,还是要注册一下事件
                // 老夫妻也可以再拍一回婚纱照吧?
                Channel.ChannelUriUpdated+=new EventHandler<NotificationChannelUriEventArgs>(Channel_ChannelUriUpdated);
                Channel.ErrorOccurred+=new EventHandler<NotificationChannelErrorEventArgs>(Channel_ErrorOccurred);
                // 把住址告诉人家,相册做好之后,数码冲印店送货上门
                System.Diagnostics.Debug.WriteLine("URI: {0}", Channel.ChannelUri.ToString());
            }
        }

        void Channel_ErrorOccurred(object sender, NotificationChannelErrorEventArgs e)
        {
            // 向上级汇报一下错误
            Dispatcher.BeginInvoke(() =>
                {
                    MessageBox.Show(e.Message);
                });
        }

        void Channel_ChannelUriUpdated(object sender, NotificationChannelUriEventArgs e)
        {
            // 搬家了记得通知一下大家新住址
            Dispatcher.BeginInvoke(() =>
                {
                    System.Diagnostics.Debug.WriteLine("URI: {0}", e.ChannelUri.ToString());
                });
        }
    }
}


 

先动行WP端,当然,同时运行两个都可以了。

在“输出”窗口中,把这个URI复制到服务器端的网页上。

 

接着,按模拟器的“开始”按钮,来到“开始”屏幕,向右滑动,看到应用程序列表,在本应用程序上长按,从弹出的菜单中选“固定到开始屏幕”.

 

然后,回到服务器端页面,填好所有参数,点击“发送”。看结果。

 

 

都看到效果了?

图片可以自己准备,png格式,173*173,随便用画图工具搞两下就行了,只是为了测试,把图片加到项目后,设置以下属性就行了。

 

OK,大家照着上面的多练几次,一定有感觉的,我不想讲理论的东西,因为没什么用。

目录
相关文章
|
1月前
|
数据可视化 数据库 C++
Qt 5.14.2揭秘高效开发:如何用VS2022快速部署Qt 5.14.2,打造无与伦比的Windows应用
Qt 5.14.2揭秘高效开发:如何用VS2022快速部署Qt 5.14.2,打造无与伦比的Windows应用
|
17天前
|
监控 安全 API
7.3 Windows驱动开发:内核监视LoadImage映像回调
在笔者上一篇文章`《内核注册并监控对象回调》`介绍了如何运用`ObRegisterCallbacks`注册`进程与线程`回调,并通过该回调实现了`拦截`指定进行运行的效果,本章`LyShark`将带大家继续探索一个新的回调注册函数,`PsSetLoadImageNotifyRoutine`常用于注册`LoadImage`映像监视,当有模块被系统加载时则可以第一时间获取到加载模块信息,需要注意的是该回调函数内无法进行拦截,如需要拦截则需写入返回指令这部分内容将在下一章进行讲解,本章将主要实现对模块的监视功能。
33 0
7.3 Windows驱动开发:内核监视LoadImage映像回调
|
4月前
|
监控 安全 API
7.2 Windows驱动开发:内核注册并监控对象回调
在笔者上一篇文章`《内核枚举进程与线程ObCall回调》`简单介绍了如何枚举系统中已经存在的`进程与线程`回调,本章`LyShark`将通过对象回调实现对进程线程的`句柄`监控,在内核中提供了`ObRegisterCallbacks`回调,使用这个内核`回调`函数,可注册一个`对象`回调,不过目前该函数`只能`监控进程与线程句柄操作,通过监控进程或线程句柄,可实现保护指定进程线程不被终止的目的。
30 0
7.2 Windows驱动开发:内核注册并监控对象回调
|
4月前
|
监控 安全 API
7.6 Windows驱动开发:内核监控FileObject文件回调
本篇文章与上一篇文章`《内核注册并监控对象回调》`所使用的方式是一样的都是使用`ObRegisterCallbacks`注册回调事件,只不过上一篇博文中`LyShark`将回调结构体`OB_OPERATION_REGISTRATION`中的`ObjectType`填充为了`PsProcessType`和`PsThreadType`格式从而实现监控进程与线程,本章我们需要将该结构填充为`IoFileObjectType`以此来实现对文件的监控,文件过滤驱动不仅仅可以用来监控文件的打开,还可以用它实现对文件的保护,一旦驱动加载则文件是不可被删除和改动的。
29 1
7.6 Windows驱动开发:内核监控FileObject文件回调
|
4月前
|
监控 安全 API
6.9 Windows驱动开发:内核枚举进线程ObCall回调
在笔者上一篇文章`《内核枚举Registry注册表回调》`中我们通过特征码定位实现了对注册表回调的枚举,本篇文章`LyShark`将教大家如何枚举系统中的`ProcessObCall`进程回调以及`ThreadObCall`线程回调,之所以放在一起来讲解是因为这两中回调在枚举是都需要使用通用结构体`_OB_CALLBACK`以及`_OBJECT_TYPE`所以放在一起来讲解最好不过。
44 1
6.9 Windows驱动开发:内核枚举进线程ObCall回调
|
4月前
|
监控 安全 API
6.8 Windows驱动开发:内核枚举Registry注册表回调
在笔者上一篇文章`《内核枚举LoadImage映像回调》`中`LyShark`教大家实现了枚举系统回调中的`LoadImage`通知消息,本章将实现对`Registry`注册表通知消息的枚举,与`LoadImage`消息不同`Registry`消息不需要解密只要找到`CallbackListHead`消息回调链表头并解析为`_CM_NOTIFY_ENTRY`结构即可实现枚举。
50 1
6.8 Windows驱动开发:内核枚举Registry注册表回调
|
4月前
|
存储 API 开发者
6.7 Windows驱动开发:内核枚举LoadImage映像回调
在笔者之前的文章`《内核特征码搜索函数封装》`中我们封装实现了特征码定位功能,本章将继续使用该功能,本次我们需要枚举内核`LoadImage`映像回调,在Win64环境下我们可以设置一个`LoadImage`映像加载通告回调,当有新驱动或者DLL被加载时,回调函数就会被调用从而执行我们自己的回调例程,映像回调也存储在数组里,枚举时从数组中读取值之后,需要进行位运算解密得到地址。
32 1
6.7 Windows驱动开发:内核枚举LoadImage映像回调
|
4月前
|
监控 安全 API
7.5 Windows驱动开发:监控Register注册表回调
在笔者前一篇文章`《内核枚举Registry注册表回调》`中实现了对注册表的枚举,本章将实现对注册表的监控,不同于32位系统在64位系统中,微软为我们提供了两个针对注册表的专用内核监控函数,通过这两个函数可以在不劫持内核API的前提下实现对注册表增加,删除,创建等事件的有效监控,注册表监视通常会通过`CmRegisterCallback`创建监控事件并传入自己的回调函数,与该创建对应的是`CmUnRegisterCallback`当注册表监控结束后可用于注销回调。
45 0
7.5 Windows驱动开发:监控Register注册表回调
|
4月前
|
存储 安全 数据安全/隐私保护
3.2 Windows驱动开发:内核CR3切换读写内存
CR3是一种控制寄存器,它是CPU中的一个专用寄存器,用于存储当前进程的页目录表的物理地址。在x86体系结构中,虚拟地址的翻译过程需要借助页表来完成。页表是由页目录表和页表组成的,页目录表存储了页表的物理地址,而页表存储了实际的物理页框地址。因此,页目录表的物理地址是虚拟地址翻译的关键之一。在操作系统中,每个进程都有自己的地址空间,地址空间中包含了进程的代码、数据和堆栈等信息。为了实现进程间的隔离和保护,操作系统会为每个进程分配独立的地址空间。在这个过程中,操作系统会将每个进程的页目录表的物理地址存储在它自己的CR3寄存器中。当进程切换时,操作系统会修改CR3寄存器的值,从而让CPU使用新的页
50 0
3.2 Windows驱动开发:内核CR3切换读写内存
|
4月前
|
编译器 C++ Windows
9.4 Windows驱动开发:内核PE结构VA与FOA转换
本章将继续探索内核中解析PE文件的相关内容,PE文件中FOA与VA,RVA之间的转换也是很重要的,所谓的FOA是文件中的地址,VA则是内存装入后的虚拟地址,RVA是内存基址与当前地址的相对偏移,本章还是需要用到`《内核解析PE结构导出表》`中所封装的`KernelMapFile()`映射函数,在映射后对其PE格式进行相应的解析,并实现转换函数。
40 0
9.4 Windows驱动开发:内核PE结构VA与FOA转换