Как записать записи NDEF в NFC-тег?
Как мне записать сообщение NDEF в NFC-тег? Нужно ли изменять файл манифеста? Пока у меня есть код для генерации сообщения NDEF:
public NdefRecord createTextRecord(String payload, Locale locale, boolean encodeInUtf8) {
byte[] langBytes = locale.getLanguage().getBytes(Charset.forName("US-ASCII"));
Charset utfEncoding = encodeInUtf8 ? Charset.forName("UTF-8") : Charset.forName("UTF-16");
byte[] textBytes = payload.getBytes(utfEncoding);
int utfBit = encodeInUtf8 ? 0 : (1 << 7);
char status = (char) (utfBit + langBytes.length);
byte[] data = new byte[1 + langBytes.length + textBytes.length];
data[0] = (byte) status;
System.arraycopy(langBytes, 0, data, 1, langBytes.length);
System.arraycopy(textBytes, 0, data, 1 + langBytes.length, textBytes.length);
NdefRecord record = new NdefRecord(NdefRecord.TNF_WELL_KNOWN,
NdefRecord.RTD_TEXT, new byte[0], data);
return record;
}
Как мне обнаружить метку?
Кто-нибудь может мне с этим помочь?
Переведено автоматически
Ответ 1
Я бы проигнорировал то, что говорится в документации Google для чтения по адресу https://developer.android.com/guide/topics/connectivity/nfc/nfc и для чтения / записи по адресу https://developer.android.com/guide/topics/connectivity/nfc/advanced-nfc#read-write поскольку это обеспечивает очень плохой пользовательский опыт записи в теги и приводит к большому количеству неудачных операций записи из-за поведения пользователя.
Чтобы обеспечить надежную запись в NFC с Android, вам следует использовать более новый и намного лучший enableReaderMode
API https://developer.android.com/reference/android/nfc/NfcAdapter
Использование этого нового API приводит к гораздо меньшему количеству неудачных операций записи и поврежденных карт, потому что вы можете контролировать, когда появляется звуковое уведомление. В старой системе, основанной на Intent
, система приостанавливает ваше приложение, а затем считывает карту и издает звуковой сигнал уведомления, затем пользователь удаляет карту перед возобновлением работы вашего приложения и имеет возможность обработать данные карты и выполнить запись на карту.
С новым enableReaderMode
API вы отключаете звук системного уведомления, и ваше приложение никогда не приостанавливается для считывания данных с NFC-карты, затем вы можете считывать и записывать данные на карту, а затем, когда вы успешно выполняете запись на карту, вы можете сами издавать звук уведомления.
Поскольку о любых ошибках ничего не сообщается, пользователь будет продолжать попытки предъявить карту до получения уведомления об успешном завершении. Требуется дополнительная логика, чтобы не писать одно и то же сообщение каждый раз, когда предъявляется одна карта или разные карты.
Несколько примеров кода (адаптированы из моего приложения NFC для обработки низкоуровневого чтения и записи (не технология Ndef-тегов))
public class NFCActivity extends AppCompatActivity implements NfcAdapter.ReaderCallback{
private NfcAdapter mNfcAdapter;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_nfc);
mNfcAdapter = NfcAdapter.getDefaultAdapter(this);
// Rest of Activity setup
}
@Override
protected void onResume() {
super.onResume();
if(mNfcAdapter!= null) {
Bundle options = new Bundle();
// Work around for some broken Nfc firmware implementations that poll the card too fast
options.putInt(NfcAdapter.EXTRA_READER_PRESENCE_CHECK_DELAY, 250);
// Enable ReaderMode for all types of card and disable platform sounds
mNfcAdapter.enableReaderMode(this,
this,
NfcAdapter.FLAG_READER_NFC_A |
NfcAdapter.FLAG_READER_NFC_B |
NfcAdapter.FLAG_READER_NFC_F |
NfcAdapter.FLAG_READER_NFC_V |
NfcAdapter.FLAG_READER_NFC_BARCODE |
NfcAdapter.FLAG_READER_NO_PLATFORM_SOUNDS,
options);
}
}
@Override
protected void onPause() {
super.onPause();
if(mNfcAdapter!= null)
mNfcAdapter.disableReaderMode(this);
}
// This method is run in another thread when a card is discovered
// !!!! This method cannot cannot direct interact with the UI Thread
// Use `runOnUiThread` method to change the UI from this method
public void onTagDiscovered(Tag tag) {
// Read and or write to Tag here to the appropriate Tag Technology type class
// in this example the card should be an Ndef Technology Type
Ndef mNdef = Ndef.get(tag);
// Check that it is an Ndef capable card
if (mNdef!= null) {
// If we want to read
// As we did not turn on the NfcAdapter.FLAG_READER_SKIP_NDEF_CHECK
// We can get the cached Ndef message the system read for us.
NdefMessage mNdefMessage = mNdef.getCachedNdefMessage();
// Or if we want to write a Ndef message
// Create a Ndef Record
NdefRecord mRecord = NdefRecord.createTextRecord("en","English String");
// Add to a NdefMessage
NdefMessage mMsg = new NdefMessage(mRecord);
// Catch errors
try {
mNdef.connect();
mNdef.writeNdefMessage(mMsg);
// Success if got to here
runOnUiThread(() -> {
Toast.makeText(getApplicationContext(),
"Write to NFC Success",
Toast.LENGTH_SHORT).show();
});
// Make a Sound
try {
Uri notification = RingtoneManager.getDefaultUri(RingtoneManager.TYPE_NOTIFICATION);
Ringtone r = RingtoneManager.getRingtone(getApplicationContext(),
notification);
r.play();
} catch (Exception e) {
// Some error playing sound
}
} catch (FormatException e) {
// if the NDEF Message to write is malformed
} catch (TagLostException e) {
// Tag went out of range before operations were complete
} catch (IOException e){
// if there is an I/O failure, or the operation is cancelled
} catch (SecurityException e){
// The SecurityException is only for Android 12L and above
// The Tag object might have gone stale by the time
// the code gets to process it, with a new one been
// delivered (for the same or different Tag)
// The SecurityException is thrown if you are working on
// a stale Tag
} finally {
// Be nice and try and close the tag to
// Disable I/O operations to the tag from this TagTechnology object, and release resources.
try {
mNdef.close();
} catch (IOException e) {
// if there is an I/O failure, or the operation is cancelled
}
}
}
}