溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

如何實現Android浮動窗口

發布時間:2020-07-30 09:18:06 來源:億速云 閱讀:711 作者:小豬 欄目:開發技術

小編這次要給大家分享的是如何實現Android浮動窗口,文章內容豐富,感興趣的小伙伴可以來了解一下,希望大家閱讀完這篇文章之后能夠有所收獲。

1.浮動窗口的實現原理

看到上圖的那個小Android圖標了吧,它不會被其他組建遮擋,也可以響應用戶的點擊和拖動事件,它的顯示和消失由WindowManager直接管理,它就是Android浮動窗口。Android浮動窗口的實現主要是靠WindowManager這個類。通過WindowManager類的addView(),updateViewLayout(),removeView()這幾個方法,我們可以直接在Window中添加,更新,移除View。

2.浮動窗口實現的具體步驟

1)既然浮動窗口的實現依賴與WindowManager,那么毫無疑問,我們得先拿到WindowManger對象??紤]到浮動窗口通常在應用程序退出后依然顯示,所以我們需要在Service中實現浮動窗口的添加和更新,當然別忘了提供給用戶一個取消浮動窗口的功能。

2)定義你要顯示的View??梢栽诓季治募卸x,也可以自定義視圖。

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;
   }
 }

看完這篇關于如何實現Android浮動窗口的文章,如果覺得文章內容寫得不錯的話,可以把它分享出去給更多人看到。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

亚洲午夜精品一区二区_中文无码日韩欧免_久久香蕉精品视频_欧美主播一区二区三区美女