Android Kiosk模式 - 防止用户“未经授权”取消固定应用程序

时间:2016-04-27 22:07:52

标签: android android-5.0-lollipop kiosk device-owner android-screen-pinning

我对Android很新,我目前正在开发一个应该在KioskMode中运行的Android应用程序,这样普通用户就无法退出应用程序或做任何其他事情。
我已经做了什么:

  • 我将我的应用设为设备所有者
  • 我在主要活动中将onCreate() - 方法中的应用视为设备所有者
  • 我在MainActivity中有一个按钮,稍后允许管理员通过输入密码退出。调用stopLockTask()

我现在遇到的问题是,任何用户都可以通过同时长按“后退”和“多任务”按钮来退出屏幕固定,因为应用程序应该运行的平板电脑具有硬件按钮,我不能简单地停用它。 (至少我不知道如何在没有设备的情况下这样做)
那么有没有办法停用此按钮组合以退出屏幕固定,或一些整洁的解决方法? 我想到的一种方法是在onLockTaskModeExiting()中的AdminReceiver类中重用应用程序,但我仍然坚持如何执行此操作。

以下是我的MainActivity和AdminReceiver类的一些代码片段:
AdminReceiver.java

public class AdminReceiver extends DeviceAdminReceiver{

@Override
public void onEnabled(Context context, Intent intent) {
    Toast.makeText(context, context.getString(R.string.device_admin_enabled), Toast.LENGTH_SHORT).show();
}

@Override
public CharSequence onDisableRequested(Context context, Intent intent) {
    return context.getString(R.string.device_admin_warning);
}

@Override
public void onDisabled(Context context, Intent intent) {
    Toast.makeText(context, context.getString(R.string.device_admin_disabled), Toast.LENGTH_SHORT).show();
}

@Override
public void onLockTaskModeEntering(Context context, Intent intent, String pkg) {
    Toast.makeText(context, context.getString(R.string.kiosk_mode_enabled), Toast.LENGTH_SHORT).show();
}

@Override
public void onLockTaskModeExiting(Context context, Intent intent) {
    Toast.makeText(context, context.getString(R.string.kiosk_mode_disabled), Toast.LENGTH_SHORT).show();
}

}

MainActivity.java

public class MainActivity extends Activity {

private DevicePolicyManager mDpm;


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

    ComponentName deviceAdmin = new ComponentName(this, AdminReceiver.class);
    mDpm = (DevicePolicyManager) getSystemService(Context.DEVICE_POLICY_SERVICE);

    if (!mDpm.isAdminActive(deviceAdmin)) {
        Toast.makeText(this, getString(R.string.not_device_admin), Toast.LENGTH_SHORT).show();
    }


    if (mDpm.isDeviceOwnerApp(getPackageName())) {
        Toast.makeText(this, getString(R.string.device_owner), Toast.LENGTH_SHORT).show();

        mDpm.setLockTaskPackages(deviceAdmin, new String[]{getPackageName()});
        startLockTask();
    } else {
        Toast.makeText(this, getString(R.string.not_device_owner), Toast.LENGTH_SHORT).show();
    }

    Button exit = (Button) findViewById(R.id.exit);
    exit.setOnClickListener(new View.OnClickListener(){
        @Override
        public void onClick(View v) {
            stopLockTask();
        }
    });}

非常感谢任何帮助。谢谢!

3 个答案:

答案 0 :(得分:1)

如果活动设置为DEFAULT HOME

<activity
        android:name=".AppActivity"
        android:label="Locked Activity">
        <intent-filter>
            <action android:name="android.intent.action.MAIN"/>
            <category android:name="android.intent.category.LAUNCHER" />
            <category android:name="android.intent.category.HOME"/>
            <category android:name="android.intent.category.DEFAULT"/>
        </intent-filter>
</activity>

每次用户按下主页按钮时,活动都可以重新启用onCreateonResumeonStart上的锁定。如果使用概述和后退键强制取消固定应用程序并且最近的活动列表为空,则用户只能按下将重新启用锁定的主页按钮。

由于问题是12个月,我仍然在寻找更有效的解决方案。

答案 1 :(得分:0)

如果锁定任务模式无意退出,只需重新启动您的活动并重新固定屏幕。

@Override
public void onLockTaskModeExiting(Context context, Intent intent) {
    SharedPreferences sharedPreferences = PreferenceManager.getDefaultSharedPreferences(context);
        if (sharedPreferences.getBoolean(PIN_SHOULD_ENABLED, false)) {
            context.startActivity(new Intent(context, YOUR_PIN_SCREEN_ACTIVITY.class).setFlags(Intent.FLAG_ACTIVITY_NEW_TASK));
    }
}

答案 2 :(得分:0)

我遇到了这个问题并在这个帖子中尝试了答案。我不推荐Xavier Lin的答案,因为它最终会创建你应用的多个实例。

幸运的是,我提出了自己的解决方案。

在扩展DeviceAdminReceiver的类中,重写onLockTaskModeExiting并在那里运行pin代码。

public class AdminReceiver extends DeviceAdminReceiver{

    @Override
    public void onLockTaskModeExiting(Context context, Intent intent) {
        super.onLockTaskModeExiting(context, intent);

        // Run pinning code here
    }
}