Проблема
Проблема, которая заставила меня написать данную статью, возникла при внедрении функционала в розничные магазины, который предполагает ввод некоторых данных. Исходные данные, сенсорный POS-Теринал, ОС Windows 10, Розница 1.0(Не имеет значения). Сенсор на POS-Терминалах не поддерживает мультитач и работает как обычная мышка. Попытки перевести систему в режим планшета и использовать сенсорную клавиатуру стандартными методами не увенчались успехом, клавиатура не появляется при активации поля ввода, плюс для не мультитач экранов режим планшета, это редкостное извращение. Информации по альтернативным способам работы с сенсорной клавиатурой найдено не было, поэтому пришлось изобретать велосипед.
Решение
И так, в обычном режиме сенсорная клавиатура не доступна и есть возможность использовать лишь экранную клавиатуру, которая в свою очередь менее удобна и так же не подразумевает свое появление при активации поля и закрыв ее придется что-то придумывать, чтобы она опять появилась (на самом деле не что-то, а можно просто выполнить команду osk.exe, экранная клавиатура появиться на экране).
Для того чтобы сенсорная клавиатура запустилась в обычном режиме нужно лишь запустить фоновый процесс:
C:Program FilesCommon Filesmicrosoft sharedinkTabTip.exe
Если процесс еще не был запущен на экране появится сенсорная клавиатура, при повторном запуске ничего происходить не будет, есть возможность просто завершать процесс и запускать заново, но вот завершить его можно лишь с повышением привилегий (выполнив команду с правами администратора). На просторах интернета был найден рабочий код, который позволяет показать клавиатуру, если она еще не открыта и скрыть, если наоборот.
#include <initguid.h>
#include <Objbase.h>
#pragma hdrstop
// 4ce576fa-83dc-4F88-951c-9d0782b4e376
DEFINE_GUID(CLSID_UIHostNoLaunch,
0x4CE576FA, 0x83DC, 0x4f88, 0x95, 0x1C, 0x9D, 0x07, 0x82, 0xB4, 0xE3, 0x76);
// 37c994e7_432b_4834_a2f7_dce1f13b834b
DEFINE_GUID(IID_ITipInvocation,
0x37c994e7, 0x432b, 0x4834, 0xa2, 0xf7, 0xdc, 0xe1, 0xf1, 0x3b, 0x83, 0x4b);
struct ITipInvocation : IUnknown
{
virtual HRESULT STDMETHODCALLTYPE Toggle(HWND wnd) = 0;
};
int WinMain(HINSTANCE hInst, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow)
{
HRESULT hr;
hr = CoInitialize(0);
ITipInvocation* tip;
hr = CoCreateInstance(CLSID_UIHostNoLaunch, 0, CLSCTX_INPROC_HANDLER | CLSCTX_LOCAL_SERVER, IID_ITipInvocation, (void**)&tip);
tip->Toggle(GetDesktopWindow());
tip->Release();
return 0;
}
Дальше обернул этот код в библиотеку и при появление окна, где требуется ввести данные открываю клавиатуру, а при его закрытие соответственно закрываю.
УправлениеКлавиатурой = Новый COMОбъект("TouchKeyboardDLL.ShKb");
УправлениеКлавиатурой.ShowKeyboard();
Итоги
Кому это может быть полезно, сейчас в связи с законом ФЗ-54, на кассовых местах может потребоваться ввод адреса электронной почты, я специально опускаю ввод телефона, так как с цифровой клавиатурой особых проблем нет, это решение можно добавить во внешние обработки для работы с кассовыми аппаратами, так же такой способ ввода может упростить жизнь, если требуется заполнять какие-то данные по клиенту на кассовом месте.
Скажу честно, мы не ввели данный функционал в эксплуатацию и не факт, что введем, у нас не запущен процесс explorer на моноблоках, что не позволяет задействовать сенсорную клавиатуру. Будем думать, что нам важнее. Исходники компоненты, выкладывать не буду, так как они делались для теста и не несут, какой-то нужной информации, кому очень захочется пишите в ЛС вышлю без вопросов. В архиве готовая библиотека и тестовая обработка, основные моменты описаны в статье. Надеюсь кому-то это пригодится.
Отличная идея, но чем был обусловлен выбор COM объекта, а не реализация NativeAPI библиотеки? Конечно и установку COM объекта можно автоматизировать, но всё-таки в отношении NativeAPI ничего особенного делать не надо, закинул ZIP архив с библиотекой в макет и устанавливай на клиентах спокойно.
Выбор COM был обусловлен более быстрой реализацией библиотеки, меньше мороки со сборкой, как я написал не факт ,что мы это запустим, поэтому делал это как тестовый вариант, если будем запускать в работу то скорее всего оберну в NativeAPI. Первый вариант был вообще консольное приложение, но там возникла проблема потери фокуса ).
Может быть компоненту неправильно установил? Подскажите, как правильно тогда.