NFC Tag запускается, не спрашивая пользователя, какое приложение использовать

мое приложение в основном просто берет идентификатор тега любых тегов

Я хочу, чтобы приложение запускалось, не спрашивая пользователя, какое приложение использовать

Прямо из того же приложения

Я нашел в Интернете учебник о том, как разработать модуль записи и чтения NFC.

но каждый раз, когда я нажимаю, система будет просить пользователя выбрать приложение (ДАЖЕ, если я закрыл приложение)

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

это мой код: -

AndroidManifest.xml

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

    <uses-sdk
        android:minSdkVersion="16"
        android:targetSdkVersion="19" />

    <uses-permission android:name="android.permission.INTERNET"/>
    <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
    <uses-permission android:name="android.permission.CAMERA" />
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
    <uses-permission android:name="android.permission.NFC" />

    <uses-feature android:name="android.hardware.camera" />
    <uses-feature android:name="android.hardware.camera.autofocus" />
    <uses-feature
        android:name="android.hardware.nfc"
        android:required="true" />

    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <activity
            android:name=".MainActivity"
            android:screenOrientation="portrait" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>

        </activity>

        <activity android:name=".TapToRegisterTag" 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" />
        </activity>

        <!-- Other Activities -->

    </application>

</manifest>

TapToRegisterTag.java

package com.nfc.test;

import android.app.Activity;
import android.app.PendingIntent;
import android.content.Intent;
import android.content.IntentFilter;
import android.content.IntentFilter.MalformedMimeTypeException;
import android.nfc.NfcAdapter;
import android.nfc.Tag;
import android.nfc.tech.Ndef;
import android.os.Bundle;
import android.util.Log;
import android.view.Window;
import android.widget.ImageView;
import android.widget.TextView;
import android.widget.Toast;

public class TapToRegisterTag extends Activity{

    // Prepare NFC Tag variables
    Tag myTag;
    String tagID;
    NfcAdapter mNfcAdapter;
    public static final String MIME_TEXT_PLAIN = "text/plain";
    public static final String TAG = "NfcDemo";

    ImageView logo;
    TextView txt;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        // TODO Auto-generated method stub
        super.onCreate(savedInstanceState);
        requestWindowFeature(Window.FEATURE_NO_TITLE);
        setContentView(R.layout.register_tag_layout);

        logo = (ImageView)findViewById(R.id.imageView2);
        txt = (TextView)findViewById(R.id.textView);

        // prepare NFC
        mNfcAdapter = NfcAdapter.getDefaultAdapter(this);

        if (mNfcAdapter == null) {
            // Stop here, we definitely need NFC
            Toast.makeText(this, "This device doesn't support NFC.", Toast.LENGTH_LONG).show();
            finish();
            return;

        }

        if (!mNfcAdapter.isEnabled()) {
            Toast.makeText(this, "NFC is disabled.", Toast.LENGTH_LONG).show();
        } else {
            Toast.makeText(this, "NFC is Enabled.", Toast.LENGTH_LONG).show();
        }

        handleIntent(getIntent());
    }

    @Override
    protected void onResume() {
        // TODO Auto-generated method stub
        super.onResume();

        /**
         * It's important, that the activity is in the foreground (resumed). Otherwise
         * an IllegalStateException is thrown. 
         */
        setupForegroundDispatch(this, mNfcAdapter);
        try{
            this.myTag = (Tag) getIntent().getParcelableExtra(NfcAdapter.EXTRA_TAG);
            this.tagID = OnetapActivity.bytesToHex(myTag.getId());
            Toast.makeText(this, "Tag ID: \n" + tagID, Toast.LENGTH_LONG).show();
        }catch (Exception ex){
            Toast.makeText(this, ex.getMessage(), Toast.LENGTH_LONG).show();
        }
    }


    @Override
    protected void onPause() {

        stopForegroundDispatch(this, mNfcAdapter);

        super.onPause();
    }

    @Override
    protected void onNewIntent(Intent intent) { 

        handleIntent(intent);

    }

    private void handleIntent(Intent intent) {
        // TODO: handle Intent

        String action = intent.getAction();
        if (NfcAdapter.ACTION_NDEF_DISCOVERED.equals(action)) {

            String type = intent.getType();
            if (MIME_TEXT_PLAIN.equals(type)) {

                //new NdefReaderTask().execute(tag);
                try{
                    this.myTag = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG);
                    this.tagID = OnetapActivity.bytesToHex(this.myTag.getId());
                    Toast.makeText(this, "Tag ID: \n" + tagID, Toast.LENGTH_LONG).show();
                }catch (Exception ex){
                    Toast.makeText(this, ex.getMessage(), Toast.LENGTH_LONG).show();
                }

            } else {
                Log.d(TAG, "Wrong mime type: " + type);
            }
        } else if (NfcAdapter.ACTION_TECH_DISCOVERED.equals(action)) {

            // In case we would still use the Tech Discovered Intent
            this.myTag = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG);
            String[] techList = this.myTag.getTechList();
            String searchedTech = Ndef.class.getName();

            for (String tech : techList) {
                if (searchedTech.equals(tech)) {
                    //new NdefReaderTask().execute(tag);

                     try{
                         myTag = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG);
                         //this.myTag = (Tag) getIntent().getParcelableExtra(NfcAdapter.EXTRA_TAG);
                         this.tagID = OnetapActivity.bytesToHex(myTag.getId());
                         Toast.makeText(this, "Tag ID: \n" + tagID, Toast.LENGTH_LONG).show();
                     }catch (Exception ex){
                         Toast.makeText(this, ex.getMessage(), Toast.LENGTH_LONG).show();
                     }

                    break;
                }
            }

        }

    }

    public static void setupForegroundDispatch(final Activity activity, NfcAdapter adapter) {
        final Intent intent = new Intent(activity.getApplicationContext(), activity.getClass());
        intent.setFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP);

        final PendingIntent pendingIntent = PendingIntent.getActivity(activity.getApplicationContext(), 0, intent, 0);

        IntentFilter[] filters = new IntentFilter[1];
        String[][] techList = new String[][]{};

        // Notice that this is the same filter as in our manifest.
        filters[0] = new IntentFilter();
        filters[0].addAction(NfcAdapter.ACTION_NDEF_DISCOVERED);
        filters[0].addCategory(Intent.CATEGORY_DEFAULT);
        try {
            filters[0].addDataType(MIME_TEXT_PLAIN);
        } catch (MalformedMimeTypeException e) {
            throw new RuntimeException("Check your mime type.");
        }

        adapter.enableForegroundDispatch(activity, pendingIntent, filters, techList);
    }


    public static void stopForegroundDispatch(final Activity activity, NfcAdapter adapter) {
        adapter.disableForegroundDispatch(activity);
    }

}

@xml/nfc_tech_filter.xml

<?xml version="1.0" encoding="utf-8"?>
<resources xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
    <tech-list>
        <tech>android.nfc.tech.Ndef</tech>
        <!-- class name -->
    </tech-list>
</resources>

<!-- 
<resources xmlns:xliff="urn:oasis:names:tc:xliff:document:1.2">
    <tech-list>
        <tech>android.nfc.tech.IsoDep</tech>
        <tech>android.nfc.tech.NfcA</tech>
        <tech>android.nfc.tech.NfcB</tech>
        <tech>android.nfc.tech.NfcF</tech>
        <tech>android.nfc.tech.NfcV</tech>
        <tech>android.nfc.tech.Ndef</tech>
        <tech>android.nfc.tech.NdefFormatable</tech>
        <tech>android.nfc.tech.MifareClassic</tech>
        <tech>android.nfc.tech.MifareUltralight</tech>
    </tech-list>
</resources>
-->

person ahmadssb    schedule 02.04.2014    source источник


Ответы (3)


Единственное, что я пока знаю, это реализация AAR (Android Application Records). Это сообщения NDEF в ваших тегах, которые только запускают приложение, указанное в AAR. Это работает очень хорошо. Однако вы должны реализовать отступы NDEF. На рынке Android есть несколько авторов AAR, которые вы можете попробовать.

person pizzaani    schedule 04.04.2014

На самом деле я нашел решение благодаря моему другу

это его ответ:

возможно, вам следует изменить «фильтр» и «технический список» на нуль

Ваш код:

adapter.enableForegroundDispatch(activity, pendingIntent, filters, techList);

Изменить на

adapter.enableForegroundDispatch(activity, pendingIntent, null, null);

Ссылка:

Если этому методу предоставлены какие-либо IntentFilters, они используются для сопоставления намерений отправки как для ACTION_NDEF_DISCOVERED, так и для ACTION_TAG_DISCOVERED. Поскольку ACTION_TECH_DISCOVERED полагается на метаданные за пределами сопоставления IntentFilter для этой отправки, намерение обрабатывается путем отдельной передачи технических списков. Каждая запись первого уровня в списке технологий представляет собой набор технологий, которые должны присутствовать для соответствия. Если какой-либо из наборов первого уровня совпадает, то отправка маршрутизируется через заданный объект PendingIntent. Другими словами, второй уровень объединяется вместе, а записи первого уровня объединяются вместе.

Если вы передадите null как для фильтров, так и для параметров techLists, это будет действовать как подстановочный знак и приведет к тому, что активность переднего плана получит все теги через намерение ACTION_TAG_DISCOVERED.

Источник:

http://developer.android.com/reference/android/nfc/NfcAdapter.html#enableForegroundDispatch%28android.app.Activity,%20android.app.PendingIntent,%20android.content.IntentFilter%5B%5D,%20java.lang.String%5B%5D%5B%5D%29

person ahmadssb    schedule 04.04.2014

Я декомпилирую один из apk Google Play и делаю этот код:

  public static void resume(Context con, Activity _activity) {
    NfcAdapter _nfcAdapter = NfcAdapter.getDefaultAdapter(con);
    if (!_isReceiveIntentEnabled) {
      if (_nfcAdapter != null) {
        _nfcAdapter.setNdefPushMessage(null, _activity, new Activity[0]);
        NfcAdapter localNfcAdapter = _nfcAdapter;
        Activity localActivity = _activity;
        int flag = PendingIntent.FLAG_NO_CREATE; // 536870912;
        Intent intent = new Intent(_activity, _activity.getClass()).addFlags(flag);
        PendingIntent localPendingIntent = PendingIntent.getActivity(_activity, 0, intent, 0);
        IntentFilter[] arrayOfIntentFilter = new IntentFilter[] { new IntentFilter("android.nfc.action.TECH_DISCOVERED") };
        String[][] techArr = new String[][] { new String[] { IsoDep.class.getName() } };
        localNfcAdapter.enableForegroundDispatch(localActivity, localPendingIntent, arrayOfIntentFilter, techArr);
      }
      _isReceiveIntentEnabled = true;
    }
  }

  public static void pause(Context con, Activity _activity) {
    NfcAdapter _nfcAdapter = NfcAdapter.getDefaultAdapter(con);
    if (_isReceiveIntentEnabled) {
      if (_nfcAdapter != null) {
        _nfcAdapter.disableForegroundDispatch(_activity);
      }
      _isReceiveIntentEnabled = false;
    }
  }

Не забудьте добавить разрешение:

<uses-permission android:name="android.permission.NFC" />
person Alexmelyon    schedule 03.07.2014