在小部件中更新Android ListView

3
我在更新App-Widget内容时遇到了问题。该小工具的外观如下: http://i.stack.imgur.com/tEGLv.jpg
该小工具由标题(按钮+文本视图)和列表视图组成。如果用户按下按钮,则列表视图会显示第二天的内容,一切都很好。但是问题是,如果我尝试从我的应用程序或自动时间段更新小部件,则列表视图不会更新。
以下是我的代码。
AppWidgetProvider:
public class StudyWidgetProvider extends AppWidgetProvider{

private static String BUTTON_BACK = "BUTTON_BACK";
private static String BUTTON_FORWARD = "BUTTON_FORWARD";

@Override
public void onUpdate(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds) {  
    context.startService(new Intent(context, UpdateService.class));
}

@Override
public void onReceive(Context context, Intent intent) {
    super.onReceive(context, intent);
    if(WidgetViewsFactory.currentDate == null){
        WidgetViewsFactory.currentDate = Calendar.getInstance();
    }

    String action = intent.getAction();
    if(action != null){
        if(action.equals(BUTTON_BACK)){
            WidgetViewsFactory.currentDate.add(Calendar.DATE, -1);
        }
        if(action.equals(BUTTON_FORWARD)){
            WidgetViewsFactory.currentDate.add(Calendar.DATE, 1);
        }
    }

    context.startService(new Intent(context, UpdateService.class));
}   
 }

更新服务负责小部件的更新。
    @Override
@Deprecated
public void onStart(Intent intent, int startId) {
    super.onStart(intent, startId);
    Log.i(LOG_TAG, "Update-Service started");

    RemoteViews widget = buildUpdate(this);
    ComponentName thisWidget = new ComponentName(this, StudyWidgetProvider.class);
    AppWidgetManager manager = AppWidgetManager.getInstance(this);

    manager.notifyAppWidgetViewDataChanged(manager.getAppWidgetIds(thisWidget), R.id.words);
    manager.updateAppWidget(thisWidget, widget);

    Log.i(LOG_TAG, "The Content!");
    if(widgetObjects == null){
        Log.i(LOG_TAG, "WidgetObjects sind NULL");
    } else {
        Log.i(LOG_TAG, "WidgetObjects sind NICHT-NULL");

        for(int i=0; i<widgetObjects.size(); i++){
            Log.i(LOG_TAG, "--- " + widgetObjects.get(i).getName());
        }
    }

    Log.i(LOG_TAG, "Update-Service ended");
}


public RemoteViews buildUpdate(Context context){
    RemoteViews retVal = new RemoteViews(context.getPackageName(), R.layout.widget);

    if(WidgetViewsFactory.currentDate == null){
        WidgetViewsFactory.currentDate = Calendar.getInstance();
    }

    AppWidgetManager appWidgetManager = AppWidgetManager.getInstance(context);
    ComponentName thisAppWidget = new ComponentName(context.getPackageName(), StudyWidgetProvider.class.getName());
    int[] appWidgetIds = appWidgetManager.getAppWidgetIds(thisAppWidget);

    loadSDCardContent(context, appWidgetManager, appWidgetIds);

    //Setzen des Widget-UIs
    Intent intent = new Intent(context, WidgetService.class);
    intent.putExtra(AppWidgetManager.EXTRA_APPWIDGET_ID, appWidgetIds);
    intent.setData(Uri.parse(intent.toUri(Intent.URI_INTENT_SCHEME)));

    retVal.setRemoteAdapter(R.id.words, intent);

    SimpleDateFormat dateFormater = new SimpleDateFormat("EEEE");
    retVal.setTextViewText(R.id.widget_day, dateFormater.format(WidgetViewsFactory.currentDate.getTime()));
    dateFormater = new SimpleDateFormat("dd.MM.yyyy");
    retVal.setTextViewText(R.id.widget_date, dateFormater.format(WidgetViewsFactory.currentDate.getTime()));

    //Button-Actions
    Intent backIntent = new Intent(context, StudyWidgetProvider.class);
    backIntent.setAction(BUTTON_BACK);
    Intent forwardIntent = new Intent(context, StudyWidgetProvider.class);
    forwardIntent.setAction(BUTTON_FORWARD);
    PendingIntent backPendingIntent = PendingIntent.getBroadcast(context, 0, backIntent, PendingIntent.FLAG_UPDATE_CURRENT);
    PendingIntent forwardPendingIntent = PendingIntent.getBroadcast(context, 0, forwardIntent, PendingIntent.FLAG_UPDATE_CURRENT);
    retVal.setOnClickPendingIntent(R.id.widget_but_back, backPendingIntent);
    retVal.setOnClickPendingIntent(R.id.widget_but_forward, forwardPendingIntent);

    Intent clickIntent = new Intent(context, WidgetActivity.class);
    PendingIntent clickPI=PendingIntent.getActivity(context, 0, clickIntent, PendingIntent.FLAG_UPDATE_CURRENT);
    retVal.setPendingIntentTemplate(R.id.words, clickPI);

    return retVal;
}

这是我如何从我的应用程序更新小部件的方式:

public RemoteViews buildUpdate(Context context){
    RemoteViews retVal = new RemoteViews(context.getPackageName(), R.layout.widget);

    if(WidgetViewsFactory.currentDate == null){
        WidgetViewsFactory.currentDate = Calendar.getInstance();
    }

    AppWidgetManager appWidgetManager = AppWidgetManager.getInstance(context);
    ComponentName thisAppWidget = new ComponentName(context.getPackageName(), StudyWidgetProvider.class.getName());
    int[] appWidgetIds = appWidgetManager.getAppWidgetIds(thisAppWidget);

    loadSDCardContent(context, appWidgetManager, appWidgetIds);

    //Setzen des Widget-UIs
    Intent intent = new Intent(context, WidgetService.class);
    intent.putExtra(AppWidgetManager.EXTRA_APPWIDGET_ID, appWidgetIds);
    intent.setData(Uri.parse(intent.toUri(Intent.URI_INTENT_SCHEME)));

    retVal.setRemoteAdapter(R.id.words, intent);

    SimpleDateFormat dateFormater = new SimpleDateFormat("EEEE");
    retVal.setTextViewText(R.id.widget_day, dateFormater.format(WidgetViewsFactory.currentDate.getTime()));
    dateFormater = new SimpleDateFormat("dd.MM.yyyy");
    retVal.setTextViewText(R.id.widget_date, dateFormater.format(WidgetViewsFactory.currentDate.getTime()));

    //Button-Actions
    Intent backIntent = new Intent(context, StudyWidgetProvider.class);
    backIntent.setAction(BUTTON_BACK);
    Intent forwardIntent = new Intent(context, StudyWidgetProvider.class);
    forwardIntent.setAction(BUTTON_FORWARD);
    PendingIntent backPendingIntent = PendingIntent.getBroadcast(context, 0, backIntent, PendingIntent.FLAG_UPDATE_CURRENT);
    PendingIntent forwardPendingIntent = PendingIntent.getBroadcast(context, 0, forwardIntent, PendingIntent.FLAG_UPDATE_CURRENT);
    retVal.setOnClickPendingIntent(R.id.widget_but_back, backPendingIntent);
    retVal.setOnClickPendingIntent(R.id.widget_but_forward, forwardPendingIntent);

    Intent clickIntent = new Intent(context, WidgetActivity.class);
    PendingIntent clickPI=PendingIntent.getActivity(context, 0, clickIntent, PendingIntent.FLAG_UPDATE_CURRENT);
    retVal.setPendingIntentTemplate(R.id.words, clickPI);

    return retVal;
}

奇怪的是,当我点击小部件的按钮时,一切都按照我所希望的方式工作。我的listview的onDatasetChanged方法被调用并且内容正在改变,但如果我在我的应用程序中尝试这样做,则什么也不会发生。
我希望你们中的某个人知道为什么这不起作用。 我已经卡在这里两天了,一点头绪也没有...
谢谢帮助!
编辑:我发现这在我的设备(MIUI V4的htc)上无法使用,但在模拟器上可以。
编辑2:我在我的设备上安装了Android 4(具有Sense 4),之后它也起作用了。看来MIUI是问题所在。

你认为user1392070是一个好的用户名选择吗? - Code Droid
发出一个stickyBroadcast以更新您的小部件。 - Code Droid
使用广播与小部件通信,而不是所有这些代码。不要直接像这样调用onUpdate。 - Code Droid
我尝试使用BroadcastReceiver进行更新,但结果相同。 - iTamp
同样的问题...希望有人有解决办法。 - JMRboosties
有什么解决方案吗?你说过,“我的listview的onDatasetChanged方法被调用并且内容正在改变”,但是你在onDataSetChanged中放了什么代码来更新列表的内容呢?谢谢。 - avlacatus
3个回答

7

我要翻译这个问题,因为我遇到了相同的问题,并找到了解决办法。

              manager.notifyAppWidgetViewDataChanged

这会向你的RemoteViewsFactory发送一个onDataSetChanged()调用。在该方法中添加构建视图的调用。这解决了我所有的问题。

我有同样的问题,请参考我的链接并给我解决方案...https://dev59.com/DHbZa4cB1Zd3GeqPD0jz - Satheesh
manager.notifyAppWidgetViewDataChanged 应该放在哪里? - natsumiyu

1

只能通过广播与小部件进行通信。不要直接调用它。如有需要,请使用stickyBroadcasts。


我尝试使用BroadcastReceiver进行更新,但结果仍然相同。 - iTamp

0

这个有效

 @Override
public void onUpdate(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds) {
    super.onUpdate(context, appWidgetManager, appWidgetIds);
    for (int widgetId : appWidgetIds) {
        RemoteViews mView = initViews(context, appWidgetManager, widgetId);
        appWidgetManager.updateAppWidget(widgetId, mView);
    }

    super.onUpdate(context, appWidgetManager, appWidgetIds);
}



@SuppressWarnings("deprecation")
@SuppressLint("NewApi")
private RemoteViews initViews(Context context,
                              AppWidgetManager widgetManager, int widgetId) {

    RemoteViews mView = new RemoteViews(context.getPackageName(),
            R.layout.widget_layout);

    Intent intent = new Intent(context, WidgetService.class);
    intent.putExtra(AppWidgetManager.EXTRA_APPWIDGET_ID, widgetId);

    intent.setData(Uri.parse(intent.toUri(Intent.URI_INTENT_SCHEME)));
    mView.setRemoteAdapter(widgetId, R.id.widgetCollectionList, intent);

    return mView;
}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接