zoukankan      html  css  js  c++  java
  • 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)访问资源的方法 

    Java代码  收藏代码
    1. Context context = createPackageContext("com.yuchen.customskin", Context.CONTEXT_IGNORE_SECURITY);  


    获取到org.yuchen.customskin对应的Context,通过返回的context对象就可以访问到org.yuchen.customskin中的任何资源。 

    例如:应用apk要获得皮肤apk中的bg.png, 
    Java代码  收藏代码
    1. Drawable drawable = context.getResources().getDrawable(R.drawable.bg);   


    这样就得到了图片的引用,其他xml资源文件的获取方式也是类似的。 

    自我实现的例子:/Files/lee0oo0/使用apk得到皮肤.rar 

    二.自定义扩展名的zip格式的皮肤 

    技术点在于如何去读取zip文件中的资源以及皮肤文件存放策略。 

    方案:如果软件每次启动都去读取SD卡上的皮肤文件,速度会比较慢。较好的做法是提供一个皮肤设置的界面,用户选择了哪一个皮肤,就把那个皮肤文件解压缩到”/data/data/[package name]/skin”路径下,这样不需要跨存储器读取,速度较快,而且不需要每次都去zip压缩包中读取,不依赖SD卡中的文件,即使皮肤压缩包文件被删除了也没有关系。 

    实现方法: 

    1. 在软件的帮助或者官网的帮助中提示用户将皮肤文件拷贝到SD卡指定路径下。 

    2. 在软件中提供皮肤设置界面。可以在菜单或者在设置中。可参考墨迹、搜狗输入法、QQ等支持换肤的软件。 

    3. 加载指定路径下的皮肤文件,读取其中的缩略图,在皮肤设置界面中显示,将用户选中的皮肤文件解压缩到”/data/data/[package name]/skin”路径下。 

    4. 软件中优先读取”/data/data/[package name]/skin/”路径下的资源。如果没有则使用apk中的资源。 

    --------------------------------------------------------------------------------- 
    http://gundumw100.iteye.com/blog/1052260 


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

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

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

    Java代码  收藏代码
    1. package org.leepood.skindemo;  
    2.   
    3. import java.util.ArrayList;  
    4. import java.util.List;  
    5.   
    6. import android.app.Activity;  
    7. import android.app.ProgressDialog;  
    8. import android.content.Context;  
    9. import android.content.Intent;  
    10. import android.content.SharedPreferences;  
    11. import android.content.pm.PackageInfo;  
    12. import android.content.pm.PackageManager;  
    13. import android.content.pm.PackageManager.NameNotFoundException;  
    14. import android.content.res.Resources;  
    15. import android.os.Bundle;  
    16. import android.os.Handler;  
    17. import android.os.Message;  
    18. import android.view.ContextMenu;  
    19. import android.view.LayoutInflater;  
    20. import android.view.MenuItem;  
    21. import android.view.View;  
    22. import android.view.ViewGroup;  
    23. import android.view.ContextMenu.ContextMenuInfo;  
    24. import android.view.View.OnCreateContextMenuListener;  
    25. import android.widget.BaseAdapter;  
    26. import android.widget.ImageView;  
    27. import android.widget.ListView;  
    28. import android.widget.TextView;  
    29. import android.widget.Toast;  
    30. import android.widget.AdapterView.AdapterContextMenuInfo;  
    31.   
    32. public class Main extends Activity implements SharedPreferences.OnSharedPreferenceChangeListener{  
    33.   
    34.     private ListView listview;  
    35.     private Context c;  
    36.     private Handler mHandler;  
    37.     private ProgressDialog pDialog;  
    38.     private SkinAdapter adapter;  
    39.     private SharedPreferences sp;  
    40.     static final int MESSAGE_SEARCHED_SKIN=0;  
    41.     static final int MESSAGE_SEARCHING_SKIN=MESSAGE_SEARCHED_SKIN+1;  
    42.     static final int MESSAGE_SEARCHED_SKIN_FOR_NONTHING=MESSAGE_SEARCHING_SKIN+1;  
    43.   
    44.     @Override  
    45.     protected void onCreate(Bundle savedInstanceState) {  
    46.   
    47.         super.onCreate(savedInstanceState);  
    48.         setContentView(R.layout.main);  
    49.         init();  
    50.         pDialog.show();  
    51.         new Thread(serachSkin).start();  
    52.   
    53.     }  
    54.   
    55.     private void init()  
    56.     {  
    57.   
    58.         c=this;  
    59.         mHandler=new Handler(){  
    60.   
    61.             @Override  
    62.             public void handleMessage(Message msg) {  
    63.                 switch(msg.what)  
    64.                 {  
    65.                 case MESSAGE_SEARCHED_SKIN:  
    66.                     ArrayList  
    67.  skins=(ArrayList  
    68. ) msg.obj;//获取skins  
    69.                     adapter=new SkinAdapter(c, skins);  
    70.                     listview.setAdapter(adapter);  
    71.                     Toast.makeText(c, "查找到已经安装的皮肤", 1).show();  
    72.                     pDialog.dismiss();  
    73.                     break;  
    74.                 case MESSAGE_SEARCHED_SKIN_FOR_NONTHING:  
    75.                     Toast.makeText(c, "未查找到任何皮肤", 1).show();  
    76.                     pDialog.dismiss();  
    77.                 }  
    78.             }  
    79.   
    80.         };  
    81.         sp=this.getSharedPreferences("config",Context.MODE_WORLD_WRITEABLE);  
    82.         sp.registerOnSharedPreferenceChangeListener(this);  
    83.   
    84.         listview=(ListView) findViewById(R.id.list);  
    85.         listview.setItemsCanFocus(false);  
    86.         listview.setChoiceMode(ListView.CHOICE_MODE_SINGLE);  
    87.   
    88.         pDialog=new ProgressDialog(this);  
    89.         pDialog.setMessage("正在查找已经安装的皮肤");  
    90.   
    91.         listview.setOnCreateContextMenuListener(new OnCreateContextMenuListener() {  
    92.   
    93.             public void onCreateContextMenu(ContextMenu menu, View v,  
    94.                     ContextMenuInfo menuInfo) {  
    95.                 menu.add("使用该主题");  
    96.   
    97.             }  
    98.         });  
    99.   
    100.     }  
    101.   
    102.     private Runnable serachSkin =new Runnable(){  
    103.   
    104.         public void run() {  
    105.             PackageManager manager=c.getPackageManager();  
    106.             List  
    107.  packages=manager.getInstalledPackages(PackageManager.PERMISSION_GRANTED);  
    108.   
    109.             ArrayList  
    110.  skins=new ArrayList  
    111. ();  
    112.             for(PackageInfo info:packages)  
    113.             {  
    114.                 //System.out.println(info.packageName);  
    115.                 if(info.packageName.startsWith("org.leepood.skin."))  
    116.                 {  
    117.                     skins.add(info);  
    118.                 }  
    119.             }  
    120.             if(skins.size()>0)  
    121.             {  
    122.                 Message msg=mHandler.obtainMessage();  
    123.                 msg.obj=skins;  
    124.                 msg.what=MESSAGE_SEARCHED_SKIN;  
    125.                 mHandler.sendMessage(msg);  
    126.             }  
    127.             else  
    128.             {  
    129.                 mHandler.sendEmptyMessage(MESSAGE_SEARCHED_SKIN_FOR_NONTHING);  
    130.             }  
    131.   
    132.         }  
    133.   
    134.     };  
    135.     private class SkinAdapter extends BaseAdapter  
    136.     {  
    137.   
    138.         LayoutInflater mInflater;  
    139.         ArrayList  
    140.  datas;  
    141.         PackageManager manager;  
    142.         public SkinAdapter(Context c,ArrayList  
    143.  datas)  
    144.         {  
    145.   
    146.             this.datas=datas;  
    147.              mInflater=LayoutInflater.from(c);  
    148.              manager=c.getPackageManager();  
    149.         }  
    150.   
    151.         public int getCount() {  
    152.   
    153.             return datas.size();  
    154.         }  
    155.   
    156.         public Object getItem(int position) {  
    157.   
    158.             return datas.get(position);  
    159.         }  
    160.   
    161.         public long getItemId(int position) {  
    162.   
    163.             return 0;  
    164.         }  
    165.   
    166.         public View getView(int position, View convertView, ViewGroup parent) {  
    167.   
    168.             if(convertView==null)  
    169.             {  
    170.                 convertView=mInflater.inflate(R.layout.skin_item, null);  
    171.             }  
    172.             ImageView icon=(ImageView) convertView.findViewById(R.id.skin_icon);  
    173.             TextView  skin_name=(TextView) convertView.findViewById(R.id.skin_name);  
    174.             PackageInfo info=datas.get(position);  
    175.             icon.setImageDrawable(info.applicationInfo.loadIcon(manager));  
    176.             skin_name.setText(info.applicationInfo.loadLabel(manager));  
    177.             return convertView;  
    178.         }  
    179.   
    180.     }  
    181.   
    182.     public void onThemeChanged(String newThemePackageName) {  
    183.         try {  
    184.   
    185.             Context themeContext=this.createPackageContext(newThemePackageName, CONTEXT_IGNORE_SECURITY);  
    186.             Resources res=themeContext.getResources();  
    187.             setControlsStyle(res);  
    188.   
    189.         } catch (NameNotFoundException e) {  
    190.   
    191.             e.printStackTrace();  
    192.         }  
    193.   
    194.     }  
    195.   
    196.     private void setControlsStyle(Resources res)  
    197.     {  
    198.         listview.setBackgroundColor(res.getColor(R.color.ListView_bg));  
    199.   
    200.     }  
    201.   
    202.     @Override  
    203.     public boolean onContextItemSelected(MenuItem item) {  
    204.         AdapterContextMenuInfo menuInfo=(AdapterContextMenuInfo)item.getMenuInfo();  
    205.         PackageInfo info=(PackageInfo) adapter.getItem(menuInfo.position);  
    206.   
    207.         sp.edit().putString("themePackage", info.packageName).commit();  
    208.         return super.onContextItemSelected(item);  
    209.     }  
    210.   
    211.     public void onSharedPreferenceChanged(SharedPreferences sharedPreferences,  
    212.             String key) {  
    213.         System.out.println("themeChange");  
    214.         onThemeChanged(sharedPreferences.getString(key, ""));  
    215.   
    216.     }  
    217.   
    218. }  


    这段代码的含义就是去查找系统中安装的包名,若以org.leepood.skin.开头则说明该包为主题包,将其加入listview中显示出来。代码中使用了多线程避免时间过长堵塞UI。程序将当前主题配置保存在SharedPreference中,为SharedPreference注册了一个监听函数,当其值发生改变时自动调用新的样式。当然,这只是个demo而已,一开始加载Activity没有去读取主题,这个可以由大家自己去实现。 

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

    Java代码  收藏代码
    1. public interface OnThemeChangedListener   
    2. {   
    3. public void onChanged(String newThemePackageName);   
    4.   
    5. }   

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

    Java代码  收藏代码
    1. public void setOnThemeChangedListener(OnThemeChangedListener listener)   
    2. {   
    3. this.listener=listener;   
    4. }   

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

    Java代码  收藏代码
    1. public void onSharedPreferenceChanged(SharedPreferences sharedPreferences,   
    2. String key) {   
    3. if(key.equals("themePackage"))   
    4. {   
    5. listener.onChanged(sp.getString("themePackage", ""));   
    6. }   
    7.   
    8. }   

    接着在继承于ThemeActivity的子类里面首先是setOnThemeChangedListener.接着用一个匿名内部类搞定。
  • 相关阅读:
    Clojure实现的简单短网址服务(Compojure、Ring、Korma库演示样例)
    android4.4系统解决“ERRORcouldn't find native method”方法
    JS window.open()属性
    网页视频播放器代码大全 + 21个为您的站点和博客提供的免费视频播放器
    理解Java的GC日志
    图像识别技术
    堆排序原理及算法实现(最大堆)
    什么是依赖注入
    Cocos2d-x3.1下实现相似iOS页面滑动指示圆点
    [Bootstrap] 6. Navigation
  • 原文地址:https://www.cnblogs.com/lee0oo0/p/2767528.html
Copyright © 2011-2022 走看看