C# Winform消息通知之系统本地通知local toast notification

目录
  • 引言
  • toast 通知的结构
  • 发送本地toast通知的操作步骤
    • 安装NuGet包Microsoft.Toolkit.Uwp.Notifications
    • 通知的发送(文本通知)
    • 处理点击通知的操作
    • 通知的卸载
    • 设置通知的过期时间
    • .NET应用使用Toast Notifications(.NET5+)
  • 添加图像
    • 使用http图像
    • 内联图像和主图
    • 徽标和剪裁(圆形图片)
    • UWP使用http图片
  • 替换或删除指定通知
    • 为 toast 设置主键
    • 根据Tag和Group删除或替换toast
  • 清除通知

引言

C#应用通过 Microsoft.Toolkit.Uwp.Notifications NuGet包可以很方便的发送本地通知(Windows 10或Windows 11 的 Toast 通知,之前系统版本未提供此类通知),适用于所有类型的应用(WPF、UWP、WinForms、控制台【实际使用各自有着不同限制】)。

后面会介绍使用任务托盘显示系统通知,从而不用使用Nuget包。

toast 通知的结构

Toast 内容的核心组件有

  • launch:定义当用户单击你的 Toast 时将传回应用的参数,允许深层链接到 Toast 所显示的正确内容。
  • visual:toast 的可视部分,包括带有文本和图像的通用绑定。
  • actions:toast 的交互部分,包括输入和操作。
  • audio:控制向用户显示 Toast 时播放的音频。

Toast 内容的可视化表示形式:

Toast 内容介绍对toast各种应用场景进行了介绍,包括文本、文本行数、图片、徽章、时间戳、进度条、按钮、输入、音频等。

发送本地toast通知的操作步骤

新建项目NotificationLocalToast

安装NuGet包Microsoft.Toolkit.Uwp.Notifications

推荐 7.0 及以上版本。

如果.NET Framework的桌面应用仍使用 packages.config 管理包,需要将其迁移到PackageReference,否则不会正确引用Windows SDK。

迁移方法是:在项目中,右键单击“引用”,然后单击“将 packages.config 迁移到 PackageReference”。需要Visual Studio 2017 15.7 及更高版本才支持。

具体参见从 packages.config 迁移到 PackageReference

注:迁移到PackageReference后,需要重启VS打开项目,否则无法正确的using引用到包库。

通知的发送(文本通知)

new ToastContentBuilder()
    .AddText("CodeMissing发来一条消息") // 标题文本
    .AddText("请检查消息内容,并及时处理")
    .Show(); // 7.0以上才提供Show方法

点击运行并发送消息,将在屏幕右下角看到toast通知

多次发送消息,会依照顺序各自依次显示(等待上一个进入操作中心或被关闭)。

处理点击通知的操作

当用户点击通知后,则会在后台线程调用ToastNotificationManagerCompat.OnActivated事件,在程序的该事件中处理交互操作。

如果应用程序关闭,将会启动exe应用(并且 ToastNotificationManagerCompat.WasCurrentProcessToastActivated()返回true,表示重新启动了进程),然后再在后台线程上调用 ToastNotificationManagerCompat.OnActivated 事件。

在主窗体的构造函数中处理该事件(如果不需要执行UI线程的操作,直接在Program.cs的Main函数入口处添加此监听事件即可)

public MainForm()
{
    InitializeComponent();
    // 监听通知激活(点击)
    ToastNotificationManagerCompat.OnActivated += toastArgs =>
    {
        // 通知参数
        ToastArguments args = ToastArguments.Parse(toastArgs.Argument);
        // 获取任何用户输入
        ValueSet userInput = toastArgs.UserInput;
        BeginInvoke(new Action( delegate
        {
            // TODO: UI线程的操作
            MessageBox.Show("Toast被激活(点击),参数是: " + toastArgs.Argument);
        }));
    };
}

当点击通知后,会显示对话框。

WPF中的(在入口程序中的)异步UI调用要通过Application.Current.Dispatcher.Invoke,它是位于System.Windows下的Application。需要添加引用PresentationFramework

// Need to dispatch to UI thread if performing UI operations
Application.Current.Dispatcher.Invoke(delegate
{
    // TODO: Show the corresponding content
    MessageBox.Show("Toast activated. Args: " + toastArgs.Argument);
});

通知的卸载

如果应用有卸载程序,应该在卸载过程中调用ToastNotificationManagerCompat.Uninstall();

如果应用是不安装的“可移植应用”,则可根据需要在退出时调用(除非想要在退出后仍然保留通知)。

ToastNotificationManagerCompat.Uninstall()在执行后,会将通知面板中相关的信息清理掉。

卸载方法将清理任何计划通知和当前通知,删除任何关联的注册表值,并删除库创建的任何关联的临时文件。

设置通知的过期时间

忽略的toast通知会转到操作中心(通知中心),后续仍可以查看。

但是,通常消息都有一定的期限,过期后则不应该继续显示或保留,本地 toast 通知的默认和最长过期时间为 3 天。

下面设置过期时间为2天:

new ToastContentBuilder()
    .Show(toast =>
    {
        toast.ExpirationTime = DateTime.Now.AddDays(2);
    });

.NET应用使用Toast Notifications(.NET5+)

.NET中如果使用Microsoft.Toolkit.Uwp.Notifications,必须指定Windows TFM,且要指定window版本,至少为net6.0-windows10.0.17763.0或更高。否则,将报错找不到Show()方法。

  • 指定Windows TFM

右键项目,编辑项目文件,将TargetFramework指定为如下:

<TargetFramework>net6.0-windows10.0.17763.0</TargetFramework>

通常,指定TFM后,启动调试会报错没有xxx的目标,确保已运行还原...等错误。

解决办法是:清理项目,并重新生成一次即可。

有时候还会报错net6.0-windows10.0.17763.0.Net框架版本不一致,需要修改为net5.0-windows10.0.17763.0。或者改为net5后生成无错再改回net6;或者直接清理项目并重新生成一次。

添加图像

使用http图像

目前仅仅具有Internet功能的UWP/MSIX/sparse应用才支持http的图像【sparse,松散或稀疏应用】。

UWP/MSIX开发在Package.appxmanifest文件中用于指定Internet的能力(默认已经开启)

其他应用,比如Winform、WPF等必须将图像下载到本地,通过本地路径引用。

并且web图片引用有200KB的限制,具体参见官方文档介绍。不过有个不太清楚的地方:app's package是什么?又如何从其中获取图片?

Images can be used from the app's package, the app's local storage, or from the web. As of the Fall Creators Update, web images can be up to 3 MB on normal connections and 1 MB on metered connections. On devices not yet running the Fall Creators Update, web images must be no larger than 200 KB.

尝试通过资源pack的方式获取图片,未成功,比如WPF图片中可以使用的"pack://application:,,,/Resources/CSharp.png"

内联图像和主图

下载图片到本地,并添加到项目中,设置图片属性:复制到输出目录“总是”或“较新复制”、生成操作“无”,确保图片生成到exe程序路径。

首先获取img文件的完整路径,并创建Uri,通过AddInlineImageAddHeroImage添加为内联图像和主图。

  • 使用file:///构建uri
var imgFileFullPath = Path.GetFullPath("Resources/CSharp.png");
var fileUriString = $"file:///{imgFileFullPath}";
var imgUri = new Uri(fileUriString);
new ToastContentBuilder()
    .AddArgument("action", "viewConversation") // 添加相关参数
    .AddArgument("conversationId", 9813)
    .AddText("CodeMissing发来一张图片") // 标题文本
    .AddText("这是C#的图片")
    .AddInlineImage(imgUri) // 内联
    .AddHeroImage(imgUri) // 主图
    .Show();

注意本地文件的Uri访问协议:"file:///FileFullPath"

  • 直接使用文件路径创建Uri

new Uri(localImgPath)也可以创建uri。

var imgFileFullPath = Path.GetFullPath("Resources/CSharp.png");
new ToastContentBuilder()
    .AddText("这是C#的图片")
    .AddAppLogoOverride(new Uri(imgFileFullPath), ToastGenericAppLogoCrop.Circle)
    .Show();

徽标和剪裁(圆形图片)

new ToastContentBuilder()
    // ....
    .AddAppLogoOverride(imgUri, ToastGenericAppLogoCrop.Circle)
    .Show();

UWP使用http图片

new ToastContentBuilder()
    .AddArgument("action", "viewConversation") // 添加相关参数
    .AddArgument("conversationId", 9813)
    .AddText("CodeMissing发来一张图片") // 标题文本
    .AddText("这是C#的图片")
    .AddInlineImage(new Uri("https://www.vippng.com/png/detail/398-3984434_c-programming-png.png"))
    .AddAppLogoOverride(new Uri("https://www.vippng.com/png/detail/398-3984434_c-programming-png.png"), ToastGenericAppLogoCrop.Circle)
    .Show();

图片来源于网络

替换或删除指定通知

替换或移除toast通知需要使用Tag属性(以及Group属性),两者构成toast的主键。

为 toast 设置主键

new ToastContentBuilder()
    .AddText("我是含有Tag和Group的消息")
    .Show(toast =>
    {
        toast.Tag = "codemissing101";
        toast.Group = "codemissing";
    });

根据Tag和Group删除或替换toast

通过使用相同的Tag和Group再次发送消息,即可以实现替换。

new ToastContentBuilder()
    .AddText("我是替换的消息")
    .Show(toast =>
    {
        toast.Tag = "codemissing101";
        toast.Group = "codemissing";
    });

History.Remove 或者 RemoveGroup 实现移除

ToastNotificationManagerCompat.History.Remove("codemissing101", "codemissing");

如果只有Tag则只指定Tag,如果Tag和Group都存在,则两者都需要指定才能匹配。

清除通知

ToastNotificationManagerCompat.History.Clear();

参考

从 C# 应用发送本地 toast 通知

以上就是C# Winform消息通知之系统本地通知local toast notification的详细内容,更多关于C# Winform系统本地消息通知的资料请关注我们其它相关文章!

(0)

相关推荐

  • C# Winform实现进度条显示

    本文实例为大家分享了C# Winform实现进度条显示的具体代码,供大家参考,具体内容如下 创建一个窗体,命名为StartForm 添加一个timer控件并更改名字为timerStart 添加一个ProgressBar控件,并调整一下属性: StartForm窗体的代码: using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Draw

  • C# Winform中DataGridView导出为Excel的实现示例

    目录 1.前言 2.效果展示 3.详细步骤 3.1 添加NPOI和NPOI.Excel包 3.2 创建NPOIHelper类 3.3 给画面添加SaveFileDialog 3.4 引入命名空间 3.5 给按钮添加click事件 4. 成功 5.写在最后 1.前言 话不多说,跟着我的步骤保证你也能成功,下面直接开始! 2.效果展示 导出前 导出后 3.详细步骤 下面是详细操作步骤,请跟着我的步伐,一步一步进行操作,保证你能够导出成功! 3.1 添加NPOI和NPOI.Excel包 首先请请确定你

  • C# Winform 分页功能的实现

    首先创建一个用户控件 如下图 用到的控件 label.button.TextBox 内部代码如下 #region 分页字段和属性 private int pageIndex = 1; /// <summary> /// 当前页数 /// </summary> public virtual int PageIndex { get { return pageIndex; } set { pageIndex = value; } } private int pageSize = 100;

  • C# Winform 实现TCP发消息

    服务端: 窗体 代码: using System; using System.Collections.Generic; using System.IO; using System.Net; using System.Net.Sockets; using System.Text; using System.Threading; using System.Windows.Forms;   namespace SocketStudy {     public partial class Form1 :

  • C# winform 请求http的实现(get,post)

    目录 一:.Net中有两个类 HttpWebRequest 和HttpWebResponse 类来实现Http的请求 二: HTTP定义了与服务器交互的不同方法 POST与GET的差异 两个简单的Get请求和Post请求 1.Get请求 2.Post请求 一个小测试 一:.Net中有两个类 HttpWebRequest 和HttpWebResponse 类来实现Http的请求 实现步骤: 1.通过WebRequest类创建一个HttpWebRequest的对象,该对象可以包含Http请求信息.2

  • C# WinForm制作登录界面的实现步骤

    在[解决方案资源管理器]中找到Form1.cs,单击,快捷键F2重命名为“Login.cs”(命名很重要,不然之后项目多了根据不知道哪个项目的内容是什么) 对窗体[Text]属性.[size]属性和[FormBoardStyle]属性进行修改 添加一个新的窗体 Ctrl+Shift+A,在弹出框中选择[Windows窗体],命名为main.cs 取消登录界面最大化最小化关闭按钮在父窗体菜单栏上显示最大化:MaximizeBox,最小化:MinimizeBox如果设置一个为False 的时候会显示

  • C# Winform消息通知之系统本地通知local toast notification

    目录 引言 toast 通知的结构 发送本地toast通知的操作步骤 安装NuGet包Microsoft.Toolkit.Uwp.Notifications 通知的发送(文本通知) 处理点击通知的操作 通知的卸载 设置通知的过期时间 .NET应用使用Toast Notifications(.NET5+) 添加图像 使用http图像 内联图像和主图 徽标和剪裁(圆形图片) UWP使用http图片 替换或删除指定通知 为 toast 设置主键 根据Tag和Group删除或替换toast 清除通知 引

  • C# Winform消息通知系统托盘气泡提示框ToolTip控件

    目录 气球状提示框的介绍和系统通知变化 消息通知的提示 ShowBalloonTip()方法及指定消息类型 NotifyIcon属性设置消息 BalloonTipIcon不同的消息类型 ToolTipText属性 ToolTip提示控件显式或主动的提示消息 一个ToolTip同时为多个控件设置提示 显式设置ToolTip 气球状提示框的介绍和系统通知变化 NotifyIcon控件表示系统右下角任务栏上的托盘图标,其ShowBalloonTip方法用于显示任务栏中一定时间的具有指定标题.消息内容和

  • iOS推送之本地通知UILocalNotification

    摘要: Notification是智能手机应用编程中非常常用的一种传递信息的机制,而且可以非常好的节省资源,不用消耗资源来不停地检查信息状态(Pooling),在iOS下应用分为两种不同的Notification种类,本地和远程.本地的Notification由iOS下NotificationManager统一管理,只需要将封装好的本地Notification对象加入到系统Notification管理机制队列中,系统会在指定的时间激发将本地Notification,应用只需设计好处理Notifi

  • Android使用JobScheduler定期推送本地通知实例代码

    Android5.0之后提供了JobService和JobScheduler,用于在稍后的某个时间点或者当满足某个特定的条件时执行一个任务.使用JobScheduler,我们可以在用户一段时间没有使用我们的app的情况下,推送本地通知来提高app的用户留存率.废话不多说,上代码: 先在app的MainActivity启动时用JobScheduler来schedule一个job.注意在onCreate中我们把用户启动app的时间记录在了shared preference里面: @Override

  • C#实现系统托盘通知的方法

    本文实例讲述了C#实现系统托盘通知的方法.分享给大家供大家参考.具体实现方法如下: namespace WindowsApplication1 { public partial class Form1 : Form { public Form1() { InitializeComponent(); } private void timer1_Tick(object sender, EventArgs e) { this.Activate(); } private void Form1_Resiz

  • 如何使用Spring AOP的通知类型及创建通知

    写在最前端 1.SpringAOP中共有六种通知类型,只要我们自定义一个类实现对应的接口,它们全都是org.springframework.aop包中的. 2.AOP的连接点可以是方法调用.方法调用本身.类初始化.对象实例化时,但是SpringAOP中全是方法调用,更简单,也最实用 通知名称 接口 前置通知 org.springframework.aop.MethodBeforeAdvice 后置返回通知 org.springframework.aop.AfterReturningAdvice

  • 获取当前系统本地时间,精确到毫秒的实例

    实例如下: #include <sys/timeb.h> #include <chrono> char* cur_time_c(char strDateTime[32]) { struct timeb tp_cur; ftime(&tp_cur); struct tm btm; #ifdef WIN32 localtime_s(&btm, &tp_cur.time); #else localtime_r(&tp_cur.time, &btm)

  • C# WinForm应用程序降低系统内存占用方法总结

    背景: 微软的 .NET FRAMEWORK 现在可谓如火如荼了.但是,.NET 一直所为人诟病的就是"胃口太大",狂吃内存,虽然微软声称 GC 的功能和智能化都很高,但是内存的回收问题,一直存在困扰,尤其是 winform 程序,其主要原因是因为.NET程序在启动时,是需要由JIT动态编译并加载的,这个加载会把所有需要的资源都加载进来,很多资源是只有启动时才用的. 以XP 系统为例子,程序启动后,打开任务管理器,会看到占用的内存量比较大,你把程序最小化,会发现该程序占用的内存迅速减小

  • 将Python中的数据存储到系统本地的简单方法

    有很多时候,我们会在python的运行过程中得到一些重要的变量,比如一个数据量很庞大的dict.而且,后面的某些程序也会用到这个dict,那么我们就最好把它存储到本地来,然后下次调用的时候,先读取本地的文件,导入到字典类型中,调用即可.这样就免去了重新学习这个字典的过程.那么在python中如何把数据存储到本地呢? 我们用到的是python中的pickle模块. 如下: import pickle data1 = {'a': [1, 2.0, 3, 4+6j], 'b': ('string',

  • 详解iOS本地推送与远程推送

    一.简介 分为本地推送和远程推送2种.可以在应用没有打开甚至手机锁屏情况下给用户以提示.它们都需要注册,注册后系统会弹出提示框(如下图)提示用户是否同意,如果同意则正常使用:如果用户不同意则下次打开程序也不会弹出该提示框,需要用户到设置里面设置.一共有三种提示类型: UIUserNotificationTypeBadge:应用图标右上角的信息提示 UIUserNotificationTypeSound:播放提示音 UIUserNotificationTypeAlert:提示框 二.本地推送 1

随机推荐