|
|
|
'''This is the Android implementation of NFC Scanning using the
|
|
|
|
built in NFC adapter of some android phones.
|
|
|
|
'''
|
|
|
|
|
|
|
|
from kivy.app import App
|
|
|
|
from kivy.clock import Clock
|
|
|
|
#Detect which platform we are on
|
|
|
|
from kivy.utils import platform
|
|
|
|
if platform != 'android':
|
|
|
|
raise ImportError
|
|
|
|
import threading
|
|
|
|
|
|
|
|
from electrum_gui.kivy.nfc_scanner import NFCBase
|
|
|
|
from jnius import autoclass, cast
|
|
|
|
from android.runnable import run_on_ui_thread
|
|
|
|
from android import activity
|
|
|
|
|
|
|
|
BUILDVERSION = autoclass('android.os.Build$VERSION').SDK_INT
|
|
|
|
NfcAdapter = autoclass('android.nfc.NfcAdapter')
|
|
|
|
PythonActivity = autoclass('org.kivy.android.PythonActivity')
|
|
|
|
JString = autoclass('java.lang.String')
|
|
|
|
Charset = autoclass('java.nio.charset.Charset')
|
|
|
|
locale = autoclass('java.util.Locale')
|
|
|
|
Intent = autoclass('android.content.Intent')
|
|
|
|
IntentFilter = autoclass('android.content.IntentFilter')
|
|
|
|
PendingIntent = autoclass('android.app.PendingIntent')
|
|
|
|
Ndef = autoclass('android.nfc.tech.Ndef')
|
|
|
|
NdefRecord = autoclass('android.nfc.NdefRecord')
|
|
|
|
NdefMessage = autoclass('android.nfc.NdefMessage')
|
|
|
|
|
|
|
|
app = None
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class ScannerAndroid(NFCBase):
|
|
|
|
''' This is the class responsible for handling the interface with the
|
|
|
|
Android NFC adapter. See Module Documentation for details.
|
|
|
|
'''
|
|
|
|
|
|
|
|
name = 'NFCAndroid'
|
|
|
|
|
|
|
|
def nfc_init(self):
|
|
|
|
''' This is where we initialize NFC adapter.
|
|
|
|
'''
|
|
|
|
# Initialize NFC
|
|
|
|
global app
|
|
|
|
app = App.get_running_app()
|
|
|
|
|
|
|
|
# Make sure we are listening to new intent
|
|
|
|
activity.bind(on_new_intent=self.on_new_intent)
|
|
|
|
|
|
|
|
# Configure nfc
|
|
|
|
self.j_context = context = PythonActivity.mActivity
|
|
|
|
self.nfc_adapter = NfcAdapter.getDefaultAdapter(context)
|
|
|
|
# Check if adapter exists
|
|
|
|
if not self.nfc_adapter:
|
|
|
|
return False
|
|
|
|
|
|
|
|
# specify that we want our activity to remain on top when a new intent
|
|
|
|
# is fired
|
|
|
|
self.nfc_pending_intent = PendingIntent.getActivity(context, 0,
|
|
|
|
Intent(context, context.getClass()).addFlags(
|
|
|
|
Intent.FLAG_ACTIVITY_SINGLE_TOP), 0)
|
|
|
|
|
|
|
|
# Filter for different types of action, by default we enable all.
|
|
|
|
# These are only for handling different NFC technologies when app is in foreground
|
|
|
|
self.ndef_detected = IntentFilter(NfcAdapter.ACTION_NDEF_DISCOVERED)
|
|
|
|
#self.tech_detected = IntentFilter(NfcAdapter.ACTION_TECH_DISCOVERED)
|
|
|
|
#self.tag_detected = IntentFilter(NfcAdapter.ACTION_TAG_DISCOVERED)
|
|
|
|
|
|
|
|
# setup tag discovery for ourt tag type
|
|
|
|
try:
|
|
|
|
self.ndef_detected.addCategory(Intent.CATEGORY_DEFAULT)
|
|
|
|
# setup the foreground dispatch to detect all mime types
|
|
|
|
self.ndef_detected.addDataType('*/*')
|
|
|
|
|
|
|
|
self.ndef_exchange_filters = [self.ndef_detected]
|
|
|
|
except Exception as err:
|
|
|
|
raise Exception(repr(err))
|
|
|
|
return True
|
|
|
|
|
|
|
|
def get_ndef_details(self, tag):
|
|
|
|
''' Get all the details from the tag.
|
|
|
|
'''
|
|
|
|
details = {}
|
|
|
|
|
|
|
|
try:
|
|
|
|
#print 'id'
|
|
|
|
details['uid'] = ':'.join(['{:02x}'.format(bt & 0xff) for bt in tag.getId()])
|
|
|
|
#print 'technologies'
|
|
|
|
details['Technologies'] = tech_list = [tech.split('.')[-1] for tech in tag.getTechList()]
|
|
|
|
#print 'get NDEF tag details'
|
|
|
|
ndefTag = cast('android.nfc.tech.Ndef', Ndef.get(tag))
|
|
|
|
#print 'tag size'
|
|
|
|
details['MaxSize'] = ndefTag.getMaxSize()
|
|
|
|
#details['usedSize'] = '0'
|
|
|
|
#print 'is tag writable?'
|
|
|
|
details['writable'] = ndefTag.isWritable()
|
|
|
|
#print 'Data format'
|
|
|
|
# Can be made readonly
|
|
|
|
# get NDEF message details
|
|
|
|
ndefMesg = ndefTag.getCachedNdefMessage()
|
|
|
|
# get size of current records
|
|
|
|
details['consumed'] = len(ndefMesg.toByteArray())
|
|
|
|
#print 'tag type'
|
|
|
|
details['Type'] = ndefTag.getType()
|
|
|
|
|
|
|
|
# check if tag is empty
|
|
|
|
if not ndefMesg:
|
|
|
|
details['Message'] = None
|
|
|
|
return details
|
|
|
|
|
|
|
|
ndefrecords = ndefMesg.getRecords()
|
|
|
|
length = len(ndefrecords)
|
|
|
|
#print 'length', length
|
|
|
|
# will contain the NDEF record types
|
|
|
|
recTypes = []
|
|
|
|
for record in ndefrecords:
|
|
|
|
recTypes.append({
|
|
|
|
'type': ''.join(map(unichr, record.getType())),
|
|
|
|
'payload': ''.join(map(unichr, record.getPayload()))
|
|
|
|
})
|
|
|
|
|
|
|
|
details['recTypes'] = recTypes
|
|
|
|
except Exception as err:
|
|
|
|
print(str(err))
|
|
|
|
|
|
|
|
return details
|
|
|
|
|
|
|
|
def on_new_intent(self, intent):
|
|
|
|
''' This function is called when the application receives a
|
|
|
|
new intent, for the ones the application has registered previously,
|
|
|
|
either in the manifest or in the foreground dispatch setup in the
|
|
|
|
nfc_init function above.
|
|
|
|
'''
|
|
|
|
|
|
|
|
action_list = (NfcAdapter.ACTION_NDEF_DISCOVERED,)
|
|
|
|
# get TAG
|
|
|
|
#tag = cast('android.nfc.Tag', intent.getParcelableExtra(NfcAdapter.EXTRA_TAG))
|
|
|
|
|
|
|
|
#details = self.get_ndef_details(tag)
|
|
|
|
|
|
|
|
if intent.getAction() not in action_list:
|
|
|
|
print('unknow action, avoid.')
|
|
|
|
return
|
|
|
|
|
|
|
|
rawmsgs = intent.getParcelableArrayExtra(NfcAdapter.EXTRA_NDEF_MESSAGES)
|
|
|
|
if not rawmsgs:
|
|
|
|
return
|
|
|
|
for message in rawmsgs:
|
|
|
|
message = cast(NdefMessage, message)
|
|
|
|
payload = message.getRecords()[0].getPayload()
|
|
|
|
print('payload: {}'.format(''.join(map(chr, payload))))
|
|
|
|
|
|
|
|
def nfc_disable(self):
|
|
|
|
'''Disable app from handling tags.
|
|
|
|
'''
|
|
|
|
self.disable_foreground_dispatch()
|
|
|
|
|
|
|
|
def nfc_enable(self):
|
|
|
|
'''Enable app to handle tags when app in foreground.
|
|
|
|
'''
|
|
|
|
self.enable_foreground_dispatch()
|
|
|
|
|
|
|
|
def create_AAR(self):
|
|
|
|
'''Create the record responsible for linking our application to the tag.
|
|
|
|
'''
|
|
|
|
return NdefRecord.createApplicationRecord(JString("org.electrum.kivy"))
|
|
|
|
|
|
|
|
def create_TNF_EXTERNAL(self, data):
|
|
|
|
'''Create our actual payload record.
|
|
|
|
'''
|
|
|
|
if BUILDVERSION >= 14:
|
|
|
|
domain = "org.electrum"
|
|
|
|
stype = "externalType"
|
|
|
|
extRecord = NdefRecord.createExternal(domain, stype, data)
|
|
|
|
else:
|
|
|
|
# Creating the NdefRecord manually:
|
|
|
|
extRecord = NdefRecord(
|
|
|
|
NdefRecord.TNF_EXTERNAL_TYPE,
|
|
|
|
"org.electrum:externalType",
|
|
|
|
'',
|
|
|
|
data)
|
|
|
|
return extRecord
|
|
|
|
|
|
|
|
def create_ndef_message(self, *recs):
|
|
|
|
''' Create the Ndef message that will be written to tag
|
|
|
|
'''
|
|
|
|
records = []
|
|
|
|
for record in recs:
|
|
|
|
if record:
|
|
|
|
records.append(record)
|
|
|
|
|
|
|
|
return NdefMessage(records)
|
|
|
|
|
|
|
|
|
|
|
|
@run_on_ui_thread
|
|
|
|
def disable_foreground_dispatch(self):
|
|
|
|
'''Disable foreground dispatch when app is paused.
|
|
|
|
'''
|
|
|
|
self.nfc_adapter.disableForegroundDispatch(self.j_context)
|
|
|
|
|
|
|
|
@run_on_ui_thread
|
|
|
|
def enable_foreground_dispatch(self):
|
|
|
|
'''Start listening for new tags
|
|
|
|
'''
|
|
|
|
self.nfc_adapter.enableForegroundDispatch(self.j_context,
|
|
|
|
self.nfc_pending_intent, self.ndef_exchange_filters, self.ndef_tech_list)
|
|
|
|
|
|
|
|
@run_on_ui_thread
|
|
|
|
def _nfc_enable_ndef_exchange(self, data):
|
|
|
|
# Enable p2p exchange
|
|
|
|
# Create record
|
|
|
|
ndef_record = NdefRecord(
|
|
|
|
NdefRecord.TNF_MIME_MEDIA,
|
|
|
|
'org.electrum.kivy', '', data)
|
|
|
|
|
|
|
|
# Create message
|
|
|
|
ndef_message = NdefMessage([ndef_record])
|
|
|
|
|
|
|
|
# Enable ndef push
|
|
|
|
self.nfc_adapter.enableForegroundNdefPush(self.j_context, ndef_message)
|
|
|
|
|
|
|
|
# Enable dispatch
|
|
|
|
self.nfc_adapter.enableForegroundDispatch(self.j_context,
|
|
|
|
self.nfc_pending_intent, self.ndef_exchange_filters, [])
|
|
|
|
|
|
|
|
@run_on_ui_thread
|
|
|
|
def _nfc_disable_ndef_exchange(self):
|
|
|
|
# Disable p2p exchange
|
|
|
|
self.nfc_adapter.disableForegroundNdefPush(self.j_context)
|
|
|
|
self.nfc_adapter.disableForegroundDispatch(self.j_context)
|
|
|
|
|
|
|
|
def nfc_enable_exchange(self, data):
|
|
|
|
'''Enable Ndef exchange for p2p
|
|
|
|
'''
|
|
|
|
self._nfc_enable_ndef_exchange()
|
|
|
|
|
|
|
|
def nfc_disable_exchange(self):
|
|
|
|
''' Disable Ndef exchange for p2p
|
|
|
|
'''
|
|
|
|
self._nfc_disable_ndef_exchange()
|