`
lovehong0306
  • 浏览: 12188 次
  • 性别: Icon_minigender_1
  • 来自: 石家庄
社区版块
存档分类
最新评论

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

阅读更多

文章出处:http://gundumw100.iteye.com/blog/1052260

 

一直很想弄清楚好多应用中是如何实现换皮肤这项功能的,花了下午点时间,查了下资料也实现了个切换主题的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.就行。 

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

 

 

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.接着用一个匿名内部类搞定。好啦,代码可以见附件啦 

分享到:
评论

相关推荐

    android theme 皮肤主题 的应用

    总结,Android主题和皮肤主题的应用涉及资源文件的管理、XML样式定义、代码中主题的设置和切换、第三方库的利用,以及动态主题切换和皮肤包的打包分发。理解并掌握这些知识点,将有助于开发出更加美观、个性化的...

    Android中英文切换

    - 对于Android应用开发者,需要在应用程序清单文件(AndroidManifest.xml)中指定支持的语言资源。通过创建不同语言的资源文件夹(如 values-zh-rCN 对应简体中文,values-en 对应英文),可以实现应用的多语言支持...

    Android 应用语言切换

    在Android应用开发中,实现语言切换是一项常见的需求,它允许用户根据自己的偏好选择应用程序的显示语言。本教程将深入探讨如何在Android应用中实现实现中英文语言切换,以提高用户体验。 首先,我们需要理解...

    在Android应用中添加修改主题功能

    总结,实现Android应用中的主题切换功能,需要理解主题的工作原理,掌握在AndroidManifest.xml和Activity中设置主题的方法,以及如何自定义和保存主题选择。通过合理的设计和实现,可以为用户提供更加个性化和舒适的...

    Android 主题切换,切换字体颜色颜色,背景色,图片引用文件等

    在Android应用开发中,主题(Theme)是一种强大的工具,它允许开发者通过全局的方式改变应用程序的外观和风格。主题切换不仅可以改变字体颜色、背景色,还可以调整图片引用以及其他UI元素的样式,为用户提供个性化的...

    android切换主题一(apk方式)

    在Android平台上,为应用切换主题是一项常见的需求,可以提供个性化的用户体验。本篇文章将深入探讨如何通过apk皮肤的方式来实现这一功能。首先,我们要理解apk皮肤的本质:它实际上是一种可替换的应用资源包,包含...

    Android应用皮肤切换Demo

    本Demo主要展示了如何在Android应用中实现皮肤切换功能,以提供更丰富的用户体验。以下将详细介绍Android应用皮肤切换的基本原理、实现步骤以及相关知识点。 1. **皮肤资源** - 在Android中,皮肤主要通过改变颜色...

    android实现离开应用程序

    在Android应用开发中,"离开应用程序"通常指的是提供一个机制让用户能够完全关闭或退出应用程序。这个功能虽然在现代Android系统中并不强制要求,因为系统会自动管理应用的生命周期,但有时候用户可能还是希望有一个...

    Android-Android主题换肤插件化换肤无缝切换

    "Android-Android主题换肤插件化换肤无缝切换"是一个专为Android应用程序设计的技术主题,它旨在实现动态的、插件化的皮肤更换,使得用户在不重启应用的情况下能够实现皮肤的平滑切换。以下是对这个主题的详细解释和...

    Android从一个应用跳转到另一个应用

    本教程将详细讲解如何在Android应用中实现从一个应用跳转到另一个应用。 首先,我们需要了解Android中的Intent机制。Intent在Android系统中扮演着消息传递的角色,它用于启动活动(Activity)、服务(Service)或...

    Android-Android日夜间模式切换

    在Android应用开发中,日夜间模式切换是一项重要的用户体验优化功能,它允许用户根据环境光线条件或个人偏好在明亮(日间)和暗淡(夜间)的主题之间自由切换。这一功能通常通过更改应用的颜色方案、字体颜色和背景...

    88个Android应用程序

    这个示例可能展示了如何在Android应用中实现网格视图(GridView)和图片切换器(ImageSwitcher)的结合。GridView用于展示大量可滚动的项目,常用于应用的主菜单或图库。ImageSwitcher则是一个可以平滑过渡显示两张...

    Android应用开发试题.pdf

    4. Intent 的认识:Intent 是 Android 应用程序之间的一种通信机制,负责实现应用程序之间的数据共享和界面间的切换。Intent 可以分为两种:显示 Intent 和隐式 Intent。 5. Android 项目工程结构的认识:Android ...

    Android设交应用常见UI实现

    本文将深入探讨Android社交应用中的常见UI实现,帮助新手理解并掌握相关知识。 首先,我们要了解Android UI设计的基本原则。Android系统遵循Material Design设计规范,它强调清晰、层次感和动态反馈,为用户提供...

    Android 主题换肤框架

    在Android应用开发中,主题换肤功能已经成为提升用户体验和个性化设置的重要组成部分。"Android 主题换肤框架"是一个专门用于实现这一功能的工具,它允许用户动态改变应用程序的主题颜色和样式,无需重新编译或安装...

    Android 轻松打造一个APP动态切换桌面图标Launcher

    1. **理解Android启动器(Launcher)**:Android启动器是系统中的一个核心组件,它负责显示应用程序的快捷方式和主屏幕小部件。开发者可以创建自定义启动器,以满足个性化需求,如动态图标切换。 2. **...

    android theme 皮肤主题 的应用 2

    这篇博文“android theme 皮肤主题的应用 2”可能是关于如何在Android应用中实现和应用自定义主题的深入探讨。尽管没有具体的描述,但我们可以从标签“源码”和“工具”推测,内容可能涵盖了如何通过源代码修改和...

    Android应用源码之更换软件主题(apk方式).zip

    这个压缩包"Android应用源码之更换软件主题(apk方式).zip"显然包含了关于如何在Android应用中实现主题更换的源代码示例,特别是通过apk方式来实现。让我们深入探讨这个主题。 首先,了解主题在Android中的概念。...

    Android实现单击按钮实现屏幕切换

    在Android开发中,屏幕切换是应用交互的基本操作之一,它涉及到Activity的...通过以上步骤,开发者可以创建一个简单的Android应用,使用户通过点击按钮在不同的Activity之间进行切换,了解Android应用的基础交互机制。

    《Android移动应用基础教程》(Android Studio)(第二版)源代码.zip

    在Android应用开发中,Android Studio是官方推荐的集成开发环境(IDE),它提供了丰富的工具和功能,如代码编辑器、调试器、构建工具等,以简化开发过程。通过这个源代码包,读者可以深入理解Android应用的构建过程...

Global site tag (gtag.js) - Google Analytics