如何在“离线”模式下检测耳机插头事件

时间:2013-05-07 00:21:15

标签: android broadcastreceiver

我知道如果我的应用程序正在运行,如何检测耳机插件事件。您必须为ACTION_HEADSET_PLUG注册广播接收器。但是你不能使用广播接收器的Manifest声明来捕获这个动作。因此,捕获耳机插件事件的唯一选择是后台服务。但它会耗尽电池等等。

我查看了一些音乐播放器并发现他们在没有任何其他服务的情况下捕获了该事件。他们是怎么做到的?有什么想法吗?

1 个答案:

答案 0 :(得分:15)

我在商店里有一个应用程序已有三年监控有线耳机和蓝牙状态,没有人抱怨电池耗尽。但那是因为我成功地使用单个服务和广播接收器来检测来自两者的事件。这是两个关键类:

public class HeadsetStateBroadcastReceiver extends BroadcastReceiver {

    public static final String[] HEADPHONE_ACTIONS = {
        Intent.ACTION_HEADSET_PLUG,
        "android.bluetooth.headset.action.STATE_CHANGED",
        "android.bluetooth.headset.profile.action.CONNECTION_STATE_CHANGED"
    };

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

        boolean broadcast = false;

        // Wired headset monitoring
        if (intent.getAction().equals(HEADPHONE_ACTIONS[0]) {
            final int state = intent.getIntExtra("state", 0);
            AudioPreferences.setWiredHeadphoneState(context, state > 0);
            broadcast = true;
        }

        // Bluetooth monitoring
        // Works up to and including Honeycomb
        if (intent.getAction().equals(HEADPHONE_ACTIONS[1])) {
            int state = intent.getIntExtra("android.bluetooth.headset.extra.STATE", 0);
            AudioPreferences.setBluetoothHeadsetState(context, state == 2);
            broadcast = true;
        }

        // Works for Ice Cream Sandwich
        if (intent.getAction().equals(HEADPHONE_ACTIONS[2])) {
            int state = intent.getIntExtra("android.bluetooth.profile.extra.STATE", 0);
            AudioPreferences.setBluetoothHeadsetState(context, state == 2);
            broadcast = true;
        }

        // Used to inform interested activities that the headset state has changed
        if (broadcast) {
            LocalBroadcastManager.getInstance(context).sendBroadcast(new Intent("headsetStateChange"));
        }

    }

}

这是我用来注册广播接收器的服务:

public class HeadsetMonitoringService extends Service {

    HeadsetStateBroadcastReceiver headsetStateReceiver;

    @Override
    public void onCreate() {

        headsetStateReceiver = new HeadsetStateBroadcastReceiver();
        final IntentFilter filter = new IntentFilter();
        for (String action: HeadsetStateBroadcastReceiver.HEADPHONE_ACTIONS) {
            filter.addAction(action);
        }

        registerReceiver(headsetStateReceiver, filter);

    }

    @Override
    public int onStartCommand(final Intent intent, final int flags, final int startId) {
        return START_STICKY;
    }

    @Override
    public void onDestroy() {
        unregisterReceiver(headsetStateReceiver);
    }

    @Override
    public IBinder onBind(final Intent intent) {
        return null;
    }

}

这是我的明显条目:

    <service
        android:name=".services.HeadsetMonitoringService"
        android:enabled="true"
        android:exported="false" >
        <intent-filter>
            <action android:name="initialiseHeadsetService" />
        </intent-filter>
    </service>

工作原理如下:

我使用on boot广播接收器向HeadsetMonitoringService发送启动服务消息(您不必这样做,您可以在应用程序启动时执行此操作)。 HeadsetMonitoringService依次注册一个广播监听器的实例,该监听器监听我感兴趣的所有耳机​​事件 - 它们保存在HEADPHONE_ACTIONS数组中。因为服务是粘性的,它会挂起 - 因此广播监听器也是如此。但是因为服务和广播监听器都是事件驱动的,所以在耳机状态发生变化之前它们不消耗任何电力。此外,由于服务是粘性的,如果操作系统意外死亡,它将重新启动 每当我收到耳机状态更改事件时,我也会触发本地广播,以便感兴趣的活动可以检查新状态并在需要时采取措施。

为了完整性,我应该指出我使用另一个类(未在此处显示)AudioPreferences来存储蓝牙和有线耳机状态的首选项,然后只要我需要知道耳机状态就可以访问它。 / p>

如果您对蓝牙耳机的状态感兴趣,您的应用程序将需要android.permission.BLUETOOTH权限。如果没有,只需从HEADPHONE_ACTIONS数组中取出蓝牙相关的操作,并从onReceive方法中删除关联的if块。