AOSP 设置-提示音和振动 添加一个带有开关(Switch)的设置项

要求

如题,在设置-提示音和振动中添加一个带有开关(Switch)的设置项。

复制代码
要实现的效果:
1、能正常显示带有开关(Switch)的设置项
2、给这个开关设置默认属性
3、开关能正常使用(具体逻辑需要自己加)

步骤

1、UI显示

可以根据上方的"振动模式下一律显示振动图标"来定位要在哪里添加。

xml 复制代码
packages/apps/Settings/res/xml/sound_settings.xml

<SwitchPreferenceCompat
        android:key="top_paddle_control_volume"
        android:title="@string/top_paddle_control_volume_title"
        android:order="-25"
        settings:controller="com.android.settings.notification.TopPaddleVolumePreferenceController"/>

只要加上上面这段xml代码就能在设置中显示了,一开始可以不加settings:controller这个配置,也会正常显示的,只是没有功能,纯UI。后续TopPaddleVolumePreferenceController是需要自己加的。

2、定义对应的controller

要给这个switch开关定义一个controller来控制开关

java 复制代码
packages/apps/Settings/src/com/android/settings/notification/TopPaddleVolumePreferenceController.java

package com.android.settings.notification;

import static com.android.settings.notification.SettingPref.TYPE_SECURE;

import android.content.Context;
import android.provider.Settings.Secure;

import com.android.settings.SettingsPreferenceFragment;
import com.android.settings.core.TogglePreferenceController;
import com.android.settings.R;
import com.android.settingslib.core.lifecycle.Lifecycle;

import android.provider.Settings;
import android.util.Log;

// public class TopPaddleVolumePreferenceController  extends SettingPrefController {
public class TopPaddleVolumePreferenceController  extends TogglePreferenceController {
    private static final String TAG = "TopPaddleVolumePreferenceController";
    private static final String TOP_PADDLE_CONTROL_VOLUME = "top_paddle_control_volume";

    public TopPaddleVolumePreferenceController(Context context, String preferenceKey) {
        super(context, preferenceKey);
    }

    @Override
    public boolean isChecked() {
        Log.d(TAG, "ENTER isChecked()");
        // Read database status, default 0 (off)
        return Settings.System.getInt(mContext.getContentResolver(),
            TOP_PADDLE_CONTROL_VOLUME, 0) != 0;
        // 经过测试,这里的TOP_PADDLE_CONTROL_VOLUME也可以替换成Settings.System.TOP_PADDLE_CONTROL_VOLUME
    }

    @Override
    public boolean setChecked(boolean isChecked) {
        Log.d(TAG, "isChecked = "+isChecked);
        // Write database status
        return Settings.System.putInt(mContext.getContentResolver(),
            TOP_PADDLE_CONTROL_VOLUME, isChecked ? 1 : 0);
        // 经过测试,这里的TOP_PADDLE_CONTROL_VOLUME也可以替换成Settings.System.TOP_PADDLE_CONTROL_VOLUME
    }

    @Override
    public int getAvailabilityStatus() {
        return AVAILABLE;
    }

    /**
     * 修复错误:覆盖 TogglePreferenceController 中的抽象方法
     * 该方法用于在搜索结果中高亮显示菜单路径
     */
    @Override
    public int getSliceHighlightMenuRes() {
        // 因为你在 sound_settings.xml 中添加,所以指向声音菜单
        return R.string.menu_key_sound; 
    }
}

3、数据初始化

需要给这个设置项定义一个变量A,对吧,点击开关时实际就是修改这个变量A的值。而且还需要给这个变量A初始化一个值,0或者1,规定这个开关是默认开还是默认关

3.1 定义switch开关的初始值

java 复制代码
frameworks\base\core\java\android\provider\Settings.java

public static final class System extends NameValueTable {
    /**
     * Whether keyboard vibration feedback is enabled. The value is boolean (1 or 0).
     *
     * @hide
     */
    @hide
    public static final String KEYBOARD_VIBRATION_ENABLED = "keyboard_vibration_enabled";
    
这样就是定义了一个变量了,相当于数据库的key,后面切换开关时修改的就是这个key。
其中@hide和@hide我看其它的变量也有,我也对应着加上了,反正如果你没有@hide编译会报错。

注意:你定义的这个KEYBOARD_VIBRATION_ENABLED是放在class System还是class Secure中很重要的。我一开始是没注意,定义在了class Secure,但是绑定初始值时用的Settings.System.TOP_PADDLE_CONTROL_VOLUME,结果就报错了,这里注意。

3.2 定义switch开关的默认值

这个默认值其实不用定义,在初始值赋值的时候可以直接给个0或者1,但是这里为了代码的规范化,咱们还是定义一下。

xml 复制代码
frameworks\base\packages\SettingsProvider\res\values\defaults.xml

<!--cmy add, Top paddle control volume, default: false -->
    <bool name="def_top_paddle_control_volume">true</bool>

3.3 绑定初始值

java 复制代码
frameworks\base\packages\SettingsProvider\src\com\android\providers\settings\DatabaseHelper.java

private void loadSystemSettings(SQLiteDatabase db) {
    // cmy add
    loadBooleanSetting(stmt, Settings.System.TOP_PADDLE_CONTROL_VOLUME,
        R.bool.def_top_paddle_control_volume);

这样修改完,然后编译刷机,就能看到switch开关被打开了,因为绑定的默认值def_top_paddle_control_volume是true。

查看变量值的adb命令

shell 复制代码
adb shell settings get system top_paddle_control_volume

这里是get system,这个system可能跟你定义变量的位置有关,前面说的,看你的变量是定义在class System还是class Secure
相关推荐
JMchen1231 天前
现代Android图像处理管道:从CameraX到OpenGL的60fps实时滤镜架构
android·图像处理·架构·kotlin·android studio·opengl·camerax
快点好好学习吧1 天前
phpize 依赖 php-config 获取 PHP 信息的庖丁解牛
android·开发语言·php
是誰萆微了承諾1 天前
php 对接deepseek
android·开发语言·php
Dxy12393102161 天前
MySQL如何加唯一索引
android·数据库·mysql
冠希陈、2 天前
PHP 判断是否是移动端,更新鸿蒙系统
android·开发语言·php
晚霞的不甘2 天前
Flutter for OpenHarmony从零到一:构建《冰火人》双人合作闯关游戏
android·flutter·游戏·前端框架·全文检索·交互
2601_949833392 天前
flutter_for_openharmony口腔护理app实战+饮食记录实现
android·javascript·flutter
独自破碎E2 天前
【滑动窗口+字符计数数组】LCR_014_字符串的排列
android·java·开发语言
stevenzqzq2 天前
compose 中 align和Arrangement的区别
android·compose
VincentWei952 天前
Compose:MutableState 和 mutableStateOf
android