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-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-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 ;
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-05 14:38:54 -07:00
import frc4388.robot.commands.RunClaw ;
2022-03-16 14:49:41 -06:00
import frc4388.robot.subsystems.ClimberRewrite ;
2022-03-05 15:15:41 -07:00
import frc4388.robot.subsystems.Claws.ClawType ;
2022-02-24 18:12:57 -07:00
import frc4388.robot.commands.AimToCenter ;
2022-03-05 11:07:54 -07:00
import frc4388.robot.commands.Shoot ;
2022-03-06 00:24:51 -07:00
import frc4388.robot.commands.TrackTarget ;
2022-01-20 18:08:05 -07:00
import frc4388.robot.subsystems.BoomBoom ;
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-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-01-11 11:05:52 -07:00
/* RobotMap */
private final RobotMap m_robotMap = new RobotMap ( ) ;
/* Subsystems */
2022-03-16 14:49:41 -06:00
private final ClimberRewrite m_robotClimber = new ClimberRewrite ( m_robotMap . shoulder , m_robotMap . elbow , m_robotMap . gyro , false ) ;
2022-03-05 15:15:41 -07:00
2022-03-10 17:06:17 -07:00
private final Claws m_robotClaws = new Claws ( m_robotMap . leftClaw , m_robotMap . rightClaw ) ;
2022-01-21 17:20:31 -07:00
2022-03-10 17:44:27 -07:00
// public final SwerveDrive m_robotSwerveDrive = new SwerveDrive(m_robotMap.leftFront, m_robotMap.leftBack,
// m_robotMap.rightFront, m_robotMap.rightBack, m_robotMap.gyro);
// private final Serializer m_robotSerializer = new Serializer(m_robotMap.serializerBelt, /*m_robotMap.serializerShooterBelt,*/ m_robotMap.serializerBeam);
// private final Intake m_robotIntake = new Intake(m_robotMap.intakeMotor, m_robotMap.extenderMotor, m_robotSerializer);
// private final Storage m_robotStorage = new Storage(m_robotMap.storageMotor, m_robotMap.beamIntake, m_robotMap.beamShooter);
// //private final LED m_robotLED = new LED(m_robotMap.LEDController);
// private final BoomBoom m_robotBoomBoom = new BoomBoom(m_robotMap.shooterFalconLeft, m_robotMap.shooterFalconRight);
// private final Hood m_robotHood = new Hood(m_robotMap.angleAdjusterMotor);
// private final Turret m_robotTurret = new Turret(m_robotMap.shooterTurret);
// private final VisionOdometry m_robotVisionOdometry = new VisionOdometry(m_robotSwerveDrive, m_robotTurret);
2022-03-05 22:57:55 -07:00
2022-03-05 23:50:32 -07:00
2022-01-11 11:05:52 -07:00
/* 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
2022-03-01 20:23:01 -07:00
/* Autonomous */
2022-02-16 22:31:00 -07:00
private PathPlannerTrajectory loadedPathTrajectory = null ;
2022-03-10 17:44:27 -07: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-01-11 11:05:52 -07:00
/**
* The container for the robot. Contains subsystems, OI devices, and commands.
*/
public RobotContainer ( ) {
configureButtonBindings ( ) ;
/* Default Commands */
2022-01-20 17:10:07 -07:00
// moves climber in xy space with two-axis input from the operator controller
2022-03-16 22:32:57 -06:00
// m_robotClimber.setDefaultCommand(
// new RunCommand(() -> m_robotClimber.setMotors(getOperatorController().getLeftX() * 0.4, getOperatorController().getLeftY() * 0.4),
// m_robotClimber));
2022-03-06 12:57:41 -07:00
2022-03-10 17:06:17 -07:00
2022-03-06 12:57:41 -07:00
// IK command
// m_robotClimber.setDefaultCommand(
// new RunCommand(() -> m_robotClimber.controlWithInput(getOperatorController().getLeftX(),
// getOperatorController().getLeftY()), m_robotClimber).withName("Climber controlWithInput defaultCommand"));
2022-03-05 15:15:41 -07:00
2022-03-05 11:12:33 -07:00
// Turret default command
2022-03-06 00:24:51 -07:00
//m_robotTurret.setDefaultCommand(new AimToCenter(m_robotTurret, m_robotSwerveDrive, m_robotVisionOdometry));
2022-03-10 17:44:27 -07:00
// m_robotTurret.setDefaultCommand(new RunCommand(() -> m_robotTurret.runTurretWithInput(getOperatorController().getLeftX()), m_robotTurret));
// //Swerve Drive
// m_robotSwerveDrive.setDefaultCommand(
// new RunCommand(() -> m_robotSwerveDrive.driveWithInput(
// getDriverController().getLeftX(),
// getDriverController().getLeftY(),
// getDriverController().getRightX(),
// getDriverController().getRightY(),
// true),
// m_robotSwerveDrive).withName("Swerve driveWithInput defaultCommand"));
// //Intake with Triggers
// m_robotIntake.setDefaultCommand(
// new RunCommand(() -> m_robotIntake.runWithTriggers(
// getOperatorController().getLeftTriggerAxis(),
// getOperatorController().getRightTriggerAxis()),
// m_robotIntake).withName("Intake runWithTriggers defaultCommand"));
// //Storage Management
// m_robotStorage.setDefaultCommand(
// new RunCommand(() -> m_robotStorage.manageStorage(),
// m_robotStorage).withName("Storage manageStorage defaultCommand"));
// //Serializer Management
// m_robotSerializer.setDefaultCommand(
// new RunCommand(() -> m_robotSerializer.setSerializerStateWithBeam(),
// m_robotSerializer).withName("Serializer setSerializerStateWithBeam defaultCommand"));
2022-01-20 17:10:07 -07:00
2022-03-05 23:47:24 -07:00
2022-01-11 11:05:52 -07:00
// continually sends updates to the Blinkin LED controller to keep the lights on
2022-03-05 15:15:41 -07:00
// m_robotLED
// .setDefaultCommand(new RunCommand(m_robotLED::updateLED,
// m_robotLED).withName("LED update defaultCommand"));
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-01-11 11:05:52 -07:00
/* Driver Buttons */
2022-02-28 19:45:29 -07:00
// "XboxController.Button.kBack" was undefined yet, 7 works just fine
2022-03-10 17:44:27 -07:00
// new JoystickButton(getDriverController(), XboxController.Button.kBack.value)
// .whenPressed(m_robotSwerveDrive::resetGyro);
2022-01-11 11:05:52 -07:00
2022-03-10 17:44:27 -07:00
// new JoystickButton(getDriverController(), XboxController.Button.kLeftBumper.value)
// // new XboxControllerRawButton(m_driverXbox,
// // XboxControllerRaw.LEFT_BUMPER_BUTTON)
// .whenPressed(() -> m_robotSwerveDrive.highSpeed(false));
2022-01-15 14:55:07 -07:00
2022-03-10 17:44:27 -07:00
// new JoystickButton(getDriverController(), XboxController.Button.kRightBumper.value)
// // new XboxControllerRawButton(m_driverXbox,
// // XboxControllerRaw.RIGHT_BUMPER_BUTTON)
// .whenPressed(() -> m_robotSwerveDrive.highSpeed(true));
2022-02-11 18:53:13 -07:00
2022-03-10 17:44:27 -07:00
// new JoystickButton(getDriverController(), XboxController.Button.kA.value)
// .whenPressed(() -> resetOdometry(new Pose2d(0, 0, new Rotation2d(0))));
2022-03-05 22:57:55 -07:00
2022-03-10 17:44:27 -07:00
// new JoystickButton(getDriverController(), XboxController.Button.kX.value) //Temp
// .whenPressed(() -> m_robotMap.leftFront.reset())
// .whenPressed(() -> m_robotMap.rightFront.reset())
// .whenPressed(() -> m_robotMap.leftBack.reset())
// .whenPressed(() -> m_robotMap.rightBack.reset());
2022-03-05 14:38:54 -07:00
2022-01-11 11:05:52 -07:00
/* Operator Buttons */
2022-03-05 14:38:54 -07:00
// run claws
2022-03-16 20:16:32 -06:00
// new JoystickButton(getOperatorController(), XboxController.Button.kY.value)
// .whileHeld(new RunCommand(() -> m_robotClaws.runClaw(ClawType.LEFT, 0.2)))
// .whenReleased(new RunCommand(() -> m_robotClaws.runClaw(ClawType.LEFT, 0.0)));
2022-03-05 14:38:54 -07:00
2022-03-16 20:16:32 -06:00
// new JoystickButton(getOperatorController(), XboxController.Button.kX.value)
// .whileHeld(new RunCommand(() -> m_robotClaws.runClaw(ClawType.LEFT, -0.2)))
// .whenReleased(new RunCommand(() -> m_robotClaws.runClaw(ClawType.LEFT, 0.0)));
2022-03-10 17:06:17 -07:00
2022-03-16 20:16:32 -06:00
// new JoystickButton(getOperatorController(), XboxController.Button.kB.value)
// .whileHeld(new RunCommand(() -> m_robotClaws.runClaw(ClawType.RIGHT, 0.2)))
// .whenReleased(new RunCommand(() -> m_robotClaws.runClaw(ClawType.RIGHT, 0.0)));
2022-03-10 17:06:17 -07:00
2022-03-16 20:16:32 -06:00
// new JoystickButton(getOperatorController(), XboxController.Button.kA.value)
// .whileHeld(new RunCommand(() -> m_robotClaws.runClaw(ClawType.RIGHT, -0.2)))
// .whenReleased(new RunCommand(() -> m_robotClaws.runClaw(ClawType.RIGHT, 0.0)));
new JoystickButton ( getOperatorController ( ) , XboxController . Button . kY . value )
2022-03-16 22:32:57 -06:00
. whileHeld ( new RunCommand ( ( ) - > m_robotClaws . setOpen ( true ) ) ) ;
2022-03-16 20:16:32 -06:00
new JoystickButton ( getOperatorController ( ) , XboxController . Button . kB . value )
2022-03-16 22:32:57 -06:00
. whileHeld ( new RunCommand ( ( ) - > m_robotClaws . setOpen ( false ) ) ) ;
2022-03-10 17:44:27 -07:00
}
2022-03-05 14:38:54 -07:00
2022-03-05 11:12:33 -07:00
/*
* // activates "BoomBoom"
2022-03-05 11:22:35 -07:00
* new JoystickButton(getOperatorController(), XboxController.Button.kA.value)
* .whenPressed(new Shoot(m_robotSwerveDrive, m_robotBoomBoom, m_robotTurret,
* m_robotHood));
2022-03-05 11:12:33 -07:00
*/
2022-03-05 22:57:55 -07:00
//Extender
2022-03-10 17:44:27 -07:00
// new JoystickButton(getOperatorController(), XboxController.Button.kX.value)
// .whenPressed(() -> m_robotIntake.runExtender(true));
2022-03-05 22:57:55 -07:00
2022-03-10 17:44:27 -07:00
// new JoystickButton(getOperatorController(), XboxController.Button.kY.value)
// .whenPressed(() -> m_robotIntake.runExtender(false));
2022-03-05 22:57:55 -07:00
2022-03-10 17:44:27 -07:00
// //Storage
// new JoystickButton(getOperatorController(), XboxController.Button.kRightBumper.value)
// .whenPressed(() -> m_robotStorage.runStorage(StorageConstants.STORAGE_SPEED))
// .whenReleased(() -> m_robotStorage.runStorage(0.0));
2022-03-05 22:57:55 -07:00
2022-03-10 17:44:27 -07:00
// new JoystickButton(getOperatorController(), XboxController.Button.kLeftBumper.value)
// .whenPressed(() -> m_robotStorage.runStorage(-StorageConstants.STORAGE_SPEED))
// .whenReleased(() -> m_robotStorage.runStorage(0.0));
2022-03-06 00:24:51 -07:00
2022-03-10 17:44:27 -07:00
// //Shooter
// new JoystickButton(getOperatorController(), XboxController.Button.kA.value)
// .whenPressed(new Shoot(m_robotSwerveDrive, m_robotBoomBoom, m_robotTurret, m_robotHood));
2022-03-06 00:24:51 -07:00
2022-03-10 17:44:27 -07:00
// new JoystickButton(getOperatorController(), XboxController.Button.kB.value)
// .whenPressed(new TrackTarget(m_robotTurret, m_robotBoomBoom, m_robotHood, m_robotSwerveDrive, m_robotVisionOdometry));
// }
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");
// }
return null ;
2022-01-11 11:05:52 -07:00
}
2022-01-29 01:16:58 -07:00
public XboxController getDriverController ( ) {
2022-01-11 11:05:52 -07:00
return m_driverXbox ;
}
/**
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-10 17:44:27 -07: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-10 17:44:27 -07:00
// public void resetOdometry(Pose2d pose) {
// m_robotSwerveDrive.resetOdometry(pose);
// }
2022-02-11 18:53:13 -07:00
2022-01-29 01:16:58 -07:00
public XboxController getOperatorController ( ) {
2022-01-11 11:05:52 -07:00
return m_operatorXbox ;
}
/**
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-10 17:44:27 -07:00
// private void autoInit() {
// 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-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
// 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)"));
// }
// /**
// * 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.
// *
// * @param watchKey The WatchKey that is being observed.
// */
// private void updateAutoChooser(WatchKey watchKey) {
// 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.");
// }
// private void loadPath(String pathName) {
// LOGGER.warning("Loading path " + pathName);
// loadedPathTrajectory = null;
// loadedPathTrajectory = PathPlanner.loadPath(PATH_EXTENSION_REMOVER.apply(Objects.requireNonNullElse(pathName, "")),
// SwerveDriveConstants.PATH_MAX_VELOCITY, SwerveDriveConstants.PATH_MAX_ACCELERATION);
// LOGGER.info("Done loading");
// }
// private void saveRecording() {
// // 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());
// }
// // public void recordPeriodic() {
// // Translation2d position = m_robotSwerveDrive.m_poseEstimator.getEstimatedPosition().getTranslation();
// // Rotation2d rotation = m_robotSwerveDrive.m_gyro.getRotation2d();
// // // FIXME: Chassis speeds are created from joystick inputs and do not reflect
// // // actual robot velocity.
// // Translation2d velocity = new Translation2d(m_robotSwerveDrive.chassisSpeeds.vxMetersPerSecond,
// // m_robotSwerveDrive.chassisSpeeds.vyMetersPerSecond);
// // Waypoint waypoint = new Waypoint(position, position, position, rotation.getDegrees(), false,
// // SwerveDriveConstants.PATH_RECORD_VELOCITY ? velocity.getNorm() : null, false);
// // pathPoints.add(waypoint);
// // }
// public PathPlannerUtil.Path createPath(Double maxVelocity, Double maxAcceleration, Boolean isReversed) {
// // Remove points whose angles to neighboring points are less than 10 degrees
// // apart.
// 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());
// 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)))
// pathPoints.set(i, null);
// else
// j = i;
// }
// pathPoints.removeIf(Objects::isNull);
// // Make control points
// pathPoints.get(0).nextControl = Optional.of(makeControlPoints(null, pathPoints.get(0).anchorPoint.orElseThrow(),
// pathPoints.get(1).anchorPoint.orElseThrow()).getSecond());
// for (int i = 1; i < pathPoints.size() - 1; i++) {
// var controls = makeControlPoints(pathPoints.get(i - 1).anchorPoint.orElseThrow(),
// pathPoints.get(i).anchorPoint.orElseThrow(), pathPoints.get(i + 1).anchorPoint.orElseThrow());
// pathPoints.get(i).prevControl = Optional.of(controls.getFirst());
// pathPoints.get(i).nextControl = Optional.of(controls.getSecond());
// }
// 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());
// // Create the path
// 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;
// }
// private static Pair<Translation2d, Translation2d> makeControlPoints(Translation2d prev, Translation2d current,
// Translation2d next) {
// 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
}