分析Android App中内置换肤功能的实现方式

Android平台api没有特意为换肤提供一套简便的机制,这可能是外国的软件更注重功能和易用,不流行换肤。系统不提供直接支持,只能自行研究。
换肤,可以认为是动态替换资源(文字、颜色、字体大小、图片、布局文件……)。这个使用编程语言来动态设置是可以做到的,例如使用View的setBackgroundResource、setTextSize、setTextColor等函数。但我们不可能在每个activity里对页面里的所有控件都通过调用这些函数来换肤,这样的程序代码难以维护、扩展,也违背了UI和代码分离的原则(android开发中UI以xml文件的方式布局)。 通常,皮肤资源会在主程序apk之外提供,以减少主程序的大小,以及方便随时提供新的皮肤扩展。
简单的来说,软件皮肤包括图标、字体、布局、交互风格等,换肤就是换掉皮肤包括的部分或所有资源。

主流应用程序换肤方式:
国内有很多的软件都支持皮肤定制,这也是与国外软件重大不同之一,国外用户注重社交、邮件等功能,国内用户则重视音乐、小说、皮肤等功能. 在写这个换肤系列之前,我也参考了其他人的一些总结. 知识点比较散,因此我对其进行了整理和进一步的优化和扩展. 当然,由于精力有限,部分换肤方式我也只写了主体功能的实现. 如果出现有误或者不够详细的地方,希望大家提出意见或者自行进行扩展.
关于其中提到的几种主流实现方式,接下来的文章里我会具体提供代码进行解释, 此次先做一个整体的概述.
目前主流的换肤从功能上可以划分几种实现方式,

1) 软件内置多个皮肤,不可由用户增加或修改:
最低的自由度,软件实现相对于后面的几种相对简单.
如果你的程序和资源都很小,可以在主程序apk中放入足够的皮肤资源.
典型应用:平板电脑Apad上QQ空间的换肤功能,实际上只是改变了Activity的背景,或这部分的资源.

2) 官方提供皮肤供下载,用户可以使用下载的皮肤:

用户可选择下载自己喜欢的皮肤,有些玩家会破解皮肤的定制方法,自己做皮肤使用,或者传到网上给大家用。
典型应用: 墨迹天气下载的皮肤就是一个zip格式的压缩包,在应用的时候把皮肤资源释放到墨迹天气应用的目录下,更换皮肤时新的皮肤资源会替换掉老的皮肤资源每次加载的时候就是从手机硬盘上读取图片,这些图片资源的命名和程序中的资源的命名保持一致,一旦找不到这些资源,可以选择到系统默认中查找。
这种实现是直接读取了外部资源文件,在程序运行时通过代码显示的替换界面的背景资源。
这种方式的优点是:皮肤资源的格式定义很随意可以是zip也可以是自定义的格式,只要程序中能够解析到资源就行,缺点是需要读取并解析文件,导致效率上会比较差.

3) 皮肤资源存在于主程序之类的APK中,即实现了APK的拆分. 这里类似于浏览器与插件的关系. 当考虑应用程序需要扩展时,则需要采用本方式实现.,这不仅仅体现在换肤的APK中.
典型应用: 手机QQ换肤的实现方式
Q的皮肤是一个无界面APK应用,这个皮肤应用中的资源和主程序的资源命名一致,通过主程序和皮肤程序共享进程实现主程序对皮肤程序中资源的访问,在程序运行时通过代码显示指定皮肤资源,缺点是在主程序中每个activity要增加复杂的使用哪种皮肤逻辑,优点是效率比较快,且使应用程序具有了良好的扩展性,降低了程序的耦合性. 包括其他类似的扩展功能,都可以利用此方式实现.

4) 官方提供皮肤制作工具或方法,用户可自制皮肤:
大致分为两种情况:
a. 应用程序主列表为一个GridView,用户可通过在设置中选择背景的颜色和按钮的风格. 直接进行替换即可.
b. 另外一种是有可视化带向导的工具。用户只要自己找一些图片、修改文字的字体替换就可以了。用户可以上传自制的皮肤,提供其他用户下载. 一般都是打包为.zip格式的,扩展名可由公司需求自定义. 例如墨迹天气皮肤扩展名是mja,搜狗输入法的皮肤扩展名是sga,它们的文件格式实际上都是zip。之后的应用就和第二种换肤方式类似了.
这种方式优点是:使用户有参与感,自由度较高。用户可根据自己的喜好定制软件的皮肤。
      
5) 改写SDK的Resource类:
此处提供一些思路:
在Android系统中,资源主要指图片和MP3类型的文件,也是用户UI包含的所有元素。谷歌在设计Android系统时,将UI界面和逻辑代码分开组建:界面设计通过XML的形式描述,具体的程序和应用逻辑则通过代码来实现;前端工程师只负责HTML和CSS的设计与架构,后端工程师则专门考虑JSP和Java的代码执行.
资源访问在Android性能架构中处于何种地位?在进行Android开发时,开发者经常用到Framework提供的资源包Framework.jar与Framework-res.apk,以及与核心资源相关的组件“Resource Manager”文件系统。
APK本身是一个简单的文件格式,也是一个压缩文件包。通过解压文件包可以释放APK文件:首先需要APK的原数据Meta INF、Manifest以及RES目录。一部分包含图片资源的应用,在资源释放时也会用到Layout。
在安装文件时,系统会将文件取出、解压后放在Dalvik Cache中。该缓存下有许多dex文件,当用户打开应用时系统会自动加载相应的类。在加载过程中,系统如需访问APK,则需对其进行解压,这样通常导致效率较为低下。而如果将dex文件放入Dalvik Cache中,则能够令加载的效率大大提升。
每个进程都有一份关于Framework的共享类和共享资源,但物理内存空间中的系统级别资源只有一份。Framework类和资源是只读的,而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);
 } 

}

皮肤管理器:

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();
  }
 } 

}

就这样,通过程序内置皮肤的基本功能完成了.若想在自己的应用中实现,仍需注意以下几点(实现起来并不复杂,此处不再写具体实现):
1.  实现多个activity的更换皮肤. 需要利用自定义MyApplication类,继承自Application. 并加入activity的集合属性.用于存储应用所有的activity。修改SkinManager,在更换皮肤时,从application中取出该集合,进行遍历并更换皮肤。
2.可以优化用户体验,通过导航栏方式进入更换皮肤界面,并可以加入预览功能,当确定修改配置后,才完成更换皮肤功能.
3. 加入style.theme等资源,实现更加复杂的皮肤更换. 具体实现同更换背景.

(0)

相关推荐

  • Android实现换肤的两种思路分析

    本文分析了Android实现换肤的两种思路.分享给大家供大家参考,具体如下: 这里来了解换肤实现及不同方案的差异和使用场合. 一.从功能上划分 1) 软件内置多个皮肤,用户不能修改: 2) 官方提供皮肤下载,用户使用下载的皮肤: 3) 官方提供皮肤制作工具或方法,用户自制皮肤. 二.皮肤定义 软件皮肤包括图标.字体.布局.交互风格等,换肤就是换掉皮肤包括的部分或所有资源. 三.皮肤与APP分离 1)打包皮肤文件 默认格式是apk.例如Launcher,它的桌面皮肤格式是一个apk: 自定义的格式

  • Android 换肤技术资料整理

    Android换肤技术总结 背景 纵观现在各种Android app,其换肤需求可以归为 - 白天/黑夜主题切换(或者别的名字,通常2套),如同花顺/自选股/天天动听等,UI表现为一个switcher. - 多种主题切换,通常为会员特权,如QQ/QQ空间. 对于第一种来说,目测应该是直接通过本地theme来做的,即所有图片/颜色的资源都在apk里面打包了. 而对于第二种,则相对复杂一些,由于作为一种线上服务,可能上架新皮肤,且那么多皮肤包放在apk里面实在太占体积了,所以皮肤资源会在选择后再进行

  • Android编程实现换肤功能实例

    本文实例讲述了Android编程实现换肤功能的方法.分享给大家供大家参考,具体如下: 本系列专题培训适用范围:初级Android程序员,即有J2SE基础和Android初级水平.J2SE基础是指掌握JAVA语法,1.5.1.6新增的语法不完全掌握也没关系.了解基本的面向对象思想.能编写简单的J2SE程序,掌握基本的调试方法,熟悉Swing更好.Android初级是指掌握Activity.Service.BroadcastReceiver.Intent.SQLite.UI组件的使用,能参照例子编写

  • Android应用开发中实现apk皮肤文件换肤的思路分析

    在android的项目开发中,都会遇到后期功能拓展增强与主程序代码变更的现实矛盾,也就是程序的灵活度. 由于linux平台的安全机制,再加上dalvik的特殊机制,各种权限壁垒,使得开发一个灵活多变的程序,变得比较困难,不像pc平台下那么容易. 这里实际上可以借鉴传统软件中扩展程序的方法: 也就是插件的实现. 如目前所有的浏览器,比如我们使用的eclipse,以及很多优秀的软件,都使用了此种方式. 这样轻松实现了软件的功能扩展,而升级功能时只用更新对应插件, 而不是需要更新整个应用,降低了程序的

  • android换肤功能 如何动态获取控件中背景图片的资源id?

    这个是在在做一个换肤功能时遇到的问题. 对于换肤,网上都有示例,可以从别的皮肤安装包中读取所要的资源,前提是你必须先持有这个资源的引用名称,像R.drawable.background(喂,这不是废话嘛).这个换肤的方案原理就是,自身应用的资源名称是R.drawable.background,那皮肤包中应该也是这个名称,然后通过这个名称获取该资源在皮肤包中的具体id,代码: //先获取本地资源引用名称,type name是R.drawable.background中的"drawable"

  • 分析Android App中内置换肤功能的实现方式

    Android平台api没有特意为换肤提供一套简便的机制,这可能是外国的软件更注重功能和易用,不流行换肤.系统不提供直接支持,只能自行研究. 换肤,可以认为是动态替换资源(文字.颜色.字体大小.图片.布局文件--).这个使用编程语言来动态设置是可以做到的,例如使用View的setBackgroundResource.setTextSize.setTextColor等函数.但我们不可能在每个activity里对页面里的所有控件都通过调用这些函数来换肤,这样的程序代码难以维护.扩展,也违背了UI和代

  • Android App中实现向右滑动销毁功能的要点解析

    今天给大家带来一个向右滑动销毁Activity的效果,Activtiy随着手指的移动而移动,该效果在Android应用中还是比较少见的,在IOS中就比较常见了,例如"网易新闻" ,"美食杰" , "淘宝"等应用采用此效果,而Android应用中"知乎"采用的也是这种滑动切换Activity的效果, 不过我发现"淘宝"并没有随着手势的移动而移动,只是捕捉到滑动手势,然后产生平滑切换界面的动画效果,这个在And

  • android使用SkinManager实现换肤功能的示例

    试着用鸿洋大神写的SkinManager实现了换肤功能. 一.配置 在app下build.gradle中添加依赖: //换肤功能 compile 'com.zhy:changeskin:4.0.2' 这样就配置好了,然后在程序入口进行初始化. 二.全局初始化 在自己创建的继承application的类中添加: //换肤sdk初始化 SkinManager.getInstance().init(this); 这个类肯定要在清单文件<application/>节点配置的. 接下来还需要注册. 三.

  • 如何通过Battery Historian分析Android APP耗电情况

    一.电量统计模块概述 耗电信息在设置 -> 电量中能够非常直观的看到.注意,Android 所有功耗统计都是通过代码估算,没有集成电路参与汇报.准确度取决于厂商 ROM 所提供的power_profile.xml文件.由于不同厂商power_profile.xml准确度及源码有差异,因此不同手机.不同版本的数据可能有较大差异. power_profile.xml直接影响统计的准确度,并且此文件无法通过应用修改.再次强调,Android 耗电估算没有硬件的参与,全靠代码估算. power_prof

  • 如何在Android App中接入微信支付

    本篇简单介绍Android App中接入微信支付,包括App内支付和扫码支付.分享+支付 pofei 微信支付 wechat 官方接入文档 App内支付 源码下载 主要流程: 1.微信支付平台注册账号​ 注:注册并申请成功以后,需要在API安全中设置你的API密钥 32个字符.建议使用 MD5加密 ,并且需要妥善的保存.因为无法查看. 2.生成预支付订单 3.生成签名参数 4.调起微信,完成支付 扫码支付 扫码支付使用的是微信统一下单API ,使用的是模式二,模式一 一直说URL参数错误,完全按

  • Android标题栏中添加返回按钮功能

    标题栏中的返回按钮在实际使用中用的比较多,今天就来讲讲我在项目开发中的使用经历,话不多说,还是直接上源码,上源码是最给力的. 一. 编写自定义类 public class CustomTitle { private static Activity mActivity; public static void getCustomTitle(Activity activity, String title) { mActivity = activity; mActivity.requestWindowF

  • Android开发中应用程序分享功能实例

    本文实例讲述了Android开发中应用程序分享功能.分享给大家供大家参考,具体如下: Intent shareIntent = new Intent(); shareIntent.setAction(Intent.ACTION_SEND); //设置类型 shareIntent.setType("text/plain"); //设置分享的主题 shareIntent.putExtra("android.intent.extra.SUBJECT", "分享&

  • python实现的批量分析xml标签中各个类别个数功能示例

    本文实例讲述了python实现的批量分析xml标签中各个类别个数功能.分享给大家供大家参考,具体如下: 文章目录 需要个脚本分析下各个目标的数目 顺带练习下多进程,自用,直接上代码: # -*- coding: utf-8 -*- # @Time : 2019/06/10 18:56 # @Author : TuanZhangSama import os import xml.etree.ElementTree as ET from multiprocessing import Pool,fre

  • Android App中进行语言的切换

    本篇简单介绍将在Android App中进行语言的切换和使用dragonFace改系统语言. 切换语言 首先需要在res 中创建个若干个不同的value文件夹(例如:values.values-en.value-ja).然后将不同的String.xml文件. 这里为 中.英.日三语切换.(value文件夹命名可以参考下面) 在res目錄下建立不同名稱的values文件來調用不同的語言包 Values文件匯總如下: 中文(中國):values-zh-rCN中文(台灣):values-zh-rTW

  • Android平台中实现数据存储的5种方式

    本文介绍Android中的5种数据存储方式,具体内容如下 数据存储在开发中是使用最频繁的,在这里主要介绍Android平台中实现数据存储的5种方式,分别是: 1 使用SharedPreferences存储数据 2 文件存储数据 3 SQLite数据库存储数据 4 使用ContentProvider存储数据 5 网络存储数据 下面将为大家一一详细介绍.  第一种:使用SharedPreferences存储数据 SharedPreferences是Android平台上一个轻量级的存储类,主要是保存一

随机推荐