Android设置选项开发及自定义Preference样式

  一个完整的Android应用程序都应该提供选项(或者叫偏好设置等等)让用户对APP的表现形式能够进行设置,比如说是否加入用户体验计划,或者是否自动升级、定时提醒、开启自启动、后台运行等等。提供一个好的设置项,会大大提升APP的用户体验。为了完成这样的功能,你不必从头开始写Activity或者Fragment,因为Android已经提供了实现这个功能的API,并且会自动将用户设置以键值对的形式存入SharedPreference(Android的四大存储方式之一)中。在3.0以前的系统,使用PreferenceActivity,这个类在api level 11(即Android 3.0)以后的api中丢弃,改用PreferenceFragment。两者的使用方式及函数调用大同小异,可以根据app的目标系统版本自己去衡量。本文主要说明两个问题,层次较浅,重在总结和说明基本用法,懂的直接飘过吧。

1. 为APP添加设置选项

  Android平台上,为应用添加设置选项是个非常容易的事儿。这里以PreferenceFragment为例进行演示,毕竟时代向前发展嘛。PreferenceFragment的父类是Fragment类,而Fragment对象必须嵌入到Activity中显示出来。由此可以确定思路,为设置新建一个activity,然后将PreferenceFragment子类对象嵌入到其中,基本上就实现了选项设置,因为数据的保存与更新自动进行。

  思路非常简单,还是贴下主要代码,顺便整理下思路,帮助理解。

  首先为设置选项设置新建一个Preference配置文件,跟layout文件也是XML文件格式,层次化清晰,注意它存储在res/xml下,而不是res/layout。系统也提供了一些比较常用了设置选项,比如PreferenceScreen,PreferenceCategory,CheckBoxPreference,EditTextPreferece,ListPreference等,如果需要你也可以很方便的实现自定义的Preference,下文将会介绍实现方法。现在新建一个Preference,命名settings.xml(更传统的命名为preference.xml)。

 1 <?xml version="1.0" encoding="utf-8"?>
 2 <PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android"
 3     xmlns:preference="http://schemas.android.com/apk/res/com.test.mytest"
 4     android:title="设置" >
 5 
 6     <PreferenceScreen
 7         android:title="关于" >
 8         <Preference android:title="意见反馈" >
 9         </Preference>
10 
11         <com.test.mytest.PreferenceWithTip
12             preference:tipstring=">"
13             preference:titlestring="自定义测试" >
14             <intent
15                 android:action="android.intent.action.VIEW"
16                 android:data="http://www.baidu.com" />
17         </com.test.mytest.PreferenceWithTip>
18 
19         <Preference android:title="常见问题" >
20         </Preference>
21         <Preference android:title="检查更新" >
22         </Preference>
23         <Preference android:title="版权声明" >
24         </Preference>
25 
26         <SwitchPreference
27             android:key="setting_test"
28             android:title="测试一下" />
29     </PreferenceScreen>
30 
31 </PreferenceScreen>

 

  然后为设置选项新建一个Acitivity,因为此处PreferenceFragment子类写的非常简单,顺便以内部类实现了。

 1 public class SettingsActivity extends Activity {
 2 
 3     @Override
 4     public void onCreate(Bundle savedInstanceState) {
 5         super.onCreate(savedInstanceState);
 6         setContentView(R.layout.settings);
 7         setTitle("选项设置");
 8         getFragmentManager().beginTransaction().replace(R.id.settings_content, 
 9                 new PrefsFragment()).commit();
10     }
11     
12     public static class PrefsFragment extends PreferenceFragment{
13 
14         @Override
15         public void onCreate(Bundle savedInstanceState) {
16             super.onCreate(savedInstanceState);
17             addPreferencesFromResource(R.xml.settings);
18         }
19         
20     }
21     
22 }

  最后就差用户点击你设计好的设置选项了,到了这里你应该猜到了,打开设置选项不过只是打开一个Intent而已。基本流程就到这里,但是一个需要获得大用户量应用的设置要比这个复杂得多,你可能还需要根据用户的设置,立即对应用的表现做出调整,可能要实现onPreferenceTreeClick(PreferenceScreen  preferenceScreen,Preference prefence)。正如前文所述,刚接触Preference,这里仅仅总结基本用法。

2. 在设置选项中使用自定义的Preference

  Preference类直接继承于Object类。在上文的settings.xml中,定义好几个Preference,Preference只提供简单的文本显示,而它的的子类CheckBoxPreference,SwitchPreference,EditTextPreference等则提供了较为复杂的UI展示,并可以保存用户的设置数据,一般来说,这些子类Preference对于应用程序更加重要。关于如果使用这些子类对象,其实很简单,他们可以像UI控件在Layout中的用法类似的应用在Preference定义的xml文件(上文定义的settings.xml)中,基本上使用了eclipse代码提示功能就可以使用,这些用法基础但不是本文的说明重点。下面旨在介绍如何定义自己的Preference,先上图看效果。

图一 自定义Preference展示

  图一展示了Preference与自定义Preference样式差别,你或许注意到第二项”自定义测试“与其他的Preference只有一个“>“符号的差别,其实这里包含了自定义一个Preference的完整步骤。说道这里,顺便说下,其实自定义Preference与自定义控件的方法和套路几乎一致。还是总结下基本步骤。

  1) 定义属性值 attr.xml

1 <?xml version="1.0" encoding="utf-8"?>
2 <resources>
3     <declare-styleable name="PreferenceWithTip">
4         <attr name="tipstring" format="string"></attr>
5         <attr name="titlestring" format="string"></attr>
6     </declare-styleable>
7 </resources>

 

  2) 设计自定义Preference的布局 preferencewithtip.xml

 1 <?xml version="1.0" encoding="utf-8"?>
 2 <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
 3     android:layout_width="match_parent"
 4     android:layout_height="match_parent"
 5     android:orientation="horizontal"
 6     android:paddingLeft="8dp"
 7     android:paddingRight="15dp"
 8     android:paddingTop="20dp"
 9     android:paddingBottom="20dp">
10     <TextView 
11         android:id="@+id/prefs_title"
12         android:layout_width="0dp"
13         android:layout_height="wrap_content"
14         android:layout_gravity="left"
15         android:gravity="left|center_vertical"
16         android:textSize="18sp"
17         android:layout_weight="1"/>
18     <TextView 
19         android:id="@+id/prefs_tip"
20         android:layout_width="0dp"
21         android:layout_height="wrap_content"
22         android:layout_gravity="right"
23         android:gravity="right|center_vertical"
24         android:textSize="18sp"
25         android:layout_weight="1"/>
26 
27 </LinearLayout>
View Code

  3) 继承Preference,实现自己的Preference类 PreferenceWithTip

 1 public class PreferenceWithTip extends Preference {
 2     private static final String TAG = "PreferenceWithTip";
 3     String pTitle = null;
 4     String tipstring = null;
 5     
 6     @SuppressLint("Recycle")
 7     public PreferenceWithTip(Context context, AttributeSet attrs, int defStyle) {
 8         super(context, attrs, defStyle);
 9         // 获取自定义参数
10         Log.i(TAG,"PreferenceWithTip invoked");
11         TypedArray ta = context.obtainStyledAttributes(attrs, R.styleable.PreferenceWithTip);
12         tipstring = ta.getString(R.styleable.PreferenceWithTip_tipstring);
13         pTitle = ta.getString(R.styleable.PreferenceWithTip_titlestring);
14         ta.recycle();
15     }
16 
17     public PreferenceWithTip(Context context, AttributeSet attrs) {
18         this(context, attrs, 0);
19     }
20 
21     @Override
22     protected void onBindView(View view) {
23         super.onBindView(view);
24         TextView pTitleView = (TextView)view.findViewById(R.id.prefs_title);
25         pTitleView.setText(pTitle);
26         TextView pTipView = (TextView)view.findViewById(R.id.prefs_tip);
27         pTipView.setText(tipstring);
28     }
29 
30     @Override
31     protected View onCreateView(ViewGroup parent) {
32         return LayoutInflater.from(getContext()).inflate(R.layout.preferencewithtip,
33                 parent, false);
34     }
35     
36     //如需更新、保存数据则需要继续编写
37     
38 }
View Code

  4) 调用。调用代码在文章的开头部分已经贴出,主要代码如下,preference是自定义的包名。

1             <com.ict.customview.PreferenceWithTip
2                 preference:tipstring=">"
3                 preference:titlestring="自定义测试" >
4                 <intent
5                     android:action="android.intent.action.VIEW"
6                     android:data="http://www.baidu.com" />
7             </com.ict.customview.PreferenceWithTip>

 

  总结一下Preference的使用还是比较简单的,自定义Preference也比较方便。但是要设计出一个漂亮的、人性化的Preference还是不那么容易,但这些都是提高用户体验的途径,值得进一步挖掘。

 

posted @ 2014-10-13 18:11  指尖的舞客  阅读(12568)  评论(0编辑  收藏  举报