android懸浮窗口

不知道有沒有童鞋用過QQ的Android版本中的一個高清版本,當你退出QQ的時候,會在你的屏幕上出現一個QQ圖標的懸浮窗,就和PC上的迅雷的懸浮窗口一樣,在所有應用的最上層,本來這個功能感覺還是挺好的,結果TX就給流氓瞭下,居然沒有做隱藏掉這個懸浮窗的功能。

今天就來為大傢簡單介紹下這個懸浮窗的實現方法吧,不過記住至少還是給用戶做一個隱藏掉這個懸浮窗的功能吧!其實實現原理還是挺簡單的,就是利用WindowManager將你的圖標View添加到窗口中,不知道大傢知道WindowManager.LayoutParams中有一個type屬性不,這裡玄機挺大的,找瞭好久,終於給我試出來有一個TYPE_PHONE可以來實現這個功能的,該類型提供與用戶交互,置於所有應用程序上方,但是在狀態欄後面。至於其他的類型,大傢還是去找API或者GG吧。
先看下效果圖吧:

device-2012-01-09-213745.png

紅框圈中得圖標可以任意拖動哦,你可以根據自己的實際需要給圖標添加功能。

核心代碼如下:
package com.apkstory.service;

import com.apkstory.R;
import com.apkstory.util.MyApplication;

import android.app.Service;
import android.content.Intent;
import android.graphics.PixelFormat;
import android.os.IBinder;
import android.view.Gravity;
import android.view.LayoutInflater;
import android.view.MotionEvent;
import android.view.View;
import android.view.WindowManager;
import android.view.View.OnTouchListener;

/**
* 懸浮窗的制作
* @author 鋒翼
* @link www.apkstory.com
*/
public class TopFloatService extends Service {

        WindowManager wm = null;
        WindowManager.LayoutParams wmParams = null;
        View view;
        private float mTouchStartX;
        private float mTouchStartY;
        private float x;
        private float y;
       
        @Override
        public void onCreate() {
                super.onCreate();
                view = LayoutInflater.from(this).inflate(R.layout.floating, null);
                createView();
        }

        private void createView() {
                // 獲取WindowManager
                wm = (WindowManager) getApplicationContext().getSystemService("window");
                // 設置LayoutParams(全局變量)相關參數
                wmParams =  ((MyApplication) getApplication()).getMywmParams();
                wmParams.type = WindowManager.LayoutParams.TYPE_PHONE;// 該類型提供與用戶交互,置於所有應用程序上方,但是在狀態欄後面
                wmParams.flags |= WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE;// 不接受任何按鍵事件
                wmParams.gravity = Gravity.LEFT | Gravity.TOP; // 調整懸浮窗口至左上角
                // 以屏幕左上角為原點,設置x、y初始值
                wmParams.x = 0;
                wmParams.y = 0;
                // 設置懸浮窗口長寬數據
                wmParams.width = WindowManager.LayoutParams.WRAP_CONTENT;
                wmParams.height = WindowManager.LayoutParams.WRAP_CONTENT;
                wmParams.format = PixelFormat.RGBA_8888;

                wm.addView(view, wmParams);

                view.setOnTouchListener(new OnTouchListener() {
                        public boolean onTouch(View v, MotionEvent event) {
                                // 獲取相對屏幕的坐標,即以屏幕左上角為原點
                                x = event.getRawX();
                                // 25是系統狀態欄的高度,也可以通過方法得到準確的值,自己微調就是瞭
                                y = event.getRawY()-25 ;
                                switch (event.getAction()) {
                                case MotionEvent.ACTION_DOWN:
                                        // 獲取相對View的坐標,即以此View左上角為原點
                                        mTouchStartX = event.getX();
                                        mTouchStartY = event.getY()+view.getHeight()/2;
                                        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(view, wmParams);
        }

        @Override
        public IBinder onBind(Intent intent) {
                return null;
        }
}

 

摘自 fhy_2008的專欄

發佈留言