【问题标题】:How to detect whether screen is touched in a background service in Android Studio?如何在Android Studio的后台服务中检测屏幕是否被触摸?
【发布时间】:2015-05-06 10:58:20
【问题描述】:

我正在做一个关于 android studio 的项目。 我需要在后台服务中检测屏幕是否被触摸(并为此弹出一条消息)。 但是,在不影响用户使用智能手机的情况下,我无法在后台服务中检测屏幕是否被触摸

当我说“在后台检测”时,我的意思是应用程序的效果不会干扰用户在屏幕上所做的事情。我下面的进展应该解释我在这里的意思:

我制作了一个应用程序,通过在 android 中实现 onTouchListener 并使用该 onTouchListener 的布局来检测用户是否在后台触摸屏幕。我按照网站上的说明做到了这一点 "http://kpbird.blogspot.com.au/2013/03/android-detect-global-touch-event.html" 我对布局进行了一些更改以覆盖整个屏幕,以便它可以检测到整个屏幕。

但是,在我运行我的应用程序后,该应用程序创建了不断检测屏幕触摸的服务,每次当用户触摸屏幕时,该触摸都会被我的服务(布局)吸收,因此用户无法正确使用他们的电话不再(如按图标启动其他应用程序)。这是因为布局覆盖了整个屏幕并阻止了对其下方图标的任何触摸信息(布局检测屏幕触摸),因此图标不知道有触摸,因此不会对用户做出反应。但我希望我的应用和服务能够让用户正常使用手机。

我听说 android 中的服务设计为不与用户交互(即屏幕触摸)并在后台工作,但我想知道是否有办法解决这个问题。

以下是我的服务代码:

public class GlobalTouchService extends Service implements OnTouchListener{

    private String TAG = this.getClass().getSimpleName();
    // window manager
    private WindowManager mWindowManager;
    // linear layout will use to detect touch event
    private LinearLayout touchLayout;
    @Override
    public IBinder onBind(Intent arg0) {
        return null;
    }
    @Override
    public void onCreate() {
        super.onCreate();
        // create linear layout
        touchLayout = new LinearLayout(this);
        // set layout width 30 px and height is equal to full screen
        LayoutParams lp = new LayoutParams(LayoutParams.MATCH_PARENT, LayoutParams.MATCH_PARENT);
        touchLayout.setLayoutParams(lp);
        // set color if you want layout visible on screen
        //touchLayout.setBackgroundColor(Color.CYAN);
        // set on touch listener
        touchLayout.setOnTouchListener(this);

        // fetch window manager object
        mWindowManager = (WindowManager) getSystemService(WINDOW_SERVICE);
        // set layout parameter of window manager
        WindowManager.LayoutParams mParams = new WindowManager.LayoutParams(
                WindowManager.LayoutParams.MATCH_PARENT, // width is equal to full screen
                WindowManager.LayoutParams.MATCH_PARENT, // height is equal to full screen
                WindowManager.LayoutParams.TYPE_PHONE, // Type Phone, These are non-application windows providing user interaction with the phone (in particular incoming calls).
                WindowManager.LayoutParams.FLAG_NOT_FOCUSABLE, // this window won't ever get key input focus
                PixelFormat.TRANSLUCENT);
        mParams.gravity = Gravity.LEFT | Gravity.TOP;
        Log.i(TAG, "add View");

        mWindowManager.addView(touchLayout, mParams);

    }


    @Override
    public void onDestroy() {
        if(mWindowManager != null) {
            if(touchLayout != null) mWindowManager.removeView(touchLayout);
        }
        super.onDestroy();
    }

    public void showAlert(View view) {
        AlertDialog.Builder myAlertBuilder = new AlertDialog.Builder(this);
        myAlertBuilder.setMessage(getString(R.string.alertMessage))
                .setTitle(getString(R.string.alertTitle))
                .setPositiveButton(getString(R.string.alertPositiveChoice), new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialog, int which) {
                        dialog.dismiss();
                    }
                })
                .setNegativeButton(getString(R.string.alertNegativeChoice), new DialogInterface.OnClickListener() {
                    @Override
                    public void onClick(DialogInterface dialog, int which) {
                        dialog.dismiss();
                        onDestroy();
                    }
                });
        AlertDialog myAlert = myAlertBuilder.create();
        myAlert.getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ALERT);
        myAlert.show();
    }


    @Override
    public boolean onTouch(View v, MotionEvent event) {
        //if (event.getAction() == MotionEvent.ACTION_DOWN || event.getAction() == MotionEvent.ACTION_UP)
        //    Log.i(TAG, "Action :" + event.getAction() + "\t X :" + event.getRawX() + "\t Y :" + event.getRawY());
        ////////////////////
        View myView = null;
        //showAlert(myView);
        ///////////////////////
        return false;
    }
}

【问题讨论】:

    标签: android service touch screen detect


    【解决方案1】:

    那里有 onTouch 方法。只需返回 false 并且触摸不会被您的应用吸收。

    请同时检查: https://stackoverflow.com/a/6384443/1723095

    编辑: 这可能不是你想要的,但如果你有根设备,你可以这样做:

    adb shell getevent dev/input/event1
    

    在我的例子中,event1 负责触摸事件,但它可能会有所不同。

    【讨论】:

    • 现在 onTouch 方法返回 false,但是在服务运行时,我仍然无法在主屏幕中按下和打开任何应用程序。我已经阅读了您发布的链接,也许 ontouch() 方法和 onTouchEvent 方法不同?我在我的服务类中实现了 ontouch() 方法,服务类实现了 onTouchListener。
    • 在本教程中,您将宽度设置为 30px,然后将其设置为:WindowManager.LayoutParams.MATCH_PARENT 这就是它覆盖整个区域并且您无法启动任何应用程序的原因。尝试为您的布局设置背景颜色,这样您就会看到它有多大。
    • 我正在制作一个应用程序,该应用程序创建一个可以检测屏幕上任何触摸的服务,因此我将宽度和高度设置为覆盖整个屏幕。现在的问题是,当我按下主屏幕中的其他应用程序图标时,用于检测屏幕触摸的服务创建的布局会消耗触摸事件。结果,我无法打开其他应用程序,并且在服务运行时无法执行任何操作。
    • 我不确定,但我认为您的情况是,尽管您忽略了布局中的触摸事件(通过返回 false),但触摸还是会转到它的父级(视图),它无论如何都会消耗事件。也许你可以缩小这个区域并使用 FLAG_WATCH_OUTSIDE_TOUCH 而不是覆盖整个屏幕
    • 我明白了。有没有办法可以到达它的父级并防止在那里消费事件?
    猜你喜欢
    • 1970-01-01
    • 2012-02-14
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2012-03-04
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多