android 皮肤切换
转载于 http://blog.csdn.net/suiyc/article/details/6329212
国内有很多的软件都支持皮肤定制,这也是与国外软件重大不同之一,国外用户注重社交、邮件等功能,国内用户则重视音乐、小说、皮肤等功能,本节课程就来讲解Android应用程序如何实现换肤功能。
软件换肤从功能上可以划分三种:
1) 软件内置多个皮肤,不可由用户增加或修改;
最低的自由度,软件实现相对于后两种最容易。
2) 官方提供皮肤供下载,用户可以使用下载的皮肤;
用户可选择下载自己喜欢的皮肤,有些玩家会破解皮肤的定制方法,自己做皮肤使用,或者传到网上给大家用。
3) 官方提供皮肤制作工具或方法,用户可自制皮肤。
这种方式使用户有参与感,自由度较高。用户可根据自己的喜好定制软件的皮肤。有些软件官网提供皮肤定制的工具或者方法,我建议最好有可视化带向导的工具。用户只要自己找一些图片、修改文字的字体替换就可以了。用户可以上传自制的皮肤,提供其他用户下载,还可以赚得一些虚拟货币或者奖品什么的。这种一般都是打包为.zip格式的。扩展名可由各公司自定义,有制作工具的话直接导出来最方便。
首先我们要弄清楚换肤的定义,软件皮肤包括图标、字体、布局、交互风格等,换肤就是换掉皮肤包括的部分或所有资源。
前面提到的三种皮肤,从软件实现上来看,它们的本质区别是皮肤是否内置到应用程序中。对于内置的实现比较简单,只要在开发应用的过程中设计几套皮肤供用户选择。这里用到的知识不超过Android基础,不详细讲解。
本节课程重点讲解如何实现皮肤与应用程序分离。
皮肤一般含有多个文件,例如图片、配置等文件,分散的文件不利于传输和使用,最好打包。打包的格式一般选择zip格式。这里分两种情况,一种是apk,例如AdwLauncher,它的桌面皮肤格式是一个apk;另一种是自定义扩展名,例如墨迹天气皮肤扩展名是mja,搜狗输入法的皮肤扩展名是sga,它们的文件格式实际上都是zip。
下面我们分别讲解。
一.apk格式
现在的问题变成了一个应用如何读取另一个apk中的资源。
在android系统中,apk之间可以相互读取数据的条件是:有同样的签名,并且AndroidManifest.xml文件中配置的android:sharedUserId属性值相同,那么两个apk运行在同一个进程中,可以互相访问任意数据。
方法如下:
1) 应用程序和皮肤程序的AndroidManifest.xml中配置
例如: android:sharedUserId="org.yuchen"
2) 文件与应用apk中对同一功能的皮肤文件名要一致
例如:应用程序的背景图片路径:/SkinDemo/res/drawable-hdpi/bg.png
那么皮肤apk中的背景图片文件路径也应该是:
CustomSkin/res/drawable-hdpi/bg.png
3)访问资源的方法
Context context = createPackageContext("com.yuchen.customskin", Context.CONTEXT_IGNORE_SECURITY);Drawable drawable = context.getResources().getDrawable(R.drawable.bg);
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;@Overrideprotected 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(){@Overridepublic void handleMessage(Message msg) {switch(msg.what){case MESSAGE_SEARCHED_SKIN:ArrayList skins=(ArrayList) msg.obj;//获取skinsadapter=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));}@Overridepublic 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, ""));}}public interface OnThemeChangedListener { public void onChanged(String newThemePackageName); } public void setOnThemeChangedListener(OnThemeChangedListener listener) { this.listener=listener; } public void onSharedPreferenceChanged(SharedPreferences sharedPreferences, String key) { if(key.equals("themePackage")) { listener.onChanged(sp.getString("themePackage", "")); } }