Android6.0源码分析之menu键弹出popupwindow菜单流程分析

例如上图,在按下菜单键后会弹出对应的菜单选项,准确来说,是在菜单键弹起后出现的一个popupwindow,那么从菜单键弹起到popupwindow创建所涉及到的历程是怎样的呢?

理论上是底层监测menu按键键值,通知framework层,framework经过一系列的处理后分发给上层,或者拦截掉

现在是假设framework层已经把menu按键分发给了用户,那么接下里就是activity进行响应,所以从Activity开始看起,整体流程如下:

从按键弹起framework层分发到activity开始,到弹出popupwindow结束整个流程图

代码所在目录如下

流程图中所涉及到的一些只是单纯的作为中介调用了一下,按着流程自己也可以看到,现在是挑一些有用的进行分析

1,第一个需要分析的就是Activity.java文件中的分发menu按键的方法

  /**
     * Called to process key events.  You can override this to intercept all
     * key events before they are dispatched to the window.  Be sure to call
     * this implementation for key events that should be handled normally.
     *
     * @param event The key event.
     *
     * @return boolean Return true if this event was consumed.
     */
    public boolean dispatchKeyEvent(KeyEvent event) {
        onUserInteraction();

        // Let action bars open menus in response to the menu key prioritized over
        // the window handling it
        if (event.getKeyCode() == KeyEvent.KEYCODE_MENU &&
                mActionBar != null && mActionBar.onMenuKeyEvent(event)) {
           //当menu按键到来时,先判断下actionBar是否为null,如果actionbar不为空再进行处理,即让actionbar响应按键事件  
           
           return true;
        }

        Window win = getWindow();//获取到当前activity的窗口实例
        if (win.superDispatchKeyEvent(event)) {
           //如果acitonbar没有对按键进行处理,则传递给activity所在窗口进行处理 
         return true;
        }
        View decor = mDecor;//先介绍一下,activity的布局的根节点为DecorView,在这里获取到activity的根节点
        if (decor == null) decor = win.getDecorView();
      //如果actionbar没有处理event,window窗口也没处理或者拦截event,则传递给view进行处理  
      return event.dispatch(this, decor != null
                ? decor.getKeyDispatcherState() : null, this);
    }

处理分发按键事件时会调用,用户也可以自己覆写,来达到按自己的意愿处理按键事件的目的,比如拦截按键传递给窗口,view等。 通过以上方法的分析可以总结出,一个menu事件在传递给activity后,如果不是menu事件就会直接交给window会向下传递,有三方可能进行处理 actionbar window view 这三个优先级由高到低,也就是说,menu事件会首先传递给actionbar,如果actionbar进行了处理并且将事件拦截下来不派发给窗口,那么menu事件到actionbar处理之后就结束了。反之则会向window和view进行传递。 2,流程中第二个值得分析的就在ActionMenuPresenter.java中

/**
     * Display the overflow menu if one is present.
     * @return true if the overflow menu was shown, false otherwise.
     */
    public boolean showOverflowMenu() {
        if (mReserveOverflow && !isOverflowMenuShowing() && mMenu != null && mMenuView != null &&
                mPostedOpenRunnable == null && !mMenu.getNonActionItems().isEmpty()) {

          //if条件进行了以下几个判断,mReserveOverflow判断了是否显示overflow按钮,isOverflowMenuShowing表示了菜单项的popupwindow是否正在显示
          //,以及要打开菜单项的进程是否已经初始化,菜单项是否不是空

            //初始化popupwindow对象
           OverflowPopup popup = new OverflowPopup(mContext, mMenu, mOverflowButton, true);
           开启线程,加载popupwindow 
          mPostedOpenRunnable = new OpenOverflowRunnable(popup);
            // Post this for later; we might still need a layout for the anchor to be right.
            ((View) mMenuView).post(mPostedOpenRunnable);

            // ActionMenuPresenter uses null as a callback argument here
            // to indicate overflow is opening.
            super.onSubMenuSelected(null);

            return true;
        }
        return false;
    }

总结下来就是,该方法先对一些必要的条件做了一些判断,比如menu是否存在,menu菜单选项是否为空,menu的popup是否已经弹出,或者正在弹出,然后在进行popup对象的实例化,并开启加载popup的线程。

private class OpenOverflowRunnable implements Runnable {
        private OverflowPopup mPopup;

        public OpenOverflowRunnable(OverflowPopup popup) {
            mPopup = popup;
        }

        public void run() {
            mMenu.changeMenuMode();
            final View menuView = (View) mMenuView;
            if (menuView != null && menuView.getWindowToken() != null && mPopup.tryShow()) {
                mOverflowPopup = mPopup;
            }
            mPostedOpenRunnable = null;
        }
    }

在加载线程中修改menu的状态模式,并且试着去弹出popup,mPopup.tryShow(); 到这里menu按下从activity分发到popup的弹起就分析完了

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏编程

python3.6爬虫之豆瓣小王子详解(一)

目标:抓取豆瓣小王子读者的书评,并保存于本地excel。 爬虫的流程:发送请求——获得页面——解析页面——抽取并储存内容。 我们第一个案例便采用:request...

1947
来自专栏ml

CEF使用的几个注意点

    CEF为chrome浏览器的切入其他浏览器中的轻量级框架。 开发的客户端的时候,这是作为界面显示的首先,可以增强客户的易变性,可塑性。 在开发的过程中(...

61310
来自专栏前端迷

ajax和fetch、axios的优缺点以及比较

前端是个发展迅速的领域,前端请求自然也发展迅速,从原生的XHR到jquery ajax,再到现在的axios和fetch。

2.4K2
来自专栏python学习之旅

Python网络爬虫笔记(五):下载、分析京东P20销售数据

1、      翻页的时候,谷歌F12的Network页签可以看到下面的请求。(这里的翻页指商品评价中1、2、3页等)

5137
来自专栏晨星先生的自留地

从协议提取到多功能RDP识别脚本

3218
来自专栏Jack-Cui

Python3网络爬虫快速入门实战解析

Python版本: Python3.x 运行平台: Windows IDE: Sublime text3 一 前言 强烈建议:请在电脑的陪同下,阅读...

8448
来自专栏ShaoYL

iOS逆向实战与工具使用(微信添加好友自动确认)

5556
来自专栏小樱的经验随笔

BugkuCTF web3

9384
来自专栏GreenLeaves

JavaScrtip之JS最佳实践

一、JavaScript之平稳退化 这边使用一个当用户点击某个页面内某个链接弹出一个新窗口的案例: JavaScript使用window对象的open()方法来...

2325
来自专栏葡萄城控件技术团队

提高性能:用RequireJS优化Wijmo Web页面

上周Wijmo 2014 V2版本刚刚发布(下载地址),  有网友下载后发现仅仅使用了40个Widgets的一小部分,还需要加载全部的jquery.wijmo-...

2055

扫码关注云+社区

领取腾讯云代金券