我尝试将 Chathead 类型在我的应用上通过Android服务过度绘制,如下图所示。
这个聊天头应用适用于Android版本5或更低版本(kitkat,棒棒糖等)。但我在Marshmallow及更高版本中尝试它,然后得到此错误。
android.view.WindowManager$BadTokenException: Unable to add window
android.view.ViewRootImpl$W@48f5767 -- permission denied for window type 2010
聊天主管代码
注意:我从Android服务
调用此功能权限列表:
<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />
<uses-permission android:name="android.permission.BIND_DEVICE_ADMIN" />
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
private WindowManager windowManager;
private ImageView chatHead;
WindowManager.LayoutParams params;
@Override
public void onCreate() {
super.onCreate();
windowManager = (WindowManager) getSystemService(WINDOW_SERVICE);
chatHead = new ImageView(this);
chatHead.setImageResource(R.drawable.settingsicon);
params = new WindowManager.LayoutParams(
WindowManager.LayoutParams.WRAP_CONTENT,
WindowManager.LayoutParams.WRAP_CONTENT,
WindowManager.LayoutParams.TYPE_PHONE,
WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE,
PixelFormat.TRANSLUCENT);
params.gravity = Gravity.TOP | Gravity.LEFT;
params.x = 0;
params.y = 100;
windowManager.addView(chatHead, params);
chatHead.setOnTouchListener(new View.OnTouchListener() {
private int initialX;
private int initialY;
private float initialTouchX;
private float initialTouchY;
@Override
public boolean onTouch(View v, MotionEvent event) {
switch (event.getAction()) {
case MotionEvent.ACTION_DOWN:
initialX = params.x;
initialY = params.y;
initialTouchX = event.getRawX();
initialTouchY = event.getRawY();
return true;
case MotionEvent.ACTION_UP:
return true;
case MotionEvent.ACTION_MOVE:
params.x = initialX + (int) (event.getRawX() - initialTouchX);
params.y = initialY + (int) (event.getRawY() - initialTouchY);
windowManager.updateViewLayout(chatHead, params);
return true;
}
return false;
}
}
}
如何在Mashmallow或更高版本的Android版本上解决此错误?
答案 0 :(得分:2)
在显示聊天头之前,请调用此方法以获得许可:
public void addOverlay() {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
if (!Settings.canDrawOverlays(this)) {
askedForOverlayPermission = true;
Intent intent = new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION, Uri.parse("package:" + getPackageName()));
startActivityForResult(intent, OVERLAY_PERMISSION_CODE);
}
}
}
之后在权限结果上如果用户允许权限,那么只有你可以显示如下的聊天头:
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
if (requestCode == OVERLAY_PERMISSION_CODE) {
askedForOverlayPermission = false;
if (Settings.canDrawOverlays(this)) {
// SYSTEM_ALERT_WINDOW permission not granted...
//Toast.makeText(MyProtector.getContext(), "ACTION_MANAGE_OVERLAY_PERMISSION Permission Granted", Toast.LENGTH_SHORT).show();
Intent serviceIntent = new Intent(Homepage.this, ChatHeadService.class);
serviceIntent.putExtra("removeUserId", friendId);
startService(serviceIntent);
} else {
Toast.makeText(MyProtector.getContext(), "ACTION_MANAGE_OVERLAY_PERMISSION Permission Denied", Toast.LENGTH_SHORT).show();
}
}
让我知道它是否对你有所帮助......祝你好运。
答案 1 :(得分:0)
从版本M(api level 23)开始,Android具有在运行时请求权限的新算法。见这里:https://developer.android.com/training/permissions/requesting.html
答案 2 :(得分:0)
似乎您的addView()方法失败。
我已经解决了该问题(在某些情况下),对于8.0以下的Android SDK使用WindowManager.LayoutParams.TYPE_PHONE
,而8.0及以上的版本使用WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY
。
您可以添加一些自动选择来授予主要兼容性(取决于活动的SDK平台),但是由于Google Play不允许8.0以下的APK,因此显然没有必要。
希望对您有所帮助。
答案 3 :(得分:0)
对我有帮助
Map<fooBarInterface, int>
答案 4 :(得分:-1)
在Orea(API级别26)中不推荐使用type = TYPE_PHONE的叠加层。 非系统应用程序仅支持TYPE_APPLICATION_OVERLAY。