您好,登錄后才能下訂單哦!
怎么在Android項目中實現(xiàn)一個全局懸浮框?很多新手對此不是很清楚,為了幫助大家解決這個難題,下面小編將為大家詳細講解,有這方面需求的人可以來學(xué)習(xí)下,希望你能有所收獲。
Androidmanifest.xml添加彈框權(quán)限
<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />
自定義懸浮窗類FloatWindow.java
public class FloatWindow implements View.OnTouchListener { private Context mContext; private WindowManager.LayoutParams mWindowParams; private WindowManager mWindowManager; private View mFloatLayout; private float mInViewX; private float mInViewY; private float mDownInScreenX; private float mDownInScreenY; private float mInScreenX; private float mInScreenY; private TextView infoText; public FloatWindow(Context context) { this.mContext = context; initFloatWindow(); } private void initFloatWindow() { LayoutInflater inflater = LayoutInflater.from(mContext); if(inflater == null) return; mFloatLayout = (View) inflater.inflate(R.layout.layout_float, null); infoText = mFloatLayout.findViewById(R.id.textView); mFloatLayout.setOnTouchListener(this); mWindowParams = new WindowManager.LayoutParams(); mWindowManager = (WindowManager) mContext.getSystemService(Context.WINDOW_SERVICE); if (Build.VERSION.SDK_INT >= 26) {//8.0新特性 mWindowParams.type = WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY; }else{ mWindowParams.type = WindowManager.LayoutParams.TYPE_SYSTEM_ALERT; } mWindowParams.format = PixelFormat.RGBA_8888; mWindowParams.flags = WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE; mWindowParams.gravity = Gravity.START | Gravity.TOP; mWindowParams.width = WindowManager.LayoutParams.WRAP_CONTENT; mWindowParams.height = WindowManager.LayoutParams.WRAP_CONTENT; } @Override public boolean onTouch(View view, MotionEvent motionEvent) { return floatLayoutTouch(motionEvent); } private boolean floatLayoutTouch(MotionEvent motionEvent) { switch (motionEvent.getAction()) { case MotionEvent.ACTION_DOWN: // 獲取相對View的坐標(biāo),即以此View左上角為原點 mInViewX = motionEvent.getX(); mInViewY = motionEvent.getY(); // 獲取相對屏幕的坐標(biāo),即以屏幕左上角為原點 mDownInScreenX = motionEvent.getRawX(); mDownInScreenY = motionEvent.getRawY() - getSysBarHeight(mContext); mInScreenX = motionEvent.getRawX(); mInScreenY = motionEvent.getRawY() - getSysBarHeight(mContext); break; case MotionEvent.ACTION_MOVE: // 更新浮動窗口位置參數(shù) mInScreenX = motionEvent.getRawX(); mInScreenY = motionEvent.getRawY() - getSysBarHeight(mContext); mWindowParams.x = (int) (mInScreenX- mInViewX); mWindowParams.y = (int) (mInScreenY - mInViewY); // 手指移動的時候更新小懸浮窗的位置 mWindowManager.updateViewLayout(mFloatLayout, mWindowParams); break; case MotionEvent.ACTION_UP: // 如果手指離開屏幕時,xDownInScreen和xInScreen相等,且yDownInScreen和yInScreen相等,則視為觸發(fā)了單擊事件。 if (mDownInScreenX == mInScreenX && mDownInScreenY == mInScreenY){ } break; } return true; } public void showFloatWindow(){ if (mFloatLayout.getParent() == null){ DisplayMetrics metrics = new DisplayMetrics(); // 默認固定位置,靠屏幕右邊緣的中間 mWindowManager.getDefaultDisplay().getMetrics(metrics); mWindowParams.x = metrics.widthPixels; mWindowParams.y = metrics.heightPixels/2 - getSysBarHeight(mContext); mWindowManager.addView(mFloatLayout, mWindowParams); } } public void updateText(final String s) { infoText.setText(s); } public void hideFloatWindow(){ if (mFloatLayout.getParent() != null) mWindowManager.removeView(mFloatLayout); } public void setFloatLayoutAlpha(boolean alpha){ if (alpha) mFloatLayout.setAlpha((float) 0.5); else mFloatLayout.setAlpha(1); } // 獲取系統(tǒng)狀態(tài)欄高度 public static int getSysBarHeight(Context contex) { Class<?> c; Object obj; Field field; int x; int sbar = 0; try { c = Class.forName("com.android.internal.R$dimen"); obj = c.newInstance(); field = c.getField("status_bar_height"); x = Integer.parseInt(field.get(obj).toString()); sbar = contex.getResources().getDimensionPixelSize(x); } catch (Exception e1) { e1.printStackTrace(); } return sbar; } }
自定義懸浮窗界面布局文件layout_float.xml
<?xml version="1.0" encoding="utf-8"?> <android.support.constraint.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" xmlns:app="http://schemas.android.com/apk/res-auto"> <ImageView android:id="@+id/imageView" android:layout_width="wrap_content" android:layout_height="wrap_content" android:src="@mipmap/float_win" app:layout_constraintStart_toStartOf="parent" app:layout_constraintTop_toTopOf="parent"/> <TextView android:id="@+id/textView" android:layout_width="wrap_content" android:layout_height="wrap_content" android:background="#00ffffff" android:text="hello" android:textSize="12sp" app:layout_constraintLeft_toLeftOf="@id/imageView" app:layout_constraintRight_toRightOf="@id/imageView" app:layout_constraintTop_toBottomOf="@id/imageView"/> </android.support.constraint.ConstraintLayout>
在Activity中使用懸浮窗。
public class MainActivity extends AppCompatActivity { private Button btnShow; FloatWindow floatWindow; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); // 權(quán)限判斷 if (Build.VERSION.SDK_INT >= 23) { if(!Settings.canDrawOverlays(getApplicationContext())) { // 啟動Activity讓用戶授權(quán) Intent intent = new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION,Uri.parse("package:" + getPackageName())); startActivityForResult(intent,10); } else { // 執(zhí)行6.0以上繪制代碼 initView(); } } else { // 執(zhí)行6.0以下繪制代碼 initView(); } } @Override protected void onResume() { super.onResume(); // 權(quán)限判斷 if (Build.VERSION.SDK_INT >= 23) { if(Settings.canDrawOverlays(getApplicationContext())) { initView(); } } else { //執(zhí)行6.0以下繪制代碼 initView(); } } private void initView() { setContentView(R.layout.activity_main); floatWindow = new FloatWindow(getApplicationContext()); btnShow = findViewById(R.id.btn_show); btnShow.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View view) { if (null != floatWindow) { floatWindow.showFloatWindow(); } } }); Button btnrefresh = findViewById(R.id.btn_refresh); btnrefresh.setOnClickListener(new View.OnClickListener() { @Override public void onClick(View view) { int random = (int) (Math.random() * 10); if (null != floatWindow) { floatWindow.updateText(String.valueOf(random)); } } }); } @Override protected void onDestroy() { super.onDestroy(); if (null != floatWindow) { floatWindow.hideFloatWindow(); } } }
看完上述內(nèi)容是否對您有幫助呢?如果還想對相關(guān)知識有進一步的了解或閱讀更多相關(guān)文章,請關(guān)注億速云行業(yè)資訊頻道,感謝您對億速云的支持。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。