Android API 26上的系统警报窗口权限未按预期工作。窗口类型2002的权限被拒绝

Android API 26上的系统警报窗口权限未按预期工作。窗口类型2002的权限被拒绝,android,permissions,android-8.0-oreo,Android,Permissions,Android 8.0 Oreo,我正在使用覆盖权限在我的应用程序中显示某些信息。在API23-25上运行它可以正常工作(根据 )。(非常感谢ceph3us!) 在API 26上尝试同样的操作时,我得到了一个错误,基本上是在调用 windowManager.addView(frameLayout, params); 谷歌是否改变了方式,覆盖工作?你知道吗,如何在Android 8(Oreo)API 26中将我的文本作为覆盖层显示在屏幕上? 谢谢你的想法 这是错误日志: 08-24 16:41:56.730 2615-2615/

我正在使用覆盖权限在我的应用程序中显示某些信息。在API23-25上运行它可以正常工作(根据

)。(非常感谢ceph3us!)

在API 26上尝试同样的操作时,我得到了一个错误,基本上是在调用

windowManager.addView(frameLayout, params);
谷歌是否改变了方式,覆盖工作?你知道吗,如何在Android 8(Oreo)API 26中将我的文本作为覆盖层显示在屏幕上? 谢谢你的想法

这是错误日志:

08-24 16:41:56.730 2615-2615/net.zwittscha.testoverlay E/AndroidRuntime: FATAL EXCEPTION: main
Process: net.zwittscha.testoverlay, PID: 2615
java.lang.RuntimeException: Unable to start activity ComponentInfo{net.zwittscha.testoverlay/net.zwittscha.testoverlay.MainActivity}: 
                android.view.WindowManager$BadTokenException: Unable to add window android.view.ViewRootImpl$W@6fa0089 -- 
                permission denied for window type 2002
 at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2817)
 at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2892)
 at android.app.ActivityThread.-wrap11(Unknown Source:0)
 at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1593)
 at android.os.Handler.dispatchMessage(Handler.java:105)
 at android.os.Looper.loop(Looper.java:164)
 at android.app.ActivityThread.main(ActivityThread.java:6541)
 at java.lang.reflect.Method.invoke(Native Method)
 at com.android.internal.os.Zygote$MethodAndArgsCaller.run(Zygote.java:240)
 at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:767)
            Caused by: android.view.WindowManager$BadTokenException: 
            Unable to add window android.view.ViewRootImpl$W@6fa0089 -- 
            permission denied for window type 2002
 at android.view.ViewRootImpl.setView(ViewRootImpl.java:789)
 at android.view.WindowManagerGlobal.addView(WindowManagerGlobal.java:356)
 at android.view.WindowManagerImpl.addView(WindowManagerImpl.java:92)
 at net.zwittscha.testoverlay.MainActivity.createOnTopView(MainActivity.java:46)
 at net.zwittscha.testoverlay.MainActivity.checkDrawOverlayPermission(MainActivity.java:66)
 at net.zwittscha.testoverlay.MainActivity.onCreate(MainActivity.java:28)
 at android.app.Activity.performCreate(Activity.java:6975)
 at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1213)
 at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2770)
 at android.app.ActivityThread.handleLaunchActivity(ActivityThread.java:2892) 
 at android.app.ActivityThread.-wrap11(Unknown Source:0) 
 at android.app.ActivityThread$H.handleMessage(ActivityThread.java:1593) 
 at android.os.Handler.dispatchMessage(Handler.java:105) 
 at android.os.Looper.loop(Looper.java:164) 
 at android.app.ActivityThread.main(ActivityThread.java:6541) 
 at java.lang.reflect.Method.invoke(Native Method) 
 at com.android.internal.os.Zygote$MethodAndArgsCaller.run(Zygote.java:240) 
 at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:767) 
在我的清单中,我有:

<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />

<uses-permission android:name="android.permission.ACTION_MANAGE_OVERLAY_PERMISSION"/>
这是DrawView:

    public class DrawView extends View {

int w;
int h;
int r;
float screenFactor;
TextPaint startTextPaint;

public DrawView(Context activity) {
    super(activity);
}

@Override
protected void onSizeChanged(int width, int height, int oldw, int oldh) {

        w = width;
        h = height;
        r = w / 2;

    screenFactor = (r / 160f);

    if (startTextPaint == null) startTextPaint = new TextPaint();

    startTextPaint.setTextSize(100);
    startTextPaint.setTextAlign(Paint.Align.CENTER);
    startTextPaint.setTypeface(Typeface.create("Roboto Condensed", Typeface.BOLD));

    super.onSizeChanged(w, h, oldw, oldh);
}

@Override
protected void onDraw(Canvas canvas) {

        startTextPaint.setARGB(255,255,0,0);
        canvas.drawText("Test", w / 2, h / 2, startTextPaint);
}
}

根据针对Android 8.0的应用程序的文档:

使用SYSTEM_ALERT_WINDOW权限的应用程序不能再使用以下窗口类型在其他应用程序和系统窗口上方显示警报窗口:

相反,应用程序必须使用名为
type\u APPLICATION\u OVERLAY
的新窗口类型

所以你的应用程序可以针对一些较低的版本。在这种情况下,您的警报窗口将

始终显示在使用类型\应用程序\覆盖窗口类型的窗口下方。如果应用程序的目标是Android 8.0(API级别26),则该应用程序将使用TYPE_APPLICATION_OVERLAY窗口类型来显示警报窗口


(引自同一来源)

Android Oreo(及未来版本)不允许使用WindowManager.LayoutParams.TYPE\u手机,因为不推荐使用WindowManager.LayoutParams.TYPE\u应用程序覆盖

if (Build.VERSION.SDK_INT < Build.VERSION_CODES.O) {
            params = new WindowManager.LayoutParams(
                    WindowManager.LayoutParams.WRAP_CONTENT,
                    WindowManager.LayoutParams.WRAP_CONTENT,
                    WindowManager.LayoutParams.TYPE_PHONE,
                    WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE,
                    PixelFormat.TRANSLUCENT);
        }else{
            params = new WindowManager.LayoutParams(
                    WindowManager.LayoutParams.WRAP_CONTENT,
                    WindowManager.LayoutParams.WRAP_CONTENT,
                    WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY,
                    WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE,
                    PixelFormat.TRANSLUCENT);
        }
}
if(Build.VERSION.SDK\u INTWindowManager.LayoutParams.TYPE_电话,
WindowManager.LayoutParams.FLAG\u不可聚焦,
像素格式(半透明);
}否则{
params=新建WindowManager.LayoutParams(
WindowManager.LayoutParams.WRAP_内容,
WindowManager.LayoutParams.WRAP_内容,

WindowManager.LayoutParams.TYPE\u应用程序\u覆盖, WindowManager.LayoutParams.FLAG\u不可聚焦, 像素格式(半透明); } }
if(Build.VERSION.SDK\u INTWindowManager.LayoutParams.TYPE\系统\错误
}
其他的
{

WindowManager.LayoutParams.TYPE\u应用程序\u覆盖 }
只有更改窗口管理器的此参数时,才能使用以下代码

int LAYOUT_FLAG = Build.VERSION.SDK_INT >= Build.VERSION_CODES.O ?
            WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY : WindowManager.LayoutParams.TYPE_PHONE;
    mWindowParams = new LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT,
            LAYOUT_FLAG, // Overlay over the other apps.
            LayoutParams.FLAG_NOT_FOCUSABLE    // This flag will enable the back key press.
                    | LayoutParams.FLAG_NOT_TOUCH_MODAL, // make the window to deliver the focus to the BG window.
            PixelFormat.TRANSPARENT);

试试这段代码,它工作得很好

 int layout_parms;
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
         layout_parms = WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY;
    } else {
            layout_parms = WindowManager.LayoutParams.TYPE_PHONE;
    }

    yourparams = new WindowManager.LayoutParams(
            WindowManager.LayoutParams.WRAP_CONTENT,
            WindowManager.LayoutParams.WRAP_CONTENT,
            layout_parms,
            WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE | WindowManager.LayoutParams.FLAG_LAYOUT_IN_SCREEN | WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE,
            PixelFormat.TRANSLUCENT);

不要直接进入你的页面,你必须先请求类似的权限(在我的项目中,主页是目标页面) 您可以在中找到更多信息


如何使覆盖布局可交互?@Jerrin A Mathews-除了覆盖onDraw()的自定义视图外,还可以使用带有按钮的布局并将此布局充气(例如,使用
视图。充气()
)和设置OnClickListeners等等。@0X0nosugar您是否知道
TYPE_ACCESSIBILITY_OVERLAY
是否可以用于具有顶级过滤器,其中触摸事件仍然传递到下面的活动?@EM creates-我非常严重地怀疑这一点,因为如果我们谈论属于第三方的活动,这将是一个安全问题派对应用程序。@0X0nosugar我感谢你的帮助。。嗯,这将给我使用最新版本Android的应用程序带来严重的问题。请为您的回答提供解释。谢谢您提供此代码片段,它可能会提供一些有限的、即时的帮助。A通过展示为什么这是一个很好的解决问题的方法,并将使它对未来有其他类似问题的读者更有用。请在您的回答中添加一些解释,包括您所做的假设。WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY在LockScreen上不起作用太棒了!只有这一个帮助很大!
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.O) {
WindowManager.LayoutParams.TYPE_SYSTEM_ERROR
}
else
{
WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY
}
int LAYOUT_FLAG = Build.VERSION.SDK_INT >= Build.VERSION_CODES.O ?
            WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY : WindowManager.LayoutParams.TYPE_PHONE;
    mWindowParams = new LayoutParams(LayoutParams.WRAP_CONTENT, LayoutParams.WRAP_CONTENT,
            LAYOUT_FLAG, // Overlay over the other apps.
            LayoutParams.FLAG_NOT_FOCUSABLE    // This flag will enable the back key press.
                    | LayoutParams.FLAG_NOT_TOUCH_MODAL, // make the window to deliver the focus to the BG window.
            PixelFormat.TRANSPARENT);
 int layout_parms;
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
         layout_parms = WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY;
    } else {
            layout_parms = WindowManager.LayoutParams.TYPE_PHONE;
    }

    yourparams = new WindowManager.LayoutParams(
            WindowManager.LayoutParams.WRAP_CONTENT,
            WindowManager.LayoutParams.WRAP_CONTENT,
            layout_parms,
            WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE | WindowManager.LayoutParams.FLAG_LAYOUT_IN_SCREEN | WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE,
            PixelFormat.TRANSLUCENT);
private static final int OVERLAY_PERMISSION_CODE =5463 ; // can be anything

@Override
public void onCreate(...) {

    if (Build.VERSION.SDK_INT >= 23) {
        if (!Settings.canDrawOverlays(this)) {

            // Open the permission page
            Intent intent = new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION,
                    Uri.parse("package:" + getPackageName()));
            startActivityForResult(intent, OVERLAY_PERMISSION_CODE);
            return;
        }
    }
}
  int LAYOUT_FLAG;

    mOverlayView = LayoutInflater.from(this).inflate(R.layout.overlay_layout, null);

    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
        LAYOUT_FLAG = WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY;
    } else {
        LAYOUT_FLAG = WindowManager.LayoutParams.TYPE_PHONE;
    }

    final WindowManager.LayoutParams params = new WindowManager.LayoutParams(
            WindowManager.LayoutParams.WRAP_CONTENT,
            WindowManager.LayoutParams.WRAP_CONTENT,
            LAYOUT_FLAG,
            WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE,
            PixelFormat.TRANSLUCENT);