Android浮動視窗實現原理及程式碼例項
1.浮動視窗的實現原理
看到上圖的那個小Android圖示了吧,它不會被其他組建遮擋,也可以響應使用者的點選和拖動事件,它的顯示和消失由WindowManager直接管理,它就是Android浮動視窗。Android浮動視窗的實現主要是靠WindowManager這個類。通過WindowManager類的addView(),updateViewLayout(),removeView()這幾個方法,我們可以直接在Window中新增,更新,移除View。
2.浮動視窗實現的具體步驟
1)既然浮動視窗的實現依賴與WindowManager,那麼毫無疑問,我們得先拿到WindowManger物件。考慮到浮動視窗通常在應用程式退出後依然顯示,所以我們需要在Service中實現浮動視窗的新增和更新,當然別忘了提供給使用者一個取消浮動視窗的功能。
2)定義你要顯示的View。可以在佈局檔案中定義,也可以自定義檢視。
3)設定必要的引數,其中有幾個比較重要的引數需要設定,具體請參考下面的程式碼。
4)將View新增到Window中,接收並處理事件,更新View。
5)在Manifest中加入對應的許可權。<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW"/>
3.浮動視窗實現程式碼
package com.spreadst.floatwindow; import android.app.Service; import android.content.Context; import android.content.Intent; import android.graphics.PixelFormat; import android.os.IBinder; import android.util.Log; import android.view.Gravity; import android.view.LayoutInflater; import android.view.MotionEvent; import android.view.View; import android.view.WindowManager; public class FloatWindowService extends Service { private static final TAG = "FloatWindowService"; private WindowManager mWindowManager; private WindowManager.LayoutParams mLayoutParams; private LayoutInflater mLayoutInflater; private View mFloatView; private int mCurrentX; private int mCurrentY; private static int mFloatViewWidth = 50; private static int mFloatViewHeight = 80; @Override public void onCreate() { // TODO Auto-generated method stub super.onCreate(); //初始化WindowManager物件和LayoutInflater物件 mWindowManager = (WindowManager) getApplicationContext().getSystemService(Context.WINDOW_SERVICE); mLayoutInflater = LayoutInflater.from(this); } @Override public void onStart(Intent intent,int startId) { // TODO Auto-generated method stub super.onStart(intent,startId); Log.i(TAG,"onStart()"); createView(); } private void createView() { // TODO Auto-generated method stub //載入佈局檔案 mFloatView = mLayoutInflater.inflate(R.layout.main,null); //為View設定監聽,以便處理使用者的點選和拖動 mFloatView.setOnTouchListener(new OnFloatViewTouchListener()); /*為View設定引數*/ mLayoutParams = new WindowManager.LayoutParams(); //設定View預設的擺放位置 mLayoutParams.gravity = Gravity.LEFT | Gravity.TOP; //設定window type mLayoutParams.type = WindowManager.LayoutParams.TYPE_PHONE; //設定背景為透明 mLayoutParams.format = PixelFormat.RGBA_8888; //注意該屬性的設定很重要,FLAG_NOT_FOCUSABLE使浮動視窗不獲取焦點,若不設定該屬性,螢幕的其它位置點選無效,應為它們無法獲取焦點 mLayoutParams.flags = WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE; //設定檢視的顯示位置,通過WindowManager更新檢視的位置其實就是改變(x,y)的值 mCurrentX = mLayoutParams.x = 50; mCurrentY = mLayoutParams.y = 50; //設定檢視的寬、高 mLayoutParams.width = 100; mLayoutParams.height = 100; //將檢視新增到Window中 mWindowManager.addView(mFloatView,mLayoutParams); } /*由於直接startService(),因此該方法沒用*/ @Override public IBinder onBind(Intent intent) { // TODO Auto-generated method stub return null; } /*該方法用來更新檢視的位置,其實就是改變(LayoutParams.x,LayoutParams.y)的值*/ private void updateFloatView() { mLayoutParams.x = mCurrentX; mLayoutParams.y = mCurrentY; mWindowManager.updateViewLayout(mFloatView,mLayoutParams); } /*處理檢視的拖動,這裡只對Move事件做了處理,使用者也可以對點選事件做處理,例如:點選浮動視窗時,啟動應用的主Activity*/ private class OnFloatViewTouchListener implements View.OnTouchListener { @Override public boolean onTouch(View v,MotionEvent event) { // TODO Auto-generated method stub Log.i(TAG,"mCurrentX: " + mCurrentX + ",mCurrentY: " + mCurrentY + ",mFloatViewWidth: " + mFloatViewWidth + ",mFloatViewHeight: " + mFloatViewHeight); /* * getRawX(),getRawY()這兩個方法很重要。通常情況下,我們使用的是getX(),getY()來獲得事件的觸發點座標, * 但getX(),getY()獲得的是事件觸發點相對與檢視左上角的座標;而getRawX(),getRawY()獲得的是事件觸發點 * 相對與螢幕左上角的座標。由於LayoutParams中的x,y是相對與螢幕的,所以需要使用getRawX(),getRawY()。 */ mCurrentX = (int) event.getRawX() - mFloatViewWidth; mCurrentY = (int) event.getRawY() - mFloatViewHeight; int action = event.getAction(); switch (action) { case MotionEvent.ACTION_DOWN: break; case MotionEvent.ACTION_MOVE: updateFloatView(); break; case MotionEvent.ACTION_UP: break; } return true; } } }
4.如何只在Launcher介面顯示浮動視窗
大家應該都熟悉360安全衛士的浮動視窗,它的浮動視窗只會在Launcher介面顯示,當用戶切到其它介面,浮動視窗自動被移除了。
要實現該功能,我們就必須知道當前所在的介面,如果只去監聽Activity的category,那麼我們只能知道什麼時候進入Launcher介面了,卻無法知道是否離開了Launcher介面。那麼360是如何實現該功能呢?大家可以反編譯一下它的程式碼。這裡提供一種可行的方法,我們的目前其實很簡單,就是要知道當前的Activity是否是Launcher介面的Activity。由於Activity是以堆疊的形式被管理的,因此,只要我們檢視棧頂的Activity是否是Launcher的Activity即可。要獲取Activity的Task資訊,需要在Manifest中新增對應許可權,<uses-permission android:name = “android.permission.GET_TASKS”/>。
private String getTopActivity(Context context) { //獲取ActivityManager物件 ActivityManager manager = (ActivityManager)getSystemService(ACTIVITY_SERVICE) ; /* * 拿到當前正在執行的Task列表,該列表按照最近使用的時間順序排列,其中的引數表示需要返回的最大列表項數目。 * 這裡我們只需要拿到處於onResume狀態的Activity所在的Task。 */ List<RunningTaskInfo> runningTaskInfos = manager.getRunningTasks(1) ; if(runningTaskInfos != null) { //拿到該task中的棧頂Activity return (runningTaskInfos.get(0).topActivity).toString() ; } else { return null; } }
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支援我們。