Android BroadcastReceiver不更新小部件

Posted

技术标签:

【中文标题】Android BroadcastReceiver不更新小部件【英文标题】:Android BroadcastReceiver not updating Widget 【发布时间】:2016-11-18 10:57:43 【问题描述】:

我尝试了所有在 *** 上找到的方法,但没有一个有效。

我正在开发一个电池监控应用程序,它包含一些用于显示信息的小部件。我使用一个名为BatteryUpdateReceiver 的通用接收器来收听ACTION_BATTERY_CHANGED 事件。我使用这个接收器来更新一个显示当前电池电量的小部件。但由于某些原因,它不会更新小部件。我尝试了很多方法,但无法弄清楚出了什么问题。我的代码如下。

BatteryUpdateReceiver.java

public void onReceive(Context context, Intent intent) 
        if (intent.getAction().equals(Intent.ACTION_BATTERY_CHANGED)) 
            Log.i(TAG, "Receiver ACTION_BATTERY_CHANGED fired");

            RemoteViews remoteViews = new RemoteViews(context.getPackageName(),
                    R.layout.widget);

            AppWidgetManager appWidgetManager = AppWidgetManager.getInstance(context.getApplicationContext());
            ComponentName thisWidget = new ComponentName(context.getApplicationContext(), BatteryWidget.class);
            int[] allWidgetIds = appWidgetManager.getAppWidgetIds(thisWidget);
            for (int i=0; i < allWidgetIds.length; i++) 
                Log.i(TAG, "WID ID: " + allWidgetIds[i]);
                appWidgetManager.updateAppWidget(allWidgetIds[i], remoteViews);
            
        
    

BatteryWidget.java

public void onUpdate(Context context, AppWidgetManager appWidgetManager, int[] appWidgetIds) 
        super.onUpdate(context, appWidgetManager, appWidgetIds);

        Log.i(TAG, "onUpdate: ");

        int currentLevel = calculateBatteryLevel(context);
        if (batteryChanged(currentLevel)) 
            batteryLevel = currentLevel;
            Log.i(TAG, "onUpdate: Battery level changed");
        
        updateViews(context);
    

    private boolean batteryChanged(int currentLevelLeft) 
        return (batteryLevel != currentLevelLeft);
    

    private int calculateBatteryLevel(Context context) 
        Log.i(TAG, "calculateBatteryLevel: ");

        Intent batteryIntent = context.getApplicationContext().registerReceiver(null, new IntentFilter(Intent.ACTION_BATTERY_CHANGED));

        int level = batteryIntent.getIntExtra(BatteryManager.EXTRA_LEVEL, 0);
        int scale = batteryIntent.getIntExtra(BatteryManager.EXTRA_SCALE, 100);
        return level * 100 / scale;
    

    private void updateViews(Context context) 
        Log.i(TAG, "updateViews: ");

        RemoteViews views = new RemoteViews(context.getPackageName(), R.layout.widget);
        views.setTextViewText(R.id.batteryText, batteryLevel + "%");

        ComponentName componentName = new ComponentName(context, BatteryWidget.class);
        AppWidgetManager appWidgetManager = AppWidgetManager.getInstance(context);
        appWidgetManager.updateAppWidget(componentName, views);
    

AndroidManifest.xml

    <receiver android:name=".lib.receivers.BatteryUpdateReceiver" android:enabled="true" />

    <receiver android:name=".widget.BatteryWidget" android:label="@string/app_name" >
         <intent-filter>
              <action android:name="android.appwidget.action.APPWIDGET_UPDATE" />
         </intent-filter>
         <meta-data
                    android:name="android.appwidget.provider" android:resource="@xml/batterywidgetinfo" />
   </receiver>

我在设备启动时使用服务注册 BatteryUpdateReceiver。

IntentFilter filterBattery = new IntentFilter();
filterBattery.addAction(Intent.ACTION_BATTERY_CHANGED);
this.mBatteryStatusReceiver = new BatteryUpdateReceiver();
getApplicationContext().registerReceiver(this.mBatteryStatusReceiver, filterBattery);
Log.i(TAG, "Receiver ACTION_BATTERY_CHANGED registered by Service");

我可以确认 BatteryUpdateReceiver 有效,因为它记录了小部件 ID。此外,如果我将接收器放在BatteryWidget 中,那么小部件就会更新。有人可以告诉我我在这里想念什么吗?我知道我可以覆盖BatteryWidget 类中的onReceive() 方法。但是我怎样才能以这种方式存档呢?

谢谢!

【问题讨论】:

【参考方案1】:

你应该从你的接收者发送一个ACTION_APPWIDGET_UPDATE广播,而不是直接调用updateAppWidget()

类似这样的:

@Override
public void onReceive(Context context, Intent intent) 
    AppWidgetManager widgetManager = AppWidgetManager.getInstance(getApplicationContext());
    int[] appWidgetIds = widgetManager.getAppWidgetIds(new ComponentName(context, BatteryWidget.class));

    Intent updateIntent = new Intent(AppWidgetManager.ACTION_APPWIDGET_UPDATE, null, getApplicationContext(), BatteryWidget.class);
    updateIntent.putExtra(AppWidgetManager.EXTRA_APPWIDGET_IDS, appWidgetIds);

    sendBroadcast(updateIntent);

【讨论】:

【参考方案2】:

在同一问题上花费了无数小时后,我终于找到了解决方案。 @earthw0rmjim 的回答把我推到了一半,但我意识到你不能从 Kotlin 广播接收器调用 sendBroadcast()。您必须调用广播上下文和应用程序上下文才能发送广播。

对于任何希望在 Kotlin 中实现相同目标的人来说,这是对我有用的代码。在我的例子中,MyWidgetProvider 是我用来处理主屏幕小部件的 AppWidgetProvider 类。

override fun onReceive(context: Context, intent: Intent) 
    val widgetManager = AppWidgetManager.getInstance(context)
    
    val appWidgetIds = widgetManager.getAppWidgetIds(ComponentName(context,MyWidgetProvider::class.java))

    val updateIntent = Intent(
        AppWidgetManager.ACTION_APPWIDGET_UPDATE,
        null,
        context,
        MyWidgetProvider::class.java
    )
    
    updateIntent.putExtra(AppWidgetManager.EXTRA_APPWIDGET_IDS, appWidgetIds)

    context.applicationContext.sendBroadcast(updateIntent)

【讨论】:

以上是关于Android BroadcastReceiver不更新小部件的主要内容,如果未能解决你的问题,请参考以下文章

Android_组件_BroadcastReceiver基础

Android BroadcastReceiver广播:基本使用

Android -- BroadCastReceiver的简单使用

Android BroadcastReceiver

android从broadcastreceiver刷新listview

Android开发实践 BroadcastReceiver