a亚洲精品_精品国产91乱码一区二区三区_亚洲精品在线免费观看视频_欧美日韩亚洲国产综合_久久久久久久久久久成人_在线区

首頁 > 系統 > Android > 正文

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

2019-10-21 21:34:02
字體:
來源:轉載
供稿:網友

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

好,開始吧。首先我們先聊權限,懸浮窗需要在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實現懸浮窗

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


注:相關教程知識閱讀請移步到Android開發頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 色综合久久88色综合天天6 | 日韩视频免费 | 美日韩一区二区三区 | 成人av网址大全 | 日本一二区视频 | 精品中文字幕一区二区三区 | 毛片免费在线观看 | 伊人无码高清 | 久久久www成人免费无遮挡大片 | 午夜高清视频在线观看 | 亚洲免费一区 | 欧美日韩一区二区三区 | 视频一二三区 | 黄a视频在线观看 | 欧美涩涩视频 | 亚洲精品91 | 成人不卡在线 | 午夜精品一区二区三区在线播放 | 亚洲精品免费看 | 亚洲欧美日韩国产 | 91精品国产综合久久婷婷香蕉 | 嫩草网站在线观看 | 成人午夜视频在线观看 | 91精彩刺激对白露脸偷拍 | 欧美日韩一区二区视频在线观看 | 亚洲中出 | 成人在线免费观看 | 久久久久黄色 | 美女网站视频免费黄 | 蜜桃av人人夜夜澡人人爽 | 久久国产精品首页 | 亚洲免费小视频 | 日本色综合 | 久久国产精品视频观看 | 日韩精品一区二区三区第95 | 99精品欧美一区二区三区 | 一区二区影院 | 欧美高清不卡 | 四虎免费在线播放 | 婷婷综合激情 | 久久人|