213 строки
4,9 КиБ
Go
213 строки
4,9 КиБ
Go
package joystick
|
|
|
|
import "encoding/binary"
|
|
|
|
type HatDirection uint8
|
|
|
|
const (
|
|
HatUp HatDirection = iota
|
|
HatRightUp
|
|
HatRight
|
|
HatRightDown
|
|
HatDown
|
|
HatLeftDown
|
|
HatLeft
|
|
HatLeftUp
|
|
HatCenter
|
|
)
|
|
|
|
type Constraint struct {
|
|
MinIn int
|
|
MaxIn int
|
|
MinOut int16
|
|
MaxOut int16
|
|
}
|
|
|
|
type AxisValue struct {
|
|
constraint Constraint
|
|
Value int
|
|
}
|
|
|
|
func fit(x, in_min, in_max int, out_min, out_max int16) int16 {
|
|
return int16((x-in_min)*(int(out_max)-int(out_min))/(in_max-in_min) + int(out_min))
|
|
}
|
|
|
|
func fitf(x, in_min, in_max, out_min, out_max float32) float32 {
|
|
return x - in_min*(out_max-out_min)/(in_max-in_min) + out_min
|
|
}
|
|
|
|
func limit(v, max int) int {
|
|
if v > max {
|
|
v = max
|
|
} else if v < -max {
|
|
v = -max
|
|
}
|
|
return v
|
|
}
|
|
|
|
type Definitions struct {
|
|
ReportID byte
|
|
ButtonCnt int
|
|
HatSwitchCnt int
|
|
AxisDefs []Constraint
|
|
descriptor []byte
|
|
}
|
|
|
|
func (c Definitions) Descriptor() []byte {
|
|
if len(c.descriptor) > 0 {
|
|
return c.descriptor
|
|
}
|
|
// TODO: build hid descriptor
|
|
return nil
|
|
}
|
|
|
|
func (c Definitions) NewState() State {
|
|
bufSize := 1
|
|
axises := make([]*AxisValue, 0, len(c.AxisDefs))
|
|
for _, v := range c.AxisDefs {
|
|
|
|
axises = append(axises, &AxisValue{
|
|
constraint: v,
|
|
Value: 0,
|
|
})
|
|
}
|
|
btnSize := (c.ButtonCnt + 7) / 8
|
|
bufSize += btnSize
|
|
if c.HatSwitchCnt > 0 {
|
|
bufSize++
|
|
}
|
|
bufSize += len(axises) * 2
|
|
initBuf := make([]byte, bufSize)
|
|
initBuf[0] = c.ReportID
|
|
return State{
|
|
buf: initBuf,
|
|
Buttons: make([]byte, btnSize),
|
|
HatSwitches: make([]HatDirection, c.HatSwitchCnt),
|
|
Axises: axises,
|
|
}
|
|
}
|
|
|
|
type State struct {
|
|
buf []byte
|
|
Buttons []byte
|
|
HatSwitches []HatDirection
|
|
Axises []*AxisValue
|
|
}
|
|
|
|
func (s State) MarshalBinary() ([]byte, error) {
|
|
s.buf = s.buf[0:1]
|
|
s.buf = append(s.buf, s.Buttons...)
|
|
if len(s.HatSwitches) > 0 {
|
|
hat := byte(0)
|
|
for _, v := range s.HatSwitches {
|
|
hat <<= 4
|
|
hat |= byte(v & 0xf)
|
|
}
|
|
s.buf = append(s.buf, hat)
|
|
}
|
|
for _, v := range s.Axises {
|
|
c := v.constraint
|
|
val := fit(v.Value, c.MinIn, c.MaxIn, c.MinOut, c.MaxOut)
|
|
s.buf = binary.LittleEndian.AppendUint16(s.buf, uint16(val))
|
|
}
|
|
return s.buf, nil
|
|
}
|
|
|
|
func (s State) Button(index int) bool {
|
|
idx := index / 8
|
|
bit := uint8(1 << (index % 8))
|
|
return s.Buttons[idx]&bit > 0
|
|
}
|
|
|
|
func (s State) SetButton(index int, push bool) {
|
|
idx := index / 8
|
|
bit := uint8(1 << (index % 8))
|
|
b := s.Buttons[idx]
|
|
b &= ^bit
|
|
if push {
|
|
b |= bit
|
|
}
|
|
s.Buttons[idx] = b
|
|
}
|
|
|
|
func (s State) Hat(index int) HatDirection {
|
|
return s.HatSwitches[index]
|
|
}
|
|
|
|
func (s State) SetHat(index int, dir HatDirection) {
|
|
s.HatSwitches[index] = dir
|
|
}
|
|
|
|
func (s State) Axis(index int) int {
|
|
return s.Axises[index].Value
|
|
}
|
|
|
|
func (s State) SetAxis(index int, v int) {
|
|
s.Axises[index].Value = v
|
|
}
|
|
|
|
func DefaultDefinitions() Definitions {
|
|
return Definitions{
|
|
ReportID: 1,
|
|
ButtonCnt: 16,
|
|
HatSwitchCnt: 1,
|
|
AxisDefs: []Constraint{
|
|
{MinIn: -32767, MaxIn: 32767, MinOut: -32767, MaxOut: 32767},
|
|
{MinIn: -32767, MaxIn: 32767, MinOut: -32767, MaxOut: 32767},
|
|
{MinIn: -32767, MaxIn: 32767, MinOut: -32767, MaxOut: 32767},
|
|
{MinIn: -32767, MaxIn: 32767, MinOut: -32767, MaxOut: 32767},
|
|
{MinIn: -32767, MaxIn: 32767, MinOut: -32767, MaxOut: 32767},
|
|
{MinIn: -32767, MaxIn: 32767, MinOut: -32767, MaxOut: 32767},
|
|
},
|
|
descriptor: []byte{
|
|
0x05, 0x01,
|
|
0x09, 0x04,
|
|
0xa1, 0x01, // COLLECTION (Application)
|
|
0x85, 0x01, // REPORT_ID (1)
|
|
0x05, 0x09, // USAGE_PAGE (Button)
|
|
0x19, 0x01, // USAGE_MINIMUM (Button 1)
|
|
0x29, 0x10, // USAGE_MAXIMUM (Button 16)
|
|
0x15, 0x00, // LOGICAL_MINIMUM (0)
|
|
0x25, 0x01, // LOGICAL_MAXIMUM (1)
|
|
0x75, 0x01, // REPORT_SIZE (1)
|
|
0x95, 0x10, // REPORT_COUNT (16)
|
|
0x55, 0x00, // Unit Exponent (-16)
|
|
0x65, 0x00, // Unit (0x00)
|
|
0x81, 0x02, // INPUT (Data/Var/Abs)
|
|
0x05, 0x01, // USAGE_PAGE (Generic Desktop Controls)
|
|
0x09, 0x39, // USAGE(Hat Switch)
|
|
0x15, 0x00, // LOGICAL_MINIMUM (0)
|
|
0x25, 0x07, // LOGICAL_MAXIMUM (7)
|
|
0x35, 0x00, // PHYSICAL_MINIMUM (0)
|
|
0x46, 0x3b, 0x01, // PHYSICAL_MAXIMUM(315)
|
|
0x65, 0x14, // UNIT (Eng Rot:Angular Pos)
|
|
0x75, 0x04, // REPORT_SIZE (4)
|
|
0x95, 0x01, // REPORT_COUNT (1)
|
|
0x81, 0x02, // INPUT (Data/Var/Abs)
|
|
0x09, 0x39, // USAGE(Hat Switch)
|
|
0x15, 0x00, // LOGICAL_MINIMUM (0)
|
|
0x25, 0x07, // LOGICAL_MAXIMUM (7)
|
|
0x35, 0x00, // PHYSICAL_MINIMUM (0)
|
|
0x46, 0x3b, 0x01, // PHYSICAL_MAXIMUM(315)
|
|
0x65, 0x14, // UNIT (Eng Rot:Angular Pos)
|
|
0x75, 0x04, // REPORT_SIZE (4)
|
|
0x95, 0x01, // REPORT_COUNT (1)
|
|
0x81, 0x02, // INPUT (Data/Var/Abs)
|
|
0x09, 0x01, // USAGE (Pointer)
|
|
0x16, 0x01, 0x80, // LOGICAL_MINIMUM (-32767)
|
|
0x26, 0xff, 0x7f, // LOGICAL_MAXIMUM (32767)
|
|
0x75, 0x10, // REPORT_SIZE (16bits)
|
|
0x95, 0x06, // REPORT_COUNT (6)
|
|
0xa1, 0x00, // COLLECTION (Physical)
|
|
0x09, 0x30, // USAGE(X)
|
|
0x09, 0x31, // USAGE(Y)
|
|
0x09, 0x32, // USAGE(Z)
|
|
0x09, 0x33, // USAGE(RX)
|
|
0x09, 0x34, // USAGE(RY)
|
|
0x09, 0x35, // USAGE(RZ)
|
|
0x81, 0x02, // INPUT (Data/Var/Abs)
|
|
0xc0, // END_COLLECTION
|
|
0xc0, // END_COLLECTION
|
|
},
|
|
}
|
|
}
|