在安卓系统中如何使用PROXIMITY_SCREEN_OFF_WAKE_LOCK?

8

我在我的应用程序中启用了接近唤醒锁屏功能,当接近传感器检测到物体时,它会关闭屏幕。但是当屏幕重新唤醒时,存在一个问题--它会进入锁屏界面而不是我的应用程序。无论屏幕关闭的时间长短如何(即使在几秒钟后传感器被清除),这种情况都会发生。以下是我使用的代码:

int PROXIMITY_SCREEN_OFF_WAKE_LOCK = 32;    
mProximityWakeLock = pm.newWakeLock(PROXIMITY_SCREEN_OFF_WAKE_LOCK, LOG_TAG);
if(!mProximityWakeLock.isHeld()){
    mProximityWakeLock.acquire();
}

有没有办法纠正这种行为?

7个回答

5

如果锁屏不是安全的,您可以使用以下方法取消锁屏:

getWindow().addFlags(WindowManager.LayoutParams.FLAG_DISMISS_KEYGUARD);

你可以在创建时调用它来防止锁屏出现,或者在需要时调用它。我与以下内容结合使用:
getWindow().addFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON);

这似乎不会影响接近锁定功能。


那不会禁用我的应用程序的锁屏功能吗?我不想这样做--我希望锁屏功能正常工作。我只是不希望在不应该锁定时(即当接近传感器释放并屏幕重新打开时)被锁定。预期的行为是屏幕返回到我的应用程序,除非已经过了足够长的超时时间以启用锁屏。 - user496854

4
我使用这段代码。
import android.content.Context;
import android.os.PowerManager;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.Toast;

public class MainActivity extends AppCompatActivity {

    private PowerManager mPowerManager;
    private PowerManager.WakeLock mWakeLock;

    private static final String TAG = "MainActivity";

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        mPowerManager = (PowerManager) getSystemService(Context.POWER_SERVICE);
    }

    public void activateSensor(View v) {
        Toast.makeText(MainActivity.this, "Proximity On", Toast.LENGTH_LONG).show();
        if (mWakeLock == null) {
            mWakeLock = mPowerManager.newWakeLock(PowerManager.PROXIMITY_SCREEN_OFF_WAKE_LOCK, "incall");
        }
        if (!mWakeLock.isHeld()) {
            Log.d(TAG, "New call active : acquiring incall (CPU only) wake lock");
            mWakeLock.acquire();
        } else {
            Log.d(TAG, "New call active while incall (CPU only) wake lock already active");
        }
    }

    public void deactivateSensor(View v) {
        Toast.makeText(MainActivity.this, "Proximity Off", Toast.LENGTH_LONG).show();
        if (mWakeLock != null && mWakeLock.isHeld()) {
            mWakeLock.release();
            Log.d(TAG, "Last call ended: releasing incall (CPU only) wake lock");
        } else {
            Log.d(TAG, "Last call ended: no incall (CPU only) wake lock were held");
        }
    }

}

在清单文件中:
<uses-permission android:name="android.permission.WAKE_LOCK" />

在我的布局中:
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    tools:context="com.example.proximitysensor.MainActivity">

    <Button
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_above="@+id/activity_main_turn_on_screen"
        android:onClick="activateSensor"
        android:text="@string/activity_main_activate_sensor" />

    <Button
        android:id="@+id/activity_main_turn_on_screen"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_alignParentBottom="true"
        android:onClick="deactivateSensor"
        android:text="@string/activity_main_deactivate_sensor" />
</RelativeLayout>

1
非常好的代码,干得好!我有一个问题。使用您的代码,我能否仅通过按下按钮关闭显示屏?(没有接近传感器!) - Dake
我不知道没有接近传感器怎么办...你可以查看这个链接(https://dev59.com/Rmw15IYBdhLWcg3wMpEY)...抱歉。 - Cabezas
这是Android的限制吗? - Dake
我在制作通话应用程序时使用了这段代码。这就是为什么用户打电话时我需要接近传感器的原因。我认为Android更喜欢限制。因为除了PROXIMITY_SCREEN_OFF_WAKE_LOCK之外,我没有找到任何代码和文档中的内容。 - Cabezas
我阅读了文档,它说如果用户按下电源按钮,则屏幕将关闭PARTIAL_WAKE_LOCK。我认为你不需要这个。 - Cabezas
显示剩余10条评论

3
除了使用PowerManager中的wakeLock,您还必须创建KeyGuard锁,如下所示:
private WakeLock mWakeLock = null;
private KeyguardLock mKeyguardLock = null;

public void enableProximitySensor() {
  PowerManager pm = (PowerManager) getApplicationContext().getSystemService(Context.POWER_SERVICE);
  mWakeLock = pm.newWakeLock(PROXIMITY_SCREEN_OFF_WAKE_LOCK, LOG_TAG);
  mWakeLock.acquire();

  KeyguardManager km = (KeyguardManager) getApplicationContext().getSystemService(Context.KEYGUARD_SERVICE);
  mKeyguardLock = mManager.newKeyguardLock( KEYGUARD_TAG );
}

public void disableProximitySensor() {
  if ( null != mWakeLock ) {
    mWakeLock.release();
    mWakeLock = null;
  }
  if ( null != mKeyguardLock ) {
    mKeyguardLock.disableKeyguard();
    mKeyguardLock = null;
  }
}

创建此锁只能在获得邻近唤醒锁的同时使用,而锁屏将在应用程序的其他时间内工作。

这会如何防止锁屏出现? - user496854
这个解决方案在同样的情况下帮了我 =) 我遇到了同样的问题 - 当手机离开耳朵时,锁屏会出现。使用KeyguardLock对我有所帮助。示例代码经过一些编辑以澄清我的想法。 - Andrey Starodubtsev
值得注意的是,这需要额外的权限(DISABLE_KEYGUARD)。 - Jay Sidri

2

谢谢,我已经在使用mProximityWakeLock.release(1)。我使用了反射使其工作,它肯定可以正常工作--当我覆盖接近传感器时屏幕会关闭,当我移开时屏幕又会亮起来。但它只是进入锁屏状态,这显然不是我想要的。 - user496854
@user496854,您能解释一下如何使用反射来使用mProximityWakeLock.release(1)吗?提前致谢。 - jegumi
没有简单的方法将它作为评论发布,但是请继续提出关于如何做到这一点的问题,并且我可以在那里将代码作为答案发布。 - user496854
对于像我一样好奇的人,mProximityWakeLock.release(1) 中的 1 是指 PowerManager 标志 RELEASE_WAIT_FOR_NO_PROXIMITY(请参见文档),它不会释放 WakeLock 直到对象不再靠近。 - Jaween

1
private WakeLock mProximityWakeLock= null;

mProximityWakeLock= pm.newWakeLock(PowerManager.FULL_WAKE_LOCK|PowerManager.ACQUIRE_CAUSES_WAKEUP , LOG_TAG);

当需要打开屏幕时获取锁定,并在需要关闭时释放锁定,即反转以上 wakeLock 将正常工作。


1

你是怎么把屏幕打开的?我成功地通过以下方式,在传感器被覆盖时将屏幕关闭:

WindowManager.LayoutParams lp = getWindow().getAttributes();
lp.buttonBrightness = WindowManager.LayoutParams.BRIGHTNESS_OVERRIDE_OFF;
lp.screenBrightness = WindowManager.LayoutParams.BRIGHTNESS_OVERRIDE_OFF;
etWindow().setAttributes(lp);

在此之后,从传感器上“释放”或“重置”盖子,将不会执行代码的其他部分:

WindowManager.LayoutParams lp = getWindow().getAttributes();
lp.buttonBrightness = WindowManager.LayoutParams.BRIGHTNESS_OVERRIDE_NONE;
lp.screenBrightness = WindowManager.LayoutParams.BRIGHTNESS_OVERRIDE_NONE;
etWindow().setAttributes(lp);

0

如果适用于您的应用程序,您可以使用

getWindow().addFlags(LayoutParams.FLAG_SHOW_WHEN_LOCKED | LayoutParams.FLAG_IGNORE_CHEEK_PRESSES);

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接