Материалы, которые использовал для создания приложений:
- Уроки по созданию приложения для взаимодействия с NFC в android studio Вам нужно только повторять за автором и на выходе вы получите функционал по чтению и записи меток.
- Статьи на инфостарте "Мобильное приложение 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
Другие разработки, которые могут вас заинтересовать:
Здравствуйте!
Идея, безусловно, полезная, но взаимодействие мобильной платформы с нативными приложениями через Intent здесь давно описана с примерами и опробована.
Если Вы владеете и 1С и Java, очень прошу Вас написать внешнюю компоненту для мобильного приложения, делающую то же самое.
Технология описана на сайте ИТС.
Я не смог найти ни одного примера с исходниками, а так хочется иметь «печку», чтобы начать танцевать.
(1) в идеале я и хотел видеть этот функционал через внешнюю компоненту , но пока не удалось полностью разобраться с этим
(1) Вот статья на эту тему:
https://infostart.ru/public/589245/
В android, NFC тесно связан с текущей activity и получение данных в фоне неактивным activity невозможно.
Вышеописанное скорее всего сделано для безопасности приложений бесконтактной оплаты.
Из ВК, невозможно создать полноценную activity, с которым будет работать NFC.
Я так же пытался сделать внешнюю компоненты для мобильной платформы, которая бы читала и писала nfc метки mifareclassic с возможностью установки ключей. Затык у меня произошёл из-за того, что мне не удавалось вызвать Activity, получающую Intent от NFC адаптера. У меня сложилось впечатление, что где-то происходит обращение к локальному указателю (в терминах jni), которые уже утратил актуальность.
Получить Intent то NFC адаптера каким-нибудь другим классом, обрабатывющим Intent-ы ( типа Service или BroadcastReceiver) и не имеющим проблем с вызовом из внешней компоненты, как у Activity, у меня не получилось.
Получать Intent то NFC могло только Actyvity, указанное адаптеру через enableForegroundDispatch.
Примечательно, что если компоненту не пересобирать, а оставить в виде Android приложение, то обработка nfc таким приложением происходила как надо.
Так же стоит указать, что действия с меткой должны происходить не по нажатию кнопки на форме, а по внесению метки в поле, а программа должна быть в состоянии готовности к определённым действиям.
Свою неудачу считаю лишь следствие недостаточности знаний, т.к., например, эти: http://www.rightscan.ru создать такую библиотеку удалось (не сочтите за рекламу) и её даже можно скачать с примером конфигурации для 1С.
(6) Ключевое слово, которое позволяет rightscan делать такие вещи — SafeDroid. Они изменили ядро ОС. В обычном андроиде ни работа с nfc без activity, на перехват клавиатуры невозможен (смздесь )
(7)Не думаю, что они в ядре что-то меняли. Их компонента, работала, как на специализированном сканере, правда у них же купленном, так и на банальном Самсунге с NFC считывателем, и на гражданских телефонах корректно лесом слало (в отдельно открытом Activity), что нет NFC адаптера. Сейчас конечно уже всё итерации не упомню, в сентябре 2017 задачу бросил. С другой стороны, если бы в специализированном сканере было пропатченное ядро, то и мои усилия бы поди увенчались успехом — в основном тестил-то я на нём.
А про изменение ядра это они сами Вам рассказали, или догадка ?
(8) А как она работает с nfc, в отдельном окне или прямо из формы 1С? Если второе, то они либо используют какие-то недокументированные возможности андроид либо, что более вероятно, возможности технологии внешних компонент. Получить nfc адаптер в андроиде можно только в главном потоке приложения и только с привязкой к foreground activity. А внешняя компонента работает в отдельном потоке и не дает возможности получить текущее окно. NDK возможностей для работы с nfc не имеет.
Интересно, работает ли их компонента на других устройствах с физической клавиатурой? Эту тему я копал глубже и с гораздо большей вероятностью утверждаю, что с помощью ТВК перехватить нажатие клавиш невозможно, если только не использовать андроид под root.
(9) В отдельном чёрном окне. Внизу по центру пишет «приложите метку», «адаптер не найден» или «адаптер выключен».
Пробовал обмануть, вызвав промежуточный класс уже из которого вызывать свою Activity. Т.е. что бы не средствами JNI на C++ на писать типа jenv->CallVoidMethod(activity_glob, runOnUiThread, runObject);
а на Яве, в среде Android, штатно, как белый человек. Безрезультатно. Промежуточный класс вызывается, работает, а с Activity облом
В jni предупреждают, не стоит использовать для всего подряд, так как встроенные функции в большинстве случаев будут быстрее.
В теории можно сделать foreground activity которая будет вызываться из ВК и взаимодействовать с nfc.
Но обмен данными с этим activity будет происходить через intent
моб. 1с -> jni -> activity nfc
моб. 1с <- jni <- activity nfc
логично что быстрее, передавать сразу intent без ВК
моб. 1с -> activity nfc
моб. 1с <- activity nfc
(11) А где эта activity будет задекларирована? Если в ВК, то ее открыть невозможно, activity должны быть включены в манифест главного приложения. Если писать отдельный apk, то для серьезной разработки это плохое решение
(6) После нашего обсуждения я нашел метод как открывать свои окна из ВК, думаю прикрутить к ним nfc со временем. Небольшая подсказка куда копать:https://developer.android.com/reference/android/app/Dialog
(12) Читайте внимательней, в (11) речь именно про плохое решение
Не суть важно где activity будет, суть в том что с ВК не уйти от передачи intent.
Т.к. даже разные activity одного приложения, передают информацию друг другу через intent.
(11) Как планируете принимать интент из 1С без ВК? Платформа умеет их только отправлять
Не планирую.
Годом ранее делал публикацию про NFC, intent возвращается также как в этой статье.
Но и без ЗапускПриложениеМобильногоУстройства и можно получить intent внутри моб. 1с, где то здесь были описаны решения, но можно просто посмотреть манифест внутри архива с платформой.
А существуют ли опубликованные в google play приложения, которые позволили бы делать то же самое?
(17) не нашел приложение в маркете с которым можно было сделать взаимодействие на чтение/запись nfc поэтому пришлось изобретать . На gitе можно поискать разработки .