システムオーバーレイウィンドウの作成(常に上)


285

常にすべてのウィンドウの上に表示される、常に操作可能なボタン/クリック可能な画像を作成しようとしています。

概念実証は

私は成功しており、現在サービスを実行しています。サービスは常に画面の左上隅にテキストを表示しますが、ユーザーは通常の方法で残りのアプリを自由に操作できます。

私がやっていることはサブクラスでViewGroupあり、それをフラグ付きでルートウィンドウマネージャーに追加しますTYPE_SYSTEM_OVERLAY。次に、このテキストの代わりにボタン/クリック可能な画像を追加して、それ自体でタッチイベントを受け取ることができるようにします。「onTouchEvent」を全体的にオーバーライドしようとしましたViewGroupが、イベントを受け取りません。

always-on-topビューグループの特定の部分でのみイベントを受信するにはどうすればよいですか?親切に提案します。

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

1
@ st0le OK多分あなたと私は一緒にそれを掘り出し、あなたの質問のリンクを私に与え、あなたがどれだけ進歩を遂げることができたかを知らせます...
coreSOLO


3
あなたonDraw()が呼ばれているのは驚くべきことです。dispatchDraw()代わりにオーバーライドすべきではありません か?groups.google.com/forum/?fromgroups#!topic/android-developers/…を
faizal

回答:


158

これは愚かな解決策かもしれません。しかし、それは機能します。改善できる場合はお知らせください。

OnCreate of your Service:私は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タッチイベント

@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:どんなエラーが出ていますか?このコードは独立しておらず、適切なプロジェクトの下に適切なファイル名を付けてプロジェクトに配置する必要があります。
Sarwar Erfan 2010

4
@pengwang:ここでは最も簡単な実装検索:docs.google.com/...
Sarwar ERFAN

6
全体のコードは、私がリンクを忘れて、私は私のミスを見つけ、正しい<用途許可をアンドロイド:名=「android.permission.SYSTEM_ALERT_WINDOW」/>私はあなたがあなたの答えを編集考えるSarwar ERFAN、それはより素晴らしいになることができます
pengwang

1
この答えは、1つの点を除いて、非常に役立ちます。コードで使用されているように、WindowManager.LayoutParams.FLAG_WATCH_OUTSIDE_TOUCHは、ドキュメントによれば、「完全な上/移動/下のジェスチャーを受け取りません」。つまり、スワイプやACTION_DOWN以外の何かを検出することはできません。まあ私はスワイプを検出することが可能であることを知っています。Waveランチャーはこの同様の概念を使用しており、スワイプを検出できるためです。これがどのように機能するか誰にも説明できますか?
ブライアン

6
@SarwarErfan-私はこれを2.3で完全に使用しましたが、onTouchイベントはAndroid 4.0、1nd 4.1で呼び出されていません。助けてください
Sanjay Singh

66

@Sam Luの回答に従い、実際に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
その画面が
ハングし

素晴らしいポスト。うまくいけば、これは将来のリリースでも無効にならないでしょう。
クックスター2013

アニメーションは動作しません... :(そして、4.2.Xでは、ビュー内のどのビューでもクリックをキャッチできません。何か提案はありますか?
Kamran Ahmed

それは機能しますが、そのショートカット(つまり、システムレベルのアラート)を使用している場合、他のダイアログが表示されません。どうして ?
Chimu 2013

@アルジュどういう意味?削除しようとしていますか?
amirlazarovich 2013年

18

アンドロイドの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 2012

1
ICSに関するこの質問を参照してください:stackoverflow.com/questions/9656185/type-system-overlay-in-ics
clemp6r

13

私はTooleap SDKの開発者の1人です。また、開発者が常に上部のウィンドウとボタンに表示する方法も提供しています。また、同様の状況にも対処しています。

ここで答えになっていない問題の1つは、Androidの「保護されたボタン」の問題です。

セキュリティで保護されたボタンには、filterTouchesWhenObscuredウィンドウがタッチされていなくても、ウィンドウの下に配置されている場合は操作できないというプロパティがあります。Androidドキュメントの引用:

ビューのウィンドウが別の表示されているウィンドウによって隠れている場合に、タッチをフィルタリングするかどうかを指定します。trueに設定すると、トースト、ダイアログ、または他のウィンドウがビューのウィンドウの上に表示されても、ビューはタッチを受けません。詳細については、{@ link android.view.View}セキュリティドキュメントをご覧ください。

このようなボタンの例は、サードパーティのAPKをインストールしようとしたときのインストールボタンです。次の行をビューレイアウトに追加すると、どのアプリでもこのようなボタンを表示できます。

android:filterTouchesWhenObscured="true"

「保護されたボタン」の上に常に最前面のウィンドウを表示すると、オーバーレイがクリックできない場合でも、オーバーレイで覆われているすべての保護されたボタンパーツがタッチに対応しなくなります。そのため、そのようなウィンドウを表示する予定がある場合は、ユーザーがウィンドウを移動または閉じる方法を提供する必要があります。また、オーバーレイの一部が透明である場合は、基になるアプリの特定のボタンが突然機能しないのはなぜかユーザーが混乱する可能性があることを考慮してください。


あなたは私の質問に答えてくださいすることができますstackoverflow.com/questions/28964771/...
ナウマンAfzaal

Android 10には、オーバーレイビューがそれらの上に表示できないいくつかのアクティビティがあります。理由を教えてください。または、ここで私の質問を参照してください。非常に役に立ちます。stackoverflow.com
59598264 /…

12

常に上部の画像ボタンで作業する

最初に私の英語をごめんなさい

コードを編集して、彼のタッチイベントをリッスンする実際の画像ボタンが背景要素にタッチコントロールを与えないようにします。

また、他の要素にタッチリスナーを提供します

ボタンの配置は下と左です

アリングメントを変更することはできますが、if要素のタッチイベントでコーディナットを変更する必要があります

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;
    }
}

これは機能しますが、画像の周りではなく、画像がクリックされたことをどのように確認できますか?
リアムW

@LiamW、タッチの座標は、この例ではkangaroo.getWidth()と比較されます。イメージをどこに描画するかに基づいて、独自のコードで何と比較するかを推測できます。
Yevgeny Simkin 2013年

非常に素晴らしい!完璧に動作します。
Yevgeny Simkin 2013年

これが機能する最小のAndroidバージョンは何ですか?私はこれを書くことに取り組んでいますが、アンドロイドに新しいので初心者の時間があります。
Noitidart 2015

6

実際には、TYPE_SYSTEM_OVERLAYの代わりにWindowManager.LayoutParams.TYPE_SYSTEM_ERRORを試すことができます。ハックのように聞こえるかもしれませんが、すべての上にビューを表示しながら、タッチイベントを取得できます。


1
私はそれを機能させるためにあらゆることを試みましたが、成功しませんでした...しかしTYPE_SYSTEM_ERRORを追加すると仕事が完了しました...実際にはTYPE_SYSTEM_OVERLAYがタッチをブロックしていた..... + 1。
Robi Kumar Tomar 2018年

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 2013

@Vinuthan What is your output、it does respond
Viswanath Lekshmanan

いいえ、応答しません。アクティビティを画面の半分以上に埋め込んで、背景とやり取りできるようにしましたが、背景とやり取りできません。
Vinuthan 2013

FLAG_WATCH_OUTSIDE_TOUCHは、バックグラウンドとの対話を防ぎます
Richard Le Mesurier

1
TYPE_PHONEが機能しました-タッチが配信されるようになりました-少なくともボタンをタップすることができます;)ありがとうございます。
2014

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;
    }

}

3

それだけを行うライブラリを見つけました:https : //github.com/recruit-lifestyle/FloatingView

ルートフォルダーにサンプルプロジェクトがあります。私はそれを実行し、それは必要に応じて動作します。背景はクリック可能です-別のアプリであっても。

ここに画像の説明を入力してください


推薦ありがとうございます、私はこの素晴らしいプロジェクトを見つけようとしていました。
アップ


1

まあ私のコードを試してみてください、少なくともそれがオーバーレイとして文字列を与えるので、それをボタンや画像に置き換えることができます。これは私の初めてのAndroidアプリLOLだとは思わないでしょう。とにかく、あなたが私よりもAndroidアプリの経験がある場合は、試してみてください

  • 「新しいWindowManager.LayoutParams」のパラメーター2と3を変更する
  • 別のイベントアプローチを試す

私は結局これを追求する時間を見つけられなかったので、私はまだその詳細が不足しています...
st0le

1

まだこのスレッドを読んでいて、これを機能させることができない場合、モーションイベントをインターセプトするこの方法はバグと見なされ、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

はい、今は機能しています。しかし、startActivityForResult(intent、OVERLAY_REQ_CODE);を呼び出した後にコールバック応答を取得しません。保護されたvoid onActivityResult(int requestCode、int resultCode、Intent data)メソッド。
Ahmad Shahwaiz

@AhmadShahwaizリンクで言及されている変更は、onActivityResultメソッドでコールバックを取得するアクティビティを使用して呼び出されます。サービスで上記の権限を確認している場合は、サービスから開始するダミーの透過アクティビティを追加する必要がある場合があります。これにより、ユーザーに権限を要求でき、onActivityResultダミーアクティビティでユーザーから結果を取得できます。ユーザーの応答に応じてウィンドウにビューを追加するかどうかをサービスに通知する必要がある場合
patilmandar2007

1

これは古い質問ですが、最近AndroidではBubblesがサポートされています。バブルはまもなくリリースされますが、現在開発者は使用を開始できます。バブルは、を使用する代わりに設計されていますSYSTEM_ALERT_WINDOW。(Facebook MessengerとMusiXMatchは同じコンセプトを使用しています)のようなアプリ。

泡

バブルは通知APIを介して作成され、通常どおり通知を送信します。バブリングしたい場合は、追加のデータを添付する必要があります。バブルの詳細については、バブルに関する公式のAndroidデベロッパーガイドをご覧ください。

弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.