中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

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

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

android實現可自由移動、監聽點擊事件的懸浮窗

發布時間:2020-10-08 22:04:11 來源:腳本之家 閱讀:237 作者:石煒賢的工作室 欄目:移動開發

最近因為項目需要,自己實現了個可以自由移動,并且長按可以跳出一個控制播放的,大的懸浮窗。

好,開始吧。首先我們先聊權限,懸浮窗需要在manifest中聲明一個權限:

<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />

然后呢,嗯,我們來講講關于懸浮窗實現的原理。

在Andriod中,所有的界面元素都要通過windowmanger來實現,像Activity、Fragment等等這些也是在其上實現。因此,我們的懸浮窗自然要通過這個實現。

這個項目中,我們自定義了兩個懸浮窗view。我們以其中一個比較簡單的為例:

我們自定義一個管理可以統一管理懸浮窗的類MyWindowManager,負責創建,刪除懸浮窗

/**
 * Created by shiwe on 2017/3/7.
 * 懸浮窗管理
 * 創建,移除
 * 單例模式
 */

public class MyWindowManager {

 private FloatNormalView normalView;
 private FloatControlView controlView;

 private static MyWindowManager instance;

 private MyWindowManager() {
 }

 public static MyWindowManager getInstance() {
  if (instance == null)
   instance = new MyWindowManager();
  return instance;
 }


 /**
  * 創建小型懸浮窗
  */
 public void createNormalView(Context context) {
  if (normalView == null)
   normalView = new FloatNormalView(context);
 }

 /**
  * 移除懸浮窗
  *
  * @param context
  */
 public void removeNormalView(Context context) {
  if (normalView != null) {
   WindowManager windowManager = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
   windowManager.removeView(normalView);
   normalView = null;
  }
 }

 /**
  * 創建小型懸浮窗
  */
 public void createControlView(Context context) {
  if (controlView == null)
   controlView = new FloatControlView(context);
 }

 /**
  * 移除懸浮窗
  *
  * @param context
  */
 public void removeControlView(Context context) {
  if (controlView != null) {
   WindowManager windowManager = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
   windowManager.removeView(controlView);
   controlView = null;
  }
 }
}

然后看看我們自定義的一個view,其繼承自LinearLayout,我們在initLayoutParams初始化這個控件的位置等其他參數;在initEvent方法中定義隨手指移動的監聽事件以及長按的監聽事件。

public class FloatNormalView extends LinearLayout {

 private Context context = null;
 private View view = null;
 private ImageView ivShowControlView = null;
 private WindowManager.LayoutParams lp = new WindowManager.LayoutParams();
 private static WindowManager windowManager;

 private float mTouchStartX;
 private float mTouchStartY;
 private float x;
 private float y;
 private boolean initViewPlace = false;
 private MyWindowManager myWindowManager;
 private boolean isControlViewShowing = false;

 public FloatNormalView(Context context) {
  super(context);
  this.context = context;
  myWindowManager = MyWindowManager.getInstance();
  LayoutInflater.from(context).inflate(R.layout.float_normal_view, this);
  view = findViewById(R.id.ll_float_normal);
  ivShowControlView = (ImageView) findViewById(R.id.iv_show_control_view);
  windowManager = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);
  initLayoutParams();
  initEvent();
 }


 /**
  * 初始化參數
  */
 private void initLayoutParams() {
  //屏幕寬高
  int screenWidth = windowManager.getDefaultDisplay().getWidth();
  int screenHeight = windowManager.getDefaultDisplay().getHeight();

  //總是出現在應用程序窗口之上。
  lp.type = WindowManager.LayoutParams.TYPE_PHONE;

  // FLAG_NOT_TOUCH_MODAL不阻塞事件傳遞到后面的窗口
  // FLAG_NOT_FOCUSABLE 懸浮窗口較小時,后面的應用圖標由不可長按變為可長按,不設置這個flag的話,home頁的劃屏會有問題
  lp.flags = WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
    | WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL;

  //懸浮窗默認顯示的位置
  lp.gravity = Gravity.START | Gravity.TOP;
  //指定位置
  lp.x = screenWidth - view.getLayoutParams().width * 2;
  lp.y = screenHeight / 2 + view.getLayoutParams().height * 2;
  //懸浮窗的寬高
  lp.width = WindowManager.LayoutParams.WRAP_CONTENT;
  lp.height = WindowManager.LayoutParams.WRAP_CONTENT;
  lp.format = PixelFormat.TRANSPARENT;
  windowManager.addView(this, lp);
 }

 /**
  * 設置懸浮窗監聽事件
  */
 private void initEvent() {
  ivShowControlView.setOnLongClickListener(new OnLongClickListener() {
   @Override
   public boolean onLongClick(View view) {
    if (!isControlViewShowing) {
     myWindowManager.createControlView(context);
     isControlViewShowing = true;
    } else {
     myWindowManager.removeControlView(context);
     isControlViewShowing = false;
    }
    return true;
   }
  });
  view.setOnTouchListener(new OnTouchListener() {
   @Override
   public boolean onTouch(View v, MotionEvent event) {
    switch (event.getAction()) {
     case MotionEvent.ACTION_DOWN:
      if (!initViewPlace) {
       initViewPlace = true;
       //獲取初始位置
       mTouchStartX += (event.getRawX() - lp.x);
       mTouchStartY += (event.getRawY() - lp.y);
      } else {
       //根據上次手指離開的位置與此次點擊的位置進行初始位置微調
       mTouchStartX += (event.getRawX() - x);
       mTouchStartY += (event.getRawY() - y);
      }
      break;
     case MotionEvent.ACTION_MOVE:
      // 獲取相對屏幕的坐標,以屏幕左上角為原點
      x = event.getRawX();
      y = event.getRawY();
      updateViewPosition();
      break;

     case MotionEvent.ACTION_UP:
      break;
    }
    return true;
   }
  });
 }

 /**
  * 更新浮動窗口位置
  */
 private void updateViewPosition() {
  lp.x = (int) (x - mTouchStartX);
  lp.y = (int) (y - mTouchStartY);
  windowManager.updateViewLayout(this, lp);
 }

最后,只需要在Activity中調用mywindowManager中調用createxxx方法就可以。

public class MainActivity extends AppCompatActivity {

 MyWindowManager myWindowManager;

 @Override
 protected void onCreate(@Nullable Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.activity_main);
  myWindowManager = MyWindowManager.getInstance();
  myWindowManager.createNormalView(this.getApplicationContext());
 }
}

最后,附上demo項目的下載地址: android實現懸浮窗

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持億速云。

向AI問一下細節

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

AI

花莲市| 黄山市| 土默特左旗| 定西市| 平顺县| 保定市| 逊克县| 双鸭山市| 榆中县| 昌图县| 工布江达县| 博白县| 阳原县| 汶川县| 辽宁省| 赫章县| 延长县| 南宫市| 和龙市| 洛宁县| 青川县| 自治县| 陇西县| 凌云县| 马龙县| 全州县| 耒阳市| 六安市| 兴国县| 和顺县| 商南县| 东兰县| 新余市| 广灵县| 宜章县| 健康| 繁峙县| 长宁区| 漠河县| 洛隆县| 葵青区|