创建系统覆盖窗口(始终在顶部)


285

我正在尝试创建一个始终操作于顶部的按钮/ clickable-image,该按钮始终保持在所有窗口的顶部。

概念证明是

我已经成功,并且现在正在运行服务。该服务始终在屏幕的左上角显示一些文本,同时用户可以正常方式与其余应用自由交互。

我正在做的是子类,ViewGroup并使用flag将其添加到根窗口管理器中TYPE_SYSTEM_OVERLAY。现在,我想添加一个按钮/可点击图像代替此文本,该文本本身可以接收触摸事件。我尝试覆盖整个“ onTouchEvent”,ViewGroup但未收到任何事件。

如何仅在始终显示在顶部的视图组的某些部分上接收事件?请提示。

public class HUD extends Service {
    HUDView mView;

    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }

    @Override
    public void onCreate() {
        super.onCreate();
        Toast.makeText(getBaseContext(),"onCreate", Toast.LENGTH_LONG).show();
        mView = new HUDView(this);
        WindowManager.LayoutParams params = new WindowManager.LayoutParams(
                WindowManager.LayoutParams.WRAP_CONTENT,
                WindowManager.LayoutParams.WRAP_CONTENT,
                WindowManager.LayoutParams.TYPE_SYSTEM_OVERLAY,
                0,
//              WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
//                      | WindowManager.LayoutParams.FLAG_NOT_TOUCHABLE,
                PixelFormat.TRANSLUCENT);
        params.gravity = Gravity.RIGHT | Gravity.TOP;
        params.setTitle("Load Average");
        WindowManager wm = (WindowManager) getSystemService(WINDOW_SERVICE);
        wm.addView(mView, params);
    }

    @Override
    public void onDestroy() {
        super.onDestroy();
        Toast.makeText(getBaseContext(),"onDestroy", Toast.LENGTH_LONG).show();
        if(mView != null)
        {
            ((WindowManager) getSystemService(WINDOW_SERVICE)).removeView(mView);
            mView = null;
        }
    }
}

class HUDView extends ViewGroup {
    private Paint mLoadPaint;

    public HUDView(Context context) {
        super(context);
        Toast.makeText(getContext(),"HUDView", Toast.LENGTH_LONG).show();

        mLoadPaint = new Paint();
        mLoadPaint.setAntiAlias(true);
        mLoadPaint.setTextSize(10);
        mLoadPaint.setARGB(255, 255, 0, 0);
    }

    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        canvas.drawText("Hello World", 5, 15, mLoadPaint);
    }

    @Override
    protected void onLayout(boolean arg0, int arg1, int arg2, int arg3, int arg4) {
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {
        //return super.onTouchEvent(event);
        Toast.makeText(getContext(),"onTouchEvent", Toast.LENGTH_LONG).show();
        return true;
    }
}

2
@coreSOLO,希望您能得到答复,我奇怪的类似问题没有得到任何答复。:) +1
st0le

3
@ st0le我保证我会花一整夜,直到找到解决方案并与您分享:)
coreSOLO 2010年

1
@ st0le好吧,也许您和我可以一起研究一下,给我您的问题的链接,并让我知道您能够取得多少进步……
coreSOLO 2010年


3
令人惊讶的是你onDraw()被打了。应该不 dispatchDraw()应该改写吗?见 groups.google.com/forum/?fromgroups#!topic/android-developers/...
faizal

Answers:


158

这可能是一个愚蠢的解决方案。但这有效。如果您可以改善它,请告诉我。

您的服务的OnCreate:我已经使用过WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH标志。这是服务的唯一变化。

@Override
    public void onCreate() {
        super.onCreate();
        Toast.makeText(getBaseContext(),"onCreate", Toast.LENGTH_LONG).show();
        mView = new HUDView(this);
        WindowManager.LayoutParams params = new WindowManager.LayoutParams(
                WindowManager.LayoutParams.TYPE_SYSTEM_OVERLAY,
                WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH,
                PixelFormat.TRANSLUCENT);
        params.gravity = Gravity.RIGHT | Gravity.TOP;
        params.setTitle("Load Average");
        WindowManager wm = (WindowManager) getSystemService(WINDOW_SERVICE);
        wm.addView(mView, params);
    }

现在,您将开始获取每个点击事件。因此,您需要在事件处理程序中进行更正。

在您的ViewGroup touch事件中

@Override
public boolean onTouchEvent(MotionEvent event) {

    // ATTENTION: GET THE X,Y OF EVENT FROM THE PARAMETER
    // THEN CHECK IF THAT IS INSIDE YOUR DESIRED AREA


    Toast.makeText(getContext(),"onTouchEvent", Toast.LENGTH_LONG).show();
    return true;
}

另外,您可能需要将此权限添加到清单中:

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

1
@pengwang:您遇到什么错误?此代码不是独立的,您需要使用适当的文件名将它们放置在项目中的适当项目下。
萨瓦尔·埃凡

4
@pengwang:在这里找到最简单的实现:docs.google.com/…–
Sarwar

6
整个代码正确,我发现了我的错误,我忘记了链接<uses-permission android:name =“ android.permission.SYSTEM_ALERT_WINDOW” /> Sarwar Erfan我认为您可以编辑答案,它会变得更加精彩
pengwang

1
除了一件事,这个答案非常有帮助。根据文档中的代码,WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH中使用了“不会收到完整的向上/向下/向下手势”。换句话说,无法检测到滑动或除ACTION_DOWN以外的任何其他内容。好吧,我知道可以检测到滑动,因为Wave Launcher使用了类似的概念,并且能够检测到滑动。谁能向我解释这是如何工作的?
布赖恩

6
@SarwarErfan-我在2.3上完美地使用了它的软木塞,但在Android 4.0(第4版)上并未调用onTouch事件。请帮忙
Sanjay Singh

66

遵循@Sam Lu的回答,Indeed Android 4.x确实阻止了某些类型的侦听外部触摸事件,但某些类型(例如TYPE_SYSTEM_ALERT)仍然有效。

    WindowManager.LayoutParams params = new WindowManager.LayoutParams(
            WindowManager.LayoutParams.MATCH_PARENT,
            WindowManager.LayoutParams.MATCH_PARENT,
            WindowManager.LayoutParams.TYPE_SYSTEM_ALERT,
            WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
                    | WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL
                    | WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH,
            PixelFormat.TRANSLUCENT);

    WindowManager wm = (WindowManager) getSystemService(WINDOW_SERVICE);
    LayoutInflater inflater = (LayoutInflater) getSystemService(LAYOUT_INFLATER_SERVICE);
    View myView = inflater.inflate(R.layout.my_view, null);
    myView.setOnTouchListener(new OnTouchListener() {
       @Override
       public boolean onTouch(View v, MotionEvent event) {
           Log.d(TAG, "touch me");
           return true;
       }
     });

    // Add layout to window manager
    wm.addView(myView, params);

权限

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

5
它使我的屏幕挂死了..任何想法
Viswanath Lekshmanan 2013年

很棒的帖子。希望将来的版本中也不会禁用此功能。
Cookster

动画无法在其上运行... :(而且,它无法在4.2.X上捕获任何视图上的点击。有任何建议吗?
Kamran Ahmed

它可以工作,但是当我使用该快捷方式(即“系统级警报”)时,未显示“其他”对话框。为什么呢
Chimu 2013年

@Arju是什么意思?您是否要删除它?
2013年

18

与Android 4.x的开始,Android团队通过添加新功能修正了一个潜在的安全问题adjustWindowParamsLw()将在其中添加FLAG_NOT_FOCUSABLEFLAG_NOT_TOUCHABLE和删除FLAG_WATCH_OUTSIDE_TOUCH标志的TYPE_SYSTEM_OVERLAY窗口。

也就是说,TYPE_SYSTEM_OVERLAY窗口将不会在ICS平台上收到任何触摸事件,因此,TYPE_SYSTEM_OVERLAY对于ICS和将来的设备而言,使用窗口当然不是可行的解决方案。


4
那有什么选择呢?
radhoo


13

我是Tooleap SDK的开发人员之一。我们还为开发人员提供了一种始终显示在顶部窗口和按钮上的方式,并且我们已经处理了类似的情况。

此处尚未解决的问题之一是Android的“安全按钮”。

安全按钮的filterTouchesWhenObscured属性意味着即使将其放置在窗口下,也无法与它们进行交互,即使该窗口未受到任何触摸也是如此。引用Android文档:

指定当视图的窗口被另一个可见窗口遮盖时是否过滤触摸。设置为true时,只要在视图窗口上方出现烤面包,对话框或其他窗口,该视图就不会受到触摸。有关更多详细信息,请参见{@link android.view.View}安全文档。

当您尝试安装第三方apk时,安装按钮就是此类按钮的示例。如果将以下行添加到视图布局,则任何应用都可以显示此类按钮:

android:filterTouchesWhenObscured="true"

如果在“安全按钮”上显示始终位于顶部的窗口,则即使覆盖层不可单击,覆盖层覆盖的所有安全按钮部分也将无法处理任何触摸。因此,如果您打算显示这样的窗口,则应该为用户提供一种移动或关闭它的方式。并且,如果叠加层的一部分是透明的,请考虑到您的用户可能会感到困惑,为什么基础应用程序中的某个按钮突然无法为他服务。


能否请您回答我的问题:stackoverflow.com/questions/28964771/...
瑙曼Afzaal

有在Android的10一些的活动,覆盖视图不能显示他们上面可以请你告诉的原因还是看我的问题在这里将是非常有益stackoverflow.com/questions/59598264/...
MATEEN乔杜里

12

始终在顶级图像按钮上工作

首先,对不起我的英语

我编辑您的代码,并使监听他的触摸事件的工作图像按钮没有对他的背景元素进行触摸控制。

也给触摸听众提供了其他元素

按键的左下角和左下角

您可以更改alingments,但是需要在if元素中的触摸事件中更改Cordinats

import android.annotation.SuppressLint;

import android.app.Service;
import android.content.Context;
import android.content.Intent;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.PixelFormat;
import android.os.IBinder;
import android.util.Log;
import android.view.Gravity;
import android.view.MotionEvent;
import android.view.View;
import android.view.View.OnTouchListener;
import android.view.ViewGroup;
import android.view.WindowManager;
import android.widget.Toast;

public class HepUstte extends Service {
    HUDView mView;

    @Override
    public void onCreate() {
        super.onCreate();   

        Toast.makeText(getBaseContext(),"onCreate", Toast.LENGTH_LONG).show();

        final Bitmap kangoo = BitmapFactory.decodeResource(getResources(),
                R.drawable.logo_l);


        WindowManager.LayoutParams params = new WindowManager.LayoutParams(
                kangoo.getWidth(), 
                kangoo.getHeight(),
                WindowManager.LayoutParams.TYPE_SYSTEM_ALERT,
                WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
                |WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL
                |WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH,
                 PixelFormat.TRANSLUCENT);






        params.gravity = Gravity.LEFT | Gravity.BOTTOM;
        params.setTitle("Load Average");
        WindowManager wm = (WindowManager) getSystemService(WINDOW_SERVICE);



        mView = new HUDView(this,kangoo);

        mView.setOnTouchListener(new OnTouchListener() {


            @Override
            public boolean onTouch(View arg0, MotionEvent arg1) {
                // TODO Auto-generated method stub
                //Log.e("kordinatlar", arg1.getX()+":"+arg1.getY()+":"+display.getHeight()+":"+kangoo.getHeight());
                if(arg1.getX()<kangoo.getWidth() & arg1.getY()>0)
                {
                 Log.d("tıklandı", "touch me");
                }
                return false;
            }
             });


        wm.addView(mView, params);



        }



    @Override
    public IBinder onBind(Intent arg0) {
        // TODO Auto-generated method stub
        return null;
    }

}



@SuppressLint("DrawAllocation")
class HUDView extends ViewGroup {


    Bitmap kangoo;

    public HUDView(Context context,Bitmap kangoo) {
        super(context);

        this.kangoo=kangoo;



    }


    protected void onDraw(Canvas canvas) {
        //super.onDraw(canvas);


        // delete below line if you want transparent back color, but to understand the sizes use back color
        canvas.drawColor(Color.BLACK);

        canvas.drawBitmap(kangoo,0 , 0, null); 


        //canvas.drawText("Hello World", 5, 15, mLoadPaint);

    }


    protected void onLayout(boolean arg0, int arg1, int arg2, int arg3, int arg4) {
    }

    public boolean onTouchEvent(MotionEvent event) {
        //return super.onTouchEvent(event);
       // Toast.makeText(getContext(),"onTouchEvent", Toast.LENGTH_LONG).show();

        return true;
    }
}

这行得通,但是如何确定我的图像被单击,而不是图像周围的东西?
Liam W

@LiamW,在此示例中将触摸的坐标与kangaroo.getWidth()进行比较。您可以根据绘制图像的位置来推断将其与自己的代码进行比较的内容。
Yevgeny Simkin

非常好!完美运作。
叶夫根尼·辛金

最低的Android版本是什么?我正在写这篇文章,但是我对Android还是陌生的,所以有一些初学者的时间。
Noitidart

6

实际上,您可以尝试使用WindowManager.LayoutParams.TYPE_SYSTEM_ERROR而不是TYPE_SYSTEM_OVERLAY。听起来好像很hack,但是它使您可以在所有内容之上显示视图,并且仍然可以获取触摸事件。


1
我尽了一切努力使其成功,但是没有成功....但是添加TYPE_SYSTEM_ERROR可以完成工作....实际上TYPE_SYSTEM_OVERLAY阻止了接触..... + 1。
Robi Kumar Tomar

6

TYPE_SYSTEM_OVERLAY自API级别26开始不建议使用此常量。请改用TYPE_APPLICATION_OVERLAY。或**适用于android 8以下的用户

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

5

尝试这个。在ICS中工作正常。如果要停止服务,只需单击状态栏中生成的通知。

 public class HUD extends Service
 {
    protected boolean foreground = false;
    protected boolean cancelNotification = false;
    private Notification notification;
    private View myView;
    protected int id = 0;
    private WindowManager wm;
    private WindowManager.LayoutParams params;
    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }
    @Override
    public void onCreate() {
        super.onCreate();
       // System.exit(0);
        Toast.makeText(getBaseContext(),"onCreate", Toast.LENGTH_SHORT).show();
        params = new WindowManager.LayoutParams(WindowManager.LayoutParams.WRAP_CONTENT, WindowManager.LayoutParams.WRAP_CONTENT,
                WindowManager.LayoutParams.TYPE_PHONE, WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL
                        | WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH | WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE, PixelFormat.TRANSLUCENT);
        params.gravity=Gravity.TOP|Gravity.LEFT;
    wm = (WindowManager) getSystemService(WINDOW_SERVICE);
    inflateview();
    foregroundNotification(1);
    //moveToForeground(1,n,true);
    }     
   @Override
    public void onDestroy() {
        super.onDestroy();
        ((NotificationManager) getSystemService(NOTIFICATION_SERVICE)).cancel(0);
        Toast.makeText(getBaseContext(),"onDestroy", Toast.LENGTH_SHORT).show();
        if(myView != null)
        {
            ((WindowManager) getSystemService(WINDOW_SERVICE)).removeView(myView);
            myView = null;
        }
    }
    protected Notification foregroundNotification(int notificationId) 
   {    
    notification = new Notification(R.drawable.ic_launcher, "my Notification", System.currentTimeMillis());    
        notification.flags = notification.flags | Notification.FLAG_ONGOING_EVENT | Notification.FLAG_ONLY_ALERT_ONCE;   
        notification.setLatestEventInfo(this, "my Notification", "my Notification", notificationIntent());          
        ((NotificationManager) getSystemService(NOTIFICATION_SERVICE)).notify(id, notification);            
        return notification;
    }
    private PendingIntent notificationIntent() {
        Intent intent = new Intent(this, stopservice.class);    
        PendingIntent pending = PendingIntent.getActivity(this, 0, intent, PendingIntent.FLAG_UPDATE_CURRENT);    
        return pending;
    }
    public void inflateview()
    {
         LayoutInflater inflater = (LayoutInflater) getSystemService(LAYOUT_INFLATER_SERVICE);
            myView = inflater.inflate(R.layout.activity_button, null);
            myView.setOnTouchListener(new OnTouchListener() {
               @Override
               public boolean onTouch(View v, MotionEvent event) {
                   Toast.makeText(getBaseContext(),"onToasttt", Toast.LENGTH_SHORT).show();
                   return false;
               }
             });
            // Add layout to window manager
            wm.addView(myView, params); 
    }
}

更新

在这里取样

要创建覆盖视图,请在设置LayoutParams时不要将类型设置为TYPE_SYSTEM_OVERLAY。

Instead set it to TYPE_PHONE.

Use the following flags:

FLAG_NOT_TOUCH_MODAL

FLAG_WATCH_OUTSIDE_TOUCH

这对我不起作用,我的屏幕与背景不互动。您知道的任何其他方法...
Vinuthan

@Vinuthan您的输出是什么,它是否响应
Viswanath Lekshmanan

不,它没有响应。我在屏幕上方填充了我的活动,以便可以与背景进行交互,但不能与背景进行交互。
Vinuthan

FLAG_WATCH_OUTSIDE_TOUCH可防止您与背景互动
Richard Le Mesurier 2014年

1
TYPE_PHONE为我工作-现在可以触摸了-至少我可以点击一个按钮;)谢谢!
vir我们

3

这是一些简单的解决方案,您所需要做的就是像在列表适配器上一样对XML布局进行膨胀,而对XML布局进行膨胀。这是您所需的所有代码。

 public class HUD extends Service {
    View mView;

    LayoutInflater inflate;
    TextView t;
    Button b;

    @Override
    public void onCreate() {
        super.onCreate();   

        Toast.makeText(getBaseContext(),"onCreate", Toast.LENGTH_LONG).show();


        WindowManager wm = (WindowManager) getSystemService(WINDOW_SERVICE);

        Display display = wm.getDefaultDisplay();  get phone display size
        int width = display.getWidth();  // deprecated - get phone display width
        int height = display.getHeight(); // deprecated - get phone display height 


        WindowManager.LayoutParams params = new WindowManager.LayoutParams(
                width, 
                height,
                WindowManager.LayoutParams.TYPE_SYSTEM_ALERT,
                WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE
                |WindowManager.LayoutParams.FLAG_NOT_TOUCH_MODAL
                |WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH,
                 PixelFormat.TRANSLUCENT);


        params.gravity = Gravity.LEFT | Gravity.CENTER;
        params.setTitle("Load Average");

        inflate = (LayoutInflater) getBaseContext()
                .getSystemService(Context.LAYOUT_INFLATER_SERVICE);

        mView = inflate.inflate(R.layout.canvas, null);

        b =  (Button) mView.findViewById(R.id.button1);
        t = (TextView) mView.findViewById(R.id.textView1);
        b.setOnClickListener(new OnClickListener() {

        public void onClick(View v) {
            // TODO Auto-generated method stub
            t.setText("yes you click me ");
        }
       });

        wm.addView(mView, params);

        }



    @Override
    public IBinder onBind(Intent arg0) {
        // TODO Auto-generated method stub
        return null;
    }

}



1

好吧,尝试我的代码,至少它会为您提供覆盖的字符串,您可以很好地用按钮或图像替换它。您不会相信这是我有史以来第一个android应用。无论如何,如果您比我对Android应用程序更有经验,请尝试

  • 在“新的WindowManager.LayoutParams”中更改参数2和3
  • 尝试一些不同的事件方法

我最终没有时间进行这项工作,因此我仍然对其细节
一无所知

1

如果有人仍在阅读此线程并且无法正常工作,很抱歉告诉您这种拦截运动事件的方法被视为bug,并在android> = 4.2中修复。

您截获的运动事件,虽然动作为ACTION_OUTSIDE,但在getX和getY中返回0。这意味着您无法在屏幕上看到所有运动位置,也无法执行任何操作。我知道文档说它将得到x和y,但事实是不会。看来这是为了阻止按键记录程序。

如果有人有解决方法,请留下您的评论。

ref: 为什么在KitKat 4.4.2上每次ACTION_OUTSIDE返回0?

https://code.google.com/p/android/issues/detail?id=72746


1

通过使用服务,您可以实现以下目标:

public class PopupService extends Service{

    private static final String TAG = PopupService.class.getSimpleName();
    WindowManager mWindowManager;
    View mView;
    String type ;

    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
//        registerOverlayReceiver();
        type = intent.getStringExtra("type");
        Utils.printLog("type = "+type);
        showDialog(intent.getStringExtra("msg"));
        return super.onStartCommand(intent, flags, startId);
    }

    private void showDialog(String aTitle)
    {
        if(type.equals("when screen is off") | type.equals("always"))
        {
            Utils.printLog("type = "+type);
            PowerManager pm = (PowerManager) getApplicationContext().getSystemService(Context.POWER_SERVICE);
            WakeLock mWakeLock = pm.newWakeLock((PowerManager.SCREEN_DIM_WAKE_LOCK | PowerManager.ACQUIRE_CAUSES_WAKEUP), "YourServie");
            mWakeLock.acquire();
            mWakeLock.release();
        }

        mWindowManager = (WindowManager) getSystemService(WINDOW_SERVICE);
        mView = View.inflate(getApplicationContext(), R.layout.dialog_popup_notification_received, null);
        mView.setTag(TAG);

        int top = getApplicationContext().getResources().getDisplayMetrics().heightPixels / 2;

        LinearLayout dialog = (LinearLayout) mView.findViewById(R.id.pop_exit);
//        android.widget.LinearLayout.LayoutParams lp = (android.widget.LinearLayout.LayoutParams) dialog.getLayoutParams();
//        lp.topMargin = top;
//        lp.bottomMargin = top;
//        mView.setLayoutParams(lp);

        final EditText etMassage = (EditText) mView.findViewById(R.id.editTextInPopupMessageReceived);

        ImageButton imageButtonSend = (ImageButton) mView.findViewById(R.id.imageButtonSendInPopupMessageReceived);
//        lp = (LayoutParams) imageButton.getLayoutParams();
//        lp.topMargin = top - 58;
//        imageButton.setLayoutParams(lp);
        imageButtonSend.setOnClickListener(new OnClickListener() {
            @Override
            public void onClick(View v) {
                Utils.printLog("clicked");
//                mView.setVisibility(View.INVISIBLE);
                if(!etMassage.getText().toString().equals(""))
                {
                    Utils.printLog("sent");
                    etMassage.setText("");
                }
            }
        });

        TextView close = (TextView) mView.findViewById(R.id.TextViewCloseInPopupMessageReceived);
        close.setOnClickListener(new OnClickListener() {

            @Override
            public void onClick(View arg0) {
                hideDialog();   
            }
        });

        TextView view = (TextView) mView.findViewById(R.id.textviewViewInPopupMessageReceived);
        view.setOnClickListener(new OnClickListener() {

            @Override
            public void onClick(View arg0) {
                hideDialog();   
            }
        });

        TextView message = (TextView) mView.findViewById(R.id.TextViewMessageInPopupMessageReceived);
        message.setText(aTitle);

        final WindowManager.LayoutParams mLayoutParams = new WindowManager.LayoutParams(
        ViewGroup.LayoutParams.MATCH_PARENT,
        ViewGroup.LayoutParams.MATCH_PARENT, 0, 0,
        WindowManager.LayoutParams.TYPE_SYSTEM_ERROR,
        WindowManager.LayoutParams.FLAG_SHOW_WHEN_LOCKED
                | WindowManager.LayoutParams.FLAG_DISMISS_KEYGUARD
//                | WindowManager.LayoutParams.FLAG_TURN_SCREEN_ON
                | WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON ,
        PixelFormat.RGBA_8888);

        mView.setVisibility(View.VISIBLE);
        mWindowManager.addView(mView, mLayoutParams);
        mWindowManager.updateViewLayout(mView, mLayoutParams);

    }

    private void hideDialog(){
        if(mView != null && mWindowManager != null){
            mWindowManager.removeView(mView);
            mView = null;
        }
    }
}

1

@Sarwar Erfan的答案不再起作用,因为Android不允许将WindowManager.LayoutParams.TYPE_SYSTEM_OVERLAY添加到窗口以使其不再可触摸,即使WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCH也是如此。

我找到了解决此问题的方法。您可以在以下问题中查看

使用WindowManager.LayoutParams.TYPE_SYSTEM_OVERLAY向窗口添加视图时,没有发生触摸事件


您提到的解决方案在棉花糖中也不起作用
Ahmad Shahwaiz

@AhmadShahwaiz请再次检查链接对于您提到的问题,我已经添加了评论
patilmandar2007'Nov

是的,现在可以正常工作了。但是在调用startActivityForResult(intent,OVERLAY_REQ_CODE)之后没有得到回调响应;在受保护的void onActivityResult(int requestCode,int resultCode,Intent data)方法中。
艾哈迈德·沙威兹

@AhmadShahwaiz链接中提到的更改将使用活动调用,我们将在onActivityResult方法中获取回调。如果您正在Service中检查以上权限,则可能需要添加一个虚拟透明活动以从Service启动,该活动可以向用户询问权限,然后在onActivityResult虚拟活动中您将从用户那里得到结果。之后,您将需要通知您的服务是否将视图添加到窗口中(取决于用户的响应)
patilmandar2007 '16

1

这是一个古老的问题,但最近Android支持Bubbles。气泡即将推出,但目前开发人员可以开始使用它们。气泡的设计可替代SYSTEM_ALERT_WINDOW。类似的应用程序(Facebook Messenger和MusiXMatch使用相同的概念)。

泡泡

气泡是通过Notification API创建的,您可以正常发送通知。如果您希望它冒泡,则需要附加一些额外的数据。有关Bubbles的更多信息,您可以转到Bubbles的官方Android开发者指南

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.