MAUI Blazor项目中如何添加一个返回服务,并支持安卓返回键
前言
MAUI Blazor中,安卓项目的返回键体验很不好,只能如同浏览器一样返回上一页。但很多时候,我们想让他返回的上一页,不一定就是实际上的上一页。而且也想让返回键去支持一些事件,按下返回键触发,例如关闭弹窗。Blazor中有NavigationManager,平时页面的跳转都是靠注入它,调用它的NavigateTo()方法。我们可以写个NavigateService类把他包装起来,再写个方法对安卓返回键做一些专门的处理,放在安卓返回键的处理事件中调用。
正文
添加所需代码
- 添加一个接口INavigateService
public interface INavigateService
{
/// <summary>
/// 存储想要返回时触发的方法
/// </summary>
event Action Action;
NavigationManager Navigation { get;protected set; }
/// <summary>
/// 存储URL历史记录
/// </summary>
List<string> HistoryUrl { get; protected set; }
/// <summary>
/// 初始化
/// </summary>
/// <param name="navigation"></param>
void Initialize(NavigationManager navigation);
/// <summary>
/// 导航
/// </summary>
/// <param name="url"></param>
void NavigateTo(string url);
/// <summary>
/// 返回上一页
/// </summary>
void NavigateToBack();
/// <summary>
/// 返回键处理
/// </summary>
/// <returns>true已处理事件,false为无事件可处理(便于做进一步处理,例如退出应用)</returns>
bool OnBackButtonPressed();
}
- 添加实现类NavigateService
public class NavigateService : INavigateService
{
public NavigationManager Navigation { get; set; } = default!;
public event Action? Action;
public void Initialize(NavigationManager navigation)
{
Navigation = navigation;
}
public List<string> HistoryUrl { get; set; } = new List<string>();
public void NavigateTo(string url)
{
var href = Navigation.ToBaseRelativePath(Navigation.Uri);
HistoryUrl.Add(href);
Navigation.NavigateTo(url);
}
public void NavigateToBack()
{
string href = string.Empty;
if (HistoryUrl.Count > 0)
{
href = HistoryUrl.Last();
}
Navigation.NavigateTo(href);
if (HistoryUrl.Count > 0)
{
HistoryUrl.RemoveAt(HistoryUrl.Count - 1);
}
}
public bool OnBackButtonPressed()
{
if (Action != null && Action?.GetInvocationList().Length > 0)
{
var delegates = Action!.GetInvocationList();
(delegates.Last() as Action)!.Invoke();
return true;
}
if (HistoryUrl.Count > 0)
{
NavigateToBack();
return true;
}
return false;
}
}
- 添加安卓返回键处理类,放在Platforms/Android下,别忘记更改命名空间
using Android.OS;
using Android.Views;
using Android.Widget;
using SwashbucklerDiary.IServices;
using Application = Android.App.Application;
namespace XXX.Platforms.Android
{
public static class BackButtonPressed
{
private static byte BackPressCounter;
public static bool OnBackButtonPressed(KeyEvent e)
{
if (e.KeyCode == Keycode.Back)
{
if (e.Action == KeyEventActions.Down)
{
var service = MauiApplication.Current.Services.GetRequiredService<INavigateService>();
bool flag = service!.OnBackButtonPressed();
if (!flag)
{
QuitApp();
}
}
return true;
}
return false;
}
public static void QuitApp()
{
if (BackPressCounter == 1)
{
Process.KillProcess(Process.MyPid());
}
else if (BackPressCounter == 0)
{
BackPressCounter++;
Toast.MakeText(Application.Context, "再按一次退出", ToastLength.Long)!.Show();
Task.Run(async () =>
{
await Task.Delay(2000);
BackPressCounter = 0;
});
}
}
}
}
4.修改Platforms/Android/MainActivity.cs,添加以下代码
public override bool DispatchKeyEvent(KeyEvent e)
{
var flag = BackButtonPressed.OnBackButtonPressed(e);
if (flag)
{
return true;
}
return base.DispatchKeyEvent(e);
}
5.Shared下添加MainLayout.razor.cs
public partial class MainLayout
{
[Inject]
NavigationManager Navigation { get; set; } = default!;
[Inject]
INavigateService NavigateService { get; set; } = default!;
protected override void OnInitialized()
{
NavigateService.Initialize(Navigation);
base.OnInitialized();
}
}
6.修改MauiProgram.cs,添加以下代码
builder.Services.AddSingleton<INavigateService, NavigateService>();
如何使用
注入INavigateService
[Inject]
protected INavigateService NavigateService { get; set; } = default!;
如果想要将跳转后的上一页保存到返回服务中,跳转页面时使用NavigateService.NavigateTo(url),当你主动调用NavigateService.NavigateToBack()或者按下安卓返回键时,会跳转回之前保存的上一页。如果想要让返回键执行什么方法,就加入到委托中NavigateService.Action += 你的方法;别忘了在执行后和组件Dispose中把你的方法移出委托NavigateService.Action -= 你的方法;
一些想法
说一下为什么没有继承NavigationManager或者在NavigateService 的构造方法中注入它,NavigationManager这个类比较特殊,也没太玩明白。无论是继承还是构造方法中注入它,得到的都是没有经过初始化的NavigationManager,可以理解为空的,后续也没有办法使用。所以最后选择在MainLayout.razor.cs中注入的NavigationManager,把它存起来。我觉得这么写肯定是不够好,只是没有更好的办法。
后记
本人才疏学浅,也没怎么认真写过文章,所以文章中可能会存在纰漏,也欢迎各位大佬指出。
MAUI Blazor项目中如何添加一个返回服务,并支持安卓返回键的更多相关文章
- 在Qt项目中如何添加一个已有的项目作为子项目
新建一个子目录项目(具体方法参见<类似Visual Studio一样,使用Qt Creator管理多个项目,创建子项目>),然后需要添加的项目移动到该子目录项目目录下,再在其pro文件中添 ...
- RK平台Android4.4 添加一个新的遥控器支持以及添加特殊按键【转】
本文转载自:http://blog.csdn.net/coding__madman/article/details/52904063 版权声明:本文为博主原创文章,未经博主允许不得转载. 瑞芯微平台 ...
- spring项目中如何添加定时器以及在定时器中自动生成sprng注入对象
最近做了一个java的项目,部门领导给了一套代码让我尽快掌握,说心里话本人真心不喜欢java的这种项目方式,各种配置各种xml文件简直头都大了,下面就将我遇到的其中一个我认为是坑的地方整理出来,希望能 ...
- Spring Cloud项目中通过Feign进行内部服务调用发生401\407错误无返回信息的问题
问题描述 最近在使用Spring Cloud改造现有服务的工作中,在内部服务的调用方式上选择了Feign组件,由于服务与服务之间有权限控制,发现通过Feign来进行调用时如果发生了401.407错误时 ...
- 在类库或winform项目中打开另一个winform项目的窗体
假设类库或winform项目为A,另一个winform项目为B.那麽在A中添加一个接口,里面有一个Show方法,然后在B中写一个类b继承这个接口,并重写这个方法,具体内容为弹出某个窗体.然后在A中另一 ...
- 在项目中全局添加FastClick导致图片上传插件在ios端失效的解决方案
---恢复内容开始--- 项目是移动端的项目,为了解决300ms的click延迟,所以在全局中加入了FastClick,引入的方式很简单,网上一大堆教程,这里不做赘述 我们就谈,我遇到的问题: 某天产 ...
- 在vue项目中如何添加eslint
随着vue的越做越好,更多的开发者选择使用vue,本篇记录如何在vue项目中添加eslint. 首先第一种就是在vue项目创建初始时就选择了创建,随着初始化一起代入到了项目当中,那么要是一开始觉得es ...
- Xcode7.2中如何添加一个Empty Application模板
大熊猫猪·侯佩原创或翻译作品.欢迎转载,转载请注明出处. 如果觉得写的不好请多提意见,如果觉得不错请多多支持点赞.谢谢! hopy ;) Xcode 6.0正式版之后已经没有所谓的Empty Appl ...
- WPF 添加 Resources Dictionary 资源 一般类库项目中无法添加资源文件(ResourceDictionary)
在文件夹或者项目右键-> Add(添加),会弹出可以快捷添加的资源,但是你会发现没有 ResourceDictionary资源可以选择. 解决此问题方法: 第一步:工程->右键->U ...
- Laravel 项目中编写第一个 Vue 组件
和 CSS 框架一样,Laravel 不强制你使用什么 JavaScript 客户端框架,但是开箱对 Vue.js 提供了良好的支持,如果你更熟悉 React 的话,也可以将默认的脚手架代码替换成 R ...
随机推荐
- ES6教程笔记
ES介绍 什么是ES ES全称 EcmaScript 是一种脚本语言的规范 Javascript为ES的实现 Ecma 是一个组织 Ecma组织 为什么要学习ES6? ES6的版本变动内容最多,具有里 ...
- Gradio入门到进阶全网最详细教程[二]:快速搭建AI算法可视化部署演示(侧重参数详解和案例实践)
Gradio入门到进阶全网最详细教程[二]:快速搭建AI算法可视化部署演示(侧重参数详解和案例实践) 相关文章:Gradio入门到进阶全网最详细教程[一]:快速搭建AI算法可视化部署演示(侧重项目搭建 ...
- sqlmap安全测试工具使用简介
SQLmap是一个自动化的SQL注入工具,其主要功能是扫描,发现并利用给定的URL的SQL注入漏洞,目前支持的数据库是MySQL,Oracle,PostgreSQL,Microsoft SQL S ...
- 使用NineData定制企业级数据库规范
1. 为什么需要数据库规范? 在企业级应用中,数据库是非常重要的一部分,它们存储着公司的核心数据,包括客户信息.订单.产品信息等等.如果这些数据没有得到妥善的管理,那么就会导致数据不一致.数据丢失.数 ...
- 2023-04-23:给定你一个整数数组 nums 我们要将 nums 数组中的每个元素移动到 A 集合 或者 B 集合中 使得 A 集合和 B 集合不为空,并且 average(A) == aver
2023-04-23:给定你一个整数数组 nums 我们要将 nums 数组中的每个元素移动到 A 集合 或者 B 集合中 使得 A 集合和 B 集合不为空,并且 average(A) == aver ...
- 2022-08-07:以下go语言代码输出什么?A:1 1;B:3 1;C:0 3;D:不能编译。 package main import ( “fmt“ “math“ ) func main
2022-08-07:以下go语言代码输出什么?A:1 1:B:3 1:C:0 3:D:不能编译. package main import ( "fmt" "math&q ...
- 2022-05-26:void add(int L, int R, int C)代表在arr[L...R]上每个数加C, int get(int L, int R)代表查询arr[L...R]上的累加
2022-05-26:void add(int L, int R, int C)代表在arr[L-R]上每个数加C, int get(int L, int R)代表查询arr[L-R]上的累加和, 假 ...
- gitlab-runner 中的 Docker-in-Docker
笔者个人理解:gitlab-runner 安装后就是一个监听状态的 runner,而通过 gitlab-runner register 注册的"实例"其实只是预定义的配置节,当消息 ...
- HCIP- ICT实战进阶ex1-MPLS
HCIP- ICT实战进阶ex1-MPLS 0 前言 由于BGP设备之间是通过TCP协议实现的跨设备互联, 所以在两台BGP设备之间的其他设备是没有配置BGP协议的, 因此中间的其他设备很可能无法学习 ...
- LeetCode 周赛 346(2023/05/21)仅 68 人 AK 的最短路问题
本文已收录到 AndroidFamily,技术和职场问题,请关注公众号 [彭旭锐] 提问. LeetCode 单周赛第 345 场 · 体验一题多解的算法之美 单周赛 345 概览 T1. 删除子串后 ...