gohook/hook.go
2021-09-26 15:09:05 -04:00

278 lines
6.0 KiB
Go

// Copyright 2016 The go-vgo Project Developers. See the COPYRIGHT
// file at the top-level directory of this distribution and at
// https://github.com/go-vgo/robotgo/blob/master/LICENSE
//
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
// option. This file may not be copied, modified, or distributed
// except according to those terms.
package hook
/*
#cgo darwin CFLAGS: -x objective-c -Wno-deprecated-declarations
#cgo darwin LDFLAGS: -framework Cocoa
#cgo linux CFLAGS:-I/usr/src
#cgo linux LDFLAGS: -L/usr/src -lX11 -lXtst
#cgo linux LDFLAGS: -lX11-xcb -lxcb -lxcb-xkb -lxkbcommon -lxkbcommon-x11
//#cgo windows LDFLAGS: -lgdi32 -luser32
#include "event/goEvent.h"
*/
import "C"
import (
"fmt"
"sync"
"time"
"unsafe"
)
const (
// Version get the gohook version
Version = "v0.31.2.113, Sierra Nevada!"
// HookEnabled honk enable status
HookEnabled = 1 // iota
HookDisabled = 2
KeyDown = 3
KeyHold = 4
KeyUp = 5
MouseUp = 6
MouseHold = 7
MouseDown = 8
MouseMove = 9
MouseDrag = 10
MouseWheel = 11
FakeEvent = 12
// Keychar could be v
CharUndefined = 0xFFFF
WheelUp = -1
WheelDown = 1
)
// Event Holds a system event
//
// If it's a Keyboard event the relevant fields are:
// Mask, Keycode, Rawcode, and Keychar,
// Keychar is probably what you want.
//
// If it's a Mouse event the relevant fields are:
// Button, Clicks, X, Y, Amount, Rotation and Direction
type Event struct {
Kind uint8 `json:"id"`
When time.Time
Mask uint16 `json:"mask"`
Reserved uint16 `json:"reserved"`
Keycode uint16 `json:"keycode"`
Rawcode uint16 `json:"rawcode"`
Keychar rune `json:"keychar"`
Button uint16 `json:"button"`
Clicks uint16 `json:"clicks"`
X int16 `json:"x"`
Y int16 `json:"y"`
Amount uint16 `json:"amount"`
Rotation int32 `json:"rotation"`
Direction uint8 `json:"direction"`
}
var (
ev = make(chan Event, 1024)
asyncon = false
lck sync.RWMutex
pressed = make(map[uint16]bool, 256)
used = []int{}
keys = map[int][]uint16{}
cbs = map[int]func(Event){}
events = map[uint8][]int{}
)
func allPressed(pressed map[uint16]bool, keys ...uint16) bool {
for _, i := range keys {
// fmt.Println(i)
if !pressed[i] {
return false
}
}
return true
}
// Register register gohook event
func Register(when uint8, cmds []string, cb func(Event)) {
key := len(used)
used = append(used, key)
tmp := []uint16{}
for _, v := range cmds {
tmp = append(tmp, Keycode[v])
}
keys[key] = tmp
cbs[key] = cb
events[when] = append(events[when], key)
// return
}
// Process return go hook process
func Process(evChan <-chan Event) (out chan bool) {
out = make(chan bool)
go func() {
for ev := range evChan {
if ev.Kind == KeyDown || ev.Kind == KeyHold {
pressed[ev.Keycode] = true
} else if ev.Kind == KeyUp {
pressed[ev.Keycode] = false
}
for _, v := range events[ev.Kind] {
if !asyncon {
break
}
if allPressed(pressed, keys[v]...) {
cbs[v](ev)
}
}
}
// fmt.Println("exiting after end (process)")
out <- true
}()
return
}
// String return formatted hook kind string
func (e Event) String() string {
switch e.Kind {
case HookEnabled:
return fmt.Sprintf("%v - Event: {Kind: HookEnabled}", e.When)
case HookDisabled:
return fmt.Sprintf("%v - Event: {Kind: HookDisabled}", e.When)
case KeyUp:
return fmt.Sprintf("%v - Event: {Kind: KeyUp, Rawcode: %v, Keychar: %v}",
e.When, e.Rawcode, e.Keychar)
case KeyHold:
return fmt.Sprintf(
"%v - Event: {Kind: KeyHold, Rawcode: %v, Keychar: %v}",
e.When, e.Rawcode, e.Keychar)
case KeyDown:
return fmt.Sprintf(
"%v - Event: {Kind: KeyDown, Rawcode: %v, Keychar: %v}",
e.When, e.Rawcode, e.Keychar)
case MouseUp:
return fmt.Sprintf(
"%v - Event: {Kind: MouseUp, Button: %v, X: %v, Y: %v, Clicks: %v}",
e.When, e.Button, e.X, e.Y, e.Clicks)
case MouseHold:
return fmt.Sprintf(
"%v - Event: {Kind: MouseHold, Button: %v, X: %v, Y: %v, Clicks: %v}",
e.When, e.Button, e.X, e.Y, e.Clicks)
case MouseDown:
return fmt.Sprintf(
"%v - Event: {Kind: MouseDown, Button: %v, X: %v, Y: %v, Clicks: %v}",
e.When, e.Button, e.X, e.Y, e.Clicks)
case MouseMove:
return fmt.Sprintf(
"%v - Event: {Kind: MouseMove, Button: %v, X: %v, Y: %v, Clicks: %v}",
e.When, e.Button, e.X, e.Y, e.Clicks)
case MouseDrag:
return fmt.Sprintf(
"%v - Event: {Kind: MouseDrag, Button: %v, X: %v, Y: %v, Clicks: %v}",
e.When, e.Button, e.X, e.Y, e.Clicks)
case MouseWheel:
return fmt.Sprintf(
"%v - Event: {Kind: MouseWheel, Amount: %v, Rotation: %v, Direction: %v}",
e.When, e.Amount, e.Rotation, e.Direction)
case FakeEvent:
return fmt.Sprintf("%v - Event: {Kind: FakeEvent}", e.When)
}
return "Unknown event, contact the mantainers."
}
// RawcodetoKeychar rawcode to keychar
func RawcodetoKeychar(r uint16) string {
lck.RLock()
defer lck.RUnlock()
return raw2key[r]
}
// KeychartoRawcode key char to rawcode
func KeychartoRawcode(kc string) uint16 {
return keytoraw[kc]
}
// Start adds global event hook to OS
// returns event channel
func Start() chan Event {
ev = make(chan Event, 1024)
go C.start_ev()
asyncon = true
go func() {
for {
if !asyncon {
return
}
C.pollEv()
time.Sleep(time.Millisecond * 50)
//todo: find smallest time that does not destroy the cpu utilization
}
}()
return ev
}
// End removes global event hook
func End() {
asyncon = false
C.endPoll()
C.stop_event()
time.Sleep(time.Millisecond * 10)
for len(ev) != 0 {
<-ev
}
close(ev)
pressed = make(map[uint16]bool, 256)
used = []int{}
keys = map[int][]uint16{}
cbs = map[int]func(Event){}
events = map[uint8][]int{}
}
// AddEvent add event listener
func AddEvent(key string) int {
cs := C.CString(key)
defer C.free(unsafe.Pointer(cs))
eve := C.add_event(cs)
geve := int(eve)
return geve
}
// StopEvent stop event listener
func StopEvent() {
C.stop_event()
}