android theme 皮肤主题 的应用 2

上次说到csipsimple里面使用皮肤的方式,是一个receiver来接收,然后setresultdata返回 数据的。
由于它是在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上有下载。

而且取消了专辑封面下载。



你可能感兴趣的:(android)