如何在Android中實現懸浮Activity
通常來說,為一款已經優化過的手機APP開發平板的版本也不是很難。使用Fragment、decompose Entities等組件就可以又快又輕松地完成。但是,最近遇到一個項目就沒有這么簡單了。我們不只要開發一個平板APP(基于ActionBarSherlock庫,也稱作ABS),還需要實現以下功能:
- 實現一個懸浮并且透明顯示的Activity,這樣用戶在進行輸入操作的時候可以看到別的Activity或者它自己的窗口背景;
- 保存懸浮的Activity的原始尺寸,包括水平方向和豎直方向;最好是Activity可以居中顯示,在小尺寸設備上可以全屏顯示,在大尺寸設備上至少顯示2/3的大小。
因為我們之前說過已經有了一個開發好的手機APP,所以如果為了講這個重新設計再編碼那就浪費時間了。以下的講解和描述都是基于已有的代碼,通過修改Activity的實現來適配平板設備,已達到代碼的最大利用率。
任務概述
這里主要有三個任務:
- 計算已有的Activity的尺寸并確定它的新坐標位置;
- 使Activity透明,可以看見其背景;
- 處理背景使其不能再和用戶進行交互。
計算窗口大小
前面提到,我們需要以一個已有的手機APP為基礎(使用了ActionBarSherlock庫),這個庫我們已經以一個外部庫的形式集成到項目中,并帶有源代碼。
如果對ABS的源碼很熟悉或者曾對原生ActionBar的構架有了解,就會發現:如果改變Activity的大小,那在調用setContentView方法的時候,功能沒有實現:ActionBar的大小和位置都沒有變化。這時要做的就是在更高的級別里操作,在系統繪制ActionBar的時候——ABS或者原生的ActionBar,這時候還沒有任何窗口的數據,這樣我們就可以按照需要調整了。
最顯而易見的方式就是改變Window的尺寸,所以這里需要以下這段代碼:
@Override public void onAttachedToWindow() { super.onAttachedToWindow();if (getResources().getBoolean(R.bool.is_tablet) && mOpenAsSmallWindow) { final View view = getWindow().getDecorView(); final WindowManager.LayoutParams lp = (WindowManager.LayoutParams) view.getLayoutParams(); lp.gravity = Gravity.CENTER; lp.width = mActivityWindowWidth; lp.height = mActivityWindowHeight; getWindowManager().updateViewLayout(view, lp); } }</pre>
代碼執行完畢后,就可以看到窗口的大小已經根據mActivityWindowWidth和mActivityWindowHeight的值發生了改變。
is_tablet檢查只對平板有效,mOpenAsSmallWindow標識表示是否讓窗口全屏顯示或者是否作為新的窗口渲染。
如果在運行的時候報錯了,錯誤日志是:
ActionBarView can only be used with android:layout_width="match_parent" (or fill_parent).
那不用擔心,我們有ABS的源碼,直接修改源碼就可以了。打開ActionBarView這個文件,然后再onMeasure方法里把拋出的異常注釋掉——這樣應該就可以解決問題了。安卓上使用的是比較特殊的權限控制機制,所以這里我們可以繼續深入的不多,也許還能粗略計算菜單項等,不過很多東西我們都不可把控。
注意:這個方法沒有在原生ActionBar上測試,因為這里只是在作者的Android版本中測試過。如果要禁用ActionBar,需要找到ActionBarSherlock這個類,然后注釋掉以下這行代碼:
// registerImplementation(ActionBarSherlockNative.class);代碼運行應該沒問題,然后就可以看到不透明背景的Activity了。
添加透明功能
在實現了Activity的尺寸和大小都改變后,現在就要讓它透明了。可以給這個平板上的Activity主題添加這個屬性:
<item name="android:windowIsTranslucent">true</item>這個屬性值可以使Activity背景透明。
看起來不錯,效果已經基本實現了。
還有一個小問題:以上測試的都是在Nexus7 android4.3上執行,而這個App在Nexus7 的android4.2版本中,Activity沒有顯示。
經過多次調試和日志檢查后,發現我們的Activity(就叫MainActivity吧)沒有被銷毀(沒有調用onDestroy方法),所以Android系統就把它忽略了,沒有繪制它。為什么呢?因為Android 4.4 KitKat新增了一個優化算法。因為這個優化算法,Android系統看到Activity為全屏顯示模式(不管WindowManager的改變),就會繪制這個MainActivity,但是它之上的東西就被忽略了,所以我們就看不到任何顯示了。
我們研究了下這個問題,然后發現對話框和其他不全屏顯示的部件都可以正常繪制,所以我們需要在主題theme里加上這幾行:
<item name="android:windowIsFloating">true</item> <item name="android:windowCloseOnTouchOutside">false</item> <item name="android:colorBackgroundCacheHint">@null</item> <item name="android:backgroundDimEnabled">true</item>這樣問題就解決了。綜上所述,我們就實現了一個懸浮、透明的Activity。
原文鏈接: azoft 翻譯: 伯樂在線 - chris
譯文鏈接: http://blog.jobbole.com/65531/