时间:2020-12-11 20:39:35 | 栏目:Android代码 | 点击:次
实现方法:(需要开启悬浮窗通知权限、允许应用在其他应用上显示)
一.利用headsup
悬挂式Notification,他是5.0中新增的,也就是API中的Headsup的Notification,可以在不打断用户操作的时候,给用户通知
二.使用Window创建悬浮窗
当window属性设置为FLAGE_NOT_FOCUSABLE
表示不需要获取焦点,也不需要接受各种输入事件,此标记会同时启用FLAGE_NOT_TOUCH_MODEL,
最终事件会直接传递给下层具有焦点的Widow
FLAGE_NOT_TOUCH_MODE
在此模式下,系统会将当前Window区域以外的单击事件传递给底层的Window,当前Window区域以内的单击事件则自己处理,如果不开启此标记,其他Window将无法接收到单击事件
FLAGE_SHOW_WHEN_LOCKED
开启此模式可以让Window显示在锁屏的桌面上(不是所有的rom都支持)
widow类型设置,每个window都有对应的z-ordered,层级大的会覆盖层级小的,Window有三种类型:应用Window(1-99),子Window(1000-1999),系统Window(2000-2999),因此,桌面悬浮窗需要系统Window,设置系统级的type有很多值,经常用的是TYPE_SYSTEM_ALERT,此设置需要添加权限<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />
但是经测试当类型设置为TYPE_TOAST的时候是不需要添加权限的( 部分rom仍旧需要权限(比如小米!!!!)小米真的是开发人员的噩梦! )
提示:6.0之后需要动态权限
注意:在某些rom下使用headsup并不会显示桌面悬浮窗,而是直接跳转到相应的界面,亲测华为,小米都是这种情况,这种情况下需要自己实现悬浮窗
具体实现:
利用headsup
manager=(NotificationManager)getSystemService(NOTIFICATION_SERVICE); notification = new NotificationCompat.Builder(this) .setVisibility(Notification.VISIBILITY_PRIVATE) .setSmallIcon(R.mipmap.ic_launcher) .setLargeIcon(BitmapFactory.decodeResource(getResources(),R.mipmap.ic_launcher)) .setFullScreenIntent(pendingIntent, false) .setContentTitle("标题") .setContentText("内容") .build(); manager.notify(1, notification);
使用Window
``private void initWindowManager(){ wm = (WindowManager) getApplicationContext().getSystemService( Context.WINDOW_SERVICE); params = new WindowManager.LayoutParams(); // 设置window type params.type = WindowManager.LayoutParams.TYPE_PHONE; /* * 如果设置为params.type = WindowManager.LayoutParams.TYPE_PHONE; 那么优先级会降低一些, * 即拉下通知栏不可见 */ params.format = PixelFormat.RGBA_8888; // 设置图片格式,效果为背景透明 // 设置Window flag params.flags = WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL | WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE | WindowManager.LayoutParams.FLAG_SHOW_WHEN_LOCKED; /* * 下面的flags属性的效果形同“锁定”。 悬浮窗不可触摸,不接受任何事件,同时不影响后面的事件响应。 * wmParams.flags=LayoutParams.FLAG_NOT_TOUCH_MODAL | * LayoutParams.FLAG_NOT_FOCUSABLE | LayoutParams.FLAG_NOT_TOUCHABLE; */ // 设置悬浮窗的长得宽 params.width = wm.getDefaultDisplay().getWidth(); params.height = 200; params.gravity = Gravity.LEFT | Gravity.TOP;} private void createFloatView(String str) { if (btn_floatView == null){ btn_floatView = new Button(getApplicationContext()); wmTag = true; } btn_floatView.setText(str); Log.i(TAG, "createFloatView: "+str); // 设置悬浮窗的Touch监听 btn_floatView.setOnTouchListener(new View.OnTouchListener() { int lastX, lastY; int paramX, paramY; public boolean onTouch(View v, MotionEvent event) { switch (event.getAction()) { case MotionEvent.ACTION_DOWN: if (MainActivity.lifeTag == 1) { Intent intent = new Intent(DataService.this, MainActivity.class); startActivity(intent); } wm.removeViewImmediate(btn_floatView); btn_floatView = null; break; case MotionEvent.ACTION_MOVE: break; } return true; } }); if (wmTag){ wm.addView(btn_floatView, params); wmTag = false; }else { wm.updateViewLayout(btn_floatView,params); } }```
效果图
总结