android - permisos - El PERMISO de SYSTEM_ALERT_WINDOW en la API 26 no funciona como se esperaba. Permiso denegado para el tipo de ventana 2002
permisos android 8 android studio (5)
Estoy usando el permiso de superposición para mostrar cierta información en mi aplicación. Al ejecutarlo en API 23 - 25 funciona bien (pedir permiso, otorgar, etc. de acuerdo con
No se puede agregar la ventana android.view.ViewRoot$W@44da9bc0 - permiso denegado para este tipo de ventana ). (Muchas gracias a ceph3us!)
Al intentar lo mismo en la API 26, recibo un error, básicamente, "permiso denegado para el tipo de ventana 2002" al llamar
windowManager.addView(frameLayout, params);
¿Cambió Google la forma, funciona la superposición? ¿Alguna idea de cómo obtener mi texto como una superposición en la pantalla en Android 8 (Oreo), API 26? Gracias por tus ideas!
Este es el registro de errores:
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)
En mi Manifiesto, tengo:
<uses-permission android:name="android.permission.SYSTEM_ALERT_WINDOW" />
<uses-permission android:name="android.permission.ACTION_MANAGE_OVERLAY_PERMISSION"/>
Este es mi MainActivity:
public class MainActivity extends AppCompatActivity {
DrawView dv;
FrameLayout frameLayout;
WindowManager windowManager;
LayoutInflater layoutInflater;
/** code to post/handler request for permission */
public final static int REQUEST_CODE = 1234;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
checkDrawOverlayPermission();
}
public void createOnTopView() {
WindowManager.LayoutParams params = new WindowManager.LayoutParams(
WindowManager.LayoutParams.WRAP_CONTENT,
WindowManager.LayoutParams.WRAP_CONTENT,
WindowManager.LayoutParams.TYPE_PHONE,
WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE,
PixelFormat.TRANSLUCENT);
params.gravity = Gravity.CENTER;
if (frameLayout == null) frameLayout = new FrameLayout(getApplicationContext());
if (dv == null) dv = new DrawView(getApplicationContext());
windowManager = (WindowManager)
getApplicationContext().getSystemService(Context.WINDOW_SERVICE);
windowManager.addView(frameLayout, params);
windowManager.addView(dv, params);
layoutInflater = (LayoutInflater)
getApplicationContext().getSystemService(Context.LAYOUT_INFLATER_SERVICE);
// Here is the place where you can inject whatever layout you want.
layoutInflater.inflate(R.layout.activity_main, frameLayout);
}
public void checkDrawOverlayPermission() {
/* check if we already have permission to draw over other apps */
if (android.os.Build.VERSION.SDK_INT > 22) {
if (!Settings.canDrawOverlays(this)) {
/* if not construct intent to request permission */
Intent intent = new Intent(Settings.ACTION_MANAGE_OVERLAY_PERMISSION,
Uri.parse("package:" + getPackageName()));
/* request permission via start activity for result */
startActivityForResult(intent, REQUEST_CODE);
}
else {
createOnTopView();
}
}
}
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
/* check if received result code
is equal our requested code for draw permission */
if (requestCode == REQUEST_CODE && android.os.Build.VERSION.SDK_INT > 22) {
/* if so check once again if we have permission */
if (Settings.canDrawOverlays(this)) {
createOnTopView();
}
}
}
}
Y este es el 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 Oreo (y futuras versiones) no permite usar WindowManager.LayoutParams.TYPE_PHONE porque está en desuso, sino que usa WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY
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);
}
}
De acuerdo con la documentación sobre los Cambios de comportamiento de Android 8.0 para aplicaciones dirigidas a Android 8.0:
Las aplicaciones que usan el permiso SYSTEM_ALERT_WINDOW ya no pueden usar los siguientes tipos de ventanas para mostrar ventanas de alerta sobre otras aplicaciones y ventanas del sistema:
TYPE_PHONE
TYPE_PRIORITY_PHONE
TYPE_SYSTEM_ALERT
TYPE_SYSTEM_OVERLAY
TYPE_SYSTEM_ERROR
En su lugar, las aplicaciones deben usar un nuevo tipo de ventana llamada
TYPE_APPLICATION_OVERLAY
.
Así que tu aplicación podría apuntar a alguna versión inferior. En este caso, su ventana de alerta se ...
siempre aparece debajo de las ventanas que usan el tipo de ventana TYPE_APPLICATION_OVERLAY. Si una aplicación se dirige a Android 8.0 (nivel de API 26), la aplicación utiliza el tipo de ventana TYPE_APPLICATION_OVERLAY para mostrar las ventanas de alerta.
(citado de la misma fuente)
Prueba este código funcionando perfectamente
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);
Usa el siguiente código
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);
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.O) {
WindowManager.LayoutParams.TYPE_SYSTEM_ERROR
}
else
{
WindowManager.LayoutParams.TYPE_APPLICATION_OVERLAY
}
Solo cambio en este parámetro del gestor de ventanas.