2026-02-20 18:30:35 -05:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
|
#
|
|
|
|
|
# Copyright (c) FIRST and other WPILib contributors.
|
|
|
|
|
# Open Source Software; you can modify and/or share it under the terms of
|
|
|
|
|
# the WPILib BSD license file in the root directory of this project.
|
|
|
|
|
#
|
|
|
|
|
|
|
|
|
|
import wpilib
|
|
|
|
|
import wpimath
|
|
|
|
|
|
|
|
|
|
from drivetrain import Drivetrain
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class MyRobot(wpilib.TimedRobot):
|
|
|
|
|
def __init__(self) -> None:
|
|
|
|
|
super().__init__()
|
|
|
|
|
self.controller = wpilib.NiDsXboxController(0)
|
|
|
|
|
|
|
|
|
|
# Slew rate limiters to make joystick inputs more gentle; 1/3 sec from 0
|
|
|
|
|
# to 1.
|
2026-03-06 14:19:15 -08:00
|
|
|
self.velocityLimiter = wpimath.SlewRateLimiter(3)
|
2026-02-20 18:30:35 -05:00
|
|
|
self.rotLimiter = wpimath.SlewRateLimiter(3)
|
|
|
|
|
|
|
|
|
|
self.drive = Drivetrain()
|
|
|
|
|
self.feedback = wpimath.LTVUnicycleController(0.020)
|
|
|
|
|
self.timer = wpilib.Timer()
|
|
|
|
|
|
|
|
|
|
# Called once at the beginning of the robot program.
|
|
|
|
|
self.trajectory = wpimath.TrajectoryGenerator.generateTrajectory(
|
|
|
|
|
wpimath.Pose2d(2, 2, wpimath.Rotation2d()),
|
|
|
|
|
[],
|
|
|
|
|
wpimath.Pose2d(6, 4, wpimath.Rotation2d()),
|
|
|
|
|
wpimath.TrajectoryConfig(2, 2),
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
def robotPeriodic(self) -> None:
|
|
|
|
|
self.drive.periodic()
|
|
|
|
|
|
|
|
|
|
def autonomousInit(self) -> None:
|
|
|
|
|
self.timer.restart()
|
|
|
|
|
self.drive.resetOdometry(self.trajectory.initialPose())
|
|
|
|
|
|
|
|
|
|
def autonomousPeriodic(self) -> None:
|
|
|
|
|
elapsed = self.timer.get()
|
|
|
|
|
reference = self.trajectory.sample(elapsed)
|
2026-03-06 14:19:15 -08:00
|
|
|
velocities = self.feedback.calculate(self.drive.getPose(), reference)
|
|
|
|
|
self.drive.drive(velocities.vx, velocities.omega)
|
2026-02-20 18:30:35 -05:00
|
|
|
|
|
|
|
|
def teleopPeriodic(self) -> None:
|
2026-03-06 14:19:15 -08:00
|
|
|
# Get the x velocity. We are inverting this because Xbox controllers return
|
2026-02-20 18:30:35 -05:00
|
|
|
# negative values when we push forward.
|
2026-03-06 14:19:15 -08:00
|
|
|
xVelocity = (
|
|
|
|
|
-self.velocityLimiter.calculate(self.controller.getLeftY())
|
|
|
|
|
* Drivetrain.kMaxVelocity
|
2026-02-20 18:30:35 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
# Get the rate of angular rotation. We are inverting this because we want a
|
|
|
|
|
# positive value when we pull to the left (remember, CCW is positive in
|
|
|
|
|
# mathematics). Xbox controllers return positive values when you pull to
|
|
|
|
|
# the right by default.
|
|
|
|
|
rot = (
|
|
|
|
|
-self.rotLimiter.calculate(self.controller.getRightX())
|
2026-03-06 14:19:15 -08:00
|
|
|
* Drivetrain.kMaxAngularVelocity
|
2026-02-20 18:30:35 -05:00
|
|
|
)
|
2026-03-06 14:19:15 -08:00
|
|
|
self.drive.drive(xVelocity, rot)
|
2026-02-20 18:30:35 -05:00
|
|
|
|
|
|
|
|
def simulationPeriodic(self) -> None:
|
|
|
|
|
self.drive.simulationPeriodic()
|