wm/vend/xgbutil/mousebind/xutil.go

154 lines
4.6 KiB
Go

package mousebind
/*
mousebind/xutil.go contains a collection of functions that modify the
Mousebinds and Mousegrabs state of an XUtil value.
They could have been placed inside the core xgbutil package, but they would
have to be exported for use by the mousebind package. In which case, the API
would become cluttered with functions that should not be used.
*/
import (
"github.com/jezek/xgb/xproto"
"github.com/jezek/xgbutil"
)
// attachMouseBindCallback associates an (event, window, mods, button)
// with a callback.
func attachMouseBindCallback(xu *xgbutil.XUtil, evtype int, win xproto.Window,
mods uint16, button xproto.Button, fun xgbutil.CallbackMouse) {
xu.MousebindsLck.Lock()
defer xu.MousebindsLck.Unlock()
// Create key
key := xgbutil.MouseKey{evtype, win, mods, button}
// Do we need to allocate?
if _, ok := xu.Mousebinds[key]; !ok {
xu.Mousebinds[key] = make([]xgbutil.CallbackMouse, 0)
}
xu.Mousebinds[key] = append(xu.Mousebinds[key], fun)
xu.Mousegrabs[key] += 1
}
// mouseKeys returns a copy of all the keys in the 'Mousebinds' map.
func mouseKeys(xu *xgbutil.XUtil) []xgbutil.MouseKey {
xu.MousebindsLck.RLock()
defer xu.MousebindsLck.RUnlock()
keys := make([]xgbutil.MouseKey, len(xu.Mousebinds))
i := 0
for key, _ := range xu.Mousebinds {
keys[i] = key
i++
}
return keys
}
// mouseBindCallbacks returns a slice of callbacks for a particular key.
func mouseCallbacks(xu *xgbutil.XUtil,
key xgbutil.MouseKey) []xgbutil.CallbackMouse {
xu.MousebindsLck.RLock()
defer xu.MousebindsLck.RUnlock()
cbs := make([]xgbutil.CallbackMouse, len(xu.Mousebinds[key]))
copy(cbs, xu.Mousebinds[key])
return cbs
}
// runMouseBindCallbacks executes every callback corresponding to a
// particular event/window/mod/button tuple.
func runMouseBindCallbacks(xu *xgbutil.XUtil, event interface{}, evtype int,
win xproto.Window, mods uint16, button xproto.Button) {
key := xgbutil.MouseKey{evtype, win, mods, button}
for _, cb := range mouseCallbacks(xu, key) {
cb.Run(xu, event)
}
}
// connectedMouseBind checks to see if there are any key binds for a particular
// event type already in play. This is to work around comparing function
// pointers (not allowed in Go), which would be used in 'Connected'.
func connectedMouseBind(xu *xgbutil.XUtil, evtype int, win xproto.Window) bool {
xu.MousebindsLck.RLock()
defer xu.MousebindsLck.RUnlock()
// Since we can't create a full key, loop through all mouse binds
// and check if evtype and window match.
for key, _ := range xu.Mousebinds {
if key.Evtype == evtype && key.Win == win {
return true
}
}
return false
}
// detachMouseBindWindow removes all callbacks associated with a particular
// window and event type (either ButtonPress or ButtonRelease)
// Also decrements the counter in the corresponding 'Mousegrabs' map
// appropriately.
func detachMouseBindWindow(xu *xgbutil.XUtil, evtype int, win xproto.Window) {
xu.MousebindsLck.Lock()
defer xu.MousebindsLck.Unlock()
// Since we can't create a full key, loop through all mouse binds
// and check if evtype and window match.
for key, _ := range xu.Mousebinds {
if key.Evtype == evtype && key.Win == win {
xu.Mousegrabs[key] -= len(xu.Mousebinds[key])
delete(xu.Mousebinds, key)
}
}
}
// mouseBindGrabs returns the number of grabs on a particular
// event/window/mods/button combination. Namely, this combination
// uniquely identifies a grab. If it's repeated, we get BadAccess.
func mouseBindGrabs(xu *xgbutil.XUtil, evtype int, win xproto.Window,
mods uint16, button xproto.Button) int {
xu.MousebindsLck.RLock()
defer xu.MousebindsLck.RUnlock()
key := xgbutil.MouseKey{evtype, win, mods, button}
return xu.Mousegrabs[key] // returns 0 if key does not exist
}
// mouseDrag true when a mouse drag is in progress.
func mouseDrag(xu *xgbutil.XUtil) bool {
return xu.InMouseDrag
}
// mouseDragSet sets whether a mouse drag is in progress.
func mouseDragSet(xu *xgbutil.XUtil, dragging bool) {
xu.InMouseDrag = dragging
}
// mouseDragStep returns the function currently associated with each
// step of a mouse drag.
func mouseDragStep(xu *xgbutil.XUtil) xgbutil.MouseDragFun {
return xu.MouseDragStepFun
}
// mouseDragStepSet sets the function associated with the step of a drag.
func mouseDragStepSet(xu *xgbutil.XUtil, f xgbutil.MouseDragFun) {
xu.MouseDragStepFun = f
}
// mouseDragEnd returns the function currently associated with the
// end of a mouse drag.
func mouseDragEnd(xu *xgbutil.XUtil) xgbutil.MouseDragFun {
return xu.MouseDragEndFun
}
// mouseDragEndSet sets the function associated with the end of a drag.
func mouseDragEndSet(xu *xgbutil.XUtil, f xgbutil.MouseDragFun) {
xu.MouseDragEndFun = f
}