应用程序强制停止后,Widget无法正常启动

时间:2011-10-10 21:00:32

标签: java android widget

我有一个小部件,它根据按下的各种小部件按钮启动到特定页面。当应用程序在内存中时(即从部署(运行/调试)),它工作正常,但是当我强制停止应用程序时,小部件不再正常运行。

当应用程序被强制停止时,第一次单击该窗口小部件正常工作,并调用相应活动的onCreate。然后,当按下主页键并按下不同的小部件按钮时,将调用上一个小部件Activity的onRestart方法。

奇怪的是,b / c没有出现日志消息,并且由于调试器无法连接,因此测试起来相当困难。有什么特别的,我做错了吗?

流程是 - > WidgetDispatchActivity (处理真正的小部件目标) - > ViewActivity > - >进一步分支(以下示例中为DetailViewActivity

小部件的

onUpdate

onUpdate(){
...


       Intent viewIntent = new Intent(context, WidgetDispatchActivity.class);
                    viewIntent.putExtra("launch", WidgetInfo.VIEW_ACTIVITY);
                    PendingIntent viewPendIntent = PendingIntent.getActivity(context, 2, viewIntent, PendingIntent.FLAG_UPDATE_CURRENT);

...
}

WidgetDispatchActivity的onCreate

 Bundle b = getIntent().getExtras();
        if (b != null && b.isEmpty()){
            Log.i(LOG_TAG, "EMPTY BUNDLE ON WIDGET!");
        }else{
            Log.i(LOG_TAG, "WIDGET BUNDLE HAS ITEMS!");
        }

        String destination = null;
        if (b != null && b.containsKey("launch")){
            destination = b.getString("launch");
            Log.i(LOG_TAG, "WIDGET ITEM FROM BUNDLE WAS: " + destination);
        }
        else{
            Log.i(LOG_TAG, " \"launch\" item was not inside bundle!");
        }

        if(destination != null) {
            Log.i(LOG_TAG, "PendingIntent received from widget!");
            Intent goTo = new Intent(this, ViewActivity.class);
               ...
}
}

logcat的

10-10 20:38:00.935: INFO/ActivityManager(58): Starting activity: Intent { act=android.intent.action.MAIN cat=[android.intent.category.HOME] flg=0x10200000 cmp=com.android.launcher/com.android.launcher2.Launcher }
10-10 20:38:00.956: INFO/DetailViewActivity(1154): ONPAUSE
10-10 20:38:00.975: WARN/InputManagerService(58): Ignoring hideSoftInput of: com.android.internal.view.IInputMethodClient$Stub$Proxy@450366d0
10-10 20:38:02.545: INFO/ActivityManager(58): Starting activity: Intent { flg=0x10000000 cmp=com.starbucks.mobilecard/com.xxx.xxx.widget.WidgetDispatchActivity bnds=[198,298][224,321] (has extras) }
10-10 20:38:02.585: INFO/ViewActivity(1154): ONRESTART
10-10 20:38:02.585: INFO/ViewActivity(1154): WIDGET LAUNCH:: == false

ViewActivity onRestart:

protected void onRestart(){
        super.onRestart();
        Log.i(LOG_TAG,"ONRESTART");

        Log.i(LOG_TAG,"WIDGET LAUNCH:: == " + WidgetInfo.getInstance().wasLaunchedFromWidget());
                 ...
}

AndroidManifest:

<activity android:name="com.xxx.xxx.ui.cards.ViewActivity" android:label="@string/_view_title" android:finishOnTaskLaunch="true" android:clearTaskOnLaunch="true" android:screenOrientation="portrait"/>

<activity android:name="com.xxx.xxx.widget.WidgetDispatchActivity" android:label="@string/widget_dispatch_title" android:finishOnTaskLaunch="true"/>

1 个答案:

答案 0 :(得分:1)

Willmel,

当Force Close实现你的onDestroy()时,阻止你的小部件工作的第一步。在这里,您需要确保与构成Widget通信的WidgetHost,HostViews和RemoteView进行对话。

此语句阻止WidgetHost从Android接收任何事件。 myHost是对象,通常是一个包含对Widget的引用的Activity。

// Stop the Widgets from fidgeting
    myHost.stopListening();
    myHost = null;

从那里,您将要删除对您可能引用的任何HostView的任何引用。只需抓取参考并将其设置为null即可完成此操作。在我的onDestroy()中,我使用以下循环...

if (appWidgets != null) {
    final int count = appWidgets.size();
    for (int i = 0; i < count; i++) {
        final Widget launcherInfo = appWidgets.get(i);
        launcherInfo.hostView = null;
    }
}

在这里,launcherInfo是我的扩展AppWidget对象。 hostView指的是我的AppWidgetHostView(这是代表我的Widget的视图)。 appWidgets数组是我的启动器正在跟踪的小部件列表。非常简单。

使用窗口小部件和WidgetHost可能会出现问题,具体取决于您的设置。上述声明在正确的地方确实有助于减轻痛苦。

FuzzicalLogic