如何在没有解锁的情况下从锁屏小部件启动活动?

时间:2013-02-02 20:15:39

标签: android android-intent android-widget

我在锁屏小部件上有一个按钮,我希望按下按钮可以在按下时启动活动。如果屏幕被锁定,我希望活动显示在锁定屏幕上,而无需用户输入PIN或模式或其他任何内容,并且当用户离开活动时锁屏应重新出现。

我知道WindowManager.LayoutParams.FLAG_SHOW_WHEN_LOCKED,如果我使用来自ADB shell的am start手动启动,我的活动 会出现在锁屏上。问题是,当我按下小部件中的按钮时,它会让我在创建活动之前输入解锁PIN。

我在我的小部件提供程序中有这个代码:

@Override
public void onUpdate(final Context context, final AppWidgetManager appWidgetManager, final int[] appWidgetIds) {
    for (final int appWidgetId : appWidgetIds) {
        // Get the RemoteViews for controlling this widget instance.
        final RemoteViews views = new RemoteViews(context.getPackageName(), R.layout.my_widget);

        // Construct an intent to launch the activity.
        final Intent intent = new Intent(context, MyActivity.class);
        intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);

        // Attach the intent to the widget's button.
        final PendingIntent pendingIntent = PendingIntent.getActivity(context, 0, intent, 0);
        views.setOnClickPendingIntent(R.id.my_button, pendingIntent);

        appWidgetManager.updateAppWidget(appWidgetId, views);
    }
}

以下是MyActivity中的代码:

public MyActivity() {
    Log.d(TAG, "Activity instantiated");
}

@Override
protected void onCreate(final Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    // Allow this activity to appear over the lock screen.
    final Window window = getWindow();
    window.addFlags(WindowManager.LayoutParams.FLAG_SHOW_WHEN_LOCKED);

    setContentView(R.layout.my_activity);
}

当我按下小部件中的按钮时,系统会提示我输入解锁PIN码。活动构造函数中的日志消息直到之后我才输入PIN,这意味着Android决定在 FLAG_SHOW_WHEN_LOCKED之前询问引脚可以有任何影响。

有没有办法告诉Android我想在屏幕仍然锁定时启动活动?也许我可以在IntentPendingIntent上设置一个标志?

2 个答案:

答案 0 :(得分:4)

您可以使用广播意图并在您的窗口小部件提供程序中收听它,然后在onRecieve方法上启动活动意图,这将不会解锁您的设备,但会在前台启动您的活动,而无需请求用户解锁。

Intent broadcastIntent = new Intent("com.yourapp.yourbroadcast");
widget.setOnClickPendingIntent(R.id.yourButton,  
                               PendingIntent.getBroadcast(ctxt, 0, broadcastIntent, PendingIntent.FLAG_UPDATE_CURRENT));

on on onRecieve(Context ctx,Intent intent):

if (intent.getAction().equals("com.yourapp.yourbroadcast"))
{ 
       final Intent intent = new Intent(context, MyActivity.class);
       intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
       ctx.startActivity(intent) ;
}

答案 1 :(得分:0)

在你的宣言中:

<receiver
        android:name=".YourWidget"
        android:label="Your Widget description" >
        <intent-filter>
            <action android:name="android.appwidget.action.APPWIDGET_UPDATE" />
            <action android:name="com.yourpackage.widgetbroadcast" />
        </intent-filter>
        <meta-data
            android:name="android.appwidget.provider"
            android:resource="@xml/wisget_yourwidget" />
</receiver>

你的Widgetclass:

public class YourWidget extends AppWidgetProvider {
    @Override
    public void onUpdate(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds) {
        for (int i = 0; i < appWidgetIds.length; i++) {
            int appWidgetId = appWidgetIds[i];      
            RemoteViews views = new RemoteViews(context.getPackageName(), R.layout.widget_layout);
            Intent intent = new Intent("com.yourpackage.widgetbroadcast");
            PendingIntent pendingIntent = PendingIntent.getBroadcast(context, 0, intent, 0);
            views.setOnClickPendingIntent(R.id.widget_yourlayout, pendingIntent);
            appWidgetManager.updateAppWidget(appWidgetId, views);
        }
    }

    @Override
    public void onReceive(Context context, Intent intent) {
        super.onReceive(context, intent);
        if (intent.getAction().equals("com.yourpackage.widgetbroadcast")) {
            Intent intentStart = new Intent(context, MainActivity.class);           
            intentStart.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
            context.startActivity(intentStart);
        }
    }
}

在您的MainActivity中:

@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);        
    Window w = getWindow();
    w.addFlags(WindowManager.LayoutParams.FLAG_SHOW_WHEN_LOCKED);
    //Your code...
}

确保将活动中的Widow-Flag设置为FLAG_SHOW_WHEN_LOCKED,而不是FLAG_DISMISS_KEYGUARD以保持屏幕锁定。 此示例将通过单击窗口小部件来启动活动。