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);
}
}
效果展示: