-
Notifications
You must be signed in to change notification settings - Fork 1
/
cSendKeys.cls
369 lines (323 loc) · 12.6 KB
/
cSendKeys.cls
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
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
VERSION 1.0 CLASS
BEGIN
MultiUse = -1 'True
END
Attribute VB_Name = "cSendKeys"
Attribute VB_GlobalNameSpace = False
Attribute VB_Creatable = True
Attribute VB_PredeclaredId = False
Attribute VB_Exposed = False
Option Explicit
Public Enum MoreKeyConstants
VK_LWIN = &H5B 'Left Windows key (Microsoft® Natural® keyboard)
VK_RWIN = &H5C 'Right Windows key (Natural keyboard)
VK_APPS = &H5D 'Applications key (Natural keyboard)
VK_SLEEP = &H5F 'Computer Sleep key
VK_RMENU = &HA5 ' Right MENU key
VK_BROWSER_BACK = &HA6 'Windows 2000/XP: Browser Back key
VK_BROWSER_FORWARD = &HA7 'Windows 2000/XP: Browser Forward key
VK_BROWSER_REFRESH = &HA8 'Windows 2000/XP: Browser Refresh key
VK_BROWSER_STOP = &HA9 'Windows 2000/XP: Browser Stop key
VK_BROWSER_SEARCH = &HAA 'Windows 2000/XP: Browser Search key
VK_BROWSER_FAVORITES = &HAB 'Windows 2000/XP: Browser Favorites key
VK_BROWSER_HOME = &HAC 'Windows 2000/XP: Browser Start and Home key
VK_VOLUME_MUTE = &HAD 'Windows 2000/XP: Volume Mute key
VK_VOLUME_DOWN = &HAE 'Windows 2000/XP: Volume Down key
VK_VOLUME_UP = &HAF 'Windows 2000/XP: Volume Up key
VK_MEDIA_NEXT_TRACK = &HB0 'Windows 2000/XP: Next Track key
VK_MEDIA_PREV_TRACK = &HB1 'Windows 2000/XP: Previous Track key
VK_MEDIA_STOP = &HB2 'Windows 2000/XP: Stop Media key
VK_MEDIA_PLAY_PAUSE = &HB3 'Windows 2000/XP: Play/Pause Media key
VK_LAUNCH_MAIL = &HB4 'Windows 2000/XP: Start Mail key
VK_LAUNCH_MEDIA_SELECT = &HB5 'Windows 2000/XP: Select Media key
VK_LAUNCH_APP1 = &HB6 'Windows 2000/XP: Start Application 1 key
VK_LAUNCH_APP2 = &HB7 'Windows 2000/XP: Start Application 2 key
VK_OEM_1 = &HBA 'Used for miscellaneous characters; it can vary by keyboard. Windows 2000/XP: For the US standard keyboard, the ';:' key
VK_OEM_PLUS = &HBB 'Windows 2000/XP: For any country/region, the '+' key
VK_OEM_COMMA = &HBC 'Windows 2000/XP: For any country/region, the ',' key
VK_OEM_MINUS = &HBD 'Windows 2000/XP: For any country/region, the '-' key
VK_OEM_PERIOD = &HBE 'Windows 2000/XP: For any country/region, the '.' key
VK_OEM_2 = &HBF 'Used for miscellaneous characters; it can vary by keyboard. Windows 2000/XP: For the US standard keyboard, the '/?' key
VK_OEM_3 = &HC0 'Used for miscellaneous characters; it can vary by keyboard. Windows 2000/XP: For the US standard keyboard, the '`~' key
'— C1–D7 Reserved
'— D8–DA Unassigned
VK_OEM_4 = &HDB 'Used for miscellaneous characters; it can vary by keyboard. Windows 2000/XP: For the US standard keyboard, the '[{' key
VK_OEM_5 = &HDC 'Used for miscellaneous characters; it can vary by keyboard. Windows 2000/XP: For the US standard keyboard, the '\|' key
VK_OEM_6 = &HDD 'Used for miscellaneous characters; it can vary by keyboard Windows 2000/XP: For the US standard keyboard, the ']}' key
VK_OEM_7 = &HDE ' Used for miscellaneous characters; it can vary by keyboard. Windows 2000/XP: For the US standard keyboard, the 'single-quote/double-quote' key
VK_OEM_8 = &HDF 'Used for miscellaneous characters; it can vary by keyboard. — E0 Reserved
'- E1 OEM specific
VK_OEM_102 = &HE2 'Windows 2000/XP: Either the angle bracket key or the backslash key on the RT 102-key keyboard
' E3–E4 OEM specific
VK_PROCESSKEY = &HE5 'Windows 95/98/Me, Windows NT 4.0, Windows 2000/XP: IME PROCESS key
' E6 OEM specific
VK_PACKET = &HE7 'Windows 2000/XP: Used to pass Unicode characters as if they were keystrokes. The VK_PACKET key is the low word of a 32-bit Virtual Key value used for non-keyboard input methods. For more information, see Remark in KEYBDINPUT, SendInput, WM_KEYDOWN, and WM_KEYUP
'— E8 Unassigned
' E9–F5 OEM specific
VK_ATTN = &HF6 'Attn key
VK_CRSEL = &HF7 'CrSel key
VK_EXSEL = &HF8 'ExSel key
VK_EREOF = &HF9 'Erase EOF key
VK_PLAY = &HFA 'Play key
VK_ZOOM = &HFB 'Zoom key
VK_NONAME = &HFC 'Reserved for future use
VK_PA1 = &HFD 'PA1 key
VK_OEM_CLEAR = &HFE 'Clear key
End Enum
Private m_colKeyMap As New Collection
Private Declare Sub keybd_event Lib "user32" ( _
ByVal bVk As Byte, ByVal bScan As Byte, _
ByVal dwFlags As Long, ByVal dwExtraInfo As Long)
Private Const KEYEVENTF_EXTENDEDKEY = &H1
Private Const KEYEVENTF_KEYUP = &H2
Private Declare Function GetVersion Lib "kernel32" () As Long
Private Declare Function VkKeyScan Lib "user32" Alias "VkKeyScanA" ( _
ByVal cChar As Byte) As Integer
Private Declare Function VkKeyScanW Lib "user32" ( _
ByVal cChar As Integer) As Integer
Private Declare Sub CopyMemory Lib "kernel32" Alias "RtlMoveMemory" ( _
lpvDest As Any, lpvSource As Any, ByVal cbCopy As Long)
Private Function nextChar(ByRef sString As String, ByVal iPos As Long, Optional ByVal lLen As Long = 0) As String
If (lLen = 0) Then lLen = Len(sString)
If (iPos + 1 <= lLen) Then
nextChar = Mid$(sString, iPos + 1, 1)
End If
End Function
Public Sub SendKeys(ByVal sKeys As String, Optional ByVal Wait As Boolean)
' The plus sign (+), caret (^), percent sign (%),
' tilde (~), and parentheses ( ) have special
' meanings to SendKeys
' Brackets ([ ]) have no special meaning to SendKeys,
' but you must enclose them in braces.
' To specify brace characters, use {{} and {}}.
' Repeating keys: {LEFT 42} do left 42 times.
' + = Shift
' ^ = Ctrl
' % = Alt
' ~ = enter
' ( = start sub expression. +(EC) = Shift then E then C
On Error GoTo errorHandler
Dim sMsg As String
Dim lErr As Long
Dim iPos As Long
Dim iNextPos As Long
Dim iLen As Long
Dim sChar As String
Dim colBrace As New Collection
Dim sContent As String
Dim sKey As String
Dim sCount As String
Dim lCount As Long
iPos = 1
iLen = Len(sKeys)
Do While iPos <= iLen
sChar = Mid$(sKeys, iPos, 1)
Select Case sChar
Case "+", "~", "%"
If nextChar(sKeys, iPos, iLen) = "(" Then
' Add to brace stack:
colBrace.Add sChar
' send key down
Select Case sChar
Case "+"
KeyDown vbKeyShift
Case "~"
KeyDown vbKeyControl
Case "%"
KeyDown vbKeyMenu
End Select
iPos = iPos + 2
Else
' Key press the key (probably not what you wanted)
Select Case sChar
Case "+"
KeyDown vbKeyShift
KeyUp vbKeyShift
Case "~"
KeyDown vbKeyControl
KeyUp vbKeyControl
Case "%"
KeyDown vbKeyMenu
KeyUp vbKeyMenu
End Select
iPos = iPos + 1
End If
Case "~"
' Enter key:
KeyDown vbKeyReturn
KeyUp vbKeyReturn
iPos = iPos + 1
Case ")"
If (colBrace.Count > 0) Then
sChar = colBrace(colBrace.Count)
' send key up
Select Case sChar
Case "+"
KeyUp vbKeyShift
Case "~"
KeyUp vbKeyControl
Case "%"
KeyUp vbKeyMenu
End Select
colBrace.Remove colBrace.Count
iPos = iPos + 1
Else
' Invalid sendkeys command:
sMsg = "Invalid sendkeys command: unmatched ) at position " & iPos
GoTo errorHandler
End If
Case "{"
' special key
If (iPos + 2 > iLen) Then
sMsg = "Invalid sendkeys command; opening { without content or closing } at position " & iPos
GoTo errorHandler
Else
iNextPos = InStr(iPos + 2, sKeys, "}")
If (iNextPos = 0) Then
sMsg = "Invalid sendkeys command; opening { without closing } at position " & iPos
GoTo errorHandler
Else
sContent = Mid$(sKeys, iPos + 1, iNextPos - iPos - 1)
iPos = iNextPos + 1
' is this a key/presses pair?
iNextPos = InStr(sContent, " ")
If (iNextPos > 0) Then
sKey = Left$(sContent, iNextPos - 1)
sCount = Mid$(sContent, iNextPos + 1)
If Not (IsNumeric(sCount)) Then
sMsg = "Invalid sendkeys command; key repetitions '" & sCount & "' is invalid near position " & iPos
lCount = CLng(sCount)
End If
Else
sKey = sContent
lCount = 1
End If
KeyPress sKey, lCount
End If
End If
Case Else
' send the key as is
KeyPress sChar, 1
iPos = iPos + 1
End Select
Loop
If (colBrace.Count > 0) Then
sMsg = "Invalid sendkeys command: more open brackets than close brackets."
GoTo errorHandler
End If
Exit Sub
errorHandler:
If Len(sMsg) = 0 Then
sMsg = Err.Description
lErr = Err.Number
End If
' If we don't clear up the shift/control/alt keys,
' then you might find other apps on the system are hard to
' use.
' Make sure you have Break on Unhandled Errors switched
' on.
Do While colBrace.Count > 0
sChar = colBrace(colBrace.Count)
' send key up
Select Case sChar
Case "+"
KeyUp vbKeyShift
Case "~"
KeyUp vbKeyControl
Case "%"
KeyUp vbKeyMenu
End Select
colBrace.Remove colBrace.Count
Loop
On Error GoTo 0
Err.Raise lErr, App.EXEName & ".cSendKeys", sMsg
Exit Sub
End Sub
Public Sub KeyPress(ByVal sKey As String, Optional ByVal lCount = 1)
Dim vKey As KeyCodeConstants
Dim l As Long
On Error Resume Next
vKey = m_colKeyMap(sKey)
On Error GoTo 0
If (vKey = 0) Then
' translate string into v key code
vKey = KeyCode(sKey)
End If
If (vKey <> 0) Then
For l = 1 To lCount
KeyDown vKey
KeyUp vKey
Next l
Else
Err.Raise 9, , "Key " & sKey & " could not be interpreted."
End If
End Sub
Public Sub KeyDown(ByVal vKey As KeyCodeConstants)
keybd_event vKey, 0, KEYEVENTF_EXTENDEDKEY, 0
End Sub
Public Sub KeyUp(ByVal vKey As KeyCodeConstants)
keybd_event vKey, 0, KEYEVENTF_EXTENDEDKEY Or KEYEVENTF_KEYUP, 0
End Sub
Public Function KeyCode(ByVal sChar As String) As KeyCodeConstants
Dim bNt As Boolean
Dim iKeyCode As Integer
Dim b() As Byte
Dim iKey As Integer
Dim vKey As KeyCodeConstants
Dim iShift As ShiftConstants
' Determine if we have Unicode support or not:
bNt = ((GetVersion() And &H80000000) = 0)
' Get the keyboard scan code for the character:
If (bNt) Then
b = sChar
CopyMemory iKey, b(0), 2
iKeyCode = VkKeyScanW(iKey)
Else
b = StrConv(sChar, vbFromUnicode)
iKeyCode = VkKeyScan(b(0))
End If
KeyCode = (iKeyCode And &HFF&)
End Function
Private Sub Class_Initialize()
m_colKeyMap.Add vbKeyBack, "BACKSPACE"
m_colKeyMap.Add vbKeyBack, "BS"
m_colKeyMap.Add vbKeyBack, "BKSP"
m_colKeyMap.Add vbKeyPause, "BREAK"
m_colKeyMap.Add vbKeyCapital, "CAPSLOCK"
m_colKeyMap.Add vbKeyDelete, "DELETE"
m_colKeyMap.Add vbKeyDelete, "DEL"
m_colKeyMap.Add vbKeyDown, "DOWN"
m_colKeyMap.Add vbKeyEnd, "END"
m_colKeyMap.Add vbKeyReturn, "ENTER"
m_colKeyMap.Add vbKeyReturn, "~"
m_colKeyMap.Add vbKeyEscape, "ESC"
m_colKeyMap.Add vbKeyHelp, "HELP"
m_colKeyMap.Add vbKeyHome, "HOME"
m_colKeyMap.Add vbKeyInsert, "INS"
m_colKeyMap.Add vbKeyInsert, "INSERT"
m_colKeyMap.Add vbKeyLeft, "LEFT"
m_colKeyMap.Add vbKeyNumlock, "NUMLOCK"
m_colKeyMap.Add vbKeyPageDown, "PGDN"
m_colKeyMap.Add vbKeyPageUp, "PGUP"
m_colKeyMap.Add vbKeyPrint, "PRTSC"
m_colKeyMap.Add vbKeyRight, "RIGHT"
m_colKeyMap.Add vbKeyScrollLock, "SCROLLLOCK"
m_colKeyMap.Add vbKeyTab, "TAB"
m_colKeyMap.Add vbKeyUp, "UP"
m_colKeyMap.Add vbKeyF1, "F1"
m_colKeyMap.Add vbKeyF2, "F2"
m_colKeyMap.Add vbKeyF3, "F3"
m_colKeyMap.Add vbKeyF4, "F4"
m_colKeyMap.Add vbKeyF5, "F5"
m_colKeyMap.Add vbKeyF6, "F6"
m_colKeyMap.Add vbKeyF7, "F7"
m_colKeyMap.Add vbKeyF8, "F8"
m_colKeyMap.Add vbKeyF9, "F9"
m_colKeyMap.Add vbKeyF10, "F10"
m_colKeyMap.Add vbKeyF11, "F11"
m_colKeyMap.Add vbKeyF12, "F12"
m_colKeyMap.Add vbKeyF13, "F13"
m_colKeyMap.Add vbKeyF14, "F14"
m_colKeyMap.Add vbKeyF15, "F15"
m_colKeyMap.Add vbKeyF16, "F16"
End Sub