Android 一体机模式:规则限制


Android Kiosk Mode: Rules for Restrictions

生活中我们随处都可见到一体机,当然了我们在这谈论的并不是卖冰激凌和柠檬水的售货一体机,我们这篇文章是关于交互式信息一体机的。

电脑的终端就是一个典型的信息一体机,用户可以使用它来执行一组有限的动作。其它常见的例子还有如 ATM 机、照相亭、自动售票机、自助登机服务终端系统、等等。

交互式信息一体机使用了各种技术:触摸屏、纸币识别器、相机、打印机、扫描仪、Wi-Fi、NFC 等等。它们的共同特点是对未经授权的活动有强有力的保障体系。这样的终端下使用的软件不允许用户去改变系统的设置,重置或安装其它应用程序。

移动一体机

移动技术 的快速发展也使一体机有了革命性的发展。智能手机和平板电脑是现在常用的一体机。移动一体机跟标准的终端相比有很多明显的优势:由于紧凑的尺寸和大量生产而更加便宜,但是也有更多的功能。移动一体机在餐馆中被用作电子菜单,在商店和展厅中的销售助手等等。无论目的是什么,重要的是当一个移动设备提供一体机的功能为目的时它就不应该被用于任何其它目的。

Share to Twitter: Mobile kiosks have a number of significant advantages over standard terminals

为了使设备在一体机模式(kiosk mode,或说展台模式)下工作,需要运行一个程序,此程序会冻结掉操作系统的常见功能且它不允许用户退出程序。一体机的应用程序可能有一个秘密隐藏的管理面板或者从服务器控制的远程配置。你也可以配置系统报告:程序会将有关用户行为的数据发送给服务器,并将相关状态通知给管理员。

下面,我们将分享一些我们在 Android 一体机应用程序 方面的经验。我们也会讨论下开发中的一些陷阱和解决方案。

Android一体机模式

把一台 Android 设备变为一体机,这是锁定其所有按键和连接器的好方法。在这种情况下,最普通也最有效的解决办法是把你的设备放置在一个防爆盒或者一个专门的地方中,然而这可能都是不可选的方法。此外,状态栏,系统对话框和虚拟小键盘仍然可以由任何用户访问,并且设置也可以更改。 你如何避免这些问题?

Android 5.0: 期待已久的 API

让我们先说个好消息开始吧:Android 5.0 中已有锁住屏幕 API 的介绍了。这个 API “pins”(别针)别住屏幕并阻止用户离开所选的应用程序。该功能可用于为职员创建的一体机,开发用于评估和审查的教育应用。

当你激活了屏幕固定模式,用户不会被系统对话框和通知所打断,同时不能打开其它应用程序,也不能回到主屏幕,状态栏也不再可见。

你可以通过设置或者软件来激活这一模式

  • 在设置中(安全 -> 屏幕固定)开启屏幕固定模式,选择所需的应用并确认你的选择。

  • 软件中激活则调用 startLockTask() 方法并确认列入锁定模式。

Android 5.0 之前如何做到兼容?

在早些的 Android SDK 中并没有提供一体机模式,也没有综合全面的 API 来冻结系统。因此,所有的不同组件应分别锁住而且在不同的版本中也是不同的方法。

定制操作系统可以大大的简化一体机模式的实现,但是我们想介绍在 Android 5.0 版本下不使用特殊固件或者 root 权限的情况下如何冻结掉不需要的东西。

重启

当设备被冻结时用户首先想到的就是重启设备了。我们的任务是确保平板电脑或者智能手机重启后,一体机应用应该自动启动。

这并不算难:在 Manifest 中声明广播接收器,重启后接受信息的权限,然后还有写个继承 Broadcastreceiver 的子类使你的程序跑起来。

AndroidManifest.xml:

<receiver android:name=".BootReceiver">  
    <intent-filter >  
        <action android:name=  
             "android.intent.action.BOOT_COMPLETED"/>  
   </intent-filter>  
</receiver>  

<uses-permission  
 android:name="android.permission.RECEIVE_BOOT_COMPLETED" />

BootReceiver.java :

public class BootReceiver extends BroadcastReceiver {  

  @Override  
  public void onReceive(Context context, Intent intent) {  
    Intent myIntent = new Intent(context, MyKioskModeActivity.class);  
    myIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);  
    context.startActivity(myIntent);  
  }  
}  

返回键

返回键则重写如下方法。

@Override
public void onBackPressed() {  
    // here we do nothing  
}  

HOME 键

HOME 键并不能截获,因此为阻止它被按下时返回主屏幕,我们指定一体机程序作为 Launcher。在 Manifest 中添加 3 行:

<intent-filter>  
    <action android:name="android.intent.action.MAIN" />  

    <category android:name="android.intent.category.HOME" />  
    <category android:name="android.intent.category.LAUNCHER" />  
    <category android:name="android.intent.category.DEFAULT" />  
</intent-filter>  

完成以上工作后,当你按下 HOME 键时,系统会弹个窗口让你选择一个 launcher 作为默认的,然后选择我们的程序就好了,HOME 键这一部分也完成了!

电源键

电源键的问题是最多的,解决掉这些问题的一个办法是把一体机程序窗口设置为锁屏屏幕。然而这个方法只能保证在 Android 4.0 版本下的系统中使用。

@Override  
public void onAttachedToWindow() {  
    getWindow().addFlags(  
        WindowManager.LayoutParams.FLAG_DISMISS_KEYGUARD);  
    getWindow().addFlags(  
        WindowManager.LayoutParams.FLAG_SHOW_WHEN_LOCKED);  
}     

系统对话框

长按 HOME 或电源键将会弹出系统对话框,这可以让你退出程序。此外还有系统升级和低电量提示窗口,这对一体机也是危险的因为可以访问到系统设置。

为了彻底摆脱系统弹框的困扰,推荐如下方法:当 Activity 失去焦点时,发送一个关闭所有系统对话框的广播。

@Override  
public void onWindowFocusChanged(boolean hasFocus) {  
  super.onWindowFocusChanged(hasFocus);  
  if(!hasFocus) {  
    Intent closeDialog =   
          new Intent(Intent.ACTION_CLOSE_SYSTEM_DIALOGS);  
    sendBroadcast(closeDialog);  
  }  
}  

虚拟键盘

在虚拟键盘中可能有一个设置按键。如果键盘是需要的,最好的解决方案是实现自己的版本或者整合现有的但须限制功能。

状态栏

状态栏中存在许多可以让你退出程序的机会,所以你必须避免它。首先要做的就是让你的程序在全屏模式下工作。

在 Android 4.0 下的系统中你也可以指定窗口的类型为 TYPE_SYSTEM_ALERT - 这种情况下一体机程序将会显示在所有系统元素的顶部。

另一个方法是当状态栏一出现时就隐藏掉,做这个工作需要到 Manifest 中申请相应权限。

<uses-permission  
        android:name="android.permission.EXPAND_STATUS_BAR"/>  

@Override  
public void onWindowFocusChanged(boolean hasFocus)<br>  
{  
   if(!hasFocus)  
   {  
           Object service  = getSystemService("statusbar");  
           Class<?> statusbarManager =   
              Class.forName("android.app.StatusBarManager");  
           Method collapse = <br>  
              statusbarManager.getMethod("collapse");  
           collapse .setAccessible(true);  
           collapse .invoke(service);  
   }  
}  

Android 4.1 开始你可以使用 SDK 中隐藏状态栏的方法。

View decorView = getWindow().getDecorView();  
int uiOptions = View.SYSTEM_UI_FLAG_FULLSCREEN;  
decorView.setSystemUiVisibility(uiOptions);  
ActionBar actionBar = getActionBar();  
actionBar.hide(); 

另一个比较流行的方法是创建一个透明的视图对象,拦截掉状态栏上的点击操作。实现它要求视图有 SYSTEM_ALERT_WINDOW 标识(flag)。

以上这些在 Android 中锁定系统元素的技术内容仅仅是几个可选的方案。为了开发出不被用户破解的一体机应用,开发者门经常发现或发明新的方法并分享它们的经验给其它开发者

与我们一起分享你在开发 Android 一体机程序的 tips 吧。你有用到我们上面说到的方法吗或者你知道其它有用的黑科技方法吗?


所属标签

无标签