A simple Kotlin library for planning 2D mobile robot paths and trajectories designed for FTC. Note: this project is in alpha and many of the APIs are incubating.
(Courtesy of FTC Team 8393, Detroit 2019 Ochoa F2)
-
Open your
ftc_app
project in Android Studio. -
Open up the Gradle file for the module you'd like to install in (probably
TeamCode/build.release.gradle
). -
Add
implementation 'com.acmerobotics.roadrunner:core:0.2.6'
to the end of thedependencies
block. -
(Android only) Although Road Runner only has a few dependencies, these may exceed the method reference limit imposed by some versions of Android. Fortunately, there are a few ways around this. For more information, see this article.
-
If you do not need to target API level 19 (i.e., in FTC, you don't need to use ZTE speeds), then just add
multiDexEnabled true
to thedefaultConfig
closure (for FTC, this is located insidebuild.common.gradle
). -
Otherwise, the next best solution is to enable Proguard (the pre Android 5.0 solution in the article is difficult to implement with the FTC SDK). To accomplish this, add the following lines to the
debug
andrelease
closures insidebuildTypes
(this is also located inbuild.common.gradle
for FTC):minifyEnabled true proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
Now download the
proguard-rules.pro
file from the quickstart and save it to your module folder (TeamCode
in the case of FTC). -
Finally, if the other solutions prove unworkable, you can download a slim jar from Releases. It lacks some of the features of the normal distribution, but it has fewer dependencies.
-
-
Sync the project (Android Studio should prompt you to do this).
-
You're finished.
Road Runner includes a simple GUI for generating trajectories from pose waypoints and constraints. You can download the latest version from the Releases tab (or build it with ./gradlew shadowJar
).
Road Runner also includes a simple IDEA/Android Studio plugin based upon the GUI. Here are some instructions for building and loading the plugin:
-
Download the latest plugin zip from Releases or build it with
./gradlew buildPlugin
. -
In Android Studio, navigate to Settings > Plugins.
-
Click the button that reads
Install plugin from disk...
. -
Select the zip archive from earlier.
-
Restart Android Studio to activate plugin changes.
-
Click on "Path Designer" on the right side of the editor and the tool window should appear.
The documentation for all releases are available online in the KDoc format (more accurately reflects the Kotlin source) and in the more traditional Javadoc format.
For a complete example of using Road Runner in FTC, see the quickstart.
Motion profiles are core to this library and effective control of robot actuators. The combination of feedforward and feedback control (for example, PID controllers) can greatly improve accuracy and reduce mechanical strain on your robot. For more information about motion profiles (and motion control in general), see the canonical talk by mentors from FRC teams 254 and 971.
Conceptually, motion profiles are time-based descriptions of your robot's kinematic state (in this case, its position, velocity, and acceleration) that obey your mechanism's constraints (maximum velocity and maximum acceleration for Road Runner). For example, here's a pre-planned motion profile with a maximum acceleration of 5 in/s^2 and a maximum velocity of 5 in/s over 10in:
As you can see, this profile is considered trapezoidal due to the shape of the velocity curve over time. All it takes to generate a profile like this in Road Runner is a single method call:
MotionProfile profile = MotionProfileGenerator.generateSimpleMotionProfile(
new MotionState(0, 0, 0), // start state
new MotionState(10, 0, 0), // goal state
5, // max vel
5 // max accel
);
or
val profile = MotionProfileGenerator.generateSimpleMotionProfile(
MotionState(0.0, 0.0, 0.0),
MotionState(10.0, 0.0, 0.0),
5.0, 5.0
)
The profile
object can then be used to find the motion state at any point in time.
While one-dimensional profiles are useful for mechanisms with one degree of freedom (for example, elevators, arms, and turrets), they aren't particularly useful by themselves for drive movement. To take full advantage of your drive's capabilities, you must also define a 2D path for it to follow.
For now, lines and quintic splines are the two built-in path constructs. However, any sufficiently continuous parametric curve can work. Here's the syntax for creating a line:
Path line = new Path(new LineSegment(
new Vector2d(10, 10), // start position
new Vector2d(20, 20) // end position
));
or
val line = Path(LineSegment(Vector2d(10.0, 10.0), Vector2d(20.0, 20.0)))
Similarly, here's a spline declaration:
Path spline = new Path(new QuinticSplineSegment(
new QuinticSplineSegment.Waypoint(10, 10, 10, 0), // start position and derivatives
new QuinticSplineSegment.Waypoint(20, 20, 0, 10) // end position and derivatives
));
or
val spline = Path(QuinticSplineSegment(
QuinticSplineSegment.Waypoint(10.0, 10.0, 10.0, 0.0),
QuinticSplineSegment.Waypoint(20.0, 20.0, 0.0, 10.0)
))
Paths also describe the robot's heading in addition to its xy position. For tank/differential drive robots (and other with nonholonomic constraints), the heading always points in the direction of the curve. However, for holonomic robots (for example, mecanum, X-drive, H-drive, or ball drive), the heading is independent (and therefore controllable). To change the default heading behavior, simply pass a heading interpolator as the second argument to the Path
constructor:
Path constantHeadingSpline = new Path(new QuinticSplineSegment(
new QuinticSplineSegment.Waypoint(10, 10, 10, 0), // start position and derivatives
new QuinticSplineSegment.Waypoint(20, 20, 0, 10) // end position and derivatives
), new ConstantInterpolator(Math.toRadians(45)));
or
val spline = Path(QuinticSplineSegment(
QuinticSplineSegment.Waypoint(10.0, 10.0, 10.0, 0.0),
QuinticSplineSegment.Waypoint(20.0, 20.0, 0.0, 10.0)
), ConstantInterpolator(Math.toRadians(45.0)))
The ConstantInterpolator
commands the robot to maintain the provided heading throughout the motion, regardless of the translational movement (this is most often used for strafing). Road runner also ships with a linear heading interpolator, a spline heading interpolator, and a "wiggle" heading interpolator. As with the path constructs, the user is free to create interpolators representing any function.
Now it's time to combine path and motion profiles. To distinguish these new entities from normal 1D motion profiles, Road Runner calls them trajectories.
Here's an example of creating a Trajectory
from a Path
:
Trajectory trajectory = new Trajectory(Arrays.asList(
new PathTrajectorySegment(path, constraints)
));
or
val trajectory = Trajectory(listOf(PathTrajectorySegment(path, constraints)))
Here constraints
is an extended version of the motion profile maximum velocity and acceleration (see DriveConstraints
). There are also special trajectory segments that aren't representable as time-independent parametric curves. For example, here's the syntax for a point turn:
Trajectory trajectory = new Trajectory(Arrays.asList(
new PointTurn(
new Pose2d(0, 0, 0), // start pose (position and heading)
Math.toRadians(90), // end heading
constraints // drive constraints
)
));
or
val trajectory = Trajectory(listOf(
PointTurn(Pose2d(0.0, 0.0, 0.0), Math.toRadians(90.0), constraints)
))
Multiple path and trajectory segments can be composed into a single trajectory. Keep in mind that each trajectory segment has its own profile. Additionally, when combining multiple paths into the same PathTrajectorySegment
, make sure that the heading interpolators don't violate the continuity (see HeadingInterpolator.respectsDerivativeContinuity()
).
Finally, TrajectoryBuilder
can be used to create trajectories in a more concise manner:
Trajectory trajectory = new TrajectoryBuilder(new Pose2d(0, 0, 0), constraints)
.splineTo(new Pose2d(30, 30, 0))
.turnTo(Math.toRadians(90))
.build()
or
val trajectory = TrajectoryBuilder(Pose2d(0.0, 0.0, 0.0), constraints)
.splineTo(Pose2d(30.0, 30.0, 0.0))
.turnTo(Math.toRadians(90.0))
.build()
Finally, we're ready to select a follower. The role of the follower is to ensure the robot accurately tracks the trajectory/path with feedback. For mecanum and swerve, it's easy: there's only a PIDVA follower. However, for tank, there are a few more options:
-
TankPIDVAFollower
: PID-based controller that minimizes displacement and cross track error. Suitable for most applications. -
RamseteFollower
: Non-linear, time-varying controller with better performance characteristics than the PIDVA follower. Unfortunately, it's more difficult to tune. -
GVFFollower
: State-of-the-art guiding vector field controller with very good performance characteristics. However, unlike the others, it tracks paths directly instead of trajectories (and therefore doesn't respect constraints as well as the others). Additionally, it's moderately more computationally intensive (thanks to some nonlinear projection).