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-03-01 20:23:01 -07:00
|
|
|
import java.io.StringWriter;
|
2022-02-16 22:31:00 -07:00
|
|
|
import java.nio.file.FileSystems;
|
|
|
|
|
import java.nio.file.Path;
|
|
|
|
|
import java.nio.file.StandardWatchEventKinds;
|
|
|
|
|
import java.nio.file.WatchEvent;
|
|
|
|
|
import java.nio.file.WatchKey;
|
2022-03-01 11:57:49 -07:00
|
|
|
import java.time.Clock;
|
|
|
|
|
import java.time.ZoneId;
|
|
|
|
|
import java.time.ZonedDateTime;
|
|
|
|
|
import java.time.format.DateTimeFormatter;
|
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;
|
2022-02-16 22:31:00 -07:00
|
|
|
import java.util.List;
|
2022-02-27 22:59:32 -07:00
|
|
|
import java.util.Objects;
|
2022-02-15 11:05:59 -07:00
|
|
|
import java.util.Optional;
|
2022-02-16 22:31:00 -07:00
|
|
|
import java.util.function.Function;
|
|
|
|
|
import java.util.logging.Level;
|
2022-02-15 11:05:59 -07:00
|
|
|
import java.util.logging.Logger;
|
2022-02-16 22:31:00 -07:00
|
|
|
import java.util.regex.Matcher;
|
|
|
|
|
import java.util.regex.Pattern;
|
2022-02-27 22:59:32 -07:00
|
|
|
import java.util.stream.Collectors;
|
2022-01-21 16:50:26 -07:00
|
|
|
|
2022-03-17 20:01:32 -06:00
|
|
|
import com.ctre.phoenix.motorcontrol.NeutralMode;
|
2022-03-10 16:47:32 -07:00
|
|
|
import com.ctre.phoenix.motorcontrol.can.WPI_TalonFX;
|
2022-02-16 22:31:00 -07:00
|
|
|
import com.diffplug.common.base.Errors;
|
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-03-10 16:47:32 -07:00
|
|
|
import org.ejml.dense.row.decomposition.hessenberg.TridiagonalDecompositionHouseholderOrig_DDRM;
|
2022-03-18 17:18:39 -06:00
|
|
|
import org.opencv.core.Point;
|
2022-03-10 16:47:32 -07:00
|
|
|
|
2022-02-27 22:59:32 -07:00
|
|
|
import edu.wpi.first.math.Pair;
|
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-16 22:31:00 -07:00
|
|
|
import edu.wpi.first.math.geometry.Translation2d;
|
2022-03-01 20:23:01 -07:00
|
|
|
import edu.wpi.first.networktables.NetworkTable;
|
|
|
|
|
import edu.wpi.first.networktables.NetworkTableInstance;
|
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-03-01 20:23:01 -07:00
|
|
|
import edu.wpi.first.wpilibj.RobotBase;
|
2022-01-29 01:16:58 -07:00
|
|
|
import edu.wpi.first.wpilibj.XboxController;
|
2022-02-16 22:31:00 -07:00
|
|
|
import edu.wpi.first.wpilibj.smartdashboard.SmartDashboard;
|
2021-11-15 16:26:16 -07:00
|
|
|
import edu.wpi.first.wpilibj2.command.Command;
|
|
|
|
|
import edu.wpi.first.wpilibj2.command.InstantCommand;
|
2022-02-16 22:31:00 -07:00
|
|
|
import edu.wpi.first.wpilibj2.command.NotifierCommand;
|
2022-03-14 15:16:21 -06:00
|
|
|
import edu.wpi.first.wpilibj2.command.ParallelCommandGroup;
|
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;
|
|
|
|
|
import frc4388.robot.Constants.*;
|
2022-03-05 14:21:48 -07:00
|
|
|
import frc4388.robot.subsystems.Claws;
|
2022-03-18 15:37:01 -06:00
|
|
|
import frc4388.robot.subsystems.Climber;
|
2022-03-05 15:15:41 -07:00
|
|
|
import frc4388.robot.subsystems.Claws.ClawType;
|
2022-02-11 18:53:13 -07:00
|
|
|
import frc4388.robot.Constants.OIConstants;
|
2022-03-14 16:56:32 -06:00
|
|
|
import frc4388.robot.Constants.ShooterConstants;
|
2022-03-05 22:57:55 -07:00
|
|
|
import frc4388.robot.Constants.StorageConstants;
|
2022-02-11 18:59:00 -07:00
|
|
|
import frc4388.robot.Constants.SwerveDriveConstants;
|
2022-03-16 11:22:51 -06:00
|
|
|
import frc4388.robot.commands.ButtonBoxCommands.RunMiddleSwitch;
|
2022-03-18 15:05:07 -06:00
|
|
|
import frc4388.robot.commands.ClimberCommands.RunClaw;
|
2022-03-18 17:18:39 -06:00
|
|
|
import frc4388.robot.commands.ClimberCommands.RunClimberPath;
|
2022-03-17 13:49:13 -06:00
|
|
|
// import frc4388.robot.commands.ButtonBoxCommands.TurretManual;
|
2022-03-14 20:10:12 -06:00
|
|
|
import frc4388.robot.commands.ExtenderIntakeCommands.ExtenderIntakeGroup;
|
|
|
|
|
import frc4388.robot.commands.ShooterCommands.AimToCenter;
|
2022-03-20 16:22:11 -06:00
|
|
|
import frc4388.robot.commands.ShooterCommands.Seek;
|
2022-03-14 20:10:12 -06:00
|
|
|
import frc4388.robot.commands.ShooterCommands.Shoot;
|
|
|
|
|
import frc4388.robot.commands.ShooterCommands.TrackTarget;
|
2022-03-16 13:59:10 -06:00
|
|
|
import frc4388.robot.commands.StorageCommands.ManageStorage;
|
2022-01-20 18:08:05 -07:00
|
|
|
import frc4388.robot.subsystems.BoomBoom;
|
2022-03-10 16:47:32 -07:00
|
|
|
import frc4388.robot.subsystems.Climber;
|
2022-03-12 15:05:32 -07:00
|
|
|
import frc4388.robot.subsystems.Extender;
|
2022-01-20 18:08:05 -07:00
|
|
|
import frc4388.robot.subsystems.Hood;
|
2022-03-05 15:32:48 -07:00
|
|
|
import frc4388.robot.subsystems.Intake;
|
2021-11-15 16:26:16 -07:00
|
|
|
import frc4388.robot.subsystems.LED;
|
2022-01-15 14:20:49 -07:00
|
|
|
import frc4388.robot.subsystems.Serializer;
|
2022-02-26 15:41:54 -07:00
|
|
|
import frc4388.robot.subsystems.Storage;
|
2021-12-02 17:51:06 -07:00
|
|
|
import frc4388.robot.subsystems.SwerveDrive;
|
2022-01-20 18:08:05 -07:00
|
|
|
import frc4388.robot.subsystems.Turret;
|
|
|
|
|
import frc4388.robot.subsystems.Vision;
|
2022-03-05 11:29:40 -07:00
|
|
|
import frc4388.robot.subsystems.VisionOdometry;
|
2022-03-05 15:15:41 -07:00
|
|
|
|
2021-11-15 16:26:16 -07:00
|
|
|
import frc4388.utility.LEDPatterns;
|
2022-02-27 22:59:32 -07:00
|
|
|
import frc4388.utility.ListeningSendableChooser;
|
2022-02-16 22:31:00 -07:00
|
|
|
import frc4388.utility.PathPlannerUtil;
|
2022-03-16 15:13:34 -06:00
|
|
|
import frc4388.utility.Vector2D;
|
2022-02-16 22:31:00 -07:00
|
|
|
import frc4388.utility.PathPlannerUtil.Path.Waypoint;
|
2022-03-12 17:33:38 -07:00
|
|
|
import frc4388.utility.controller.ButtonBox;
|
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-02-25 01:33:32 -07:00
|
|
|
private static final Logger LOGGER = Logger.getLogger(RobotContainer.class.getSimpleName());
|
2022-03-14 20:10:12 -06:00
|
|
|
|
|
|
|
|
// RobotMap
|
2022-03-07 15:26:05 -07:00
|
|
|
public final RobotMap m_robotMap = new RobotMap();
|
2022-01-11 11:05:52 -07:00
|
|
|
|
|
|
|
|
/* Subsystems */
|
2022-03-20 18:09:16 -06:00
|
|
|
public final Climber m_robotClimber = new Climber(m_robotMap.elbow);
|
2022-03-18 14:13:37 -06:00
|
|
|
public final Claws m_robotClaws = new Claws(m_robotMap.leftClaw, m_robotMap.rightClaw);
|
2022-03-05 15:32:48 -07:00
|
|
|
public final SwerveDrive m_robotSwerveDrive = new SwerveDrive(m_robotMap.leftFront, m_robotMap.leftBack, m_robotMap.rightFront, m_robotMap.rightBack, m_robotMap.gyro);
|
2022-03-07 15:26:05 -07:00
|
|
|
public final Serializer m_robotSerializer = new Serializer(m_robotMap.serializerBelt, /*m_robotMap.serializerShooterBelt,*/ m_robotMap.serializerBeam);
|
2022-03-14 20:10:12 -06:00
|
|
|
public final Intake m_robotIntake = new Intake(m_robotMap.intakeMotor);
|
2022-03-12 15:05:32 -07:00
|
|
|
public final Extender m_robotExtender = new Extender(m_robotMap.extenderMotor);
|
2022-03-05 22:57:55 -07:00
|
|
|
|
2022-03-07 15:26:05 -07:00
|
|
|
public final Storage m_robotStorage = new Storage(m_robotMap.storageMotor);
|
2022-03-06 11:14:30 -07:00
|
|
|
// private final LED m_robotLED = new LED(m_robotMap.LEDController);
|
2022-03-12 20:39:05 -07:00
|
|
|
public final BoomBoom m_robotBoomBoom = new BoomBoom(m_robotMap.shooterFalconLeft, m_robotMap.shooterFalconRight);
|
2022-03-12 22:19:42 -07:00
|
|
|
public final Hood m_robotHood = new Hood(m_robotMap.angleAdjusterMotor);
|
2022-03-07 15:26:05 -07:00
|
|
|
public final Turret m_robotTurret = new Turret(m_robotMap.shooterTurret);
|
2022-03-13 19:59:55 -06:00
|
|
|
public final VisionOdometry m_robotVisionOdometry = new VisionOdometry(m_robotSwerveDrive, m_robotTurret);
|
2022-03-17 20:01:32 -06:00
|
|
|
|
2022-03-14 20:10:12 -06:00
|
|
|
// Controllers
|
2022-03-20 15:24:48 -06:00
|
|
|
private final static XboxController m_driverXbox = new DeadbandedXboxController(OIConstants.XBOX_DRIVER_ID);
|
|
|
|
|
private final static XboxController m_operatorXbox = new DeadbandedXboxController(OIConstants.XBOX_OPERATOR_ID);
|
2022-03-12 19:02:37 -07:00
|
|
|
private final ButtonBox m_buttonBox = new ButtonBox(OIConstants.BUTTON_BOX_ID);
|
2022-01-11 11:05:52 -07:00
|
|
|
|
2022-03-14 20:10:12 -06:00
|
|
|
// Autonomous
|
2022-02-16 22:31:00 -07:00
|
|
|
private PathPlannerTrajectory loadedPathTrajectory = null;
|
2022-03-18 14:13:37 -06:00
|
|
|
private final ListeningSendableChooser<File> autoChooser = new ListeningSendableChooser<>(this::loadPath);
|
2022-03-01 11:57:49 -07:00
|
|
|
private final List<Waypoint> pathPoints = new ArrayList<>();
|
2022-03-01 20:23:01 -07:00
|
|
|
private final NetworkTableInstance networkTableInstance = NetworkTableInstance.getDefault();
|
|
|
|
|
private final NetworkTable recordingNetworkTable = networkTableInstance.getTable("Recording");
|
2022-02-27 22:59:32 -07:00
|
|
|
|
2022-03-05 11:12:33 -07:00
|
|
|
private static final DateTimeFormatter RECORDING_FILE_NAME_FORMATTER = DateTimeFormatter
|
|
|
|
|
.ofPattern("'Recording' yyyy-MM-dd HH-mm-ss.SSS'.path'");
|
2022-03-01 11:57:49 -07:00
|
|
|
private static final Clock SYSTEM_CLOCK = Clock.system(ZoneId.systemDefault());
|
|
|
|
|
private static final Path PATHPLANNER_DIRECTORY = Filesystem.getDeployDirectory().toPath().resolve("pathplanner");
|
2022-03-03 23:56:08 -07:00
|
|
|
// Function that removes the ".path" from the end of a string.
|
2022-03-05 11:12:33 -07:00
|
|
|
private static final Function<CharSequence, String> PATH_EXTENSION_REMOVER = ((Function<CharSequence, Matcher>) Pattern
|
|
|
|
|
.compile(".path")::matcher).andThen(m -> m.replaceFirst(""));
|
2022-03-12 17:43:17 -07:00
|
|
|
|
2022-03-12 22:19:42 -07:00
|
|
|
public static boolean softLimits = true;
|
2022-03-18 17:18:39 -06:00
|
|
|
|
2022-03-18 17:39:06 -06:00
|
|
|
// control mode switching
|
2022-03-18 18:29:53 -06:00
|
|
|
public static enum ControlMode { SHOOTER, CLIMBER };
|
|
|
|
|
public static ControlMode currentControlMode = ControlMode.SHOOTER;
|
2022-03-18 17:18:39 -06:00
|
|
|
|
2022-03-18 17:39:06 -06:00
|
|
|
// turret mode switching
|
|
|
|
|
private enum TurretMode { MANUAL, AUTONOMOUS };
|
|
|
|
|
private TurretMode currentTurretMode = TurretMode.MANUAL;
|
|
|
|
|
|
|
|
|
|
// climber mode switching
|
2022-03-18 17:18:39 -06:00
|
|
|
private enum ClimberMode { MANUAL, AUTONOMOUS };
|
|
|
|
|
private ClimberMode currentClimberMode = ClimberMode.MANUAL;
|
2022-01-11 11:05:52 -07:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* The container for the robot. Contains subsystems, OI devices, and commands.
|
|
|
|
|
*/
|
|
|
|
|
public RobotContainer() {
|
|
|
|
|
configureButtonBindings();
|
|
|
|
|
/* Default Commands */
|
2022-03-20 20:18:02 -06:00
|
|
|
// Swerve Drive with Input
|
2022-03-07 15:26:05 -07:00
|
|
|
m_robotSwerveDrive.setDefaultCommand(
|
2022-03-20 00:50:01 -06:00
|
|
|
new RunCommand(() -> {
|
2022-03-20 20:18:02 -06:00
|
|
|
if (RobotContainer.currentControlMode.equals(ControlMode.SHOOTER)) {
|
2022-03-20 00:50:01 -06:00
|
|
|
m_robotSwerveDrive.driveWithInput( getDriverController().getLeftX(),
|
|
|
|
|
getDriverController().getLeftY(),
|
|
|
|
|
getDriverController().getRightX(),
|
|
|
|
|
getDriverController().getRightY(),
|
2022-03-20 20:18:02 -06:00
|
|
|
true); }
|
|
|
|
|
if (RobotContainer.currentControlMode.equals(ControlMode.CLIMBER)) {
|
|
|
|
|
m_robotSwerveDrive.driveWithInput( 0,
|
|
|
|
|
0,
|
|
|
|
|
0,
|
|
|
|
|
0,
|
|
|
|
|
true);
|
|
|
|
|
}}
|
2022-03-20 00:50:01 -06:00
|
|
|
, m_robotSwerveDrive).withName("Swerve driveWithInput defaultCommand"));
|
2022-03-18 17:18:39 -06:00
|
|
|
|
2022-03-20 20:18:02 -06:00
|
|
|
// Intake with Triggers
|
2022-03-05 22:57:55 -07:00
|
|
|
m_robotIntake.setDefaultCommand(
|
|
|
|
|
new RunCommand(() -> m_robotIntake.runWithTriggers(
|
2022-03-07 15:26:05 -07:00
|
|
|
getOperatorController().getLeftTriggerAxis(),
|
|
|
|
|
getOperatorController().getRightTriggerAxis()),
|
2022-03-05 22:57:55 -07:00
|
|
|
m_robotIntake).withName("Intake runWithTriggers defaultCommand"));
|
2022-03-20 00:16:49 -06:00
|
|
|
m_robotBoomBoom.setDefaultCommand(new RunCommand(() -> m_robotBoomBoom.runDrumShooter(0.0), m_robotBoomBoom));
|
2022-03-18 17:18:39 -06:00
|
|
|
|
2022-03-20 20:18:02 -06:00
|
|
|
// Serializer Manual
|
2022-03-08 16:14:34 -07:00
|
|
|
m_robotSerializer.setDefaultCommand(
|
2022-03-18 15:09:08 -06:00
|
|
|
new RunCommand(() -> m_robotSerializer.setSerializer(getOperatorController().getLeftTriggerAxis() * 0.8),//m_robotSerializer.setSerializerStateWithBeam(),
|
2022-03-08 16:14:34 -07:00
|
|
|
m_robotSerializer).withName("Serializer setSerializerStateWithBeam defaultCommand"));
|
2022-03-20 00:16:49 -06:00
|
|
|
|
2022-03-20 20:18:02 -06:00
|
|
|
// Turret Manual
|
2022-03-20 00:16:49 -06:00
|
|
|
m_robotTurret.setDefaultCommand(
|
|
|
|
|
new RunCommand(() -> {
|
2022-03-20 12:09:41 -06:00
|
|
|
if (RobotContainer.currentControlMode.equals(ControlMode.SHOOTER)) { m_robotTurret.runTurretWithInput(getOperatorController().getLeftX()); }
|
|
|
|
|
if (RobotContainer.currentControlMode.equals(ControlMode.CLIMBER)) { m_robotTurret.runTurretWithInput(0); }
|
2022-03-20 00:16:49 -06:00
|
|
|
}, m_robotTurret));
|
2022-03-18 23:56:01 -06:00
|
|
|
|
2022-03-20 20:18:02 -06:00
|
|
|
// Hood Manual
|
2022-03-18 23:56:01 -06:00
|
|
|
m_robotHood.setDefaultCommand(
|
|
|
|
|
new RunCommand(() -> {
|
2022-03-20 12:09:41 -06:00
|
|
|
if (RobotContainer.currentControlMode.equals(ControlMode.SHOOTER)) { m_robotHood.runHood(getOperatorController().getRightY()); }
|
|
|
|
|
if (RobotContainer.currentControlMode.equals(ControlMode.CLIMBER)) { m_robotHood.runHood(0); }
|
2022-03-18 23:56:01 -06:00
|
|
|
}, m_robotHood));
|
|
|
|
|
|
2022-03-20 20:18:02 -06:00
|
|
|
//Climber Manual
|
2022-03-20 00:16:49 -06:00
|
|
|
m_robotClimber.setDefaultCommand(
|
2022-03-20 20:18:02 -06:00
|
|
|
new RunCommand(() -> {
|
|
|
|
|
if (RobotContainer.currentControlMode.equals(ControlMode.SHOOTER)) { m_robotClimber.setMotors(0.0); }
|
|
|
|
|
if (RobotContainer.currentControlMode.equals(ControlMode.CLIMBER)) { m_robotClimber.setMotors(getOperatorController().getRightY()); }
|
|
|
|
|
}, m_robotClimber));
|
2022-03-10 17:44:27 -07:00
|
|
|
// autoInit();
|
|
|
|
|
// recordInit();
|
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() {
|
2022-03-05 22:57:55 -07:00
|
|
|
|
2022-03-20 20:18:02 -06:00
|
|
|
//! Driver Buttons
|
2022-01-11 11:05:52 -07:00
|
|
|
|
2022-03-06 00:48:38 -07:00
|
|
|
// Start > Calibrate Odometry
|
2022-03-06 11:23:40 -07:00
|
|
|
new JoystickButton(getDriverController(), XboxController.Button.kBack.value)
|
2022-03-06 00:48:38 -07:00
|
|
|
.whenPressed(() -> resetOdometry(new Pose2d(0, 0, new Rotation2d(0))));
|
2022-03-20 20:18:02 -06:00
|
|
|
|
|
|
|
|
// Start > Calibrate Odometry
|
2022-03-06 11:23:40 -07:00
|
|
|
new JoystickButton(getDriverController(), XboxController.Button.kStart.value)
|
2022-03-20 20:18:02 -06:00
|
|
|
.whenPressed(m_robotSwerveDrive::resetGyro);
|
|
|
|
|
|
2022-03-06 00:48:38 -07:00
|
|
|
// Left Bumper > Shift Down
|
2022-03-18 13:04:20 -06:00
|
|
|
new JoystickButton(getDriverController(), XboxController.Button.kLeftBumper.value)
|
|
|
|
|
.whenPressed(() -> m_robotSwerveDrive.highSpeed(false));
|
2022-03-20 20:18:02 -06:00
|
|
|
|
2022-03-18 13:04:20 -06:00
|
|
|
// Right Bumper > Shift Up
|
2022-03-18 23:11:30 -06:00
|
|
|
new JoystickButton(getDriverController(), XboxController.Button.kRightBumper.value)
|
|
|
|
|
.whenPressed(() -> m_robotSwerveDrive.highSpeed(true));
|
|
|
|
|
|
2022-03-20 18:09:16 -06:00
|
|
|
|
2022-03-20 16:22:11 -06:00
|
|
|
|
2022-03-20 20:18:02 -06:00
|
|
|
//! Operator Buttons
|
|
|
|
|
|
|
|
|
|
// Right Bumper > Storage Out
|
|
|
|
|
new JoystickButton(getOperatorController(), XboxController.Button.kRightBumper.value)
|
|
|
|
|
.whileHeld(new RunCommand(() -> m_robotStorage.runStorage(-StorageConstants.STORAGE_SPEED)))
|
|
|
|
|
.whenReleased(new RunCommand(() -> m_robotStorage.runStorage(0.0)));
|
|
|
|
|
|
|
|
|
|
// Left Bumper > Storage In
|
|
|
|
|
new JoystickButton(getOperatorController(), XboxController.Button.kLeftBumper.value)
|
|
|
|
|
.whileHeld(new RunCommand(() -> m_robotStorage.runStorage(StorageConstants.STORAGE_SPEED)))
|
|
|
|
|
.whenReleased(new RunCommand(() -> m_robotStorage.runStorage(0.0)));
|
|
|
|
|
|
|
|
|
|
// B > Toggle claws
|
2022-03-20 16:22:11 -06:00
|
|
|
new JoystickButton(getOperatorController(), XboxController.Button.kB.value)
|
2022-03-20 20:52:05 -06:00
|
|
|
.whenPressed(new InstantCommand(() -> m_robotClaws.toggleClaws(), m_robotClaws));
|
2022-03-20 14:25:38 -06:00
|
|
|
|
2022-03-20 20:18:02 -06:00
|
|
|
// X > Toggles extender in and out
|
2022-03-20 21:21:45 -06:00
|
|
|
// new JoystickButton(getOperatorController(), XboxController.Button.kX.value)
|
|
|
|
|
// .whenPressed(new ExtenderIntakeGroup(m_robotIntake, m_robotExtender));
|
2022-03-16 20:16:32 -06:00
|
|
|
|
2022-03-20 20:18:02 -06:00
|
|
|
// A > Spit Out Ball
|
|
|
|
|
new JoystickButton(getOperatorController(), XboxController.Button.kA.value)
|
2022-03-20 20:52:05 -06:00
|
|
|
.whileHeld(new RunCommand(() -> m_robotTurret.gotoMidpoint(), m_robotTurret))
|
2022-03-20 20:18:02 -06:00
|
|
|
.whileHeld(new RunCommand(() -> m_robotBoomBoom.runDrumShooter(0.25)));
|
|
|
|
|
|
|
|
|
|
// Y > Full aim command
|
2022-03-14 15:16:21 -06:00
|
|
|
// new JoystickButton(getOperatorController(), XboxController.Button.kY.value)
|
2022-03-20 21:21:45 -06:00
|
|
|
// .whileHeld(new Seek(m_robotSwerveDrive, m_robotBoomBoom, m_robotTurret, m_robotHood, m_robotVisionOdometry));
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//! Test Buttons
|
|
|
|
|
new JoystickButton(getOperatorController(), XboxController.Button.kY.value)
|
|
|
|
|
.whenPressed(new Shoot(m_robotSwerveDrive, m_robotBoomBoom, m_robotTurret, m_robotHood, m_robotVisionOdometry, false, false));
|
2022-03-20 20:18:02 -06:00
|
|
|
|
2022-03-20 21:21:45 -06:00
|
|
|
new JoystickButton(getOperatorController(), XboxController.Button.kX.value)
|
|
|
|
|
.whileHeld(new TrackTarget(m_robotSwerveDrive, m_robotTurret, m_robotBoomBoom, m_robotHood, m_robotVisionOdometry));
|
2022-03-05 22:57:55 -07:00
|
|
|
|
2022-03-20 20:18:02 -06:00
|
|
|
// new JoystickButton(getOperatorController(), XboxController.Button.kY.value)
|
|
|
|
|
// .whileHeld(new RunCommand(() -> m_robotClaws.setOpen(true)));
|
2022-03-05 22:57:55 -07:00
|
|
|
|
2022-03-20 20:18:02 -06:00
|
|
|
// new JoystickButton(getOperatorController(), XboxController.Button.kB.value)
|
|
|
|
|
// .whileHeld(new RunCommand(() -> m_robotClaws.setOpen(false)));
|
2022-03-05 22:57:55 -07:00
|
|
|
|
2022-03-20 20:18:02 -06:00
|
|
|
// new JoystickButton(getOperatorController(), XboxController.Button.kA.value)
|
|
|
|
|
// .whenPressed(new Shoot(m_robotSwerveDrive, m_robotBoomBoom, m_robotTurret, m_robotHood));
|
2022-03-13 17:57:36 -06:00
|
|
|
|
2022-03-20 20:18:02 -06:00
|
|
|
// new JoystickButton(getOperatorController(), XboxController.Button.kA.value)
|
|
|
|
|
// .whenPressed(new RunCommand(() -> m_robotBoomBoom.runDrumShooter(0.25)))
|
|
|
|
|
// .whenReleased(new RunCommand(() -> m_robotBoomBoom.runDrumShooter(0.0)));
|
2022-03-10 16:47:32 -07:00
|
|
|
|
2022-03-05 22:57:55 -07:00
|
|
|
|
|
|
|
|
|
2022-03-20 20:18:02 -06:00
|
|
|
//! Button Box Buttons
|
|
|
|
|
// Left Switch > Disables soft limits on press, release resets encoders (all for turret, hood, climber, and extender)
|
2022-03-12 19:02:37 -07:00
|
|
|
new JoystickButton(getButtonBox(), ButtonBox.Button.kLeftSwitch.value)
|
2022-03-20 20:18:02 -06:00
|
|
|
.whenPressed(new InstantCommand(() -> m_robotTurret.setTurretSoftLimits(false), m_robotTurret))
|
|
|
|
|
.whenPressed(new InstantCommand(() -> m_robotTurret.calibrationSpeed = 0.3, m_robotTurret))
|
2022-03-19 21:03:01 -06:00
|
|
|
|
2022-03-20 20:18:02 -06:00
|
|
|
.whenPressed(new InstantCommand(() -> m_robotHood.setHoodSoftLimits(false), m_robotHood))
|
|
|
|
|
.whenPressed(new InstantCommand(() -> m_robotHood.calibrationSpeed = 0.3, m_robotHood))
|
2022-03-19 21:03:01 -06:00
|
|
|
|
2022-03-20 20:18:02 -06:00
|
|
|
.whenPressed(new InstantCommand(() -> m_robotExtender.setExtenderSoftLimits(false), m_robotExtender))
|
2022-03-19 21:03:01 -06:00
|
|
|
|
2022-03-20 20:18:02 -06:00
|
|
|
.whenReleased(new InstantCommand(() -> m_robotTurret.setTurretSoftLimits(true), m_robotTurret))
|
|
|
|
|
.whenReleased(new InstantCommand(() -> m_robotTurret.calibrationSpeed = 1.0, m_robotTurret))
|
2022-03-19 21:03:01 -06:00
|
|
|
|
2022-03-20 20:18:02 -06:00
|
|
|
.whenReleased(new InstantCommand(() -> m_robotHood.setHoodSoftLimits(true), m_robotHood))
|
|
|
|
|
.whenReleased(new InstantCommand(() -> m_robotHood.calibrationSpeed = 1.0, m_robotHood))
|
2022-03-19 21:03:01 -06:00
|
|
|
|
2022-03-20 20:18:02 -06:00
|
|
|
.whenReleased(new InstantCommand(() -> m_robotExtender.setExtenderSoftLimits(true), m_robotExtender))
|
2022-03-06 00:24:51 -07:00
|
|
|
|
2022-03-20 20:18:02 -06:00
|
|
|
.whenReleased(new InstantCommand(() -> m_robotTurret.m_boomBoomRotateEncoder.setPosition(0), m_robotTurret))
|
|
|
|
|
.whenReleased(new InstantCommand(() -> m_robotHood.m_angleEncoder.setPosition(0), m_robotHood))
|
|
|
|
|
.whenReleased(new InstantCommand(() -> m_robotExtender.setEncoder(0), m_robotExtender))
|
|
|
|
|
.whenReleased(new InstantCommand(() -> ExtenderIntakeGroup.setDirectionToOut(), m_robotIntake, m_robotExtender))
|
|
|
|
|
.whenReleased(new InstantCommand(() -> m_robotClimber.setEncoders(0), m_robotClimber));
|
2022-03-12 17:33:38 -07:00
|
|
|
|
2022-03-20 20:18:02 -06:00
|
|
|
// Middle Switch > Climber and Shooter mode switching
|
2022-03-18 17:37:11 -06:00
|
|
|
new JoystickButton(getButtonBox(), ButtonBox.Button.kMiddleSwitch.value)
|
2022-03-20 20:18:02 -06:00
|
|
|
.whenPressed(new InstantCommand(() -> this.currentControlMode = ControlMode.CLIMBER))
|
|
|
|
|
.whenReleased(new InstantCommand(() -> this.currentControlMode = ControlMode.SHOOTER));
|
2022-03-18 15:09:08 -06:00
|
|
|
|
2022-03-20 20:18:02 -06:00
|
|
|
// Left Button > Extender In
|
2022-03-12 19:02:37 -07:00
|
|
|
new JoystickButton(getButtonBox(), ButtonBox.Button.kLeftButton.value)
|
2022-03-20 20:18:02 -06:00
|
|
|
.whileHeld(new RunCommand(() -> m_robotExtender.runExtender(-1.0), m_robotExtender))
|
|
|
|
|
.whenReleased(new RunCommand(() -> m_robotExtender.runExtender(0.0), m_robotExtender));
|
2022-03-12 17:33:38 -07:00
|
|
|
|
2022-03-20 20:18:02 -06:00
|
|
|
// Left Button > Extender Out
|
2022-03-12 19:02:37 -07:00
|
|
|
new JoystickButton(getButtonBox(), ButtonBox.Button.kRightButton.value)
|
2022-03-20 20:18:02 -06:00
|
|
|
.whileHeld(new RunCommand(() -> m_robotExtender.runExtender(1.0), m_robotExtender))
|
|
|
|
|
.whenReleased(new RunCommand(() -> m_robotExtender.runExtender(0.0), m_robotExtender));
|
2022-03-12 17:33:38 -07:00
|
|
|
}
|
2022-01-11 11:05:52 -07:00
|
|
|
|
2022-03-20 19:31:42 -06:00
|
|
|
/**
|
|
|
|
|
* Generate autonomous
|
|
|
|
|
* @param maxVel max velocity for the path (null to override default value of 5.0)
|
|
|
|
|
* @param maxAccel max acceleration for the path (null to override default value of 5.0)
|
|
|
|
|
* @param inputs strings (paths) or commands you want to run (in order)
|
|
|
|
|
* @return array of commands
|
|
|
|
|
*/
|
|
|
|
|
public Command[] buildAuto(Double maxVel, Double maxAccel, Object... inputs) {
|
|
|
|
|
maxVel = Objects.requireNonNullElse(maxVel, SwerveDriveConstants.PATH_MAX_VELOCITY);
|
|
|
|
|
maxAccel = Objects.requireNonNullElse(maxAccel, SwerveDriveConstants.PATH_MAX_ACCELERATION);
|
|
|
|
|
|
|
|
|
|
ArrayList<Command> commands = new ArrayList<Command>();
|
|
|
|
|
|
|
|
|
|
PIDController xController = SwerveDriveConstants.X_CONTROLLER;
|
|
|
|
|
PIDController yController = SwerveDriveConstants.Y_CONTROLLER;
|
|
|
|
|
ProfiledPIDController thetaController = SwerveDriveConstants.THETA_CONTROLLER;
|
|
|
|
|
thetaController.enableContinuousInput(-Math.PI, Math.PI);
|
|
|
|
|
|
|
|
|
|
// parse input
|
|
|
|
|
for (int i=0; i<inputs.length; i++) {
|
|
|
|
|
if (inputs[i] instanceof String) {
|
|
|
|
|
|
|
|
|
|
PathPlannerTrajectory traj = PathPlanner.loadPath(inputs[i].toString(), maxVel, maxAccel);
|
|
|
|
|
PathPlannerState initState = (PathPlannerState) traj.sample(0);
|
|
|
|
|
commands.add(new InstantCommand(() -> m_robotSwerveDrive.resetOdometry(new Pose2d(initState.poseMeters.getTranslation(), initState.holonomicRotation))));
|
|
|
|
|
commands.add(new PPSwerveControllerCommand(
|
|
|
|
|
traj,
|
|
|
|
|
m_robotSwerveDrive::getOdometry,
|
|
|
|
|
m_robotSwerveDrive.m_kinematics,
|
|
|
|
|
xController,
|
|
|
|
|
yController,
|
|
|
|
|
thetaController,
|
|
|
|
|
m_robotSwerveDrive::setModuleStates,
|
|
|
|
|
m_robotSwerveDrive));
|
|
|
|
|
}
|
|
|
|
|
if (inputs[i] instanceof Command) {
|
|
|
|
|
commands.add((Command) inputs[i]);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
commands.add(new InstantCommand(() -> m_robotSwerveDrive.stopModules()));
|
|
|
|
|
Command[] ret = new Command[commands.size()];
|
|
|
|
|
ret = commands.toArray(ret);
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
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
|
|
|
|
|
*/
|
|
|
|
|
public Command getAutonomousCommand() {
|
2022-03-10 17:44:27 -07:00
|
|
|
// if (loadedPathTrajectory != null) {
|
|
|
|
|
// PIDController xController = SwerveDriveConstants.X_CONTROLLER;
|
|
|
|
|
// PIDController yController = SwerveDriveConstants.Y_CONTROLLER;
|
|
|
|
|
// ProfiledPIDController thetaController = SwerveDriveConstants.THETA_CONTROLLER;
|
|
|
|
|
// thetaController.enableContinuousInput(-Math.PI, Math.PI);
|
|
|
|
|
|
|
|
|
|
// PathPlannerState initialState = loadedPathTrajectory.getInitialState();
|
|
|
|
|
// Pose2d initialPosition = new Pose2d(initialState.poseMeters.getTranslation(), initialState.holonomicRotation);
|
|
|
|
|
// return new SequentialCommandGroup(
|
|
|
|
|
// new InstantCommand(m_robotSwerveDrive.m_gyro::reset),
|
|
|
|
|
// new InstantCommand(() -> m_robotSwerveDrive.resetOdometry(initialPosition)),
|
|
|
|
|
// new PPSwerveControllerCommand(loadedPathTrajectory, m_robotSwerveDrive::getOdometry,
|
|
|
|
|
// m_robotSwerveDrive.m_kinematics, xController, yController, thetaController,
|
|
|
|
|
// m_robotSwerveDrive::setModuleStates, m_robotSwerveDrive),
|
|
|
|
|
// new InstantCommand(m_robotSwerveDrive::stopModules)).withName("Run Autonomous Path");
|
|
|
|
|
// } else {
|
|
|
|
|
// LOGGER.severe("No auto selected.");
|
|
|
|
|
// return new RunCommand(() -> {
|
|
|
|
|
// }).withName("No Autonomous Path");
|
|
|
|
|
// }
|
2022-03-20 19:31:42 -06:00
|
|
|
|
|
|
|
|
// ! this will run each of the specified PathPlanner paths in sequence.
|
|
|
|
|
// * return new SequentialCommandGroup(buildAuto(5.0, 5.0, "Path1", "Path2", "Path3"));
|
|
|
|
|
|
|
|
|
|
// ! this will run each of the specified PathPlanner paths in sequence, while simultaneously running the intake throughout all the paths.
|
|
|
|
|
// * return new ParallelCommandGroup(buildAuto(null,
|
|
|
|
|
// * null,
|
|
|
|
|
// * new SequentialCommandGroup(buildAuto(5.0, 5.0, "Path1", "Path2", "Path3")),
|
|
|
|
|
// * new RunCommand(() -> m_robotIntake.runAtOutput(0.5))));
|
|
|
|
|
|
2022-03-20 21:21:45 -06:00
|
|
|
return new SequentialCommandGroup(buildAuto(1.0, 1.0, new InstantCommand(() -> m_robotSwerveDrive.m_gyro.reset()), "Move Forward")); // test command
|
2022-01-11 11:05:52 -07:00
|
|
|
}
|
|
|
|
|
|
2022-03-20 15:24:48 -06:00
|
|
|
public static XboxController getDriverController() {
|
2022-01-11 11:05:52 -07:00
|
|
|
return m_driverXbox;
|
|
|
|
|
}
|
|
|
|
|
|
2022-03-20 15:24:48 -06:00
|
|
|
public static XboxController getOperatorController() {
|
2022-03-12 17:33:38 -07:00
|
|
|
return m_operatorXbox;
|
|
|
|
|
}
|
|
|
|
|
|
2022-03-12 19:02:37 -07:00
|
|
|
public ButtonBox getButtonBox() {
|
|
|
|
|
return m_buttonBox;
|
2022-03-12 17:33:38 -07:00
|
|
|
}
|
|
|
|
|
|
2022-03-12 22:19:42 -07:00
|
|
|
public static void setSoftLimits(boolean set) {
|
|
|
|
|
softLimits = set;
|
|
|
|
|
}
|
|
|
|
|
|
2022-01-11 11:05:52 -07:00
|
|
|
/**
|
2022-02-17 19:52:05 -07:00
|
|
|
* Get odometry.
|
2022-03-05 11:12:33 -07:00
|
|
|
*
|
2022-02-17 19:52:05 -07:00
|
|
|
* @return Odometry
|
2022-01-11 11:05:52 -07:00
|
|
|
*/
|
2022-03-18 14:09:25 -06:00
|
|
|
public Pose2d getOdometry() {
|
|
|
|
|
return m_robotSwerveDrive.getOdometry();
|
|
|
|
|
}
|
2022-01-11 11:05:52 -07:00
|
|
|
|
|
|
|
|
/**
|
2022-02-17 19:52:05 -07:00
|
|
|
* Set odometry to given pose.
|
2022-03-05 11:12:33 -07:00
|
|
|
*
|
2022-02-17 19:52:05 -07:00
|
|
|
* @param pose Pose to set odometry to.
|
2022-01-11 11:05:52 -07:00
|
|
|
*/
|
2022-03-18 14:09:25 -06:00
|
|
|
public void resetOdometry(Pose2d pose) {
|
|
|
|
|
m_robotSwerveDrive.resetOdometry(pose);
|
|
|
|
|
}
|
2022-02-11 18:53:13 -07:00
|
|
|
|
2022-01-11 11:05:52 -07:00
|
|
|
/**
|
2022-03-05 11:12:33 -07:00
|
|
|
* Creates a WatchKey for the path planner directory and registers it with the
|
|
|
|
|
* WatchService.
|
|
|
|
|
* Then creates a NotifierCommand that will update the auto chooser with the
|
|
|
|
|
* latest path files.
|
2022-03-03 23:56:08 -07:00
|
|
|
* Finally, adds the existing path files to the auto chooser
|
|
|
|
|
*/
|
2022-03-18 14:09:25 -06:00
|
|
|
private void autoInit() {
|
2022-03-18 14:13:37 -06:00
|
|
|
try {
|
|
|
|
|
WatchKey watchKey = PATHPLANNER_DIRECTORY.register(FileSystems.getDefault().newWatchService(),
|
|
|
|
|
StandardWatchEventKinds.ENTRY_CREATE, StandardWatchEventKinds.ENTRY_MODIFY,
|
|
|
|
|
StandardWatchEventKinds.ENTRY_DELETE);
|
|
|
|
|
// TODO: Store this and other commands as fields so they can be rescheduled.
|
|
|
|
|
new NotifierCommand(() -> updateAutoChooser(watchKey), 0.5) {
|
|
|
|
|
@Override
|
|
|
|
|
public boolean runsWhenDisabled() {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}.withName("Path Watcher").schedule();
|
|
|
|
|
} catch (IOException exception) {
|
|
|
|
|
LOGGER.log(Level.SEVERE, "Exception with path file watcher.", exception);
|
|
|
|
|
}
|
|
|
|
|
Arrays.stream(PATHPLANNER_DIRECTORY.toFile().listFiles())
|
|
|
|
|
.filter(file -> file.getName().endsWith(".path")).sorted(Comparator.comparingLong(File::lastModified))
|
|
|
|
|
.forEachOrdered(file -> autoChooser.addOption(file.getName(), file));
|
|
|
|
|
SmartDashboard.putData("Auto Chooser", autoChooser);
|
2022-03-18 14:09:25 -06:00
|
|
|
}
|
2022-02-16 22:31:00 -07:00
|
|
|
|
2022-03-03 23:56:08 -07:00
|
|
|
/**
|
2022-03-05 11:12:33 -07:00
|
|
|
* Creates a button on the SmartDashboard that will record the path of the
|
|
|
|
|
* robot.
|
2022-01-11 11:05:52 -07:00
|
|
|
*/
|
2022-03-10 17:44:27 -07:00
|
|
|
|
2022-02-16 22:31:00 -07:00
|
|
|
public void recordInit() {
|
|
|
|
|
SmartDashboard.putData("Recording",
|
|
|
|
|
new RunCommand(this::recordPeriodic) {
|
|
|
|
|
@Override
|
|
|
|
|
public void end(boolean interupted) {
|
|
|
|
|
new InstantCommand(RobotContainer.this::saveRecording) {
|
|
|
|
|
@Override
|
|
|
|
|
public boolean runsWhenDisabled() {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}.withName("Save Recording").schedule();
|
|
|
|
|
}
|
|
|
|
|
}.withName("Record Path (Cancel to Save)"));
|
|
|
|
|
}
|
2022-03-10 17:44:27 -07:00
|
|
|
|
2022-03-03 23:56:08 -07:00
|
|
|
/**
|
|
|
|
|
* Called when a file is created, modified, or deleted.
|
|
|
|
|
* Adds newly created .path files to the SendableChooser.
|
|
|
|
|
* Reloads the path if the currently selected file is modified.
|
2022-03-05 11:12:33 -07:00
|
|
|
*
|
2022-03-03 23:56:08 -07:00
|
|
|
* @param watchKey The WatchKey that is being observed.
|
|
|
|
|
*/
|
2022-03-01 11:57:49 -07:00
|
|
|
private void updateAutoChooser(WatchKey watchKey) {
|
2022-03-18 15:09:08 -06:00
|
|
|
List<WatchEvent<?>> watchEvents = watchKey.pollEvents();
|
|
|
|
|
if (!watchEvents.isEmpty()) {
|
|
|
|
|
List<WatchEvent<?>> pathWatchEvents = watchEvents.stream()
|
|
|
|
|
.filter(e -> e.kind().type().isAssignableFrom(Path.class)).collect(Collectors.toList());
|
|
|
|
|
for (WatchEvent<?> pathWatchEvent : pathWatchEvents) {
|
|
|
|
|
Path watchEventPath = (Path) pathWatchEvent.context();
|
|
|
|
|
File watchEventFile = watchEventPath.toFile();
|
|
|
|
|
String watchEventFileName = watchEventFile.getName();
|
|
|
|
|
if (watchEventFileName.endsWith(".path")) {
|
|
|
|
|
if (pathWatchEvent.kind().equals(StandardWatchEventKinds.ENTRY_CREATE)) {
|
|
|
|
|
LOGGER.log(Level.WARNING, "PathPlanner file {0} created. Options added to SendableChooser.",
|
|
|
|
|
watchEventFileName);
|
|
|
|
|
autoChooser.addOption(watchEventFile.getName(), watchEventFile);
|
|
|
|
|
} else if (pathWatchEvent.kind().equals(StandardWatchEventKinds.ENTRY_MODIFY)) {
|
|
|
|
|
LOGGER.log(Level.WARNING, "PathPlanner file {0} modified.", watchEventFileName);
|
|
|
|
|
if (watchEventFileName.equals(autoChooser.getSelected().getName())) {
|
|
|
|
|
LOGGER.log(Level.SEVERE, "PathPlanner file {0} already selected. Reloading path.", watchEventFileName);
|
|
|
|
|
loadPath(watchEventFileName);
|
|
|
|
|
}
|
|
|
|
|
} else if (pathWatchEvent.kind().equals(StandardWatchEventKinds.ENTRY_DELETE)) {
|
|
|
|
|
LOGGER.log(Level.SEVERE,
|
|
|
|
|
"PathPlanner file {0} deleted. Removing options from SendableChooser not yet implemented.",
|
|
|
|
|
watchEventFileName);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (!watchKey.reset())
|
|
|
|
|
LOGGER.severe("File watch key invalid.");
|
2022-03-01 11:57:49 -07:00
|
|
|
}
|
2022-03-10 17:44:27 -07:00
|
|
|
|
2022-03-01 11:57:49 -07:00
|
|
|
private void loadPath(String pathName) {
|
|
|
|
|
LOGGER.warning("Loading path " + pathName);
|
|
|
|
|
loadedPathTrajectory = null;
|
2022-03-05 11:12:33 -07:00
|
|
|
loadedPathTrajectory = PathPlanner.loadPath(PATH_EXTENSION_REMOVER.apply(Objects.requireNonNullElse(pathName, "")),
|
|
|
|
|
SwerveDriveConstants.PATH_MAX_VELOCITY, SwerveDriveConstants.PATH_MAX_ACCELERATION);
|
2022-03-01 11:57:49 -07:00
|
|
|
LOGGER.info("Done loading");
|
|
|
|
|
}
|
2022-03-10 17:44:27 -07:00
|
|
|
|
2022-02-16 22:31:00 -07:00
|
|
|
private void saveRecording() {
|
2022-03-18 15:09:08 -06:00
|
|
|
// IMPORTANT: Had to chown the pathplanner folder in order to save autos.
|
|
|
|
|
File outputFile = PATHPLANNER_DIRECTORY
|
|
|
|
|
.resolve(ZonedDateTime.now(SYSTEM_CLOCK).format(RECORDING_FILE_NAME_FORMATTER)).toFile();
|
|
|
|
|
LOGGER.log(Level.WARNING, "Creating path {0}.", outputFile.getPath());
|
|
|
|
|
if (!pathPoints.isEmpty() && Boolean.TRUE.equals(Errors.log().getWithDefault(outputFile::createNewFile, false))) {
|
|
|
|
|
// TODO: Change to use measured maximum velocity and acceleration.
|
|
|
|
|
var path = createPath(null, null, false);
|
|
|
|
|
if (RobotBase.isReal())
|
|
|
|
|
path.write(outputFile);
|
|
|
|
|
StringWriter writer = new StringWriter();
|
|
|
|
|
path.write(writer);
|
|
|
|
|
recordingNetworkTable.getEntry(outputFile.getName()).setString(writer.toString());
|
|
|
|
|
autoChooser.setDefaultOption(outputFile.getName(), outputFile);
|
|
|
|
|
LOGGER.log(Level.INFO, "Recorded path to {0}.", outputFile.getPath());
|
|
|
|
|
} else
|
|
|
|
|
LOGGER.log(Level.SEVERE, "Unable to record path to {0}", outputFile.getPath());
|
2022-02-16 22:31:00 -07:00
|
|
|
}
|
2022-03-10 17:44:27 -07:00
|
|
|
|
2022-02-16 22:31:00 -07:00
|
|
|
public void recordPeriodic() {
|
|
|
|
|
Translation2d position = m_robotSwerveDrive.m_poseEstimator.getEstimatedPosition().getTranslation();
|
|
|
|
|
Rotation2d rotation = m_robotSwerveDrive.m_gyro.getRotation2d();
|
2022-03-05 11:12:33 -07:00
|
|
|
// FIXME: Chassis speeds are created from joystick inputs and do not reflect
|
|
|
|
|
// actual robot velocity.
|
2022-03-06 18:02:04 -07:00
|
|
|
Translation2d velocity = new Translation2d(m_robotSwerveDrive.getChassisSpeeds()[0],
|
|
|
|
|
m_robotSwerveDrive.getChassisSpeeds()[1]);
|
2022-03-05 11:12:33 -07:00
|
|
|
Waypoint waypoint = new Waypoint(position, position, position, rotation.getDegrees(), false,
|
|
|
|
|
SwerveDriveConstants.PATH_RECORD_VELOCITY ? velocity.getNorm() : null, false);
|
2022-02-16 22:31:00 -07:00
|
|
|
pathPoints.add(waypoint);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public PathPlannerUtil.Path createPath(Double maxVelocity, Double maxAcceleration, Boolean isReversed) {
|
2022-03-05 11:12:33 -07:00
|
|
|
// Remove points whose angles to neighboring points are less than 10 degrees
|
|
|
|
|
// apart.
|
2022-02-27 22:59:32 -07:00
|
|
|
int j = 0;
|
|
|
|
|
for (int i = 1; i < pathPoints.size() - 1; i++) {
|
|
|
|
|
var prev = pathPoints.get(j).anchorPoint.orElseThrow();
|
|
|
|
|
var current = pathPoints.get(i).anchorPoint.orElseThrow();
|
|
|
|
|
var next = pathPoints.get(i + 1).anchorPoint.orElseThrow();
|
|
|
|
|
var fromPrevious = current.minus(prev);
|
|
|
|
|
var toNext = next.minus(current);
|
|
|
|
|
var angleFromPrevious = new Rotation2d(fromPrevious.getX(), fromPrevious.getY());
|
|
|
|
|
var angleToNext = new Rotation2d(toNext.getX(), toNext.getY());
|
2022-03-05 11:12:33 -07:00
|
|
|
if (Math.abs(angleFromPrevious.minus(angleToNext).getDegrees()) < SwerveDriveConstants.MIN_WAYPOINT_ANGLE
|
|
|
|
|
|| (next.getDistance(prev) < SwerveDriveConstants.MIN_WAYPOINT_DISTANCE
|
|
|
|
|
&& pathPoints.get(i).velOverride.map(v -> v < SwerveDriveConstants.MIN_WAYPOINT_VELOCITY).orElse(false)))
|
2022-02-27 22:59:32 -07:00
|
|
|
pathPoints.set(i, null);
|
2022-02-28 18:57:53 -07:00
|
|
|
else
|
|
|
|
|
j = i;
|
2022-02-27 22:59:32 -07:00
|
|
|
}
|
|
|
|
|
pathPoints.removeIf(Objects::isNull);
|
|
|
|
|
// Make control points
|
2022-03-05 11:12:33 -07:00
|
|
|
pathPoints.get(0).nextControl = Optional.of(makeControlPoints(null, pathPoints.get(0).anchorPoint.orElseThrow(),
|
|
|
|
|
pathPoints.get(1).anchorPoint.orElseThrow()).getSecond());
|
2022-02-27 22:59:32 -07:00
|
|
|
for (int i = 1; i < pathPoints.size() - 1; i++) {
|
2022-03-05 11:12:33 -07:00
|
|
|
var controls = makeControlPoints(pathPoints.get(i - 1).anchorPoint.orElseThrow(),
|
|
|
|
|
pathPoints.get(i).anchorPoint.orElseThrow(), pathPoints.get(i + 1).anchorPoint.orElseThrow());
|
2022-02-27 22:59:32 -07:00
|
|
|
pathPoints.get(i).prevControl = Optional.of(controls.getFirst());
|
|
|
|
|
pathPoints.get(i).nextControl = Optional.of(controls.getSecond());
|
|
|
|
|
}
|
2022-03-05 11:12:33 -07:00
|
|
|
pathPoints.get(pathPoints.size() - 1).prevControl = Optional
|
|
|
|
|
.of(makeControlPoints(pathPoints.get(pathPoints.size() - 2).anchorPoint.orElseThrow(),
|
|
|
|
|
pathPoints.get(pathPoints.size() - 1).anchorPoint.orElseThrow(), null).getFirst());
|
2022-02-27 22:59:32 -07:00
|
|
|
// Create the path
|
2022-02-16 22:31:00 -07:00
|
|
|
PathPlannerUtil.Path path = new PathPlannerUtil.Path();
|
|
|
|
|
path.waypoints = Optional.ofNullable(pathPoints.toArray(PathPlannerUtil.Path.Waypoint[]::new));
|
|
|
|
|
path.maxVelocity = Optional.ofNullable(maxVelocity);
|
|
|
|
|
path.maxAcceleration = Optional.ofNullable(maxAcceleration);
|
|
|
|
|
path.isReversed = Optional.ofNullable(isReversed);
|
|
|
|
|
pathPoints.clear();
|
|
|
|
|
return path;
|
|
|
|
|
}
|
2022-02-27 22:59:32 -07:00
|
|
|
|
2022-03-05 11:12:33 -07:00
|
|
|
private static Pair<Translation2d, Translation2d> makeControlPoints(Translation2d prev, Translation2d current,
|
|
|
|
|
Translation2d next) {
|
2022-02-27 22:59:32 -07:00
|
|
|
var line = Objects.requireNonNullElse(next, current).minus(Objects.requireNonNullElse(prev, current)).div(4);
|
|
|
|
|
return Pair.of(current.minus(line), current.plus(line));
|
|
|
|
|
}
|
2021-11-15 16:26:16 -07:00
|
|
|
}
|