2022-01-11 11:05:52 -07:00
|
|
|
// 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.
|
2021-11-15 16:26:16 -07:00
|
|
|
|
|
|
|
|
package frc4388.robot;
|
|
|
|
|
|
2022-02-15 11:05:59 -07:00
|
|
|
import java.io.File;
|
2022-02-11 18:53:13 -07:00
|
|
|
import java.io.IOException;
|
2022-02-10 21:18:00 -07:00
|
|
|
import java.util.ArrayList;
|
2022-02-15 11:05:59 -07:00
|
|
|
import java.util.Arrays;
|
|
|
|
|
import java.util.Comparator;
|
|
|
|
|
import java.util.Optional;
|
|
|
|
|
import java.util.logging.Logger;
|
2022-01-21 16:50:26 -07:00
|
|
|
|
2022-02-11 17:42:48 -07:00
|
|
|
import com.ctre.phoenix.motorcontrol.TalonFXControlMode;
|
|
|
|
|
import com.ctre.phoenix.motorcontrol.can.TalonFX;
|
2022-01-29 14:39:46 -07:00
|
|
|
import com.pathplanner.lib.PathPlanner;
|
2022-02-04 19:02:41 -07:00
|
|
|
import com.pathplanner.lib.PathPlannerTrajectory;
|
2022-02-07 19:49:27 -07:00
|
|
|
import com.pathplanner.lib.PathPlannerTrajectory.PathPlannerState;
|
2022-02-04 19:02:41 -07:00
|
|
|
import com.pathplanner.lib.commands.PPSwerveControllerCommand;
|
|
|
|
|
|
2022-01-29 14:39:46 -07:00
|
|
|
import edu.wpi.first.math.controller.PIDController;
|
|
|
|
|
import edu.wpi.first.math.controller.ProfiledPIDController;
|
2022-01-22 15:55:04 -07:00
|
|
|
import edu.wpi.first.math.geometry.Pose2d;
|
2022-01-29 14:39:46 -07:00
|
|
|
import edu.wpi.first.math.geometry.Rotation2d;
|
2022-02-15 11:05:59 -07:00
|
|
|
import edu.wpi.first.wpilibj.Filesystem;
|
2022-02-11 18:53:13 -07:00
|
|
|
import edu.wpi.first.wpilibj.GenericHID;
|
2022-01-29 01:16:58 -07:00
|
|
|
import edu.wpi.first.wpilibj.XboxController;
|
2022-02-15 11:05:59 -07:00
|
|
|
import edu.wpi.first.wpilibj.command.PrintCommand;
|
2021-11-15 16:26:16 -07:00
|
|
|
import edu.wpi.first.wpilibj2.command.Command;
|
2022-02-03 20:53:43 -07:00
|
|
|
import edu.wpi.first.wpilibj2.command.InstantCommand;
|
2021-11-15 16:26:16 -07:00
|
|
|
import edu.wpi.first.wpilibj2.command.RunCommand;
|
2022-02-03 20:53:43 -07:00
|
|
|
import edu.wpi.first.wpilibj2.command.SequentialCommandGroup;
|
2021-11-15 16:26:16 -07:00
|
|
|
import edu.wpi.first.wpilibj2.command.button.JoystickButton;
|
2022-02-11 18:53:13 -07:00
|
|
|
import frc4388.robot.Constants.LEDConstants;
|
|
|
|
|
import frc4388.robot.Constants.OIConstants;
|
2022-02-11 18:59:00 -07:00
|
|
|
import frc4388.robot.Constants.SwerveDriveConstants;
|
2021-11-15 16:26:16 -07:00
|
|
|
import frc4388.robot.subsystems.LED;
|
2021-12-02 17:51:06 -07:00
|
|
|
import frc4388.robot.subsystems.SwerveDrive;
|
2021-11-15 16:26:16 -07:00
|
|
|
import frc4388.utility.LEDPatterns;
|
2022-01-29 01:16:58 -07:00
|
|
|
import frc4388.utility.controller.DeadbandedXboxController;
|
2021-11-15 16:26:16 -07:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* This class is where the bulk of the robot should be declared. Since
|
|
|
|
|
* Command-based is a "declarative" paradigm, very little robot logic should
|
|
|
|
|
* actually be handled in the {@link Robot} periodic methods (other than the
|
|
|
|
|
* scheduler calls). Instead, the structure of the robot (including subsystems,
|
|
|
|
|
* commands, and button mappings) should be declared here.
|
|
|
|
|
*/
|
|
|
|
|
public class RobotContainer {
|
2022-01-11 11:05:52 -07:00
|
|
|
/* RobotMap */
|
|
|
|
|
private final RobotMap m_robotMap = new RobotMap();
|
|
|
|
|
|
|
|
|
|
/* Subsystems */
|
|
|
|
|
private final SwerveDrive m_robotSwerveDrive = new SwerveDrive(
|
2022-02-11 18:53:13 -07:00
|
|
|
m_robotMap.leftFront, m_robotMap.leftBack, m_robotMap.rightFront, m_robotMap.rightBack, m_robotMap.gyro);
|
2022-01-11 11:05:52 -07:00
|
|
|
|
2022-02-11 17:42:48 -07:00
|
|
|
private final TalonFX m_testMotor = new TalonFX(23);
|
|
|
|
|
|
2022-01-11 11:05:52 -07:00
|
|
|
private final LED m_robotLED = new LED(m_robotMap.LEDController);
|
|
|
|
|
|
|
|
|
|
/* Controllers */
|
2022-01-29 01:16:58 -07:00
|
|
|
private final XboxController m_driverXbox = new DeadbandedXboxController(OIConstants.XBOX_DRIVER_ID);
|
|
|
|
|
private final XboxController m_operatorXbox = new DeadbandedXboxController(OIConstants.XBOX_OPERATOR_ID);
|
2022-01-11 11:05:52 -07:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* The container for the robot. Contains subsystems, OI devices, and commands.
|
|
|
|
|
*/
|
|
|
|
|
public RobotContainer() {
|
|
|
|
|
configureButtonBindings();
|
|
|
|
|
/* Default Commands */
|
|
|
|
|
// drives the swerve drive with a two-axis input from the driver controller
|
2022-01-15 11:31:15 -07:00
|
|
|
m_robotSwerveDrive.setDefaultCommand(
|
|
|
|
|
new RunCommand(() -> m_robotSwerveDrive.driveWithInput(
|
2022-02-11 18:53:13 -07:00
|
|
|
getDriverController().getLeftX(),
|
|
|
|
|
getDriverController().getLeftY(),
|
|
|
|
|
getDriverController().getRightX(),
|
|
|
|
|
getDriverController().getRightY(),
|
|
|
|
|
true),
|
|
|
|
|
m_robotSwerveDrive));
|
2022-01-11 11:05:52 -07:00
|
|
|
|
|
|
|
|
// continually sends updates to the Blinkin LED controller to keep the lights on
|
|
|
|
|
m_robotLED.setDefaultCommand(new RunCommand(m_robotLED::updateLED, m_robotLED));
|
2022-02-11 17:42:48 -07:00
|
|
|
|
|
|
|
|
m_testMotor.set(TalonFXControlMode.PercentOutput, 0);
|
2022-01-11 11:05:52 -07:00
|
|
|
}
|
2022-01-15 11:31:15 -07:00
|
|
|
|
2022-01-11 11:05:52 -07:00
|
|
|
/**
|
|
|
|
|
* Use this method to define your button->command mappings. Buttons can be
|
|
|
|
|
* created by instantiating a {@link GenericHID} or one of its subclasses
|
|
|
|
|
* ({@link edu.wpi.first.wpilibj.Joystick} or {@link XboxController}), and then
|
|
|
|
|
* passing it to a {@link edu.wpi.first.wpilibj2.command.button.JoystickButton}.
|
|
|
|
|
*/
|
|
|
|
|
private void configureButtonBindings() {
|
|
|
|
|
/* Driver Buttons */
|
2022-01-29 01:16:58 -07:00
|
|
|
new JoystickButton(getDriverController(), XboxController.Button.kY.value)
|
2022-02-11 18:53:13 -07:00
|
|
|
// new XboxControllerRawButton(m_driverXbox, XboxControllerRaw.Y_BUTTON)
|
|
|
|
|
.whenPressed(m_robotSwerveDrive.m_gyro::reset);
|
2022-01-15 14:55:07 -07:00
|
|
|
|
2022-01-29 01:16:58 -07:00
|
|
|
new JoystickButton(getDriverController(), XboxController.Button.kLeftBumper.value)
|
2022-02-11 18:53:13 -07:00
|
|
|
// new XboxControllerRawButton(m_driverXbox,
|
|
|
|
|
// XboxControllerRaw.LEFT_BUMPER_BUTTON)
|
|
|
|
|
.whenPressed(() -> m_robotSwerveDrive.highSpeed(false));
|
2022-01-15 14:55:07 -07:00
|
|
|
|
2022-01-29 01:16:58 -07:00
|
|
|
new JoystickButton(getDriverController(), XboxController.Button.kRightBumper.value)
|
2022-02-11 18:53:13 -07:00
|
|
|
// new XboxControllerRawButton(m_driverXbox,
|
|
|
|
|
// XboxControllerRaw.RIGHT_BUMPER_BUTTON)
|
|
|
|
|
.whenPressed(() -> m_robotSwerveDrive.highSpeed(true));
|
|
|
|
|
|
2022-01-29 01:16:58 -07:00
|
|
|
new JoystickButton(getDriverController(), XboxController.Button.kA.value)
|
2022-02-05 11:50:49 -07:00
|
|
|
.whenPressed(() -> resetOdometry(new Pose2d(0, 0, new Rotation2d(0))));
|
2022-01-15 14:55:07 -07:00
|
|
|
|
2022-01-11 11:05:52 -07:00
|
|
|
/* Operator Buttons */
|
|
|
|
|
// activates "Lit Mode"
|
2022-01-29 01:16:58 -07:00
|
|
|
new JoystickButton(getOperatorController(), XboxController.Button.kA.value)
|
2022-02-11 18:53:13 -07:00
|
|
|
// new XboxControllerRawButton(m_driverXbox, XboxControllerRaw.A_BUTTON)
|
2022-01-11 11:05:52 -07:00
|
|
|
.whenPressed(() -> m_robotLED.setPattern(LEDPatterns.LAVA_RAINBOW))
|
|
|
|
|
.whenReleased(() -> m_robotLED.setPattern(LEDConstants.DEFAULT_PATTERN));
|
|
|
|
|
}
|
|
|
|
|
|
2022-02-11 17:42:48 -07:00
|
|
|
public Command[] buildAuto(double maxVel, double maxAccel, Object... inputs) {
|
2022-02-15 11:05:59 -07:00
|
|
|
ArrayList<Command> commands = new ArrayList<>();
|
2022-02-10 21:18:00 -07:00
|
|
|
commands.add(new InstantCommand(() -> m_robotSwerveDrive.m_gyro.reset()));
|
|
|
|
|
|
2022-02-05 10:08:37 -07:00
|
|
|
PIDController xController = SwerveDriveConstants.X_CONTROLLER;
|
|
|
|
|
PIDController yController = SwerveDriveConstants.Y_CONTROLLER;
|
|
|
|
|
ProfiledPIDController thetaController = SwerveDriveConstants.THETA_CONTROLLER;
|
|
|
|
|
thetaController.enableContinuousInput(-Math.PI, Math.PI);
|
2022-02-10 21:18:00 -07:00
|
|
|
|
|
|
|
|
// parse input
|
2022-02-15 11:05:59 -07:00
|
|
|
for (int i = 0; i < inputs.length; i++) {
|
2022-02-11 17:42:48 -07:00
|
|
|
if (inputs[i] instanceof String) {
|
|
|
|
|
PathPlannerTrajectory traj = PathPlanner.loadPath(inputs[i].toString(), maxVel, maxAccel);
|
2022-02-10 21:18:00 -07:00
|
|
|
PathPlannerState initState = (PathPlannerState) traj.sample(0);
|
|
|
|
|
|
|
|
|
|
commands.add(new InstantCommand(() -> m_robotSwerveDrive.resetOdometry(new Pose2d(initState.poseMeters.getTranslation(), initState.holonomicRotation))));
|
2022-02-15 11:05:59 -07:00
|
|
|
commands.add(new PPSwerveControllerCommand(traj, m_robotSwerveDrive::getOdometry, m_robotSwerveDrive.m_kinematics, xController, yController, thetaController, m_robotSwerveDrive::setModuleStates, m_robotSwerveDrive));
|
|
|
|
|
} else if (inputs[i] instanceof Command)
|
2022-02-11 17:42:48 -07:00
|
|
|
commands.add((Command) inputs[i]);
|
2022-02-10 21:18:00 -07:00
|
|
|
}
|
2022-02-15 11:05:59 -07:00
|
|
|
|
|
|
|
|
commands.add(new InstantCommand(m_robotSwerveDrive::stopModules));
|
2022-02-10 21:18:00 -07:00
|
|
|
Command[] ret = new Command[commands.size()];
|
|
|
|
|
ret = commands.toArray(ret);
|
|
|
|
|
return ret;
|
2022-02-04 19:02:41 -07:00
|
|
|
}
|
|
|
|
|
|
2022-01-11 11:05:52 -07:00
|
|
|
/**
|
|
|
|
|
* Use this to pass the autonomous command to the main {@link Robot} class.
|
|
|
|
|
*
|
|
|
|
|
* @return the command to run in autonomous
|
2022-02-11 18:53:13 -07:00
|
|
|
* @throws IOException
|
2022-01-11 11:05:52 -07:00
|
|
|
*/
|
2022-02-11 18:53:13 -07:00
|
|
|
public Command getAutonomousCommand() throws IOException {
|
2022-01-29 14:39:46 -07:00
|
|
|
// https://github.com/mjansen4857/pathplanner/wiki <-- Pathplanner Wiki
|
|
|
|
|
|
2022-02-15 11:05:59 -07:00
|
|
|
// Command[] trajCommands = buildAuto(0.5, 0.5, "Move Forward", "Move Down");
|
|
|
|
|
// SequentialCommandGroup ret = new SequentialCommandGroup(trajCommands);
|
|
|
|
|
// return (new ParallelCommandGroup(buildAuto(0.1, 0.1, ret, new InstantCommand(() -> m_testMotor.set(TalonFXControlMode.PercentOutput, 0.2)))));
|
|
|
|
|
Optional<File> recordedAuto = Arrays.stream(Filesystem.getDeployDirectory().toPath().resolve("pathplanner").toFile().listFiles()).filter(f -> f.getName().startsWith("recording")).max(Comparator.comparingLong(File::lastModified));
|
|
|
|
|
if (recordedAuto.isPresent()) {
|
|
|
|
|
String name = recordedAuto.orElseThrow().getName().replace(".path", "");
|
|
|
|
|
Logger.getLogger(this.getClass().getName()).fine(name);
|
|
|
|
|
return new SequentialCommandGroup(buildAuto(1.0, 1.0, name));
|
|
|
|
|
}
|
|
|
|
|
Logger.getLogger(this.getClass().getName()).severe("No recorded autos found.");
|
|
|
|
|
return new InstantCommand();
|
|
|
|
|
// return new PrintCommand("No recorded autos found.");
|
2022-01-11 11:05:52 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Add your docs here.
|
|
|
|
|
*/
|
2022-01-29 01:16:58 -07:00
|
|
|
public XboxController getDriverController() {
|
2022-01-11 11:05:52 -07:00
|
|
|
return m_driverXbox;
|
|
|
|
|
}
|
|
|
|
|
|
2022-01-22 15:55:04 -07:00
|
|
|
public Pose2d getOdometry() {
|
|
|
|
|
return m_robotSwerveDrive.getOdometry();
|
|
|
|
|
}
|
2022-01-24 17:00:06 -07:00
|
|
|
|
2022-02-05 11:50:49 -07:00
|
|
|
public void resetOdometry(Pose2d pose) {
|
2022-01-29 14:39:46 -07:00
|
|
|
m_robotSwerveDrive.resetOdometry(pose);
|
2022-01-24 17:00:06 -07:00
|
|
|
}
|
2022-02-11 18:53:13 -07:00
|
|
|
|
2022-01-11 11:05:52 -07:00
|
|
|
/**
|
|
|
|
|
* Add your docs here.
|
|
|
|
|
*/
|
2022-01-29 01:16:58 -07:00
|
|
|
public XboxController getOperatorController() {
|
2022-01-11 11:05:52 -07:00
|
|
|
return m_operatorXbox;
|
|
|
|
|
}
|
2021-11-15 16:26:16 -07:00
|
|
|
}
|