From 88183af74117f37da9b6feabbb3757c6bc071b44 Mon Sep 17 00:00:00 2001 From: Svyatoslav2 <51791931+Svyatoslav2@users.noreply.github.com> Date: Mon, 7 Dec 2020 00:15:19 +0300 Subject: [PATCH] support for NFC (#383) NFC module was added for reading / writing data from NFC Ndef tags. --- app/src/main/AndroidManifest.xml | 10 + .../main/java/com/termux/api/NfcActivity.java | 301 ++++++++++++++++++ .../com/termux/api/TermuxApiReceiver.java | 4 + 3 files changed, 315 insertions(+) create mode 100644 app/src/main/java/com/termux/api/NfcActivity.java diff --git a/app/src/main/AndroidManifest.xml b/app/src/main/AndroidManifest.xml index 75a70e82..87dc49fe 100644 --- a/app/src/main/AndroidManifest.xml +++ b/app/src/main/AndroidManifest.xml @@ -27,6 +27,7 @@ tools:ignore="ProtectedPermissions" /> + @@ -57,6 +58,15 @@ tools:ignore="GoogleAppIndexingWarning"> + + + + + + + + + 0) + out.name("error").value(error); + out.name("nfcPresent").value(null != adapter); + if(null!=adapter) + out.name("nfcActive").value(adapter.isEnabled()); + out.endObject(); + } + }); + } + + @Override + protected void onCreate(@Nullable Bundle savedInstanceState) { + super.onCreate(savedInstanceState); + Intent intent = this.getIntent(); + if (intent != null) { + mode = intent.getStringExtra("mode"); + if (null == mode) + mode = "noData"; + param =intent.getStringExtra("param"); + if (null == param) + param = "noData"; + value=intent.getStringExtra("value"); + if (null == socket_input) socket_input = intent.getStringExtra("socket_input"); + if (null == socket_output) socket_output = intent.getStringExtra("socket_output"); + if (mode == "noData") { + errorNfc(this, intent,""); + finish(); + } + } + + NfcAdapter adapter = NfcAdapter.getDefaultAdapter(this); + if((null==adapter)||(!adapter.isEnabled())){ + errorNfc(this,intent,""); + finish(); + } + } + + @Override + protected void onResume() { + super.onResume(); + adapter = NfcAdapter.getDefaultAdapter(this); + Intent intentNew = new Intent(this, NfcActivity.class).addFlags(Intent.FLAG_RECEIVER_REPLACE_PENDING); + PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, intentNew, 0); + IntentFilter[] intentFilter = new IntentFilter[]{ + new IntentFilter(NfcAdapter.ACTION_NDEF_DISCOVERED), + new IntentFilter(NfcAdapter.ACTION_TAG_DISCOVERED), + new IntentFilter(NfcAdapter.ACTION_TECH_DISCOVERED)}; + adapter.enableForegroundDispatch(this, pendingIntent, intentFilter, null); + } + + @Override + protected void onNewIntent(Intent intent) { + intent.putExtra("socket_input", socket_input); + intent.putExtra("socket_output", socket_output); + + if (NfcAdapter.ACTION_TAG_DISCOVERED.equals(intent.getAction())) { + try { + postResult(this, intent); + } + catch (Exception e) + { + Log.e("Termix-api.NfcAction",e.getMessage()); + } + finish(); + } + super.onNewIntent(intent); + } + + @Override + protected void onPause() { + adapter.disableForegroundDispatch(this); + super.onPause(); + } + + @Override + protected void onDestroy() { + socket_input = null; + socket_output = null; + super.onDestroy(); + } + + protected void postResult(final Context context, Intent intent) { + ResultReturner.returnData(context, intent, new ResultReturner.ResultJsonWriter() { + @Override + public void writeJson(JsonWriter out) throws Exception { + Log.e("NFC","postResult"); + try + { + switch (mode) { + case "write": + switch (param) { + case "text": + Log.e("NFC","-->write"); + onReceiveNfcWrite(context, intent); + Log.e("NFC","<--write"); + break; + default: + onUnexpectedAction(out, "Wrong Params", "Should be text for TAG"); + break; + } + break; + case "read": + switch (param){ + case "short": + readNDEFTag(intent,out); + break; + case "full": + readFullNDEFTag(intent,out); + break; + case "noData": + readNDEFTag(intent,out); + break; + default: + onUnexpectedAction(out, "Wrong Params", "Should be correct param value"); + break; + } + break; + default: + onUnexpectedAction(out, "Wrong Params", "Should be correct mode value "); + break; + } + } + catch (Exception e){ + onUnexpectedAction(out, "exception", e.getMessage()); + } + } + }); + } + public void onReceiveNfcWrite( final Context context, Intent intent) throws Exception { + { + Log.e("NFC","---->onReceiveNfcWrite"); + NfcAdapter adapter = NfcAdapter.getDefaultAdapter(context); + Tag tag = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG); + NdefRecord record = NdefRecord.createTextRecord("en", value); + NdefMessage msg = new NdefMessage(new NdefRecord[]{record}); + Ndef ndef = Ndef.get(tag); + ndef.connect(); + ndef.writeNdefMessage(msg); + ndef.close(); + } + } + + + public void readNDEFTag(Intent intent, JsonWriter out) throws Exception { + NfcAdapter adapter = NfcAdapter.getDefaultAdapter(this); + Parcelable[] msgs = intent.getParcelableArrayExtra(NfcAdapter.EXTRA_NDEF_MESSAGES); + Tag tag = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG); + Ndef ndefTag = Ndef.get(tag); + boolean bNdefPresent = false; + String strs[] = tag.getTechList(); + for (String s: strs){ + if (s.equals("android.nfc.tech.Ndef")) + bNdefPresent = true; + } + if (!bNdefPresent){ + onUnexpectedAction(out, "Wrong Technology","termux API support only NFEF Tag"); + return; + } + NdefMessage[] nmsgs = new NdefMessage[msgs.length]; + if (msgs.length == 1) { + nmsgs[0] = (NdefMessage) msgs[0]; + NdefRecord records[] = nmsgs[0].getRecords(); + out.beginObject(); + if (records.length >0 ) { + { + out.name("Record"); + if (records.length > 1) + out.beginArray(); + for (NdefRecord record: records){ + out.beginObject(); + int pos = 1 + record.getPayload()[0]; + pos = (NdefRecord.TNF_WELL_KNOWN==record.getTnf())?(int)record.getPayload()[0]+1:0; + int len = record.getPayload().length - pos; + byte msg[] = new byte[len]; + System.arraycopy(record.getPayload(), pos, msg, 0, len); + out.name("Payload").value(new String(msg)); + out.endObject(); + } + if (records.length > 1) + out.endArray(); + } + } + out.endObject(); + } + } + + public void readFullNDEFTag(Intent intent, JsonWriter out) throws Exception { + NfcAdapter adapter = NfcAdapter.getDefaultAdapter(this); + Tag tag = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG); + Ndef ndefTag = Ndef.get(tag); + Parcelable[] msgs = intent.getParcelableArrayExtra(NfcAdapter.EXTRA_NDEF_MESSAGES); + + String strs[] = tag.getTechList(); + boolean bNdefPresent = false; + for (String s: strs){ + if (s.equals("android.nfc.tech.Ndef")) + bNdefPresent = true; + } + if (!bNdefPresent){ + onUnexpectedAction(out, "Wrong Technology","termux API support only NFEF Tag"); + return; + } + NdefMessage[] nmsgs = new NdefMessage[msgs.length]; + out.beginObject(); + { + byte[] tagID = tag.getId(); + StringBuilder sp = new StringBuilder(); + for (byte tagIDpart : tagID) { sp.append(String.format("%02x", tagIDpart)); } + out.name("id").value(sp.toString()); + out.name("typeTag").value(ndefTag.getType()); + out.name("maxSize").value(ndefTag.getMaxSize()); + out.name("techList"); + { + out.beginArray(); + String[] tlist = tag.getTechList(); + for (String str : tlist) { + out.value(str); + } + out.endArray(); + } + if (msgs.length == 1) { + Log.e("NFC", "-->> readFullNDEFTag - 06"); + nmsgs[0] = (NdefMessage) msgs[0]; + NdefRecord records[] = nmsgs[0].getRecords(); + { + out.name("record"); + if (records.length > 1) + out.beginArray(); + for (NdefRecord record : records) { + out.beginObject(); + out.name("type").value(new String(record.getType())); + out.name("tnf").value(record.getTnf()); + if (records[0].toUri() != null) out.name("URI").value(record.toUri().toString()); + out.name("mime").value(record.toMimeType()); + int pos = 1 + record.getPayload()[0]; + pos = (NdefRecord.TNF_WELL_KNOWN==record.getTnf())?(int)record.getPayload()[0]+1:0; + int len = record.getPayload().length - pos; + byte msg[] = new byte[len]; + System.arraycopy(record.getPayload(), pos, msg, 0, len); + out.name("payload").value(new String(msg)); + out.endObject(); + } + if (records.length > 1) out.endArray(); + } + } + + } + out.endObject(); + } + + protected void onUnexpectedAction(JsonWriter out,String error, String description) throws Exception { + out.beginObject(); + out.name("error").value(error); + out.name("description").value(description); + out.endObject(); + out.flush(); + } +} diff --git a/app/src/main/java/com/termux/api/TermuxApiReceiver.java b/app/src/main/java/com/termux/api/TermuxApiReceiver.java index 5addbd3f..a1fc917b 100644 --- a/app/src/main/java/com/termux/api/TermuxApiReceiver.java +++ b/app/src/main/java/com/termux/api/TermuxApiReceiver.java @@ -5,6 +5,7 @@ import android.content.ComponentName; import android.content.Context; import android.content.Intent; +import android.nfc.NfcAdapter; import android.os.Build; import android.provider.Settings; import android.widget.Toast; @@ -115,6 +116,9 @@ private void doWork(Context context, Intent intent) { MicRecorderAPI.onReceive(context, intent); } break; + case "Nfc": + context.startActivity(new Intent(context, NfcActivity.class).putExtras(intent.getExtras()).addFlags(Intent.FLAG_ACTIVITY_NEW_TASK)); + break; case "NotificationList": ComponentName cn = new ComponentName(context, NotificationService.class); String flat = Settings.Secure.getString(context.getContentResolver(), "enabled_notification_listeners");