Android系统SettingsPrivider分析与修改方法

        继上次分析Settings的文章Android 系统Settings概要之后,现要增加一些系统中没有的设置项,因上次只是猜测DatabaseHelper.java 在创建数据库时将defaults.xml中的配制存入了数据库中,因此现在来分析下SettingsPrivider源码(源码还是官方android-4.4_r1版本).

第一,主要分析DatabaseHelper.java文件:

    1. 数据库文件为 settings.db  定义的表有人个 ,里面system , secure ,global 是比主要的三个。

    private static final String TAG = "SettingsProvider";
    private static final String DATABASE_NAME = "settings.db";

    // Please, please please. If you update the database version, check to make sure the
    // database gets upgraded properly. At a minimum, please confirm that 'upgradeVersion'
    // is properly propagated through your change.  Not doing so will result in a loss of user
    // settings.
    private static final int DATABASE_VERSION = 98;

    private Context mContext;
    private int mUserHandle;

    private static final HashSet<String> mValidTables = new HashSet<String>();

    private static final String TABLE_SYSTEM = "system";
    private static final String TABLE_SECURE = "secure";
    private static final String TABLE_GLOBAL = "global";

    static {
        mValidTables.add(TABLE_SYSTEM);
        mValidTables.add(TABLE_SECURE);
        mValidTables.add(TABLE_GLOBAL);
        mValidTables.add("bluetooth_devices");
        mValidTables.add("bookmarks");

        // These are old.
        mValidTables.add("favorites");
        mValidTables.add("gservices");
        mValidTables.add("old_favorites");
    }

   2. 看onCreate()方法,创建表和各表的索引,然后调用了loadBookmarks(db); loadVolumeLevels(db);  loadSettings(db); 三个方法,loadBookmarks(db);是存系统几个常用的应用的intent ( 如,通讯录app,emailApp, 短信app,),loadVolumeLevels(db); 初始化的是与声音相关的配制,  loadSettings(db);方法就是从defaults.xml文件中读取默认值存入相应的表中,验证了之前 的猜测。

    @Override
    public void onCreate(SQLiteDatabase db) {
        db.execSQL("CREATE TABLE system (" +
                    "_id INTEGER PRIMARY KEY AUTOINCREMENT," +
                    "name TEXT UNIQUE ON CONFLICT REPLACE," +
                    "value TEXT" +
                    ");");
        db.execSQL("CREATE INDEX systemIndex1 ON system (name);");

        createSecureTable(db);

        // Only create the global table for the singleton 'owner' user
        if (mUserHandle == UserHandle.USER_OWNER) {
            createGlobalTable(db);
        }

        db.execSQL("CREATE TABLE bluetooth_devices (" +
                    "_id INTEGER PRIMARY KEY," +
                    "name TEXT," +
                    "addr TEXT," +
                    "channel INTEGER," +
                    "type INTEGER" +
                    ");");

        db.execSQL("CREATE TABLE bookmarks (" +
                    "_id INTEGER PRIMARY KEY," +
                    "title TEXT," +
                    "folder TEXT," +
                    "intent TEXT," +
                    "shortcut INTEGER," +
                    "ordering INTEGER" +
                    ");");

        db.execSQL("CREATE INDEX bookmarksIndex1 ON bookmarks (folder);");
        db.execSQL("CREATE INDEX bookmarksIndex2 ON bookmarks (shortcut);");

        // Populate bookmarks table with initial bookmarks
        boolean onlyCore = false;
        try {
            onlyCore = IPackageManager.Stub.asInterface(ServiceManager.getService(
                    "package")).isOnlyCoreApps();
        } catch (RemoteException e) {
        }
        if (!onlyCore) {
            loadBookmarks(db);
        }

        // Load initial volume levels into DB
        loadVolumeLevels(db);

        // Load inital settings values
        loadSettings(db);
    }

3  .    loadSettings(db); 里面的三个方法者是从defaults.xml文件中读取默认值存入相应的表中,loadSystemSettings(db)是存入System表中,loadSecureSettings(db)是存入 secure表中。

    

private void loadSettings(SQLiteDatabase db) {
        loadSystemSettings(db);
        loadSecureSettings(db);
        // The global table only exists for the 'owner' user
        if (mUserHandle == UserHandle.USER_OWNER) {
            loadGlobalSettings(db);
        }
    }


loadSetting()方法中SQLiteStatement将值存入了数据库中:

 private void loadSystemSettings(SQLiteDatabase db) {
        SQLiteStatement stmt = null;
        try {
            stmt = db.compileStatement("INSERT OR IGNORE INTO system(name,value)"
                    + " VALUES(?,?);");

            loadBooleanSetting(stmt, Settings.System.DIM_SCREEN,
                    R.bool.def_dim_screen);
            //..............省略..................

            loadIntegerSetting(stmt, Settings.System.POINTER_SPEED,
                    R.integer.def_pointer_speed);
        } finally {
            if (stmt != null) stmt.close();
        }
    }



第二,那如何来增加设置项(例如要在设置中增加一项 账户信息):

      1. 在Settings 中增加界面相关的代码:

          1.a.    Settings/res/xml/settings_headers.xml文件中增加:

           <header
            android:id="@+id/account_info"
            android:fragment="com.android.settings.accounts.AccountInfoSettings"
            android:title="@string/account_info_label"
            android:icon="@drawable/ic_menu_add_dark"/>

         1.b.   增加字符串图标等 ,如  Settings/res/values/strings.xml文件中:
           <string name="account_info_label" >"账号信息"</string>

         1.c  相关类与代码的编写:如 创建Fragment:AccountInfoSettings 并写代码,里面的代码会Settings.System.put**的代码将操作SettingsProvider    
   
     2. 修改SettingsPrivider

       2. a   defalult.xml

         <bool name="def_acctount_islogin">false</bool>
         <string name="def_acctount_name">请设置昵称</string>

       2. b  DatabaseHelper.java 在loadSecureSettings(SQLiteDatabase db) 方法中增加

            loadBooleanSetting(stmt, Settings.Secure.ACCOUNT_ISLOGIN,
                    R.bool.def_account_islogin);
            loadStringSetting(stmt, Settings.Secure.ACCOUNT_NAME,
                    R.string.def_account_name);

      3. 在/android-4.4_r1/frameworks/base/core/java/android/provider/Settings.java 文件中找到内部静态类:public static final class Secure extends NameValueTable {..........} 在Secure类中增加:

        public static final String ACCOUNT_ISLOGIN = "def_account_islogin";
        public static final String ACCOUNT_NAME = "def_account_name";

         

     到此修改基本完成,可以编译,运行,测试了。实际开发中的步骤应该是3.2,1按倒着的顺序来做^-^.

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值