文档目录

本节内容:

  • 简介

    • 发送模式
    • 通知类型
    • 通知数据
    • 通知重要性
    • 关于通知持久化
  • 订阅通知
  • 发布通知
  • 用户通知管理器
  • 实时通知
    • 客户端
  • 通知存储
  • 通知定义

简介

通知用来告知用户系统里特定的事件发生了,ABP提供一个发布/订阅,它基于实时通知基础框架。

发送模式

有两种方式可以发送通知给用户:

  • 用户订阅一个特定的通知类型,然后我们发布一个此类型的通知,它会分发给所有订阅的用户,这就是发布/订阅模式。
  • 我们可以直接发送一个通知给目标用户(users)。

通知类型

有两种通知类型:

  • 一般通知:任意的通知类型,“如果一个用户发送一个好友请求,那么通知我”就是一个此类型的通知。
  • 实体通知:与一个特定的实体关联,“如果一个用户给这张(photo)图片发了评论,那么通知我”就是一个基于实体的通知,因为它与一个特定的photo实体关联,用户可能想为某些图片发出通知,而不是所有图片。

通知数据

一个通知通常包含一个通知数据,例如:“如果一个用户发送一个好友请求,那么通知我”该通知可能有两个数据属性:发送者名字(谁发送了这个好友请求)和备注(发送者写在请求里的信息),很明显,通知数据类型与通知类型是紧密联系的,不同的通知类型有不同的数据类型。

通知数据是可选的,有些通知可能就不需要数据。ABP已经提供足以满足大部分情况的预定义通知数据类型,简单的信息可以用MessageNotificationData,本地化和可参数化的通知信息可以使用LocalizableMessageNotificationData,我们会在以后章节看到使用示例。

通知重要性

通知重要性有5个级别,定义在NotificationSeverity枚举里:Info,Success,Warn,ErrorFatal,默认为Info

关于通知持久化

查看通知存储小节获取更多信息。

订阅通知

INotificationSubscriptionManager提供了用来订阅通知的API,例如:

public class MyService : ITransientDependency
{private readonly INotificationSubscriptionManager _notificationSubscriptionManager;public MyService(INotificationSubscriptionManager notificationSubscriptionManager){_notificationSubscriptionManager = notificationSubscriptionManager;}//订阅一个一般通知public async Task Subscribe_SentFrendshipRequest(int? tenantId, long userId){await _notificationSubscriptionManager.SubscribeAsync(new UserIdentifier(tenantId, userId), "SentFrendshipRequest");    }//订阅一个实体通知public async Task Subscribe_CommentPhoto(int? tenantId, long userId, Guid photoId){await _notificationSubscriptionManager.SubscribeAsync(new UserIdentifier(tenantId, userId), "CommentPhoto", new EntityIdentifier(typeof(Photo), photoId));   }
} 

首先,我们注入INotificationSubscriptionManager,第一方法订阅一个一般通知,当某人发送一个好友请求时,用户会收到通知,第二个订阅一个与特定实体(Photo)相关的通知,当某人给特定照片写评论后,用户会收到通知。

每一个通知类型都应当有一个唯一的名称(如示例中的SentFrendshipRequest 和CommentPhoto)。

INotificationSubscriptionManager还有UnsubscribeAsync,IsSubscribedAsyncGetSubscriptionsAsyn等方法来管理订阅。

发布通知

INotificationPublisher用来发布通知,例如:

public class MyService : ITransientDependency
{private readonly INotificationPublisher _notiticationPublisher;public MyService(INotificationPublisher notiticationPublisher){_notiticationPublisher = notiticationPublisher;}//发送一个一般通知给一个特定用户public async Task Publish_SentFrendshipRequest(string senderUserName, string friendshipMessage, UserIdentifier targetUserId){await _notiticationPublisher.PublishAsync("SentFrendshipRequest", new SentFrendshipRequestNotificationData(senderUserName, friendshipMessage), userIds: new[] { targetUserId });}//发送一个实体通知给一个特定用户public async Task Publish_CommentPhoto(string commenterUserName, string comment, Guid photoId, UserIdentifier photoOwnerUserId){await _notiticationPublisher.PublishAsync("CommentPhoto", new CommentPhotoNotificationData(commenterUserName, comment), new EntityIdentifier(typeof(Photo), photoId), userIds: new[] { photoOwnerUserId });}//发送一个一般通知给所有当前租户(在会话里)里的订阅它的用户public async Task Publish_LowDisk(int remainingDiskInMb){//Example "LowDiskWarningMessage" content for English -> "Attention! Only {remainingDiskInMb} MBs left on the disk!" var data = new LocalizableMessageNotificationData(new LocalizableString("LowDiskWarningMessage", "MyLocalizationSourceName"));data["remainingDiskInMb"] = remainingDiskInMb;await _notiticationPublisher.PublishAsync("System.LowDisk", data, severity: NotificationSeverity.Warn);   }
} 

在第一个例子里, 我们发布一个通知给一个单独的用户,SentFrendshipRequestNotificationData应该继承于NotificationData,如下所示:

[Serializable]
public class SentFrendshipRequestNotificationData : NotificationData
{public string SenderUserName { get; set; }public string FriendshipMessage { get; set; }public SentFrendshipRequestNotificationData(string senderUserName, string friendshipMessage){SenderUserName = senderUserName;FriendshipMessage = friendshipMessage;}
} 

第二个例子,我们发送一个通知给一个特定用户并传递一个特定实体,通知数据类不需要可序列化(因为默认使用JSON序列器),但建议把它标记为可序列化,因为你可能会在不同应用中移动而且你将来也可能想使用二进制可序列化器。当然如前面所述,通知数据是可选的,并不是所有的通知都需要它。

注意:如果我们发布一个通知给一个特定用户,那么这个用户不需要订阅这个通知。

第三个例子,我们没有定义一个专用的通知数据类,而是直接使用内容的LocalizableMessageNotificationData类,并使用基于字典的数据,并发布一个“Warn”通知,LocalizableMessageNotificationData可以存储基于字典的任意数据(如果自定义的通知数据类继承于NotificationData类,那么也可以这么用),我们使用“remainingDiskInMb”作为本地化参数,本地化信息可以包含这些参数(如示例中的“Attention!Only{remainingDiskInMb} MBs left on the disk!”),我们在客户端小节里可以看到如果本地化。

用户通知管理器

IUserNotificationManager用来管理用户的通知,它get,update或delete一个用户的通知,你可以用它为你的应用准备一个通知列表页面。

实时通知

虽然你可以用IUserNotificationManager来查询通知,但我们通常想推送一个实时通知到客户端。

通知系统使用IRealTimeNotifier来发送实时通知给用户,这可以使用任何类型的实时通讯系统实现,我们可以用一个单独实现的SignalR包,启动模板已经安装了SignalR,查看SignalR集成文档获取更多信息。

注意:通知系统用一个后台作业异步调用IRealTimeNotifier,所以,通知可能会有一点点的延迟。

客户端

当接收到一个实时的通知,ABP在客户端触发一个全局事件,你可以用如下的方式注册来得到通知:

abp.event.on('abp.notifications.received', function (userNotification) {console.log(userNotification);
}); 

每接收到一个实时通知,都会触发abp.notifications.received事件事件,你可以像上面那样注册该事件来获取通知,查看javascript事件总线文档获取事件更多信息。一个收到的“System.LowDisk”通知的Json示例:

{"userId": 2,"state": 0,"notification": {"notificationName": "System.LowDisk","data": {"message": {"sourceName": "MyLocalizationSourceName","name": "LowDiskWarningMessage"},"type": "Abp.Notifications.LocalizableMessageNotificationData","properties": {"remainingDiskInMb": "42"}},"entityType": null,"entityTypeName": null,"entityId": null,"severity": 0,"creationTime": "2016-02-09T17:03:32.13","id": "0263d581-3d8a-476b-8e16-4f6a6f10a632"},"id": "4a546baf-bf17-4924-b993-32e420a8d468"
}

在这个对象里:

  • userId:当前用户Id,通常你不需要这个因为你知道当前的户。
  • state:UserNotificationState的枚举值,0:Unread(未读),1:Read(已读)。
  • notification:通知明细:
    • notificationName:这个通知的唯一名称(发布时也用这个名称)。
    • data:通知数据,在这个示例里,我们使用LocalizableMessageNotificationData(和之前发布的示例一致):
      • message:本地化的消息信息,我们可以用sourceName和name来本地化界面上的消息。
      • type:通知数据类型,全类型名,包含命名空间,当在处理通知数据时,我们可以检查这个类型。
      • properties:基于字典的自定义属性。
    • entityType、entityTypeName和entityId:实体信息(如果这是一个实体实时通知)
    • severity:一个NotificationSeverity枚举值,0:Info,1:Success,2:Warn,3:Error,4:Fatal。
    • id:通知Id。
  • id:用户通知Id。

当然,你不会只是日志这个通知,你可以使用通知数据把通知信息显示给用户,例如:

abp.event.on('abp.notifications.received', function (userNotification) {if (userNotification.notification.data.type === 'Abp.Notifications.LocalizableMessageNotificationData') {var localizedText = abp.localization.localize(userNotification.notification.data.message.name,userNotification.notification.data.message.sourceName);$.each(userNotification.notification.data.properties, function (key, value) {localizedText = localizedText.replace('{' + key + '}', value);});alert('New localized notification: ' + localizedText);} else if (userNotification.notification.data.type === 'Abp.Notifications.MessageNotificationData') {alert('New simple notification: ' + userNotification.notification.data.message);}
});

为了能处理通知数据,我们应当检查这个数据类型,这个例子简单的从通知数据里获取消息,如果是本地化的消息(LocalizableMessageNotificationData),我们本地化这个消息并替换参数, 如果是简单消息(MessageNotificationData),我们直接获取这个消息,当然,在真实的项目里,我们不会使用alert函数,我们可以使用abp.notify api来显示良好的UI通知。

如果你想实现上面这样的逻辑,有一个更容易且富有弹性的方式,当接收到一个推送的通知,你只需要一行代码来显示UI通知:

abp.event.on('abp.notifications.received', function (userNotification) {abp.notifications.showUiNotifyForUserNotification(userNotification);
});

这显示一个UI通知,如下所示(上面描述的推送的System.LowDisk通知):

它可工作于内容的通知数据类型(LocalizableMessageNotificationData和MessageNotificationData),如果你是自定义的通知数据类型,那么你应该像下面这样注册数据格式:

abp.notifications.messageFormatters['MyProject.MyNotificationDataType'] = function(userNotification) {return ...; //此处格式化并返回消息
};

因此,showUiNotifyForUserNotification可以为你的数据类型创建显示的消息,如果你只是需要格式的消息,你可以直接使用abp.notifications.getFormattedMessageFromUserNotification

(userNotification), 它内部被showUiNotifyForUserNotification。

启动模板包含了当接收到一个推送信息后显示UI通知的代码。

通知存储

通知系统使用INotificationStore来持久化通知,该接口实现后才能使通知系统正常工作,你可以自己实现它或使用已经实现它的module-zero。

通知定义

你不需要在用前先定义一个通知,你只管使用任何通知名称而无需定义,但是,定义它可能给你带来额外的好处,例如,定义后你可以在你的应用里检查所有的通知。鉴于这种情况,我们可以为我们的模块定义一个通知供应器,如下所示:

public class MyAppNotificationProvider : NotificationProvider
{public override void SetNotifications(INotificationDefinitionContext context){context.Manager.Add(new NotificationDefinition("App.NewUserRegistered",displayName: new LocalizableString("NewUserRegisteredNotificationDefinition", "MyLocalizationSourceName"),permissionDependency: new SimplePermissionDependency("App.Pages.UserManagement")));}
}

“Abp.NewUserRegistered”是这个通知的唯一名称,我们定义了一个可本地化的displayName(当在UI上订阅这个通知时可以显示它),最后,我们声明这个通知只对拥有“App.Pages.UserManagerment”许可的用户可用。

这里还有一些其它参数,你可以在代码中检查它,一个通知的定义只有名称是必需的。

在定义好这个通知供应器后,我们应该在模块的PreInitialize事件里注册它,如下所示:

public class AbpZeroTemplateCoreModule : AbpModule
{public override void PreInitialize(){Configuration.Notifications.Providers.Add<MyAppNotificationProvider>();}//...
}

最后,你可以在你应用中注入并使用INotificationDefinitionManager来获取通知定义,接着你可能想准备一个允许用户自己订阅这些通知的页面。

kid1412附:英文原文:http://www.aspnetboilerplate.com/Pages/Documents/Notification-System

转载于:https://www.cnblogs.com/kid1412/p/6023163.html

ABP文档 - 通知系统相关推荐

  1. 1、ABP 文档介绍

    ABP 文档介绍 Quick Start Getting Started Architecture Infrastructure ABP是一个开源且文档友好的应用程序框架.ABP不仅仅是一个框架,它还 ...

  2. 95. 基于Notes/Domino的文档工作流系统(七)

    本文继续剖析基于Notes/Domino的文档工作流系统的设计和代码,以方便用户能应用和创建自己的工作流.(CSDN的下载资源一旦上传就不能修改,很不方便,现已将下载地址改到GitHub,若发现下载有 ...

  3. Confluence文档共享系统

    Confluence文档共享系统 转载于:https://www.cnblogs.com/cuiqq/p/10966967.html

  4. 针对文档加密系统,如何不破解将文档解密上传

    故事的开端 我们系统是对外系统,里面有表格上传功能,前几天有客户反馈,上传表格无法解析,远程后发现客户的电脑安装有文档加密系统,这种文档加密系统是为了 防止公司内部人员将公司机密的文档信息泄露出去而强 ...

  5. java毕业设计汽车维修管理系统源码+lw文档+mybatis+系统+mysql数据库+调试

    java毕业设计汽车维修管理系统源码+lw文档+mybatis+系统+mysql数据库+调试 java毕业设计汽车维修管理系统源码+lw文档+mybatis+系统+mysql数据库+调试 本源码技术栈 ...

  6. java毕业设计物业管理系统源码+lw文档+mybatis+系统+mysql数据库+调试

    java毕业设计物业管理系统源码+lw文档+mybatis+系统+mysql数据库+调试 java毕业设计物业管理系统源码+lw文档+mybatis+系统+mysql数据库+调试 本源码技术栈: 项目 ...

  7. java毕业设计融呗智慧金融微资讯移动平台服务端源码+lw文档+mybatis+系统+mysql数据库+调试

    java毕业设计融呗智慧金融微资讯移动平台服务端源码+lw文档+mybatis+系统+mysql数据库+调试 java毕业设计融呗智慧金融微资讯移动平台服务端源码+lw文档+mybatis+系统+my ...

  8. java毕业设计物业信息管理系统源码+lw文档+mybatis+系统+mysql数据库+调试

    java毕业设计物业信息管理系统源码+lw文档+mybatis+系统+mysql数据库+调试 java毕业设计物业信息管理系统源码+lw文档+mybatis+系统+mysql数据库+调试 本源码技术栈 ...

  9. java毕业设计企业员工考勤系统的设计与实现源码+lw文档+mybatis+系统+mysql数据库+调试

    java毕业设计企业员工考勤系统的设计与实现源码+lw文档+mybatis+系统+mysql数据库+调试 java毕业设计企业员工考勤系统的设计与实现源码+lw文档+mybatis+系统+mysql数 ...

  10. java毕业设计汽车售后服务管理系统源码+lw文档+mybatis+系统+mysql数据库+调试

    java毕业设计汽车售后服务管理系统源码+lw文档+mybatis+系统+mysql数据库+调试 java毕业设计汽车售后服务管理系统源码+lw文档+mybatis+系统+mysql数据库+调试 本源 ...

最新文章

  1. javascript选择器_如何通过选择正确JavaScript选择器来避免沮丧
  2. 寒假每日一题(入门组)【week7 完结】
  3. 【27前端】base标签带有href属性会让chrome里的svg元素url失效
  4. 面试题11. 旋转数组的最小数字
  5. 首批国产特斯拉车主,被“割了韭菜”
  6. “超人”助阵,IE静音很简单
  7. 2018年最新税收分类编码_这是我们在2018年推出的所有免费编码课程
  8. 《简明python教程》沈洁元
  9. 12款好用超赞的国外搜索资源网站 ,开发者们的标配,你都知道吗?不知道就OUT了...
  10. 知码开门 装箱问题
  11. 智能对话系统原理和实践
  12. 计算机网络统考outlook操作视频,网络教育计算机统考Outlook
  13. Python HackerRank 刷题 Maximum Subarray Sum
  14. Laya和TypeScript制作H5——开始
  15. HC-05蓝牙模块与STM32控制led灯亮详解
  16. 帆软内SQL语句的使用
  17. BLE 协议栈(Master,Slave;Standby,Advertiser,Scanner,Initiator;连接流程,连接参数)
  18. OpenVINO 2021r4.1 - 瞎搞YOLOV5 模型转换,INT8量化及C++推理实现
  19. 计算机地图制图成图方法,第五章 地图成图方法.ppt
  20. 嘉立创 EDA 专业版原理图绘制

热门文章

  1. 谷歌学术、github、Sci-Hub镜像网址总结
  2. Android应用层读写设备节点
  3. Java多用户商城系统B2B2C源码-(九)服务链路追踪(Spring Cloud Sleuth)
  4. oracle经纬度换算成xy坐标,经纬度换算成xy坐标(经纬度转换xy坐标算法)
  5. 树莓派小实验 | 制作一个带快门的照相机 录像机
  6. Linux du命令和df命令
  7. 三角形求高公式计算机,直角三角形求高公式 直角三角形怎么求高
  8. 外币兑换系统(某专业大作业)
  9. 蛐蛐播放器 android,蛐蛐五线谱播放器
  10. scala怎么取绝对值