替换Android桌面的相关问题:

1、想将home screen换成自己写的activity,该如何实现?

在 你要设置为home screen的那个activity的androidManifest.xml中的标签中加上这几句 话
运行后,重启模拟器会弹出一个选择进入哪个界面的对话框

2、怎样将系统默认的home screen删除?

重新编译launcher源码,去掉配置文件中的home属性和HOME属性。。
在自己的activity中加入这两个属性,然后重新烧系统。 

以下是定制Android的完整思路和步骤:


如果你要定制一个Android系统,你想用你自己的Launcher(Home)作主界面来替换Android自己的Home,而且不希望用户安装的Launcher来替换掉你的Launcher.
我们可以通过修改Framework来实现这样的功能。

这里以Android2.1的源代码为例来实际说明。

1)首先了解一下Android的启动过程。
Android系统的启动先从Zygote开始启动,然后……(中间的过程就不说了)…..一直到了SystemServer(framework)这个地方,看到这段代码:

1. nativepublicstaticvoidinit1(String[] args);
2.  
3. publicstaticvoidmain(String[] args) {
4. if(SamplingProfilerIntegration.isEnabled()) {
5. SamplingProfilerIntegration.start();
6. timer = newTimer();
7. timer.schedule(newTimerTask() {
8. @Override
9. publicvoidrun() {
10. SamplingProfilerIntegration.writeSnapshot(“system_server”);
11. }
12. }, SNAPSHOT_INTERVAL, SNAPSHOT_INTERVAL);
13. }
14.  
15. // The system server has to run all of the time, so it needs to be
16. // as efficient as possible with its memory usage.
17. VMRuntime.getRuntime().setTargetHeapUtilization(0.8f);
18.  
19. System.loadLibrary(“android_servers”);
20. init1(args);
21. }
22.  
23. publicstaticfinalvoidinit2() {
24. Log.i(TAG, “Entered the Android system server!”);
25. Thread thr = newServerThread();
26. thr.setName(“android.server.ServerThread”);
27. thr.start();
28. }
29. }


从SystemServer的main函数开始启动各种服务。


首先启动init1,然后启动init2.
从上面的注释可以看到:init1这个方法时被Zygote调用来初始化系统的,init1会启动native的服务如SurfaceFlinger,AudioFlinger等等,这些工作做完以后会回调init2来启动Android的service。

这里我们主要来关注init2的过程。
init2中启动ServerThread线程,
ServerThread中启动了一系列的服务,比如这些:



1. ActivityManagerService
2. EntropyService
3. PowerManagerService
4. TelephonyRegistry
5. PackageManagerService
6. AccountManagerService
7. BatteryService
8. HardwareService
9. Watchdog
10. SensorService
11. BluetoothService
12. StatusBarService
13. ClipboardService
14. InputMethodManagerService
15. NetStatService
16. ConnectivityService
17. AccessibilityManagerService
18. NotificationManagerService
19. MountService
20. DeviceStorageMonitorService
21. LocationManagerService
22. SearchManagerService
23. FallbackCheckinService
24. WallpaperManagerService
25. AudioService
26. BackupManagerService
27. AppWidgetService

这些大大小小的服务起来以后,开始
((ActivityManagerService)ActivityManagerNative.getDefault()).systemReady()
在systemReady后开始开始启动Launcher。

在寻找Launcher的时候是根据HOME的filter(在Manifest中定义的Android:name=”android.intent.category.HOME” />)来过滤。
然后根据filter出来的HOME来启动,如果只有一个HOME,则启动这个HOME,如果用户自己装了HOME,那就会弹出来一个列表供用户选择。

我们现在希望从这里弹出我们自己定制的Launcher,同时也不希望弹出选择HOME的界面,我们不希望用户修改我们的home,比如我们的home上放了好多广告,以及强制安装的程序,不希望用户把它干掉。

我们可以通过这样来实现:

2) 定义一个私有的filter选项,然后用这个选项来过滤HOME.
一般情况下我们使用Manifest中定义的

在Intent.java(frameworks/base/core/java/android/content/Intent.java)中添加两行代码

    1. //添加CATEGORY_HOME_FIRST
    2. @SdkConstant(SdkConstantType.INTENT_CATEGORY)
    3. publicstaticfinalString CATEGORY_HOME_FIRST = “android.intent.category.HOME_FIRST”;

    3)修改和CATEGORY_HOME相关的所有的地方,都改成HOME_FIRST,主要是framework中的这几个地方:


    1. frameworks/base/services/java/com/android/server/am/ActivityManagerService.java中
    2. //intent.addCategory(Intent.CATEGORY_HOME);
    3. 改成intent.addCategory(Intent.CATEGORY_HOME_FIRST);
    4. //if (r.intent.hasCategory(Intent.CATEGORY_HOME)) {
    5. 改成if(r.intent.hasCategory(Intent.CATEGORY_HOME_FIRST)) { //Intent.CATEGORY_HOME -> Intent.CATEGORY_HOME_FIRST
    6.  
    7. frameworks/base/services/java/com/android/server/am/HistoryRecorder.java中
    8. // _intent.hasCategory(Intent.CATEGORY_HOME) &&
    9. 改成 _intent.hasCategory(Intent.CATEGORY_HOME_FIRST) && //Intent.CATEGORY_HOME->Intent.CATEGORY_HOME_FIRST
    10.  
    11. frameworks/policies/base/mid/com/android/internal/policy/impl/MidWindowManager.java中
    12. //mHomeIntent.addCategory(Intent.CATEGORY_HOME);
    13. 改成 mHomeIntent.addCategory(Intent.CATEGORY_HOME_FIRST);
    14.  
    15. frameworks/policies/base/mid/com/android/internal/policy/impl/RecentApplicationsDialog.java中
    16. //new Intent(Intent.ACTION_MAIN).addCategory(Intent.CATEGORY_HOME),0);
    17. 改成 newIntent(Intent.ACTION_MAIN).addCategory(Intent.CATEGORY_HOME_FIRST),0);
    18.  
    19. frameworks/policies/base/phone/com/android/internal/policy/impl/PhoneWindowManager.java中
    20. //mHomeIntent.addCategory(Intent.CATEGORY_HOME);
    21. 改成 mHomeIntent.addCategory(Intent.CATEGORY_HOME_FIRST);
    22.  
    23. frameworks/policies/base/phone/com/android/internal/policy/impl/RecentApplicationsDialog.java中
    24. //ResolveInfo homeInfo = pm.resolveActivity(new Intent(Intent.ACTION_MAIN).addCategory(Intent.CATEGORY_HOME),0);
    25. 改成 ResolveInfo homeInfo = pm.resolveActivity(newIntent(Intent.ACTION_MAIN).addCategory(Intent.CATEGORY_HOME_FIRST),0);
    26.  
     4) 写一个自己的Launcher.
     可以参考Android sample中的Launcher,或者android源代码中的 /packages/apps/Launcher 来写。
     在Launcher中标记其是不是Launcher的最关键的代码时Manifest中的filter:android:name=”android.intent.category.HOME”
     现在我们定义了自己的filter,那么,我们在我们自己写的Launcher中将Manifest改为:27. @drawable/icon” android:label=”@string/app_name”>
    28. android:label=”@string/app_name”>

    然后将编译好的apk放到/out/target/product/generic/system/app目录下。

    5)将Android自带的Launcher删除掉,包括源代码(packages/apps/Launcher)和apk(/out/target/product/generic/system/app/Launcher.apk)。

    6)
    做完这些工作,就可以重新编译Android了,我们可以编译修改过的几个相关的包。
    如果之前编译过了Android源码,可以用mmm命令来编译部分的改动。
    这里需要这样编译:


    1. $ . build/envsetup.sh
    2. $ mmm frameworks/base
    3. $ mmm frameworks/base/services/java
    4. $ mmm frameworks/policies/base/mid
    5. $ mmm frameworks/policies/base/phone

    7)
    编译完成后重新生成img文件。

    1. $ make snod

    现在可以启动Android模拟器来看效果了。
    首先设置环境变量:
    $ export ANDROID_PRODUCT_OUT= ./out/target/product/generic
    然后切换到
    $ cd ./out/host/linux-x86/bin
    运行
    $ ./emulator

    这样我们启动的模拟器里面用的image就是我们刚才编译好的自己定制的东西了。
    从模拟器上可以看到启动的Launcher是我们自己的Launcher,不会出现默认的Launcher了,也不会出现选择界面。

    9)我们再验证一下,如果用户装上了一个其他的Launcher(Home)会怎么样。
    从网上找一个一般的Launcher或者自己写一个一般的Launcher装上去,重新启动,不会出现选择界面。
    按HOME键也不会出来两个HOME来选择。

    这样我们就牢牢控制了用户的桌面。

    只有我们自己定制的HOME才能装上。 这对于定制Android设备的厂商很有用处。