A MicroPython motion-control library for a differential-drive robot built on the LEGO SPIKE Prime hub, using Pybricks. Hyperlib provides odometry, PID-based control, and a full suite of movement functions for autonomous navigation.
Odometry is not magic; it cannot pinpoint exactly where your robot tracks every single minute movement of both wheels. It is just an approximation, and the error keeps increasing as the robot travels. Therefore, odometry in the World Robot Olympiad program is just a method to cut corners, to travel to approximate positions. If you want exact positions, you must plan waypoints (walls, black line to follow, colors on the map,...) which the robot can track using other kinds of sensors
These functions can NEVER make the robot arrive at the desired position/heading with 100% accuracy. The idea behind these functions is that the robot keeps steering to its target to MINIMIZE the error between the robot's and the desired position/heading, not completely EFFACING it. Thus, if you want the robot to be 100% on the target, it will run forever and can never complete this task. Instead, as the robot approaches an acceptable margin of error, it will stop. But don't worry! Since odometry tracking is always on, the robot will ALWAYS be able to approximate where it is and where it isn't.
WE ENCOURAGE YOU TO ADD FUNCTIONS OF YOUR OWN TO THE PROGRAM! Many functions may be essential to you that we have not built yet! So, don't hesitate to study the method and programming style used to build these functions to construct your own! Also, you must firmly understand the MATHEMATICAL PROOF and LOGIC under the algorithms and functions to build your own programmes. Contact the author (Lam) if you have any question to ask!
βββ Versions/
β βββ Hyperlib_V1_4/
β βββ Hyperlib_V1_5/
β βββ Hyperlib_V1_6/
β βββ Hyperlib_v1_6.py β Main library
β βββ main_program.py β Your program goes here
βββ Hyperlib V1.3.py
βββ README.md
- Place
Hyperlib_v1_5.pyandmain_program.pyin the same directory on your SPIKE Prime hub. - Write your autonomous routine inside
main_program.py:
from Hyperlib_v1_5 import *
async def main():
await OdomIni(0, 0, 0) # Start at origin, facing 0Β°
await SwingMM(2, -300) # Swing turn 300 mm on right wheel
await TurnGyro(0) # Turn back to face 0Β°
run_task(main())| Parameter | Value | Description |
|---|---|---|
| Left Motor | Port A | Reversed (CounterClockwise) |
| Right Motor | Port E | Standard |
WheelD |
based on your robot | Wheel diameter |
Axle |
based on your robot | Distance between wheels |
LoopTime |
10 ms | Control loop update frequency |
MinSpeed |
input your motors' min dps (Β°/s) | Minimum motor speed |
MaxSpeed |
input your motors' max dps (Β°/s) | Maximum motor speed |
MaxAcc |
input your motors' max acceleration (Β°/sΒ²) | Maximum acceleration |
| Constant | Value | Purpose |
|---|---|---|
GKp |
20 | Proportional gain for gyro heading correction |
GKd |
250 | Derivative gain for gyro heading correction |
ENCKp |
1 | Proportional gain for encoder straightening |
ENCKd |
10 | Derivative gain for encoder straightening |
DistanceKp |
4 | Proportional gain for distance-based speed |
GyroSingleTurnKp |
20 | Gain for gyro-based swing turns |
ENCSingleTurnKp |
12 | Gain for encoder-based swing turns |
DoubleTurnKp |
8 | Gain for two-motor point turns |
Set OdometrySwitch = False to disable automatic odometry updates inside movement functions.
Initializes the robot's position and heading. Resets the IMU to theta, sets coordinates to (x, y), and zeroes motor encoders.
Updates (odomX, odomY) using arc odometry β accurate for both straight paths and curves. Uses wheel encoder deltas combined with IMU heading. Returns (odomX, odomY, current_theta).
Moves straight for a set distance (mm) using only wheel encoders for straightness correction (PD-controller on left/right encoder difference). No gyro required.
Moves straight for a set distance (mm) while maintaining a target heading using the IMU gyro (PD-controller on heading error).
Point turn using both motors to reach a target heading via the IMU.
Thetaβ target heading in degreesEarlyExitβ acceptable heading error to stop (default5Β°)Forwardβ face forward (True) or backward (False)Directionβ0= shortest path,1= force clockwise,2= force counterclockwise
Point turn using both motors, controlled purely by wheel encoders. Each wheel travels turn_mm in opposite directions. Positive = clockwise.
Turns in place to face a target coordinate (TargetX, TargetY) using odometry and the IMU.
Pivot turn using one motor while the other is held, controlled by the IMU gyro.
Side = 1β left motor moves, right heldSideβ 1 β right motor moves, left heldThetaβ target heading in degrees
Pivot turn using one motor while the other is held, controlled by the moving wheel's encoder.
turn_mmβ distance the moving wheel travels (mm). Positive = clockwise.
Swing turn to face a specific field coordinate using one motor and the IMU gyro.
Drives to a target coordinate (targetX, targetY) using a P-controller for distance-based speed and a PD-controller for heading correction. Stops within ArrivalThreshold mm of the target.
MoveToPoRo(targetX, targetY, targetAngle, offsetDistance, finalDistance, offsetAngle, MaxSpeed, MinSpeed, ArrivalThreshold, minRatio, Forward, DKp, Stopping)
Drives to a coordinate while controlling the final approach angle (Position + Orientation). Places a virtual waypoint behind the target and smoothly blends the robot's heading toward targetAngle as it gets closer. Ideal for docking or alignment tasks.
| Parameter | Default | Description |
|---|---|---|
targetAngle |
β | Desired heading when arriving (degrees) |
offsetDistance |
250 mm | Distance behind target for virtual waypoint |
offsetAngle |
20Β° | Angular offset of virtual waypoint |
minRatio |
0.8 | Minimum blend ratio during final approach |
ArrivalThreshold |
25 mm | Distance considered as "arrived" |
Brake both motors for 40 ms, then hold them in place.
Converts motor encoder degrees to millimeters of travel using the configured wheel diameter.
Returns the signed heading error in degrees (range β180Β° to 180Β°) between the current IMU heading and a target angle.
β οΈ First run: Before running any program, update your robot's physical specs at the top ofHyperlib_v1_5.pyto match your actual hardware:Incorrect values will cause all distance and odometry calculations to be wrong.WheelD = 62 # β your wheel diameter in mm Axle = 170 # β your axle width in mm (wheel center to wheel center)
- All movement functions are
asyncand must be called withawait, or composed using Pybricks'multitask/run_task. - The Bluetooth button is configured as the stop button to prevent accidental program termination during a run.
- For debugging, use the
view()function inmain_program.pyto continuously print the current IMU heading to the console. - Hyperlib uses arc odometry in
OdomUpdatefor accurate position tracking on both straight and curved paths.
| Version | File | Notes |
|---|---|---|
| v1.5 | Hyperlib_v1_5.py |
Added arc odometry, encoder-based turns (TurnMM, SwingMM), improved PID gains |
| v1.4 | Hyperlib_V1_4/ |
Previous stable version |
| v1.3 | Hyperlib V1.3.py |
Initial release |
Odometry is a method to estimate the robot's position. There are two main odometry methods: linear odometry and arc odometry. During straight movements, linear odometry is best suited for use, whereas during curved moment, arc odometry is the optimal method
Here is the mathematical proof of it, you can try to look for symmetries in these graph and the code to better understand odometry:
Mr. Phong and his colleagues have been working with us since day one of robotics, and I want to dedicate a section of this library to honor their contributions to building our knowledge and curiosity, which have allowed us to build this library.
π€author: @Lamcao2109 π₯contributors: @DuCpHancm



