Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

Commit e875a191 authored by Treehugger Robot's avatar Treehugger Robot Committed by Android (Google) Code Review
Browse files

Merge "[Physical Keyboard] Add Repeat key toggle" into main

parents 7d94abc8 cffb21b1
Loading
Loading
Loading
Loading
+4 −0
Original line number Original line Diff line number Diff line
@@ -4696,6 +4696,10 @@
    <string name="keyboard_a11y_settings">Physical keyboard accessibility</string>
    <string name="keyboard_a11y_settings">Physical keyboard accessibility</string>
    <!-- Summary for the button to trigger the 'Physical keyboard accessibility' page. [CHAR LIMIT=NONE] -->
    <!-- Summary for the button to trigger the 'Physical keyboard accessibility' page. [CHAR LIMIT=NONE] -->
    <string name="keyboard_a11y_settings_summary">Sticky keys, Bounce keys, Mouse keys</string>
    <string name="keyboard_a11y_settings_summary">Sticky keys, Bounce keys, Mouse keys</string>
    <!-- Title for the keyboard repeat key option. [CHAR LIMIT=60] -->
    <string name="keyboard_repeat_key_title">Repeat Keys</string>
    <!-- Summary for the keyboard repeat key option. [CHAR LIMIT=NONE] -->
    <string name="keyboard_repeat_key_summary">Hold down a key to repeat its character until the key is released</string>
    <!-- Title text for per IME subtype keyboard layout. [CHAR LIMIT=35] -->
    <!-- Title text for per IME subtype keyboard layout. [CHAR LIMIT=35] -->
    <string name="ime_label_title"><xliff:g id="ime_label" example="Gboard">%s</xliff:g> layout</string>
    <string name="ime_label_title"><xliff:g id="ime_label" example="Gboard">%s</xliff:g> layout</string>
+9 −0
Original line number Original line Diff line number Diff line
@@ -15,6 +15,7 @@
-->
-->
<PreferenceScreen
<PreferenceScreen
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:settings="http://schemas.android.com/apk/res-auto"
    android:title="@string/physical_keyboard_title">
    android:title="@string/physical_keyboard_title">
    <!-- Additional preference screens are inserted here programmatically
    <!-- Additional preference screens are inserted here programmatically
         with low order values to set the key map of each attached keyboard. -->
         with low order values to set the key map of each attached keyboard. -->
@@ -31,6 +32,14 @@
            android:title="@string/modifier_keys_settings"
            android:title="@string/modifier_keys_settings"
            android:summary="@string/modifier_keys_settings_summary"
            android:summary="@string/modifier_keys_settings_summary"
            android:fragment="com.android.settings.inputmethod.ModifierKeysSettings" />
            android:fragment="com.android.settings.inputmethod.ModifierKeysSettings" />

        <SwitchPreferenceCompat
            android:key="physical_keyboard_repeat_key"
            android:title="@string/keyboard_repeat_key_title"
            android:summary="@string/keyboard_repeat_key_summary"
            android:defaultValue="false"
            settings:controller="com.android.settings.inputmethod.KeyboardRepeatKeysController" />

        <Preference
        <Preference
            android:key="physical_keyboard_a11y"
            android:key="physical_keyboard_a11y"
            android:title="@string/keyboard_a11y_settings"
            android:title="@string/keyboard_a11y_settings"
+76 −0
Original line number Original line Diff line number Diff line
/*
 * Copyright 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.inputmethod;

import android.content.Context;
import android.hardware.input.InputSettings;
import android.net.Uri;
import android.provider.Settings;

import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.lifecycle.LifecycleObserver;
import androidx.preference.PreferenceScreen;
import androidx.preference.SwitchPreferenceCompat;

public class KeyboardRepeatKeysController extends
        InputSettingPreferenceController implements
        LifecycleObserver {

    @Nullable
    private SwitchPreferenceCompat mSwitchPreferenceCompat;

    public KeyboardRepeatKeysController(@NonNull Context context,
            @NonNull String key) {
        super(context, key);
    }

    @Override
    public void displayPreference(@NonNull PreferenceScreen screen) {
        super.displayPreference(screen);
        mSwitchPreferenceCompat = screen.findPreference(getPreferenceKey());
    }

    @Override
    public int getAvailabilityStatus() {
        return InputSettings.isRepeatKeysFeatureFlagEnabled() ? AVAILABLE : UNSUPPORTED_ON_DEVICE;
    }

    @Override
    public boolean isChecked() {
        return InputSettings.isRepeatKeysEnabled(mContext);
    }

    @Override
    public boolean setChecked(boolean isChecked) {
        InputSettings.setRepeatKeysEnabled(mContext, isChecked);
        return true;
    }

    @Override
    protected void onInputSettingUpdated() {
        if (mSwitchPreferenceCompat != null) {
            mSwitchPreferenceCompat.setChecked(InputSettings.isRepeatKeysEnabled(mContext));
        }
    }

    @Override
    protected Uri getSettingUri() {
        return Settings.Secure.getUriFor(
                Settings.Secure.KEY_REPEAT_ENABLED);
    }
}
+14 −2
Original line number Original line Diff line number Diff line
@@ -48,8 +48,8 @@ import androidx.preference.TwoStatePreference;
import com.android.internal.util.Preconditions;
import com.android.internal.util.Preconditions;
import com.android.settings.R;
import com.android.settings.R;
import com.android.settings.Settings;
import com.android.settings.Settings;
import com.android.settings.SettingsPreferenceFragment;
import com.android.settings.core.SubSettingLauncher;
import com.android.settings.core.SubSettingLauncher;
import com.android.settings.dashboard.DashboardFragment;
import com.android.settings.keyboard.Flags;
import com.android.settings.keyboard.Flags;
import com.android.settings.overlay.FeatureFactory;
import com.android.settings.overlay.FeatureFactory;
import com.android.settings.search.BaseSearchIndexProvider;
import com.android.settings.search.BaseSearchIndexProvider;
@@ -65,7 +65,7 @@ import java.util.Objects;
// TODO(b/327638540): Update implementation of preference here and reuse key preferences and
// TODO(b/327638540): Update implementation of preference here and reuse key preferences and
//  controllers between here and A11y Setting page.
//  controllers between here and A11y Setting page.
@SearchIndexable
@SearchIndexable
public final class PhysicalKeyboardFragment extends SettingsPreferenceFragment
public final class PhysicalKeyboardFragment extends DashboardFragment
        implements InputManager.InputDeviceListener,
        implements InputManager.InputDeviceListener,
        KeyboardLayoutDialogFragment.OnSetupKeyboardLayoutsListener {
        KeyboardLayoutDialogFragment.OnSetupKeyboardLayoutsListener {


@@ -79,6 +79,7 @@ public final class PhysicalKeyboardFragment extends SettingsPreferenceFragment
    private static final String KEYBOARD_SHORTCUTS_HELPER = "keyboard_shortcuts_helper";
    private static final String KEYBOARD_SHORTCUTS_HELPER = "keyboard_shortcuts_helper";
    private static final String MODIFIER_KEYS_SETTINGS = "modifier_keys_settings";
    private static final String MODIFIER_KEYS_SETTINGS = "modifier_keys_settings";
    private static final String EXTRA_AUTO_SELECTION = "auto_selection";
    private static final String EXTRA_AUTO_SELECTION = "auto_selection";
    private static final String TAG = "KeyboardAndTouchA11yFragment";
    private static final Uri sVirtualKeyboardSettingsUri = Secure.getUriFor(
    private static final Uri sVirtualKeyboardSettingsUri = Secure.getUriFor(
            Secure.SHOW_IME_WITH_HARD_KEYBOARD);
            Secure.SHOW_IME_WITH_HARD_KEYBOARD);
    private static final Uri sAccessibilityBounceKeysUri = Secure.getUriFor(
    private static final Uri sAccessibilityBounceKeysUri = Secure.getUriFor(
@@ -118,6 +119,16 @@ public final class PhysicalKeyboardFragment extends SettingsPreferenceFragment
    static final String EXTRA_BT_ADDRESS = "extra_bt_address";
    static final String EXTRA_BT_ADDRESS = "extra_bt_address";
    private String mBluetoothAddress;
    private String mBluetoothAddress;


    @Override
    protected String getLogTag() {
        return TAG;
    }

    @Override
    protected int getPreferenceScreenResId() {
        return R.xml.physical_keyboard_settings;
    }

    @Override
    @Override
    public void onSaveInstanceState(Bundle outState) {
    public void onSaveInstanceState(Bundle outState) {
        outState.putParcelable(EXTRA_AUTO_SELECTION, mAutoInputDeviceIdentifier);
        outState.putParcelable(EXTRA_AUTO_SELECTION, mAutoInputDeviceIdentifier);
@@ -126,6 +137,7 @@ public final class PhysicalKeyboardFragment extends SettingsPreferenceFragment


    @Override
    @Override
    public void onCreatePreferences(Bundle bundle, String s) {
    public void onCreatePreferences(Bundle bundle, String s) {
        super.onCreatePreferences(bundle, s);
        Activity activity = Preconditions.checkNotNull(getActivity());
        Activity activity = Preconditions.checkNotNull(getActivity());
        addPreferencesFromResource(R.xml.physical_keyboard_settings);
        addPreferencesFromResource(R.xml.physical_keyboard_settings);
        mIm = Preconditions.checkNotNull(activity.getSystemService(InputManager.class));
        mIm = Preconditions.checkNotNull(activity.getSystemService(InputManager.class));
+83 −0
Original line number Original line Diff line number Diff line
/*
 * Copyright 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.inputmethod;

import static com.android.input.flags.Flags.FLAG_KEYBOARD_REPEAT_KEYS;

import static com.google.common.truth.Truth.assertThat;

import android.content.Context;
import android.hardware.input.InputSettings;
import android.platform.test.annotations.DisableFlags;
import android.platform.test.annotations.EnableFlags;
import android.platform.test.flag.junit.SetFlagsRule;

import com.android.settings.core.BasePreferenceController;

import org.junit.Before;
import org.junit.Rule;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.robolectric.RobolectricTestRunner;
import org.robolectric.RuntimeEnvironment;
import org.robolectric.annotation.Config;

@RunWith(RobolectricTestRunner.class)
@Config(shadows = {
        com.android.settings.testutils.shadow.ShadowFragment.class,
})
public class KeyboardRepeatKeysControllerTest {
    @Rule
    public final SetFlagsRule mSetFlagsRule = new SetFlagsRule();
    private Context mContext;

    private KeyboardRepeatKeysController mKeyboardRepeatKeysController;

    @Before
    public void setUp() {
        mContext = RuntimeEnvironment.application;
        mKeyboardRepeatKeysController = new KeyboardRepeatKeysController(mContext,
                "physical_keyboard_repeat_key");
    }

    @Test
    @EnableFlags(FLAG_KEYBOARD_REPEAT_KEYS)
    public void getAvailabilityStatus_flagIsEnabled_isAvailable() {
        assertThat(mKeyboardRepeatKeysController.getAvailabilityStatus())
                .isEqualTo(BasePreferenceController.AVAILABLE);
    }

    @Test
    @DisableFlags(FLAG_KEYBOARD_REPEAT_KEYS)
    public void getAvailabilityStatus_flagIsDisabled_notSupport() {
        assertThat(mKeyboardRepeatKeysController.getAvailabilityStatus())
                .isEqualTo(BasePreferenceController.UNSUPPORTED_ON_DEVICE);
    }

    @Test
    public void isChecked_sameWithInputSettingValue() {
        boolean isRepeatKeysEnabled = InputSettings.isRepeatKeysEnabled(mContext);
        assertThat(mKeyboardRepeatKeysController.isChecked()).isEqualTo(isRepeatKeysEnabled);
    }

    @Test
    public void setChecked_updatesInputSettingValue() {
        mKeyboardRepeatKeysController.setChecked(false);

        assertThat(InputSettings.isRepeatKeysEnabled(mContext)).isEqualTo(false);
    }
}