新闻中心
Blazor 怎么实现一个全局通知服务
Blazor全局通知服务通过创建NotificationService管理通知队列并触发UI更新,配合NotificationItem组件渲染,注册为scoped服务后在任意组件中注入调用Show()方法即可显示通知。

Blazor 实现全局通知服务,核心是创建一个可被任意组件注入、跨页面共享状态的 Service,并配合一个统一的 UI 组件(如 Toast 或 Banner)来渲染通知。关键在于状态管理 + 组件通信 + 生命周期控制。
1. 定义通知数据模型和通知服务
先定义一个轻量通知类,包含类型(Success/Info/Error/Warning)、内容、持续时间、是否自动关闭等字段:
public class Notification
{
public string Id { get; } = Guid.NewGuid().ToString();
public string Message { get; set; } = string.Empty;
public NotificationType Type { get; set; } = NotificationType.Info;
public int DurationMs { get; set; } = 5000;
public bool AutoDismiss { get; set; } = true;
}
public enum NotificationType
{
Info, Success, Warning, Error
}
然后创建 NotificationService,用 List<notification></notification> 管理队列,提供添加、移除、清空方法。注意用 INotifyPropertyChanged 或 Blazor 的 StateHasChanged() 机制触发 UI 更新——推荐用 EventCallback 或 Subject<notification></notification>(配合 Rx.NET)或更简单的方式:直接暴露一个 EventCallback 委托供 UI 订阅:
public class NotificationService
{
private readonly List<Notification> _notifications = new();
public event Action? OnNotificationsChanged;
public IReadOnlyList<Notification> Notifications => _notifications.AsReadOnly();
public void Show(string message, NotificationType type = NotificationType.Info, int durationMs = 5000)
{
var notification = new Notification
{
Message = message,
Type = type,
DurationMs = durationMs
};
_notifications.Add(notification);
OnNotificationsChanged?.Invoke();
if (notification.AutoDismiss)
{
_ = Task.Delay(durationMs).ContinueWith(_ =>
{
Dismiss(notification.Id);
});
}
}
public void Dismiss(string id)
{
_notifications.RemoveAll(n => n.Id == id);
OnNotificationsChanged?.Invoke();
}
public void Clear()
{
_notifications.Clear();
OnNotificationsChanged?.Invoke();
}
}
2. 注册服务并注入到根组件
在 Program.cs(.NET 6+)中注册为 scoped 服务:
builder.Services.AddScoped<NotificationService>();
在 App.razor 或 MainLayout.razor 顶部注入服务,并订阅变化:
@inject NotificationService NotificationService
@if (NotificationService.Notifications.Any())
{
<div class="notification-container">
@foreach (var notification in NotificationService.Notifications)
{
<NotificationItem Notification="@notification" OnDismiss="() => NotificationService.Dismiss(notification.Id)" />
}
</div>
}
@code {
private void HandleNotificationsChanged() => StateHasChanged();
protected override void OnInitialized()
{
NotificationService.OnNotificationsChanged += HandleNotificationsChanged;
}
public void Dispose()
{
NotificationService.OnNotificationsChanged -= HandleNotificationsChanged;
}
}
3. 创建可复用的通知 UI 组件(NotificationItem.razor)
封装单条通知的样式与交互,支持手动关闭、动画入场/离场(可用 CSS transition 或第三方库如 blazored-toast):
Figma
Figma 是一款基于云端的 UI 设计工具,可以在线进行产品原型、设计、评审、交付等工作。
1371
查看详情
@using System.Text.RegularExpressions <div class="notification @GetCssClass()" @key="Notification.Id"> <span class="notification-icon">@GetIcon()</span> <span class="notification-message">@Notification.Message</span> <button class="notification-close" @onclick="() => OnDismiss.InvokeAsync()">✕</button> </div> @code { [Parameter] public Notification Notification { get; set; } = default!; [Parameter] public EventCallback OnDismiss { get; set; } private string GetCssClass() => $"notification-{Notification.Type.ToString().ToLower()}"; private string GetIcon() { return Notification.Type switch { NotificationType.Success => "✓", NotificationType.Error => "✗", NotificationType.Warning => "⚠", _ => "ℹ" }; } }
CSS 示例(加到 wwwroot/css/app.css 或组件内):
.notification-container {
position: fixed;
top: 1rem;
right: 1rem;
z-index: 1000;
display: flex;
flex-direction: column;
gap: 0.5rem;
max-width: 350px;
}
.notification {
padding: 0.75rem 1rem;
border-radius: 4px;
box-shadow: 0 2px 8px rgba(0,0,0,0.15);
display: flex;
align-items: center;
gap: 0.5rem;
animation: slideInRight 0.3s ease-out;
transition: opacity 0.3s, transform 0.3s;
}
.notification:not(:first-child) { margin-top: 0.25rem; }
.notification-info { background: #e3f2fd; color: #1565c0; }
.notification-success { background: #e8f5e9; color: #2e7d32; }
.notification-warning { background: #fff8e1; color: #f57c00; }
.notification-error { background: #ffebee; color: #c62828; }
.notification-close {
background: none;
border: none;
font-size: 1.2rem;
cursor: pointer;
margin-left: auto;
color: inherit;
opacity: 0.7;
}
.notification-close:hover { opacity: 1; }
@keyframes slideInRight {
from { transform: translateX(100%); opacity: 0; }
to { transform: translateX(0); opacity: 1; }
}
4. 在任意组件中使用通知
哪里需要就在哪里 @inject NotificationService,调用 Show() 即可:
@inject NotificationService NotificationService
<button @onclick="ShowSuccess">显示成功通知</button>
@code {
private void ShowSuccess() =>
NotificationService.Show("操作已成功!", NotificationType.Success);
}
支持异步操作后通知:
private async Task SubmitForm()
{
try
{
await httpClient.PostAsJsonAsync("/api/s*e", model);
NotificationService.Show("保存成功", NotificationType.Success);
}
catch
{
NotificationService.Show("保存失败,请重试", NotificationType.Error);
}
}
基本上就这些。不复杂但容易忽略的是:服务生命周期要匹配(scoped 最常用)、UI 订阅/取消订阅要配对、避免内存泄漏(尤其用委托事件时),以及通知叠加时的 z-index 和定位控制。进阶可加队列限流、点击跳转、自定义模板、声音提示等。
以上就是Blazor 怎么实现一个全局通知服务的详细内容,更多请关注其它相关文章!
# 相关文章
# seo能带来哪些好处
# 千岛湖优化网站价格
# 内江专业网站建设服务
# 清溪麻涌网站建设
# 武清眼镜网站建设
# 营销策略推广ppt
# 全渠道营销推广哪家好
# seo途牛
# 网站建设优化方法 s
# 宁波营销推广公司电话
# 跳转
# 解决问题
# 自定义
# 中文网
# css
# 就在
# 序列化
# 进阶
# 的是
# 如何将
# red
# .net
# 组件渲染
# nas
# switch
# ai
# app
# json
# js
相关栏目:
【
科技资讯46185 】
【
网络学院92790 】
相关推荐:
Excel函数批量查找替换超快方法_Excel用REPLACE和FIND函数秒级替换
mcjs网页版流畅运行 mcjs低配电脑畅玩入口
mysql如何设置表访问权限_mysql表访问权限配置
《北京人工智能产业白皮书(2025)》发布:全年核心产值预计突破 4500 亿元
深入理解J*aScript Promise异步执行与微任务队列
漫蛙2正版漫画站 漫蛙2网页版快速访问入口
包子漫画官方网站阅读入口-包子漫画在线漫画官网直达链接
poki免费入口快捷访问 poki人气小游戏直接玩站点
J*a里如何使用N*igableMap进行导航操作_可导航Map操作技巧解析
必由学官网快捷入口 必由学网页版在线学习平台
一加Ace 6T支持全新明眸护眼:通过了最严苛的护眼小金标认证
VS Code远程开发时如何处理文件权限问题
一加手机电池耗电快怎么办_一加手机电池耗电快的解决方法
Node.js 中使用 node-cron 实现定时 API 数据抓取与处理
MAC如何将整个网页截长图_MAC使用Safari的导出为PDF或第三方工具
解决Python单元测试中Mock异常方法调用计数为零的问题
电脑安装程序提示“错误1722”怎么办_Windows Installer服务问题解决【教程】
qq浏览器打开空白页怎么办 qq浏览器启动后显示白屏的解决教程
Lar*el递归关系中排除子孙节点的策略
美团外卖商家服务中心入口 美团商家版官网入口
Win11怎么查看显卡显存 Win11显示适配器属性及专用视频内存查询
Golang如何处理RPC请求负载均衡_Golang RPC请求负载均衡策略与实践
整合Supabase认证与Django模型:跨模式迁移的解决方案
内存检查:在VS Code中调试C++时的内存视图
处理动态列数据:J*a ArrayList的正确初始化与字符累加教程
Basecamp怎样用留言钉固定重点_Basecamp用留言钉固定重点【重点标记】
J*aScript打印功能_j*ascript输出控制
机构:以往存储涨价周期小米利润率实际上有所改善 能转嫁给消费者等
c++如何使用折叠表达式(Fold Expressions)_c++17可变参数模板新技巧
蛙漫2台版漫画地址 Manwa2正版网页版链接
KFC套餐升级怎么获取优惠代码_KFC套餐升级活动与优惠代码获取方法
PHP URL参数传递与500错误调试指南
DLsite中文平台入口 DLsite官网内容在线查看
钉钉视频会议声音异常如何处理 钉钉会议音频修复技巧
Golang如何优雅处理error_Golang error处理最佳实践总结
Win11怎么设置鼠标主按键_Win11鼠标左右键功能互换
《明末:渊虚之羽》设计师谈设计角色:那会刚毕业 充满激情
Go语言中Map值调用指针接收器方法的限制与应对
C++如何实现一个智能指针_手动实现C++ shared_ptr的引用计数功能
Typer应用中动态命令行参数的解析与处理
SteamMachine定价或为699美元 大家想入手吗?
Mac怎么锁定备忘录_Mac备忘录加密设置教程
汽水音乐在线版入口_汽水音乐网页播放手册
C++ vector二维数组定义_C++ vector of vector用法
限制HTML日期输入框的日期选择范围
外媒分析《GTA6》定价:卖100美元可以但真没必要!
在FastAPI中利用lifespan与依赖注入高效管理Redis连接池
Win10自动更新怎么关闭 Win10永久关闭系统更新的两种方法【终极版】
天眼查怎么看公司融资情况 天眼查企业融资历史查询步骤【攻略】
邮政编码查询不到怎么办_邮政编码查询不到的常见原因与对策


2025-12-13
浏览次数:次
返回列表
xt.RegularExpressions
<div class="notification @GetCssClass()" @key="Notification.Id">
<span class="notification-icon">@GetIcon()</span>
<span class="notification-message">@Notification.Message</span>
<button class="notification-close" @onclick="() => OnDismiss.InvokeAsync()">✕</button>
</div>
@code {
[Parameter] public Notification Notification { get; set; } = default!;
[Parameter] public EventCallback OnDismiss { get; set; }
private string GetCssClass() => $"notification-{Notification.Type.ToString().ToLower()}";
private string GetIcon()
{
return Notification.Type switch
{
NotificationType.Success => "✓",
NotificationType.Error => "✗",
NotificationType.Warning => "⚠",
_ => "ℹ"
};
}
}