国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 系統 > Android > 正文

Android懸浮窗按鈕實現點擊并顯示/隱藏多功能列表

2019-10-21 21:44:48
字體:
來源:轉載
供稿:網友

前言

最近在一個項目中,需要制作錄屏的功能,原先是在應用中有錄屏/控制的按鈕,思考之下覺得這種效果并不好,因此就想制作一個可以懸浮的懸浮窗,這樣不論手機在什么界面中都可以對錄屏功能進行控制。

這里就來構建一個桌面的懸浮窗,使用了DataBinding的MVVM模式,這些方面就不再多提。

FloatNormalView

這個是一個普通的懸浮窗,懸浮窗只有一個按鈕,點擊按鈕顯示更多的按鈕。 

首先是頁面布局:

<?xml version="1.0" encoding="utf-8"?><layout xmlns:android="http://schemas.android.com/apk/res/android"> <data>  <variable   name="viewModel"   type="com.example.zjt.floatrecorder.FloatNormalViewModel"/> </data> <LinearLayout  android:layout_width="50dp"  android:layout_height="50dp"  android:gravity="center">  <RelativeLayout   android:id="@+id/root"   android:layout_width="wrap_content"   android:layout_height="wrap_content"   android:orientation="vertical">   <!-- 圖標,點擊后彈出后面的按鈕 -->   <ImageView    android:id="@+id/float_id"    android:layout_width="40dp"    android:layout_height="40dp"    android:background="@drawable/ic_launcher_background"    android:onClick="@{viewModel::onControlClick}"/>  </RelativeLayout> </LinearLayout></layout>

下面一步步的介紹這個懸浮窗的創建。

1 懸浮窗的顯示

// 創建WindowManager對象private WindowManager windowManager;windowManager = (WindowManager) context.getSystemService(Context.WINDOW_SERVICE);// 創建懸浮窗的LayoutParams private void initLayoutParams() {  try {   DisplayMetrics metrics = new DisplayMetrics();   windowManager.getDefaultDisplay().getMetrics(metrics);   screenWidth = metrics.widthPixels;   screenHeight = metrics.heightPixels;   lp = new WindowManager.LayoutParams();   if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {    lp.type = WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY;   } else {    lp.type = WindowManager.LayoutParams.TYPE_TOAST;   }   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 = 0;   lp.width = WindowManager.LayoutParams.WRAP_CONTENT;   lp.height = WindowManager.LayoutParams.WRAP_CONTENT;   lp.format = PixelFormat.TRANSPARENT;  } catch (Exception e) {  } }

上面分別創建了控制懸浮窗顯示的WindowManager和控制懸浮窗布局的LayoutParams 
然后使用如下代碼就可展示懸浮窗了:

public void show() {  if (!isShowing) {   isShowing = true;   windowManager.addView(this, lp);  } }

想要移除懸浮窗也很簡單,如下代碼:

public void dismiss() {  if (isShowing) {   isShowing = false;   windowManager.removeView(this);  } }

2 觸摸事件

觸摸事件可以使得懸浮窗跟隨手指進行移動

// 界面FloatLayoutBinding layoutBinding = DataBindingUtil.inflate(LayoutInflater.from(context),R.layout.float_layout,this,false);FloatNormalViewModel floatNormalViewModel = new FloatNormalViewModel(context,layoutBinding,onClickCallback);layoutBinding.setViewModel(floatNormalViewModel);addView(layoutBinding.getRoot());view = layoutBinding.root;isShowControlView = layoutBinding.floatId;//這就是控制按鈕// 控制的變量private float downX, downY;private float moveX, moveY;// 觸摸事件isShowControlView.setOnTouchListener(new OnTouchListener() {  @Override  public boolean onTouch(View view, MotionEvent motionEvent) {   switch (motionEvent.getActionMasked()) {    case MotionEvent.ACTION_DOWN:     downX = motionEvent.getRawX();     downY = motionEvent.getRawY();     break;    case MotionEvent.ACTION_MOVE:     moveX = motionEvent.getRawX() - downX;     moveY = motionEvent.getRawY() - downY;     downX += moveX;     downY += moveY;     updateViewPosition();     break;    }    return false; }});private void updateViewPosition() {  lp.x += (int) (moveX);  lp.y += (int) (moveY);  windowManager.updateViewLayout(this, lp);}

3 點擊事件

點擊事件是實現了一個回調函數,因為點擊事件的邏輯不應該在此處完成,應當交給主布局進行控制,所以定義了一個點擊接口。 

這里事件的處理順序是:點擊了按鈕后,按鈕將點擊事件通過回調函數來處理,而回調函數是由創建這個View的Activity或者Fragment、Service等提供的,就將事件處理交到了外部。

// 點擊的接口public interface OnClickCallback { public void onClick(View view);}// 控制按鈕點擊事件public void onControlClick(View view){ if(onClickCallback != null)  onClickCallback.onClick(view);}

多功能懸浮窗

多功能懸浮窗與上面類似,只不過在點擊事件上較多而已。 

而如何完成兩個懸浮窗的切換呢,就可以利用之前所使用的OnClickCallback回調接口了,將一個顯示、另一個隱藏即可,且兩個懸浮窗若采用同一個LayoutParams就可以讓兩個顯示在同一個位置。

private void init() {  floatNormalView = new FloatNormalView(context, new OnClickCallback() {   @Override   public void onClick(View view) {    floatControlView.setLayoutParams(floatNormalView.getLayoutParams());    floatControlView.show();    floatNormalView.dismiss();   }  });  floatControlView = new FloatControlView(context, new OnClickCallback() {   @Override   public void onClick(View view) {    floatNormalView.setLayoutParams(floatControlView.getLayoutParams());    floatNormalView.show();    floatControlView.dismiss();   }  }, new FloatControlViewModel.OnVisibleChangeListener() {   @Override   public void onChange(boolean isVisible) {    if (isControlVisible) {     floatControlView.show();     floatNormalView.dismiss();    } else {     floatControlView.dismiss();     floatNormalView.show();    }   }  });  floatNormalView.show(); }

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


注:相關教程知識閱讀請移步到Android開發頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 苏尼特左旗| 腾冲县| 南汇区| 新泰市| 永泰县| 吉木萨尔县| 腾冲县| 遂昌县| 英吉沙县| 子洲县| 丹阳市| 汉源县| 万宁市| 邵东县| 连江县| 沙河市| 和龙市| 天全县| 区。| 方正县| 闵行区| 诸城市| 钟祥市| 砚山县| 金秀| 涿鹿县| 东乡族自治县| 普定县| 九龙县| 保靖县| 德惠市| 门头沟区| 竹北市| 毕节市| 交城县| 嘉荫县| 柳州市| 南投市| 江北区| 武宣县| 桑植县|