Android 打造自己的个性化应用(二):应用程序内置资源实现换肤功能
通过应用程序内置资源实现换肤,典型的应用为QQ空间中换肤的实现. 应用场景为: 应用一般不大,且页面较少,风格相对简单,一般只用实现部分资源或者只用实现背景的更换.
此种换肤方式实现的思路:
1. 把几套皮肤放在res/drawable目录里,然后用SharedPreferences来记录当前皮肤的资源id.然后在程序启动时加载Activity背景。
2. 主要的实现在皮肤管理器SkinManager类中. 将皮肤资源的ID加入集合中. 由该类同一调度皮肤更换,如初始化皮肤,获取当前皮肤符号以及具体的对应资源的更换皮肤.
接下来看一下效果图:


内置皮肤的实现相对比较简单,下面直接上代码:
AndroidMainfest.xml
- <?xml version="1.0" encoding="utf-8"?>
- <manifest xmlns:android="http://schemas.android.com/apk/res/android"
- package="com.tony.skindemo"
- android:versionCode="1"
- android:versionName="1.0" >
- <uses-sdk android:minSdkVersion="8" />
- <application
- android:icon="@drawable/ic_launcher"
- android:label="@string/app_name" >
- <activity
- android:label="@string/app_name"
- android:name="com.tony.skindemo.SkinDemoActivity" >
- <intent-filter >
- <action android:name="android.intent.action.MAIN" />
- <category android:name="android.intent.category.LAUNCHER" />
- </intent-filter>
- </activity>
- </application>
- </manifest>
布局文件:
main.xml
- <?xml version="1.0" encoding="utf-8"?>
- <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
- android:layout_width="fill_parent"
- android:layout_height="fill_parent"
- android:orientation="vertical" >
- <TextView
- android:textColor="#ff00ff"
- android:text="程序皮肤更换"
- android:layout_width="fill_parent"
- android:layout_height="wrap_content" />
- <RadioGroup
- android:id="@+id/skin_options"
- android:layout_width="fill_parent"
- android:layout_height="wrap_content"
- >
- <RadioButton
- android:layout_weight="1"
- android:id="@+id/radioButton1"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:text="风格1" />
- <RadioButton
- android:layout_weight="1"
- android:id="@+id/radioButton2"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:text="风格2" />
- <RadioButton
- android:layout_weight="1"
- android:id="@+id/radioButton3"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:text="风格3" />
- <RadioButton
- android:layout_weight="1"
- android:id="@+id/radioButton4"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:text="风格4" />
- <RadioButton
- android:layout_weight="1"
- android:id="@+id/radioButton5"
- android:layout_width="wrap_content"
- android:layout_height="wrap_content"
- android:text="风格5" />
- </RadioGroup>
- </LinearLayout>
程序主Activity
- package com.tony.skindemo;
- import android.app.Activity;
- import android.os.Bundle;
- import android.view.MotionEvent;
- import android.view.Window;
- import android.view.WindowManager;
- import android.widget.RadioButton;
- import android.widget.RadioGroup;
- import android.widget.RadioGroup.OnCheckedChangeListener;
- public class SkinDemoActivity extends Activity {
- private SkinSettingManager mSettingManager;
- private RadioButton radioButton1;
- private RadioButton radioButton2;
- private RadioButton radioButton3;
- private RadioButton radioButton4;
- private RadioButton radioButton5;
- @Override
- public void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- // 取消标题栏
- requestWindowFeature(Window.FEATURE_NO_TITLE);
- // 完成窗体的全屏显示 // 取消掉状态栏
- getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,
- WindowManager.LayoutParams.FLAG_FULLSCREEN);
- setContentView(R.layout.main);
- // 初始化皮肤
- mSettingManager = new SkinSettingManager(this);
- mSettingManager.initSkins();
- //通过单选按钮设置皮肤(可自定义更换的方式,如导航栏,也可以加上预览功能,此处不再实现)
- radioButton1 = (RadioButton) findViewById(R.id.radioButton1);
- radioButton2 = (RadioButton) findViewById(R.id.radioButton2);
- radioButton3 = (RadioButton) findViewById(R.id.radioButton3);
- radioButton4 = (RadioButton) findViewById(R.id.radioButton4);
- radioButton5 = (RadioButton) findViewById(R.id.radioButton5);
- RadioGroup radioGroup = (RadioGroup) findViewById(R.id.skin_options);
- radioGroup.setOnCheckedChangeListener(new OnCheckedChangeListener() {
- @Override
- public void onCheckedChanged(RadioGroup group, int checkedId) {
- switch (checkedId) {
- case R.id.radioButton1:
- mSettingManager.changeSkin(1);
- break;
- case R.id.radioButton2:
- mSettingManager.changeSkin(2);
- break;
- case R.id.radioButton3:
- mSettingManager.changeSkin(3);
- break;
- case R.id.radioButton4:
- mSettingManager.changeSkin(4);
- break;
- case R.id.radioButton5:
- mSettingManager.changeSkin(5);
- break;
- default:
- break;
- }
- }
- });
- }
- // 这里为了简单实现,实现换肤
- public boolean onTouchEvent(MotionEvent event) {
- mSettingManager.toggleSkins();
- return super.onTouchEvent(event);
- }
- }
- <pre name="code" class="java"></pre><pre></pre><p></p><pre></pre>皮肤管理器:<p></p><p></p><pre name="code" class="java">package com.tony.skindemo;
- import android.app.Activity;
- import android.content.SharedPreferences;
- /**
- * 皮肤管理器
- * @author tony
- *
- */
- public class SkinSettingManager {
- public final static String SKIN_PREF = "skinSetting";
- public SharedPreferences skinSettingPreference;
- private int[] skinResources = { R.drawable.default_wallpaper,
- R.drawable.wallpaper_c,R.drawable.wallpaper_d,R.drawable.wallpaper_f,
- R.drawable.wallpaper_g
- };
- private Activity mActivity;
- public SkinSettingManager(Activity activity) {
- this.mActivity = activity;
- skinSettingPreference = mActivity.getSharedPreferences(SKIN_PREF, 3);
- }
- /**
- * 获取当前程序的皮肤序号
- *
- * @return
- */
- public int getSkinType() {
- String key = "skin_type";
- return skinSettingPreference.getInt(key, 0);
- }
- /**
- * 把皮肤序号写到全局设置里去
- *
- * @param j
- */
- public void setSkinType(int j) {
- SharedPreferences.Editor editor = skinSettingPreference.edit();
- String key = "skin_type";
- editor.putInt(key, j);
- editor.commit();
- }
- /**
- * 获取当前皮肤的背景图资源id
- *
- * @return
- */
- public int getCurrentSkinRes() {
- int skinLen = skinResources.length;
- int getSkinLen = getSkinType();
- if(getSkinLen >= skinLen){
- getSkinLen = 0;
- }
- return skinResources[getSkinLen];
- }
- public void toggleSkins(){
- int skinType = getSkinType();
- if(skinType == skinResources.length - 1){
- skinType = 0;
- }else{
- skinType ++;
- }
- setSkinType(skinType);
- mActivity.getWindow().setBackgroundDrawable(null);
- try {
- mActivity.getWindow().setBackgroundDrawableResource(getCurrentSkinRes());
- } catch (Throwable e) {
- e.printStackTrace();
- }
- }
- /**
- * 用于初始化皮肤
- */
- public void initSkins(){
- mActivity.getWindow().setBackgroundDrawableResource(getCurrentSkinRes());
- }
- /**
- * 随即切换一个背景皮肤
- */
- public void changeSkin(int id) {
- setSkinType(id);
- mActivity.getWindow().setBackgroundDrawable(null);
- try {
- mActivity.getWindow().setBackgroundDrawableResource(getCurrentSkinRes());
- } catch (Throwable e) {
- e.printStackTrace();
- }
- }
- }
- </pre><br><p></p><p>就这样,通过程序内置皮肤的基本功能完成了.</p><p>若想在自己的应用中实现,仍需注意以下几点(实现起来并不复杂,此处不再写具体实现):<br></p><p>1. 实现多个activity的更换皮肤. 需要利用自定义MyApplication类,继承自Application. 并加入activity的集合属性.用于存储应用所有的activity<br></p><p> 修改SkinManager,在更换皮肤时,从application中取出该集合,进行遍历并更换皮肤</p><p><br></p><p>2. 可以优化用户体验,通过导航栏方式进入更换皮肤界面,并可以加入预览功能,当确定修改配置后,才完成更换皮肤功能.</p><p>3. 加入style.theme等资源,实现更加复杂的皮肤更换. 具体实现同更换背景.<br></p><p><br></p><p><br></p><p><br></p><p><br></p>
Android 打造自己的个性化应用(二):应用程序内置资源实现换肤功能的更多相关文章
- 一种简单的实现:Android一键换肤功能
现在的APP开发,通常会提供APP的换肤功能,网上流传的换肤代码和实现手段过于复杂,我把原作者的代码重新整理抽取出来,转换成Eclipse项目,重新整理成正确.可直接运行的项目. 代码运行结果如图. ...
- Android一键换肤功能:一种简单的实现
Android一键换肤功能:一种简单的实现 现在的APP开发,通常会提供APP的换肤功能,网上流传的换肤代码和实现手段过于复杂,这里有一个开源实现,我找了一大堆,发现这个项目相对较为简洁:htt ...
- Android 打造自己的个性化应用(一):应用程序换肤主流方式的分析与概述
Android平台api没有特意为换肤提供一套简便的机制,这可能是外国的软件更注重功能和易用,不流行换肤.系统不提供直接支持,只能自行研究. 换肤,可以认为是动态替换资源(文字.颜色.字体大小.图片. ...
- Android 换肤功能的实现(Apk插件方式)
一.概述 由于Android 没有提供一套统一的换肤机制,我猜可能是因为国外更注重功能和体验的原因 所以国内如果要做一个漂亮的换肤方案,需要自己去实现. 目前换肤的方法大概有三种方案: (1)把皮肤资 ...
- Unit05: JavaScript对象概述 、 常用内置对象一 、 常用内置对象二 、 常用内置对象三
Unit05: JavaScript对象概述 . 常用内置对象一 . 常用内置对象二 . 常用内置对象三 常用内置对象使用演示: <!DOCTYPE html> <html> ...
- Flex AIR应用换肤功能(Android和IOS)
说明 换肤功能,即将整个应用的皮肤都进行更换,其实质,是动态加载swf文件的过程,而这些swf文件则有css文件编译而来. 关于换肤功能,在android和ios系统的实现方式是不同的.主要原因,是因 ...
- Android QMUI实战:实现APP换肤功能,并自动适配手机深色模式
Android换肤功能已不是什么新鲜事了,市面上有很多第三方的换肤库和实现方案. 之所以选择腾讯的QMUI库来演示APP的换肤功能,主要原因: 1.换肤功能的实现过程较简单.容易理解: 2.能轻松适配 ...
- Android 打造自己的个性化应用(四):仿墨迹天气实现-->自定义扩展名的zip格式的皮肤
在这里谈一下墨迹天气的换肤实现方式,不过首先声明我只是通过反编译以及参考了一些网上其他资料的方式推测出的换肤原理, 在这里只供参考. 若大家有更好的方式, 欢迎交流. 墨迹天气下载的皮肤就是一个zip ...
- Android 打造自己的个性化应用(三):应用程序的插件化
在android的项目开发中,都会遇到后期功能拓展增强与主程序代码变更的现实矛盾,也就是程序的灵活度. 由于linux平台的安全机制,再加上dalvik的特殊机制,各种权限壁垒,使得开发一个灵活多变的 ...
随机推荐
- Gstreamer 中的playback插件
1. PLAYBACK插件基本介绍 在早期的版本中同时存在playbin和playbin2,但是在最新的版本中,playbin2已经稳定,取代了playbin, playbin不再进行维护.下面是官网 ...
- CSS基础知识笔记(一)
css 样式由选择符和声明组成,而声明又由属性和值组成: 选择符: 又称选择器,指明网页中要应用样式规则的元素,如本例中是网页中所有的段(p)的文字将变成蓝色,而其他的元素(如ol)不会受到影响. 声 ...
- css3标签
-moz代表firefox浏览器私有属性 -ms代表ie浏览器私有属性 -webkit代表chrome.safari私有属性 -o代表opera私有属性 border-radius:2em; 向div ...
- VS2015 新Web项目(C#6)出现CS1617异常的解决
VS2015 新Web项目(C#6)出现CS1617错误的解决 VS2015新增了对C#6的支持. 在新的Web项目模板中通过引入nuget包Microsoft.CodeDom.Providers.D ...
- Qt sqlite
原文地址:http://blog.csdn.net/NewBee520/article/details/8247236QSqlDatabase类实现了数据库连接的操作 QSqlQuery类用来执行SQ ...
- T-SQL语句——UNION, EXCEPT, INTERSECT
UNION,EXCEPT, INTERSECT关键字用于对集合的查询,它们的作用分别为: UNION:合并两个或多个 SELECT 语句的结果集,并把重复结果去除: UNIONALL:合并两个或多个 ...
- CSS常见BUG
CSS Hack IE条件注释: 所有IE:<!--[if IE]> css code <![endif]--> IE6以上:<!--[if gt IE 6]> c ...
- 如何在异步请求时设置RequestHeader
一.为何要用到setRequestHeader 通常在HTTP协议里,客户端像服务器取得某个网页的时候,必须发送一个HTTP协议的头文件,告诉服务器客户端要下载什么信息以及相关的参数.而 XMLHTT ...
- HDU 3516 Tree Construction (四边形不等式)
题意:给定一些点(xi,yi)(xj,yj)满足:i<j,xi<xj,yi>yj.用下面的连起来,使得所有边的长度最小? 思路:考虑用区间表示,f[i][j]表示将i到j的点连起来的 ...
- 《Programming WPF》翻译 第7章 6.视频和3-D
原文:<Programming WPF>翻译 第7章 6.视频和3-D 虽然详细地讨论视频和3-D超越了这本书的范围,但是获得这些特征的支持是值得的. 视频由MediaElement类型支 ...