Android中实现应用切换主题机制

一直很想弄清楚好多应用中是如何实现换皮肤这项功能的,花了下午点时间,查了下资料也实现了个切换主题的Demo;

首先要感谢下这位大哥,参阅了下他写的文件 http://www.eoeandroid.com/forum-viewthread-tid-31756-highlight-%E7%9A%AE%E8%82%A4.html

好了,废话不多说了,该切换主题的demo里面一共实现了两个功能,其一,搜索已经安装的皮肤,其二,应用安装的皮肤。

主项目包名为org.leepood.skindemo,主题项目的包名为org.leepood.skin.blue,org.leepood.skin.red,等等,只要前缀是org.leepood.skin.就行。

首先是查找已安装主题的代码:

Java代码 复制代码 收藏代码
  1. packageorg.leepood.skindemo;
  2. importjava.util.ArrayList;
  3. importjava.util.List;
  4. importandroid.app.Activity;
  5. importandroid.app.ProgressDialog;
  6. importandroid.content.Context;
  7. importandroid.content.Intent;
  8. importandroid.content.SharedPreferences;
  9. importandroid.content.pm.PackageInfo;
  10. importandroid.content.pm.PackageManager;
  11. importandroid.content.pm.PackageManager.NameNotFoundException;
  12. importandroid.content.res.Resources;
  13. importandroid.os.Bundle;
  14. importandroid.os.Handler;
  15. importandroid.os.Message;
  16. importandroid.view.ContextMenu;
  17. importandroid.view.LayoutInflater;
  18. importandroid.view.MenuItem;
  19. importandroid.view.View;
  20. importandroid.view.ViewGroup;
  21. importandroid.view.ContextMenu.ContextMenuInfo;
  22. importandroid.view.View.OnCreateContextMenuListener;
  23. importandroid.widget.BaseAdapter;
  24. importandroid.widget.ImageView;
  25. importandroid.widget.ListView;
  26. importandroid.widget.TextView;
  27. importandroid.widget.Toast;
  28. importandroid.widget.AdapterView.AdapterContextMenuInfo;
  29. publicclassMainextendsActivityimplementsSharedPreferences.OnSharedPreferenceChangeListener{
  30. privateListViewlistview;
  31. privateContextc;
  32. privateHandlermHandler;
  33. privateProgressDialogpDialog;
  34. privateSkinAdapteradapter;
  35. privateSharedPreferencessp;
  36. staticfinalintMESSAGE_SEARCHED_SKIN=0;
  37. staticfinalintMESSAGE_SEARCHING_SKIN=MESSAGE_SEARCHED_SKIN+1;
  38. staticfinalintMESSAGE_SEARCHED_SKIN_FOR_NONTHING=MESSAGE_SEARCHING_SKIN+1;
  39. @Override
  40. protectedvoidonCreate(BundlesavedInstanceState){
  41. super.onCreate(savedInstanceState);
  42. setContentView(R.layout.main);
  43. init();
  44. pDialog.show();
  45. newThread(serachSkin).start();
  46. }
  47. privatevoidinit()
  48. {
  49. c=this;
  50. mHandler=newHandler(){
  51. @Override
  52. publicvoidhandleMessage(Messagemsg){
  53. switch(msg.what)
  54. {
  55. caseMESSAGE_SEARCHED_SKIN:
  56. ArrayList
  57. skins=(ArrayList
  58. )msg.obj;//获取skins
  59. adapter=newSkinAdapter(c,skins);
  60. listview.setAdapter(adapter);
  61. Toast.makeText(c,"查找到已经安装的皮肤",1).show();
  62. pDialog.dismiss();
  63. break;
  64. caseMESSAGE_SEARCHED_SKIN_FOR_NONTHING:
  65. Toast.makeText(c,"未查找到任何皮肤",1).show();
  66. pDialog.dismiss();
  67. }
  68. }
  69. };
  70. sp=this.getSharedPreferences("config",Context.MODE_WORLD_WRITEABLE);
  71. sp.registerOnSharedPreferenceChangeListener(this);
  72. listview=(ListView)findViewById(R.id.list);
  73. listview.setItemsCanFocus(false);
  74. listview.setChoiceMode(ListView.CHOICE_MODE_SINGLE);
  75. pDialog=newProgressDialog(this);
  76. pDialog.setMessage("正在查找已经安装的皮肤");
  77. listview.setOnCreateContextMenuListener(newOnCreateContextMenuListener(){
  78. publicvoidonCreateContextMenu(ContextMenumenu,Viewv,
  79. ContextMenuInfomenuInfo){
  80. menu.add("使用该主题");
  81. }
  82. });
  83. }
  84. privateRunnableserachSkin=newRunnable(){
  85. publicvoidrun(){
  86. PackageManagermanager=c.getPackageManager();
  87. List
  88. packages=manager.getInstalledPackages(PackageManager.PERMISSION_GRANTED);
  89. ArrayList
  90. skins=newArrayList
  91. ();
  92. for(PackageInfoinfo:packages)
  93. {
  94. //System.out.println(info.packageName);
  95. if(info.packageName.startsWith("org.leepood.skin."))
  96. {
  97. skins.add(info);
  98. }
  99. }
  100. if(skins.size()>0)
  101. {
  102. Messagemsg=mHandler.obtainMessage();
  103. msg.obj=skins;
  104. msg.what=MESSAGE_SEARCHED_SKIN;
  105. mHandler.sendMessage(msg);
  106. }
  107. else
  108. {
  109. mHandler.sendEmptyMessage(MESSAGE_SEARCHED_SKIN_FOR_NONTHING);
  110. }
  111. }
  112. };
  113. privateclassSkinAdapterextendsBaseAdapter
  114. {
  115. LayoutInflatermInflater;
  116. ArrayList
  117. datas;
  118. PackageManagermanager;
  119. publicSkinAdapter(Contextc,ArrayList
  120. datas)
  121. {
  122. this.datas=datas;
  123. mInflater=LayoutInflater.from(c);
  124. manager=c.getPackageManager();
  125. }
  126. publicintgetCount(){
  127. returndatas.size();
  128. }
  129. publicObjectgetItem(intposition){
  130. returndatas.get(position);
  131. }
  132. publiclonggetItemId(intposition){
  133. return0;
  134. }
  135. publicViewgetView(intposition,ViewconvertView,ViewGroupparent){
  136. if(convertView==null)
  137. {
  138. convertView=mInflater.inflate(R.layout.skin_item,null);
  139. }
  140. ImageViewicon=(ImageView)convertView.findViewById(R.id.skin_icon);
  141. TextViewskin_name=(TextView)convertView.findViewById(R.id.skin_name);
  142. PackageInfoinfo=datas.get(position);
  143. icon.setImageDrawable(info.applicationInfo.loadIcon(manager));
  144. skin_name.setText(info.applicationInfo.loadLabel(manager));
  145. returnconvertView;
  146. }
  147. }
  148. publicvoidonThemeChanged(StringnewThemePackageName){
  149. try{
  150. ContextthemeContext=this.createPackageContext(newThemePackageName,CONTEXT_IGNORE_SECURITY);
  151. Resourcesres=themeContext.getResources();
  152. setControlsStyle(res);
  153. }catch(NameNotFoundExceptione){
  154. e.printStackTrace();
  155. }
  156. }
  157. privatevoidsetControlsStyle(Resourcesres)
  158. {
  159. listview.setBackgroundColor(res.getColor(R.color.ListView_bg));
  160. }
  161. @Override
  162. publicbooleanonContextItemSelected(MenuItemitem){
  163. AdapterContextMenuInfomenuInfo=(AdapterContextMenuInfo)item.getMenuInfo();
  164. PackageInfoinfo=(PackageInfo)adapter.getItem(menuInfo.position);
  165. sp.edit().putString("themePackage",info.packageName).commit();
  166. returnsuper.onContextItemSelected(item);
  167. }
  168. publicvoidonSharedPreferenceChanged(SharedPreferencessharedPreferences,
  169. Stringkey){
  170. System.out.println("themeChange");
  171. onThemeChanged(sharedPreferences.getString(key,""));
  172. }
  173. }
package org.leepood.skindemo; import java.util.ArrayList; import java.util.List; import android.app.Activity; import android.app.ProgressDialog; import android.content.Context; import android.content.Intent; import android.content.SharedPreferences; import android.content.pm.PackageInfo; import android.content.pm.PackageManager; import android.content.pm.PackageManager.NameNotFoundException; import android.content.res.Resources; import android.os.Bundle; import android.os.Handler; import android.os.Message; import android.view.ContextMenu; import android.view.LayoutInflater; import android.view.MenuItem; import android.view.View; import android.view.ViewGroup; import android.view.ContextMenu.ContextMenuInfo; import android.view.View.OnCreateContextMenuListener; import android.widget.BaseAdapter; import android.widget.ImageView; import android.widget.ListView; import android.widget.TextView; import android.widget.Toast; import android.widget.AdapterView.AdapterContextMenuInfo; public class Main extends Activity implements SharedPreferences.OnSharedPreferenceChangeListener{ private ListView listview; private Context c; private Handler mHandler; private ProgressDialog pDialog; private SkinAdapter adapter; private SharedPreferences sp; static final int MESSAGE_SEARCHED_SKIN=0; static final int MESSAGE_SEARCHING_SKIN=MESSAGE_SEARCHED_SKIN+1; static final int MESSAGE_SEARCHED_SKIN_FOR_NONTHING=MESSAGE_SEARCHING_SKIN+1; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); init(); pDialog.show(); new Thread(serachSkin).start(); } private void init() { c=this; mHandler=new Handler(){ @Override public void handleMessage(Message msg) { switch(msg.what) { case MESSAGE_SEARCHED_SKIN: ArrayList skins=(ArrayList ) msg.obj;//获取skins adapter=new SkinAdapter(c, skins); listview.setAdapter(adapter); Toast.makeText(c, "查找到已经安装的皮肤", 1).show(); pDialog.dismiss(); break; case MESSAGE_SEARCHED_SKIN_FOR_NONTHING: Toast.makeText(c, "未查找到任何皮肤", 1).show(); pDialog.dismiss(); } } }; sp=this.getSharedPreferences("config",Context.MODE_WORLD_WRITEABLE); sp.registerOnSharedPreferenceChangeListener(this); listview=(ListView) findViewById(R.id.list); listview.setItemsCanFocus(false); listview.setChoiceMode(ListView.CHOICE_MODE_SINGLE); pDialog=new ProgressDialog(this); pDialog.setMessage("正在查找已经安装的皮肤"); listview.setOnCreateContextMenuListener(new OnCreateContextMenuListener() { public void onCreateContextMenu(ContextMenu menu, View v, ContextMenuInfo menuInfo) { menu.add("使用该主题"); } }); } private Runnable serachSkin =new Runnable(){ public void run() { PackageManager manager=c.getPackageManager(); List packages=manager.getInstalledPackages(PackageManager.PERMISSION_GRANTED); ArrayList skins=new ArrayList (); for(PackageInfo info:packages) { //System.out.println(info.packageName); if(info.packageName.startsWith("org.leepood.skin.")) { skins.add(info); } } if(skins.size()>0) { Message msg=mHandler.obtainMessage(); msg.obj=skins; msg.what=MESSAGE_SEARCHED_SKIN; mHandler.sendMessage(msg); } else { mHandler.sendEmptyMessage(MESSAGE_SEARCHED_SKIN_FOR_NONTHING); } } }; private class SkinAdapter extends BaseAdapter { LayoutInflater mInflater; ArrayList datas; PackageManager manager; public SkinAdapter(Context c,ArrayList datas) { this.datas=datas; mInflater=LayoutInflater.from(c); manager=c.getPackageManager(); } public int getCount() { return datas.size(); } public Object getItem(int position) { return datas.get(position); } public long getItemId(int position) { return 0; } public View getView(int position, View convertView, ViewGroup parent) { if(convertView==null) { convertView=mInflater.inflate(R.layout.skin_item, null); } ImageView icon=(ImageView) convertView.findViewById(R.id.skin_icon); TextView skin_name=(TextView) convertView.findViewById(R.id.skin_name); PackageInfo info=datas.get(position); icon.setImageDrawable(info.applicationInfo.loadIcon(manager)); skin_name.setText(info.applicationInfo.loadLabel(manager)); return convertView; } } public void onThemeChanged(String newThemePackageName) { try { Context themeContext=this.createPackageContext(newThemePackageName, CONTEXT_IGNORE_SECURITY); Resources res=themeContext.getResources(); setControlsStyle(res); } catch (NameNotFoundException e) { e.printStackTrace(); } } private void setControlsStyle(Resources res) { listview.setBackgroundColor(res.getColor(R.color.ListView_bg)); } @Override public boolean onContextItemSelected(MenuItem item) { AdapterContextMenuInfo menuInfo=(AdapterContextMenuInfo)item.getMenuInfo(); PackageInfo info=(PackageInfo) adapter.getItem(menuInfo.position); sp.edit().putString("themePackage", info.packageName).commit(); return super.onContextItemSelected(item); } public void onSharedPreferenceChanged(SharedPreferences sharedPreferences, String key) { System.out.println("themeChange"); onThemeChanged(sharedPreferences.getString(key, "")); } }
这段代码的含义就是去查找系统中安装的包名,若以org.leepood.skin.开头则说明该包为主题包,将其加入listview中显示出来。代码中使用了多线程避免时间过长堵塞UI。程序将当前主题配置保存在SharedPreference中,为SharedPreference注册了一个监听函数,当其值发生改变时自动调用新的样式。当然,这只是个demo而已,一开始加载Activity没有去读取主题,这个可以由大家自己去实现。
最后贴张图片:




Android实现主题切换机制2
昨天花了点时间实现了主题的切换,但是里面还是不够灵活,回去想了想可以用继承和回调函数来进一步灵活更改主题,现在记录下我的实现办法
首先一个自定义类ThemeActivity继承自Activity,这个类是以后所有Activity的父类,在这个类里面定义了一个接口

public interface OnThemeChangedListener
{
public void onChanged(String newThemePackageName);

}
接下来,首先是要给ThemeActivity注册一个主题切换的listener,代码如下:

public void setOnThemeChangedListener(OnThemeChangedListener listener)
{
this.listener=listener;
}
然后就是注册一个SharedPreference来监听xml的变化,当发生改变的时候自动去调用listener.onChanged方法,将新的主题包名传递过去,代码如下:

public void onSharedPreferenceChanged(SharedPreferences sharedPreferences,
String key) {
if(key.equals("themePackage"))
{
listener.onChanged(sp.getString("themePackage", ""));
}

}
接着在继承于ThemeActivity的子类里面首先是setOnThemeChangedListener.接着用一个匿名内部类搞定。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值