Search code examples
androidnfccordova-pluginsonnewintent

NFC tag(for NfcA) scan works only from the second time


I wrote a custom plugin to read blocks of data from an NfcA(i.e.non-ndef) tag. It seems to work fine , but only after the second scan. I am using Activity intent to derive the "NfcAdapter.EXTRA_TAG" to later use it for reading the values. I am also updating the Intents in onNewIntent(). OnNewIntent gets called after the second scan and after that I get result all the time.But in the first scan onNewIntent does not gets called, hence I end up using the Activity tag that does not have "NfcAdapter.EXTRA_TAG", hence I get null. Please see the my code below.

SE_NfcA.java(my native code for plugin)

 @Override    
 public boolean execute(String action, JSONArray args, CallbackContext callbackContext) throws JSONException {      
    String Result = "";
    String TypeOfTalking = "";                   
    if (action.contains("TalkToNFC")) 
    {
        JSONObject arg_object = args.getJSONObject(0);
        TypeOfTalking = arg_object.getString("type");           

        if(TypeOfTalking != "")
        {
          if (TypeOfTalking.contains("readBlock")) 
            {
                if(TypeOfTalking.contains("@"))
                {
                    try
                    {
                        String[] parts = TypeOfTalking.split("@");
                        int index = Integer.parseInt(parts[1]);
                        Result = Readblock(cordova.getActivity().getIntent(),(byte)index);
                        callbackContext.success(Result);
                    }
                    catch(Exception e)
                    {
                        callbackContext.error("Exception Reading "+ TypeOfTalking + "due to "+ e.toString());
                        return false;
                    }
                }               
            }                   
          else 
            {
                return false;
            }           
        }
        else 
            {
                return false;
            }       
    }
    else 
        {
            return false;
        }      
    return true;
}

@Override
public void onNewIntent(Intent intent) {    
ShowAlert("onNewIntent called");
Tag tagFromIntent = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG);
super.onNewIntent(intent);
getActivity().setIntent(intent);
savedTag = tagFromIntent;
savedIntent = intent;   
}

@Override
public void onPause(boolean multitasking) {
Log.d(TAG, "onPause " + getActivity().getIntent());
   super.onPause(multitasking);
    if (multitasking) {
        // nfc can't run in background       
        stopNfc();
    }
}

@Override
public void onResume(boolean multitasking) {
Log.d(TAG, "onResume " + getActivity().getIntent());
super.onResume(multitasking);
startNfc();
}

public String Readblock(Intent Intent,byte block) throws IOException{       
    byte[] response = new byte[]{};
    if(Intent != null)
    {
        Tag myTag = Intent.getParcelableExtra(NfcAdapter.EXTRA_TAG);
        if(savedTag != null)
            myTag = savedTag;

        if(myTag != null)
        {           
            try{                
            Reader nTagReader = new Reader(myTag);                          
                nTagReader.close();
                nTagReader.connect();
                nTagReader.SectorSelect(Sector.Sector0);                    
                response = nTagReader.fast_read(block, block);
                nTagReader.close();             
                return ConvertH(response);                  
            }catch(Exception e){
                ShowAlert(e.toString());                    
            }
    }
        else
            ShowAlert("myTag is null.");            
    }       
    return null;
}

 private void createPendingIntent() {
        if (pendingIntent == null) {
            Activity activity = getActivity();
            Intent intent = new Intent(activity, activity.getClass());
            intent.addFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP| Intent.FLAG_ACTIVITY_CLEAR_TOP);
            pendingIntent = PendingIntent.getActivity(activity, 0, intent, 0);
        }
    }

 private void startNfc() {
      createPendingIntent(); // onResume can call startNfc before execute
        getActivity().runOnUiThread(new Runnable() {
            public void run() {
                NfcAdapter nfcAdapter = NfcAdapter.getDefaultAdapter(getActivity());
                if (nfcAdapter != null && !getActivity().isFinishing()) {
                    try {
                        nfcAdapter.enableForegroundDispatch(getActivity(), getPendingIntent(), getIntentFilters(), getTechLists());

                        if (p2pMessage != null) {
                            nfcAdapter.setNdefPushMessage(p2pMessage, getActivity());
                        }
                    } catch (IllegalStateException e) {
                        // issue 110 - user exits app with home button while nfc is initializing
                        Log.w(TAG, "Illegal State Exception starting NFC. Assuming application is terminating.");
                    }
                }
            }
        });
    }

    private void stopNfc() {
        Log.d(TAG, "stopNfc");
        getActivity().runOnUiThread(new Runnable() {
            public void run() {
                NfcAdapter nfcAdapter = NfcAdapter.getDefaultAdapter(getActivity());
                if (nfcAdapter != null) {
                    try {
                        nfcAdapter.disableForegroundDispatch(getActivity());
                    } catch (IllegalStateException e) {
                        // issue 125 - user exits app with back button while nfc
                        Log.w(TAG, "Illegal State Exception stopping NFC. Assuming application is terminating.");
                    }
                }
            }
        });
    }

    private Activity getActivity() {
        return this.cordova.getActivity();
    }

    private PendingIntent getPendingIntent() {
        return pendingIntent;
    }

    private IntentFilter[] getIntentFilters() {
        return intentFilters.toArray(new IntentFilter[intentFilters.size()]);
    }

    private String[][] getTechLists() {
        //noinspection ToArrayCallWithZeroLengthArrayArgument
        return techLists.toArray(new String[0][0]);
    }

}

My index.js file nfc.addTagDiscoveredListener(

        function(nfcEvent){
          console.log(nfcEvent.tag.id);
            alert(nfcEvent.tag.id);
            window.echo("readBlock@88");//call to plugin
        },
        function() {
            alert("Listening for NFC tags.");
        },
        function() {
            alert("NFC activation failed.");
        }

    );

SE_NfcA.js(plugin interface for interaction b/w index.js and SE_NfcA.java)

window.echo = function(natureOfTalk)
    {
        alert("Inside JS Interface, arg =" + natureOfTalk);
        cordova.exec(function(result){alert("Result is : "+result);},
                     function(error){alert("Some Error happened : "+ error);},
                     "SE_NfcA","TalkToNFC",[{"type": natureOfTalk}]);
    };

I guess I have messed up with the Intents/Activity Life-Cycle, please help. TIA!


Solution

  • I found a tweak/hack and made it to work. Before making any call to read or write, I made one dummy Initialize call.

    window.echo("Initialize");
    window.echo("readBlock@88");//call to plugin to read.
    

    And in the native code of the plugin, on receiving the "Initialize" token I made a startNFC() call.

     else if(TypeOfTalking.equalsIgnoreCase("Initialize"))
           {
               startNfc();
           }