安卓 3.1 USB 主机 - 广播接收机未收到USB_DEVICE_ATTACHED

2022-09-01 03:50:22

我研究了 developer.android.com USB主机的描述和示例,以检测连接和分离的USB设备。

如果我在附加设备时使用清单文件中的意向过滤器来启动我的应用程序,它工作得很好:插入,设备被检测到,android请求启动应用程序的权限,设备信息显示在表中。

我正在开发的应用程序不应该只有在设备附加/分离时才启动/完成(例如,数据管理目的)。另外,如果应用程序已经在运行,我不希望弹出打开对话框。因此,我决定在设备连接时不直接启动活动,而是注册一个 BroadcastReceiver,如果设备处于打开状态/分离状态,它应该(稍后)通知该活动。此接收器可以很好地识别分离操作,但不能识别连接操作。

我是否缺少权限或数据属性或类似内容?本教程和示例没有说明其他必要的属性。

下面是清单文件:

<?xml version="1.0" encoding="utf-8"?>
<manifest 
  xmlns:android="http://schemas.android.com/apk/res/android"
  package="de.visira.smartfdr"
  android:versionCode="1"
  android:versionName="1.0">

<uses-sdk android:minSdkVersion="12" />
<uses-feature android:name="android.hardware.usb.host" />

<application android:icon="@drawable/icon" android:label="@string/app_name">


    <receiver android:name=".usb.Detector">
        <intent-filter>
            <action android:name="android.hardware.usb.action.USB_DEVICE_ATTACHED" />
            <action android:name="android.hardware.usb.action.USB_DEVICE_DETACHED" />
        </intent-filter>

        <meta-data android:name="android.hardware.usb.action.USB_DEVICE_ATTACHED"
            android:resource="@xml/device_filter" />
        <meta-data android:name="android.hardware.usb.action.USB_DEVICE_DETACHED"
            android:resource="@xml/device_filter" />
    </receiver>
</application>

接收器:

public class FDRDetector extends BroadcastReceiver{

@Override
public void onReceive(Context context, Intent intent) {
    String action = intent.getAction();

    Toast.makeText(context, "Action: " + action, 3).show();
            // pops up only if action == DETACHED
}

我不明白为什么相同的意图过滤器有效,如果我在活动上使用它们,但如果它们应用于接收器,则不然?即使我在代码中设置接收器和过滤器,也无法识别附加。

我的工作环境:IDE:带有Android插件的Eclipse 3.7

设备: 宏碁图标选项卡 A500

安卓: 3.1

提前致谢


答案 1

啊哈!我想通了。我遇到了完全相同的问题。

它的要点是 - 如果您在设备插入时自动启动应用程序(使用清单文件),那么Android系统似乎获得了ACTION_USB_DEVICE_ATTACHED意图,然后由于它知道您的应用程序想要在这种情况下运行,它实际上会向您的应用程序发送android.intent.action.MAIN意图。它从不向应用程序发送ACTION_USB_DEVICE_ATTACHED操作,因为它认为它已经知道应用程序在这种情况下要执行的操作。

我刚刚确定了问题,我想我有一个解决方案,但我可以告诉你我发现了什么:

即使你的应用正在运行且位于前台,当你插入 USB 设备并且 Android 系统获得ACTION_USB_DEVICE_ATTACHED意图时,它将在你的活动中调用 onResume()。

不幸的是,你不能只这样做:

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

    Intent intent = getIntent();
    Log.d(TAG, "intent: " + intent);
    String action = intent.getAction();

    if (UsbManager.ACTION_USB_DEVICE_ATTACHED.equals(action)) {
        //do something
    } 
}

因为意图将返回为android.intent.action.MAIN,而不是ACTION_USB_DEVICE_ATTACHED。

令人讨厌的是,如果您只是离开应用程序,但不要拔下USB,您也会获得android.intent.action.MAIN。我想把设备放到睡眠状态并重新唤醒它也会做同样的事情。

因此,从我发现的情况来看,您无法直接获得意图,但是看起来您可以依靠在插入USB设备时调用Resume(),因此解决方案是每次获得onResume时检查USB是否连接。您也可以在USB断开连接时设置一个标志,因为USB断开连接的意图当然会触发。

因此,总的来说,您的广播接收器可能如下所示:

// BroadcastReceiver when remove the device USB plug from a USB port  
BroadcastReceiver mUsbReceiver = new BroadcastReceiver() {
    public void onReceive(Context context, Intent intent) {
        String action = intent.getAction();
         if (UsbManager.ACTION_USB_DEVICE_DETACHED.equals(action)) {                
        usbConnected=false;             
        }
    }
};

你会在 onCreate 内部有这个:

  // listen for new devices
 IntentFilter filter = new IntentFilter();
 filter.addAction(UsbManager.ACTION_USB_DEVICE_DETACHED);
 registerReceiver(mUsbReceiver, filter);

它位于清单中的活动标记内:

        <intent-filter>
            <action android:name="android.hardware.usb.action.USB_DEVICE_ATTACHED" />
        </intent-filter>

        <meta-data android:name="android.hardware.usb.action.USB_DEVICE_ATTACHED"
            android:resource="@xml/device_filter" />

在 /res/xml/ 文件夹中将有一个device_filter.xml文件,如下所示:

<?xml version="1.0" encoding="utf-8"?>

<resources>
    <usb-device vendor-id="1027" product-id="24577" />  
    <usb-device vendor-id="1118" product-id="688" /> 
</resources>

(当然,使用您需要的任何供应商 ID 和产品 ID)

然后你的 onCreate 看起来像这样:

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

    Intent intent = getIntent();
    Log.d(TAG, "intent: " + intent);
    String action = intent.getAction();


    if (usbConnected==false ) {
        //check to see if USB is now connected
    } 
}

我没有特定的代码来检查USB是否连接,因为我实际上还没有深入研究。我正在使用一个库,如果可以的话,它就会连接,所以对于我的应用程序,我可以启动这个循环,我很好。

将清单中活动的启动模式设置为“singleTask”可能也很重要,以防止它在已经运行时再次运行,否则插入USB设备将只启动应用程序的第二个实例!

因此,我的清单中的整个活动标记如下所示:

    <activity
        android:label="@string/app_name"
        android:name="com.awitness.common.TorqueTablet"
        android:theme="@android:style/Theme.Holo.NoActionBar.Fullscreen"
        android:screenOrientation="landscape"
        android:configChanges="orientation|keyboardHidden" 
        android:launchMode="singleTask"
        >
        <intent-filter >
            <action android:name="android.intent.action.MAIN" />
            <category android:name="android.intent.category.HOME"/>
        <category android:name="android.intent.category.DEFAULT" />
        <category android:name="android.intent.category.LAUNCHER" />
        </intent-filter> 

        <intent-filter>
            <action android:name="android.hardware.usb.action.USB_DEVICE_ATTACHED" />
        </intent-filter>

        <meta-data android:name="android.hardware.usb.action.USB_DEVICE_ATTACHED"
            android:resource="@xml/device_filter" />

    </activity>

无论如何,我希望这对某人有所帮助!我很惊讶我已经无法找到解决方案了!


答案 2

接下来@Gusdor的有见地的评论(+1):我实现了一个检查,正如@Gusdor指出的那样,当你的活动设置为或时,它会被调用。然后,不要像接受的答案所建议的那样检查布尔标志,只需使用 .例如onNewIntent()launchModesingleTasksingleTopLocalBroadcastManager

@Override
protected void onNewIntent(Intent intent) {
    super.onNewIntent(intent);
    if (UsbManager.ACTION_USB_ACCESSORY_ATTACHED.equals(intent.getAction())) {
        LocalBroadcastManager.getInstance(this).sendBroadcast(intent);
    }
}

然后,无论您在何处注册现有的(系统)USB 广播接收器,只需向本地广播管理器实例注册相同的接收器,即

@Override
protected void onResume() {
    super.onResume();
    myContext.registerReceiver(myUsbBroadcastReceiver, myIntent); // system receiver
    LocalBroadcastManager.getInstance(myContext).registerReceiver(myUsbBroadcastReceiver, intent); // local receiver
}

@Override
protected void onPause() {
    super.onResume();
    myContext.unregisterReceiver(myUsbBroadcastReceiver); // system receiver
    LocalBroadcastManager.getInstance(myContext).unregisterReceiver(myUsbBroadcastReceiver); // local receiver
}

您可以发送另一个系统广播而不是本地广播,但我认为您无法使用该操作(系统会将其视为潜在的安全风险),因此您必须定义自己的操作。没什么大不了的,但为什么要打扰,特别是因为本地广播没有IPC开销。UsbManager.ACTION_USB_ACCESSORY_ATTACHED


推荐