Android Input输入系统之五:KeyEvent按键调节音量加减流程

本文主要是介绍Android Input输入系统之五:KeyEvent按键调节音量加减流程,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

《Android按键Input KeyEvent》
《Android Input输入系统之一:KeyEvent事件监听及事件分发流程》
《Android Input输入系统之二:KeyEvent注入事件及事件分发流程》
《Android Input输入系统之三:KeyEvent事件分发和上层应用层对事件的接收》
《Android Input输入系统之四:KeyEvent事件中的InputChannel通信》
《Android Input输入系统之五:KeyEvent按键调节音量加减流程》

在上一章节,我们讲到了将按键消息传递分发给View。

在ViewPostImeInputStage阶段,调用了processKeyEvent(),其中:

// Deliver the key to the view hierarchy.
if (mView.dispatchKeyEvent(event)) {return FINISH_HANDLED;
}

通过dispatchKeyEvent将按键消息分发给了View.

那我们就从dispatchKeyEvent()说起。

View的根类是DecorView。

最终都会调用到DecorView中的dispatchKeyEvent()。
frameworks\base\core\java\com\android\internal\policy\DecorView.java

    @Overridepublic boolean dispatchKeyEvent(KeyEvent event) {final int keyCode = event.getKeyCode();final int action = event.getAction();final boolean isDown = action == KeyEvent.ACTION_DOWN;if (isDown && (event.getRepeatCount() == 0)) {// First handle chording of panel key: if a panel key is held// but not released, try to execute a shortcut in it.if ((mWindow.mPanelChordingKey > 0) && (mWindow.mPanelChordingKey != keyCode)) {boolean handled = dispatchKeyShortcutEvent(event);if (handled) {return true;}}// If a panel is open, perform a shortcut on it without the// chorded panel keyif ((mWindow.mPreparedPanel != null) && mWindow.mPreparedPanel.isOpen) {if (mWindow.performPanelShortcut(mWindow.mPreparedPanel, keyCode, event, 0)) {return true;}}}if (!mWindow.isDestroyed()) {final Window.Callback cb = mWindow.getCallback();final boolean handled = cb != null && mFeatureId < 0 ? cb.dispatchKeyEvent(event): super.dispatchKeyEvent(event);if (handled) {return true;}}return isDown ? mWindow.onKeyDown(mFeatureId, event.getKeyCode(), event): mWindow.onKeyUp(mFeatureId, event.getKeyCode(), event);}

cb 是一个Activity或者Dialog对象。用于回调上层应用app中监听的dispatchKeyEvent();

如果经过所有应用层的View,Activity等都没有消耗掉key事件,最终会走到PhoneWindow中:

return isDown ? mWindow.onKeyDown(mFeatureId, event.getKeyCode(), event): mWindow.onKeyUp(mFeatureId, event.getKeyCode(), event);

frameworks\base\core\java\com\android\internal\policy\PhoneWindow.java

    /*** A key was pressed down and not handled by anything else in the window.** @see #onKeyUp* @see android.view.KeyEvent*/protected boolean onKeyDown(int featureId, int keyCode, KeyEvent event) {/* ***************************************************************************** HOW TO DECIDE WHERE YOUR KEY HANDLING GOES.** If your key handling must happen before the app gets a crack at the event,* it goes in PhoneWindowManager.** If your key handling should happen in all windows, and does not depend on* the state of the current application, other than that the current* application can override the behavior by handling the event itself, it* should go in PhoneFallbackEventHandler.** Only if your handling depends on the window, and the fact that it has* a DecorView, should it go here.* ****************************************************************************/final KeyEvent.DispatcherState dispatcher =mDecor != null ? mDecor.getKeyDispatcherState() : null;//Log.i(TAG, "Key down: repeat=" + event.getRepeatCount()//        + " flags=0x" + Integer.toHexString(event.getFlags()));switch (keyCode) {case KeyEvent.KEYCODE_VOLUME_UP:case KeyEvent.KEYCODE_VOLUME_DOWN:case KeyEvent.KEYCODE_VOLUME_MUTE: {// If we have a session send it the volume command, otherwise// use the suggested stream.if (mMediaController != null) {mMediaController.dispatchVolumeButtonEventAsSystemService(event);} else {getMediaSessionManager().dispatchVolumeKeyEventAsSystemService(event,mVolumeControlStreamType);}return true;}//省略一部分代码//...}//省略一部分代码//...}

因为我们只分析音量的调节,这里列出来了KeyEvent.KEYCODE_VOLUME_UP的按键消息。

下面即使我们比较熟悉的音量调节流程了。我们走一遍这个流程。

这个流程分两个:
//UI 按钮画面上进行音量调节
dispatchVolumeButtonEventAsSystemService()

//硬件按键进行调节
dispatchVolumeKeyEventAsSystemService()

我们需要关注的是按键的调节。按钮调节逻辑简单跟一下。

按钮执行音量调节

dispatchVolumeButtonEventAsSystemService
frameworks\base\media\java\android\media\session\MediaController.java

public void dispatchVolumeButtonEventAsSystemService(@NonNull KeyEvent keyEvent) {switch (keyEvent.getAction()) {case KeyEvent.ACTION_DOWN: {int direction = 0;switch (keyEvent.getKeyCode()) {case KeyEvent.KEYCODE_VOLUME_UP:direction = AudioManager.ADJUST_RAISE;break;case KeyEvent.KEYCODE_VOLUME_DOWN:direction = AudioManager.ADJUST_LOWER;break;case KeyEvent.KEYCODE_VOLUME_MUTE:direction = AudioManager.ADJUST_TOGGLE_MUTE;break;}try {mSessionBinder.adjustVolume(mContext.getPackageName(), mCbStub, true, direction,AudioManager.FLAG_SHOW_UI);} catch (RemoteException e) {Log.wtf(TAG, "Error calling adjustVolumeBy", e);}}case KeyEvent.ACTION_UP: {final int flags = AudioManager.FLAG_PLAY_SOUND | AudioManager.FLAG_VIBRATE| AudioManager.FLAG_FROM_KEY;try {mSessionBinder.adjustVolume(mContext.getPackageName(), mCbStub, true, 0, flags);} catch (RemoteException e) {Log.wtf(TAG, "Error calling adjustVolumeBy", e);}}}
}

/frameworks/base/services/core/java/com/android/server/media/MediaSessionRecord.java
通过ISessionController的AIDL方式调用到MediaSessionRecord中的adjustVolume().
通过SessionCb中的回调,调用mCb.onAdjustVolume()即MediaSession中的onAdjustVolume().

\frameworks\base\media\java\android\media\session\MediaSession.java

@Override
public void onAdjustVolume(String packageName, int pid, 

这篇关于Android Input输入系统之五:KeyEvent按键调节音量加减流程的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/733588

相关文章

通过Docker容器部署Python环境的全流程

《通过Docker容器部署Python环境的全流程》在现代化开发流程中,Docker因其轻量化、环境隔离和跨平台一致性的特性,已成为部署Python应用的标准工具,本文将详细演示如何通过Docker容... 目录引言一、docker与python的协同优势二、核心步骤详解三、进阶配置技巧四、生产环境最佳实践

MyBatis分页查询实战案例完整流程

《MyBatis分页查询实战案例完整流程》MyBatis是一个强大的Java持久层框架,支持自定义SQL和高级映射,本案例以员工工资信息管理为例,详细讲解如何在IDEA中使用MyBatis结合Page... 目录1. MyBATis框架简介2. 分页查询原理与应用场景2.1 分页查询的基本原理2.1.1 分

JWT + 拦截器实现无状态登录系统

《JWT+拦截器实现无状态登录系统》JWT(JSONWebToken)提供了一种无状态的解决方案:用户登录后,服务器返回一个Token,后续请求携带该Token即可完成身份验证,无需服务器存储会话... 目录✅ 引言 一、JWT 是什么? 二、技术选型 三、项目结构 四、核心代码实现4.1 添加依赖(pom

redis-sentinel基础概念及部署流程

《redis-sentinel基础概念及部署流程》RedisSentinel是Redis的高可用解决方案,通过监控主从节点、自动故障转移、通知机制及配置提供,实现集群故障恢复与服务持续可用,核心组件包... 目录一. 引言二. 核心功能三. 核心组件四. 故障转移流程五. 服务部署六. sentinel部署

SpringBoot集成XXL-JOB实现任务管理全流程

《SpringBoot集成XXL-JOB实现任务管理全流程》XXL-JOB是一款轻量级分布式任务调度平台,功能丰富、界面简洁、易于扩展,本文介绍如何通过SpringBoot项目,使用RestTempl... 目录一、前言二、项目结构简述三、Maven 依赖四、Controller 代码详解五、Service

基于Python实现自动化邮件发送系统的完整指南

《基于Python实现自动化邮件发送系统的完整指南》在现代软件开发和自动化流程中,邮件通知是一个常见且实用的功能,无论是用于发送报告、告警信息还是用户提醒,通过Python实现自动化的邮件发送功能都能... 目录一、前言:二、项目概述三、配置文件 `.env` 解析四、代码结构解析1. 导入模块2. 加载环

Android协程高级用法大全

《Android协程高级用法大全》这篇文章给大家介绍Android协程高级用法大全,本文结合实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友跟随小编一起学习吧... 目录1️⃣ 协程作用域(CoroutineScope)与生命周期绑定Activity/Fragment 中手

linux系统上安装JDK8全过程

《linux系统上安装JDK8全过程》文章介绍安装JDK的必要性及Linux下JDK8的安装步骤,包括卸载旧版本、下载解压、配置环境变量等,强调开发需JDK,运行可选JRE,现JDK已集成JRE... 目录为什么要安装jdk?1.查看linux系统是否有自带的jdk:2.下载jdk压缩包2.解压3.配置环境

MySQL 临时表与复制表操作全流程案例

《MySQL临时表与复制表操作全流程案例》本文介绍MySQL临时表与复制表的区别与使用,涵盖生命周期、存储机制、操作限制、创建方法及常见问题,本文结合实例代码给大家介绍的非常详细,感兴趣的朋友跟随小... 目录一、mysql 临时表(一)核心特性拓展(二)操作全流程案例1. 复杂查询中的临时表应用2. 临时

Linux查询服务器系统版本号的多种方法

《Linux查询服务器系统版本号的多种方法》在Linux系统管理和维护工作中,了解当前操作系统的版本信息是最基础也是最重要的操作之一,系统版本不仅关系到软件兼容性、安全更新策略,还直接影响到故障排查和... 目录一、引言:系统版本查询的重要性二、基础命令解析:cat /etc/Centos-release详