android 4.0.3上的小部件问题

时间:2013-07-16 10:28:20

标签: android android-intent android-widget android-alarms android-broadcast

我已经尝试了大约7,8个教程并在这里阅读至少30,40个答案但是徒劳...我无法在我的手机上使用任何小部件。

每个小部件实现都不会更新我的小部件上的文本,也不会在点击时注册小部件。

例如本教程:http://blog.doityourselfandroid.com/2011/05/24/developing-android-home-screenwidgets/你可以下载代码,它可以在我的朋友手机(android 2.3)上正常运行但在我的手机上我可以看到接收器在Log Cat中触发但没有更新?

这是我目前的简单代码:

1.manifest

<receiver android:name=".TestWidgetProvider">
        <meta-data android:name="android.appwidget.provider"
            android:resource="@xml/testwidgetprovider"
            />
        <intent-filter >
            <action android:name="android.appwidget.action.APPWIDGET_UPDATE"/>
        </intent-filter>

    </receiver>
    <receiver android:name=".TestReceiver"/>
  1. TestWidgetProvider

    @Override
    public void onUpdate(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds){
    
    
    Log.d("CSV", "Called on onUpdate");
    
    AlarmManager am=(AlarmManager) context.getSystemService(Context.ALARM_SERVICE);
    Calendar cal=Calendar.getInstance();
    cal.add(Calendar.SECOND, 30);
    
    Intent intent=new Intent(context, TestReceiver.class);
    
    
    int N=appWidgetIds.length;
    for (int i=0; i<N; i++) {
    
        RemoteViews tt=new RemoteViews(context.getPackageName(), R.layout.testwidget);
    
        tt.setTextViewText(R.id.textView1,"UPDATED" +i);
    
        appWidgetManager.updateAppWidget(appWidgetIds[i], tt);
    
        Log.d("CSV", "Update widget, i = " +i + " widgetId=" + appWidgetIds[i] );
    
    }
    
    
    
    intent.putExtra("id", appWidgetIds);
    intent.putExtra("duz", appWidgetIds.length);
    
    PendingIntent p1=PendingIntent.getBroadcast(context, 1, intent, 0);
    am.setRepeating(AlarmManager.RTC_WAKEUP, cal.getTimeInMillis(), 10000, p1);
    
    super.onUpdate(context, appWidgetManager, appWidgetIds);
    

    }

  2. TestReceiver

    public class TestReceiver extends BroadcastReceiver {
    
    
    private static final String tag="TestReceiver";
    @Override
    public void onReceive(Context context, Intent intent) {
    // TODO Auto-generated method stub
    
    
    int[] ID;
    Bundle extras=intent.getExtras();
    ID=extras.getIntArray("id");
    int duz=intent.getIntExtra("duz", 0);
    
    Log.d("CSV", "update widget id=" +ID);
    
    
    double b=Math.random();
    int i=(int) ((b*10+1)%10);
    
    RemoteViews rv=new RemoteViews(context.getApplicationContext().getPackageName(), R.layout.testwidget);
    AppWidgetManager appWidgetManager = AppWidgetManager.getInstance(context.getApplicationContext());
    
    
    for (int a=0; a<duz; i++) {
    
        rv.setTextViewText(R.id.textView1, "update random: " + a);
        appWidgetManager.updateAppWidget(ID[a], rv);
        Log.d("CSV", "Update widget, a = " +a + " widgetId=" + ID[a] );
    
    }
    
    
    /*
    ComponentName provider = new ComponentName(context, TestWidgetProvider.class);
    appWidgetManager.updateAppWidget ( provider, rv);
    */
    

    }

  3. widget xml

    <?xml version="1.0" encoding="utf-8"?>
       <appwidget-provider xmlns:android="http://schemas.android.com/apk/res/android" 
        android:minWidth="200dp"
    android:minHeight="120dp"
    android:updatePeriodMillis="0"
    android:initialLayout="@layout/testwidget"
    android:resizeMode="horizontal|vertical"
    >
    </appwidget-provider>
    
  4. 请帮助我被困了好几天......

    编辑1。我添加了第一个答案建议的主要活动,但没有变化。

    当我第一次在屏幕上设置小部件时,文本也没有实际更新,就像它无法找到小部件一样,但是第二次我将小部件放在屏幕上它会根据代码更改其文本在onEnable?

    编辑2: 我已经发布了一个解决方案,现在解决了我的问题,并更新了小部件

    编辑3.我将Richard Le Mesurier的答案标记为正确,因为他的演示代码详细介绍了在进行网络操作时通过服务管理小部件的细节。它涵盖了我的问题以及我要提出的许多其他问题。如果你打算用小工具做一些认真的工作,我建议你学习这个演示。

2 个答案:

答案 0 :(得分:1)

答案2(通过工作演示链接更新)

正如@JanBo所承诺的那样,关于我如何解决这个问题的更多细节(免责声明 - 这只是我的观点,我希望对如何改进提出建设性意见。)

JanBo有一个更新方法有问题,一个失败。我建议他将它们组合成一个方法,在两种情况下都可以调用。

我在GitHub上有一个名为StackFlairWidget的Proof-Of-Concept演示小部件应用程序。它显示用户的Stack Overflow“天赋”图像:

(这个项目正在进行中 - 使用POC Complete提交可能更容易看到与此问题相关的代码。它有一些错误,但比头版本更容易理解。)

这是一个允许以不同方式配置多个小部件的演示。你可以,例如在项目中搜索“appWidgetId”,以显示我如何围绕各种方法传递小部件ID,使用户能够正确配置小部件。

它显示了将小部件更新代码卸载到服务中的线程中的模式,以免使您的BroadcastReceiver过度紧张 - 您可能需要做一些与项目增长类似的事情。

因为它只是一个演示应用程序,它包含许多评论,一些过于复杂的代码(从生产应用程序中借用),但也有一些可怕的黑客(我道歉)。

基本上,正如您所说,您的更新代码是正确的。我的fwiw在FlairWidgetService班。它使用图像更新给定的小部件:

public void updateWidget(int appWidgetId, Bitmap flair)
{
    try
    {
        // set remote views
        RemoteViews widgetUi = new RemoteViews(getPackageName(), R.layout.widget);
        widgetUi.setImageViewBitmap(R.id.ivFlair, flair);

        // open Settings on click
        Intent flairSettings = new Intent(this, SettingsActivity.class);
        flairSettings.putExtra(IntentExtra.Key.APP_WIDGET_ID, appWidgetId);
        PendingIntent pendingIntentIp =
                PendingIntent.getActivity(this, appWidgetId, flairSettings,
                        PendingIntent.FLAG_UPDATE_CURRENT);
        widgetUi.setOnClickPendingIntent(R.id.ivFlair, pendingIntentIp);

        /* UPDATE THE WIDGET INSTANCE */
        try
        {
            AppWidgetManager widgetManager = AppWidgetManager.getInstance(this);
            widgetManager.updateAppWidget(appWidgetId, widgetUi);
        }
        catch (Exception e)
        {
            Dbug.log("Failed to update widget");
        }
    }
    catch (Exception e)
    {
        Dbug.log("Failed to update widget");
    }
    finally
    {
        // clean up
        FlairWidgetService.this.stopSelf();
    }
}

这不是唯一的方法,但我保证我会提供一些东西......祝你好运。


答案1(没有帮助海报,按照编辑1)

看起来你遇到了我认为在HoneyComb中发生的安全升级。

基本上,在较新版本的Android中,如果没有首先启动Activity,则无法让小部件正常工作。

所以在项目中添加一个活动,先运行它,我认为你的小部件会更新。

一个很好的例子是使用“设置”样式活动,或者可能使用“关于”框类型的事物来告诉用户有关您的窗口小部件的信息。 (我在我的小部件中使用设置)。


欲了解更多信息:

答案 1 :(得分:0)

如果我在TestReceiver

中使用此代码,它会起作用
    ComponentName thisAppWidget = new ComponentName(context.getPackageName(), TestWidgetProvider.class.getName());
     AppWidgetManager appWidgetManager = AppWidgetManager.getInstance(context.getApplicationContext());
     int ids[] = appWidgetManager.getAppWidgetIds(thisAppWidget);

     appWidgetManager.updateAppWidget(ids, tt);

并在AppWidgetProvider中的onUpdate中:

    ComponentName provider = new ComponentName(context, TestWidgetProvider.class);
    appWidgetManager.updateAppWidget ( provider, tt); //tt is RemoteViews tt=...

我不确定为什么,但这是我手机唯一有用的组合?

如果有人对此有更多了解,请解释,因为我需要制作一个复杂的服务,并结合警报管理器进行更新......

@Richard Le Mesurier感谢android 4.0需要为小部件提供活动的信息,并没有在任何地方找到它。