Flutter и работа с нативным кодом на примере библиотеки для работы с NFC
Как исправлять недочеты в нативных библиотеках при работе с флаттером на примере NFC
В большинстве случаев при работе с флаттером разработчику никак не приходится взаимодействовать с нативным кодом (Java/Kotlin и ObjectiveC/Swift), потому что инструментарий флаттера довольно богат. Но иногда возникают случаи, когда без этого не обойтись. Самым частым примером является работа с датчиками телефона.
Рассмотрим проблему: представим что вам нужно сделать чтение данных карты с NFC. Вы выбрали плагин, в нашем случае это был nfc_manager 3.1.0, который позволяет это делать, но на android из него не возвращается номер карты, который вам необходим. Что делать? С такой проблемой мы недавно столкнулись и сейчас расскажем как это решается.
Для решения данной проблемы нужно выполнить несколько шагов.
Во-первых необходимо скопировать код библиотеки к себе в проект. Должно получиться так:
Так же небольшой совет: при наличии нескольких таких пакетов лучше вынести их в отдельную папку на том же уровне вложенности и назвать ее, например, packages.
Затем нам необходимо локализовать момент, в котором не происходит считывание номера банковской карты. Для старта считывания из flutter вызывается метод «startSession», который, в свою очередь, вызывает нативный метод «Nfc#startSession». Внутри него все устроено довольно просто:
adapter.enableReaderMode(activity, {
val handle = UUID.randomUUID().toString()
tags[handle] = it
activity.runOnUiThread { channel.invokeMethod("onDiscovered", getTagMap(it).toMutableMap().apply { put("handle", handle) }) }
}, getFlags(call.argument>("pollingOptions")!!), null)
Теперь нам необходимо немного модифицировать данный код, чтобы мы могли отправить в flutter номер карты.
Для этого сначала в файл build.gradle добавим импорт соответствующей нативной библиотеки:
implementation 'com.github.devnied.emvnfccard: library:3.0.1'
Затем нам нужно сделать реализацию для интерфейса из этой библиотеки IProvider, который передает команды на карту. Необходимая для нас реализация выглядит так:
class Provider : IProvider {
private var mTagCom: IsoDep? = null
@Throws(CommunicationException::class)
override fun transceive(pCommand: ByteArray): ByteArray {
val response: ByteArray = try {
// send command to emv card
mTagCom!!.transceive(pCommand)
} catch (e: IOException) {
throw CommunicationException(e.message)
}
return response
}
override fun getAt(): ByteArray {
// For NFC-A
return mTagCom!!.historicalBytes
// For NFC-B
// return mTagCom.getHiLayerResponse();
}
fun setmTagCom(mTagCom: IsoDep?) {
this.mTagCom = mTagCom
}
}
Затем отредактируем непосредственно сам код перед отправкой данных в flutter в »onDiscovered». Для чтения данных карты нам необходимо добавить выше созданную реализацию интерфейса, а также конфиг и парсер. Код выглядит примерно так:
val provider: IProvider = Provider()
(provider as Provider).setmTagCom(IsoDep.get(it))
val config: EmvTemplate.Config = EmvTemplate.Config()
.setContactLess(true) // Enable contact less reading (default: true)
.setReadAllAids(true) // Read all aids in card (default: true)
.setReadTransactions(true) // Read all transactions (default: true)
.setReadCplc(false) // Read and extract CPCLC data (default: false)
.setRemoveDefaultParsers(false) // Remove default parsers for GeldKarte and EmvCard (default: false)
.setReadAt(true) // Read and extract ATR/ATS and description
val parser = EmvTemplate.Builder() //
.setProvider(provider) // Define provider
.setConfig(config) // Define config
//.setTerminal(terminal) (optional) you can define a custom terminal implementation to create APDU
.build()
Затем нам необходимо подключиться к карте и считать необходимые для нас данные:
val mIsoDep = IsoDep.get(it)
mIsoDep.connect()
val card = parser.readEmvCard()
И наконец отправим необходимы для нас данные в flutter, добавив к map полученные данные карты:
activity.runOnUiThread {
channel.invokeMethod("onDiscovered", getTagMap(it).toMutableMap().apply {
put("handle", handle)
put("cardNumber", card.cardNumber)
put("holderFirstName", card.holderFirstname)
put("holderLastName", card.holderLastname)
put("type", card.type.getName())
})
}
Теперь мы можем получать данные карты с NFC.