Skip to content

Commit bce4cf7

Browse files
aymanbagabasalexbrainman
authored andcommitted
windows: add GetKeyboardLayout & ToUnicodeEx
These are used along with GetForegroundWindow and GetWindowThreadProcessId to determine the current user layout and translate the base key the user has pressed. Change-Id: Ib833ba7ab54213d83e889ff74c5bc0ace5edbe95 GitHub-Last-Rev: 2afe997 GitHub-Pull-Request: #188 Reviewed-on: https://go-review.googlesource.com/c/sys/+/574755 LUCI-TryBot-Result: Go LUCI <golang-scoped@luci-project-accounts.iam.gserviceaccount.com> Reviewed-by: Cherry Mui <cherryyz@google.com> Reviewed-by: Than McIntosh <thanm@google.com> Reviewed-by: Alex Brainman <alex.brainman@gmail.com> Reviewed-by: Ayman Bagabas <ayman.bagabas@gmail.com>
1 parent 0eac9b5 commit bce4cf7

4 files changed

+101
-2
lines changed

windows/syscall_windows.go

+10-2
Original file line numberDiff line numberDiff line change
@@ -17,8 +17,10 @@ import (
1717
"unsafe"
1818
)
1919

20-
type Handle uintptr
21-
type HWND uintptr
20+
type (
21+
Handle uintptr
22+
HWND uintptr
23+
)
2224

2325
const (
2426
InvalidHandle = ^Handle(0)
@@ -211,6 +213,10 @@ func NewCallbackCDecl(fn interface{}) uintptr {
211213
//sys OpenProcess(desiredAccess uint32, inheritHandle bool, processId uint32) (handle Handle, err error)
212214
//sys ShellExecute(hwnd Handle, verb *uint16, file *uint16, args *uint16, cwd *uint16, showCmd int32) (err error) [failretval<=32] = shell32.ShellExecuteW
213215
//sys GetWindowThreadProcessId(hwnd HWND, pid *uint32) (tid uint32, err error) = user32.GetWindowThreadProcessId
216+
//sys LoadKeyboardLayout(name *uint16, flags uint32) (hkl Handle, err error) [failretval==0] = user32.LoadKeyboardLayoutW
217+
//sys UnloadKeyboardLayout(hkl Handle) (err error) = user32.UnloadKeyboardLayout
218+
//sys GetKeyboardLayout(tid uint32) (hkl Handle) = user32.GetKeyboardLayout
219+
//sys ToUnicodeEx(vkey uint32, scancode uint32, keystate *byte, pwszBuff *uint16, cchBuff int32, flags uint32, hkl Handle) (ret int32) = user32.ToUnicodeEx
214220
//sys GetShellWindow() (shellWindow HWND) = user32.GetShellWindow
215221
//sys MessageBox(hwnd HWND, text *uint16, caption *uint16, boxtype uint32) (ret int32, err error) [failretval==0] = user32.MessageBoxW
216222
//sys ExitWindowsEx(flags uint32, reason uint32) (err error) = user32.ExitWindowsEx
@@ -1368,9 +1374,11 @@ func SetsockoptLinger(fd Handle, level, opt int, l *Linger) (err error) {
13681374
func SetsockoptInet4Addr(fd Handle, level, opt int, value [4]byte) (err error) {
13691375
return Setsockopt(fd, int32(level), int32(opt), (*byte)(unsafe.Pointer(&value[0])), 4)
13701376
}
1377+
13711378
func SetsockoptIPMreq(fd Handle, level, opt int, mreq *IPMreq) (err error) {
13721379
return Setsockopt(fd, int32(level), int32(opt), (*byte)(unsafe.Pointer(mreq)), int32(unsafe.Sizeof(*mreq)))
13731380
}
1381+
13741382
func SetsockoptIPv6Mreq(fd Handle, level, opt int, mreq *IPv6Mreq) (err error) {
13751383
return syscall.EWINDOWS
13761384
}

windows/syscall_windows_test.go

+47
Original file line numberDiff line numberDiff line change
@@ -1437,3 +1437,50 @@ uintptr_t beep(void) {
14371437
t.Fatal("LoadLibraryEx unexpectedly found beep.dll")
14381438
}
14391439
}
1440+
1441+
func TestGetKeyboardLayout(t *testing.T) {
1442+
fg := windows.GetForegroundWindow()
1443+
tid, err := windows.GetWindowThreadProcessId(fg, nil)
1444+
if err != nil {
1445+
t.Fatalf("GetWindowThreadProcessId failed: %v", err)
1446+
}
1447+
1448+
// We don't care about the result, just that it doesn't crash.
1449+
_ = windows.GetKeyboardLayout(tid)
1450+
}
1451+
1452+
func TestToUnicodeEx(t *testing.T) {
1453+
var utf16Buf [16]uint16
1454+
1455+
// Arabic (101) Keyboard Identifier
1456+
// See https://learn.microsoft.com/en-us/windows-hardware/manufacture/desktop/windows-language-pack-default-values
1457+
ara, err := windows.UTF16PtrFromString("00000401")
1458+
if err != nil {
1459+
t.Fatalf("UTF16PtrFromString failed: %v", err)
1460+
}
1461+
araLayout, err := windows.LoadKeyboardLayout(ara, windows.KLF_ACTIVATE)
1462+
if err != nil {
1463+
t.Fatalf("LoadKeyboardLayout failed: %v", err)
1464+
}
1465+
1466+
var keyState [256]byte
1467+
ret := windows.ToUnicodeEx(
1468+
0x41, // 'A' vkCode
1469+
0x1e, // 'A' scanCode
1470+
&keyState[0],
1471+
&utf16Buf[0],
1472+
int32(len(utf16Buf)),
1473+
0x4, // don't change keyboard state
1474+
araLayout,
1475+
)
1476+
1477+
if ret != 1 {
1478+
t.Errorf("ToUnicodeEx failed, wanted 1, got %d", ret)
1479+
}
1480+
if utf16Buf[0] != 'ش' {
1481+
t.Errorf("ToUnicodeEx failed, wanted 'ش', got %q", utf16Buf[0])
1482+
}
1483+
if err := windows.UnloadKeyboardLayout(araLayout); err != nil {
1484+
t.Errorf("UnloadKeyboardLayout failed: %v", err)
1485+
}
1486+
}

windows/types_windows.go

+11
Original file line numberDiff line numberDiff line change
@@ -3418,3 +3418,14 @@ type DCB struct {
34183418
EvtChar byte
34193419
wReserved1 uint16
34203420
}
3421+
3422+
// Keyboard Layout Flags.
3423+
// See https://learn.microsoft.com/en-us/windows/win32/api/winuser/nf-winuser-loadkeyboardlayoutw
3424+
const (
3425+
KLF_ACTIVATE = 0x00000001
3426+
KLF_SUBSTITUTE_OK = 0x00000002
3427+
KLF_REORDER = 0x00000008
3428+
KLF_REPLACELANG = 0x00000010
3429+
KLF_NOTELLSHELL = 0x00000080
3430+
KLF_SETFORPROCESS = 0x00000100
3431+
)

windows/zsyscall_windows.go

+33
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

0 commit comments

Comments
 (0)