XBox One Wireless control of Servo using Analogue Stick withevdev

1k Views Asked by At

I'm using Evdev to read the XBOX ONE inputs of A, B, X, Y to great success. However, I am battling to pick up the Analog Stick inputs. Can anyone help me with the Python code for this?

I am trying to control a Servo.

This is my code so far, it works "perfectly". I need to know how to read the outupts of the (xbox)joystick so i can us "GPIO.PWM" for the servo.

    from evdev import InputDevice, categorize, ecodes, KeyEvent
    import RPi.GPIO as GPIO
    from time import sleep

    GPIO.setmode(GPIO.BOARD)
    GPIO.setup(8, GPIO.OUT, initial=GPIO.LOW)
    GPIO.setup(7, GPIO.OUT, initial=GPIO.LOW)
    GPIO.setup(11, GPIO.OUT, initial=GPIO.LOW)
    GPIO.setup(12, GPIO.OUT, initial=GPIO.LOW)


    gamepad = InputDevice('/dev/input/event2')


    #evdev takes care of polling the controller in a loop
    for event in gamepad.read_loop():
    if event.type == ecodes.EV_KEY:
    keyevent = categorize(event)
    if keyevent.keystate == KeyEvent.key_down:
        if keyevent.keycode[0] == "BTN_A":
            print "Button A Pressed"
            GPIO.output (8, GPIO.HIGH)
        elif keyevent.keycode[0] == "BTN_B":
            print "Button B Pressed"
            GPIO.output (7, GPIO.HIGH)
        elif keyevent.keycode[0] == "BTN_WEST":
            print "Button Y Pressed"
            GPIO.output (11, GPIO.HIGH)
        elif keyevent.keycode[0] == "BTN_NORTH":
            print "Button X Pressed"
            GPIO.output (12, GPIO.HIGH)
    if keyevent.keystate == KeyEvent.key_up:
        if keyevent.keycode[0] == "BTN_A":
            print "Button A Released"
            GPIO.output (8, GPIO.LOW)
        elif keyevent.keycode[0] == "BTN_B":
            print "Button B Released"
            GPIO.output (7, GPIO.LOW)
        elif keyevent.keycode[0] == "BTN_WEST":
            print "Button Y Released"
            GPIO.output (11, GPIO.LOW)
        elif keyevent.keycode[0] == "BTN_NORTH":
            print "Button X Released"
            GPIO.output (12, GPIO.LOW)
1

There are 1 best solutions below

0
On

I am working on a similar project to control a tank-style robot with an XBox One Wireless controller, and I've mapped out the absolute axis, with an example of the left stick moving foreward/backward, and the right stick turning.

Python is not my favorite language, and there is probably a better method of calibrating, but I slapped this together. Hope it helps.

from evdev import list_devices, InputDevice, categorize, ecodes

CENTER_TOLERANCE = 350
STICK_MAX = 65536

dev = InputDevice( list_devices()[0] )
axis = {
    ecodes.ABS_X: 'ls_x', # 0 - 65,536   the middle is 32768
    ecodes.ABS_Y: 'ls_y',
    ecodes.ABS_Z: 'rs_x',
    ecodes.ABS_RZ: 'rs_y',
    ecodes.ABS_BRAKE: 'lt', # 0 - 1023
    ecodes.ABS_GAS: 'rt',

    ecodes.ABS_HAT0X: 'dpad_x', # -1 - 1
    ecodes.ABS_HAT0Y: 'dpad_y'
}

center = {
    'ls_x': STICK_MAX/2,
    'ls_y': STICK_MAX/2,
    'rs_x': STICK_MAX/2,
    'rs_y': STICK_MAX/2
}

last = {
    'ls_x': STICK_MAX/2,
    'ls_y': STICK_MAX/2,
    'rs_x': STICK_MAX/2,
    'rs_y': STICK_MAX/2
}

for event in dev.read_loop():

    # calibrate zero on Y button
    if event.type == ecodes.EV_KEY:
        if categorize(event).keycode[0] == "BTN_WEST":
            center['ls_x'] = last['ls_x']
            center['ls_y'] = last['ls_y']
            center['rs_x'] = last['rs_x']
            center['rs_y'] = last['rs_y']
            print( 'calibrated' )

    #read stick axis movement
    elif event.type == ecodes.EV_ABS:
        if axis[ event.code ] in [ 'ls_x', 'ls_y', 'rs_x', 'rs_y' ]:
            last[ axis[ event.code ] ] = event.value

            value = event.value - center[ axis[ event.code ] ]

            if abs( value ) <= CENTER_TOLERANCE:
                value = 0

            if axis[ event.code ] == 'rs_x':
                if value < 0:
                    print('left')
                else:
                    print('right')
                print( value )

            elif axis[ event.code ] == 'ls_y':
                if value < 0:
                    print('foreward')
                else:
                    print('backward')
                print( value )