这个是在在做一个换肤功能时遇到的问题。

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

//先获取本地资源引用名称,type name是R.drawable.background中的"drawable",entry name是"background"
String resTypeName = getContext().getResources().getResourceTypeName(id);
String resEntryName = getContext().getResources().getResourceEntryName(id);
//然后创建皮肤包所在应用的Context
Context apk = getContext().createPackageContext(packageName,
Context.CONTEXT_IGNORE_SECURITY)
//然后就是获取皮肤包中的资源id了
int drawavleId = apk.getResources().getIdentifier(resEntryName, resTypeName,
apk.getPackageName());

这个换肤方案中,每个Activity在切换皮肤时,需要遍历整个layout,判断控件如果id中包含“skin”的字符,意味这个控件是需要换肤的控件,这个控件的id可以先保存下来。

遍历视图的代码

private List skinViewList = new ArrayList (); private void scanViewGroup(ViewGroup group, List skinViewList, Resources res) {
//first we need check if this ViewGroup have a background
if(group.getId() != View.NO_ID
&& res.getResourceEntryName(group.getId()).contains(SKIN_PATTERN)
&& !skinViewList.contains(group)) {
skinViewList.add(group.getId());
}
//second check its child view
View child;
for(int i = 0; i < group.getChildCount(); i++) {
child = group.getChildAt(i);
if(child instanceof ViewGroup) {
scanViewGroup((ViewGroup)child, skinViewList, res);
} else if(child.getId() == View.NO_ID) {
return;
} else {
int viewId = child.getId();
String entryName = res.getResourceEntryName(viewId);
Log("scanViewGroup(), entryName of this childView : " + entryName);
if(entryName.contains(SKIN_PATTERN) && !skinViewList.contains(child))
skinViewList.add(child.getId());
}
}
}

问题来了,本地应用中,你持有一个控件,比如Button,它的id可以直接调用button.getId()方法获取,但是它的背景图片background呢,我们可以调用button.getBackground()方法获取其对象,但是却没有方法可以获取这个资源图片的引用名称,也就无法得到它的具体id了。后面想到的方案就是,在每次Activity初始化的时候,我们事先遍历每一个控件的属性集AttributeSet,有需要换肤的控件,将其android:background这个属性的值保存下来,为此,需要重载Activity的onCreateView(String name, Context context, AttributeSet attrs)方法,这个方法我的理解是在Activity中每个控件(包括LinearLayout、TextView、Button等等)初始化前会调用,我也打了log,进行了验证,其中attrs参数就是该控件的属性集,这就是我们需要的,代码:

//先判断前面扫描的skinViewList是否为空,不为空意味着有控件需要换肤
if(skinViewList != null && skinViewList.size() > 0) {
int viewId = -1, backgroundId = -1;
for(int i = 0; i < attrs.getAttributeCount(); i++) {
if(attrs.getAttributeName(i).equals("id")) {
viewId = attrs.getAttributeResourceValue(i, -1);
}
if(attrs.getAttributeName(i).equals("background")) {
backgroundId = attrs.getAttributeResourceValue(i, -1);
}
}
//check if background drawable need save
if(viewId != -1 && backgroundId != -1 &&
drawableIdList != null && !drawableIdList.containsKey(viewId)) {
drawableIdList.put(viewId, backgroundId);
Log("add to drawableIdList, viewId = " + viewId
+ ", backgroundId = " + backgroundId);
}
}

有了这个backgroundId,就能获取该资源的引用名称R.drawable.background,然后我们就能通过名称从其他包获取对应的资源文件了,从而可以执行换肤操作。而且,通过这个方法,不只可以获取图片资源的id,也能获取字符串如R.string.title,字体颜色如R.color.red,字体大小如R.dimens.text_size_small等等属性,从而扩大换肤的范围。