Android BroadcastReceiver android.intent.action.BATTERY_CHANGED - PullRequest
7 голосов
/ 28 июня 2011

Я могу прослушать Намерение Intent.ACTION_BATTERY_CHANGED в своей деятельности, и оно прекрасно работает. Я попытался определить получателя в манифесте:

<receiver android:name="com.beargreaves.battery.BatteryReceiver"> 
    <intent-filter> 
        <action android:name="android.intent.action.BATTERY_CHANGED" />
    </intent-filter> 
</receiver>

но это не сработало, я прочитал здесь пост о том, что он должен быть определен программно. Вместо того, чтобы регистрироваться для получателя в моей Активности, я хотел добиться этого в Сервисе, чтобы он постоянно отслеживал. Я успешно достиг этого, и это работает, но я хотел проверить свою работу, чтобы убедиться, что это правильный подход.

Я начал с расширения BroadcastReceiver:

public class BatteryReceiver extends BroadcastReceiver{
    @Override
    public void onReceive(Context context, Intent intent) {
        Bundle bundle = intent.getExtras();

        if(null == bundle)
            return;

        boolean isPresent = intent.getBooleanExtra("present", false);
        String technology = intent.getStringExtra("technology");
        int plugged = intent.getIntExtra("plugged", -1);
        int scale = intent.getIntExtra("scale", -1);
        int health = intent.getIntExtra("health", 0);
        int status = intent.getIntExtra("status", 0);
        int rawlevel = intent.getIntExtra("level", -1);
        int level = 0;

        Log.d("Debug","Battery Receiver OnReceive");

        if(isPresent) {
            if (rawlevel >= 0 && scale > 0) {
                level = (rawlevel * 100) / scale;

                Log.d("Debug","BatterReceiver: " + level);

                Toast.makeText(context,"Battery Receiver: " + level + "\t" + status + "Raw: " + rawlevel,Toast.LENGTH_LONG).show();

                if(level <60) {
                    /*
                     * Only invoke the service when level below threshold
                     */
                    Intent i = new Intent(context, BatteryService.class);
                    i.putExtra("level", level);
                    context.startService(i);
                }
            }
        }
    }
}

А затем я использую Службу, чтобы сначала зарегистрировать получателя в onCreate(), а затем обработать события в onStart(). Мой BroadcastReceiver запускает Сервис, если уровень ниже порогового.

public class BatteryService extends Service {
    /*
     * First Call to onStart we don't want to do anything
     */
    boolean avoidFirst = false;
    private BroadcastReceiver receiver;

    @Override
    public IBinder onBind(Intent intent) {
        // TODO Auto-generated method stub
        return null;
    }

    @Override
    public void onCreate() {
        super.onCreate();

        IntentFilter filter = new IntentFilter(Intent.ACTION_BATTERY_CHANGED);
        receiver = new BatteryReceiver();
        registerReceiver(receiver, filter);
    }

    @Override
    public void onStart(Intent intent, int startId) {
        Log.d("Debug","Battery Service On Start");
        int level = intent.getIntExtra("level", -1);

        if(avoidFirst) {
            if(level != -1) {
                Log.d("Debug","Battery Alert Notifying..... " + level);
                SMSManager.sendSMS(BatteryService.this, "<number redacted>", "Battery Level Aleart: " + level);
                stopSelf();
            }
        } else {
            avoidFirst = true;
        }
    }

    @Override
    public void onDestroy() {
        // TODO Auto-generated method stub
        super.onDestroy();
        PreferenceUtil.updatePreference(BatteryService.this, "battery_monitor_on", false);
        unregisterReceiver(receiver);
    }
}

Это правильный подход? Будь то зарегистрируй получателя в onCreate() и затем запусти Сервис, когда событие получено. Сначала я попытался не использовать Сервис, но потом у меня нет возможности зарегистрировать получателя, так как это не может быть достигнуто в манифесте. Во-вторых, возникла исключительная ситуация, когда я попытался отправить текстовое сообщение в onReceive(). Я прочитал, что onReceive() не должен запускать какие-либо темы.

Заранее спасибо

1 Ответ

2 голосов
/ 02 июля 2012

Да, это выглядит как очень хороший подход. Это довольно похоже (на самом деле несколько лучше, я бы сказал), чем то, что я делаю с моим приложением (которое хорошо работает в течение многих лет).

...