android悬浮提示
⑴ android 系统级的悬浮窗实现
当我们在使用的app的时候,如果需要实时观测到某个功能的实时进度并且不影响其他的操作的时候或者不影响使用其他应用的时候,系统级的悬浮球是个非常不错的选择。
public class QueueUpFloatService extends Service {
/**
* 启动服务并传值
*
* @param activity 启动服务的activity
* @param modeBean 数据对象
*/
public static void launchService(Activity activity, ModeBean modeBean) {
try {
Intent intent =new Intent(activity, QueueUpFloatService.class);
Bundle bundle =new Bundle();
bundle.putSerializable(KEY_MODEL, modeBean);
intent.putExtras(bundle);
activity.startService(intent);
}catch (Exception e) {
e.printStackTrace();
}
}
@Override
public void onCreate() {
super.onCreate();
}
@Override
public IBinder onBind(Intent intent) {
return null;
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
return super.onStartCommand(intent, flags, startId);
}
@Override
public void onDestroy() {
super.onDestroy();
}
}
@Override
public void onCreate() {
super.onCreate();
//加一点简单的动画
buttonScale = (ScaleAnimation) AnimationUtils.loadAnimation(this, R.anim.anim_float);
windowManager = (WindowManager) getSystemService(WINDOW_SERVICE);
layoutParams =new WindowManager.LayoutParams();
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
layoutParams.type = WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY;
}else {
layoutParams.type = WindowManager.LayoutParams.TYPE_PHONE;
}
layoutParams.format = PixelFormat.RGBA_8888;
layoutParams.gravity = Gravity.LEFT | Gravity.TOP;
layoutParams.flags = WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL | WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE;
layoutParams.width = ScreenUtils.dp2px(66);
layoutParams.height = ScreenUtils.dp2px(66);
layoutParams.x = ScreenUtils.getRealWidth() - ScreenUtils.dp2px(60);
layoutParams.y = ScreenUtils.deviceHeight() *2 /3;
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
ModeBean modeBean = (ModeBean) intent.getExtras().getSerializable(KEY_MODEL);
LayoutInflater layoutInflater = LayoutInflater.from(this);
floatView = layoutInflater.inflate(R.layout.view_float, null);
RelativeLayout rlFloatParent =floatView.findViewById(R.id.rl_float_parent);
rlFloatParent.startAnimation(buttonScale);
TextView tvIndex =floatView.findViewById(R.id.tv_queue_index);
tvIndex.setText(modeBean.title);
floatView.findViewById(R.id.iv_close_float).setOnClickListener(v -> stopSelf());
//修改悬浮球的滑动实现
floatView.setOnTouchListener(new FloatingOnTouchListener());
windowManager.addView(floatView, layoutParams);
return super.onStartCommand(intent, flags, startId);
}
private class View.OnTouchListener {
private int x;
private int y;
private long downTime;
@SuppressLint("ClickableViewAccessibility")
@Override
public boolean onTouch(View view, MotionEvent event) {
switch (event.getAction()) {
case MotionEvent.ACTION_DOWN:
downTime = System.currentTimeMillis();
x = (int) event.getRawX();
y = (int) event.getRawY();
break;
case MotionEvent.ACTION_MOVE:
int nowX = (int) event.getRawX();
int nowY = (int) event.getRawY();
int movedX = nowX -x;
int movedY = nowY -y;
x = nowX;
y = nowY;
layoutParams.x =layoutParams.x + movedX;
layoutParams.y =layoutParams.y + movedY;
windowManager.updateViewLayout(view, layoutParams);
break;
case MotionEvent.ACTION_UP:
/* *
* 这里根据手指按下和抬起的时间差来判断点击事件还是滑动事件
* */
if ((System.currentTimeMillis() -downTime) <200) {
//检测应用在前台还是后台
if (AppUtils.isAppIsInBackground()) {
AppUtils.moveToFront(CloseActivityUtils.activityList.get(CloseActivityUtils.activityList.size() -1).getClass());
} else {
//检测栈顶是否为SecondActivity 不是就打开SecondActivity
if (!CloseActivityUtils.activityList.get(CloseActivityUtils.activityList.size() -1)
.getClass().getSimpleName().contains("SecondActivity")) {
SecondActivity.launchActivity(CloseActivityUtils.activityList.get(CloseActivityUtils.activityList.size() -1));
}
}
}
break;
default:
break;
}
return false;
}
}
@Override
public void onDestroy() {
super.onDestroy();
if (null ==floatView) {
return;
}
windowManager.removeView(floatView);
windowManager=null;
}
⑵ 安卓手机的悬浮球怎么设置
若使用的是vivo手机,进入设置--快捷与辅助/更多设置--悬浮球中,将悬浮球开启即可。
关闭的方法:进入设置--快捷与辅助/更多设置--悬浮球中,将悬浮球关闭即可。
注:目丛桐前支持悬浮球功能的机型有:X60t、iQOO Z3、iQOO Neo5、S9/春念S9e、Y31s标准版、S7t、iQOO 7、Y31s、X60/X60 Pro/X60 Pro+、iQOO U3、Y30标准版、Y52s、iQOO U1x、S7e、Y3s、Y30、Y73s、iQOO 5系列、S7、iQOO U1、iQOO Z1x、X50系列、Y70s、iQOO Z1、iQOO Neo3、Y50、S6、NEX 3S、Z6、iQOO 3、X30 Pro/X30、iQOO Neo 855竞速版、U3、Y9s、Z5i、S5、Y5s、iQOO Neo 855版、U3x、NEX 3、Z5/Z5x、Y7s、X27/X27Pro、S1/S1Pro、Z3x、iQOO Pro/iQOO/iQOO Neo、NEX双屏版、Y81s、Y70、X23、Y97、Y91、NEX/NEX旗舰版、Z1、Z3i、U1、X21/X21i、Y71、Y85、X20、Y79、Y75s、Y83、扒郑困Y3、X20Plus(需升级至最新版本)。
可进入手机设置--快捷与辅助/更多设置--查看是否有“悬浮球”功能。
⑶ 安卓手机开启无障碍功能与悬浮窗的方法
怎么开启辅助服务?
使用优Q的功能必须开服辅助服务,辅助服务是安卓官方提供的功能。
打开优Q点击【我的】点击权限设置,开启即可。如果无法快速开启,请参考如下的方法。
部分手机辅助可通过快捷键开启,优Q仍提示未开启,将辅助服务关闭一下再打开就可以了。
如果仍然无效,请先关闭优Q然后重启手机。
VIVO设置-更多设置-辅助功能-开启-优Q
OPPO设置-其他设置-辅助功能-开启-优Q
小米设置-更多设置-无障碍-开启-优Q
魅族设置-辅助功能-无障碍-开启-优Q
华为设置-高级设置-辅助功能-开启-优Q
联想设置-高级设置-辅助功能-开启-优Q
三星设置-辅助功能-开启-优Q
乐视设置-辅助功能-开启-优Q
360设置-辅助功能-开启优Q
锤子设置-全局高级设置-辅助功能-开启-优Q
一加设置-其他高级设置-无障碍-开启-优Q
怎么开启悬浮窗?
部分高版本手机打开优Q会自动跳转到悬浮窗开关。
打开优Q点击【我的】点击权限设置,开启悬浮窗即可。如果无法快速开启,请参考如下的方法。
不同的手机打开悬浮窗权限的方法不一样,常见的方法是在应用权限管理或手机管家进行设置。
VIVO
系统6.0:设置-更多设置-权限管理-优Q-开启悬浮窗系统
5.0:i管家-软件管理-悬浮窗管理-开启-"优Q
OPPO手机管家-隐私权限-悬浮窗权限管理-开启
优Q
小米安全中心-应用权限管理-优Q-允许显示悬浮窗
魅族手机管家-应用权限管理-优Q-开启桌面悬浮窗
华为设置权限管理-悬浮窗开启优Q
联想安全中心-权限管理-显示悬浮窗-
允许"优Q显示悬浮窗
三星设置-应用程序-应用程序管理器-右上角更多-
可出现在顶部的应用程序-开启优Q
乐视设置-隐私权限-应用权限管理器-应用列表
优Q-允许显示悬浮窗
360安全卫士-软件管理-权限管理-优Q-
允许显示悬浮窗
锤子设置-安全中心-应用程序权限管理-悬浮窗-
开启优Q
一加设置-应用程序-特殊访问权限-出现在其他应用上--找到优Q打开即可
三条鱼软件官网
⑷ Android实现类似qq,微信消息悬浮窗通知
实现方法:(需要开启悬浮窗通知权限、允许应用在其他应用上显示)
悬挂式Notification,他是5.0中新增的,也就是API中的Headsup的Notification,可以在不打断用户操作的时候,给用户通知
注意:在某些rom下使用headsup并不会显示桌面悬浮窗,而是直接跳转到相应的界面,亲测华为,小米都是这种情况,这种情况下需要自己实现悬浮窗
具体实现:
⑸ android8.0之悬浮窗和通知栏
悬浮窗:
使用场景:例如微信在视频的时候,点击Home键,视频小窗口仍然会在屏幕上显示;
注意事项:
1、一般需要在后台进行操作的时候才需要悬浮窗,这样悬浮窗才有意义;
2、API Level >= 23的时候,需要在AndroidManefest.xml文件中声明权限SYSTEM_ALERT_WINDOW才能在其他应用上绘制控件。
<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />;除了这个权限外,我们还需要在系统设置里面对本应用进行设置悬浮窗权限。该权限在应用中需要启动Settings.ACTION_MANAGE_OVERLAY_PERMISSION来让用户手动设置权限:startActivityForResult(new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION, Uri.parse("package:" + getPackageName())), REQUEST_CODE);
3、LayoutParam设置:LayoutParam里的type变量。这个变量是用来指定窗口类型的。在设置这个变量时,需要注意一个坑,那就是需要对不同版本的Android系统进行适配。
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
layoutParams.type = WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY;
} else {
layoutParams.type = WindowManager.LayoutParams.TYPE_PHONE;
};在Android 8.0之前,悬浮窗口设置可以为TYPE_PHONE,这种类型是用于提供用户交互操作的非应用窗口。
而Android 8.0对系统和API行为做了修改,包括使用SYSTEM_ALERT_WINDOW权限的应用无法再使用一下窗口类型来在其他应用和窗口上方显示提醒窗口:
- TYPE_PHONE
- TYPE_PRIORITY_PHONE
- TYPE_SYSTEM_ALERT
- TYPE_SYSTEM_OVERLAY
- TYPE_SYSTEM_ERROR
如果需要实现在其他应用和窗口上方显示提醒窗口,那么必须该为TYPE_APPLICATION_OVERLAY的新类型;
如果在Android 8.0以上版本仍然使用TYPE_PHONE类型的悬浮窗口,则会出现如下异常信息:
android.view.WindowManager$BadTokenException: Unable to add window android.view.ViewRootImpl$W@f8ec928 -- permission denied for window type 2002;
具体实现:
1、Activity:
public void startFloatingService(View view) {
...
if (!Settings.canDrawOverlays(this)) {
Toast.makeText(this, "当前无权限,请授权", Toast.LENGTH_SHORT);
startActivityForResult(new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION, Uri.parse("package:" + getPackageName())), 0);
} else {
startService(new Intent(MainActivity.this, FloatingService.class));
}
}
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
if (requestCode == 0) {
if (!Settings.canDrawOverlays(this)) {
Toast.makeText(this, "授权失败", Toast.LENGTH_SHORT).show();
} else {
Toast.makeText(this, "授权成功", Toast.LENGTH_SHORT).show();
startService(new Intent(MainActivity.this, FloatingService.class));
}
}
}
2、service:
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
showFloatingWindow();
return super.onStartCommand(intent, flags, startId);
}
private void showFloatingWindow() {
if (Settings.canDrawOverlays(this)) {
// 获取WindowManager服务
WindowManager windowManager = (WindowManager) getSystemService(WINDOW_SERVICE);
// 新建悬浮窗控件
Button button = new Button(getApplicationContext());
button.setText("Floating Window");
button.setBackgroundColor(Color.BLUE);
// 设置LayoutParam
WindowManager.LayoutParams layoutParams = new WindowManager.LayoutParams();
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
layoutParams.type = WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY;
} else {
layoutParams.type = WindowManager.LayoutParams.TYPE_PHONE;
}
layoutParams.format = PixelFormat.RGBA_8888;
layoutParams.width = 500;
layoutParams.height = 100;
layoutParams.x = 300;
layoutParams.y = 300;
// 将悬浮窗控件添加到WindowManager
windowManager.addView(button, layoutParams);
}
}
效果展示: