システムオーバーレイウィンドウを作成する(常に最前面に表示) 質問する

システムオーバーレイウィンドウを作成する(常に最前面に表示) 質問する

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

概念実証は

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

私がやっていることは、サブクラス化ViewGroupして、フラグを使用してルート ウィンドウ マネージャーに追加することです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;
    }
}

ベストアンサー1

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

サービスの 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タッチイベントで

@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" />

おすすめ記事