Android浮動(dòng)窗口實(shí)現(xiàn)原理及代碼實(shí)例
1.浮動(dòng)窗口的實(shí)現(xiàn)原理
看到上圖的那個(gè)小Android圖標(biāo)了吧,它不會(huì)被其他組建遮擋,也可以響應(yīng)用戶的點(diǎn)擊和拖動(dòng)事件,它的顯示和消失由WindowManager直接管理,它就是Android浮動(dòng)窗口。Android浮動(dòng)窗口的實(shí)現(xiàn)主要是靠WindowManager這個(gè)類。通過WindowManager類的addView(),updateViewLayout(),removeView()這幾個(gè)方法,我們可以直接在Window中添加,更新,移除View。
2.浮動(dòng)窗口實(shí)現(xiàn)的具體步驟
1)既然浮動(dòng)窗口的實(shí)現(xiàn)依賴與WindowManager,那么毫無疑問,我們得先拿到WindowManger對(duì)象??紤]到浮動(dòng)窗口通常在應(yīng)用程序退出后依然顯示,所以我們需要在Service中實(shí)現(xiàn)浮動(dòng)窗口的添加和更新,當(dāng)然別忘了提供給用戶一個(gè)取消浮動(dòng)窗口的功能。
2)定義你要顯示的View??梢栽诓季治募卸x,也可以自定義視圖。
3)設(shè)置必要的參數(shù),其中有幾個(gè)比較重要的參數(shù)需要設(shè)置,具體請(qǐng)參考下面的代碼。
4)將View添加到Window中,接收并處理事件,更新View。
5)在Manifest中加入對(duì)應(yīng)的權(quán)限。<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW"/>
3.浮動(dòng)窗口實(shí)現(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對(duì)象和LayoutInflater對(duì)象 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設(shè)置監(jiān)聽,以便處理用戶的點(diǎn)擊和拖動(dòng) mFloatView.setOnTouchListener(new OnFloatViewTouchListener()); /*為View設(shè)置參數(shù)*/ mLayoutParams = new WindowManager.LayoutParams(); //設(shè)置View默認(rèn)的擺放位置 mLayoutParams.gravity = Gravity.LEFT | Gravity.TOP; //設(shè)置window type mLayoutParams.type = WindowManager.LayoutParams.TYPE_PHONE; //設(shè)置背景為透明 mLayoutParams.format = PixelFormat.RGBA_8888; //注意該屬性的設(shè)置很重要,F(xiàn)LAG_NOT_FOCUSABLE使浮動(dòng)窗口不獲取焦點(diǎn),若不設(shè)置該屬性,屏幕的其它位置點(diǎn)擊無效,應(yīng)為它們無法獲取焦點(diǎn) mLayoutParams.flags = WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE; //設(shè)置視圖的顯示位置,通過WindowManager更新視圖的位置其實(shí)就是改變(x,y)的值 mCurrentX = mLayoutParams.x = 50; mCurrentY = mLayoutParams.y = 50; //設(shè)置視圖的寬、高 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; } /*該方法用來更新視圖的位置,其實(shí)就是改變(LayoutParams.x,LayoutParams.y)的值*/ private void updateFloatView() { mLayoutParams.x = mCurrentX; mLayoutParams.y = mCurrentY; mWindowManager.updateViewLayout(mFloatView, mLayoutParams); } /*處理視圖的拖動(dòng),這里只對(duì)Move事件做了處理,用戶也可以對(duì)點(diǎn)擊事件做處理,例如:點(diǎn)擊浮動(dòng)窗口時(shí),啟動(dòng)應(yīng)用的主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()這兩個(gè)方法很重要。通常情況下,我們使用的是getX(),getY()來獲得事件的觸發(fā)點(diǎn)坐標(biāo), * 但getX(),getY()獲得的是事件觸發(fā)點(diǎn)相對(duì)與視圖左上角的坐標(biāo);而getRawX(),getRawY()獲得的是事件觸發(fā)點(diǎn) * 相對(duì)與屏幕左上角的坐標(biāo)。由于LayoutParams中的x,y是相對(duì)與屏幕的,所以需要使用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界面顯示浮動(dòng)窗口
大家應(yīng)該都熟悉360安全衛(wèi)士的浮動(dòng)窗口,它的浮動(dòng)窗口只會(huì)在Launcher界面顯示,當(dāng)用戶切到其它界面,浮動(dòng)窗口自動(dòng)被移除了。
要實(shí)現(xiàn)該功能,我們就必須知道當(dāng)前所在的界面,如果只去監(jiān)聽Activity的category,那么我們只能知道什么時(shí)候進(jìn)入Launcher界面了,卻無法知道是否離開了Launcher界面。那么360是如何實(shí)現(xiàn)該功能呢?大家可以反編譯一下它的代碼。這里提供一種可行的方法,我們的目前其實(shí)很簡(jiǎn)單,就是要知道當(dāng)前的Activity是否是Launcher界面的Activity。由于Activity是以堆棧的形式被管理的,因此,只要我們查看棧頂?shù)腁ctivity是否是Launcher的Activity即可。要獲取Activity的Task信息,需要在Manifest中添加對(duì)應(yīng)權(quán)限,<uses-permission android:name = “android.permission.GET_TASKS”/>。
private String getTopActivity(Context context) { //獲取ActivityManager對(duì)象 ActivityManager manager = (ActivityManager)getSystemService(ACTIVITY_SERVICE) ; /* * 拿到當(dāng)前正在運(yùn)行的Task列表,該列表按照最近使用的時(shí)間順序排列,其中的參數(shù)表示需要返回的最大列表項(xiàng)數(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; } }
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- android 實(shí)現(xiàn)按鈕浮動(dòng)在鍵盤上方的實(shí)例代碼
- Android實(shí)現(xiàn)支付寶螞蟻森林水滴浮動(dòng)效果
- Android實(shí)現(xiàn)圖片浮動(dòng)隨意拖拽效果
- Android 浮動(dòng)編輯框的具體實(shí)現(xiàn)代碼
- Android利用浮動(dòng)窗口提示用戶操作
- 安卓(android)仿電商app商品詳情頁按鈕浮動(dòng)效果
- Android自定義ViewGroup實(shí)現(xiàn)標(biāo)簽浮動(dòng)效果
- Android應(yīng)用中制作選中后圖標(biāo)變大浮動(dòng)效果的代碼分享
相關(guān)文章
Android Listview notifyDataSetChanged() 不起作用的
這篇文章主要介紹了Android Listview notifyDataSetChanged()不起作用的解決方法,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2024-08-08Android實(shí)現(xiàn)滑動(dòng)加載數(shù)據(jù)的方法
這篇文章主要介紹了Android實(shí)現(xiàn)滑動(dòng)加載數(shù)據(jù)的方法,實(shí)例分析了Android通過滑動(dòng)實(shí)現(xiàn)動(dòng)態(tài)加載數(shù)據(jù)的技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-07-07RecyclerVIew實(shí)現(xiàn)懸浮吸頂效果
這篇文章主要為大家詳細(xì)介紹了RecyclerVIew實(shí)現(xiàn)懸浮吸頂效果,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-09-09Android使用開源框架ANDROID-IMAGE-INDICATOR實(shí)現(xiàn)圖片輪播部署
這篇文章主要為大家詳細(xì)介紹了Android使用開源框架ANDROID-IMAGE-INDICATOR實(shí)現(xiàn)圖片輪播部署,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-01-01Android 監(jiān)聽軟鍵盤狀態(tài)的實(shí)例詳解
這篇文章主要介紹了Android 監(jiān)聽軟鍵盤狀態(tài)的實(shí)例詳解的相關(guān)資料,希望通過本文能掌握這樣的知識(shí),需要的朋友可以參考下2017-09-09Android中使用Kotlin實(shí)現(xiàn)一個(gè)簡(jiǎn)單的登錄界面
Kotlin 是一種在 Java 虛擬機(jī)上運(yùn)行的靜態(tài)類型編程語言,被稱之為 Android 世界的Swift,由 JetBrains 設(shè)計(jì)開發(fā)并開源。接下來本文通過實(shí)例代碼給大家講解Android中使用Kotlin實(shí)現(xiàn)一個(gè)簡(jiǎn)單的登錄界面,一起看看吧2017-09-09