|
| 1 | +/*Taken straight from 4915 |
| 2 | + * https://github.com/Spartronics4915/2025-Reefscape/blob/unstable/src/main/java/com/spartronics4915/frc2025/commands/autos/PositionPIDCommand.java#L30 |
| 3 | + */ |
| 4 | +package frc.robot.util; |
| 5 | + |
| 6 | +import static edu.wpi.first.units.Units.Centimeter; |
| 7 | +import static edu.wpi.first.units.Units.Degrees; |
| 8 | +import static edu.wpi.first.units.Units.Meters; |
| 9 | +import static edu.wpi.first.units.Units.Seconds; |
| 10 | +import static frc.robot.subsystems.drive.DriveConstants.*; |
| 11 | + |
| 12 | +import com.pathplanner.lib.config.PIDConstants; |
| 13 | +import com.pathplanner.lib.controllers.PPHolonomicDriveController; |
| 14 | +import com.pathplanner.lib.trajectory.PathPlannerTrajectoryState; |
| 15 | +import edu.wpi.first.math.MathUtil; |
| 16 | +import edu.wpi.first.math.geometry.Pose2d; |
| 17 | +import edu.wpi.first.math.kinematics.ChassisSpeeds; |
| 18 | +import edu.wpi.first.networktables.BooleanPublisher; |
| 19 | +import edu.wpi.first.networktables.DoublePublisher; |
| 20 | +import edu.wpi.first.networktables.NetworkTableInstance; |
| 21 | +import edu.wpi.first.units.measure.Time; |
| 22 | +import edu.wpi.first.wpilibj.Timer; |
| 23 | +import edu.wpi.first.wpilibj2.command.Command; |
| 24 | +import edu.wpi.first.wpilibj2.command.button.Trigger; |
| 25 | +import frc.robot.subsystems.drive.Drive; |
| 26 | + |
| 27 | +import java.util.function.BooleanSupplier; |
| 28 | + |
| 29 | +public class PositionPIDCommand extends Command { |
| 30 | + |
| 31 | + public Drive mSwerve; |
| 32 | + public final Pose2d goalPose; |
| 33 | + private PPHolonomicDriveController mDriveController = |
| 34 | + new PPHolonomicDriveController(new PIDConstants(1),new PIDConstants(1)); //NOT TUNED |
| 35 | + // PPHolonomicController is the built in path following controller for holonomic drive trains; |
| 36 | + |
| 37 | + private final Trigger endTrigger; |
| 38 | + private final Trigger endTriggerDebounced; |
| 39 | + |
| 40 | + private final Timer timer = new Timer(); |
| 41 | + |
| 42 | + private final BooleanPublisher endTriggerLogger = |
| 43 | + NetworkTableInstance.getDefault() |
| 44 | + .getTable("logging") |
| 45 | + .getBooleanTopic("PositionPIDEndTrigger") |
| 46 | + .publish(); |
| 47 | + private final DoublePublisher xErrLogger = |
| 48 | + NetworkTableInstance.getDefault().getTable("logging").getDoubleTopic("X Error").publish(); |
| 49 | + private final DoublePublisher yErrLogger = |
| 50 | + NetworkTableInstance.getDefault().getTable("logging").getDoubleTopic("Y Error").publish(); |
| 51 | + |
| 52 | + private final BooleanSupplier booleanSupplier; |
| 53 | + |
| 54 | + private PositionPIDCommand(Drive mSwerve, Pose2d goalPose) { |
| 55 | + this.mSwerve = mSwerve; |
| 56 | + this.goalPose = goalPose; |
| 57 | + |
| 58 | + booleanSupplier = |
| 59 | + () -> { |
| 60 | + Pose2d diff = mSwerve.getPose().relativeTo(goalPose); |
| 61 | + |
| 62 | + var rotation = |
| 63 | + MathUtil.isNear( |
| 64 | + 0.0, |
| 65 | + diff.getRotation().getRotations(), |
| 66 | + kRotationTolerance.getRotations(), |
| 67 | + 0.0, |
| 68 | + 1.0); |
| 69 | + |
| 70 | + var position = diff.getTranslation().getNorm() < kPositionTolerance.in(Meters); |
| 71 | + |
| 72 | + // TODO: ADD SPEED FUNCTION |
| 73 | + // var speed = mSwerve.getSpeed() < kSpeedTolerance.in(MetersPerSecond); |
| 74 | + |
| 75 | + // System.out.println("end trigger conditions R: "+ rotation + "\tP: " + position + "\tS: |
| 76 | + // " + speed); |
| 77 | + |
| 78 | + return rotation && position; // && speed; |
| 79 | + }; |
| 80 | + |
| 81 | + endTrigger = new Trigger(booleanSupplier); |
| 82 | + |
| 83 | + endTriggerDebounced = endTrigger.debounce(kEndTriggerDebounce.in(Seconds)); |
| 84 | + } |
| 85 | + |
| 86 | + public static Command generateCommand( |
| 87 | + Drive swerve, Pose2d goalPose, Time timeout, Boolean freeze) { |
| 88 | + return new PositionPIDCommand(swerve, goalPose) |
| 89 | + .withTimeout(timeout) |
| 90 | + .finallyDo( |
| 91 | + () -> { |
| 92 | + if (freeze) { |
| 93 | + swerve.stopWithX(); |
| 94 | + } |
| 95 | + }); |
| 96 | + } |
| 97 | + |
| 98 | + @Override |
| 99 | + public void initialize() { |
| 100 | + endTriggerLogger.accept(endTrigger.getAsBoolean()); |
| 101 | + timer.restart(); |
| 102 | + } |
| 103 | + |
| 104 | + @Override |
| 105 | + public void execute() { |
| 106 | + PathPlannerTrajectoryState goalState = new PathPlannerTrajectoryState(); |
| 107 | + goalState.pose = goalPose; |
| 108 | + |
| 109 | + endTriggerLogger.accept(endTrigger.getAsBoolean()); |
| 110 | + ChassisSpeeds speeds = |
| 111 | + mDriveController.calculateRobotRelativeSpeeds(mSwerve.getPose(), goalState); |
| 112 | + mSwerve.runVelocity(speeds); |
| 113 | + |
| 114 | + xErrLogger.accept(mSwerve.getPose().getX() - goalPose.getX()); |
| 115 | + yErrLogger.accept(mSwerve.getPose().getY() - goalPose.getY()); |
| 116 | + } |
| 117 | + |
| 118 | + @Override |
| 119 | + public void end(boolean interrupted) { |
| 120 | + endTriggerLogger.accept(endTrigger.getAsBoolean()); |
| 121 | + timer.stop(); |
| 122 | + |
| 123 | + Pose2d diff = mSwerve.getPose().relativeTo(goalPose); |
| 124 | + |
| 125 | + System.out.println( |
| 126 | + "Adjustments to alginment took: " |
| 127 | + + timer.get() |
| 128 | + + " seconds and interrupted = " |
| 129 | + + interrupted |
| 130 | + + "\nPosition offset: " |
| 131 | + + Centimeter.convertFrom(diff.getTranslation().getNorm(), Meters) |
| 132 | + + " cm" |
| 133 | + + "\nRotation offset: " |
| 134 | + + diff.getRotation().getMeasure().in(Degrees) |
| 135 | + + " deg" |
| 136 | + + "\nVelocity value: " |
| 137 | + + "Speed Not Implemented" // mSwerve.getSpeed() + "m/s" |
| 138 | + // We dont have a get speed function yet |
| 139 | + ); |
| 140 | + } |
| 141 | + |
| 142 | + @Override |
| 143 | + public boolean isFinished() { |
| 144 | + return endTriggerDebounced.getAsBoolean(); |
| 145 | + } |
| 146 | +} |
0 commit comments