Взаимодействие с NFC через мобильную платформу 1С










Цель была научиться взаимодействовать с внешними приложениями из приложения на мобильной платформе 1С. Для данной цели выбрал тему чтение/запись NFC меток. Было разработано: 1) приложение в android studio, которое взаимодействует с NFC 2) приложение на мобильное платформе 1с, которое взаимодействует с разработанным приложением в android studio.

Материалы, которые использовал для создания приложений:

  1. Уроки по созданию приложения для взаимодействия с NFC в android studio   Вам нужно только повторять за автором и на выходе вы получите функционал по чтению и записи меток.
  2. Статьи на инфостарте "Мобильное приложение 1С и приложение Java. Совместная работа через Intent. " в них очень подробно расписано, как "подружить" приложения:

Всегда актуально: Курс Дмитрия Шерстобитова на курсах-1с.рф

Функционал, который получилось реализовать:

  • Получение состояния (включено/выключено)
  • Переход в настройки
  • Запись (в тестовой записи текст вшит в приложение на android studio, в обычной записи текст передается из приложения на мобильной платформе 1С)
  • Чтение с передачей ID метки и содержимого в приложение на МП 1С. Т.к поддерживается только Ndef то формат передается просто текстом.
  • Установка внешнего приложения из приложения на МП 1С.

Приведу пример кода некоторых функций на стороне МП 1С и android studio:

Получение состояния (включено/выключено)

Код в 1С:

&НаКлиенте
Процедура ПолучитьСостояниеNFC()
#Если МобильноеПриложениеКлиент Тогда
НовВз = Новый ЗапускПриложенияМобильногоУстройства();
НовВз.Действие="ru.mp.intent.action.NFCstate";
НовВз.Запустить(Истина);

Для Каждого Стр Из НовВз.ДополнительныеДанные Цикл
Если Стр.Ключ = "state" Тогда
СостояниеNFC = Стр.Значение;
КонецЕсли;
КонецЦикла;
#КонецЕсли
КонецПроцедуры

Код в android studio:

NfcAdapter nfcAdapter;
 
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.nfc_state);
 
nfcAdapter = NfcAdapter.getDefaultAdapter(this);
 
//++ определим состояние NFC Урок 1
String textState = new String();
Boolean StateNfc = false;
if (nfcAdapter != null && nfcAdapter.isEnabled()) {
textState = "NFC Включено";
android.widget.Toast.makeText(this, textState, Toast.LENGTH_SHORT).show();
StateNfc = true;
}else{
textState = "NFC Выключено";
android.widget.Toast.makeText(this, textState, Toast.LENGTH_SHORT).show();
StateNfc = false;
}
//-- определим состояние NFC Урок 1
 
Intent intent = new Intent();
intent.putExtra("state", StateNfc);
setResult(RESULT_OK, intent);
finish();
 
}

Переход в настройки

Код в 1С:

&НаКлиенте
Процедура ВключитьОтключитьNFC(Команда)
#Если МобильноеПриложениеКлиент Тогда
НовВз = Новый ЗапускПриложенияМобильногоУстройства();
НовВз.Действие="ru.mp.intent.action.OnOffNFC";
НовВз.Запустить(Истина);
#КонецЕсли
КонецПроцедуры

Код в android studio:

public  void ActionNFCSettings(){
if (android.os.Build.VERSION.SDK_INT >= 16) {
startActivity(new Intent(android.provider.Settings.ACTION_NFC_SETTINGS));
} else {
startActivity(new Intent(android.provider.Settings.ACTION_WIRELESS_SETTINGS));
}
 
this.finish();
}

Тестовая запись

@Override
protected void onNewIntent(Intent intent) {
 
//урок 3
if(intent.hasExtra(NfcAdapter.EXTRA_TAG)) {
//android.widget.Toast.makeText(this, "NFC ТЕГ", Toast.LENGTH_SHORT).show();
 
Tag tag = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG);
NdefMessage ndefMessage = createNdefMessage("Android studio");
 
writeNdefMessage(tag,ndefMessage);
setResult(RESULT_OK);
finish();
 
}
}

Записать

Код в 1С:

&НаКлиенте
Процедура ЗаписатьNFC(Команда)
#Если МобильноеПриложениеКлиент Тогда
НовВз = Новый ЗапускПриложенияМобильногоУстройства();
НовВз.Действие="ru.mp.intent.action.WriteNFC";
НовВз.ДополнительныеДанные.Добавить("textNFC", "Мобильная платформа 1С");
НовВз.Запустить(Истина);
#КонецЕсли
КонецПроцедуры

Код в android studio:

NfcAdapter nfcAdapter;
public static String textNFC;
 
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.nfc_write);
 
nfcAdapter = NfcAdapter.getDefaultAdapter(this);
 
if (nfcAdapter != null && nfcAdapter.isEnabled()) {
 
}else{
android.widget.Toast.makeText(this, "NFC Выключено", Toast.LENGTH_SHORT).show();
this.finish();
}
 
Intent intent = getIntent();
textNFC = intent.getStringExtra("textNFC");
 
}
 
@Override
protected void onNewIntent(Intent intent) {
 
if(intent.hasExtra(NfcAdapter.EXTRA_TAG)) {
//android.widget.Toast.makeText(this, "NFC ТЕГ", Toast.LENGTH_SHORT).show();
Tag tag = intent.getParcelableExtra(NfcAdapter.EXTRA_TAG);
NdefMessage ndefMessage = createNdefMessage(textNFC);
 
writeNdefMessage(tag,ndefMessage);
setResult(RESULT_OK);
finish();
}
}

Тестировал только на Nexus 5 и на карте, которая поддерживает Ndef (ее хар-ки в изображениях).

Использовалась мобильная платформа 8.3.11, технологическая платформа 8.3.10

Содержимое архива:

1)  Два apk:

  • приложение на мобильной платформе 1С
  • приложение на android stuido

2) Конфигурация приложения 1С и архив проекта на android studio

Другие разработки, которые могут вас заинтересовать:

17 Comments

  1. saa@kuzov.org

    Здравствуйте!

    Идея, безусловно, полезная, но взаимодействие мобильной платформы с нативными приложениями через Intent здесь давно описана с примерами и опробована.

    Если Вы владеете и 1С и Java, очень прошу Вас написать внешнюю компоненту для мобильного приложения, делающую то же самое.

    Технология описана на сайте ИТС.

    Я не смог найти ни одного примера с исходниками, а так хочется иметь «печку», чтобы начать танцевать.

    Reply
  2. leobrn

    (1) в идеале я и хотел видеть этот функционал через внешнюю компоненту , но пока не удалось полностью разобраться с этим

    Reply
  3. CyberCerber

    (1) Вот статья на эту тему:

    https://infostart.ru/public/589245/

    Reply
  4. PloAl

    В android, NFC тесно связан с текущей activity и получение данных в фоне неактивным activity невозможно.

    Вышеописанное скорее всего сделано для безопасности приложений бесконтактной оплаты.

    Из ВК, невозможно создать полноценную activity, с которым будет работать NFC.

    Reply
  5. user808274

    Я так же пытался сделать внешнюю компоненты для мобильной платформы, которая бы читала и писала nfc метки mifareclassic с возможностью установки ключей. Затык у меня произошёл из-за того, что мне не удавалось вызвать Activity, получающую Intent от NFC адаптера. У меня сложилось впечатление, что где-то происходит обращение к локальному указателю (в терминах jni), которые уже утратил актуальность.

    Получить Intent то NFC адаптера каким-нибудь другим классом, обрабатывющим Intent-ы ( типа Service или BroadcastReceiver) и не имеющим проблем с вызовом из внешней компоненты, как у Activity, у меня не получилось.

    Получать Intent то NFC могло только Actyvity, указанное адаптеру через enableForegroundDispatch.

    Примечательно, что если компоненту не пересобирать, а оставить в виде Android приложение, то обработка nfc таким приложением происходила как надо.

    Так же стоит указать, что действия с меткой должны происходить не по нажатию кнопки на форме, а по внесению метки в поле, а программа должна быть в состоянии готовности к определённым действиям.

    Свою неудачу считаю лишь следствие недостаточности знаний, т.к., например, эти: http://www.rightscan.ru создать такую библиотеку удалось (не сочтите за рекламу) и её даже можно скачать с примером конфигурации для 1С.

    Reply
  6. IgorKissil

    (6) Ключевое слово, которое позволяет rightscan делать такие вещи — SafeDroid. Они изменили ядро ОС. В обычном андроиде ни работа с nfc без activity, на перехват клавиатуры невозможен (см здесь)

    Reply
  7. user808274

    (7)Не думаю, что они в ядре что-то меняли. Их компонента, работала, как на специализированном сканере, правда у них же купленном, так и на банальном Самсунге с NFC считывателем, и на гражданских телефонах корректно лесом слало (в отдельно открытом Activity), что нет NFC адаптера. Сейчас конечно уже всё итерации не упомню, в сентябре 2017 задачу бросил. С другой стороны, если бы в специализированном сканере было пропатченное ядро, то и мои усилия бы поди увенчались успехом — в основном тестил-то я на нём.

    А про изменение ядра это они сами Вам рассказали, или догадка ?

    Reply
  8. IgorKissil

    (8) А как она работает с nfc, в отдельном окне или прямо из формы 1С? Если второе, то они либо используют какие-то недокументированные возможности андроид либо, что более вероятно, возможности технологии внешних компонент. Получить nfc адаптер в андроиде можно только в главном потоке приложения и только с привязкой к foreground activity. А внешняя компонента работает в отдельном потоке и не дает возможности получить текущее окно. NDK возможностей для работы с nfc не имеет.

    Интересно, работает ли их компонента на других устройствах с физической клавиатурой? Эту тему я копал глубже и с гораздо большей вероятностью утверждаю, что с помощью ТВК перехватить нажатие клавиш невозможно, если только не использовать андроид под root.

    Reply
  9. user808274

    (9) В отдельном чёрном окне. Внизу по центру пишет «приложите метку», «адаптер не найден» или «адаптер выключен».

    Пробовал обмануть, вызвав промежуточный класс уже из которого вызывать свою Activity. Т.е. что бы не средствами JNI на C++ на писать типа jenv->CallVoidMethod(activity_glob, runOnUiThread, runObject);

    а на Яве, в среде Android, штатно, как белый человек. Безрезультатно. Промежуточный класс вызывается, работает, а с Activity облом

    Reply
  10. PloAl

    В jni предупреждают, не стоит использовать для всего подряд, так как встроенные функции в большинстве случаев будут быстрее.

    В теории можно сделать foreground activity которая будет вызываться из ВК и взаимодействовать с nfc.

    Но обмен данными с этим activity будет происходить через intent

    моб. 1с -> jni -> activity nfc

    моб. 1с <- jni <- activity nfc

    логично что быстрее, передавать сразу intent без ВК

    моб. 1с -> activity nfc

    моб. 1с <- activity nfc

    Reply
  11. IgorKissil

    (11) А где эта activity будет задекларирована? Если в ВК, то ее открыть невозможно, activity должны быть включены в манифест главного приложения. Если писать отдельный apk, то для серьезной разработки это плохое решение

    Reply
  12. IgorKissil

    (6) После нашего обсуждения я нашел метод как открывать свои окна из ВК, думаю прикрутить к ним nfc со временем. Небольшая подсказка куда копать: https://developer.android.com/reference/android/app/Dialog

    Reply
  13. PloAl

    (12) Читайте внимательней, в (11) речь именно про плохое решение

    Не суть важно где activity будет, суть в том что с ВК не уйти от передачи intent.

    Т.к. даже разные activity одного приложения, передают информацию друг другу через intent.

    Reply
  14. IgorKissil

    (11) Как планируете принимать интент из 1С без ВК? Платформа умеет их только отправлять

    Reply
  15. PloAl

    Не планирую.

    Годом ранее делал публикацию про NFC, intent возвращается также как в этой статье.

    Но и без ЗапускПриложениеМобильногоУстройства и можно получить intent внутри моб. 1с, где то здесь были описаны решения, но можно просто посмотреть манифест внутри архива с платформой.

    Reply
  16. TPID

    А существуют ли опубликованные в google play приложения, которые позволили бы делать то же самое?

    Reply
  17. leobrn

    (17) не нашел приложение в маркете с которым можно было сделать взаимодействие на чтение/запись nfc поэтому пришлось изобретать . На gitе можно поискать разработки .

    Reply

Leave a Comment

Ваш адрес email не будет опубликован. Обязательные поля помечены *