android - 如何模拟来自其他应用程序的标签触摸

标签 android android-intent nfc intentfilter ndef

我想为我的应用程序模拟触摸事件。我的 list 就像

<activity
    android:name=".activity.TagActivity_"
    android:label="@string/app_name"
    android:launchMode="singleTask"
    android:noHistory="true"
    android:permission="android.permission.NFC"
    android:screenOrientation="portrait" >

    <intent-filter>
        <action android:name="android.nfc.action.TECH_DISCOVERED" />
    </intent-filter>
    <meta-data
        android:name="android.nfc.action.TECH_DISCOVERED"
        android:resource="@xml/nfc_tech_filter" />

    <intent-filter>
        <action android:name="android.nfc.action.NDEF_DISCOVERED" />
        <category android:name="android.intent.category.DEFAULT" />
        <data
            android:host="ext"
            android:pathPrefix="/abc:d"
            android:scheme="vnd.android.nfc" />
    </intent-filter>
    <intent-filter>
        <action android:name="android.nfc.action.NDEF_DISCOVERED"/>
        <category android:name="android.intent.category.DEFAULT"/>
        <data android:mimeType="text/plain" />
    </intent-filter>
</activity>

问题(1):我想从其他应用程序调用我的应用程序。我怎么做?我当前的代码是这样的:

try {
    final Intent intent = new Intent(NfcAdapter.ACTION_NDEF_DISCOVERED);
    NdefMessage ndefMessage = buildNdefMessage(getTagData());
    intent.putExtra(NfcAdapter.EXTRA_NDEF_MESSAGES, new NdefMessage[] {ndefMessage});
    startActivity(intent);
} catch (Exception e) {
    e.printStackTrace();
}

但这并没有调用我的应用程序。可能是因为数据类型和路径前缀不匹配。我如何在开始 Activity 时通过它?

问题(2):出于临时目的,我添加了

<intent-filter>
    <action android:name="android.nfc.action.NDEF_DISCOVERED" />
    <category android:name="android.intent.category.DEFAULT" />
</intent-filter>

让它工作,我的应用程序被调用。但是在读取标签时我检查了标签类型,由于我没有传递任何标签类型,我的应用程序崩溃了。那么如何创建标签实例呢?没有构造函数。我的意思是我不能执行 Ndef ndef = new Ndef();。而且我基本上没有标签,所以我无法执行 Ndef ndef = Ndef.get(tag);

最佳答案

关于问题(1):如何调用NDEF_DISCOVERED intent filter

您的 Activity “.activity.TagActivity_”过滤器有两个不同的NDEF_DISCOVERED。如果Intent bundle 含“vnd.android.nfc://ext/abc:d”形式的 URI,则第一个匹配。如果Intent bundle 含“文本/纯文本”数据类型,则第二个匹配。

因此,您需要将匹配的 URI 或匹配的 MIME 类型添加到用于启动该 Activity 的 Intent 。

  1. MIME 类型“text/plain”:

    final Intent intent = new Intent(NfcAdapter.ACTION_NDEF_DISCOVERED);
    intent.setType("text/plain");
    startActivity(intent);
    
  2. URI“vnd.android.nfc://ext/abc:d”:

    final Intent intent = new Intent(NfcAdapter.ACTION_NDEF_DISCOVERED);
    intent.setData(Uri.parse("vnd.android.nfc://ext/abc:d"));
    startActivity(intent);
    

两者中的任何一个都将开始您的 Activity 。

注意您为接收 Activity 指定了android:permission="android.permission.NFC"。你通常不会那样做。指定此权限意味着您的发送应用程序需要具有该权限。

关于问题(2):如何传递mock标签对象?

可以使用反射创建模拟标记对象实例(请注意,这不是公共(public) Android SDK 的一部分,因此在未来的 Android 版本中可能会失败)。

  1. 通过反射获取createMockTag()方法:

    Class tagClass = Tag.class;
    Method createMockTagMethod = tagClass.getMethod("createMockTag", byte[].class, int[].class, Bundle[].class);
    
  2. 定义一些常量来准备一个有效的模拟 NDEF 标签实例:

    final int TECH_NFC_A = 1;
    final String EXTRA_NFC_A_SAK = "sak";    // short (SAK byte value)
    final String EXTRA_NFC_A_ATQA = "atqa";  // byte[2] (ATQA value)
    
    final int TECH_NDEF = 6;
    final String EXTRA_NDEF_MSG = "ndefmsg";              // NdefMessage (Parcelable)
    final String EXTRA_NDEF_MAXLENGTH = "ndefmaxlength";  // int (result for getMaxSize())
    final String EXTRA_NDEF_CARDSTATE = "ndefcardstate";  // int (1: read-only, 2: read/write, 3: unknown)
    final String EXTRA_NDEF_TYPE = "ndeftype";            // int (1: T1T, 2: T2T, 3: T3T, 4: T4T, 101: MF Classic, 102: ICODE)
    
  3. 创建 NDEF 消息:

    NdefMessage ndefMessage = new NdefMessage(NdefRecord.createMime("text/plain"), "Text".getBytes("US-ASCII"));
    
  4. 使用该 NDEF 消息为 Type 2 标签创建技术附加包:

    Bundle nfcaBundle = new Bundle();
    nfcaBundle.putByteArray(EXTRA_NFC_A_ATQA, new byte[]{ (byte)0x44, (byte)0x00 }); //ATQA for Type 2 tag
    nfcaBundle.putShort(EXTRA_NFC_A_SAK , (short)0x00); //SAK for Type 2 tag
    
    Bundle ndefBundle = new Bundle();
    ndefBundle.putInt(EXTRA_NDEF_MAXLENGTH, 48); // maximum message length: 48 bytes
    ndefBundle.putInt(EXTRA_NDEF_CARDSTATE, 1); // read-only
    ndefBundle.putInt(EXTRA_NDEF_TYPE, 2); // Type 2 tag
    ndefBundle.putParcelable(EXTRA_NDEF_MSG, ndefMessage);  // add an NDEF message
    
  5. 为您的标签准备一个防冲突标识符/UID(参见 Tag.getId() 方法)。例如。类型 2 标签的 7 字节 UID:

    byte[] tagId = new byte[] { (byte)0x3F, (byte)0x12, (byte)0x34, (byte)0x56, (byte)0x78, (byte)0x90, (byte)0xAB };
    
  6. 然后您可以通过调用createMockTag() 方法来创建模拟标记实例

    Tag mockTag = (Tag)createMockTagMethod.invoke(null,
            tagId,                                     // tag UID/anti-collision identifier (see Tag.getId() method)
            new int[] { TECH_NFC_A, TECH_NDEF },       // tech-list
            new Bundle[] { nfcaBundle, ndefBundle });  // array of tech-extra bundles, each entry maps to an entry in the tech-list
    

一旦你创建了那个模拟标签对象,你就可以将它作为 NDEF_DISCOVERED Intent 的一部分发送:

Intent ndefIntent = new Intent(NfcAdapter.ACTION_NDEF_DISCOVERED);
ndefIntent.setType("text/plain");
ndefIntent.putExtra(NfcAdapter.EXTRA_ID, tagId);
ndefIntent.putExtra(NfcAdapter.EXTRA_TAG, mockTag);
ndefIntent.putExtra(NfcAdapter.EXTRA_NDEF_MESSAGES, new NdefMessage[]{ ndefMessage });

您可以选择将目标 Activity 明确定义为接收组件:

ndefIntent.setComponent(...); // or equivalent

然后您可以将此 Intent 发送到您的 Activity :

startActivity(ndefIntent);

接收者然后可以使用 moch 标签对象来检索技术类的实例(例如 Ndef.get(tag))但是任何需要 IO 操作的方法都将失败.

关于android - 如何模拟来自其他应用程序的标签触摸,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28046115/

相关文章:

android - 要求用户打开位置

java - 防止应用程序在读取 NFC 标签时暂停

android - IntentService 和 CursorLoader 同步问题

android - 如何为主机卡仿真的 STORE DATA 定义 APDU?

nfc - ISO 14443-3 防冲突协议(protocol)不正确

Android:compileDebugKotlin 失败并出现 java.lang.ClassCastException

android - 在 Android 中运行 tensorflow 模型

android - 在android中,在哪里存储json数据而不是使用sqlite?

android - 服务在特定时间运行 Android

安卓 : destroy activity before intent to another activity