Android6.0 源码修改之Settings音量调节界面增加通话音量调节

简介: Android6.0 源码修改之Settings音量调节界面增加通话音量调节

Android6.0 源码修改之Settings音量调节界面增加通话音量调节


前言


今天客户提了个需求,因为我们的设备在正常情况下无法调节通话音量,只有在打电话过程中,按物理音量加减键才能出现调节通话音量seekBar,很不方便,于是乎需求就来了。需要优化两个地方


1、在正常情况下,按物理音量加减键都显示 通话音量调节seekBar,可方便快速调节通话音量

2、在Settings中提示音界面点击设置进入,增加通话音量调节seekBar

20190416181909188.png

20190416182007485.png

修改前

20190416182030130.png


20190416182119988.png


修改后

实现

第一个功能


先来完成第一个功能,还是通过Hierarchy View查看布局结构,查找到布局文件id为volume_dialog,通过在源码中搜索找到位于SystemUI中,volume_dialog.xml

源码位置 frameworks\base\packages\SystemUI\res\layout\volume_dialog.xml

对应的java类为 frameworks\base\packages\SystemUI\src\com\android\systemui\volume\VolumeDialog.java

修改代码

addRow(AudioManager.STREAM_VOICE_CALL,
            R.drawable.ic_volume_voice, R.drawable.ic_volume_voice, true);

原来的第四个参数为false,修改为true即可显示通话音量seekBar

为了便于说明,我们跟进addRow()中查看

private void addRow(int stream, int iconRes, int iconMuteRes, boolean important) {
    final VolumeRow row = initRow(stream, iconRes, iconMuteRes, important);
    if (!mRows.isEmpty()) {
        final View v = new View(mContext);
        v.setId(android.R.id.background);
        final int h = mContext.getResources()
                .getDimensionPixelSize(R.dimen.volume_slider_interspacing);
        final LinearLayout.LayoutParams lp =
                new LinearLayout.LayoutParams(LinearLayout.LayoutParams.MATCH_PARENT, h);
        mDialogContentView.addView(v, mDialogContentView.getChildCount() - 1, lp);
        row.space = v;
    }
  ...
}


传递的参数对应important,从字面意思理解重要对应显示,继续查看initRow都做了什么

private VolumeRow initRow(final int stream, int iconRes, int iconMuteRes, boolean important) {
    final VolumeRow row = new VolumeRow();
    row.stream = stream;
    row.iconRes = iconRes;
    row.iconMuteRes = iconMuteRes;
    row.important = important;
    row.view = mDialog.getLayoutInflater().inflate(R.layout.volume_dialog_row, null);
    row.view.setTag(row);
    row.header = (TextView) row.view.findViewById(R.id.volume_row_header);
    mSpTexts.add(row.header);
    row.slider = (SeekBar) row.view.findViewById(R.id.volume_row_slider);
    row.slider.setOnSeekBarChangeListener(new VolumeSeekBarChangeListener(row));
    // forward events above the slider into the slider
    row.view.setOnTouchListener(new OnTouchListener() {
        private final Rect mSliderHitRect = new Rect();
        private boolean mDragging;
        @SuppressLint("ClickableViewAccessibility")
        @Override
        public boolean onTouch(View v, MotionEvent event) {
            row.slider.getHitRect(mSliderHitRect);
            if (!mDragging && event.getActionMasked() == MotionEvent.ACTION_DOWN
                    && event.getY() < mSliderHitRect.top) {
                mDragging = true;
            }
            if (mDragging) {
                event.offsetLocation(-mSliderHitRect.left, -mSliderHitRect.top);
                row.slider.dispatchTouchEvent(event);
                if (event.getActionMasked() == MotionEvent.ACTION_UP
                        || event.getActionMasked() == MotionEvent.ACTION_CANCEL) {
                    mDragging = false;
                }
                return true;
            }
            return false;
        }
    });
    row.icon = (ImageButton) row.view.findViewById(R.id.volume_row_icon);
    row.icon.setImageResource(iconRes);
    row.icon.setOnClickListener(new OnClickListener() {
        @Override
        public void onClick(View v) {
            Events.writeEvent(mContext, Events.EVENT_ICON_CLICK, row.stream, row.iconState);
            mController.setActiveStream(row.stream);
            if (row.stream == AudioManager.STREAM_RING) {
                final boolean hasVibrator = mController.hasVibrator();
                if (mState.ringerModeInternal == AudioManager.RINGER_MODE_NORMAL) {
                    if (hasVibrator) {
                        mController.setRingerMode(AudioManager.RINGER_MODE_VIBRATE, false);
                    } else {
                        final boolean wasZero = row.ss.level == 0;
                        mController.setStreamVolume(stream, wasZero ? row.lastAudibleLevel : 0);
                    }
                } else {
                    mController.setRingerMode(AudioManager.RINGER_MODE_NORMAL, false);
                    if (row.ss.level == 0) {
                        mController.setStreamVolume(stream, 1);
                    }
                }
            } else {
                final boolean vmute = row.ss.level == 0;
                mController.setStreamVolume(stream, vmute ? row.lastAudibleLevel : 0);
            }
            row.userAttempt = 0;  // reset the grace period, slider should update immediately
        }
    });
    row.settingsButton = (ImageButton) row.view.findViewById(R.id.volume_settings_button);
    row.settingsButton.setOnClickListener(mClickSettings);
    return row;
}

从上面可看出,将一些变量都保存到了VolumeRow中,设置了icon的点击事件,将当前对应的音量类型设置为最低(禁音), 设置seekBar的改变事件。通过过滤日志,查找到控制音量类型的显示和隐藏的代码块updateRowsH()

private boolean isVisibleH(VolumeRow row, boolean isActive) {
    return mExpanded && row.view.getVisibility() == View.VISIBLE
            || (mExpanded && (row.important || isActive))
            || !mExpanded && isActive;
}
private void updateRowsH() {
    if (D.BUG) Log.d(TAG, "updateRowsH");
    final VolumeRow activeRow = getActiveRow();
    updateFooterH();
    updateExpandButtonH();
    if (!mShowing) {
        trimObsoleteH();
    }
    // apply changes to all rows
    for (VolumeRow row : mRows) {
        final boolean isActive = row == activeRow;
        final boolean visible = isVisibleH(row, isActive);
        Log.e(TAG, "row==" + row.stream + " isActive=="+isActive + " visible="+visible);
        Util.setVisOrGone(row.view, visible);
        Util.setVisOrGone(row.space, visible && mExpanded);
        final int expandButtonRes = mExpanded ? R.drawable.ic_volume_settings : 0;
        if (expandButtonRes != row.cachedExpandButtonRes) {
            row.cachedExpandButtonRes = expandButtonRes;
            if (expandButtonRes == 0) {
                row.settingsButton.setImageDrawable(null);
            } else {
                row.settingsButton.setImageResource(expandButtonRes);
            }
        }
        Util.setVisOrInvis(row.settingsButton, false);
        updateVolumeRowHeaderVisibleH(row);
        row.header.setAlpha(mExpanded && isActive ? 1 : 0.5f);
        updateVolumeRowSliderTintH(row, isActive);
    }
}


遍历已经添加的音量类型集合mRows,依次判断是否处于活动状态,再和开始设置的important属性比较。mExpanded是否展开,默认只显示铃声音量控制,点击下拉的按钮,才完全显示其它的音量控制


mExpanded && row.view.getVisibility() == View.VISIBLE || (mExpanded && (row.important || isActive)) || !mExpanded && isActive


true && false || (true && (true || false)) || false && true —>true


好了,至此分析完毕,重新mmm push SystemUI.apk 查看效果


第二个功能

源码位置


Settings\res_ext\xml\edit_profile_prefs.xml

Settings\src\com\mediatek\audioprofile\Editprofile.java

Settings\src\com\mediatek\audioprofile\VolumeSeekBarPreference.java


在edit_profile_prefs.xml中仿照原来的Alarm volume和Ring volume,新增加一个Call volume


<!-- Media volume -->
<com.mediatek.audioprofile.VolumeSeekBarPreference
        android:key="media_volume"
        android:icon="@*android:drawable/ic_audio_vol"
        android:title="@string/media_volume_option_title" />
<!-- Alarm volume -->
<com.mediatek.audioprofile.VolumeSeekBarPreference
        android:key="alarm_volume"
        android:icon="@*android:drawable/ic_audio_alarm"
        android:title="@string/alarm_volume_option_title" />
<!-- Ring volume -->
<com.mediatek.audioprofile.VolumeSeekBarPreference
        android:key="ring_volume"
        android:icon="@*android:drawable/ic_audio_ring_notif"
        android:title="@string/ring_volume_option_title" />
<!-- Call volume -->
<com.mediatek.audioprofile.VolumeSeekBarPreference
        android:key="call_volume"
        android:icon="@drawable/ic_volume_voice"
        android:title="@string/call_volume_option_title" />

对应的drawable文件时从SystemUI中拷贝过来的,ic_volume_voice.xml

<vector xmlns:android="http://schemas.android.com/apk/res/android"
android:height="24.0dp"
android:viewportHeight="48.0"
android:viewportWidth="48.0"
android:width="24.0dp" >
<path
    android:fillColor="#ff727272"
    android:pathData="M13.25,21.59c2.88,5.66 7.51,10.29 13.18,13.17l4.4,-4.41c0.55,-0.55 1.34,-0.71 2.03,-0.49C35.1,30.6 37.51,31.0 40.0,31.0c1.11,0.0 2.0,0.89 2.0,2.0l0.0,7.0c0.0,1.11 -0.89,2.0 -2.0,2.0C21.22,42.0 6.0,26.78 6.0,8.0c0.0,-1.1 0.9,-2.0 2.0,-2.0l7.0,0.0c1.11,0.0 2.0,0.89 2.0,2.0 0.0,2.4 0.4,4.9 1.14,7.1 0.2,0.6 0.06,1.48 -0.49,2.03l-4.4,4.42z" />
</vector>

接下来对应到 Editprofile.java 文件中,可以看到 KEY_ALARM_VOLUME 对应的preference初始化,依旧照葫芦画瓢,添加 KEY_CALL_VOLUME

private void initVolume(PreferenceScreen parent) {
    initVolumePreference(KEY_MEDIA_VOLUME, AudioManager.STREAM_MUSIC);
    initVolumePreference(KEY_ALARM_VOLUME, AudioManager.STREAM_ALARM);
    initVolumePreference(KEY_CALL_VOLUME, AudioManager.STREAM_VOICE_CALL);
    if (mVoiceCapable) {
        mVolume = initVolumePreference(KEY_RING_VOLUME, AudioManager.STREAM_RING);
        parent.removePreference(parent.findPreference(KEY_NOTIFICATION_VOLUME));
    } else {
        mVolume = initVolumePreference(KEY_NOTIFICATION_VOLUME,
                AudioManager.STREAM_NOTIFICATION);
        parent.removePreference(parent.findPreference(KEY_RING_VOLUME));
    }
}

重新编译,push替换后发现,UI倒是出来了,但是无法滑动,事情果然没那么简单,继续查看 initVolumePreference()

private VolumeSeekBarPreference initVolumePreference(String key, int stream) {
    Log.d("@M_" + TAG, "Init volume preference, key = " + key + ",stream = " + stream);
    final VolumeSeekBarPreference volumePref = (VolumeSeekBarPreference) findPreference(key);
    volumePref.setStream(stream);
    volumePref.setCallback(mVolumeCallback);
    volumePref.setProfile(mKey);
    return volumePref;
}


保存了当前的音量调节类型,设置seekBar回调事件,接下来看看回调处理了什么

private final class VolumePreferenceCallback implements VolumeSeekBarPreference.Callback {
    private SeekBarVolumizer mCurrent;
    @Override
    public void onSampleStarting(SeekBarVolumizer sbv) {
        if (mCurrent != null && mCurrent != sbv) {
            mCurrent.stopSample();
        }
        mCurrent = sbv;
        if (mCurrent != null) {
            mHandler.removeMessages(H.STOP_SAMPLE);
            mHandler.sendEmptyMessageDelayed(H.STOP_SAMPLE, SAMPLE_CUTOFF);
        }
    }
    public void onStreamValueChanged(int stream, int progress) {
        if (stream == AudioManager.STREAM_RING) {
            mHandler.removeMessages(H.UPDATE_RINGER_ICON);
            mHandler.obtainMessage(H.UPDATE_RINGER_ICON, progress, 0).sendToTarget();
        }
    }
    public void stopSample() {
        if (mCurrent != null) {
            mCurrent.stopSample();
        }
    }
    public void ringtoneChanged() {
        if (mCurrent != null) {
            mCurrent.ringtoneChanged();
        } else {
            mVolume.getSeekBar().ringtoneChanged();
        }
    }
}


当我们点击或者是滑动seekBar时,会根据当前设置的音量大小播放一段短暂的默认铃音,当铃音未播放完成时,再次点击将不进行播放。继续跟进 VolumeSeekBarPreference.java 中

@Override
protected void onBindView(View view) {
    super.onBindView(view);
    if (mStream == 0) {
        Log.w(TAG, "No stream found, not binding volumizer  ");
        return;
    }
    getPreferenceManager().registerOnActivityStopListener(this);
    final SeekBar seekBar = (SeekBar) view.findViewById(com.android.internal.R.id.seekbar);
    if (seekBar == mSeekBar) {
        return;
    }
    mSeekBar = seekBar;
    final SeekBarVolumizer.Callback sbvc = new SeekBarVolumizer.Callback() {
        @Override
        public void onSampleStarting(SeekBarVolumizer sbv) {
            if (mCallback != null) {
                mCallback.onSampleStarting(sbv);
            }
        }
    };
    final Uri sampleUri = mStream == AudioManager.STREAM_MUSIC ? getMediaVolumeUri() : null;
    if (mVolumizer == null) {
        mVolumizer = new SeekBarVolumizer(getContext(), mStream, sampleUri, sbvc, mKey);
    }
    //mVolumizer.setProfile(mKey);
    mVolumizer.setSeekBar(mSeekBar);
}

mStream == 0, 直接return,会不会和这有关系呢,来看下各个音量调节类型对应的int值

/** Used to identify the volume of audio streams for phone calls */
public static final int STREAM_VOICE_CALL = 0;
/** Used to identify the volume of audio streams for the phone ring and message alerts */
public static final int STREAM_RING = 2;
/** Used to identify the volume of audio streams for music playback */
public static final int STREAM_MUSIC = 3;
/** Used to identify the volume of audio streams for alarms */
public static final int STREAM_ALARM = 4;


我们新增的 STREAM_VOICE_CALL对应的mStream正好为0,直接给return掉了,所以修改为 mStream < 0 即可,重新编译 push 发现成功了。


具体的音量调节逻辑在 packages\apps\Settings\src\com\mediatek\audioprofile\SeekBarVolumizer.java 中,感兴趣的可继续深究,肯定离不开调用 .setStreamVolume()方法,大概看了一眼,主要是onProgressChanged()回调,通过postSetVolume(progress)方法,发送MSG_SET_STREAM_VOLUME消息,最终调用saveVolume()

目录
相关文章
|
6月前
|
Ubuntu 开发工具 Android开发
Repo下载AOSP源码:基于ubuntu22.04 环境配置,android-12.0.0_r32
本文介绍了在基于Ubuntu 22.04的环境下配置Python 3.9、安装repo工具、下载和同步AOSP源码包以及处理repo同步错误的详细步骤。
391 0
Repo下载AOSP源码:基于ubuntu22.04 环境配置,android-12.0.0_r32
|
4月前
|
XML 数据可视化 Android开发
Android应用界面
Android应用界面中的布局和控件使用,包括相对布局、线性布局、表格布局、帧布局、扁平化布局等,以及AdapterView及其子类如ListView的使用方法和Adapter接口的应用。
50 0
Android应用界面
|
5月前
|
XML Android开发 UED
💥Android UI设计新风尚!掌握Material Design精髓,让你的界面颜值爆表!🎨
随着移动应用市场的蓬勃发展,用户对界面设计的要求日益提高。为此,掌握由Google推出的Material Design设计语言成为提升应用颜值和用户体验的关键。本文将带你深入了解Material Design的核心原则,如真实感、统一性和创新性,并通过丰富的组件库及示例代码,助你轻松打造美观且一致的应用界面。无论是色彩搭配还是动画效果,Material Design都能为你的Android应用增添无限魅力。
125 1
|
5月前
|
Web App开发 网络协议 Android开发
Android平台一对一音视频通话方案大比拼:WebRTC VS RTMP VS RTSP,谁才是王者?
【9月更文挑战第4天】本文详细对比了在Android平台上实现一对一音视频通话时常用的WebRTC、RTMP及RTSP三种技术方案。从技术原理、性能表现与开发难度等方面进行了深入分析,并提供了示例代码。WebRTC适合追求低延迟和高质量的场景,但开发成本较高;RTMP和RTSP则在简化开发流程的同时仍能保持较好的传输效果,适用于不同需求的应用场景。
281 1
|
6月前
|
开发工具 git 索引
repo sync 更新源码 android-12.0.0_r34, fatal: 不能重置索引文件至版本 ‘v2.27^0‘。
本文描述了在更新AOSP 12源码时遇到的repo同步错误,并提供了通过手动git pull更新repo工具来解决这一问题的方法。
223 1
|
6月前
|
开发工具 uml git
AOSP源码下载方法,解决repo sync错误:android-13.0.0_r82
本文分享了下载AOSP源码的方法,包括如何使用repo工具和处理常见的repo sync错误,以及配置Python环境以确保顺利同步特定版本的AOSP代码。
851 0
AOSP源码下载方法,解决repo sync错误:android-13.0.0_r82
|
6月前
|
Java Android开发 芯片
使用Android Studio导入Android源码:基于全志H713 AOSP,方便解决编译、编码问题
本文介绍了如何将基于全志H713芯片的AOSP Android源码导入Android Studio以解决编译和编码问题,通过操作步骤的详细说明,展示了在Android Studio中利用代码提示和补全功能快速定位并修复编译错误的方法。
351 0
使用Android Studio导入Android源码:基于全志H713 AOSP,方便解决编译、编码问题
|
6月前
|
Android开发 iOS开发 C#
Xamarin.Forms:从零开始的快速入门指南——打造你的首个跨平台移动应用,轻松学会用C#和XAML构建iOS与Android通用界面的每一个步骤
【8月更文挑战第31天】Xamarin.Forms 是一个强大的框架,让开发者通过单一共享代码库构建跨平台移动应用,支持 iOS、Android 和 Windows。使用 C# 和 XAML,它简化了多平台开发流程并保持一致的用户体验。本指南通过创建一个简单的 “HelloXamarin” 应用演示了 Xamarin.Forms 的基本功能和工作原理。
151 0
|
6月前
|
开发工具 Android开发 git
全志H713 Android 11 :给AOSP源码,新增一个Product
本文介绍了在全志H713 Android 11平台上新增名为myboard的产品的步骤,包括创建新的device目录、编辑配置文件、新增内核配置、记录差异列表以及编译kernel和Android系统的详细过程。
394 0
|
6月前
|
Ubuntu 开发工具 Android开发
Repo下载、编译AOSP源码:基于Ubuntu 21.04,android-12.1.0_r27
文章记录了作者在Ubuntu 21.04服务器上配置环境、下载并编译基于Android 12.1.0_r27版本的AOSP源码的过程,包括解决编译过程中遇到的问题和错误处理方法。
367 0

热门文章

最新文章

  • 1
    如何修复 Android 和 Windows 不支持视频编解码器的问题?
  • 2
    【08】flutter完成屏幕适配-重建Android,增加GetX路由,屏幕适配,基础导航栏-多版本SDK以及gradle造成的关于fvm的使用(flutter version manage)-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
  • 3
    当flutter react native 等混开框架-并且用vscode-idea等编译器无法打包apk,打包安卓不成功怎么办-直接用android studio如何打包安卓apk -重要-优雅草卓伊凡
  • 4
    【04】flutter补打包流程的签名过程-APP安卓调试配置-结构化项目目录-完善注册相关页面-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程
  • 5
    APP-国内主流安卓商店-应用市场-鸿蒙商店上架之必备前提·全国公安安全信息评估报告如何申请-需要安全评估报告的资料是哪些-优雅草卓伊凡全程操作
  • 6
    【09】flutter首页进行了完善-采用android studio 进行真机调试开发-增加了直播间列表和短视频人物列表-增加了用户中心-卓伊凡换人优雅草Alex-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草Alex
  • 7
    Android经典面试题之Kotlin中Lambda表达式和匿名函数的区别
  • 8
    【02】仿站技术之python技术,看完学会再也不用去购买收费工具了-本次找了小影-感觉页面很好看-本次是爬取vue需要用到Puppeteer库用node.js扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-优雅草卓伊凡
  • 9
    【01】仿站技术之python技术,看完学会再也不用去购买收费工具了-用python扒一个app下载落地页-包括安卓android下载(简单)-ios苹果plist下载(稍微麻烦一丢丢)-客户的麻将软件需要下载落地页并且要做搜索引擎推广-本文用python语言快速开发爬取落地页下载-优雅草卓伊凡
  • 10
    Android 13 SystemUI 启动流程