Data Exchange Format (NDEF) read and write support for NFC cards
This library supports NDEF messages. NDEF is composed of a message with records in it. Every record can be a know type or a specific type. This library fully support all root type of messages.
NDEF messages are used on Mifare Cards and is included into this library as well. You have a full example using 23 different NFC readers MFRC522, PN532 and PN1850 build in.
Reading NDEF from a card
This operation only require a valid NFC reader which implement CardTransceiver
, see the class.
From the code below, the transceiver is a PN532 used with a serial port on Windows. This is just for convenience, it can be any supported reader connected on any interface and any OS.
This is a complete example from initializing the reader, detecting the card, extracting the message and displaying the detailed messages.
Important: NDEF is supported for both Mifare and Ultralight cards.
// Create a PN532
var pn532 = new Pn532("COM4", debugLevel);
byte[]? retData = null;
while (true)
{
retData = pn532.ListPassiveTarget(MaxTarget.One, TargetBaudRate.B106kbpsTypeA);
if (retData is object)
{
break;
}
// Give time to PN532 to process
Thread.Sleep(200);
}
if (retData is null)
{
return;
}
Debug.WriteLine();
// Check if it is a valid card
var card = pn532.TryDecode106kbpsTypeA(retData.AsSpan().Slice(1));
if (card is not object)
{
Debug.WriteLine("Can't read properly the card");
return;
}
// Create the Mifare card
MifareCard mifareCard = new MifareCard(pn532, card.TargetNumber) { BlockNumber = 0, Command = MifareCardCommand.AuthenticationA };
mifareCard.SetCapacity(card.Atqa, card.Sak);
mifareCard.SerialNumber = card.NfcId;
// Read an extract the NDEF message
// This is where you can write as well, format the card, check the card see next sections
mifareCard.TryReadNdefMessage(out NdefMessage message);
if (message.Records.Count == 0)
{
Debug.WriteLine("Sorry, there is no NDEF message in this card or I can't find them");
}
// Display the messages
foreach (NdefRecord msg in message.Records)
{
Debug.WriteLine("Record header:");
Debug.WriteLine($" Is first message: {msg.Header.IsFirstMessage}, is last message: {msg.Header.IsLastMessage}");
Debug.Write($" Type name format: {msg.Header.TypeNameFormat}");
if (msg.Header.PayloadType is object)
{
Debug.WriteLine($", Payload type: {BitConverter.ToString(msg.Header.PayloadType)}");
}
else
{
Debug.WriteLine("");
}
Debug.WriteLine($" Is composed: {msg.Header.IsComposedMessage}, is Id present: {msg.Header.MessageFlag.HasFlag(MessageFlag.IdLength)}, Id Length value: {msg.Header.IdLength}");
Debug.WriteLine($" Payload Length: {msg.Payload?.Length}, is short message= {msg.Header.MessageFlag.HasFlag(MessageFlag.ShortRecord)}");
if (msg.Payload is object)
{
Debug.WriteLine($"Payload: {BitConverter.ToString(msg.Payload)}");
}
else
{
Debug.WriteLine("No payload");
}
if (UriRecord.IsUriRecord(msg))
{
var urirec = new UriRecord(msg);
Debug.WriteLine($" Type {nameof(UriRecord)}, Uri Type: {urirec.UriType}, Uri: {urirec.Uri}, Full URI: {urirec.FullUri}");
}
if (TextRecord.IsTextRecord(msg))
{
var txtrec = new TextRecord(msg);
Debug.WriteLine($" Type: {nameof(TextRecord)}, Encoding: {txtrec.Encoding}, Language: {txtrec.LanguageCode}, Text: {txtrec.Text}");
}
if (GeoRecord.IsGeoRecord(msg))
{
var geo = new GeoRecord(msg);
Debug.WriteLine($" Type: {nameof(GeoRecord)}, Lat: {geo.Latitude}, Long: {geo.Longitude}");
}
if (MediaRecord.IsMediaRecord(msg))
{
var media = new MediaRecord(msg);
Debug.WriteLine($" Type: {nameof(MediaRecord)}, Payload Type = {media.PayloadType}");
if (media.IsTextType)
{
var ret = media.TryGetPayloadAsText(out string payloadAsText);
if (ret)
{
Debug.WriteLine($" Payload as Text:");
Debug.WriteLine($"{payloadAsText}");
}
else
{
Debug.WriteLine($"Can't convert the payload as a text");
}
}
}
Debug.WriteLine("");
}
Writing NDEF to a card
From the previous example, you will still need to get a card. From there, to create and write messages, it's quite straight forward:
// Create the NDEF message
NdefMessage message = new();
// Create a Text record
TextRecord recordText = new("I ❤ .NET nanoFramework", "en", Encoding.UTF8);
// Add the record to the message
message.Records.Add(recordText);
// Create a Geo message
GeoRecord geoRecord = new(2.1234, -1.2345);
// Add the record to the message
message.Records.Add(geoRecord);
// Write the message, by using the default Key A
var res = mifareCard.WriteNdefMessage(message);
if (res)
{
Debug.WriteLine($"Writing successful");
}
else
{
Debug.WriteLine($"Error writing to the card");
}
Note that if you want to set specific permission and have read only with the default NDEF Keys, you can authenticate and write the NDEF message with the Key B:
// Your secret Key B (here the default one)
mifareCard.KeyB = new byte[6] { 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF };
// This will write using this secret key B
var res = mifareCard.WriteNdefMessage(message, false);
Format a card to NDEF
You can format a card, this will be done using Key B, by default it will use the Default Key B. You can pass as well the Key B in the parameters.
var ret = mifareCard.FormatNdef();
string msg = ret ? "Formatting successful." : "Error formatting card.";
Debug.WriteLine(msg);
Check if the card is NDEF formatted
You can as well check is properly NDEF formatted:
var ret = mifareCard.IsFormattedNdef();
var isForm = ret ? string.Empty : " not";
Debug.WriteLine($"This card is{isForm} NDEF formatted");
Card type supported
NDEF per se is fully independent of cards, so the class can be used independently. A Mifare implementation has been done. All Mifare 1K, 2K and 4K are supported. The Mifare 300 are not.