summaryrefslogtreecommitdiff
path: root/plugins/HwHotKeys/src/HwHotKeys_DllMain.cpp
blob: 63d0a172fa9ca6283ab916203e1b09b4cf2f9c5f (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
/* ============================================================================
Hardware HotKeys plugin for Miranda NG.
Copyright © Eugene f2065, http://f2065.narod.ru, f2065 mail.ru, ICQ 35078112

This program is free software; you can redistribute it and / or
modify it under the terms of the GNU General Public License
as published by the Free Software Foundation; either version 2
of the License, or(at your option) any later version.

This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.See the
GNU General Public License for more details.

You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111 - 1307, USA.
============================================================================ */



#include "stdafx.h"

CMPlugin g_plugin;
int &hLangpack(g_plugin.m_hLang);
HWND hDialogWnd = nullptr; // хэндл окна настроек, он глобально используется для вывода туда в реалтайме сканкодов клавы из хука
HHOOK hHook;
CLIST_INTERFACE *pcli;

// там хранятся настройки - сканкоды кнопок для закрытия/показа/чтения, на которые должны реагировать соответствующие действия
DWORD code_Close = 0;
DWORD code_HideShow = 0;
DWORD code_ReadMsg = 0;
// в _tmp копируются настройки на время диалога настроек
DWORD code_Close_tmp = 0;
DWORD code_HideShow_tmp = 0;
DWORD code_ReadMsg_tmp = 0;
// формат сканкодов тут такой:
// младшие 0...7 биты - это аппаратный сканкод (как он приходит в KbdLLHookStruct.scanCode&0xFF)
// 8 бит - это как бы часть сканкода, флаг LLKHF_EXTENDED ((LOBYTE(KbdLLHookStruct.flags)&1) - часть кнопок с этим флагом 
// (авторам 101-кнопочной клавы не хватило 255 сканкодов, т.к. там много старых кнопок которые уже давно вымерли типа F13-F20).
// итак, плагин оперирует 9-битными сканкодами (имена кнопок всех этих сканкодов прописаны в таблице *key_tab[0x200]).
// биты 9-19 - не используются.
// биты 20-31 - это флаги регистров, Shift/Ctrl/Alt/Win. На 4 кнопки - 12 битов. Потому что каждая кнопка 3 имеет бита - левая, правая, или любая.
// битовые маски регистров опеределены в key_flag_sr и соседних именах

// плагин оперирует 10 - битными "сканкодами" собственного формата.
// младшие 0...7 биты - это обычно аппаратный сканкод (как он приходит в KbdLLHookStruct.scanCode&0xFF)
// 8 бит - это как бы часть сканкода, флаг LLKHF_EXTENDED ((LOBYTE(KbdLLHookStruct.flags)&1) - часть кнопок с этим флагом 
// (авторам 101-кнопочной клавы не хватило 255 сканкодов, т.к. там много старых кнопок которые уже давно вымерли типа F13-F20).
// 9 бит - это значит что в битах 0...7 не аппаратный сканкод, а VK-код (KbdLLHookStruct.vkCode), и 8 бит при этом всегда 0.
// сделано это потому что некоторые USB-клавиатуры не выдают в хук аппаратные скандоды, и если там 0 - то будет использован VK-код.
// (имена кнопок всех этих сканкодов прописаны в таблице *key_tab[0x300]).
// биты 10-19 - не используются.
// биты 20-31 - это флаги регистров, Shift/Ctrl/Alt/Win. На 4 кнопки - 12 битов. Потому что каждая кнопка 3 имеет бита - левая, правая, или любая.
// битовые маски регистров опеределены в key_flag_sr и соседних именах




DWORD key_code = 0; // там постоянно обновляется сканкод текущей нажатой комбинации
DWORD key_code_raw = 0;
DWORD key_code_assign = 0;  // во время настроек там хранится сканкод последней нажатой комбинации, который можно назначить для действия.

CHAR key_name_buffer[150]; // буфер куда печатается имя кнопки в настройках. ANSI. 
// Самая длинная в теории строка - «L_Shift + L_Ctrl + L_Alt + L_Win + R_Shift + R_Ctrl + R_Alt + R_Win + MSO_Print, HP_TouchPadOff» - менее 100 букв.

// ============================================================================

PLUGININFOEX PluginInfoEx = {
	sizeof(PLUGININFOEX),
	__PLUGIN_NAME,
	PLUGIN_MAKE_VERSION(__MAJOR_VERSION, __MINOR_VERSION, __RELEASE_NUM, __BUILD_NUM),
	__DESCRIPTION_MIR,
	__AUTHOR,
	__COPYRIGHT,
	__AUTHORWEB,
	UNICODE_AWARE,
	// {315B3800-8258-44C4-B60E-58C50B9303B6} - GUID для плагина HwHotKeys
	{ 0x315b3800, 0x8258, 0x44c4, { 0xb6, 0xe, 0x58, 0xc5, 0xb, 0x93, 0x3, 0xb6 } }
};

extern "C" __declspec(dllexport) PLUGININFOEX * MirandaPluginInfoEx(DWORD)
{
	return &PluginInfoEx;
}

// ============================================================================

extern "C" __declspec(dllexport) int Load(void)
{
	// загружаем (из БД) настройки плагина
	code_Close = db_get_dw(0, __DbModName, "Close", 0);
	code_HideShow = db_get_dw(0, __DbModName, "HideShow", 0);
	code_ReadMsg = db_get_dw(0, __DbModName, "ReadMsg", 0);

	// установка кода локализации - макрос mir_getLP(PlgInfoEx)
	mir_getLP(&PluginInfoEx);

	// Интерфейс контактлиста - макрос заполняет CLIST_INTERFACE *pcli;
	pcli = Clist_GetInterface();

	// регистрация диалога опций
	HookEvent(ME_OPT_INITIALISE, initializeOptions);

	// установка хука для низкоуровневой обработки хоткеев
	hHook = SetWindowsHookExA(WH_KEYBOARD_LL, key_hook, g_plugin.getInst(), 0);

	//	StringCbPrintfA(key_name_buffer, 100, "hHook = 0x%x, Err = %u", hHook, GetLastError);
	//	MessageBoxA(0, key_name_buffer, 0, 0);

	return 0;
}

// ====================================

int initializeOptions(WPARAM wParam, LPARAM)
{
	OPTIONSDIALOGPAGE OptDlgPg = { sizeof(OptDlgPg) };
	OptDlgPg.position = 100000000;
	OptDlgPg.hInstance = g_plugin.getInst();
	OptDlgPg.flags = ODPF_BOLDGROUPS | ODPF_UNICODE;
	OptDlgPg.pszTemplate = MAKEINTRESOURCEA(dlg_options);
	OptDlgPg.szGroup.w = LPGENW("Customize");
	OptDlgPg.szTitle.w = LPGENW("Hardware HotKeys");
	OptDlgPg.pfnDlgProc = OptDlgProc;
	OptDlgPg.hLangpack = hLangpack;
	Options_AddPage(wParam, &OptDlgPg);
	return 0;
}

// ============================================================================

extern "C" __declspec(dllexport) int Unload(void)
{
	UnhookWindowsHookEx(hHook);
	return 0;
}

// ============================================================================