Android浮動窗口實現(xiàn)原理及代碼實例
1.浮動窗口的實現(xiàn)原理
看到上圖的那個小Android圖標了吧,它不會被其他組建遮擋,也可以響應用戶的點擊和拖動事件,它的顯示和消失由WindowManager直接管理,它就是Android浮動窗口。Android浮動窗口的實現(xiàn)主要是靠WindowManager這個類。通過WindowManager類的addView(),updateViewLayout(),removeView()這幾個方法,我們可以直接在Window中添加,更新,移除View。
2.浮動窗口實現(xiàn)的具體步驟
1)既然浮動窗口的實現(xiàn)依賴與WindowManager,那么毫無疑問,我們得先拿到WindowManger對象??紤]到浮動窗口通常在應用程序退出后依然顯示,所以我們需要在Service中實現(xiàn)浮動窗口的添加和更新,當然別忘了提供給用戶一個取消浮動窗口的功能。
2)定義你要顯示的View??梢栽诓季治募卸x,也可以自定義視圖。
3)設置必要的參數(shù),其中有幾個比較重要的參數(shù)需要設置,具體請參考下面的代碼。
4)將View添加到Window中,接收并處理事件,更新View。
5)在Manifest中加入對應的權限。<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW"/>
3.浮動窗口實現(xiàn)代碼
package com.spreadst.floatwindow;
import android.app.Service;
import android.content.Context;
import android.content.Intent;
import android.graphics.PixelFormat;
import android.os.IBinder;
import android.util.Log;
import android.view.Gravity;
import android.view.LayoutInflater;
import android.view.MotionEvent;
import android.view.View;
import android.view.WindowManager;
public class FloatWindowService extends Service {
private static final TAG = "FloatWindowService";
private WindowManager mWindowManager;
private WindowManager.LayoutParams mLayoutParams;
private LayoutInflater mLayoutInflater;
private View mFloatView;
private int mCurrentX;
private int mCurrentY;
private static int mFloatViewWidth = 50;
private static int mFloatViewHeight = 80;
@Override
public void onCreate() {
// TODO Auto-generated method stub
super.onCreate();
//初始化WindowManager對象和LayoutInflater對象
mWindowManager = (WindowManager) getApplicationContext().getSystemService(Context.WINDOW_SERVICE);
mLayoutInflater = LayoutInflater.from(this);
}
@Override
public void onStart(Intent intent, int startId) {
// TODO Auto-generated method stub
super.onStart(intent, startId);
Log.i(TAG, "onStart()");
createView();
}
private void createView() {
// TODO Auto-generated method stub
//加載布局文件
mFloatView = mLayoutInflater.inflate(R.layout.main, null);
//為View設置監(jiān)聽,以便處理用戶的點擊和拖動
mFloatView.setOnTouchListener(new OnFloatViewTouchListener());
/*為View設置參數(shù)*/
mLayoutParams = new WindowManager.LayoutParams();
//設置View默認的擺放位置
mLayoutParams.gravity = Gravity.LEFT | Gravity.TOP;
//設置window type
mLayoutParams.type = WindowManager.LayoutParams.TYPE_PHONE;
//設置背景為透明
mLayoutParams.format = PixelFormat.RGBA_8888;
//注意該屬性的設置很重要,F(xiàn)LAG_NOT_FOCUSABLE使浮動窗口不獲取焦點,若不設置該屬性,屏幕的其它位置點擊無效,應為它們無法獲取焦點
mLayoutParams.flags = WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE;
//設置視圖的顯示位置,通過WindowManager更新視圖的位置其實就是改變(x,y)的值
mCurrentX = mLayoutParams.x = 50;
mCurrentY = mLayoutParams.y = 50;
//設置視圖的寬、高
mLayoutParams.width = 100;
mLayoutParams.height = 100;
//將視圖添加到Window中
mWindowManager.addView(mFloatView, mLayoutParams);
}
/*由于直接startService(),因此該方法沒用*/
@Override
public IBinder onBind(Intent intent) {
// TODO Auto-generated method stub
return null;
}
/*該方法用來更新視圖的位置,其實就是改變(LayoutParams.x,LayoutParams.y)的值*/
private void updateFloatView() {
mLayoutParams.x = mCurrentX;
mLayoutParams.y = mCurrentY;
mWindowManager.updateViewLayout(mFloatView, mLayoutParams);
}
/*處理視圖的拖動,這里只對Move事件做了處理,用戶也可以對點擊事件做處理,例如:點擊浮動窗口時,啟動應用的主Activity*/
private class OnFloatViewTouchListener implements View.OnTouchListener {
@Override
public boolean onTouch(View v, MotionEvent event) {
// TODO Auto-generated method stub
Log.i(TAG, "mCurrentX: " + mCurrentX + ",mCurrentY: "
+ mCurrentY + ",mFloatViewWidth: " + mFloatViewWidth
+ ",mFloatViewHeight: " + mFloatViewHeight);
/*
* getRawX(),getRawY()這兩個方法很重要。通常情況下,我們使用的是getX(),getY()來獲得事件的觸發(fā)點坐標,
* 但getX(),getY()獲得的是事件觸發(fā)點相對與視圖左上角的坐標;而getRawX(),getRawY()獲得的是事件觸發(fā)點
* 相對與屏幕左上角的坐標。由于LayoutParams中的x,y是相對與屏幕的,所以需要使用getRawX(),getRawY()。
*/
mCurrentX = (int) event.getRawX() - mFloatViewWidth;
mCurrentY = (int) event.getRawY() - mFloatViewHeight;
int action = event.getAction();
switch (action) {
case MotionEvent.ACTION_DOWN:
break;
case MotionEvent.ACTION_MOVE:
updateFloatView();
break;
case MotionEvent.ACTION_UP:
break;
}
return true;
}
}
}
4.如何只在Launcher界面顯示浮動窗口
大家應該都熟悉360安全衛(wèi)士的浮動窗口,它的浮動窗口只會在Launcher界面顯示,當用戶切到其它界面,浮動窗口自動被移除了。
要實現(xiàn)該功能,我們就必須知道當前所在的界面,如果只去監(jiān)聽Activity的category,那么我們只能知道什么時候進入Launcher界面了,卻無法知道是否離開了Launcher界面。那么360是如何實現(xiàn)該功能呢?大家可以反編譯一下它的代碼。這里提供一種可行的方法,我們的目前其實很簡單,就是要知道當前的Activity是否是Launcher界面的Activity。由于Activity是以堆棧的形式被管理的,因此,只要我們查看棧頂?shù)腁ctivity是否是Launcher的Activity即可。要獲取Activity的Task信息,需要在Manifest中添加對應權限,<uses-permission android:name = “android.permission.GET_TASKS”/>。
private String getTopActivity(Context context) {
//獲取ActivityManager對象
ActivityManager manager = (ActivityManager)getSystemService(ACTIVITY_SERVICE) ;
/*
* 拿到當前正在運行的Task列表,該列表按照最近使用的時間順序排列,其中的參數(shù)表示需要返回的最大列表項數(shù)目。
* 這里我們只需要拿到處于onResume狀態(tài)的Activity所在的Task。
*/
List<RunningTaskInfo> runningTaskInfos = manager.getRunningTasks(1) ;
if(runningTaskInfos != null) {
//拿到該task中的棧頂Activity
return (runningTaskInfos.get(0).topActivity).toString() ;
} else {
return null;
}
}
以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關文章
Android Listview notifyDataSetChanged() 不起作用的
這篇文章主要介紹了Android Listview notifyDataSetChanged()不起作用的解決方法,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友參考下吧2024-08-08
Android實現(xiàn)滑動加載數(shù)據(jù)的方法
這篇文章主要介紹了Android實現(xiàn)滑動加載數(shù)據(jù)的方法,實例分析了Android通過滑動實現(xiàn)動態(tài)加載數(shù)據(jù)的技巧,具有一定參考借鑒價值,需要的朋友可以參考下2015-07-07
Android使用開源框架ANDROID-IMAGE-INDICATOR實現(xiàn)圖片輪播部署
這篇文章主要為大家詳細介紹了Android使用開源框架ANDROID-IMAGE-INDICATOR實現(xiàn)圖片輪播部署,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-01-01
Android 監(jiān)聽軟鍵盤狀態(tài)的實例詳解
這篇文章主要介紹了Android 監(jiān)聽軟鍵盤狀態(tài)的實例詳解的相關資料,希望通過本文能掌握這樣的知識,需要的朋友可以參考下2017-09-09
Android中使用Kotlin實現(xiàn)一個簡單的登錄界面
Kotlin 是一種在 Java 虛擬機上運行的靜態(tài)類型編程語言,被稱之為 Android 世界的Swift,由 JetBrains 設計開發(fā)并開源。接下來本文通過實例代碼給大家講解Android中使用Kotlin實現(xiàn)一個簡單的登錄界面,一起看看吧2017-09-09

