Android操作系統本身就是一個巨大的開源軟件倉庫,熟悉它既可以了解到Android系統的設計框架,也可以獲得高效的應用程序編寫方式。本文所分析的源碼來自於Google官方的AOSP源碼4.0.1_r1,手機的Android版本是CM 4.2.2。對於Android系統分析而言,手機的操作系統版本和源碼版本有些不一致的地方不會有太大的影響,但是如果需要將源碼中的修改安裝到手機里面的話,最不容易遇到問題的辦法就是保持手機中系統的版本和源碼的版本完全一致了。
1.程序入口發現
每一個應用程序都應當有各自的入口,使用JAVA編寫的Android應用程序也不例外。確認應用的入口是確認應用在系統中對應源碼的第一步,也是應用程序分析的第一步。我采用的辦法是將需要分析的短信app運行起來,通過Android自帶工具獲取當前占據主界面的activity名稱。
Stackoverflow中的回答http://stackoverflow.com/questions/13193592/adb-android-getting-the-name-of-the-current-activity給出了這個問題的三種解決辦法:
1)打Hierarchy View(Window->Open Perspective->Other->Hierarchy View),在Windows欄中用粗體顯示了當前占據屏幕activity和package,如下圖所示。其中com.android.mms是應用的包名,com.adnroid.mms.ui.ConversationList是具體的Activity名稱。
2)也可以直接打開Windows欄進行查看。(Window->Show View->Others->Windows),結果和上圖一致。
3) 使用dumpsys命令:adb shell "dumpsys window windows | grep -E 'mCurrentFocus|mFocusedApp'" ,結果如下:
dumpsys使用方法見:http://stackoverflow.com/questions/11201659/whats-android-adb-shell-dumpsys-tool-and-its-benefits
2.程序功能分析
確定了短信app的報名以及主Activity之后,可以通過軟件搜索(這里用的是everything)確定短信app在源碼中的位置是 packages\apps\Mms\src\com\android\mms\ui,並且短信app的啟動activity就是繼承自ListActivity的ConversationList。
1)ActionBar的構造
在onCreateOptionsMenu函數中通過導入conversation_list_menu加載menu項目。
public boolean onCreateOptionsMenu(Menu menu) { getMenuInflater().inflate(R.menu.conversation_list_menu, menu);
conversation_list_menu.xml中設置了:發送、搜索、設置、刪除所有等5個按鈕。
<menu xmlns:android="http://schemas.android.com/apk/res/android"> <item android:id="@+id/action_compose_new" android:title="@string/new_message" android:icon="@drawable/ic_menu_msg_compose_holo_dark" android:showAsAction="always|withText" /> <item android:id="@+id/search" android:title="@string/menu_search" android:icon="@drawable/ic_menu_search_holo_dark" android:showAsAction="ifRoom|collapseActionView" android:actionViewClass="android.widget.SearchView" /> <item android:id="@+id/action_settings" android:title="@string/menu_preferences" android:icon="@android:drawable/ic_menu_preferences" /> <item android:id="@+id/action_delete_all" android:title="@string/menu_delete_all" android:icon="@drawable/ic_menu_trash_holo_dark" /> <item android:id="@+id/action_debug_dump" android:title="@string/menu_debug_dump" /> </menu>
oncreate函數中的setupActionBar(),從功能上看是用來設置Actionbar的view的,並且可以顯示未讀短信的條數,但是在CM的手機上沒有看到對應的view。
private void setupActionBar() { ActionBar actionBar = getActionBar(); ViewGroup v = (ViewGroup)LayoutInflater.from(this) .inflate(R.layout.conversation_list_actionbar, null); actionBar.setDisplayOptions(ActionBar.DISPLAY_SHOW_CUSTOM, ActionBar.DISPLAY_SHOW_CUSTOM); actionBar.setCustomView(v, new ActionBar.LayoutParams(ActionBar.LayoutParams.WRAP_CONTENT, ActionBar.LayoutParams.WRAP_CONTENT, Gravity.CENTER_VERTICAL | Gravity.RIGHT)); mUnreadConvCount = (TextView)v.findViewById(R.id.unread_conv_count); }
2)listview的構造
在oncreate()中設置listview的屬性,其中setOnCreateContextMenuListener設置了ContextMenu,通過這個選項的設置,在用戶單擊了listview一欄之后,調用onContextItemSelected函數中的MENU_VIEW: {openThread(threadId);break;}。可以跳轉至發送短信界面。
mQueryHandler = new ThreadListQueryHandler(getContentResolver()); ListView listView = getListView(); listView.setOnCreateContextMenuListener(mConvListOnCreateContextMenuListener); listView.setOnKeyListener(mThreadListKeyListener); listView.setChoiceMode(ListView.CHOICE_MODE_MULTIPLE_MODAL); listView.setMultiChoiceModeListener(new ModeCallback()); // Tell the list view which view to display when the list is empty View emptyView = findViewById(R.id.empty); listView.setEmptyView(emptyView);
initListAdapter();
listview的初始化,oncreate函數中的initListAdapter()如下:
private void initListAdapter() { mListAdapter = new ConversationListAdapter(this, null); mListAdapter.setOnContentChangedListener(mContentChangedListener); setListAdapter(mListAdapter); getListView().setRecyclerListener(mListAdapter); }