Connect to a Wiimote
Fleetingusing bluez
Following the documentation of https://github.com/xwiimote/xwiimote/blob/master/doc/PROTOCOL and using https://github.com/TurpIF/PyWiimote as an example.
-
press 1 + 2 on the wiimote
-
find the device
import bluetooth return bluetooth.discover_devices()
['CC:9E:00:B9:7F:EF']
-
keep pressing 1 + 2
-
connect to the device and listen to some events once connected
from bluetooth import BluetoothSocket, L2CAP s = BluetoothSocket(L2CAP) s.connect(('CC:9E:00:B9:7F:EF', 0x13)) return [s.recv(16) for i in range(10)]
[b'\xa10\x00\x03', b'\xa10\x00\x02', b'\xa10\x00\x00', b'\xa10\x00\x08', b'\xa10\x00\x00', b'\xa10\x01\x00', b'\xa10\x00\x00', b'\xa10\x02\x00', b'\xa10\x00\x00', b'\xa10\x08\x00']
-
in the future, it helps first running the program, then pressing 1 and 2. The first received data will be about 1 and 2.
-
map the received data to pressed buttons
- relaxed state: b’\xa10\x00\x00'
- 1: b’\xa10\x00\x02'
- 2: b’\xa10\x00\x01'
- home: b’\xa10\x00\x80'
- -: b’\xa10\x00\x10'
- +: b’\xa10\x10\x00'
- A: b’\xa10\x00\x08'
- down: b’\xa10\x04\x00'
- up: b’\xa10\x08\x00'
- right: b’\xa10\x02\x00'
- left: b’\xa10\x01\x00'
- z: b’\xa10\x00\x04'
- unplug nunchuk: b’\xa1 \x00\x00\x00\x00\x00c'
- plug nunchuk: b’\xa1 \x00\x00\x02\x00\x00c'
Once the nunchuk is plugged, the device stops sending data.
To control the device
-
vibration
from bluetooth import BluetoothSocket, L2CAP import time s = BluetoothSocket(L2CAP) s.connect(('CC:9E:00:B9:7F:EF', 0x11)) s.send(''.join(map(chr, (0x52, 0x15, 0x01)))) time.sleep(1) s.send(''.join(map(chr, (0x52, 0x15, 0x00))))
receive b’\xa1 \x00\x00\x00\x00\x00b' b’\xa1 \x00\x00\x00\x00\x00a'
-
led number can be 0 (no led) or 1 to 4
from bluetooth import BluetoothSocket, L2CAP import time s = BluetoothSocket(L2CAP) s.connect(('CC:9E:00:B9:7F:EF', 0x11)) s.send(''.join(map(chr, (0x52, 0x11, int(number << 4)))))
Nothing received
-
ask to get only the buttons
from bluetooth import BluetoothSocket, L2CAP import time s = BluetoothSocket(L2CAP) s.connect(('CC:9E:00:B9:7F:EF', 0x11)) n = 1 s.send(''.join(map(chr, (0x52, 0x12, 0x00, 0x30))))
receive b’\xa10\x00\x00'
-
buttons + acceleration
from bluetooth import BluetoothSocket, L2CAP import time s = BluetoothSocket(L2CAP) s.connect(('CC:9E:00:B9:7F:EF', 0x11)) n = 1 s.send(''.join(map(chr, (0x52, 0x12, 0x00, 0x31))))
Receive a lot of acceleration data b’\xa11\x00@\x8a\x7f\x94' b’\xa11`@\x88\x7f\x94' b’\xa11\x00`\x87~\x97' b’\xa11`\x00\x85\x7f\x9b' b’\xa11 `\x84\x7f\x9c' b’\xa11@ \x82\x7f\x9c' b’\xa11 \x00\x81\x7f\x9b' b’\xa11 \x00\x80~\x9b'
in android
Something like this should do it in my a python runtime on android. Yet it seems that the support for low level bluetooth stuff in the android API was introduced very late (android 10) .
And even in android 11, the connection stalled in my few attempts to make it work.
def run():
from jnius import autoclass
from android.broadcast import BroadcastReceiver
from jnius import autoclass, cast
from android.broadcast import BroadcastReceiver
# Import necessary Java classes
BluetoothAdapter = autoclass('android.bluetooth.BluetoothAdapter')
BluetoothDevice = autoclass('android.bluetooth.BluetoothDevice')
BluetoothSocket = autoclass('android.bluetooth.BluetoothSocket')
UUID = autoclass('java.util.UUID')
# Get BluetoothAdapter and IntentFilter classes
BluetoothAdapter = autoclass('android.bluetooth.BluetoothAdapter')
IntentFilter = autoclass('android.content.IntentFilter')
# Get the default Bluetooth adapter
bluetooth_adapter = BluetoothAdapter.getDefaultAdapter()
# Define the target device's address and L2CAP port
l2cap_psm = 0x13 # L2CAP Protocol/Service Multiplexor (PSM) value
# Get the Bluetooth adapter and the remote device
bluetooth_adapter = BluetoothAdapter.getDefaultAdapter()
if not bluetooth_adapter:
raise Exception("Bluetooth is not supported on this device")
if not bluetooth_adapter.isEnabled():
raise Exception("Bluetooth is not enabled")
device_address = "CC:9E:00:B9:7F:EF"
device = bluetooth_adapter.getRemoteDevice(device_address)
print("Create an L2CAP socket")
socket = device.createInsecureL2capChannel(l2cap_psm)
print("connecting")
# Connect to the remote device
socket.connect()
print("Receive data")
try:
input_stream = socket.getInputStream()
for i in range(10):
buffer = bytearray(16)
num_bytes = input_stream.read(buffer)
print(buffer[:num_bytes])
finally:
socket.close()