我有一个小部件,它根据按下的各种小部件按钮启动到特定页面。当应用程序在内存中时(即从部署(运行/调试)),它工作正常,但是当我强制停止应用程序时,小部件不再正常运行。
当应用程序被强制停止时,第一次单击该窗口小部件正常工作,并调用相应活动的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"/>
答案 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