由于它是在onresume中才调用 主题的,所以会有一段时间先看到一些旧的资源,然后才刷新界面,这样的好处是选择了不同的主题马上会看到结果。显然也有不好一面了。就是每次onresume都会调用一次。
总得来说,主题变换都是通过 getThemePackageName,得到主题apk的包名。
Resources themeResources = null;
themeResources = pm.getResourcesForApplication(themePackage);
然后得到资源。
int resourceID = themeResources.getIdentifier(resourceName, "color", themePackage);
if (resourceID != 0) {
view.setTextColor(themeResources.getColor(resourceID));
}
最后应用。
把应用主题放在oncreate里面,就更适合了,只不过不是每次选择了主题就会马上看到效果,需要重启这个activity或fragment。
上面这段是apollo的应用 方式。
private void initActionBar():
ThemeUtils.setTextColor(this, actionBarTitle, "action_bar_title_color");
ThemeUtils.initThemeChooser(this, actionBarUp, "action_bar_up", THEME_ITEM_BACKGROUND);
apollo是cm团队制作的音乐播放器,(据代码来看是根据google原来的music源码修改来的,使用上了fragment,viewpager,actionbar),当然它只能在v14以上的系统使用,效果不错。(其实我已经修改了部分,现在运行在2.3.3上 了,actionbar的功能还没有修改完成。还有popmenu)
从csipsimple,apollo的主题应用来看,都不会是创建一个Context对象,原因我想大概是Context容易造成内存的泄露,所以它使用了全局的对象。使用pm.getResourcesForApplication这样的方式来查找对应的资源文件。有听说这个方法稍微慢一些。因为getIdentifier速度的问题,很显然的,根据名字查询当然比不上根据id查询了(文档里这么说地 )
下面传几张图来看看效果。分别是默认的light效果与orange皮肤的效果(网上下载 的。)
还有两张是2.3.3系统的效果(修改了apollo的源码,ActionBar明显还没有处理,准备使用的是sdk里面的 actionbarcompat,因为原来的源码不多,所以不引入重量级的组件abs了。况且2.3.x迟早要被4.x替代的。)
源码在:https://github.com/CyanogenMod/android_packages_apps_Apollo.git
我下载的是branch apollo_jellybean这个比master更新,应该是针对jellybean的,我猜迟早是要放入master主分支中的。
如果想学皮肤制作的可以参考下。不错的示例,
说了半天,还没有说这次的重点。 上次csipimple使用一个接收器来接收广播,这次apollo不是,但原理相同,只要找得到包就可以了,皮肤没有源码,直接反编译了,里面没有java代码(annotation我觉得不算是) manifest文件也简单。 <manifest android:versionCode="2" android:versionName="1.1" package="com.lehoang.orangetheme" xmlns:android="http://schemas.android.com/apk/res/android"> <uses-sdk android:minSdkVersion="7" /> <application android:label="@string/themeTitle" android:icon="@drawable/ic_launcher"> <activity android:label="@string/themeTitle" android:name="com.lehoang.orangetheme.GoogleMusicThemeActivity"> <intent-filter> <action android:name="android.intent.action.MAIN" /> <action android:name="com.andrew.apollo.THEMES" /> <category android:name="android.intent.category.DEFAULT" /> </intent-filter> </activity> </application> </manifest> 关键在于com.andrew.apollo.THEMES这个, 而com.lehoang.orangetheme.GoogleMusicThemeActivity这段无关紧要的,其实没有这个Activity的java代码。 查找: Intent intent = new Intent("com.andrew.apollo.THEMES"); intent.addCategory("android.intent.category.DEFAULT"); PackageManager pm = getPackageManager(); List<ResolveInfo> themes = pm.queryIntentActivities(intent, 0); String[] entries = new String[themes.size() + 1]; String[] values = new String[themes.size() + 1]; entries[0] = APOLLO; values[0] = APOLLO; for (int i = 0; i < themes.size(); i++) { String appPackageName = (themes.get(i)).activityInfo.packageName.toString(); String themeName = (themes.get(i)).loadLabel(pm).toString(); entries[i + 1] = themeName; values[i + 1] = appPackageName; } themeLp.setEntries(entries); themeLp.setEntryValues(values); ThemePreview themePreview = (ThemePreview)findPreference(THEME_PREVIEW); themePreview.setTheme(themePackage); 可见android.intent.action.MAIN这个用处没有体现 。 这里使用了queryIntentActivities来查找对应的Activity,与查找receiver一样。结果是查找到这个包名,和主题包。 android:name="com.lehoang.orangetheme.GoogleMusicThemeActivity" 这个似乎就没有用到了,不过最好与包名一样,起个与众不同的就可以了。 主题选择后的应用: public void applyTheme(View v) { ThemePreview themePreview = (ThemePreview)findPreference(THEME_PREVIEW); String packageName = themePreview.getValue().toString(); ThemeUtils.setThemePackageName(this, packageName); Intent intent = new Intent(); intent.setClass(this, MusicLibrary.class); intent.setFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP | Intent.FLAG_ACTIVITY_NEW_TASK); startActivity(intent); finish(); } 如果选择了主题但没有应用,是看不到地,而这两个程序的普遍做法是将主题对应的包名等信息放在sp中,然后在程序启动时就可以直接读取了。 想要皮肤效果立刻见效,只有重启Activity了。如果一个程序的主页比较重要,而应用 主题不改变当前的程序中的内容,这种方法就不太可行了。它直接启动一个新的。 每一个皮肤的制作都遵循一定的规则的,规则当然由主程序定了,当主程序变换时,皮肤如果不变,有可能得不到对应的资源,所以可以读取皮肤的版本号,版本名称或其它的一些信息来判断皮肤是否对应当前的主程序。 另外说下apollo在ui设计方面可以学习的地方: popmenu,这个是v11后的东西,前面的系统就免试了。这是在bottombar中右侧竖线点击后的弹出的效果。 以前一直想弄一个上面使用tabs,然后再加入一些navigation_list mode这样可以下拉的spinner,而apollo直接把actionbar隐藏了,自己弄了一个scrollingtab,跟ActionBar的tabs是一样的效果。(我的思想被束缚了,一直想是否系统有现成的这样控件)。 对于底部的bottombar同样是使用了一个viewpager,我一直以为。像viewpager这样的东西算是重量级的控件了,通常用在大块区域,没想到,放到底部这样小块的地方,也是相当不错的效果。 列表中的spinner的效果(三角形的右下角)这也算是v14引入的吧。现在越来越多的程序会有这样的控件背景按钮了。 专辑中的列表项(要有封面背景才看得出效果),下面文字半透明的,加上背景,效果不错。 截图会有些图不正常,但程序没有问题,是ddms的截图的问题。 600k左右的代码,把一个音乐播放器弄成这样,还有自动搜索封面等 信息(google搜索的估计,使用了一个包,可能多数音乐搜索结果不能让人满意,所以我自己修改时去了。),有好多东西值得去学习的。 顺便说一下,如果你想自己修改源码,需要注意MediaPlayer.setNextMediaPlayer这个在之前的版本是没有的,所以在oncompletelistener里面需要加入一个mCurrentPlayer.start()启动播放,不然只有单曲的了。放完了,不会自动下一首。 audioeffect在2.1版本也没有,但2.3,还是可以看到效果的 最后感谢cm团队。 下载了几个主题然后,重新打包,现在可以用于2.x的系统了。 除了优点外,apollo还有一个缺点,就是一开始就加载了所有的Fragment,当然是浪费了资源了。修改这些,需要改变PagerAdapter的addfragment方式,然后在musiclibrary里面的不是使用new XXFragemnt,而是将Class放进驻,然后通过反射在PagerAdapter里面使用,获取实例。 最后将所有apk打包发布,比较大,是因为加了vlc的视频播放功能,它还有自动扫描视频,还没有处理,现在还是直接查找媒体库的内容。 以com.开头的是v14以后就是4.x以后的系统皮肤,也是原版的,其它的皮肤是重新打包,适用于2.x系统的。由于本程序主要针对2.x系统,在4。x上直接使用原版就好了。可以用里面的四个主题,皮肤是找了挺久的,只有在play上有下载。 而且取消了专辑封面下载。