肖锐(Cooki)个人原创,欢迎转载,转载请注明地址,肖锐(Cooki)的技术博客 http://blog.youkuaiyun.com/xiao0026
IOS:
在程序中添加
[[UIApplication sharedApplication] setIdleTimerDisabled:YES];
Android:
锁定屏幕对于移动终端来说是非常有必要的,但是对于机顶盒产品就没有这个必要了。所以本文介绍一下怎样让Android设备永不锁屏。Android系统的锁屏时间存放在Setting数据库中,字段为Settings.System.SCREEN_OFF_TIMEOUT。查看SettingsProvider源码,查看如下文件的源码:
frameworks/base/packages/SettingsProvider/src/com/android/providers/settings/DatabaseHelper.java
查看loadSystemSettings()函数的代码如下:
01.
private
void
loadSystemSettings(SQLiteDatabase db) {
02.
SQLiteStatement stmt = db.compileStatement(
"INSERT OR IGNORE INTO system(name,value)"
03.
+
" VALUES(?,?);"
);
04.
05.
Resources r = mContext.getResources();
06.
07.
loadBooleanSetting(stmt, Settings.System.DIM_SCREEN,
08.
R.bool.def_dim_screen);
09.
loadSetting(stmt, Settings.System.STAY_ON_WHILE_PLUGGED_IN,
10.
"1"
.equals(SystemProperties.get(
"ro.kernel.qemu"
)) ?
1
:
0
);
11.
loadIntegerSetting(stmt, Settings.System.SCREEN_OFF_TIMEOUT,
12.
R.integer.def_screen_off_timeout);
13.
14.
// Set default cdma emergency tone
15.
loadSetting(stmt, Settings.System.EMERGENCY_TONE,
0
);
16.
17.
// Set default cdma call auto retry
18.
loadSetting(stmt, Settings.System.CALL_AUTO_RETRY,
0
);
19.
20.
// Set default cdma DTMF type
21.
loadSetting(stmt, Settings.System.DTMF_TONE_TYPE_WHEN_DIALING,
0
);
22.
23.
// Set default hearing aid
24.
loadSetting(stmt, Settings.System.HEARING_AID,
0
);
25.
26.
// Set default tty mode
27.
loadSetting(stmt, Settings.System.TTY_MODE,
0
);
28.
29.
loadBooleanSetting(stmt, Settings.System.AIRPLANE_MODE_ON,
30.
R.bool.def_airplane_mode_on);
31.
32.
loadStringSetting(stmt, Settings.System.AIRPLANE_MODE_RADIOS,
33.
R.string.def_airplane_mode_radios);
34.
35.
loadStringSetting(stmt, Settings.System.AIRPLANE_MODE_TOGGLEABLE_RADIOS,
36.
R.string.airplane_mode_toggleable_radios);
37.
38.
loadBooleanSetting(stmt, Settings.System.AUTO_TIME,
39.
R.bool.def_auto_time);
// Sync time to NITZ
40.
41.
loadIntegerSetting(stmt, Settings.System.SCREEN_BRIGHTNESS,
42.
R.integer.def_screen_brightness);
43.
44.
loadBooleanSetting(stmt, Settings.System.SCREEN_BRIGHTNESS_MODE,
45.
R.bool.def_screen_brightness_automatic_mode);
46.
47.
loadDefaultAnimationSettings(stmt);
48.
49.
loadBooleanSetting(stmt, Settings.System.ACCELEROMETER_ROTATION,
50.
R.bool.def_accelerometer_rotation);
51.
52.
loadDefaultHapticSettings(stmt);
53.
54.
stmt.close();
55.
}
从代码中我们可以看出,假如Settings.System.SCREEN_OFF_TIMEOUT没有初始化的话(系统首次启动,这个字段肯定是没有初始化的),将会利用资源中的R.integer.def_screen_off_timeout来初始化。我们为了让系统永不锁屏,只需要把资源R.integer.def_screen_off_timeout设为-1即可。查看文件
frameworks/base/packages/SettingsProvider/res/values/defaults.xml
可以找到R.integer.def_screen_off_timeout的定义。
1.
<integer name=
"def_screen_off_timeout"
>
60000
</integer>
发现默认值为60000ms,也就是60s。我们只需要把这个参数改为-1。然后重新编译SettingsProvider模块,就OK了。
另外为了防止用户进入系统后,修改锁屏时间,在Setting模块中删除对锁屏时间的设置。这样Android设备就永不锁屏了。
后来发现我的Android设备烧录好之后第一次启动永远不会锁屏,但是设备重启之后开机就进入锁屏状态,解锁之后就再也不会锁屏了(因为永远不超时)。看来“革命尚未成功,同志仍需努力”啊。那么为什么启动之后没有进入锁屏状态呢?是不会系统有把超时锁屏的值给修改了呢?我通过sqlite3去查看settings.db的内容,发现超时锁屏的值仍然是-1。说明启动之后,系统并没有去数据库中查看屏幕超时锁屏的值,就直接锁屏了。
但是怎样才能开机之后不进入锁屏状态呢?这个是个非常费思量的问题。经过go,我知道锁屏的代码在LockScreen.java中,然后顺藤摸瓜,终于找到了可以设置锁屏功能开关的位置。代码位于:
frameworks/policies/base/phone/com/android/internal/policy/impl/KeyguardViewMediator.java
该文件中有一个变量定义如下:
1.
/**
2.
* External apps (like the phone app) can tell us to disable the keygaurd.
3.
*/
4.
private
boolean
mExternallyEnabled =
true
;
mExternallyEnabled是用来管理是否开启屏幕锁的关键。默认值是打开屏锁,根据注释可以知道他是希望应用程序来修改这个值。但是经过加打印信息发现开机的时候没有任何应用程序会修改它。修改这个值调用如下函数:
01.
/**
02.
* Same semantics as {@link WindowManagerPolicy#enableKeyguard}; provide
03.
* a way for external stuff to override normal keyguard behavior. For instance
04.
* the phone app disables the keyguard when it receives incoming calls.
05.
*/
06.
public
void
setKeyguardEnabled(
boolean
enabled) {
07.
synchronized
(
this
) {
08.
if
(DEBUG) Log.d(TAG,
"setKeyguardEnabled("
+ enabled +
")"
);
09.
10.
mExternallyEnabled = enabled;
11.
12.
if
(!enabled && mShowing) {
13.
if
(mExitSecureCallback !=
null
) {
14.
if
(DEBUG) Log.d(TAG,
"in process of verifyUnlock request, ignoring"
);
15.
// we're in the process of handling a request to verify the user
16.
// can get past the keyguard. ignore extraneous requests to disable / reenable
17.
return
;
18.
}
19.
20.
// hiding keyguard that is showing, remember to reshow later
21.
if
(DEBUG) Log.d(TAG,
"remembering to reshow, hiding keyguard, "
22.
+
"disabling status bar expansion"
);
23.
mNeedToReshowWhenReenabled =
true
;
24.
hideLocked();
25.
}
else
if
(enabled && mNeedToReshowWhenReenabled) {
26.
// reenabled after previously hidden, reshow
27.
if
(DEBUG) Log.d(TAG,
"previously hidden, reshowing, reenabling "
28.
+
"status bar expansion"
);
29.
mNeedToReshowWhenReenabled =
false
;
30.
31.
if
(mExitSecureCallback !=
null
) {
32.
if
(DEBUG) Log.d(TAG,
"onKeyguardExitResult(false), resetting"
);
33.
mExitSecureCallback.onKeyguardExitResult(
false
);
34.
mExitSecureCallback =
null
;
35.
resetStateLocked();
36.
}
else
{
37.
showLocked();
38.
39.
// block until we know the keygaurd is done drawing (and post a message
40.
// to unblock us after a timeout so we don't risk blocking too long
41.
// and causing an ANR).
42.
mWaitingUntilKeyguardVisible =
true
;
43.
mHandler.sendEmptyMessageDelayed(KEYGUARD_DONE_DRAWING, KEYGUARD_DONE_DRAWING_TIMEOUT_MS);
44.
if
(DEBUG) Log.d(TAG,
"waiting until mWaitingUntilKeyguardVisible is false"
);
45.
while
(mWaitingUntilKeyguardVisible) {
46.
try
{
47.
wait();
48.
}
catch
(InterruptedException e) {
49.
Thread.currentThread().interrupt();
50.
}
51.
}
52.
if
(DEBUG) Log.d(TAG,
"done waiting for mWaitingUntilKeyguardVisible"
);
53.
}
54.
}
55.
}
56.
}
经过上面的讨论我们可以发现我们有两个解决方法:
1、定义变量的时候,给其初始化为false。
2、在launcher模块启动的时候,调用setKeyguardEnabled方法,关闭锁屏功能。
我懒得修改Laucher模块,我的解决方法就是在定义mExternallyEnabled时修改其初始值为false。各位朋友可以根据自己的实际情况选择解决方案。我的代码如下:
1.
/**
2.
* External apps (like the phone app) can tell us to disable the keygaurd.
3.
*/
4.
private
boolean
mExternallyEnabled =
false
;
这样修改之后,Android设备开机之后,默认不会进入锁屏状态,除非你在应用程序中调用setKeyguardEnabled方法显示打开这个功能。因为设置的超时时间为-1,则永远也不会进入锁屏界面。完全满足了我的需求,终于大功告成了。