Android6.0 源碼修改之Settings音量調節界面增加通話音量調節

来源:https://www.cnblogs.com/cczheng-666/archive/2019/04/23/10757358.html
-Advertisement-
Play Games

前言 今天客戶提了個需求,因為我們的設備在正常情況下無法調節通話音量,只有在打電話過程中,按物理音量加減鍵才能出現調節通話音量seekBar,很不方便,於是乎需求就來了。需要優化兩個地方 1、在正常情況下,按物理音量加減鍵都顯示 通話音量調節seekBar,可方便快速調節通話音量 2、在Settin ...


前言

今天客戶提了個需求,因為我們的設備在正常情況下無法調節通話音量,只有在打電話過程中,按物理音量加減鍵才能出現調節通話音量seekBar,很不方便,於是乎需求就來了。需要優化兩個地方

1、在正常情況下,按物理音量加減鍵都顯示 通話音量調節seekBar,可方便快速調節通話音量

2、在Settings中提示音界面點擊設置進入,增加通話音量調節seekBar

在這裡插入圖片描述
在這裡插入圖片描述

修改前

在這裡插入圖片描述
在這裡插入圖片描述

修改後

實現

第一個功能

先來完成第一個功能,還是通過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()


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 加鎖的主要目的是為了防止併發操作時導致的數據不一致等問題,鎖分為共用鎖(S)、更新鎖(U)、排他鎖(X),共用鎖與更新只是單向相容?傳說中的單相思? 事務 事務能保證數據操作的原子性,要麼內部操作都提交,要麼都回退。事務內部某個地方出錯時,可以回滾前面的操作,比如更新、刪除等。 共用鎖 共用鎖允許並 ...
  • 在資料庫的鎖機制中介紹過,資料庫管理系統(DBMS)中的併發控制的任務是確保在多個事務同時存取資料庫中同一數據時不破壞事務的隔離性和統一性以及資料庫的統一性。 樂觀併發控制(樂觀鎖)和悲觀併發控制(悲觀鎖)是併發控制主要採用的技術手段。 無論是悲觀鎖還是樂觀鎖,都是人們定義出來的概念,可以認為是一種 ...
  • 一、統計語句 1、--統計當前【>當天00點以後的數據】 SELECT * FROM 表 WHERE CONVERT(Nvarchar, dateandtime, 111) = CONVERT(Nvarchar, GETDATE(), 111) ORDER BY dateandtime DESC 2 ...
  • 安裝redis-5.0.4 修改配置文件 創建命令鏈接 設置redis開機自啟動 啟動redis服務 測試連接 安裝過程可能出現的問題 1.CentOS預設沒有安裝gcc,這會導致我們無法make成功 2.make時報如下錯誤: zmalloc.h:50:31: fatal error: jemal ...
  • [20190423]那個更快的疑問3.txt--//前一陣子,做了11g在單表單條記錄唯一索引掃描的測試,摘要如下:--//參考鏈接:http://blog.itpub.net/267265/viewspace-2636321/http://blog.itpub.net/267265/viewspa ...
  • 【問題描述】 開發反饋,有一個SQL Server數據同步的作業,從Table1 拉取數據,主鍵是ID, 每次拉取批次數據的SQL語句是 select top (15) from Table1(NOLOCK) where ID ?,?代表的是上次同步批次中最後一個ID號。 某一次拉取到的數據為 ID ...
  • count()是聚合函數,對於返回的結果集,一行行地判斷,累計值加1,最後返回累計值,count(*)、count(主鍵ID)和count(1)表示返回滿足條件的結果集的總行數。 count()聚合函數統計非NULL與NULL值的區別: 1、count(欄位)不統計NULL記錄,即表示滿足條件的數據 ...
  • 版權聲明:本文為HaiyuKing原創文章,轉載請註明出處! 前言 這個Demo只是記錄華為推送的集成,不能運行。 另外,因為可能用到存儲許可權,所以還需要搭配運行時許可權申請功能。 使用步驟 一、項目組織結構圖 註意事項: 1、 導入類文件後需要change包名以及重新import R文件路徑 2、 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...