1. WindowManager,Window的介紹
1.1. Window
Window表示一個窗口。對于Android里的Window,我們可以類比Windows系統(tǒng)中的Window,在Windows中,每打開一個軟件,都會彈出一個窗口,這個窗口右上角有最小化,最大化,關(guān)閉按鈕,做了某些操作時,也可能會彈出一個窗口,下面可能會有確定,取消之類的按鈕,這些都是Windows系統(tǒng)中的窗口。如圖所示:
Android中所有的界面都是顯示在一個個Window中的,包括Activity,Dialog,Toast,甚至狀態(tài)欄,最近應(yīng)用列表,都是在Window中顯示的。只是我們看不到這些Window的邊框,只能看到里面的內(nèi)容。其實(shí) Window并不能真正的顯示內(nèi)容,它只是一個虛擬的"框",真正能顯示內(nèi)容的是View。Window是View的直接管理者,觸摸事件也是先由Window接收,然后傳遞給View的。
Window是一個抽象類,在Android手機(jī)中,Window的實(shí)現(xiàn)類是PhoneWindow。
1.2. WindowManager
WindowManager是Window的管理者,對應(yīng)著系統(tǒng)底層的一個服務(wù):WindowManagerService。
我們無法直接訪問Window,要操作Window,必須通過WindowManager。WindowManager有三個常用方法:addView,removeView,updateViewLayout我們可以通過WindowManager往屏幕上添加/刪除一個Window,或者通過它修改一個Window的布局參數(shù)。
WindowManager是一個接口,在Android中,WindowManager的實(shí)現(xiàn)類WindowManagerImpl。
2. WindowManager使用詳解
2.1. 往屏幕上添加一個Window
調(diào)用WindowManager的addView方法即可。
顯示效果如下:
2.2. 代碼詳解
上面的代碼表示:調(diào)用 WindowManager的 addView(View view,WindowManager。LayoutParams lp) 方法,往屏幕上添加一個Window,這個Window中顯示的內(nèi)容為第一個參數(shù)設(shè)置的View,Window的顯示位置以及其他屬性由第二個參數(shù) WindowManager。LayoutParams指定。
這個方法很簡單,但是 WindowManager。LayoutParams 中有兩個字段比較重要,這里詳細(xì)說一下。
1.flags
用來控制Window的顯示特性,有很多可取的值,不同的的值表示不同的顯示特性, 如果希望Window具有多個值的特性, 可以使用 “|” 將這些值進(jìn)行按位或運(yùn)算。這里介紹幾個比較常用的取值:
2.type
用來表示W(wǎng)indow的類型,Window有三種大的類型,分別是應(yīng)用Window,子Window和系統(tǒng)Window。Activity的Window就是一種應(yīng)用Window,Dialog的Window是一種子Window,子Window不能單獨(dú)存在,必須附屬在特定的父Window中,這也就是為什么Dialog的Context必須是Activity。系統(tǒng)Window大都是(不是全部)需要聲明權(quán)限才能創(chuàng)建,獨(dú)立應(yīng)用Window之外,比如Toast,狀態(tài)欄等等。
Window是分層的,層級大的會覆蓋在層級小的之上,三大類Window中,應(yīng)用Window層級范圍是1-99,子Window是1000-1999,系統(tǒng)Window是2000-2999,這就是為什么Dialog顯示在Activity之上,而Toast又可以顯示在Dialog之上。如圖:
type這個字段的取值有很多,不同的值表示不同的層級范圍,具體可以看文檔,一般來說,如果這個取值為系統(tǒng)Window層級范圍內(nèi)的值,則需要聲明權(quán)限:
<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW"/>
另外,type會影響flags的效果,比如,如果type設(shè)置為 TYPE_TOAST,則無論怎樣設(shè)置flags,這個Window都無法接收觸摸事件。
2.3. 查看屏幕上的Window
我們再往屏幕上加一個PopupWindow和一個Dialog,當(dāng)前界面如下:
在Eclipse中,點(diǎn)擊菜單 Window - Open Perspective - Others,選擇 HierarchyView,打開,選擇Windows面板,可以看到當(dāng)前屏幕中所有的Window:
我們添加的Window在其中顯示的標(biāo)題為AddWindow,另外,我們可以看到還有別的幾個Window,比如 PopupWindow,MainActivity,加粗的那一個其實(shí)是MainActivity中彈出的Dialog,還能看到 StatusBar(狀態(tài)欄),RecentsPanel(最近應(yīng)用列表)等等,這也證明了我們前面說的,Android中所有的界面都是顯示在Window中的。
3桌面懸浮窗實(shí)現(xiàn)思路
3.1. 在桌面上顯示W(wǎng)indow
如果我們在Activity中使用WindowManager添加Window,當(dāng)Activity退出時,添加的Window也會被回收掉。所以要想在桌面上顯示懸浮窗,可以在Service中使用WindowManager添加Window,這樣只要服務(wù)不停止,就可以一直顯示。當(dāng)服務(wù)啟動時,在其onCreate方法中,使用WindowManager的 addView方法添加一個系統(tǒng)Window,當(dāng)服務(wù)銷毀時,可以在其 onDestroy中使用WindowManager的removeView 方法移除Window。大體是這樣的思路,代碼就不再給出了。
3.2.. 讓這個Window隨手指移動
要想讓這個Window能接收事件,需要給他設(shè)置相應(yīng)的flags(只要不包含F(xiàn)LAG_NOT_TOUCHABLE即可),另外其type也不能是 TYPE_TOAST??梢允褂茫篢YPE_PRIORITY_PHONE,表示比來去電界面的Window級別還要高一些(來去電界面的Window是系統(tǒng)Window)。
注意添加權(quán)限:
然后給Window里的View設(shè)置onTouchListener,重寫onTouch方法:
這樣就實(shí)現(xiàn)了Window隨著手指拖動而移動了。
本文版權(quán)歸傳智播客Android培訓(xùn)學(xué)院所有,歡迎轉(zhuǎn)載,轉(zhuǎn)載請注明作者出處。謝謝!
作者:傳智播客Android培訓(xùn)學(xué)院
首發(fā):http://m.xamj520.com/Android