UWP不归路——自定义后退事件管理

双击退出

UWP 中可以调用如下方法对后退按钮进行事件处理,比如页面导航,退出全屏,双击退出等等。

// 注册后退事件
SystemNavigationManager.GetForCurrentView().BackRequested += PageBackRequested;

// 后退事件处理
private void PageBackRequested(object sender, BackRequestedEventArgs e)
{
    e.Handled = true;// 阻止后面注册的事件继续执行
    // TODO
}

// 注销
SystemNavigationManager.GetForCurrentView().BackRequested -= PageBackRequested;

通过上面的调用已经可以对后退按钮定制不同的点击效果,但是,还存在一个问题。应用中肯定会有很多的页面,每个页面对于后退按钮的处理需求肯定会有所不同,多次注册后退事件(连续注册不注销)是难免的,注册后退事件实际上是把每个事件处理依次加入到一个事件队列中去,每当点击后退按钮的时候,就会按照先后顺序依次执行事件处理,因此就会存在后面处理和前面的处理产生冲突导致无法实现预期中的效果,还需要做一些费劲的特殊处理才能正常使用。

所以可以做一个简单的封装,实现只处理最后注册的事件

using System;
using System.Collections;
using Windows.UI.Core;

namespace indi.anyesu.UWP.Core.Managers
{
    //
    // 自定义后退事件管理:
    //     允许只调用最后注册的后退事件,而SystemNavigationManager的后退事件是按顺序依次执行的。
    public sealed class BackEventManager
    {

        private static Stack BackEventStack = new Stack();

        //
        // 注册后退事件
        //
        public static void Register(EventHandler<BackRequestedEventArgs> PageBackRequested)
        {
            if (BackEventStack.Count > 0)
            {
                SystemNavigationManager.GetForCurrentView().BackRequested -= BackEventStack.Peek() as EventHandler<BackRequestedEventArgs>;
            }
            SystemNavigationManager.GetForCurrentView().BackRequested += PageBackRequested;// 注册到系统自带的后退事件队列
            BackEventStack.Push(PageBackRequested);
        }

        //
        // 注销最后注册的后退事件
        //
        public static void Unregister(EventHandler<BackRequestedEventArgs> PageBackRequested)
        {
            if (BackEventStack.Count > 0)
            {
                var top = BackEventStack.Peek() as EventHandler<BackRequestedEventArgs>;
                if (PageBackRequested.Equals(top))
                {
                    SystemNavigationManager.GetForCurrentView().BackRequested -= PageBackRequested;
                    BackEventStack.Pop();
                    if (BackEventStack.Count > 0)
                    {
                        SystemNavigationManager.GetForCurrentView().BackRequested += BackEventStack.Peek() as EventHandler<BackRequestedEventArgs>;
                    }
                }
            }
        }
    }
}

主要思路是将所有注册的事件处理压入 BackEventStack 这个栈当中,保证系统自带的后退事件处理队列当中最多只有一个事件处理,即最后注册的那个。调用方法如下:

BackEventManager.Register(PageBackRequested);// 注册后退事件
BackEventManager.Unregister(PageBackRequested);// 注销后退事件
// 注意: 最好保证注册和注销成对出现(如在页面的OnNavigatedTo方法中注册,OnNavigatedFrom方法中注销),避免出现冲突。

有了这个后退事件管理器之后,可以在 APP 初始化的时候完成后退事件的注册,实现一个统一的后退事件处理,特殊页面特殊处理,这样就不用到处贴代码了,维护起来更方便。
App.xaml.cs 中修改 OnLaunched 方法,如下所示:

/// <summary>
/// 在应用程序由最终用户正常启动时进行调用。
/// 将在启动应用程序以打开特定文件等情况下使用。
/// </summary>
/// <param name="e">有关启动请求和过程的详细信息。</param>
protected override async void OnLaunched(LaunchActivatedEventArgs e)
{
    Frame rootFrame = Window.Current.Content as Frame;
    // 不要在窗口已包含内容时重复应用程序初始化,
    // 只需确保窗口处于活动状态
    if (rootFrame == null)
    {
        // 创建要充当导航上下文的框架,并导航到第一页
        rootFrame = new Frame();
        rootFrame.NavigationFailed += OnNavigationFailed;
        // 下面这句话是关键
        rootFrame.Navigated += OnNavigated;// 注册页面加载完毕事件
        if (e.PreviousExecutionState == ApplicationExecutionState.Terminated)
        {
            //TODO: 从之前挂起的应用程序加载状态
        }
        // 将框架放在当前窗口中
        Window.Current.Content = rootFrame;
    }
    if (rootFrame.Content == null)
    {
        // 当导航堆栈尚未还原时,导航到第一页,
        // 并通过将所需信息作为导航参数传入来配置
        rootFrame.Navigate(typeof(MainPage), e.Arguments);
    }
    // 如果是移动端,则设置可以设置顶部状态栏(电量、时间...)的颜色、是否显示
    if (Windows.Foundation.Metadata.ApiInformation.IsTypePresent("Windows.UI.ViewManagement.StatusBar"))
    {
        StatusBar statusBar = StatusBar.GetForCurrentView();
        // statusBar.ForegroundColor = Colors.White;// 设置背景色
        await statusBar.HideAsync();// 隐藏状态栏
    }
    Window.Current.Activate();// 确保当前窗口处于活动状态
}
// 在OnNavigated方法中进行事件的注册和控制后退按钮的显示与否
private void OnNavigated(object sender, NavigationEventArgs e)
{
    //显示标题栏后退按钮
    //SystemNavigationManager.GetForCurrentView().AppViewBackButtonVisibility = ((Frame)sender).CanGoBack ? AppViewBackButtonVisibility.Visible : AppViewBackButtonVisibility.Collapsed;
    SystemNavigationManager.GetForCurrentView().AppViewBackButtonVisibility = AppViewBackButtonVisibility.Visible;// 在PC客户端左上角显示后退按钮
    BackEventManager.Register(PageBackRequested);// 注册后退事件
}

这里最关键的一句就是 rootFrame.Navigated += OnNavigated;PageBackRequested 方法中做了统一的后退事件处理,比如在我的应用中,我希望显示 MainPage 时后退按钮能够双击彻底退出应用,显示其他页面时后退按钮能够处理正常的页面后退导航,具体处理如下:

private void PageBackRequested(object sender, BackRequestedEventArgs e)
{
    e.Handled = true;
    GoBack();
}
/// <summary>
/// 自定义全局后退事件处理
/// </summary>
public static async void GoBack()
{
    var rootFrame = Window.Current.Content as Frame;// App的根Frame
    if (rootFrame == null)
    {
        return;
    }
    if (rootFrame.CurrentSourcePageType == typeof(MainPage))// 判断rootFrame 当前页面类型是否为MainPage
    {
        if (!IsQuit)// IsQuit表示是否已点击了后退按钮,用来处理双击事件
        {
            IsQuit = true;
            await (rootFrame.Content as MainPage).showMessage("再按一次返回键退出", Colors.Red);// 异步调出页面的提示框
            IsQuit = false;
        }
        else
        {
            Current.Exit();// 彻底退出App
        }
    }
    else if (rootFrame.CanGoBack)
    {
        rootFrame.GoBack();
    }
}

在MainPage中

public async Task showMessage(string msg, Color color)
{
    Hint.Text = msg;// 设置提示框文本内容
    if (color != null)
    {
        if (color == Colors.Green)
        {
            color = Color.FromArgb(255, 91, 159, 82);
        }
        messageBorder.Background = new SolidColorBrush(color);// 设置提示框背景色
    }
    message.Visibility = Visibility.Visible;// 显示提示框
    await Task.Delay(1500);// 延时1500ms
    message.Visibility = Visibility.Collapsed;// 隐藏提示框
}

//以下为xaml内容,一个自定义的文本提示框
<Grid x:Name="message" RelativePanel.AlignVerticalCenterWithPanel="True" RelativePanel.AlignHorizontalCenterWithPanel="True" Visibility="Collapsed">
    <Border x:Name="messageBorder" CornerRadius="10" Background="#5B9F52"></Border>
    <ScrollViewer VerticalAlignment="Center" MaxHeight="120" VerticalScrollBarVisibility="Auto" BorderThickness="0">
        <TextBlock x:Name="Hint" Foreground="White" RelativePanel.AlignHorizontalCenterWithPanel="True" RelativePanel.AlignVerticalCenterWithPanel="True" VerticalAlignment="Center" HorizontalAlignment="Center" TextAlignment="Center" Margin="0" TextWrapping="Wrap" Padding="10" MinWidth="120" MinHeight="30" FontFamily="Resources/FontAwesome.otf#FontAwesome" FontSize="16"/>
    </ScrollViewer>
</Grid>

总结


在我的开发过程中,这个后退事件管理器已经基本满足所有需求了,不过每次执行后退事件的时候都很任性的抛弃了之前注册的事件,之后会考虑加入与前面注册的事件共存的方法并做一些优化,以便灵活地适应更多的通用需求。


转载请注明出处:https://www.jianshu.com/p/55d6cd6b632d

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,214评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,307评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,543评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,221评论 1 279
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,224评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,007评论 1 284
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,313评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,956评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,441评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,925评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,018评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,685评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,234评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,240评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,464评论 1 261
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,467评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,762评论 2 345

推荐阅读更多精彩内容

  •   JavaScript 与 HTML 之间的交互是通过事件实现的。   事件,就是文档或浏览器窗口中发生的一些特...
    霜天晓阅读 3,473评论 1 11
  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,089评论 1 32
  • 有时候不是不懂,只是不想懂;有时候不是不知道,只是不想说出来; 有时候不是不明白,而是明白了也不知道该怎么做,于是...
    久殇阅读 175评论 2 5
  • 宝哥,70后,和阿成是一个大院的发小,准确说是宝哥是阿成哥哥阿伟的同学,大阿成两岁,是大院的孩子王,阿成从小就跟在...
    冀泰来阅读 733评论 1 1
  • -1- 这两天,江先生的一个大学同学,小涛(化名),突然和他联系起来。 江先生说,要还他钱,他拒绝了。说手里不紧急...
    大麦07阅读 750评论 2 1