青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

xiaoguozi's Blog
Pay it forword - 我并不覺(jué)的自豪,我所嘗試的事情都失敗了······習(xí)慣原本生活的人不容易改變,就算現(xiàn)狀很糟,他們也很難改變,在過(guò)程中,他們還是放棄了······他們一放棄,大家就都是輸家······讓愛傳出去,很困難,也無(wú)法預(yù)料,人們需要更細(xì)心的觀察別人,要隨時(shí)注意才能保護(hù)別人,因?yàn)樗麄兾幢刂雷约阂裁础ぁぁぁぁ?/span>
原文:
http://1622511.blog.51cto.com/1612511/581011

如果你要定制一個(gè)Android系統(tǒng),你想用你自己的Launcher(Home)作主界面來(lái)替換Android自己的Home,而且不希望用戶安裝的Launcher來(lái)替換掉你的Launcher.
我們可以通過(guò)修改Framework來(lái)實(shí)現(xiàn)這樣的功能。

這里以Android2.1的源代碼為例來(lái)實(shí)際說(shuō)明。

1)首先了解一下Android的啟動(dòng)過(guò)程。
  Android系統(tǒng)的啟動(dòng)先從Zygote開始啟動(dòng),然后......(中間的過(guò)程就不說(shuō)了).....一直到了SystemServer(framework)這個(gè)地方,看到這段代碼:

      /**
     * This method is called from Zygote to initialize the system. This will cause the native
     * services (SurfaceFlinger, AudioFlinger, etc..) to be started. After that it will call back
     * up into init2() to start the Android services.
     */
    native public static void init1(String[] args);

    public static void main(String[] args) {
        if (SamplingProfilerIntegration.isEnabled()) {
            SamplingProfilerIntegration.start();
            timer = new Timer();
            timer.schedule(new TimerTask() {
                @Override
                public void run() {
                    SamplingProfilerIntegration.writeSnapshot("system_server");
                }
            }, SNAPSHOT_INTERVAL, SNAPSHOT_INTERVAL);
        }

        // The system server has to run all of the time, so it needs to be
        // as efficient as possible with its memory usage.
        VMRuntime.getRuntime().setTargetHeapUtilization(0.8f);

        System.loadLibrary("android_servers");
        init1(args);
    }

    public static final void init2() {
        Log.i(TAG, "Entered the Android system server!");
        Thread thr = new ServerThread();
        thr.setName("android.server.ServerThread");
        thr.start();
    }
}

從SystemServer的main函數(shù)開始啟動(dòng)各種服務(wù)。
首先啟動(dòng)init1,然后啟動(dòng)init2.
從上面的注釋可以看到:init1這個(gè)方法時(shí)被Zygote調(diào)用來(lái)初始化系統(tǒng)的,init1會(huì)啟動(dòng)native的服務(wù)如SurfaceFlinger,AudioFlinger等等,這些工作做完以后會(huì)回調(diào)init2來(lái)啟動(dòng)Android的service。

這里我們主要來(lái)關(guān)注init2的過(guò)程。
init2中啟動(dòng)ServerThread線程,
ServerThread中啟動(dòng)了一系列的服務(wù),比如這些:

ActivityManagerService
EntropyService
PowerManagerService
TelephonyRegistry
PackageManagerService
AccountManagerService
BatteryService
HardwareService
Watchdog
SensorService
BluetoothService
StatusBarService
ClipboardService
InputMethodManagerService
NetStatService
ConnectivityService
AccessibilityManagerService
NotificationManagerService
MountService
DeviceStorageMonitorService
LocationManagerService
SearchManagerService
FallbackCheckinService
WallpaperManagerService
AudioService
BackupManagerService
AppWidgetService

這些大大小小的服務(wù)起來(lái)以后,開始
 ((ActivityManagerService)ActivityManagerNative.getDefault()).systemReady()
在systemReady后開始開始啟動(dòng)Launcher。

在尋找Launcher的時(shí)候是根據(jù)HOME的filter(在Manifest中定義的<category android:name="android.intent.category.HOME" />)來(lái)過(guò)濾。
然后根據(jù)filter出來(lái)的HOME來(lái)啟動(dòng),如果只有一個(gè)HOME,則啟動(dòng)這個(gè)HOME,如果用戶自己裝了HOME,那就會(huì)彈出來(lái)一個(gè)列表供用戶選擇。

我們現(xiàn)在希望從這里彈出我們自己定制的Launcher,同時(shí)也不希望彈出選擇HOME的界面,我們不希望用戶修改我們的home,比如我們的home上放了好多廣告,以及強(qiáng)制安裝的程序,不希望用戶把它干掉。

我們可以通過(guò)這樣來(lái)實(shí)現(xiàn):

2) 定義一個(gè)私有的filter選項(xiàng),然后用這個(gè)選項(xiàng)來(lái)過(guò)濾HOME.
   一般情況下我們使用Manifest中定義的<category android:name="android.intent.category.HOME"來(lái)過(guò)濾的,我們現(xiàn)在增加一個(gè)私有的HOME_FIRST過(guò)濾。

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

    //lixinso:添加CATEGORY_HOME_FIRST
    @SdkConstant(SdkConstantType.INTENT_CATEGORY)
    public static final String CATEGORY_HOME_FIRST = "android.intent.category.HOME_FIRST";

3)修改和CATEGORY_HOME相關(guān)的所有的地方,都改成HOME_FIRST,主要是framework中的這幾個(gè)地方:

    frameworks/base/services/java/com/android/server/am/ActivityManagerService.java中
    //intent.addCategory(Intent.CATEGORY_HOME);
    改成intent.addCategory(Intent.CATEGORY_HOME_FIRST); //lixinso:
    //if (r.intent.hasCategory(Intent.CATEGORY_HOME)) {
    改成if (r.intent.hasCategory(Intent.CATEGORY_HOME_FIRST)) { //lixinso: Intent.CATEGORY_HOME -> Intent.CATEGORY_HOME_FIRST
 
   frameworks/base/services/java/com/android/server/am/HistoryRecorder.java中
   // _intent.hasCategory(Intent.CATEGORY_HOME) &&
   改成 _intent.hasCategory(Intent.CATEGORY_HOME_FIRST) && //lixinso: Intent.CATEGORY_HOME->Intent.CATEGORY_HOME_FIRST

   frameworks/policies/base/mid/com/android/internal/policy/impl/MidWindowManager.java中
   //mHomeIntent.addCategory(Intent.CATEGORY_HOME); 
   改成 mHomeIntent.addCategory(Intent.CATEGORY_HOME_FIRST); //lixinso
 
  frameworks/policies/base/mid/com/android/internal/policy/impl/RecentApplicationsDialog.java中
   //new Intent(Intent.ACTION_MAIN).addCategory(Intent.CATEGORY_HOME),0);
   改成 new Intent(Intent.ACTION_MAIN).addCategory(Intent.CATEGORY_HOME_FIRST),0); //lixinso

  frameworks/policies/base/phone/com/android/internal/policy/impl/PhoneWindowManager.java中
   //mHomeIntent.addCategory(Intent.CATEGORY_HOME);
   改成 mHomeIntent.addCategory(Intent.CATEGORY_HOME_FIRST); //lixinso

  frameworks/policies/base/phone/com/android/internal/policy/impl/RecentApplicationsDialog.java中
   //ResolveInfo homeInfo = pm.resolveActivity(new Intent(Intent.ACTION_MAIN).addCategory(Intent.CATEGORY_HOME),0);
   改成 ResolveInfo homeInfo = pm.resolveActivity(new Intent(Intent.ACTION_MAIN).addCategory(Intent.CATEGORY_HOME_FIRST),0); //lixinso



4) 寫一個(gè)自己的Launcher.
   可以參考android sample中的Launcher,或者android源代碼中的 /packages/apps/Launcher 來(lái)寫。
   在Launcher中標(biāo)記其是不是Launcher的最關(guān)鍵的代碼時(shí)Manifest中的filter:android:name="android.intent.category.HOME"
   現(xiàn)在我們定義了自己的filter,那么,我們?cè)谖覀冏约簩懙腖auncher中將Manifest改為:
    <application  android:process="android.process.acore3" android:icon="@drawable/icon" android:label="@string/app_name">
        <activity android:name=".FirstAppActivity"
                  android:label="@string/app_name">
            <intent-filter>
                                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.HOME_FIRST" />
                <category android:name="android.intent.category.DEFAULT" />
                <category android:name="android.intent.category.MONKEY" />
            </intent-filter>
        </activity>
    </application>

然后將編譯好的apk放到/out/target/product/generic/system/app目錄下。

5)將Android自帶的Launcher刪除掉,包括源代碼(packages/apps/Launcher)和apk(/out/target/product/generic/system/app/Launcher.apk)。

6)
做完這些工作,就可以重新編譯Android了,我們可以編譯修改過(guò)的幾個(gè)相關(guān)的包。
如果之前編譯過(guò)了Android源碼,可以用mmm命令來(lái)編譯部分的改動(dòng)。
這里需要這樣編譯:

$ . build/envsetup.sh 
$ mmm frameworks/base
$ mmm frameworks/base/services/java
$ mmm frameworks/policies/base/mid
$ mmm frameworks/policies/base/phone

7)
編譯完成后重新生成img文件。
$ make snod

8) 現(xiàn)在可以啟動(dòng)Android模擬器來(lái)看效果了。
首先設(shè)置環(huán)境變量:
$ export ANDROID_PRODUCT_OUT= ./out/target/product/generic
然后切換到
$ cd ./out/host/linux-x86/bin
運(yùn)行
$ ./emulator

這樣我們啟動(dòng)的模擬器里面用的image就是我們剛才編譯好的自己定制的東西了。
從模擬器上可以看到啟動(dòng)的Launcher是我們自己的Launcher,不會(huì)出現(xiàn)默認(rèn)的Launcher了,也不會(huì)出現(xiàn)選擇界面。

9)我們?cè)衮?yàn)證一下,如果用戶裝上了一個(gè)其他的Launcher(Home)會(huì)怎么樣。
  從網(wǎng)上找一個(gè)一般的Launcher或者自己寫一個(gè)一般的Launcher裝上去,重新啟動(dòng),不會(huì)出現(xiàn)選擇界面。
  按HOME鍵也不會(huì)出來(lái)兩個(gè)HOME來(lái)選擇。


這樣我們就牢牢控制了用戶的桌面。
只有我們自己定制的HOME才能裝上。 這對(duì)于定制Android設(shè)備的廠商很有用處。
posted on 2013-02-17 14:51 小果子 閱讀(1526) 評(píng)論(0)  編輯 收藏 引用 所屬分類: 學(xué)習(xí)筆記Android & Ios
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            亚洲风情亚aⅴ在线发布| 亚洲美女精品成人在线视频| 国产精品成人在线| 久久精品女人| 亚洲欧美激情视频| 亚洲婷婷在线| 99在线|亚洲一区二区| 免费亚洲视频| 欧美va亚洲va国产综合| 亚洲电影免费观看高清| 亚洲国产高清自拍| 亚洲蜜桃精久久久久久久| 亚洲国产成人av| 亚洲欧洲日本在线| 日韩视频在线一区二区| 亚洲综合第一页| 欧美在线播放高清精品| 欧美99久久| 国产精品国产福利国产秒拍| 国产揄拍国内精品对白| 91久久精品国产91久久性色tv| 99在线|亚洲一区二区| 久久riav二区三区| 亚洲高清av| 亚洲综合精品四区| 久久成人久久爱| 欧美日韩成人激情| 韩日欧美一区二区| 亚洲一区二区三区三| 欧美成人精品一区二区| 亚洲一区在线免费观看| 女人天堂亚洲aⅴ在线观看| 欧美日韩国产成人在线91| 亚洲肉体裸体xxxx137| 国产精品爱啪在线线免费观看| 国产欧美日韩免费看aⅴ视频| 亚洲国产成人在线视频| 亚洲一区二区日本| 蘑菇福利视频一区播放| 亚洲欧美www| 欧美精品一区二区在线播放| 国产真实乱子伦精品视频| 中文高清一区| 亚洲国产高清视频| 久久久www| 国产九区一区在线| 日韩午夜在线观看视频| 欧美成年人网站| 午夜一区二区三视频在线观看| 欧美日韩国产在线看| 91久久精品国产91久久性色| 久久久久久久一区| 亚洲一二三级电影| 欧美视频一区二区| 一本久久a久久免费精品不卡| 欧美成人综合| 久久综合精品一区| 影音先锋久久精品| 久久免费的精品国产v∧| 亚洲欧美制服另类日韩| 国产精品久久久久影院色老大| 一本色道久久加勒比精品| 亚洲高清久久久| 美女主播一区| 亚洲三级影片| 最新中文字幕亚洲| 欧美日韩国产三级| 亚洲一区bb| 在线亚洲一区观看| 国产精品国产三级国产| 欧美伊人久久久久久久久影院| 亚洲网友自拍| 国产精品外国| 久久久久久97三级| 久久精品伊人| 亚洲精品国产视频| 亚洲精品国产精品乱码不99| 欧美三级韩国三级日本三斤| 午夜精品视频一区| 久久成年人视频| 亚洲国产91精品在线观看| 最新日韩av| 国产精品香蕉在线观看| 久久一区激情| 女仆av观看一区| 亚洲视频在线一区| 午夜伦欧美伦电影理论片| 激情文学一区| 亚洲精品国产精品国自产观看浪潮 | 久久国产99| 亚洲日本一区二区三区| 亚洲视频久久| 性色一区二区三区| 亚洲精品免费观看| 欧美日韩蜜桃| 欧美一区成人| 另类亚洲自拍| 亚洲在线网站| 久久精品国产亚洲aⅴ| 亚洲精品国精品久久99热一| 一区二区三区高清不卡| 国内精品久久久久久 | 欧美激情第一页xxx| 一区二区黄色| 久久精品中文字幕一区二区三区| 亚洲毛片网站| 久久国产精品72免费观看| 一区二区三区久久久| 久久九九国产| 欧美一区二区高清在线观看| 欧美电影免费| 久久精品理论片| 欧美精品自拍偷拍动漫精品| 噜噜爱69成人精品| 国产精品久在线观看| 亚洲国产精品久久| 国产在线欧美日韩| 在线视频免费在线观看一区二区| 在线日韩欧美视频| 午夜精品亚洲| 亚洲免费视频成人| 欧美福利视频在线| 免费成人av在线| 国产三区精品| 亚洲桃色在线一区| 在线中文字幕不卡| 欧美高清视频一区二区三区在线观看| 久久精品一区二区三区不卡| 欧美特黄一级| 日韩亚洲不卡在线| 一区二区三区www| 欧美激情中文字幕乱码免费| 欧美大成色www永久网站婷| 影音先锋成人资源站| 久久精品免费看| 久久阴道视频| 韩国成人理伦片免费播放| 亚洲欧美日韩中文在线制服| 亚洲欧美日韩另类精品一区二区三区| 欧美日韩国产精品自在自线| 亚洲精品日韩精品| 亚洲午夜一区二区| 欧美视频中文一区二区三区在线观看| 亚洲久久在线| 亚洲影音一区| 国产日韩在线亚洲字幕中文| 久久国产乱子精品免费女| 久久人人九九| 亚洲福利在线视频| 欧美aa在线视频| 亚洲精品国产无天堂网2021| 亚洲视频在线一区| 国产精品伦一区| 欧美亚洲视频在线看网址| 免费观看日韩av| 中文在线资源观看网站视频免费不卡 | 欧美激情第3页| 亚洲成色最大综合在线| 久久综合久久美利坚合众国| 亚洲国产精品一区二区久| 日韩亚洲一区二区| 欧美午夜片欧美片在线观看| 午夜精品在线| 欧美顶级少妇做爰| 一区二区三区.www| 国产日韩精品在线| 美日韩精品视频| 99国产一区| 久久久99免费视频| 亚洲黄色在线| 国产精品久久久久7777婷婷| 久久国产精品黑丝| 亚洲精品综合| 久久精品国产精品亚洲综合| 亚洲欧洲日产国产网站| 欧美午夜大胆人体| 久久久久久香蕉网| 99人久久精品视频最新地址| 久久久久国产成人精品亚洲午夜| 亚洲国产精品第一区二区| 欧美三级小说| 亚洲欧美国产视频| 亚洲第一成人在线| 午夜精品久久久久久久白皮肤| 亚洲国产成人在线视频| 欧美视频在线观看一区二区| 久久精品2019中文字幕| 这里只有精品视频| 欧美第一黄色网| 欧美亚洲一区二区在线观看| 99riav久久精品riav| 国产一区二区三区四区| 欧美日韩你懂的| 久久婷婷影院| 性色一区二区三区| 夜夜嗨av一区二区三区四区| 欧美国产日韩精品| 久久久久久久久伊人| 午夜精品久久久久久| 最新日韩精品|