色综合图-色综合图片-色综合图片二区150p-色综合图区-玖玖国产精品视频-玖玖香蕉视频

您的位置:首頁技術文章
文章詳情頁

Android浮動窗口實現(xiàn)原理及代碼實例

瀏覽:3日期:2022-09-23 09:12:38

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。可以在布局文件中定義,也可以自定義視圖。

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; } }

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

標簽: Android
相關文章:
主站蜘蛛池模板: 亚洲欧美日本国产综合在线 | 日韩在线视频免费不卡一区 | 亚洲一区www | 先锋影音xfyy5566男人资源 | 在线观看日本视频免费 | 日韩一级一片 | 欧美.亚洲.日本一区二区三区 | 亚洲欧美日韩国产综合 | 免费一级毛片在线播放视频 | 亚洲 欧美 丝袜 | 91香蕉国产 | 久久精品在线免费观看 | 精品国产欧美一区二区 | 亚洲精品久久久久久久福利 | 久久99精品久久久久久野外 | 香蕉超级碰碰碰97视频在线观看 | 久久伊人男人的天堂网站 | 视频一区精品 | 萌白酱在线喷水福利视频 | 欧美视频精品在线观看 | 欧美成年 | 在线亚洲黄色 | 国产91精品久久久久999 | 亚洲国产第一区二区三区 | 九九免费精品视频 | 在线欧美精品二区三区 | 亚洲经典三级 | 亚洲国产精品91 | 免费看成人毛片日本久久 | 亚洲欧美精品久久 | 成人免费观看一区二区 | 成人国产在线视频在线观看 | 日韩欧美视频一区二区 | 免费人欧美成又黄又爽的视频 | 国内视频自拍 | 精品乱人伦一区二区 | 玖玖在线免费视频 | 真实国产普通话对白乱子子伦视频 | 国产高清免费 | 亚洲一级高清在线中文字幕 | 亚洲精品在线视频观看 |