Merge "Add master audio balance setting"
This commit is contained in:
109
res/layout/preference_balance_slider.xml
Normal file
109
res/layout/preference_balance_slider.xml
Normal file
@@ -0,0 +1,109 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<!-- Copyright (C) 2019 The Android Open Source Project
|
||||||
|
|
||||||
|
Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
you may not use this file except in compliance with the License.
|
||||||
|
You may obtain a copy of the License at
|
||||||
|
|
||||||
|
http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
|
||||||
|
Unless required by applicable law or agreed to in writing, software
|
||||||
|
distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
See the License for the specific language governing permissions and
|
||||||
|
limitations under the License.
|
||||||
|
-->
|
||||||
|
|
||||||
|
<LinearLayout
|
||||||
|
xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:minHeight="?android:attr/listPreferredItemHeightSmall"
|
||||||
|
android:gravity="center_vertical"
|
||||||
|
android:paddingStart="?android:attr/listPreferredItemPaddingStart"
|
||||||
|
android:paddingEnd="?android:attr/listPreferredItemPaddingEnd"
|
||||||
|
android:clickable="false"
|
||||||
|
android:orientation="horizontal">
|
||||||
|
|
||||||
|
<LinearLayout
|
||||||
|
android:id="@+id/icon_frame"
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:minWidth="56dp"
|
||||||
|
android:gravity="start|center_vertical"
|
||||||
|
android:orientation="horizontal"
|
||||||
|
android:paddingEnd="12dp"
|
||||||
|
android:paddingTop="4dp"
|
||||||
|
android:paddingBottom="4dp">
|
||||||
|
<com.android.internal.widget.PreferenceImageView
|
||||||
|
android:id="@android:id/icon"
|
||||||
|
android:layout_width="24dp"
|
||||||
|
android:layout_height="24dp"/>
|
||||||
|
</LinearLayout>
|
||||||
|
|
||||||
|
<LinearLayout
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:orientation="vertical"
|
||||||
|
android:layout_marginTop="8dp"
|
||||||
|
android:layout_marginBottom="8dp">
|
||||||
|
|
||||||
|
<LinearLayout
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="wrap_content">
|
||||||
|
<TextView
|
||||||
|
android:id="@android:id/title"
|
||||||
|
android:layout_width="0dp"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:layout_weight="1"
|
||||||
|
android:singleLine="true"
|
||||||
|
android:textAppearance="@*android:style/TextAppearance.Material.Subhead"
|
||||||
|
android:textColor="?android:attr/textColorPrimary"
|
||||||
|
android:ellipsize="marquee"
|
||||||
|
android:fadingEdge="horizontal"/>
|
||||||
|
<LinearLayout
|
||||||
|
android:id="@android:id/widget_frame"
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="match_parent"
|
||||||
|
android:gravity="center_vertical"
|
||||||
|
android:orientation="vertical"/>
|
||||||
|
</LinearLayout>
|
||||||
|
|
||||||
|
<LinearLayout
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:orientation="vertical">
|
||||||
|
|
||||||
|
<com.android.settings.accessibility.BalanceSeekBar
|
||||||
|
android:id="@*android:id/seekbar"
|
||||||
|
android:layout_gravity="center_vertical"
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="wrap_content"/>
|
||||||
|
|
||||||
|
<LinearLayout
|
||||||
|
android:layout_width="match_parent"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:layout_gravity="center_vertical"
|
||||||
|
android:orientation="horizontal">
|
||||||
|
<TextView
|
||||||
|
android:id="@+id/left_text"
|
||||||
|
android:text="@string/accessibility_toggle_master_balance_left_label"
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:layout_weight="1"
|
||||||
|
android:maxLines="1"
|
||||||
|
android:textAlignment="viewStart" />
|
||||||
|
<TextView
|
||||||
|
android:id="@+id/right_text"
|
||||||
|
android:text="@string/accessibility_toggle_master_balance_right_label"
|
||||||
|
android:layout_width="wrap_content"
|
||||||
|
android:layout_height="wrap_content"
|
||||||
|
android:layout_weight="1"
|
||||||
|
android:maxLines="1"
|
||||||
|
android:textAlignment="viewEnd" />
|
||||||
|
|
||||||
|
</LinearLayout>
|
||||||
|
</LinearLayout>
|
||||||
|
</LinearLayout>
|
||||||
|
|
||||||
|
</LinearLayout>
|
@@ -38,6 +38,9 @@
|
|||||||
|
|
||||||
<dimen name="volume_seekbar_side_margin">8dip</dimen>
|
<dimen name="volume_seekbar_side_margin">8dip</dimen>
|
||||||
|
|
||||||
|
<dimen name="balance_seekbar_center_marker_height">14dp</dimen>
|
||||||
|
<dimen name="balance_seekbar_center_marker_width">1dp</dimen>
|
||||||
|
|
||||||
<dimen name="crypt_clock_size">100sp</dimen>
|
<dimen name="crypt_clock_size">100sp</dimen>
|
||||||
|
|
||||||
<dimen name="divider_height">3dip</dimen>
|
<dimen name="divider_height">3dip</dimen>
|
||||||
|
@@ -4717,6 +4717,12 @@
|
|||||||
<string name="accessibility_toggle_master_mono_title">Mono audio</string>
|
<string name="accessibility_toggle_master_mono_title">Mono audio</string>
|
||||||
<!-- Summary for the accessibility preference for master mono. [CHAR LIMIT=50] -->
|
<!-- Summary for the accessibility preference for master mono. [CHAR LIMIT=50] -->
|
||||||
<string name="accessibility_toggle_master_mono_summary">Combine channels when playing audio</string>
|
<string name="accessibility_toggle_master_mono_summary">Combine channels when playing audio</string>
|
||||||
|
<!-- Title for the accessibility preference for master balance. [CHAR LIMIT=35] -->
|
||||||
|
<string name="accessibility_toggle_master_balance_title">Audio balance</string>
|
||||||
|
<!-- 'Left' balance text for the accessibility preference for master balance. [CHAR LIMIT=20] -->
|
||||||
|
<string name="accessibility_toggle_master_balance_left_label">Left</string>
|
||||||
|
<!-- 'Right' balance text for the accessibility preference for master balance. [CHAR LIMIT=20] -->
|
||||||
|
<string name="accessibility_toggle_master_balance_right_label">Right</string>
|
||||||
|
|
||||||
<!-- Option heading to leave the timeout requirement for accessibility users at its default level. [CHAR LIMIT=35] -->
|
<!-- Option heading to leave the timeout requirement for accessibility users at its default level. [CHAR LIMIT=35] -->
|
||||||
<string name="accessibility_timeout_default">Default</string>
|
<string name="accessibility_timeout_default">Default</string>
|
||||||
|
@@ -130,6 +130,10 @@
|
|||||||
android:summary="@string/accessibility_toggle_master_mono_summary"
|
android:summary="@string/accessibility_toggle_master_mono_summary"
|
||||||
android:persistent="false"/>
|
android:persistent="false"/>
|
||||||
|
|
||||||
|
<com.android.settings.accessibility.BalanceSeekBarPreference
|
||||||
|
android:key="seekbar_master_balance"
|
||||||
|
android:title="@string/accessibility_toggle_master_balance_title" />
|
||||||
|
|
||||||
<Preference
|
<Preference
|
||||||
android:key="hearing_aid_preference"
|
android:key="hearing_aid_preference"
|
||||||
android:summary="@string/accessibility_hearingaid_not_connected_summary"
|
android:summary="@string/accessibility_hearingaid_not_connected_summary"
|
||||||
|
152
src/com/android/settings/accessibility/BalanceSeekBar.java
Normal file
152
src/com/android/settings/accessibility/BalanceSeekBar.java
Normal file
@@ -0,0 +1,152 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2019 The Android Open Source Project
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package com.android.settings.accessibility;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
import android.content.res.ColorStateList;
|
||||||
|
import android.content.res.Resources;
|
||||||
|
import android.graphics.Canvas;
|
||||||
|
import android.graphics.Color;
|
||||||
|
import android.graphics.Paint;
|
||||||
|
import android.graphics.Rect;
|
||||||
|
import android.os.UserHandle;
|
||||||
|
import android.provider.Settings;
|
||||||
|
import android.util.AttributeSet;
|
||||||
|
import android.widget.SeekBar;
|
||||||
|
|
||||||
|
import com.android.settings.R;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A custom seekbar for the balance setting.
|
||||||
|
*
|
||||||
|
* Adds a center line indicator between left and right, which snaps to if close.
|
||||||
|
* Updates Settings.System for balance on progress changed.
|
||||||
|
*/
|
||||||
|
public class BalanceSeekBar extends SeekBar {
|
||||||
|
private static final String TAG = "BalanceSeekBar";
|
||||||
|
private final Context mContext;
|
||||||
|
private final Object mListenerLock = new Object();
|
||||||
|
private OnSeekBarChangeListener mOnSeekBarChangeListener;
|
||||||
|
private final OnSeekBarChangeListener mProxySeekBarListener = new OnSeekBarChangeListener() {
|
||||||
|
@Override
|
||||||
|
public void onStopTrackingTouch(SeekBar seekBar) {
|
||||||
|
synchronized(mListenerLock) {
|
||||||
|
if (mOnSeekBarChangeListener != null) {
|
||||||
|
mOnSeekBarChangeListener.onStopTrackingTouch(seekBar);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onStartTrackingTouch(SeekBar seekBar) {
|
||||||
|
synchronized(mListenerLock) {
|
||||||
|
if (mOnSeekBarChangeListener != null) {
|
||||||
|
mOnSeekBarChangeListener.onStartTrackingTouch(seekBar);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) {
|
||||||
|
if (fromUser) {
|
||||||
|
// Snap to centre when within the specified threshold
|
||||||
|
if (progress != mCenter
|
||||||
|
&& progress > mCenter - mSnapThreshold
|
||||||
|
&& progress < mCenter + mSnapThreshold) {
|
||||||
|
progress = mCenter;
|
||||||
|
seekBar.setProgress(progress); // direct update (fromUser becomes false)
|
||||||
|
}
|
||||||
|
final float balance = (progress - mCenter) * 0.01f;
|
||||||
|
Settings.System.putFloatForUser(mContext.getContentResolver(),
|
||||||
|
Settings.System.MASTER_BALANCE, balance, UserHandle.USER_CURRENT);
|
||||||
|
}
|
||||||
|
// If fromUser is false, the call is a set from the framework on creation or on
|
||||||
|
// internal update. The progress may be zero, ignore (don't change system settings).
|
||||||
|
|
||||||
|
// after adjusting the seekbar, notify downstream listener.
|
||||||
|
// note that progress may have been adjusted in the code above to mCenter.
|
||||||
|
synchronized(mListenerLock) {
|
||||||
|
if (mOnSeekBarChangeListener != null) {
|
||||||
|
mOnSeekBarChangeListener.onProgressChanged(seekBar, progress, fromUser);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// Percentage of max to be used as a snap to threshold
|
||||||
|
private static final float SNAP_TO_PERCENTAGE = 0.03f;
|
||||||
|
private final Paint mCenterMarkerPaint;
|
||||||
|
private final Rect mCenterMarkerRect;
|
||||||
|
// changed in setMax()
|
||||||
|
private float mSnapThreshold;
|
||||||
|
private int mCenter;
|
||||||
|
|
||||||
|
public BalanceSeekBar(Context context, AttributeSet attrs) {
|
||||||
|
this(context, attrs, com.android.internal.R.attr.seekBarStyle);
|
||||||
|
}
|
||||||
|
|
||||||
|
public BalanceSeekBar(Context context, AttributeSet attrs, int defStyleAttr) {
|
||||||
|
this(context, attrs, defStyleAttr, 0 /* defStyleRes */);
|
||||||
|
}
|
||||||
|
|
||||||
|
public BalanceSeekBar(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
|
||||||
|
super(context, attrs, defStyleAttr, defStyleRes);
|
||||||
|
mContext = context;
|
||||||
|
Resources res = getResources();
|
||||||
|
mCenterMarkerRect = new Rect(0 /* left */, 0 /* top */,
|
||||||
|
res.getDimensionPixelSize(R.dimen.balance_seekbar_center_marker_width),
|
||||||
|
res.getDimensionPixelSize(R.dimen.balance_seekbar_center_marker_height));
|
||||||
|
mCenterMarkerPaint = new Paint();
|
||||||
|
// TODO use a more suitable colour?
|
||||||
|
mCenterMarkerPaint.setColor(Color.BLACK);
|
||||||
|
mCenterMarkerPaint.setStyle(Paint.Style.FILL);
|
||||||
|
// Remove the progress colour
|
||||||
|
setProgressTintList(ColorStateList.valueOf(Color.TRANSPARENT));
|
||||||
|
|
||||||
|
super.setOnSeekBarChangeListener(mProxySeekBarListener);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setOnSeekBarChangeListener(OnSeekBarChangeListener listener) {
|
||||||
|
synchronized(mListenerLock) {
|
||||||
|
mOnSeekBarChangeListener = listener;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Note: the superclass AbsSeekBar.setMax is synchronized.
|
||||||
|
@Override
|
||||||
|
public synchronized void setMax(int max) {
|
||||||
|
super.setMax(max);
|
||||||
|
// update snap to threshold
|
||||||
|
mCenter = max / 2;
|
||||||
|
mSnapThreshold = max * SNAP_TO_PERCENTAGE;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Note: the superclass AbsSeekBar.onDraw is synchronized.
|
||||||
|
@Override
|
||||||
|
protected synchronized void onDraw(Canvas canvas) {
|
||||||
|
// Draw a vertical line at 50% that represents centred balance
|
||||||
|
int seekBarCenter = (canvas.getHeight() - getPaddingBottom()) / 2;
|
||||||
|
canvas.save();
|
||||||
|
canvas.translate((canvas.getWidth() - mCenterMarkerRect.right) / 2,
|
||||||
|
seekBarCenter - (mCenterMarkerRect.bottom / 2));
|
||||||
|
canvas.drawRect(mCenterMarkerRect, mCenterMarkerPaint);
|
||||||
|
canvas.restore();
|
||||||
|
super.onDraw(canvas);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@@ -0,0 +1,70 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (C) 2019 The Android Open Source Project
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package com.android.settings.accessibility;
|
||||||
|
|
||||||
|
import android.content.Context;
|
||||||
|
import android.media.AudioSystem;
|
||||||
|
import android.os.UserHandle;
|
||||||
|
import android.provider.Settings;
|
||||||
|
import android.util.AttributeSet;
|
||||||
|
import android.view.View;
|
||||||
|
import android.widget.ImageView;
|
||||||
|
import android.widget.SeekBar;
|
||||||
|
|
||||||
|
import androidx.core.content.res.TypedArrayUtils;
|
||||||
|
import androidx.preference.Preference;
|
||||||
|
import androidx.preference.PreferenceViewHolder;
|
||||||
|
|
||||||
|
import com.android.settings.R;
|
||||||
|
import com.android.settings.widget.SeekBarPreference;
|
||||||
|
|
||||||
|
/** A slider preference that directly controls audio balance **/
|
||||||
|
public class BalanceSeekBarPreference extends SeekBarPreference {
|
||||||
|
private static final String TAG = "BalanceSeekBarPreference";
|
||||||
|
private final Context mContext;
|
||||||
|
private BalanceSeekBar mSeekBar;
|
||||||
|
private ImageView mIconView;
|
||||||
|
|
||||||
|
public BalanceSeekBarPreference(Context context, AttributeSet attrs) {
|
||||||
|
super(context, attrs, TypedArrayUtils.getAttr(context,
|
||||||
|
R.attr.preferenceStyle,
|
||||||
|
android.R.attr.preferenceStyle));
|
||||||
|
mContext = context;
|
||||||
|
setLayoutResource(R.layout.preference_balance_slider);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void onBindViewHolder(PreferenceViewHolder view) {
|
||||||
|
super.onBindViewHolder(view);
|
||||||
|
mSeekBar = (BalanceSeekBar) view.findViewById(com.android.internal.R.id.seekbar);
|
||||||
|
mIconView = (ImageView) view.findViewById(com.android.internal.R.id.icon);
|
||||||
|
init();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void init() {
|
||||||
|
if (mSeekBar == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
final float balance = Settings.System.getFloatForUser(
|
||||||
|
mContext.getContentResolver(), Settings.System.MASTER_BALANCE,
|
||||||
|
0.f /* default */, UserHandle.USER_CURRENT);
|
||||||
|
// Rescale balance to range 0-200 centered at 100.
|
||||||
|
mSeekBar.setMax(200);
|
||||||
|
mSeekBar.setProgress((int)(balance * 100.f) + 100);
|
||||||
|
mSeekBar.setEnabled(isEnabled());
|
||||||
|
}
|
||||||
|
}
|
Reference in New Issue
Block a user