編輯:關於Android編程
前面簡單描述下了Preference的家族構成和基本知識,相信對於Preference早已不會陌生,肯定也躍躍欲試了吧,這篇文章就給大家總結下Preference、PreferenceActivity、PreferenceGroup、RingtonePreference的普通應用和管理,還有通過一些測試來驗證一些機制和原理。
PreferenceActivity是一個抽象類,繼承於ListActivity,以列表形式視圖來展現界面,加載的整個View也是基於ListActivity中那個ListView的,其最主要的優勢在於添加Preference後可讓其狀態持久化儲存(通過SharedPreferences,一般存儲在/data/data//shared_prefs文件夾下的默認名為“app package name”+”_preferences.xml”的文件裡),比如說用戶勾選CheckBox後退出應用,下一次進入到這一界面時候,對應的是CheckBox依然是被勾選狀態,如果要實現這樣的機制,我們自己也可以實現,但是沒有必要,因為Android已經替我們實現了,就是我們的這一系列的主角——Preference,Preference會自動地替我們去保存這些狀態對應的值到對應的SharedPreferences文件裡,而當我們每次啟動的時候Acitivity(PreferenceActivity)會自動根據key去獲取相關數據,完成用戶界面的更新。我們手機當中的系統設置就是及其典型的Preference的應用,也正是由於工作中需要去客制化Settings,才有了這一系列的文章。
上圖是我們定制的Settings模塊中的對應的部分SharedPreferences。關於Preference對應的SharedPreferences往往很容易被我們忽視兩點
並非我們第一次打開相應界面之後就會自動創建對應的SharedPreferences文件,而是在我們改變了原有狀態時候<喎?/kf/ware/vc/" target="_blank" class="keylink">vc3Ryb25nPjwvcD4NCjxwPjxzdHJvbmc+sqK3x8v509C1xFByZWZlcmVuY2W8sMbk19PA4La8u+G0tL2oo6y99r321eu21NDo0qq8x8K817TMrLXEUHJlZmVyZW5jZTwvc3Ryb25nPqGjPC9wPg0KPGgyIGlkPQ=="2preferenceactivity的初始化">2、PreferenceActivity的初始化
PreferenceActivity其實和普通的Activity本質上來說區別不大,只不過多了些自動去讀取SharedPrefrences的值來更新界面和其他一些邏輯,所以初始化本質上來說並無很大的區別,但是與普通Activity的layout不同,PreferenceActivity的layout我們可以理解成為兩個部分:其他View和一個id為android.R.id.list的ListView,那麼我們可以理解成為當我們在onCreate方法裡先調用setContentView完成整個Activity的View的構建(layout文件裡必須包含id為android.R.id.list的listView,否則會報E/AndroidRuntime: Caused by: java.lang.RuntimeException: Your content must have a ListView whose id attribute is ‘android.R.id.list’),再調用addPreferencesFromResource來完成Preference界面的構建;當然也可以只調用addPreferencesFromResource方法。 在res文件夾下新建xml文件夾,再在xml文件中新建對應的xml資源,xml資源類似我們在使用普通Activity時的layout文件,PreferenceActivity獨特之處在於並不是使用普通的layout文件,而是使用res下xml文件夾下的xml資源文件 res/xml/demo_preference.xml 也可通過代碼去構造對象,添加容器之後再調用setPreferenceScreen(PreferenceScreen preferenceScreen)把容器對象設置到Activity上 前一篇文章我們講述了Preference家族的基類(接下來我們所要介紹的其他子類Preference一定是繼承了他的所有屬性也可以理解成都是在繼承他所展示的UI效果及交互功能的基礎上升級的)而且Preference可以實例化,那麼我們就可以把他看成對應的一個組件,其實和我們熟悉的TextView一樣,所以我們需要使用的時候第一步肯定是先得到他的對象——而構造對象我們都可以通過兩種方式:通過其對應的構造方法或者其他方法和通過xml映射(或許說法不夠准確),接著第二步PreferenceActivity的初始化,再接著根據業務設置相關監聽。 前面Android進階——Preference詳解之初識Preference及Preference系(一)已經講過PreferenceScreen只能作為top-level節點,而構造對象我們都可以通過兩種方式:通過其對應的構造方法或者其他方法和通過映射xml(或者講法不夠准確)。 res/xml/test_preference.xml 常見的Preference的事件有兩個:setOnPreferenceClickListener和onPreferenceChange 在xml文件的Preference標簽中,我們可以添加intent來為我們快速實現一種意圖,比如說快速打開一個網頁,或者快速啟動一個Activity等等,還可以使用extra給intent標簽加參數來傳遞參數(再通過getIntent().getStringExtra(“key”)來獲取)。 點擊這個Preference則會自動去調用浏覽器打開http://www.hao123.com網頁 啟動指定類 PreferenceScreen和PreferenceCategory沒有新增的屬性,所有屬性全部繼承自Preference。其中PreferenceScreen作為頂級容器,PreferenceCategory作為次級容器(類似於SQL Group by功能暫且這麼理解吧),雖然他們也是可以單獨使用的,但是並不能響應onPreferenceClick和onPreferenceChange事件。 布局和MainActivity的代碼依然很簡單和前面類似 RingtonePreference起作用就是供我們選擇系統鈴聲的,除了Preference共有的屬性外還新增了自己的幾個獨特屬性。 應用也很簡單,與Preference大同小異(注意看圖) Preference的管理主要包含Preference的創建、添加、移除和尋找特定Preference,Preference並沒有直接提供相關替換的方法。 主要是通過各自對應Preference的構造方法或者直接在xml文件中定義來創建對應的Preference 添加只要是就是調用PreferenceGroup的addPreference(Preference preference)方法 來添加至容器PreferenceGroup。 對於PreferenceActivity的findPreference(key)方法,我們只需要知道key值就可以找到同一xml文件下相應的Preference,勿需考慮層級和嵌套關系 對於PreferenceGroup的removePreference(Preference preference)和removeAll()方法,都是針對某個PreferenceGroup來處理的,所以我們必須考慮層級嵌套關系,可以分為兩步:先找到PreferenceGroup和要刪除的Preference,再調用PreferenceGroup的removePreference執行刪除動作。 首先這是我們的布局 測試主體代碼 這篇文章主要介紹了Preference家族樹中頂級成員和次級成員的應用和簡單原理的說明,也基本把幾乎所有相關的知識點都涉及了,Preference的基本語法都是一樣的,區別在於各自不同的特性。1、繼承PreferenceActivity實現具體業務類,重寫相關生命周期方法
public class MainActivity extends PreferenceActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
addPreferencesFromResource(R.xml.demo_preference);
}
...
}
2、通過addPreferencesFromResource(xml資源id)加載靜態xml資源文件 或者 完全通過代碼構造對象再動態添加
private void createPreference(){
PreferenceScreen preferenceScreen = this.getPreferenceManager().createPreferenceScreen(this);//先構建PreferenceScreen對象得到一個布局容器
this.setPreferenceScreen(preferenceScreen);//設置容器
CheckBoxPreference checkBoxPreference=new CheckBoxPreference(this);//構建一個子Preference,待添加到容器中
checkBoxPreference.setKey(CHECKBOXPRERENCE_KEY);//設置key
checkBoxPreference.setTitle("The Title Of CheckBoxPreference");//設置title
checkBoxPreference.setSummary("Some summay for CheckBoxPreference");
preferenceScreen.addPreference(checkBoxPreference);//添加到容器中
}
二、Preference的使用
1、構造Preference容器和Preference對象
2、初始化PreferenceActivity和設置相關監聽
設置Preference對應的SharedPrefernces值監聽
preference.setOnPreferenceClickListener(new Preference.OnPreferenceClickListener() {
@Override
public boolean onPreferenceClick(Preference preference) {
//當接收到Click事件之後觸發
return true;
}
});
preference.setOnPreferenceChangeListener(new Preference.OnPreferenceChangeListener() {
@Override
public boolean onPreferenceChange(Preference preference, Object newValue) {
//如果值改變了我們可以通過監聽這個事件來處理
return true;
}
});
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);//activity_main裡必須存在id為android.R.id.list的ListView否則報E/AndroidRuntime: Caused by: java.lang.RuntimeException: Your content must have a ListView whose id attribute is 'android.R.id.list'
addPreferencesFromResource(R.xml.test_preference);
mContext=getApplicationContext();
preference=findPreference(PREFERENCE_KEY);
preference.setOnPreferenceChangeListener(new Preference.OnPreferenceChangeListener() {
@Override
public boolean onPreferenceChange(Preference preference, Object newValue) {
//如果值改變了我們可以通過監聽這個事件來獲取新值
Toast.makeText(mContext, String.format("Preference的值為%s", newValue),Toast.LENGTH_LONG).show();
return true;
}
});
//設置Preference的點擊事件監聽
preference.setOnPreferenceClickListener(new Preference.OnPreferenceClickListener() {
@Override
public boolean onPreferenceClick(Preference preference) {
//當接收到Click事件之後觸發
Toast.makeText(mContext, "Preference Clicked",Toast.LENGTH_LONG).show();
return true;
}
});
}
3、Preference的常用xml屬性
xml屬性
說明
android:icon
Preference的icon
android:summary
副標題、說明(小字體顯示)
android:title
標題
android:key
對應的SharedPrefrences的key值
android:layout
PreferenceActivity的布局文件即如果設置了這個屬性則會覆蓋原來的UI
android:fragment
應用在PreferenceActivity中時,當用戶點擊這個Preference時,啟動一個新的Fragment
android:enabled
設置Preference是否可用,false陰影狀態不可操作
android:order
Preference的排序,整數類型如”100”
android:persistent
true時,系統會幫助我們去保存該設置,即使重啟後依然能記憶之前的設置,這也是所謂的持久化 。這裡 將 android:persistent設置為False,表明不需要讓系統去做持久化,開發者系統通過自己的方式去實現持久化。
android:selectable
Preference是否可選,false可以點擊但無響應
android:shouldDisableView
當View disabled的時候是否Preference也一樣Disabled.
android:widgetLayout
自定義Preference用到給子Preference定義布局。
4、Preference的xml文件中的常用標簽intent、extra
三、PreferenceScreen和PreferenceCategory
四、RingtonePreference的應用
新增屬性
說明
android:ringtoneType
鈴聲類型:ringtone、notification、all、alarm
android:showDefault
布爾值是否顯示默認鈴聲
android:showSilent
布爾值是否顯示靜音
五、Preference的管理
1、創建Preference
2、addPreference添加Preference
3、findPreference尋找特定Preference
4、removePreference或removeAll移除Preference
public class MainActivity extends PreferenceActivity {
private Context mContext;
private Preference preference;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
addPreferencesFromResource(R.xml.test_preference);
//removePreferenceByKey();
mContext=getApplicationContext();
preference=findPreference("key_checkbox_child");//只要是同一個xml文件下的所有Preference都能通過key直接找到
preference.setOnPreferenceChangeListener(new Preference.OnPreferenceChangeListener() {
@Override
public boolean onPreferenceChange(Preference preference, Object newValue) {
Toast.makeText(mContext, String.format("Preference的值為%s", newValue),Toast.LENGTH_LONG).show();
return true;
}
});
preference.setOnPreferenceClickListener(new Preference.OnPreferenceClickListener() {
@Override
public boolean onPreferenceClick(Preference preference) {
Toast.makeText(mContext, "Preference Clicked",Toast.LENGTH_LONG).show();
//getFragmentManager().beginTransaction().replace(android.R.id.content,new MainFragment()).commit();
return false;
}
});
}
private void removePreferenceByKey(){
/*PreferenceGroup preferenceGroup=(PreferenceGroup)findPreference("key_category");//先找到PreferenceGroup
Preference preference=findPreference("key_edtprefs");//再找到要刪除的Preference*/
/*(PreferenceGroup)findPreference("key_child_prefscreen").removePreference(findPreference("key_checkbox_child"));//執行刪除key為key_checkbox_child的Preference**/
getPreferenceScreen().removePreference(findPreference("key_category"));//刪除掉key_category及對應節點下的所有節點
//ERROR//getPreferenceScreen().removePreference(findPreference("key_edtprefs"));//無效,因為getPreferenceScreen獲得的是當前的頂級容器,而key_edtprefs不是它的直接字節點
////((PreferenceGroup)findPreference("key_category")).removeAll();//僅刪除掉key_category下對應Preference節點下的所有子節點
}
小結
線程與進程的最大區別就是是否共享父進程的地址空間,內核角度來看沒有線程與進程之分,都用task_struct結構體來表示,調度器操作的實體便是task_st
AdapterViewFlipper簡介AdapterViewFlipper繼承了AdapterViewAnimater。每次只能顯示一個組件,用showPrevious
當用戶在用一個App的時候,Activity會在生命周期不同的狀態下轉換。當用戶離開或重新回到一個Activity,我們可以在生命周期的回調函數中來寫一些相關的操作。比如
本文提到的所有數字模型制作,全部是用3D MAX建立模型,即使是不同的驅動引擎,對模型的要求基本是相同的。當一個VR模型制作完成時,它所包含的基本內容包括場景尺寸、單位,