[Catalyst] Introduce a AccessibilitySeekBarPreference contains tool tip window

Test: atest PreviewSizeSeekBarControllerTest TextReadingPreviewControllerTest
Bug: 372776688
Flag: com.android.settings.flags.catalyst_text_reading_screen
Change-Id: Ie93d2f26b1521e931ce648f0140894b153259f81
This commit is contained in:
Sunny Shao
2024-12-10 10:27:49 +08:00
parent dc80bddc1e
commit 9777bbe38f
6 changed files with 103 additions and 57 deletions

View File

@@ -25,7 +25,7 @@
android:key="preview" android:key="preview"
android:selectable="false"/> android:selectable="false"/>
<com.android.settings.widget.LabeledSeekBarPreference <com.android.settings.accessibility.AccessibilitySeekBarPreference
android:key="font_size" android:key="font_size"
android:summary="@string/short_summary_font_size" android:summary="@string/short_summary_font_size"
android:title="@string/title_font_size" android:title="@string/title_font_size"
@@ -35,7 +35,7 @@
settings:iconStartContentDescription="@string/font_size_make_smaller_desc" settings:iconStartContentDescription="@string/font_size_make_smaller_desc"
settings:keywords="@string/keywords_font_size" /> settings:keywords="@string/keywords_font_size" />
<com.android.settings.widget.LabeledSeekBarPreference <com.android.settings.accessibility.AccessibilitySeekBarPreference
android:key="display_size" android:key="display_size"
android:summary="@string/screen_zoom_short_summary" android:summary="@string/screen_zoom_short_summary"
android:title="@string/screen_zoom_title" android:title="@string/screen_zoom_title"

View File

@@ -0,0 +1,66 @@
/*
* Copyright (C) 2024 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.os.Bundle
import android.os.Parcelable
import android.util.AttributeSet
import com.android.settings.widget.LabeledSeekBarPreference
/**
* Add a custom AccessibilitySeekBarPreference with tool tip window for font size and display size.
*/
open class AccessibilitySeekBarPreference(context: Context, attrs: AttributeSet?) :
LabeledSeekBarPreference(context, attrs) {
var needsQSTooltipReshow = false
private var tooltipWindow: AccessibilityQuickSettingsTooltipWindow? = null
override fun onSaveInstanceState(): Parcelable {
val state = Bundle()
state.putParcelable(null, super.onSaveInstanceState())
if (needsQSTooltipReshow || tooltipWindow?.isShowing == true) {
state.putBoolean(KEY_SAVED_QS_TOOLTIP_RESHOW, /* value= */ true)
}
return state
}
override fun onRestoreInstanceState(state: Parcelable?) {
val bundle = state as Bundle
super.onRestoreInstanceState(bundle.getParcelable(null, Parcelable::class.java))
if (bundle.containsKey(KEY_SAVED_QS_TOOLTIP_RESHOW)) {
needsQSTooltipReshow = bundle.getBoolean(KEY_SAVED_QS_TOOLTIP_RESHOW)
}
}
/** To generate a tooltip window and return it. */
fun createTooltipWindow(): AccessibilityQuickSettingsTooltipWindow =
AccessibilityQuickSettingsTooltipWindow(context).also { tooltipWindow = it }
/** To dismiss the tooltip window. */
fun dismissTooltip() {
val tooltip = tooltipWindow
if (tooltip?.isShowing == true) {
tooltip.dismiss()
tooltipWindow = null
}
}
companion object {
private const val KEY_SAVED_QS_TOOLTIP_RESHOW = "qs_tooltip_reshow"
}
}

View File

@@ -19,7 +19,6 @@ package com.android.settings.accessibility;
import android.app.Activity; import android.app.Activity;
import android.content.ComponentName; import android.content.ComponentName;
import android.content.Context; import android.content.Context;
import android.os.Bundle;
import android.os.Handler; import android.os.Handler;
import android.widget.SeekBar; import android.widget.SeekBar;
@@ -28,31 +27,27 @@ import androidx.preference.PreferenceScreen;
import com.android.settings.R; import com.android.settings.R;
import com.android.settings.core.BasePreferenceController; import com.android.settings.core.BasePreferenceController;
import com.android.settings.widget.LabeledSeekBarPreference;
import com.android.settingslib.core.lifecycle.LifecycleObserver; import com.android.settingslib.core.lifecycle.LifecycleObserver;
import com.android.settingslib.core.lifecycle.events.OnCreate;
import com.android.settingslib.core.lifecycle.events.OnDestroy; import com.android.settingslib.core.lifecycle.events.OnDestroy;
import com.android.settingslib.core.lifecycle.events.OnSaveInstanceState; import com.android.settingslib.core.lifecycle.events.OnStart;
import com.android.settingslib.core.lifecycle.events.OnStop;
import com.google.android.setupcompat.util.WizardManagerHelper; import com.google.android.setupcompat.util.WizardManagerHelper;
import java.util.Optional; import java.util.Optional;
/** /**
* The controller of {@link LabeledSeekBarPreference} that listens to display size and font size * The controller of {@link AccessibilitySeekBarPreference} that listens to display size and font
* settings changes and updates preview size threshold smoothly. * size settings changes and updates preview size threshold smoothly.
*/ */
abstract class PreviewSizeSeekBarController extends BasePreferenceController implements abstract class PreviewSizeSeekBarController extends BasePreferenceController implements
TextReadingResetController.ResetStateListener, LifecycleObserver, OnCreate, TextReadingResetController.ResetStateListener, LifecycleObserver, OnStart, OnStop,
OnDestroy, OnSaveInstanceState { OnDestroy {
private final PreviewSizeData<? extends Number> mSizeData; private final PreviewSizeData<? extends Number> mSizeData;
private static final String KEY_SAVED_QS_TOOLTIP_RESHOW = "qs_tooltip_reshow";
private boolean mSeekByTouch; private boolean mSeekByTouch;
private Optional<ProgressInteractionListener> mInteractionListener = Optional.empty(); private Optional<ProgressInteractionListener> mInteractionListener = Optional.empty();
private LabeledSeekBarPreference mSeekBarPreference; private AccessibilitySeekBarPreference mSeekBarPreference;
private int mLastProgress; private int mLastProgress;
private boolean mNeedsQSTooltipReshow = false;
private AccessibilityQuickSettingsTooltipWindow mTooltipWindow;
private final Handler mHandler; private final Handler mHandler;
private String[] mStateLabels = null; private String[] mStateLabels = null;
@@ -101,30 +96,21 @@ abstract class PreviewSizeSeekBarController extends BasePreferenceController imp
} }
@Override @Override
public void onCreate(Bundle savedInstanceState) { public void onStart() {
// Restore the tooltip. if (mSeekBarPreference.getNeedsQSTooltipReshow()) {
if (savedInstanceState != null mHandler.post(this::showQuickSettingsTooltipIfNeeded);
&& savedInstanceState.containsKey(KEY_SAVED_QS_TOOLTIP_RESHOW)) {
mNeedsQSTooltipReshow = savedInstanceState.getBoolean(KEY_SAVED_QS_TOOLTIP_RESHOW);
} }
} }
@Override
public void onStop() {
// all the messages/callbacks will be removed.
mHandler.removeCallbacksAndMessages(null);
}
@Override @Override
public void onDestroy() { public void onDestroy() {
// remove runnables in the queue. mSeekBarPreference.dismissTooltip();
mHandler.removeCallbacksAndMessages(null);
final boolean isTooltipWindowShowing = mTooltipWindow != null && mTooltipWindow.isShowing();
if (isTooltipWindowShowing) {
mTooltipWindow.dismiss();
}
}
@Override
public void onSaveInstanceState(Bundle outState) {
final boolean isTooltipWindowShowing = mTooltipWindow != null && mTooltipWindow.isShowing();
if (mNeedsQSTooltipReshow || isTooltipWindowShowing) {
outState.putBoolean(KEY_SAVED_QS_TOOLTIP_RESHOW, /* value= */ true);
}
} }
void setInteractionListener(ProgressInteractionListener interactionListener) { void setInteractionListener(ProgressInteractionListener interactionListener) {
@@ -148,9 +134,6 @@ abstract class PreviewSizeSeekBarController extends BasePreferenceController imp
mSeekBarPreference.setProgress(initialIndex); mSeekBarPreference.setProgress(initialIndex);
mSeekBarPreference.setContinuousUpdates(true); mSeekBarPreference.setContinuousUpdates(true);
mSeekBarPreference.setOnSeekBarChangeListener(mSeekBarChangeListener); mSeekBarPreference.setOnSeekBarChangeListener(mSeekBarChangeListener);
if (mNeedsQSTooltipReshow) {
mHandler.post(this::showQuickSettingsTooltipIfNeeded);
}
setSeekbarStateDescription(mSeekBarPreference.getProgress()); setSeekbarStateDescription(mSeekBarPreference.getProgress());
} }
@@ -216,7 +199,8 @@ abstract class PreviewSizeSeekBarController extends BasePreferenceController imp
return; return;
} }
if (!mNeedsQSTooltipReshow && AccessibilityQuickSettingUtils.hasValueInSharedPreferences( if (!mSeekBarPreference.getNeedsQSTooltipReshow()
&& AccessibilityQuickSettingUtils.hasValueInSharedPreferences(
mContext, tileComponentName)) { mContext, tileComponentName)) {
// Returns if quick settings tooltip only show once. // Returns if quick settings tooltip only show once.
return; return;
@@ -228,14 +212,15 @@ abstract class PreviewSizeSeekBarController extends BasePreferenceController imp
// is not ready when we would like to show the tooltip. If the seekbar is not ready, // is not ready when we would like to show the tooltip. If the seekbar is not ready,
// we give up showing the tooltip and also do not reshow it in the future. // we give up showing the tooltip and also do not reshow it in the future.
if (mSeekBarPreference.getSeekbar() != null) { if (mSeekBarPreference.getSeekbar() != null) {
mTooltipWindow = new AccessibilityQuickSettingsTooltipWindow(mContext); final AccessibilityQuickSettingsTooltipWindow tooltipWindow =
mTooltipWindow.setup(getTileTooltipContent(), mSeekBarPreference.createTooltipWindow();
tooltipWindow.setup(getTileTooltipContent(),
R.drawable.accessibility_auto_added_qs_tooltip_illustration); R.drawable.accessibility_auto_added_qs_tooltip_illustration);
mTooltipWindow.showAtTopCenter(mSeekBarPreference.getSeekbar()); tooltipWindow.showAtTopCenter(mSeekBarPreference.getSeekbar());
} }
AccessibilityQuickSettingUtils.optInValueToSharedPreferences(mContext, AccessibilityQuickSettingUtils.optInValueToSharedPreferences(mContext,
tileComponentName); tileComponentName);
mNeedsQSTooltipReshow = false; mSeekBarPreference.setNeedsQSTooltipReshow(false);
} }
/** Returns the accessibility Quick Settings tile component name. */ /** Returns the accessibility Quick Settings tile component name. */

View File

@@ -33,7 +33,6 @@ import com.android.settings.accessibility.TextReadingPreferenceFragment.EntryPoi
import com.android.settings.core.BasePreferenceController; import com.android.settings.core.BasePreferenceController;
import com.android.settings.core.instrumentation.SettingsStatsLog; import com.android.settings.core.instrumentation.SettingsStatsLog;
import com.android.settings.display.PreviewPagerAdapter; import com.android.settings.display.PreviewPagerAdapter;
import com.android.settings.widget.LabeledSeekBarPreference;
import java.util.Objects; import java.util.Objects;
@@ -58,8 +57,8 @@ class TextReadingPreviewController extends BasePreferenceController implements
private int mLastDisplayProgress; private int mLastDisplayProgress;
private long mLastCommitTime; private long mLastCommitTime;
private TextReadingPreviewPreference mPreviewPreference; private TextReadingPreviewPreference mPreviewPreference;
private LabeledSeekBarPreference mFontSizePreference; private AccessibilitySeekBarPreference mFontSizePreference;
private LabeledSeekBarPreference mDisplaySizePreference; private AccessibilitySeekBarPreference mDisplaySizePreference;
@EntryPoint @EntryPoint
private int mEntryPoint; private int mEntryPoint;

View File

@@ -29,7 +29,6 @@ import static org.mockito.Mockito.when;
import android.app.Activity; import android.app.Activity;
import android.content.ComponentName; import android.content.ComponentName;
import android.content.Intent; import android.content.Intent;
import android.os.Bundle;
import android.view.LayoutInflater; import android.view.LayoutInflater;
import android.widget.PopupWindow; import android.widget.PopupWindow;
import android.widget.SeekBar; import android.widget.SeekBar;
@@ -44,7 +43,6 @@ import androidx.test.ext.junit.rules.ActivityScenarioRule;
import com.android.settings.R; import com.android.settings.R;
import com.android.settings.SettingsPreferenceFragment; import com.android.settings.SettingsPreferenceFragment;
import com.android.settings.testutils.shadow.ShadowFragment; import com.android.settings.testutils.shadow.ShadowFragment;
import com.android.settings.widget.LabeledSeekBarPreference;
import com.android.settingslib.testutils.shadow.ShadowInteractionJankMonitor; import com.android.settingslib.testutils.shadow.ShadowInteractionJankMonitor;
import com.google.android.setupcompat.util.WizardManagerHelper; import com.google.android.setupcompat.util.WizardManagerHelper;
@@ -77,7 +75,7 @@ public class PreviewSizeSeekBarControllerTest {
private Activity mContext; private Activity mContext;
private PreviewSizeSeekBarController mSeekBarController; private PreviewSizeSeekBarController mSeekBarController;
private FontSizeData mFontSizeData; private FontSizeData mFontSizeData;
private LabeledSeekBarPreference mSeekBarPreference; private AccessibilitySeekBarPreference mSeekBarPreference;
private PreferenceScreen mPreferenceScreen; private PreferenceScreen mPreferenceScreen;
private TestFragment mFragment; private TestFragment mFragment;
@@ -109,7 +107,7 @@ public class PreviewSizeSeekBarControllerTest {
mPreferenceScreen = spy(new PreferenceScreen(mContext, /* attrs= */ null)); mPreferenceScreen = spy(new PreferenceScreen(mContext, /* attrs= */ null));
when(mPreferenceScreen.getPreferenceManager()).thenReturn(mPreferenceManager); when(mPreferenceScreen.getPreferenceManager()).thenReturn(mPreferenceManager);
doReturn(mPreferenceScreen).when(mFragment).getPreferenceScreen(); doReturn(mPreferenceScreen).when(mFragment).getPreferenceScreen();
mSeekBarPreference = spy(new LabeledSeekBarPreference(mContext, /* attrs= */ null)); mSeekBarPreference = spy(new AccessibilitySeekBarPreference(mContext, /* attrs= */ null));
mSeekBarPreference.setKey(FONT_SIZE_KEY); mSeekBarPreference.setKey(FONT_SIZE_KEY);
LayoutInflater inflater = LayoutInflater.from(mContext); LayoutInflater inflater = LayoutInflater.from(mContext);
@@ -246,12 +244,11 @@ public class PreviewSizeSeekBarControllerTest {
@Test @Test
@Config(shadows = ShadowFragment.class) @Config(shadows = ShadowFragment.class)
public void restoreValueFromSavedInstanceState_showTooltipView() { public void enabledNeedsQSTooltipReshow_showTooltipView() {
final Bundle savedInstanceState = new Bundle(); mSeekBarPreference.setNeedsQSTooltipReshow(true);
savedInstanceState.putBoolean(KEY_SAVED_QS_TOOLTIP_RESHOW, /* value= */ true);
mSeekBarController.onCreate(savedInstanceState);
mSeekBarController.displayPreference(mPreferenceScreen); mSeekBarController.displayPreference(mPreferenceScreen);
mSeekBarController.onStart();
ShadowLooper.idleMainLooper(); ShadowLooper.idleMainLooper();
assertThat(getLatestPopupWindow().isShowing()).isTrue(); assertThat(getLatestPopupWindow().isShowing()).isTrue();

View File

@@ -28,7 +28,6 @@ import androidx.test.core.app.ApplicationProvider;
import com.android.settings.display.PreviewPagerAdapter; import com.android.settings.display.PreviewPagerAdapter;
import com.android.settings.testutils.shadow.ShadowInteractionJankMonitor; import com.android.settings.testutils.shadow.ShadowInteractionJankMonitor;
import com.android.settings.widget.LabeledSeekBarPreference;
import org.junit.Before; import org.junit.Before;
import org.junit.Test; import org.junit.Test;
@@ -54,8 +53,8 @@ public class TextReadingPreviewControllerTest {
private final Context mContext = ApplicationProvider.getApplicationContext(); private final Context mContext = ApplicationProvider.getApplicationContext();
private TextReadingPreviewController mPreviewController; private TextReadingPreviewController mPreviewController;
private TextReadingPreviewPreference mPreviewPreference; private TextReadingPreviewPreference mPreviewPreference;
private LabeledSeekBarPreference mFontSizePreference; private AccessibilitySeekBarPreference mFontSizePreference;
private LabeledSeekBarPreference mDisplaySizePreference; private AccessibilitySeekBarPreference mDisplaySizePreference;
@Mock @Mock
private DisplaySizeData mDisplaySizeData; private DisplaySizeData mDisplaySizeData;
@@ -73,8 +72,8 @@ public class TextReadingPreviewControllerTest {
mPreviewPreference = spy(new TextReadingPreviewPreference(mContext, /* attr= */ null)); mPreviewPreference = spy(new TextReadingPreviewPreference(mContext, /* attr= */ null));
mPreviewController = new TextReadingPreviewController(mContext, PREVIEW_KEY, fontSizeData, mPreviewController = new TextReadingPreviewController(mContext, PREVIEW_KEY, fontSizeData,
mDisplaySizeData); mDisplaySizeData);
mFontSizePreference = new LabeledSeekBarPreference(mContext, /* attr= */ null); mFontSizePreference = new AccessibilitySeekBarPreference(mContext, /* attr= */ null);
mDisplaySizePreference = new LabeledSeekBarPreference(mContext, /* attr= */ null); mDisplaySizePreference = new AccessibilitySeekBarPreference(mContext, /* attr= */ null);
} }
@Test @Test