Android Prefence 总结

PreferenceActivity通常用在设置界面,用于保存设置的状态数据。在Android系统源码中,绝大多数应用程序的UI布局采用了Preference的布局结构,而不是我们平时在模拟器中构建应用程序时使用的View布局结构,例如,Setting模块中布局。Preference布局结构和View的布局结构本质上还是大同小异,Preference的优点在于布局界面的可控性和高效率以及可存储值的简洁性(每个PreferenPreferencece存储在相对应下的SharedPreference文件夹下)。 下面,我们对比Preference和View下得各个子控件,对他们的家庭元素在宏观上有个更好的把握性。

 

  在应用程序中,我们可以通过代码的方式来访问该sharedPreference文件,继而可以对其进行读取甚至任何操作。代码如下: 

// 得到我们的存储Preferences值的对象,然后对其进行相应操作  

SharedPreferences shp = PreferenceManager.getDefaultSharedPreferences(this);  

boolean apply_wifiChecked = shp.getBoolean("apply_wifi", false);

 

文件保存路径:Android系统会将Preference元素的值存储在sharedPreference文件中。该文件位于data/data/[packgename]/shared_prefs/文件下,命名约定为:packagename_preferencse.xml

 

控件含义         Preference 控件家庭         View控件家庭       

文本框              Preference                  TextView           

单选框           CheckPreference             CheckBox           

输入文本框           EditTextPreference          EditText           

列表框             ListPreference              ListView          

铃声            RingtonePreference          ——               

    其实在Android源码系统中还有很多的”未完工”的Preference, 没有为它们提供PI接口,例如SeekBarPreference。有兴趣的同学可以参考源码,具体路径为:frameworks/base/core/java/preference。

 

组合控件:

PreferenceCategory :类似于LinearLayou、RelativeLayout,用于组合一组Preference,使布局更具备层次感 。

PreferenceScreen  : 所有Preference元素的根节点。

显示Preference布局结构的方法为:新建Activity继承PreferenceActivity,然后在onCreate()方法中通过addPreferencesFromResource(R.xml.custom_preference) (我们自定义的Preference 布局)。

 

Preference元素的通用XML Attributes说明:

android:key :          每个Preference控件独一无二的”ID”,唯一表示此Preference。          

android:defaultValue : 默认值。 例如,CheckPreference的默认值可为”true”,默认为选中状态;EditTextPreference的默认值可为”110” 。

android:enabled :      表示该Preference是否可用状态。     

  android:title :        每个Preference在PreferenceScreen布局上显示的标题——大标题

    android:summary :      每个Preference在PreferenceScreen布局上显示的标题——小标题(可以没有)

    android:persistent:    表示Preference元素所对应的值是否写入sharedPreferen文件中,如果是true,则表示写入;否则,则表示不写入该Preference元素的值。

    android:dependency:    表示一个Preference(用A表示)的可用状态依赖另外一个Preference(用B表示)。B可用,则A可用;B不可用,则A不可用。

  android:disableDependentsState:  与android:dependency相反。B可用,则A不可用;B不可用,则A可用。

 

常用的方法则包括:

     getKey()        setKey()

getSummary()    setSummary()

  getText()       setText() //getXXX()代表取得xxx属性的值。

 

一个简易的效果图如下:

 

Preference的跳转:

方法一:在配置每个Preference元素节点时,我们可以显示为点击它时所跳转的Intent。点击该Preference,跳转至目标Intent。除非在onPreferenceTreeClick()方法中进行抉择。在xml中配置如下:

<Preference android:key="wifi_setting" android:title="Wi-Fi设置"  

    android:summary="设置和管理无线接入点" android:dependency="apply_wifi">  

    <!-- 点击时 自定义一个默认跳转Intent  action指定隐式Intent -->  

    <!-- action指定隐式Intent ; targetPackage和targetClass指定显示Intent-->  

    <intent android:action="com.xsl.seemAction" android:targetPackage="com.xsl.qin" android:targetClass="com.xsl.qin.MainActivity" />  

</Preference>

方法二:可以实现onPreferenceTreeClick()创建新的intent显示的进行跳转。

 

  接下来,对每个Preference的的独有XML Attributes和方法进行一下总结,使大家有更好的深入理解。

1、EditPreference 

方法:

getEditText()  返回的是我们在该控件中输入的文本框值

  getText()     返回的是我们之前sharedPreferen文件保存的值

2、ListPreference

XML Attributes:

  android:dialogTitle:弹出控件对话框时显示的标题

    android:entries:类型为array,控件欲显示的文本

android:entryValues:类型为array,与文本相对应的key-value键值对,value保存至sharedPreference文件

    说明:entries和entryValue属性使用的数组为定义在资源文件arrays.xml的数组名:

方法:

    CharSequence[]    getEntries(): 返回的是控件显示文本的一个”key”数组,对应于属性android:entries

   CharSequence[]    getEntryValues():返回的一个”value”数组,对应于属性android: entryValues

CharSequence  getEntry(): 返回当前选择文本

    String           getValue() :返回当前选中文本选中的value 。

与之对应的还有它们所对应的setXXX()方法,可以参考SDK进行分析。

3、RingtonePreference

XML Attributes:

android:ringtoneType:响铃的铃声类型,主要有:ringtone(音乐)、notification(通知)、alarm(闹铃)、all(所有可用声 音类型)。

   android:showDefault :默认铃声,可以使用系统(布尔值---true,false)的或者自定义的铃声

   android:showSilent  :指定铃声是否为静音。指定铃声包括系统默认铃声或者自定义的铃声

 

分析Preference事件

  在PreferenceActivity方法中,一个比较重要的监听点击事件方法为:

/* 说 明 : 当Preference控件被点击时,触发该方法。

     * 参数说明: preference   点击的对象。

     * 返回值:   true  代表点击事件已成功捕捉,无须执行默认动作或者返回上层调用链。例如,不跳转至默认Intent。 false 代表执行默认动作并且返回上层调用链。例如,跳转至默认Intent。

     */

public boolean onPreferenceTreeClick(PreferenceScreen preferenceScreen, Preference preference)

 

Preference相关的两个重要监听接口。

    Preference.OnPreferenceChangeListener  该监听器的一个重要方法如下:

   /*

 * 说明:  当Preference的元素值发送改变时,触发该事件。

 * 返回值:true  代表将新值写入sharedPreference文件中。 false 则不将新值写入sharedPreference文件

 */

boolean onPreferenceChange(Preference preference,Object objValue)

 

   Preference.OnPreferenceClickListener      该监听器的一个重要方法如下:

    /*

 * 说明:当点击控件时触发发生,可以做相应操作。

 */

public boolean onPreferenceClick(Preference preference)                  

    那么当一个Preference控件实现这两个接口时,当被点击或者值发生改变时,触发方法是如何执行的呢?事实上,

触发规则如下:

      1.先调用onPreferenceClick()方法,如果该方法返回true,则不再调用onPreferenceTreeClick方法;

       如果onPreferenceClick方法返回false,则继续调用onPreferenceTreeClick方法。

      2.onPreferenceChange的方法独立与其他两种方法的运行。也就是说,它总是会运行。

 

      补充:点击某个Preference控件后,会先回调onPreferenceChange()方法,即是否保存值,然后再回调onPreferenceClick以及onPreferenceTreeClick()方法,因此在onPreferenceClick/  onPreferenceTreeClick方法中我们得到的控件值就是最新的Preference控件值。

public class HelloPreference extends PreferenceActivity implements
        Preference.OnPreferenceClickListener,
        Preference.OnPreferenceChangeListener {
    private static String TAG = "HelloPreference";
    private CheckBoxPreference mapply_wifiPreference;       //打开wifi
    private CheckBoxPreference mapply_internetPreference;   //Internet共享
    private ListPreference 	 depart_valuePreference;      	//部门设置
    private EditTextPreference number_editPreference;        //输入电话号码
    private Preference mwifi_settingPreference;             	//wifi设置
    private String oldDeptId; // 旧部门的名称

    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        addPreferencesFromResource(R.xml.mypreference);
        //根据key值找到控件
        mapply_wifiPreference = (CheckBoxPreference) findPreference("apply_wifi");
        mapply_internetPreference = (CheckBoxPreference) findPreference("apply_internet");
        depart_valuePreference = (ListPreference) findPreference("depart_value");
        number_editPreference = (EditTextPreference) findPreference("number_edit");
        mwifi_settingPreference = (Preference) findPreference("wifi_setting");

        // 设置监听器
        mapply_internetPreference.setOnPreferenceClickListener(this);
        mapply_internetPreference.setOnPreferenceChangeListener(this);
        depart_valuePreference.setOnPreferenceClickListener(this);
        depart_valuePreference.setOnPreferenceChangeListener(this);
        number_editPreference.setOnPreferenceClickListener(this);
        number_editPreference.setOnPreferenceChangeListener(this);
        mwifi_settingPreference.setOnPreferenceClickListener(this);

        // 得到我们的存储Preferences值的对象,然后对其进行相应操作
        SharedPreferences shp = PreferenceManager.getDefaultSharedPreferences(this);
        boolean apply_wifiChecked = shp.getBoolean("apply_wifi", false);
    }

    // 对控件进行的一些操作
    private void operatePreference(Preference preference) {
        if (preference == mapply_wifiPreference){                  //点击了    "打开wifi"
            Log.i(TAG, " Wifi CB, and isCheckd ="+ mapply_wifiPreference.isChecked());
        }else if (preference.getKey().equals("apply_internet")){   //点击了"Internet共享"
            Log.i(TAG, " internet CB, and isCheckd = "+mapply_internetPreference.isChecked());
        }else if (preference == depart_valuePreference){           //点击了 "部门设置"
            Log.i(TAG, " department CB,and selectValue = "+ depart_valuePreference.getValue() + ", Text="+ depart_valuePreference.getEntry());
        }else if (preference.getKey().equals("wifi_setting")) {    //点击了"wifi设置"
            mwifi_settingPreference.setTitle("its turn me.");
        }else if (preference == number_editPreference)             //点击了"输入电话号码"
            Log.i(TAG, "Old Value="+ number_editPreference.getText() + ", New Value="+ number_editPreference.getEditText().toString());
    }
    // 点击事件触发
    @Override
    public boolean onPreferenceClick(Preference preference) {
        // TODO Auto-generated method stub
        Log.i(TAG, "onPreferenceClick----->"+String.valueOf(preference.getKey()));
        // 对控件进行操作
        operatePreference(preference);
        return false;
    }
        //点击事件触发
    public boolean onPreferenceTreeClick(PreferenceScreen preferenceScreen,
            Preference preference) {
        Log.i(TAG, "onPreferenceTreeClick----->"+preference.getKey());
        // 对控件进行操作
        operatePreference(preference);
        if (preference.getKey().equals("wifi_setting")) {
            // 创建一个新的Intent,
            // 函数如果返回true, 则跳转至该自定义的新的Intent ;
            // 函数如果返回false,则跳转至xml文件中配置的Intent ;
            Intent i = new Intent(HelloPreference.this, OtherActivity.class);  //OtherActivity只是一个简单的Activity
            i.putExtra("type", "wifi");
            startActivity(i);
            return true;
        }
        return false;
    }

    // 当Preference的值发生改变时触发该事件,true则以新值更新控件的状态,false则do noting
    public boolean onPreferenceChange(Preference preference, Object objValue) {
        Log.i(TAG, "onPreferenceChange----->"+String.valueOf(preference.getKey()));
        if (preference == mapply_wifiPreference){
            Log.i(TAG, "Wifi CB, and isCheckd = " + String.valueOf(objValue));
        }else if (preference.getKey().equals("apply_internet")) {
            Log.i(TAG, "internet CB, and isCheckd = "+ String.valueOf(objValue));
            return false;  //不保存该新值
        }else if (preference == depart_valuePreference){
            Log.i(TAG, "  Old Value"+ depart_valuePreference.getValue()+" NewDeptName"+objValue);
        }else if (preference.getKey().equals("wifi_setting")) {
            Log.i(TAG, "change" + String.valueOf(objValue));
            mwifi_settingPreference.setTitle("its turn me.");  //重新设置title
        } else if (preference == number_editPreference) {
            Log.i(TAG, "Old Value = " + String.valueOf(objValue));
            return false; // 不保存更新值
        }
        return true;  //保存更新后的值
    }
}

  

Demo中preference.xml的布局文件如下:
<?xml version="1.0" encoding="utf-8"?>
<PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android">

    <PreferenceCategory android:title="我的位置"
        android:key="set_local" />
    <CheckBoxPreference android:key="apply_wireless"
        android:title="使用无线网络" android:summary="使用无线网络在应用程序(例如Google地图)中查看位置"
        android:defaultValue="true">
    </CheckBoxPreference>
    <CheckBoxPreference android:key="apply_gps"
        android:title="使用GPS" android:summary="定位到街道级别(需要消耗更多的电量以及天气允许)">
    </CheckBoxPreference>

    <PreferenceCategory android:title="无线和网络设置"></PreferenceCategory>

    <CheckBoxPreference android:key="apply_fly"
        android:title="飞行模式" android:summary="禁用所有无线连接" >
    </CheckBoxPreference>

    <CheckBoxPreference android:key="apply_internet"
        android:title="Internet共享" android:summary="禁用通过USB共享Internet连接">
    </CheckBoxPreference>

    <CheckBoxPreference android:key="apply_wifi"
        android:title="Wi-Fi" android:summary="打开Wi-Fi">
    </CheckBoxPreference>
    <Preference android:key="wifi_setting" android:title="Wi-Fi设置"
        android:summary="设置和管理无线接入点" android:dependency="apply_wifi">
        <!-- 点击时 自定义一个默认跳转Intent  action指定隐式Intent -->
        <!-- action指定隐式Intent ; targetPackage和targetClass指定显示Intent-->
        <intent android:action="com.feixun.action.seemAction"
            android:targetPackage="com.feixun.qin" 				  android:targetClass="com.feixun.qin.MainActivity" />
    </Preference>
    <CheckBoxPreference android:key="apply_bluetooth"
        android:title="蓝牙" android:summary="启用蓝牙">
    </CheckBoxPreference>
    <Preference android:key="bluetooth_setting" android:title="蓝牙设置"
        android:summary="管理连接、设备设备名称和可检测性" android:dependency="apply_bluetooth">
    </Preference>
    <EditTextPreference android:key="number_edit"
        android:title="输入电话号码" android:defaultValue="123">
    </EditTextPreference>
    <ListPreference android:key="depart_value"
        android:title="部门设置" android:dialogTitle="选择部门" android:entries="@array/department"
        android:entryValues="@array/department_value">
    </ListPreference>
    <RingtonePreference android:key="ring_key"
        android:title="铃声" android:ringtoneType="all" android:showDefault="true"
        android:showSilent="true">
    </RingtonePreference>
</PreferenceScreen> 

  

posted @ 2012-10-24 14:31  滴水瓦  阅读(1246)  评论(0编辑  收藏  举报