Android之懸浮視窗實現(WindowManager)

來源:互聯網
上載者:User

    工作中遇到一些項目需要把表單顯示在最上層,像來電彈窗顯示電話號碼等資訊、攔截簡訊資訊顯示給使用者或者遊戲中實現聲音的調節,我們想這些資料放在最上層,activity就滿足不了我們的需求了,有些開發人員使用了迴圈顯示toast的方式,toast是不能獲得焦點的,這種方法是不可取的。這個時候,我們如何處理呢?
       原來,整個Android的視窗機制是基於一個叫做 WindowManager,這個介面可以添加view到螢幕,也可以從螢幕刪除view。它面向的對象一端是螢幕,另一端就是View,直接忽略我們以前的Activity或者Dialog之類的東東。其實我們的Activity或者Dialog底層的實現也是通過WindowManager,這個WindowManager是全域的,整個系統就是這個唯一的東東。它是顯示View的最底層了。

   WindowManager主要用來管理視窗的一些狀態、屬性、view增加、刪除、更新、視窗順序、訊息收集和處理等。通過Context.getSystemService(Context.WINDOW_SERVICE)的方式可以獲得WindowManager的執行個體.

WindowManager繼承自ViewManager,裡面涉及到視窗管理的三個重要方法,分別是:

     * addView();

     * updateViewLayout();

     * removeView(); 

 

如下:

 

 

可以移動的懸浮框實現代碼如下:

public class WindowManageDemoActivity extends Activity { private WindowManager mWindowManager; private WindowManager.LayoutParams param; private FloatView mLayout;    /** Called when the activity is first created. */    @Override    public void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.main);                      showView();    }    private void showView(){     mLayout=new FloatView(getApplicationContext());     mLayout.setBackgroundResource(R.drawable.faceback_head);     //擷取WindowManager     mWindowManager=(WindowManager)getApplicationContext().getSystemService(Context.WINDOW_SERVICE);        //設定LayoutParams(全域變數)相關參數     param = ((MyApplication)getApplication()).getMywmParams();     param.type=WindowManager.LayoutParams.TYPE_SYSTEM_ALERT;     // 系統提示類型,重要     param.format=1;     param.flags = WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE; // 不能搶佔聚焦點     param.flags = param.flags | WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH;     param.flags = param.flags | WindowManager.LayoutParams.FLAG_LAYOUT_NO_LIMITS; // 排版不受限制           param.alpha = 1.0f;             param.gravity=Gravity.LEFT|Gravity.TOP;   //調整懸浮視窗至左上方        //以螢幕左上方為原點,設定x、y初始值     param.x=0;     param.y=0;                //設定懸浮視窗長寬資料     param.width=140;     param.height=140;                //顯示myFloatView映像     mWindowManager.addView(mLayout, param);         }    @Override    public void onDestroy(){     super.onDestroy();     //在程式退出(Activity銷毀)時銷毀懸浮視窗     mWindowManager.removeView(mLayout);    }}

FloatView代碼:

public class FloatView extends View { private float mTouchStartX;    private float mTouchStartY;    private float x;    private float y;        private WindowManager wm=(WindowManager)getContext().getApplicationContext().getSystemService(Context.WINDOW_SERVICE);    private WindowManager.LayoutParams wmParams = ((MyApplication)getContext().getApplicationContext()).getMywmParams(); public FloatView(Context context) {  super(context);    // TODO Auto-generated constructor stub }   @Override  public boolean onTouchEvent(MotionEvent event) {         //擷取相對螢幕的座標,即以螢幕左上方為原點         x = event.getRawX();         y = event.getRawY()-25;   //25是系統狀態列的高度      Log.i("currP", "currX"+x+"====currY"+y);      switch (event.getAction()) {         case MotionEvent.ACTION_DOWN:          //擷取相對View的座標,即以此View左上方為原點          mTouchStartX =  event.getX();                  mTouchStartY =  event.getY();                             Log.i("startP", "startX"+mTouchStartX+"====startY"+mTouchStartY);                          break;         case MotionEvent.ACTION_MOVE:                          updateViewPosition();             break;         case MotionEvent.ACTION_UP:          updateViewPosition();          mTouchStartX=mTouchStartY=0;          break;         }         return true;  }    private void updateViewPosition(){  //更新浮動視窗位置參數  wmParams.x=(int)( x-mTouchStartX);  wmParams.y=(int) (y-mTouchStartY);     wm.updateViewLayout(this, wmParams);       }}

最後,還有需要注意的是,如果要用懸浮視窗,需要在AndroidManifest.xml中加入如下的許可權:
        <uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />


MyApplication代碼:

public class MyApplication extends Application { /**  * 建立全域變數  * 全域變數一般都比較傾向於建立一個單獨的資料類檔案,並使用static靜態變數  *   * 這裡使用了在Application中添加資料的方法實現全域變數  * 注意在AndroidManifest.xml中的Application節點添加android:name=".MyApplication"屬性  *   */ private WindowManager.LayoutParams wmParams=new WindowManager.LayoutParams(); public WindowManager.LayoutParams getMywmParams(){  return wmParams; }}
相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.