Почему функция базового класса onReceive () не вызывает onUpdate ()? - PullRequest
0 голосов
/ 17 января 2012

У меня есть довольно простое приложение виджетов на главном экране, которое показывает тост по нажатию кнопки.К счастью, он отображает тост только при включении питания (или сразу после обновления apk в эмуляторе).Нажатие кнопки имеет ожидающее намерение, которое отправляет ACTION_APPWIDGET_UPDATE, но функция onReceive () базового класса не обрабатывает его.

Вот мой код:

public class WordWidget extends AppWidgetProvider {

    @Override
    public void onReceive(Context ctxt, Intent intent) {

        Log.d("WordWidget.onReceive", "onReceive");
        Log.d("WordWidget.onReceive", intent.getAction());

        if (intent.getAction()==null) {
            ctxt.startService(new Intent(ctxt, UpdateService.class));
        } else {
            super.onReceive(ctxt, intent);
        }
    }

    @Override
    public void onUpdate(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds) {

        // To prevent any ANR timeouts, we perform the update in a service
        Log.d("WordWidget.onUpdate", "onUpdate");

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

    public static class UpdateService extends Service {
        @Override
        public void onStart(Intent intent, int startId) {
            Log.d("UpdateService.onStart", "onStart");

            // Build the widget update for today
            RemoteViews updateViews = buildUpdate(this);

            // Push update for this widget to the home screen
            ComponentName thisWidget = new ComponentName(this, WordWidget.class);
            AppWidgetManager manager = AppWidgetManager.getInstance(this);
            manager.updateAppWidget(thisWidget, updateViews);
        }

        public RemoteViews buildUpdate(Context context) {

            Log.d("UpdateService.buildUpdate", "buildUpdate");

            RemoteViews updateViews = new RemoteViews(context.getPackageName(), R.layout.widget_word);

            Intent defineIntent = new Intent(context, WordWidget.class);
            defineIntent.setAction(AppWidgetManager.ACTION_APPWIDGET_UPDATE);

            PendingIntent pendingIntent = PendingIntent.getBroadcast(context, 0, defineIntent, 0);

            updateViews.setOnClickPendingIntent(R.id.widget, pendingIntent);

            showToast(context);

            return updateViews;
        }

        private void showToast(Context context){
            Log.d("UpdateService.showToast", "showToast");
            Toast.makeText(context, "It is working...", Toast.LENGTH_LONG).show();          
        }

        @Override
        public IBinder onBind(Intent intent) {
            // We don't need to bind to this service
            return null;
        }
    }
}

На power-вверх (или сразу после переустановки) мой виджет показывает тост, вот LogCat:

01-17 13:03:10.855: I/ActivityManager(72): Start proc com.example.android.simplewiktionary for broadcast com.example.android.simplewiktionary/.WordWidget: pid=564 uid=10036 gids={3003}
01-17 13:03:11.045: D/WordWidget.onReceive(564): onReceive
01-17 13:03:11.045: D/WordWidget.onReceive(564): android.appwidget.action.APPWIDGET_UPDATE
01-17 13:03:11.054: D/WordWidget.onUpdate(564): onUpdate
01-17 13:03:11.075: D/UpdateService.onStart(564): onStart
01-17 13:03:11.075: D/UpdateService.buildUpdate(564): buildUpdate
01-17 13:03:11.094: D/UpdateService.showToast(564): showToast
01-17 13:03:12.655: D/dalvikvm(72): GC_EXPLICIT freed 971K, 47% free 13550K/25159K, paused 7ms+52ms

Теперь, когда я начал, если я нажму на кнопку, вот LogCat:

01-17 13:04:16.095: D/dalvikvm(126): GC_EXPLICIT freed 72K, 14% free 14016K/16135K, paused 3ms+3ms
01-17 13:04:16.277: D/WordWidget.onReceive(564): onReceive
01-17 13:04:16.277: D/WordWidget.onReceive(564): android.appwidget.action.APPWIDGET_UPDATE
01-17 13:04:21.365: D/dalvikvm(564): GC_EXPLICIT freed 71K, 5% free 6307K/6595K, paused 3ms+2ms

Как видите, onUpdate () не вызывается функцией базового класса onReceive () ...

Не могли бы вы помочь мне с этим?

Спасибо.

Ответы [ 2 ]

3 голосов
/ 18 января 2012

Я наконец нашел причину, по которой мой собственный onUpdate () не был выполнен super.onReceive () при нажатии кнопки.

Создание намерения с действием, установленным в ACTION_APPWIDGET_UPDATE, недостаточно для запускавызов моей переопределенной функции onUpdate () с помощью вызова super.onReceive ().В соответствии с кодом функции onReceive () класса AppWidgetProvider намерение должно содержать некоторые данные в поле «Дополнительно».Вот почему onUpdate () вызывается правильно в первый раз (когда это реальное намерение ACTION_APPWIDGET_UPDATE от системы), а не когда я нажимаю на кнопку ...

Вот код из AppWidgetProvider.Java-файл:

// BEGIN_INCLUDE(onReceive)
public void  [More ...] onReceive(Context context, Intent intent) {
 // Protect against rogue update broadcasts (not really a security issue,
 // just filter bad broacasts out so subclasses are less likely to crash).
 String action = intent.getAction();
 if (AppWidgetManager.ACTION_APPWIDGET_UPDATE.equals(action)) {
     Bundle extras = intent.getExtras();
     if (extras != null) {
         int[] appWidgetIds = extras.getIntArray(AppWidgetManager.EXTRA_APPWIDGET_IDS);
         if (appWidgetIds != null && appWidgetIds.length > 0) {
             this.onUpdate(context, AppWidgetManager.getInstance(context), appWidgetIds);
         }
     }
 }
 else if (AppWidgetManager.ACTION_APPWIDGET_DELETED.equals(action)) {
     Bundle extras = intent.getExtras();
     if (extras != null && extras.containsKey(AppWidgetManager.EXTRA_APPWIDGET_ID)) {
         final int appWidgetId = extras.getInt(AppWidgetManager.EXTRA_APPWIDGET_ID);
         this.onDeleted(context, new int[] { appWidgetId });
     }
 }
 else if (AppWidgetManager.ACTION_APPWIDGET_ENABLED.equals(action)) {
     this.onEnabled(context);
 }
 else if (AppWidgetManager.ACTION_APPWIDGET_DISABLED.equals(action)) {
     this.onDisabled(context);
 }
}

Спасибо.

0 голосов
/ 17 января 2012

Я бы порекомендовал вместо того, чтобы пытаться вызвать onUpdate, установить PendingIntent для запуска службы:

Intent intent = new Intent(context, UpdateService.class);    
PendingIntent pi = PendingIntent.getService(context, 0, intent, 0);

Дайте мне знать, если это не работает или еслиэто не жизнеспособное решение для вас.

...