前言

MAUI Blazor中,安卓项目的返回键体验很不好,只能如同浏览器一样返回上一页。但很多时候,我们想让他返回的上一页,不一定就是实际上的上一页。而且也想让返回键去支持一些事件,按下返回键触发,例如关闭弹窗。Blazor中有NavigationManager,平时页面的跳转都是靠注入它,调用它的NavigateTo()方法。我们可以写个NavigateService类把他包装起来,再写个方法对安卓返回键做一些专门的处理,放在安卓返回键的处理事件中调用。

正文

添加所需代码

  1. 添加一个接口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();
}
  1. 添加实现类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;
}
}
  1. 添加安卓返回键处理类,放在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项目中如何添加一个返回服务,并支持安卓返回键的更多相关文章

  1. 在Qt项目中如何添加一个已有的项目作为子项目

    新建一个子目录项目(具体方法参见<类似Visual Studio一样,使用Qt Creator管理多个项目,创建子项目>),然后需要添加的项目移动到该子目录项目目录下,再在其pro文件中添 ...

  2. RK平台Android4.4 添加一个新的遥控器支持以及添加特殊按键【转】

    本文转载自:http://blog.csdn.net/coding__madman/article/details/52904063 版权声明:本文为博主原创文章,未经博主允许不得转载. 瑞芯微平台 ...

  3. spring项目中如何添加定时器以及在定时器中自动生成sprng注入对象

    最近做了一个java的项目,部门领导给了一套代码让我尽快掌握,说心里话本人真心不喜欢java的这种项目方式,各种配置各种xml文件简直头都大了,下面就将我遇到的其中一个我认为是坑的地方整理出来,希望能 ...

  4. Spring Cloud项目中通过Feign进行内部服务调用发生401\407错误无返回信息的问题

    问题描述 最近在使用Spring Cloud改造现有服务的工作中,在内部服务的调用方式上选择了Feign组件,由于服务与服务之间有权限控制,发现通过Feign来进行调用时如果发生了401.407错误时 ...

  5. 在类库或winform项目中打开另一个winform项目的窗体

    假设类库或winform项目为A,另一个winform项目为B.那麽在A中添加一个接口,里面有一个Show方法,然后在B中写一个类b继承这个接口,并重写这个方法,具体内容为弹出某个窗体.然后在A中另一 ...

  6. 在项目中全局添加FastClick导致图片上传插件在ios端失效的解决方案

    ---恢复内容开始--- 项目是移动端的项目,为了解决300ms的click延迟,所以在全局中加入了FastClick,引入的方式很简单,网上一大堆教程,这里不做赘述 我们就谈,我遇到的问题: 某天产 ...

  7. 在vue项目中如何添加eslint

    随着vue的越做越好,更多的开发者选择使用vue,本篇记录如何在vue项目中添加eslint. 首先第一种就是在vue项目创建初始时就选择了创建,随着初始化一起代入到了项目当中,那么要是一开始觉得es ...

  8. Xcode7.2中如何添加一个Empty Application模板

    大熊猫猪·侯佩原创或翻译作品.欢迎转载,转载请注明出处. 如果觉得写的不好请多提意见,如果觉得不错请多多支持点赞.谢谢! hopy ;) Xcode 6.0正式版之后已经没有所谓的Empty Appl ...

  9. WPF 添加 Resources Dictionary 资源 一般类库项目中无法添加资源文件(ResourceDictionary)

    在文件夹或者项目右键-> Add(添加),会弹出可以快捷添加的资源,但是你会发现没有 ResourceDictionary资源可以选择. 解决此问题方法: 第一步:工程->右键->U ...

  10. Laravel 项目中编写第一个 Vue 组件

    和 CSS 框架一样,Laravel 不强制你使用什么 JavaScript 客户端框架,但是开箱对 Vue.js 提供了良好的支持,如果你更熟悉 React 的话,也可以将默认的脚手架代码替换成 R ...

随机推荐

  1. 基于海思H3520DV400和QT5.9设计的车载终端DVR控制平台

    ​ 目录 前言: 说明: 功能介绍: 设计思路: 详细设计: QT界面设计: 代码实现: 注意事项: (一)QT运行慢问题 (二)QT图层隐藏问题 (三)鼠标问题 (四)字体问题 (五)主界面图案 ( ...

  2. c# 异步进阶———— 自定义 taskschedule[三]

    前言 我们知道我们的task async 和 await 是基于线程池进行调度的. 但是async 和 await 也就是使用了默认的task调度,让其在线程池中运行. 但是线程池是榨干机器性能为本质 ...

  3. SpringBoot开启日志级别

    #开启logging logging.level.org.springframework.boot.autoconfigure: error logging: level: main.blog.map ...

  4. Python-解三元一次方程

    需要解的方程组为: x + y + z = 26 x - y = 1 2x - y + z = 18 下面进入代码实现: 1.导入数学计算库 numpy import numpy as np 2.生成 ...

  5. 解决VM虚拟机中IP或域名不能ping通

    c4548abb-da65-4f7d-827f-e95dca25a13d 问题 无法ping通域名, 检查事项 确定在同一个子网,能访问DNS服务器. DNS服务器正确设置了正反向解析,且DNS服务器 ...

  6. 2021-03-25:如何把长url转换为短url?

    2021-03-25:如何把长url转换为短url? 福大大 答案2021-03-25: 1.长url和短url一一映射.想采用某种算法,把长url和短url做一一映射.后来发现,这种方法是行不通的. ...

  7. vue-router几大坑

    如今vue使用率很高,踩坑这就是很平常的了,使用了几年坑都依然没踩完,纠结呀 一.router.js配置要点 大家都知道vue 是组件化开发,页面很多路由难免, 这里是路由配置router.js 最外 ...

  8. js代理(Proxy) 和 反射(Reflection)

    在实际开发中经常会遇到js抛出的错误,但是我们有没有想过自己去接管js异常验证,根据自己的需求抛出异常呢?原本也许不行,但是在es6出来后就可以做到了 一.代理(Proxy) 什么是'代理' 呢?代理 ...

  9. web自动化09-frame切换、多窗口切换

    frame切换 1.html代码: <frameset cols="25%,50%,25%"> <frame src="a.htm"> ...

  10. Ascend C sqrt算子实战

    摘要:编写一个Ascend C的sqrt算子,并通过内核调用方式在cpu和npu模式下进行验证. 本文分享自华为云社区<[2023 · CANN训练营第一季]--Ascend C sqrt算子实 ...