溫馨提示×

溫馨提示×

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

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

android如何實(shí)現(xiàn)桌面移動懸浮窗口

發(fā)布時間:2021-04-17 09:41:05 來源:億速云 閱讀:310 作者:小新 欄目:移動開發(fā)

這篇文章將為大家詳細(xì)講解有關(guān)android如何實(shí)現(xiàn)桌面移動懸浮窗口,小編覺得挺實(shí)用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。

現(xiàn)在很多應(yīng)用都有這樣的功能,比如360等安全衛(wèi)士,手機(jī)管家之內(nèi)的應(yīng)用。

效果圖:

android如何實(shí)現(xiàn)桌面移動懸浮窗口

一、實(shí)現(xiàn)原理及移動思路

調(diào)用WindowManager,并設(shè)置WindowManager.LayoutParams的相關(guān)屬性,通過WindowManager的addView方法創(chuàng)建View,這樣產(chǎn)生出來的View根據(jù)WindowManager.LayoutParams屬性不同,效果也就不同了。比如創(chuàng)建系統(tǒng)頂級窗口,實(shí)現(xiàn)懸浮窗口效果!然后通過覆寫懸浮View中onTouchEvent方法來改變windowMananager.LayoutParams中x和y的值來實(shí)現(xiàn)自由移動懸浮窗口。

二、示例代碼

先看看懸浮View的代碼把

package com.example.suspend;
 
import android.content.Context;
import android.util.Log;
import android.view.MotionEvent;
import android.view.WindowManager;
import android.widget.ImageView;
 
public class MyFloatView extends ImageView {
 private float mTouchStartX;
 private float mTouchStartY;
 private float x;
 private float y;
 
 private WindowManager wm = (WindowManager) getContext()
  .getApplicationContext().getSystemService("window");
 // 此wmParams變量為獲取的全局變量,用以保存懸浮窗口的屬性
 private WindowManager.LayoutParams wmParams = ((MyApplication) getContext()
  .getApplicationContext()).getMywmParams();
 
 public MyFloatView(Context context) {
 super(context);
 }
 
 @Override
 public boolean onTouchEvent(MotionEvent event) {
 // TODO Auto-generated method stub
 // 獲取相對屏幕的坐標(biāo),即以屏幕左上角為原點(diǎn)
 x = event.getRawX();
 y = event.getRawY() - 25; // 25是系統(tǒng)狀態(tài)欄的高度
 Log.i("currP", "currX" + x + "====currY" + y);
 
 switch (event.getAction()) {
 case MotionEvent.ACTION_DOWN: // 捕獲手指觸摸按下動作
  // 獲取相對View的坐標(biāo),即以此View左上角為原點(diǎn)
  mTouchStartX = event.getX();
  mTouchStartY = event.getY();
  Log.i("startP", "startX" + mTouchStartX + "====startY"
   + mTouchStartY);
  break;
 case MotionEvent.ACTION_MOVE://捕獲手指觸摸移動動作      
  updateViewPosition(); 
  break;
 case MotionEvent.ACTION_UP://捕獲手指觸摸離開動作 
  updateViewPosition(); 
  mTouchStartX=mTouchStartY=0; 
  break;
 default:
  break;
 }
 return true;
 }
 
 private void updateViewPosition() {
 //更新浮動窗口位置參數(shù) 
 wmParams.x = (int)(x - mTouchStartX);
 wmParams.y = (int)(y - mTouchStartY);
 wm.updateViewLayout(this, wmParams);//刷新顯示  
 }
}

上面的wmParams變量(即WindowManager.LayoutParams)的存儲采用了extends Application的方式來創(chuàng)建全局變量,
示例代碼如下:

package com.example.suspend;
 
import android.app.Application;
import android.view.WindowManager;
 
public class MyApplication extends Application {
 private WindowManager.LayoutParams wmParams = new WindowManager.LayoutParams();
 
 @Override
 public void onCreate() {
 // TODO Auto-generated method stub
 super.onCreate();
 }
 
 public WindowManager.LayoutParams getMywmParams() {
 return wmParams;
 
 }
 
}

再來看一看Activity中的代碼:

package com.example.suspend;
 
import android.app.Activity;
import android.graphics.PixelFormat;
import android.os.Bundle;
import android.view.Gravity;
import android.view.WindowManager;
import android.view.WindowManager.LayoutParams;
 
public class MainActivity extends Activity {
 private WindowManager wm;
 private WindowManager.LayoutParams wmParams;
 private MyFloatView myFV;
 
 @Override
 protected void onCreate(Bundle savedInstanceState) {
 super.onCreate(savedInstanceState);
 setContentView(R.layout.activity_main);
 // 創(chuàng)建懸浮窗口
 createFloatView();
 }
 
 private void createFloatView() {
 myFV = new MyFloatView(getApplicationContext());
 myFV.setImageResource(R.drawable.ic_launcher);
 // 獲取WindowManager
 wm = (WindowManager) getApplicationContext().getSystemService("window");
 // 設(shè)置LayoutParams(全局變量)相關(guān)參數(shù)
 wmParams = ((MyApplication) getApplication()).getMywmParams();
 
 wmParams.type = LayoutParams.TYPE_PHONE;// 設(shè)置window type
 wmParams.format = PixelFormat.RGBA_8888;// 設(shè)置圖片格式,效果為背景透明
 // 設(shè)置Window flag
 wmParams.flags = LayoutParams.FLAG_NOT_TOUCH_MODAL
  | LayoutParams.FLAG_NOT_FOCUSABLE;
 /*
  * 
  * 下面的flags屬性的效果形同“鎖定”。
  * 
  * 懸浮窗不可觸摸,不接受任何事件,同時不影響后面的事件響應(yīng)。
  * 
  * wmParams.flags=LayoutParams.FLAG_NOT_TOUCH_MODAL
  * 
  * | LayoutParams.FLAG_NOT_FOCUSABLE
  * 
  * | LayoutParams.FLAG_NOT_TOUCHABLE;
  */
 wmParams.gravity = Gravity.LEFT | Gravity.TOP;// 調(diào)整懸浮窗口至左上角,便于調(diào)整坐標(biāo)
 // 以屏幕左上角為原點(diǎn),設(shè)置x、y初始值
 wmParams.x = 0;
 wmParams.y = 0;
 // 設(shè)置懸浮窗口長寬數(shù)據(jù)
 wmParams.width = 40;
 wmParams.height = 40;
 // 顯示myFloatView圖像
 wm.addView(myFV, wmParams);
 }
 
 @Override
 protected void onDestroy() {
 // TODO Auto-generated method stub
 super.onDestroy();
 // 在程序退出(Activity銷毀)時銷毀懸浮窗口
 wm.removeView(myFV);
 }
}

最后,別忘了在AndroidManifest.xml中添加權(quán)限:

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

關(guān)于“android如何實(shí)現(xiàn)桌面移動懸浮窗口”這篇文章就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,使各位可以學(xué)到更多知識,如果覺得文章不錯,請把它分享出去讓更多的人看到。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI