commit d861e109fce18a59aeeb733ff6f91d96e7132395
Author: C4llSqin <83995467+C4llSqin@users.noreply.github.com>
Date: Mon Jan 8 11:38:08 2024 -0700
Initial commit
diff --git a/.gitattributes b/.gitattributes
new file mode 100644
index 0000000..dfe0770
--- /dev/null
+++ b/.gitattributes
@@ -0,0 +1,2 @@
+# Auto detect text files and perform LF normalization
+* text=auto
diff --git a/.github/workflows/gradle.yml b/.github/workflows/gradle.yml
new file mode 100644
index 0000000..9eaf0d0
--- /dev/null
+++ b/.github/workflows/gradle.yml
@@ -0,0 +1,23 @@
+name: Java CI
+
+on:
+ push:
+ branches:
+ - master
+ pull_request:
+ branches:
+ - master
+
+jobs:
+ build:
+ runs-on: ubuntu-latest
+ steps:
+ - uses: actions/checkout@v1
+ - name: Set up JDK 1.8
+ uses: actions/setup-java@v1
+ with:
+ java-version: 1.12
+ - name: Change wrapper permissions
+ run: chmod +x ./gradlew
+ - name: Build with Gradle
+ run: ./gradlew build
diff --git a/.gitignore b/.gitignore
new file mode 100644
index 0000000..a8d1911
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1,172 @@
+# This gitignore has been specially created by the WPILib team.
+# If you remove items from this file, intellisense might break.
+
+### C++ ###
+# Prerequisites
+*.d
+
+# Compiled Object files
+*.slo
+*.lo
+*.o
+*.obj
+
+# Precompiled Headers
+*.gch
+*.pch
+
+# Compiled Dynamic libraries
+*.so
+*.dylib
+*.dll
+
+# Fortran module files
+*.mod
+*.smod
+
+# Compiled Static libraries
+*.lai
+*.la
+*.a
+*.lib
+
+# Executables
+*.exe
+*.out
+*.app
+
+### Java ###
+# Compiled class file
+*.class
+
+# Log file
+*.log
+
+# BlueJ files
+*.ctxt
+
+# Mobile Tools for Java (J2ME)
+.mtj.tmp/
+
+# Package Files #
+*.jar
+*.war
+*.nar
+*.ear
+*.zip
+*.tar.gz
+*.rar
+
+# virtual machine crash logs, see http://www.java.com/en/download/help/error_hotspot.xml
+hs_err_pid*
+
+### Linux ###
+*~
+
+# temporary files which can be created if a process still has a handle open of a deleted file
+.fuse_hidden*
+
+# KDE directory preferences
+.directory
+
+# Linux trash folder which might appear on any partition or disk
+.Trash-*
+
+# .nfs files are created when an open file is removed but is still being accessed
+.nfs*
+
+### macOS ###
+# General
+.DS_Store
+.AppleDouble
+.LSOverride
+
+# Icon must end with two \r
+Icon
+
+# Thumbnails
+._*
+
+# Files that might appear in the root of a volume
+.DocumentRevisions-V100
+.fseventsd
+.Spotlight-V100
+.TemporaryItems
+.Trashes
+.VolumeIcon.icns
+.com.apple.timemachine.donotpresent
+
+# Directories potentially created on remote AFP share
+.AppleDB
+.AppleDesktop
+Network Trash Folder
+Temporary Items
+.apdisk
+
+### VisualStudioCode ###
+.vscode/*
+!.vscode/settings.json
+!.vscode/tasks.json
+!.vscode/launch.json
+!.vscode/extensions.json
+
+### Windows ###
+# Windows thumbnail cache files
+Thumbs.db
+ehthumbs.db
+ehthumbs_vista.db
+
+# Dump file
+*.stackdump
+
+# Folder config file
+[Dd]esktop.ini
+
+# Recycle Bin used on file shares
+$RECYCLE.BIN/
+
+# Windows Installer files
+*.cab
+*.msi
+*.msix
+*.msm
+*.msp
+
+# Windows shortcuts
+*.lnk
+
+### Gradle ###
+.gradle
+/build/
+
+# Ignore Gradle GUI config
+gradle-app.setting
+
+# Avoid ignoring Gradle wrapper jar file (.jar files are usually ignored)
+!gradle-wrapper.jar
+
+# Cache of project
+.gradletasknamecache
+
+# # Work around https://youtrack.jetbrains.com/issue/IDEA-116898
+# gradle/wrapper/gradle-wrapper.properties
+
+# # VS Code Specific Java Settings
+# DO NOT REMOVE .classpath and .project
+.classpath
+.project
+.settings/
+bin/
+
+# IntelliJ
+*.iml
+*.ipr
+*.iws
+.idea/
+out/
+
+# Fleet
+.fleet
+
+# Simulation GUI and other tools window save file
+*-window.json
diff --git a/.idea/.gitignore b/.idea/.gitignore
new file mode 100644
index 0000000..26d3352
--- /dev/null
+++ b/.idea/.gitignore
@@ -0,0 +1,3 @@
+# Default ignored files
+/shelf/
+/workspace.xml
diff --git a/.idea/gradle.xml b/.idea/gradle.xml
new file mode 100644
index 0000000..4edc9d5
--- /dev/null
+++ b/.idea/gradle.xml
@@ -0,0 +1,17 @@
+
+
It is advised to statically import this class (or one of its inner classes) wherever the + * constants are needed, to reduce verbosity. + */ +public final class Constants { + public static final class SwerveDriveConstants { + + public static final double MAX_ROT_SPEED = 1.5; + public static final double MIN_ROT_SPEED = 0.8; + public static double ROTATION_SPEED = MAX_ROT_SPEED; + public static double ROT_CORRECTION_SPEED = 10; // MIN_ROT_SPEED; + + public static final double CORRECTION_MIN = 10; + public static final double CORRECTION_MAX = 50; + + public static final double SLOW_SPEED = 0.8; + public static final double FAST_SPEED = 1.0; + public static final double TURBO_SPEED = 4.0; + + public static final class IDs { + public static final int LEFT_FRONT_WHEEL_ID = 2; + public static final int LEFT_FRONT_STEER_ID = 3; + public static final int LEFT_FRONT_ENCODER_ID = 10; + + public static final int RIGHT_FRONT_WHEEL_ID = 4; + public static final int RIGHT_FRONT_STEER_ID = 5; + public static final int RIGHT_FRONT_ENCODER_ID = 11; + + public static final int LEFT_BACK_WHEEL_ID = 6; + public static final int LEFT_BACK_STEER_ID = 7; + public static final int LEFT_BACK_ENCODER_ID = 12; + + public static final int RIGHT_BACK_WHEEL_ID = 8; + public static final int RIGHT_BACK_STEER_ID = 9; + public static final int RIGHT_BACK_ENCODER_ID = 13; + } + + public static final class PIDConstants { + public static final int SWERVE_SLOT_IDX = 0; + public static final int SWERVE_PID_LOOP_IDX = 1; + public static final Gains SWERVE_GAINS = new Gains(0.5, 0.0, 0.0, 0.0, 0, 1.0); + } + + public static final class AutoConstants { + public static final Gains X_CONTROLLER = new Gains(0.8, 0.0, 0.0); + public static final Gains Y_CONTROLLER = new Gains(0.8, 0.0, 0.0); + public static final Gains THETA_CONTROLLER = new Gains(-0.8, 0.0, 0.0); + public static final TrapezoidProfile.Constraints THETA_CONSTRAINTS = new TrapezoidProfile.Constraints(Math.PI/2, Math.PI/2); // TODO: tune + + public static final double PATH_MAX_VEL = 0.3; // TODO: find the actual value + public static final double PATH_MAX_ACC = 0.3; // TODO: find the actual value + } + + public static final class Conversions { + public static final int CANCODER_TICKS_PER_ROTATION = 4096; + + public static final double JOYSTICK_TO_METERS_PER_SECOND_FAST = 4.8; + public static final double JOYSTICK_TO_METERS_PER_SECOND_SLOW = 0.8; + + public static final double MOTOR_REV_PER_WHEEL_REV = 5.12; + public static final double MOTOR_REV_PER_STEER_REV = 12.8; + + public static final double TICKS_PER_MOTOR_REV = 2048; + public static final double WHEEL_DIAMETER_INCHES = 3.9; + public static final double INCHES_PER_WHEEL_REV = WHEEL_DIAMETER_INCHES * Math.PI; + + public static final double WHEEL_REV_PER_MOTOR_REV = 1 / MOTOR_REV_PER_WHEEL_REV; + public static final double TICKS_PER_WHEEL_REV = TICKS_PER_MOTOR_REV * MOTOR_REV_PER_WHEEL_REV; + public static final double TICKS_PER_INCH = TICKS_PER_WHEEL_REV / INCHES_PER_WHEEL_REV; + public static final double INCHES_PER_TICK = 1 / TICKS_PER_INCH; + + public static final double TICK_TIME_TO_SECONDS = 10; + public static final double SECONDS_TO_TICK_TIME = 1 / TICK_TIME_TO_SECONDS; + } + + public static final class Configurations { + public static final double OPEN_LOOP_RAMP_RATE = 0.2; // TODO: find the actual value + public static final double CLOSED_LOOP_RAMP_RATE = 0.2; // TODO: find the actual value + public static final double NEUTRAL_DEADBAND = 0.04; // TODO: find the actual value + } + + public static final double MAX_SPEED_FEET_PER_SECOND = 5; // TODO: find the actual value + public static final double MAX_ANGULAR_SPEED_FEET_PER_SECOND = 2 * 2 * Math.PI; // TODO: find the actual value + + // dimensions + public static final double WIDTH = 18.5; + public static final double HEIGHT = 18.5; + public static final double HALF_WIDTH = WIDTH / 2.d; + public static final double HALF_HEIGHT = HEIGHT / 2.d; + + // misc + public static final int TIMEOUT_MS = 30; + public static final int SMARTDASHBOARD_UPDATE_FRAME = 2; + } + + public static final class VisionConstants { + public static final String NAME = "photonCamera"; + + public static final int LIME_HIXELS = 640; + public static final int LIME_VIXELS = 480; + + public static final double H_FOV = 59.6; + public static final double V_FOV = 45.7; + + public static final double LIME_HEIGHT = 6.0; + public static final double LIME_ANGLE = 55.0; + + // public static final double HIGH_TARGET_HEIGHT = 46.0; + public static final double HIGH_TAPE_HEIGHT = 44.0; + + // public static final double MID_TARGET_HEIGHT = 34.0; + public static final double MID_TAPE_HEIGHT = 24.0; + + public static final double APRIL_HEIGHT = -1.0; // TODO: find actual value + + } + + public static final class DriveConstants { + public static final int DRIVE_PIGEON_ID = 6; + + public static final int SMARTDASHBOARD_UPDATE_FRAME = 2; + } + + public static final class LEDConstants { + public static final int LED_SPARK_ID = 0; + + public static final LEDPatterns DEFAULT_PATTERN = LEDPatterns.FOREST_WAVES; + } + + public static final class OIConstants { + public static final int XBOX_DRIVER_ID = 0; + public static final int XBOX_OPERATOR_ID = 1; + } +} diff --git a/src/main/java/frc4388/robot/Main.java b/src/main/java/frc4388/robot/Main.java new file mode 100644 index 0000000..ad2d494 --- /dev/null +++ b/src/main/java/frc4388/robot/Main.java @@ -0,0 +1,29 @@ +/*----------------------------------------------------------------------------*/ +/* Copyright (c) 2018 FIRST. All Rights Reserved. */ +/* Open Source Software - may be modified and shared by FRC teams. The code */ +/* must be accompanied by the FIRST BSD license file in the root directory of */ +/* the project. */ +/*----------------------------------------------------------------------------*/ + +package frc4388.robot; + +import edu.wpi.first.wpilibj.RobotBase; + +/** + * Do NOT add any static variables to this class, or any initialization at all. + * Unless you know what you are doing, do not modify this file except to + * change the parameter class to the startRobot call. + */ +public final class Main { + private Main() { + } + + /** + * Main initialization function. Do not perform any initialization here. + * + *
If you change your main robot class, change the parameter type. + */ + public static void main(String... args) { + RobotBase.startRobot(Robot::new); + } +} diff --git a/src/main/java/frc4388/robot/Robot.java b/src/main/java/frc4388/robot/Robot.java new file mode 100644 index 0000000..e555b09 --- /dev/null +++ b/src/main/java/frc4388/robot/Robot.java @@ -0,0 +1,131 @@ +/*----------------------------------------------------------------------------*/ +/* Copyright (c) 2017-2019 FIRST. All Rights Reserved. */ +/* Open Source Software - may be modified and shared by FRC teams. The code */ +/* must be accompanied by the FIRST BSD license file in the root directory of */ +/* the project. */ +/*----------------------------------------------------------------------------*/ + +package frc4388.robot; + +import edu.wpi.first.wpilibj.TimedRobot; +import edu.wpi.first.wpilibj2.command.Command; +import edu.wpi.first.wpilibj2.command.CommandScheduler; +import frc4388.utility.DeferredBlock; +import frc4388.utility.RobotTime; + +/** + * The VM is configured to automatically run this class, and to call the + * functions corresponding to each mode, as described in the TimedRobot + * documentation. If you change the name of this class or the package after + * creating this project, you must also update the build.gradle file in the + * project. + */ +public class Robot extends TimedRobot { + Command m_autonomousCommand; + + private RobotTime m_robotTime = RobotTime.getInstance(); + private RobotContainer m_robotContainer; + + /** + * This function is run when the robot is first started up and should be + * used for any initialization code. + */ + @Override + public void robotInit() { + // Instantiate our RobotContainer. This will perform all our button bindings, and put our + // autonomous chooser on the dashboard. + m_robotContainer = new RobotContainer(); + } + + /** + * This function is called every robot packet, no matter the mode. Use + * this for items like diagnostics that you want ran during disabled, + * autonomous, teleoperated and test. + * + *
This runs after the mode specific periodic functions, but before
+ * LiveWindow and SmartDashboard integrated updating.
+ */
+ @Override
+ public void robotPeriodic() {
+ m_robotTime.updateTimes();
+ // Runs the Scheduler. This is responsible for polling buttons, adding newly-scheduled
+ // commands, running already-scheduled commands, removing finished or interrupted commands,
+ // and running subsystem periodic() methods. This must be called from the robot's periodic
+ // block in order for anything in the Command-based framework to work.
+ CommandScheduler.getInstance().run();
+ }
+
+ /**
+ * This function is called once each time the robot enters Disabled mode.
+ * You can use it to reset any subsystem information you want to clear when
+ * the robot is disabled.
+ */
+ @Override
+ public void disabledInit() {
+ m_robotTime.endMatchTime();
+ }
+
+ @Override
+ public void disabledPeriodic() {
+ }
+
+ @Override
+ public void disabledExit() {
+ DeferredBlock.execute();
+ }
+
+ /**
+ * This autonomous runs the autonomous command selected by your {@link RobotContainer} class.
+ */
+ @Override
+ public void autonomousInit() {
+ m_autonomousCommand = m_robotContainer.getAutonomousCommand();
+
+ /*
+ * String autoSelected = SmartDashboard.getString("Auto Selector",
+ * "Default"); switch(autoSelected) { case "My Auto": autonomousCommand
+ * = new MyAutoCommand(); break; case "Default Auto": default:
+ * autonomousCommand = new ExampleCommand(); break; }
+ */
+
+ // schedule the autonomous command (example)
+ if (m_autonomousCommand != null) {
+ m_autonomousCommand.schedule();
+ }
+ m_robotTime.startMatchTime();
+ }
+
+ /**
+ * This function is called periodically during autonomous.
+ */
+ @Override
+ public void autonomousPeriodic() {
+ }
+
+ @Override
+ public void teleopInit() {
+ // This makes sure that the autonomous stops running when
+ // teleop starts running. If you want the autonomous to
+ // continue until interrupted by another command, remove
+ // this line or comment it out.
+ if (m_autonomousCommand != null) {
+ m_autonomousCommand.cancel();
+ }
+ m_robotTime.startMatchTime();
+ }
+
+ /**
+ * This function is called periodically during operator control.
+ */
+ @Override
+ public void teleopPeriodic() {
+
+ }
+
+ /**
+ * This function is called periodically during test mode.
+ */
+ @Override
+ public void testPeriodic() {
+ }
+}
diff --git a/src/main/java/frc4388/robot/RobotContainer.java b/src/main/java/frc4388/robot/RobotContainer.java
new file mode 100644
index 0000000..23c1b0f
--- /dev/null
+++ b/src/main/java/frc4388/robot/RobotContainer.java
@@ -0,0 +1,119 @@
+/*----------------------------------------------------------------------------*/
+/* Copyright (c) 2018-2019 FIRST. All Rights Reserved. */
+/* Open Source Software - may be modified and shared by FRC teams. The code */
+/* must be accompanied by the FIRST BSD license file in the root directory of */
+/* the project. */
+/*----------------------------------------------------------------------------*/
+
+package frc4388.robot;
+
+import edu.wpi.first.wpilibj.Joystick;
+import edu.wpi.first.wpilibj2.command.Command;
+import edu.wpi.first.wpilibj2.command.InstantCommand;
+import edu.wpi.first.wpilibj2.command.RunCommand;
+import edu.wpi.first.wpilibj2.command.button.JoystickButton;
+import frc4388.robot.Constants.*;
+import frc4388.robot.commands.Swerve.JoystickPlayback;
+import frc4388.robot.commands.Swerve.JoystickRecorder;
+import frc4388.robot.subsystems.LED;
+import frc4388.utility.LEDPatterns;
+import frc4388.utility.controller.IHandController;
+import frc4388.utility.controller.XboxController;
+
+/**
+ * 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 {
+ /* RobotMap */
+ private final RobotMap m_robotMap = new RobotMap();
+
+ /* Subsystems */
+ private final LED m_robotLED = new LED(m_robotMap.LEDController);
+
+ /* Controllers */
+ private final XboxController m_driverXbox = new XboxController(OIConstants.XBOX_DRIVER_ID);
+ private final XboxController m_operatorXbox = new XboxController(OIConstants.XBOX_OPERATOR_ID);
+
+ /**
+ * The container for the robot. Contains subsystems, OI devices, and commands.
+ */
+ public RobotContainer() {
+ configureButtonBindings();
+
+ /* Default Commands */
+ // drives the robot with a two-axis input from the driver controller
+ // continually sends updates to the Blinkin LED controller to keep the lights on
+ m_robotLED.setDefaultCommand(new RunCommand(() -> m_robotLED.updateLED(), m_robotLED));
+ }
+
+ /**
+ * 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 */
+ // test command to spin the robot while pressing A on the driver controller
+ // new JoystickButton(getDeadbandedDriverController(), XboxController.RIGHT_BUMPER_BUTTON)
+ // .whileTrue(new JoystickRecorder(m_robotSwerveDrive,
+ // () -> getDeadbandedDriverController().getLeftX(),
+ // () -> getDeadbandedDriverController().getLeftY(),
+ // () -> getDeadbandedDriverController().getRightX(),
+ // () -> getDeadbandedDriverController().getRightY(),
+ // "Blue1Path.txt"))
+ // .onFalse(new InstantCommand());
+
+ // new JoystickButton(getDeadbandedDriverController(), XboxController.LEFT_BUMPER_BUTTON)
+ // .onTrue(new JoystickPlayback(m_robotSwerveDrive, "Blue1Path.txt"))
+ // .onFalse(new InstantCommand());
+
+ /* Operator Buttons */
+ // activates "Lit Mode"
+ new JoystickButton(getOperatorJoystick(), XboxController.A_BUTTON)
+ .whenPressed(() -> m_robotLED.setPattern(LEDPatterns.LAVA_RAINBOW))
+ .whenReleased(() -> m_robotLED.setPattern(LEDConstants.DEFAULT_PATTERN));
+ }
+
+ /**
+ * Use this to pass the autonomous command to the main {@link Robot} class.
+ *
+ * @return the command to run in autonomous
+ */
+ public Command getAutonomousCommand() {
+ // no auto
+ return new InstantCommand();
+ }
+
+ /**
+ * Add your docs here.
+ */
+ public IHandController getDriverController() {
+ return m_driverXbox;
+ }
+
+ /**
+ * Add your docs here.
+ */
+ public IHandController getOperatorController() {
+ return m_operatorXbox;
+ }
+
+ /**
+ * Add your docs here.
+ */
+ public Joystick getOperatorJoystick() {
+ return m_operatorXbox.getJoyStick();
+ }
+
+ /**
+ * Add your docs here.
+ */
+ public Joystick getDriverJoystick() {
+ return m_driverXbox.getJoyStick();
+ }
+}
diff --git a/src/main/java/frc4388/robot/RobotMap.java b/src/main/java/frc4388/robot/RobotMap.java
new file mode 100644
index 0000000..5f17853
--- /dev/null
+++ b/src/main/java/frc4388/robot/RobotMap.java
@@ -0,0 +1,40 @@
+/*----------------------------------------------------------------------------*/
+/* Copyright (c) 2018-2019 FIRST. All Rights Reserved. */
+/* Open Source Software - may be modified and shared by FRC teams. The code */
+/* must be accompanied by the FIRST BSD license file in the root directory of */
+/* the project. */
+/*----------------------------------------------------------------------------*/
+
+package frc4388.robot;
+
+import com.ctre.phoenix.motorcontrol.InvertType;
+import com.ctre.phoenix.motorcontrol.NeutralMode;
+import com.ctre.phoenix.motorcontrol.can.WPI_TalonFX;
+import com.ctre.phoenix.sensors.PigeonIMU;
+
+import edu.wpi.first.wpilibj.motorcontrol.Spark;
+import edu.wpi.first.wpilibj.drive.DifferentialDrive;
+import frc4388.robot.Constants.LEDConstants;
+import frc4388.utility.RobotGyro;
+
+/**
+ * Defines and holds all I/O objects on the Roborio. This is useful for unit
+ * testing and modularization.
+ */
+public class RobotMap {
+
+ public RobotMap() {
+ configureLEDMotorControllers();
+ configureDriveMotorControllers();
+ }
+
+ /* LED Subsystem */
+ public final Spark LEDController = new Spark(LEDConstants.LED_SPARK_ID);
+
+ void configureLEDMotorControllers() {
+
+ }
+
+ void configureDriveMotorControllers() {
+ }
+}
diff --git a/src/main/java/frc4388/robot/commands/Autos/PlaybackChooser.java b/src/main/java/frc4388/robot/commands/Autos/PlaybackChooser.java
new file mode 100644
index 0000000..2438a38
--- /dev/null
+++ b/src/main/java/frc4388/robot/commands/Autos/PlaybackChooser.java
@@ -0,0 +1,91 @@
+package frc4388.robot.commands.Autos;
+
+import java.io.File;
+import java.util.ArrayList;
+import java.util.HashMap;
+
+import edu.wpi.first.wpilibj.shuffleboard.BuiltInWidgets;
+import edu.wpi.first.wpilibj.shuffleboard.ComplexWidget;
+import edu.wpi.first.wpilibj.shuffleboard.Shuffleboard;
+import edu.wpi.first.wpilibj.smartdashboard.SendableChooser;
+import edu.wpi.first.wpilibj2.command.Command;
+import edu.wpi.first.wpilibj2.command.InstantCommand;
+import frc4388.robot.commands.Swerve.JoystickPlayback;
+import frc4388.robot.subsystems.SwerveDrive;
+
+public class PlaybackChooser {
+ private final ArrayList NavX:
+ * Calibrate the gyro by running for a number of samples and computing the center value. Then use
+ * the center value as the Accumulator center value for subsequent measurements. It's important to
+ * make sure that the robot is not moving while the centering calculations are in progress, this
+ * is typically done when the robot is first turned on while it's sitting at rest before the
+ * competition starts.
+ *
+ * Pigeon:
+ * Calibrate the gyro by collecting data at a range of tempuratures. Allow pigeon to cool, then boot
+ * into calibration mode. For faster calibration, use a heat lamp to heat up the pigeon. Once the pigeon
+ * has seen a reasonable range of tempuratures, it will exit calibration mode. It's important to
+ * make sure that the robot is not moving while the tempurature calculations are in progress, this
+ * is typically done when the robot is first turned on while it's sitting at rest before the
+ * competition starts.
+ */
+ @Override
+ public void calibrate() {
+ if (m_isGyroAPigeon) {
+ m_pigeon.calibrate();
+ } else {
+ m_navX.calibrate();
+ }
+ }
+
+ @Override
+ public void reset() {
+ resetZeroValues();
+
+ if (m_isGyroAPigeon) {
+ m_pigeon.setYaw(0);
+ } else {
+ m_navX.reset();
+ }
+ }
+
+ /**
+ * Get Yaw, Pitch, and Roll data.
+ *
+ * @return ypr_deg Array with yaw[0], pitch[1], and roll[2] data.
+ * Yaw is within [-368,640, +368,640] degrees.
+ * Pitch is within [-90,+90] degrees.
+ * Roll is within [-90,+90] degrees.
+ */
+ private double[] getPigeonAngles() {
+ double[] ypr = new double[3];
+ m_pigeon.getYawPitchRoll(ypr);
+
+ return new double[] {ypr[0], (ypr[1] - pitchZero), (ypr[2] - rollZero)};
+ }
+
+ @Override
+ public double getAngle() {
+ if (m_isGyroAPigeon) {
+ return getPigeonAngles()[0];
+ } else {
+ return m_navX.getAngle();
+ }
+ }
+
+ public double getYaw() {
+ return this.getAngle();
+ }
+
+ /**
+ * Gets an absolute heading of the robot
+ * @return heading from -180 to 180 degrees
+ */
+ public double getHeading() {
+ return getHeading(getAngle());
+ }
+
+ /**
+ * Gets an absolute heading of the robot
+ * @return heading from -180 to 180 degrees
+ */
+ public double getHeading(double angle) {
+ return Math.IEEEremainder(angle, 360);
+ }
+
+ /**
+ * Returns the current pitch value (in degrees, from -90 to 90)
+ * reported by the sensor. Pitch is a measure of rotation around
+ * the Y Axis.
+ * @return The current pitch value in degrees (-90 to 90).
+ */
+ public double getPitch() {
+ if (m_isGyroAPigeon) {
+ return MathUtil.clamp(getPigeonAngles()[1], -90, 90);
+ } else {
+ return MathUtil.clamp(m_navX.getPitch(), -90, 90);
+ }
+ }
+
+ /**
+ * Returns the current roll value (in degrees, from -90 to 90)
+ * reported by the sensor. Roll is a measure of rotation around
+ * the X Axis.
+ * @return The current roll value in degrees (-90 to 90).
+ */
+ public double getRoll() {
+ if (m_isGyroAPigeon) {
+ return MathUtil.clamp(getPigeonAngles()[2], -90, 90);
+ } else {
+ return MathUtil.clamp(m_navX.getRoll(), -90, 90);
+ }
+ }
+
+ @Override
+ public double getRate() {
+ if (m_isGyroAPigeon) {
+ return m_deltaPigeonAngle / m_robotTime.m_deltaTime * 1000;
+ } else {
+ return m_navX.getRate();
+ }
+ }
+
+ public WPI_Pigeon2 getPigeon(){
+ return m_pigeon;
+ }
+
+ public AHRS getNavX(){
+ return m_navX;
+ }
+
+ @Override
+ public void close() throws Exception {
+
+ }
+}
diff --git a/src/main/java/frc4388/utility/RobotTime.java b/src/main/java/frc4388/utility/RobotTime.java
new file mode 100644
index 0000000..694f850
--- /dev/null
+++ b/src/main/java/frc4388/utility/RobotTime.java
@@ -0,0 +1,79 @@
+/*----------------------------------------------------------------------------*/
+/* Copyright (c) 2018-2019 FIRST. All Rights Reserved. */
+/* Open Source Software - may be modified and shared by FRC teams. The code */
+/* must be accompanied by the FIRST BSD license file in the root directory of */
+/* the project. */
+/*----------------------------------------------------------------------------*/
+
+package frc4388.utility;
+
+/**
+ * Keeps track of Robot times like time passed, delta time, etc
+ * All times are in milliseconds
+ */
+public class RobotTime {
+ private long m_currTime = System.currentTimeMillis();
+ public long m_deltaTime = 0;
+
+ private long m_startRobotTime = m_currTime;
+ public long m_robotTime = 0;
+ public long m_lastRobotTime = 0;
+
+ private long m_startMatchTime = 0;
+ public long m_matchTime = 0;
+ public long m_lastMatchTime = 0;
+
+ public long m_frameNumber = 0;
+
+ /**
+ * Private constructor prevents other classes from instantiating
+ */
+ private RobotTime(){}
+
+ private static RobotTime instance = null;
+
+ /**
+ * Gets the instance of Robot Time. If there is no instance running one will be created.
+ * @return instance of Robot Time
+ */
+ public static RobotTime getInstance() {
+ if (instance == null) {
+ instance = new RobotTime();
+ }
+ return instance;
+ }
+
+ /**
+ * Call this once per periodic loop.
+ */
+ public void updateTimes() {
+ m_lastRobotTime = m_robotTime;
+ m_lastMatchTime = m_matchTime;
+
+ m_currTime = System.currentTimeMillis();
+ m_robotTime = m_currTime - m_startRobotTime;
+ m_deltaTime = m_robotTime - m_lastRobotTime;
+ m_frameNumber++;
+
+ if (m_startMatchTime != 0) {
+ m_matchTime = m_currTime - m_startMatchTime;
+ }
+ }
+
+ /**
+ * Call this in both the auto and periodic inits
+ */
+ public void startMatchTime() {
+ if (m_startMatchTime == 0) {
+ m_startMatchTime = m_currTime;
+ }
+ }
+
+ /**
+ * Call this in disabled init
+ */
+ public void endMatchTime() {
+ m_startMatchTime = 0;
+ m_matchTime = 0;
+ }
+}
diff --git a/src/main/java/frc4388/utility/RobotUnits.java b/src/main/java/frc4388/utility/RobotUnits.java
new file mode 100644
index 0000000..9e07312
--- /dev/null
+++ b/src/main/java/frc4388/utility/RobotUnits.java
@@ -0,0 +1,27 @@
+// 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.
+
+package frc4388.utility;
+
+/** Aarav's good units class (better than WPILib)
+ * @author Aarav Shah */
+
+public class RobotUnits {
+ // constants
+
+ // angle conversions
+ public static double degreesToRadians(final double degrees) {return degrees * Math.PI / 180;}
+
+ public static double radiansToDegrees(final double radians) {return radians / Math.PI * 180;}
+
+ // falcon conversions
+ public static double falconTicksToRotations(final double ticks) {return ticks / 2048;}
+
+ public static double falconRotationsToTicks(final double rotations) {return rotations * 2048;}
+
+ // distance conversions
+ public static double metersToFeet(final double meters) {return meters * 3.28084;}
+
+ public static double feetToMeters(final double feet) {return feet / 3.28084;}
+}
\ No newline at end of file
diff --git a/src/main/java/frc4388/utility/UtilityStructs.java b/src/main/java/frc4388/utility/UtilityStructs.java
new file mode 100644
index 0000000..e8b10cc
--- /dev/null
+++ b/src/main/java/frc4388/utility/UtilityStructs.java
@@ -0,0 +1,12 @@
+package frc4388.utility;
+
+public class UtilityStructs {
+ public static class TimedOutput {
+ public double leftX = 0.0;
+ public double leftY = 0.0;
+ public double rightX = 0.0;
+ public double rightY = 0.0;
+
+ public long timedOffset = 0;
+ }
+}
diff --git a/src/main/java/frc4388/utility/controller/IHandController.java b/src/main/java/frc4388/utility/controller/IHandController.java
new file mode 100644
index 0000000..13aa763
--- /dev/null
+++ b/src/main/java/frc4388/utility/controller/IHandController.java
@@ -0,0 +1,21 @@
+package frc4388.utility.controller;
+
+/**
+ * Add your docs here.
+ */
+public interface IHandController {
+
+ public double getLeftXAxis();
+
+ public double getLeftYAxis();
+
+ public double getRightXAxis();
+
+ public double getRightYAxis();
+
+ public double getLeftTriggerAxis();
+
+ public double getRightTriggerAxis();
+
+ public int getDpadAngle();
+}
diff --git a/src/main/java/frc4388/utility/controller/XboxController.java b/src/main/java/frc4388/utility/controller/XboxController.java
new file mode 100644
index 0000000..8b8f0f8
--- /dev/null
+++ b/src/main/java/frc4388/utility/controller/XboxController.java
@@ -0,0 +1,218 @@
+package frc4388.utility.controller;
+
+import edu.wpi.first.wpilibj.Joystick;
+
+/**
+ * This is a wrapper for the WPILib Joystick class that represents an XBox
+ * controller.
+ * @author frc1675
+ */
+public class XboxController implements IHandController
+{
+ public static final int LEFT_X_AXIS = 0;
+ public static final int LEFT_Y_AXIS = 1;
+ public static final int LEFT_TRIGGER_AXIS = 2;
+ public static final int RIGHT_TRIGGER_AXIS = 3;
+ public static final int RIGHT_X_AXIS = 4;
+ public static final int RIGHT_Y_AXIS = 5;
+ public static final int LEFT_RIGHT_DPAD_AXIS = 6;
+ public static final int TOP_BOTTOM_DPAD_AXIS = 6;
+
+ public static final int A_BUTTON = 1;
+ public static final int B_BUTTON = 2;
+ public static final int X_BUTTON = 3;
+ public static final int Y_BUTTON = 4;
+ public static final int LEFT_BUMPER_BUTTON = 5;
+ public static final int RIGHT_BUMPER_BUTTON = 6;
+ public static final int BACK_BUTTON = 7;
+ public static final int START_BUTTON = 8;
+
+ public static final int LEFT_JOYSTICK_BUTTON = 9;
+ public static final int RIGHT_JOYSTICK_BUTTON = 10;
+
+ private static final double LEFT_DPAD_TOLERANCE = -0.9;
+ private static final double RIGHT_DPAD_TOLERANCE = 0.9;
+ private static final double BOTTOM_DPAD_TOLERANCE = -0.9;
+ private static final double TOP_DPAD_TOLERANCE = 0.9;
+
+ private static final double LEFT_TRIGGER_TOLERANCE = 0.5;
+ private static final double RIGHT_TRIGGER_TOLERANCE = 0.5;
+
+ private static final double RIGHT_AXIS_UP_TOLERANCE = -0.9;
+ private static final double RIGHT_AXIS_DOWN_TOLERANCE = 0.9;
+ private static final double RIGHT_AXIS_RIGHT_TOLERANCE = 0.9;
+ private static final double RIGHT_AXIS_LEFT_TOLERANCE = -0.9;
+
+ private static final double LEFT_AXIS_UP_TOLERANCE = -0.9;
+ private static final double LEFT_AXIS_DOWN_TOLERANCE = 0.9;
+ private static final double LEFT_AXIS_RIGHT_TOLERANCE = 0.9;
+ private static final double LEFT_AXIS_LEFT_TOLERANCE = -0.9;
+
+ private static final double DEADZONE = 0.1;
+
+ private Joystick m_stick;
+
+ /**
+ * Add your docs here.
+ */
+ public XboxController(int portNumber){
+ m_stick = new Joystick(portNumber);
+ }
+
+ /**
+ * Add your docs here.
+ */
+ public Joystick getJoyStick() {
+ return m_stick;
+ }
+
+ /**
+ * Checks if the input falls within the deadzone.
+ * @param input from an axis on the controller
+ * @return true if input falls in deadzone, false if input falls outside deadzone
+ */
+ private boolean inDeadZone(double input){
+ return (Math.abs(input) < DEADZONE);
+ }
+
+ /**
+ * Updates an input to have a deadzone around the 0 position
+ * @param input from an axis on the controller
+ * @return updated input
+ */
+ private double getAxisWithDeadZoneCheck(double input){
+ if(inDeadZone(input)){
+ return 0.0;
+ } else {
+ return input;
+ }
+ }
+
+ public boolean getAButton(){
+ return m_stick.getRawButton(A_BUTTON);
+ }
+
+ public boolean getXButton(){
+ return m_stick.getRawButton(X_BUTTON);
+ }
+
+ public boolean getBButton(){
+ return m_stick.getRawButton(B_BUTTON);
+ }
+
+ public boolean getYButton(){
+ return m_stick.getRawButton(Y_BUTTON);
+ }
+
+ public boolean getBackButton(){
+ return m_stick.getRawButton(BACK_BUTTON);
+ }
+
+ public boolean getStartButton(){
+ return m_stick.getRawButton(START_BUTTON);
+ }
+
+ public boolean getLeftBumperButton(){
+ return m_stick.getRawButton(LEFT_BUMPER_BUTTON);
+ }
+
+ public boolean getRightBumperButton(){
+ return m_stick.getRawButton(RIGHT_BUMPER_BUTTON);
+ }
+
+ public boolean getLeftJoystickButton(){
+ return m_stick.getRawButton(LEFT_JOYSTICK_BUTTON);
+ }
+
+ public boolean getRightJoystickButton(){
+ return m_stick.getRawButton(RIGHT_JOYSTICK_BUTTON);
+ }
+
+ public double getLeftXAxis(){
+ return getAxisWithDeadZoneCheck(m_stick.getRawAxis(LEFT_X_AXIS));
+ }
+
+ public double getLeftYAxis(){
+ return getAxisWithDeadZoneCheck(m_stick.getRawAxis(LEFT_Y_AXIS));
+ }
+
+ public double getRightXAxis(){
+ return getAxisWithDeadZoneCheck(m_stick.getRawAxis(RIGHT_X_AXIS));
+ }
+
+ public double getRightYAxis(){
+ return getAxisWithDeadZoneCheck(m_stick.getRawAxis(RIGHT_Y_AXIS));
+ }
+
+ public double getLeftTriggerAxis(){
+ return getAxisWithDeadZoneCheck(m_stick.getRawAxis(LEFT_TRIGGER_AXIS));
+ }
+
+ public double getRightTriggerAxis(){
+ return getAxisWithDeadZoneCheck(m_stick.getRawAxis(RIGHT_TRIGGER_AXIS));
+ }
+
+ /**
+ * Get the angle input from the dpad.
+ * @return -1 if nothing is pressed, or the angle of the button pressed. 0 = up, 90 = right, etc.
+ */
+ public int getDpadAngle() {
+ return m_stick.getPOV(0);
+ }
+
+ public boolean getDPadLeft(){
+ return (m_stick.getRawAxis(LEFT_RIGHT_DPAD_AXIS) < LEFT_DPAD_TOLERANCE);
+ }
+
+ public boolean getDPadRight(){
+ return (m_stick.getRawAxis(LEFT_RIGHT_DPAD_AXIS) > RIGHT_DPAD_TOLERANCE);
+ }
+
+ public boolean getDPadTop(){
+ return (m_stick.getRawAxis(TOP_BOTTOM_DPAD_AXIS) < TOP_DPAD_TOLERANCE);
+ }
+
+ public boolean getDPadBottom(){
+ return (m_stick.getRawAxis(TOP_BOTTOM_DPAD_AXIS) > BOTTOM_DPAD_TOLERANCE);
+ }
+
+ public boolean getLeftTrigger(){
+ return (getLeftTriggerAxis() > LEFT_TRIGGER_TOLERANCE);
+ }
+
+ public boolean getRightTrigger(){
+ return (getRightTriggerAxis() > RIGHT_TRIGGER_TOLERANCE);
+ }
+
+ public boolean getRightAxisUpTrigger(){
+ return (getRightYAxis() < RIGHT_AXIS_UP_TOLERANCE);
+ }
+
+ public boolean getRightAxisDownTrigger(){
+ return (getRightYAxis() > RIGHT_AXIS_DOWN_TOLERANCE);
+ }
+
+ public boolean getRightAxisLeftTrigger(){
+ return (getRightXAxis() > RIGHT_AXIS_LEFT_TOLERANCE);
+ }
+
+ public boolean getRightAxisRightTrigger(){
+ return (getRightXAxis() > RIGHT_AXIS_RIGHT_TOLERANCE);
+ }
+
+ public boolean getLeftAxisUpTrigger(){
+ return (getLeftYAxis() < LEFT_AXIS_UP_TOLERANCE);
+ }
+
+ public boolean getLeftAxisDownTrigger(){
+ return (getLeftYAxis() > LEFT_AXIS_DOWN_TOLERANCE);
+ }
+
+ public boolean getLeftAxisLeftTrigger(){
+ return (getLeftXAxis() > LEFT_AXIS_LEFT_TOLERANCE);
+ }
+
+ public boolean getLeftAxisRightTrigger(){
+ return (getLeftXAxis() > LEFT_AXIS_RIGHT_TOLERANCE);
+ }
+}
\ No newline at end of file
diff --git a/src/main/java/frc4388/utility/controller/XboxTriggerButton.java b/src/main/java/frc4388/utility/controller/XboxTriggerButton.java
new file mode 100644
index 0000000..8c2fe88
--- /dev/null
+++ b/src/main/java/frc4388/utility/controller/XboxTriggerButton.java
@@ -0,0 +1,68 @@
+package frc4388.utility.controller;
+
+import edu.wpi.first.wpilibj2.command.button.Button;
+
+/**
+ * Mapping for the Xbox controller triggers to allow triggers to be defined as
+ * buttons in {@link frc4388.robot.OI}. Checks to see if the given trigger
+ * exceeds a tolerance defined in {@link XboxController}.
+ */
+public class XboxTriggerButton extends Button {
+ public static final int RIGHT_TRIGGER = 0;
+ public static final int LEFT_TRIGGER = 1;
+ public static final int RIGHT_AXIS_UP_TRIGGER = 2;
+ public static final int RIGHT_AXIS_DOWN_TRIGGER = 3;
+ public static final int RIGHT_AXIS_RIGHT_TRIGGER = 4;
+ public static final int RIGHT_AXIS_LEFT_TRIGGER = 5;
+ public static final int LEFT_AXIS_UP_TRIGGER = 6;
+ public static final int LEFT_AXIS_DOWN_TRIGGER = 7;
+ public static final int LEFT_AXIS_RIGHT_TRIGGER = 8;
+ public static final int LEFT_AXIS_LEFT_TRIGGER = 9;
+
+ private XboxController m_controller;
+ private int m_trigger;
+
+ /**
+ * Creates a Trigger-Button mapped to a specific Xbox controller and trigger
+ */
+ public XboxTriggerButton(XboxController controller, int trigger) {
+ m_controller = controller;
+ m_trigger = trigger;
+ }
+
+ /** {@inheritDoc} */
+ // @Override
+ public boolean get() {
+ if (m_trigger == RIGHT_TRIGGER) {
+ return m_controller.getRightTrigger();
+ }
+ else if (m_trigger == LEFT_TRIGGER) {
+ return m_controller.getLeftTrigger();
+ }
+ else if (m_trigger == RIGHT_AXIS_UP_TRIGGER) {
+ return m_controller.getRightAxisUpTrigger();
+ }
+ else if (m_trigger == RIGHT_AXIS_DOWN_TRIGGER) {
+ return m_controller.getRightAxisDownTrigger();
+ }
+ else if (m_trigger == RIGHT_AXIS_RIGHT_TRIGGER) {
+ return m_controller.getRightAxisRightTrigger();
+ }
+ else if (m_trigger == RIGHT_AXIS_LEFT_TRIGGER) {
+ return m_controller.getRightAxisLeftTrigger();
+ }
+ else if (m_trigger == LEFT_AXIS_UP_TRIGGER) {
+ return m_controller.getLeftAxisUpTrigger();
+ }
+ else if (m_trigger == LEFT_AXIS_DOWN_TRIGGER) {
+ return m_controller.getLeftAxisDownTrigger();
+ }
+ else if (m_trigger == LEFT_AXIS_RIGHT_TRIGGER) {
+ return m_controller.getLeftAxisRightTrigger();
+ }
+ else if (m_trigger == LEFT_AXIS_LEFT_TRIGGER) {
+ return m_controller.getLeftAxisLeftTrigger();
+ }
+ return false;
+ }
+}
diff --git a/src/test/java/frc4388/mocks/MockPigeonIMU.java b/src/test/java/frc4388/mocks/MockPigeonIMU.java
new file mode 100644
index 0000000..ecddde7
--- /dev/null
+++ b/src/test/java/frc4388/mocks/MockPigeonIMU.java
@@ -0,0 +1,54 @@
+/*----------------------------------------------------------------------------*/
+/* Copyright (c) 2018-2019 FIRST. All Rights Reserved. */
+/* Open Source Software - may be modified and shared by FRC teams. The code */
+/* must be accompanied by the FIRST BSD license file in the root directory of */
+/* the project. */
+/*----------------------------------------------------------------------------*/
+
+package frc4388.mocks;
+
+import com.ctre.phoenix.ErrorCode;
+import com.ctre.phoenix.sensors.PigeonIMU;
+
+/**
+ * Add your docs here.
+ */
+public class MockPigeonIMU extends PigeonIMU {
+ public int m_deviceNumber;
+ public double currentYaw;
+ public double currentPitch;
+ public double currentRoll;
+
+ public MockPigeonIMU(int deviceNumber) {
+ super(deviceNumber);
+ m_deviceNumber = deviceNumber;
+ }
+
+ @Override
+ public ErrorCode setYaw(double angleDeg) {
+ currentYaw = angleDeg;
+ return ErrorCode.OK;
+ }
+
+ /**
+ * @param currentPitch the Pitch to set
+ */
+ public void setCurrentPitch(double currentPitch) {
+ this.currentPitch = currentPitch;
+ }
+
+ /**
+ * @param currentRoll the Roll to set
+ */
+ public void setCurrentRoll(double currentRoll) {
+ this.currentRoll = currentRoll;
+ }
+
+ @Override
+ public ErrorCode getYawPitchRoll(double[] ypr_deg) {
+ ypr_deg[0] = currentYaw;
+ ypr_deg[1] = currentPitch;
+ ypr_deg[2] = currentRoll;
+ return ErrorCode.OK;
+ }
+}
diff --git a/src/test/java/frc4388/robot/subsystems/LEDSubsystemTest.old b/src/test/java/frc4388/robot/subsystems/LEDSubsystemTest.old
new file mode 100644
index 0000000..8fcbf53
--- /dev/null
+++ b/src/test/java/frc4388/robot/subsystems/LEDSubsystemTest.old
@@ -0,0 +1,59 @@
+/*----------------------------------------------------------------------------*/
+/* Copyright (c) 2018-2019 FIRST. All Rights Reserved. */
+/* Open Source Software - may be modified and shared by FRC teams. The code */
+/* must be accompanied by the FIRST BSD license file in the root directory of */
+/* the project. */
+/*----------------------------------------------------------------------------*/
+
+package frc4388.robot.subsystems;
+
+import static org.junit.Assert.assertEquals;
+import static org.mockito.Mockito.mock;
+
+import org.junit.Test;
+
+import edu.wpi.first.wpilibj.*;
+import frc4388.robot.Constants.LEDConstants;
+import frc4388.utility.LEDPatterns;
+
+/**
+ * Add your docs here.
+ */
+public class LEDSubsystemTest {
+ @Test
+ public void testConstructor() {
+ // Arrange
+ Spark ledController = mock(Spark.class);
+
+ // Act
+ LED led = new LED(ledController);
+
+ // Assert
+ assertEquals(LEDConstants.DEFAULT_PATTERN.getValue(), led.getPattern().getValue(), 0.0001);
+ }
+
+ @Test
+ public void testPatterns() {
+ // Arrange
+ Spark ledController = mock(Spark.class);
+ LED led = new LED(ledController);
+
+ // Act
+ led.setPattern(LEDPatterns.RAINBOW_RAINBOW);
+
+ // Assert
+ assertEquals(LEDPatterns.RAINBOW_RAINBOW.getValue(), led.getPattern().getValue(), 0.0001);
+
+ // Act
+ led.setPattern(LEDPatterns.BLUE_BREATH);
+
+ // Assert
+ assertEquals(LEDPatterns.BLUE_BREATH.getValue(), led.getPattern().getValue(), 0.0001);
+
+ // Act
+ led.setPattern(LEDPatterns.SOLID_BLACK);
+
+ // Assert
+ assertEquals(LEDPatterns.SOLID_BLACK.getValue(), led.getPattern().getValue(), 0.0001);
+ }
+}
diff --git a/src/test/java/frc4388/utility/RobotGyroUtilityTest.old b/src/test/java/frc4388/utility/RobotGyroUtilityTest.old
new file mode 100644
index 0000000..8c0b1e5
--- /dev/null
+++ b/src/test/java/frc4388/utility/RobotGyroUtilityTest.old
@@ -0,0 +1,184 @@
+/*----------------------------------------------------------------------------*/
+/* Copyright (c) 2018-2019 FIRST. All Rights Reserved. */
+/* Open Source Software - may be modified and shared by FRC teams. The code */
+/* must be accompanied by the FIRST BSD license file in the root directory of */
+/* the project. */
+/*----------------------------------------------------------------------------*/
+
+package frc4388.utility;
+
+import static org.junit.Assert.*;
+import static org.mockito.Mockito.*;
+
+import com.kauailabs.navx.frc.AHRS;
+
+import org.junit.*;
+
+import frc4388.mocks.MockPigeonIMU;
+import frc4388.robot.Constants.DriveConstants;
+
+/**
+ * Add your docs here.
+ */
+public class RobotGyroUtilityTest {
+ // TODO UNTESTED: most functions for NavX
+
+ private RobotGyro gyroPigeon;
+ private RobotGyro gyroNavX;
+
+ @Test
+ public void testConstructor() {
+ // Arrange
+ MockPigeonIMU pigeon = new MockPigeonIMU(DriveConstants.DRIVE_PIGEON_ID);
+ AHRS navX = mock(AHRS.class);
+ gyroPigeon = new RobotGyro(pigeon);
+ gyroNavX = new RobotGyro(navX);
+
+ // Assert
+ assertEquals(true, gyroPigeon.m_isGyroAPigeon);
+ assertEquals(pigeon, gyroPigeon.getPigeon());
+ assertEquals(null, gyroPigeon.getNavX());
+ assertEquals(false, gyroNavX.m_isGyroAPigeon);
+ assertEquals(navX, gyroNavX.getNavX());
+ assertEquals(null, gyroNavX.getPigeon());
+ }
+
+ @Test
+ public void testHeadingPigeon() {
+ // Arrange
+ MockPigeonIMU pigeon = new MockPigeonIMU(DriveConstants.DRIVE_PIGEON_ID);
+ gyroPigeon = new RobotGyro(pigeon);
+
+ // Act & Assert
+ assertEquals(-90, gyroPigeon.getHeading(270), 0.0001);
+ assertEquals(-45, gyroPigeon.getHeading(315), 0.0001);
+ assertEquals(-60, gyroPigeon.getHeading(-60), 0.0001);
+ assertEquals(30, gyroPigeon.getHeading(30), 0.0001);
+ assertEquals(0, gyroPigeon.getHeading(0), 0.0001);
+ assertEquals(180, gyroPigeon.getHeading(180), 0.0001);
+ assertEquals(-180, gyroPigeon.getHeading(-180), 0.0001);
+ assertEquals(97, gyroPigeon.getHeading(1537), 0.0001);
+ assertEquals(99, gyroPigeon.getHeading(-2781), 0.0001);
+ }
+
+ @Test
+ public void testYawPitchRollPigeon() {
+ // Arrange
+ MockPigeonIMU pigeon = new MockPigeonIMU(DriveConstants.DRIVE_PIGEON_ID);
+ gyroPigeon = new RobotGyro(pigeon);
+
+ // Assert
+ assertEquals(0, gyroPigeon.getAngle(), 0.0001);
+
+ // Act
+ pigeon.setYaw(40);
+
+ // Assert
+ assertEquals(40, gyroPigeon.getAngle(), 0.0001);
+
+ // Act
+ gyroPigeon.reset();
+
+ // Assert
+ assertEquals(0, gyroPigeon.getAngle(), 0.0001);
+
+ // Act
+ pigeon.setYaw(-1457);
+ pigeon.setCurrentPitch(100);
+ pigeon.setCurrentRoll(100);
+
+ // Assert
+ assertEquals(-1457, gyroPigeon.getAngle(), 0.0001);
+ assertEquals(90, gyroPigeon.getPitch(), 0.0001);
+ assertEquals(90, gyroPigeon.getRoll(), 0.0001);
+
+ // Act
+ pigeon.setCurrentPitch(45);
+ pigeon.setCurrentRoll(45);
+
+ // Assert
+ assertEquals(45, gyroPigeon.getPitch(), 0.0001);
+ assertEquals(45, gyroPigeon.getRoll(), 0.0001);
+
+ // Act
+ pigeon.setCurrentPitch(0);
+ pigeon.setCurrentRoll(0);
+
+ // Assert
+ assertEquals(0, gyroPigeon.getPitch(), 0.0001);
+ assertEquals(0, gyroPigeon.getRoll(), 0.0001);
+
+ // Act
+ pigeon.setCurrentPitch(-60);
+ pigeon.setCurrentRoll(-60);
+
+ // Assert
+ assertEquals(-60, gyroPigeon.getPitch(), 0.0001);
+ assertEquals(-60, gyroPigeon.getRoll(), 0.0001);
+
+ // Act
+ pigeon.setCurrentPitch(-90);
+ pigeon.setCurrentRoll(-90);
+
+ // Assert
+ assertEquals(-90, gyroPigeon.getPitch(), 0.0001);
+ assertEquals(-90, gyroPigeon.getRoll(), 0.0001);
+
+ // Act
+ pigeon.setCurrentPitch(-100);
+ pigeon.setCurrentRoll(-100);
+
+ // Assert
+ assertEquals(-90, gyroPigeon.getPitch(), 0.0001);
+ assertEquals(-90, gyroPigeon.getRoll(), 0.0001);
+ }
+
+ @Test
+ public void testRatesPigeon() {
+ // Arrange
+ MockPigeonIMU pigeon = new MockPigeonIMU(DriveConstants.DRIVE_PIGEON_ID);
+ gyroPigeon = new RobotGyro(pigeon);
+ RobotTime robotTime = RobotTime.getInstance();
+ gyroPigeon.updatePigeonDeltas();
+
+ // Act
+ robotTime.m_deltaTime = 5;
+ pigeon.setYaw(0);
+ gyroPigeon.updatePigeonDeltas();
+
+ // Assert
+ assertEquals(0, gyroPigeon.getRate(), 1);
+
+ // Act
+ robotTime.m_deltaTime = 5;
+ pigeon.setYaw(90);
+ gyroPigeon.updatePigeonDeltas();
+
+ // Assert
+ assertEquals(18000, gyroPigeon.getRate(), 0.001);
+
+ // Act
+ robotTime.m_deltaTime = 5;
+ pigeon.setYaw(90);
+ gyroPigeon.updatePigeonDeltas();
+
+ // Assert
+ assertEquals(0, gyroPigeon.getRate(), 0.001);
+
+ // Act
+ robotTime.m_deltaTime = 3;
+ pigeon.setYaw(-30);
+ gyroPigeon.updatePigeonDeltas();
+
+ // Assert
+ assertEquals(-40000, gyroPigeon.getRate(), 0.001);
+
+ // Act
+ robotTime.m_deltaTime = 6;
+ pigeon.setYaw(690);
+ gyroPigeon.updatePigeonDeltas();
+
+ // Assert
+ assertEquals(120000, gyroPigeon.getRate(), 0.001);
+ }
+}
diff --git a/src/test/java/frc4388/utility/RobotTimeUtilityTest.old b/src/test/java/frc4388/utility/RobotTimeUtilityTest.old
new file mode 100644
index 0000000..2c31a34
--- /dev/null
+++ b/src/test/java/frc4388/utility/RobotTimeUtilityTest.old
@@ -0,0 +1,104 @@
+/*----------------------------------------------------------------------------*/
+/* Copyright (c) 2018-2019 FIRST. All Rights Reserved. */
+/* Open Source Software - may be modified and shared by FRC teams. The code */
+/* must be accompanied by the FIRST BSD license file in the root directory of */
+/* the project. */
+/*----------------------------------------------------------------------------*/
+
+package frc4388.utility;
+
+import static org.junit.Assert.*;
+
+import org.junit.*;
+
+/**
+ * Add your docs here.
+ */
+public class RobotTimeUtilityTest {
+ RobotTime robotTime = RobotTime.getInstance();
+
+ @Test
+ public void testUpdateTimes() {
+ // Arrange
+ long lastTime;
+ robotTime.m_deltaTime = 0;
+ robotTime.m_robotTime = 0;
+ robotTime.m_lastRobotTime = 0;
+ robotTime.m_frameNumber = 0;
+ robotTime.endMatchTime();
+ robotTime.m_lastMatchTime = 0;
+
+ // Assert
+ assertEquals(0, robotTime.m_deltaTime);
+ assertEquals(0, robotTime.m_robotTime);
+ assertEquals(0, robotTime.m_lastRobotTime);
+ assertEquals(0, robotTime.m_frameNumber);
+ lastTime = robotTime.m_robotTime;
+
+ // Act
+ wait(1);
+ robotTime.updateTimes();
+
+ // Assert
+ assertEquals(true, robotTime.m_deltaTime > 0);
+ assertEquals(true, robotTime.m_robotTime > 0);
+ assertEquals(lastTime, robotTime.m_lastRobotTime);
+ assertEquals(1, robotTime.m_frameNumber);
+ lastTime = robotTime.m_robotTime;
+
+ // Act
+ wait(1);
+ robotTime.updateTimes();
+
+ // Assert
+ assertEquals(true, robotTime.m_deltaTime > 0);
+ assertEquals(true, robotTime.m_robotTime > 0);
+ assertEquals(lastTime, robotTime.m_lastRobotTime);
+ assertEquals(2, robotTime.m_frameNumber);
+ }
+
+ @Test
+ public void testMatchTime() {
+ // Arrange
+ long lastTime;
+
+ // Assert
+ assertEquals(0, robotTime.m_matchTime);
+ assertEquals(0, robotTime.m_lastMatchTime);
+ lastTime = robotTime.m_matchTime;
+
+ // Act
+ robotTime.startMatchTime();
+ wait(1);
+ robotTime.updateTimes();
+
+ // Assert
+ assertEquals(true, robotTime.m_matchTime > 0);
+ assertEquals(lastTime, robotTime.m_lastMatchTime);
+ lastTime = robotTime.m_matchTime;
+
+ // Act
+ wait(1);
+ robotTime.updateTimes();
+ robotTime.endMatchTime();
+
+ // Assert
+ assertEquals(0, robotTime.m_matchTime);
+ assertEquals(lastTime, robotTime.m_lastMatchTime);
+ lastTime = robotTime.m_matchTime;
+
+ // Act
+ wait(1);
+ robotTime.updateTimes();
+
+ // Assert
+ assertEquals(0, robotTime.m_matchTime);
+ assertEquals(lastTime, robotTime.m_lastMatchTime);
+ }
+
+ private void wait(int millis) {
+ try {
+ Thread.sleep(millis);
+ } catch (Exception e) {}
+ }
+}
diff --git a/template.config.js b/template.config.js
new file mode 100644
index 0000000..54f2425
--- /dev/null
+++ b/template.config.js
@@ -0,0 +1,31 @@
+/**
+ * This file is a configuration file generated by the `Template` extension on `vscode`
+ * @see https://marketplace.visualstudio.com/items?itemName=yongwoo.template
+ */
+module.exports = {
+ // You can change the template path to another path
+ templateRootPath: "./.templates",
+ // After copying the template file the `replaceFileTextFn` function is executed
+ replaceFileTextFn: (fileText, templateName, utils) => {
+ // @see https://www.npmjs.com/package/change-case
+ const { changeCase } = utils;
+ // You can change the text in the file
+ return fileText
+ .replace(/__templateName__/gm, templateName)
+ .replace(
+ /__templateNameToPascalCase__/gm,
+ changeCase.pascalCase(templateName)
+ )
+ .replace(
+ /__templateNameToParamCase__/gm,
+ changeCase.paramCase(templateName)
+ );
+ },
+ replaceFileNameFn: (fileName, templateName, utils) => {
+ const { path } = utils;
+ // @see https://nodejs.org/api/path.html#path_path_parse_path
+ const { base } = path.parse(fileName);
+ // You can change the file name
+ return base;
+ }
+};
diff --git a/vendordeps/NavX.json b/vendordeps/NavX.json
new file mode 100644
index 0000000..33d81f6
--- /dev/null
+++ b/vendordeps/NavX.json
@@ -0,0 +1,39 @@
+{
+ "fileName": "NavX.json",
+ "name": "KauaiLabs_navX_FRC",
+ "version": "2023.0.4",
+ "uuid": "cb311d09-36e9-4143-a032-55bb2b94443b",
+ "mavenUrls": [
+ "https://dev.studica.com/maven/release/2023/"
+ ],
+ "jsonUrl": "https://dev.studica.com/releases/2023/NavX.json",
+ "javaDependencies": [
+ {
+ "groupId": "com.kauailabs.navx.frc",
+ "artifactId": "navx-frc-java",
+ "version": "2023.0.4"
+ }
+ ],
+ "jniDependencies": [],
+ "cppDependencies": [
+ {
+ "groupId": "com.kauailabs.navx.frc",
+ "artifactId": "navx-frc-cpp",
+ "version": "2023.0.4",
+ "headerClassifier": "headers",
+ "sourcesClassifier": "sources",
+ "sharedLibrary": false,
+ "libName": "navx_frc",
+ "skipInvalidPlatforms": true,
+ "binaryPlatforms": [
+ "linuxathena",
+ "linuxraspbian",
+ "linuxarm32",
+ "linuxarm64",
+ "linuxx86-64",
+ "osxuniversal",
+ "windowsx86-64"
+ ]
+ }
+ ]
+}
\ No newline at end of file
diff --git a/vendordeps/Phoenix.json b/vendordeps/Phoenix.json
new file mode 100644
index 0000000..614dc3a
--- /dev/null
+++ b/vendordeps/Phoenix.json
@@ -0,0 +1,423 @@
+{
+ "fileName": "Phoenix.json",
+ "name": "CTRE-Phoenix (v5)",
+ "version": "5.31.0+23.2.2",
+ "frcYear": 2023,
+ "uuid": "ab676553-b602-441f-a38d-f1296eff6537",
+ "mavenUrls": [
+ "https://maven.ctr-electronics.com/release/"
+ ],
+ "jsonUrl": "https://maven.ctr-electronics.com/release/com/ctre/phoenix/Phoenix5-frc2023-latest.json",
+ "javaDependencies": [
+ {
+ "groupId": "com.ctre.phoenix",
+ "artifactId": "api-java",
+ "version": "5.31.0"
+ },
+ {
+ "groupId": "com.ctre.phoenix",
+ "artifactId": "wpiapi-java",
+ "version": "5.31.0"
+ }
+ ],
+ "jniDependencies": [
+ {
+ "groupId": "com.ctre.phoenix",
+ "artifactId": "cci",
+ "version": "5.31.0",
+ "isJar": false,
+ "skipInvalidPlatforms": true,
+ "validPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "linuxathena"
+ ],
+ "simMode": "hwsim"
+ },
+ {
+ "groupId": "com.ctre.phoenix.sim",
+ "artifactId": "cci-sim",
+ "version": "5.31.0",
+ "isJar": false,
+ "skipInvalidPlatforms": true,
+ "validPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "osxuniversal"
+ ],
+ "simMode": "swsim"
+ },
+ {
+ "groupId": "com.ctre.phoenix6",
+ "artifactId": "tools",
+ "version": "23.2.2",
+ "isJar": false,
+ "skipInvalidPlatforms": true,
+ "validPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "linuxathena"
+ ],
+ "simMode": "hwsim"
+ },
+ {
+ "groupId": "com.ctre.phoenix6.sim",
+ "artifactId": "tools-sim",
+ "version": "23.2.2",
+ "isJar": false,
+ "skipInvalidPlatforms": true,
+ "validPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "osxuniversal"
+ ],
+ "simMode": "swsim"
+ },
+ {
+ "groupId": "com.ctre.phoenix6.sim",
+ "artifactId": "simTalonSRX",
+ "version": "23.2.2",
+ "isJar": false,
+ "skipInvalidPlatforms": true,
+ "validPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "osxuniversal"
+ ],
+ "simMode": "swsim"
+ },
+ {
+ "groupId": "com.ctre.phoenix6.sim",
+ "artifactId": "simTalonFX",
+ "version": "23.2.2",
+ "isJar": false,
+ "skipInvalidPlatforms": true,
+ "validPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "osxuniversal"
+ ],
+ "simMode": "swsim"
+ },
+ {
+ "groupId": "com.ctre.phoenix6.sim",
+ "artifactId": "simVictorSPX",
+ "version": "23.2.2",
+ "isJar": false,
+ "skipInvalidPlatforms": true,
+ "validPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "osxuniversal"
+ ],
+ "simMode": "swsim"
+ },
+ {
+ "groupId": "com.ctre.phoenix6.sim",
+ "artifactId": "simPigeonIMU",
+ "version": "23.2.2",
+ "isJar": false,
+ "skipInvalidPlatforms": true,
+ "validPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "osxuniversal"
+ ],
+ "simMode": "swsim"
+ },
+ {
+ "groupId": "com.ctre.phoenix6.sim",
+ "artifactId": "simCANCoder",
+ "version": "23.2.2",
+ "isJar": false,
+ "skipInvalidPlatforms": true,
+ "validPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "osxuniversal"
+ ],
+ "simMode": "swsim"
+ },
+ {
+ "groupId": "com.ctre.phoenix6.sim",
+ "artifactId": "simProTalonFX",
+ "version": "23.2.2",
+ "isJar": false,
+ "skipInvalidPlatforms": true,
+ "validPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "osxuniversal"
+ ],
+ "simMode": "swsim"
+ },
+ {
+ "groupId": "com.ctre.phoenix6.sim",
+ "artifactId": "simProCANcoder",
+ "version": "23.2.2",
+ "isJar": false,
+ "skipInvalidPlatforms": true,
+ "validPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "osxuniversal"
+ ],
+ "simMode": "swsim"
+ },
+ {
+ "groupId": "com.ctre.phoenix6.sim",
+ "artifactId": "simProPigeon2",
+ "version": "23.2.2",
+ "isJar": false,
+ "skipInvalidPlatforms": true,
+ "validPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "osxuniversal"
+ ],
+ "simMode": "swsim"
+ }
+ ],
+ "cppDependencies": [
+ {
+ "groupId": "com.ctre.phoenix",
+ "artifactId": "wpiapi-cpp",
+ "version": "5.31.0",
+ "libName": "CTRE_Phoenix_WPI",
+ "headerClassifier": "headers",
+ "sharedLibrary": true,
+ "skipInvalidPlatforms": true,
+ "binaryPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "linuxathena"
+ ],
+ "simMode": "hwsim"
+ },
+ {
+ "groupId": "com.ctre.phoenix",
+ "artifactId": "api-cpp",
+ "version": "5.31.0",
+ "libName": "CTRE_Phoenix",
+ "headerClassifier": "headers",
+ "sharedLibrary": true,
+ "skipInvalidPlatforms": true,
+ "binaryPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "linuxathena"
+ ],
+ "simMode": "hwsim"
+ },
+ {
+ "groupId": "com.ctre.phoenix",
+ "artifactId": "cci",
+ "version": "5.31.0",
+ "libName": "CTRE_PhoenixCCI",
+ "headerClassifier": "headers",
+ "sharedLibrary": true,
+ "skipInvalidPlatforms": true,
+ "binaryPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "linuxathena"
+ ],
+ "simMode": "hwsim"
+ },
+ {
+ "groupId": "com.ctre.phoenix6",
+ "artifactId": "tools",
+ "version": "23.2.2",
+ "libName": "CTRE_PhoenixTools",
+ "headerClassifier": "headers",
+ "sharedLibrary": true,
+ "skipInvalidPlatforms": true,
+ "binaryPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "linuxathena"
+ ],
+ "simMode": "hwsim"
+ },
+ {
+ "groupId": "com.ctre.phoenix.sim",
+ "artifactId": "wpiapi-cpp-sim",
+ "version": "5.31.0",
+ "libName": "CTRE_Phoenix_WPISim",
+ "headerClassifier": "headers",
+ "sharedLibrary": true,
+ "skipInvalidPlatforms": true,
+ "binaryPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "osxuniversal"
+ ],
+ "simMode": "swsim"
+ },
+ {
+ "groupId": "com.ctre.phoenix.sim",
+ "artifactId": "api-cpp-sim",
+ "version": "5.31.0",
+ "libName": "CTRE_PhoenixSim",
+ "headerClassifier": "headers",
+ "sharedLibrary": true,
+ "skipInvalidPlatforms": true,
+ "binaryPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "osxuniversal"
+ ],
+ "simMode": "swsim"
+ },
+ {
+ "groupId": "com.ctre.phoenix.sim",
+ "artifactId": "cci-sim",
+ "version": "5.31.0",
+ "libName": "CTRE_PhoenixCCISim",
+ "headerClassifier": "headers",
+ "sharedLibrary": true,
+ "skipInvalidPlatforms": true,
+ "binaryPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "osxuniversal"
+ ],
+ "simMode": "swsim"
+ },
+ {
+ "groupId": "com.ctre.phoenix6.sim",
+ "artifactId": "tools-sim",
+ "version": "23.2.2",
+ "libName": "CTRE_PhoenixTools_Sim",
+ "headerClassifier": "headers",
+ "sharedLibrary": true,
+ "skipInvalidPlatforms": true,
+ "binaryPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "osxuniversal"
+ ],
+ "simMode": "swsim"
+ },
+ {
+ "groupId": "com.ctre.phoenix6.sim",
+ "artifactId": "simTalonSRX",
+ "version": "23.2.2",
+ "libName": "CTRE_SimTalonSRX",
+ "headerClassifier": "headers",
+ "sharedLibrary": true,
+ "skipInvalidPlatforms": true,
+ "binaryPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "osxuniversal"
+ ],
+ "simMode": "swsim"
+ },
+ {
+ "groupId": "com.ctre.phoenix6.sim",
+ "artifactId": "simTalonFX",
+ "version": "23.2.2",
+ "libName": "CTRE_SimTalonFX",
+ "headerClassifier": "headers",
+ "sharedLibrary": true,
+ "skipInvalidPlatforms": true,
+ "binaryPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "osxuniversal"
+ ],
+ "simMode": "swsim"
+ },
+ {
+ "groupId": "com.ctre.phoenix6.sim",
+ "artifactId": "simVictorSPX",
+ "version": "23.2.2",
+ "libName": "CTRE_SimVictorSPX",
+ "headerClassifier": "headers",
+ "sharedLibrary": true,
+ "skipInvalidPlatforms": true,
+ "binaryPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "osxuniversal"
+ ],
+ "simMode": "swsim"
+ },
+ {
+ "groupId": "com.ctre.phoenix6.sim",
+ "artifactId": "simPigeonIMU",
+ "version": "23.2.2",
+ "libName": "CTRE_SimPigeonIMU",
+ "headerClassifier": "headers",
+ "sharedLibrary": true,
+ "skipInvalidPlatforms": true,
+ "binaryPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "osxuniversal"
+ ],
+ "simMode": "swsim"
+ },
+ {
+ "groupId": "com.ctre.phoenix6.sim",
+ "artifactId": "simCANCoder",
+ "version": "23.2.2",
+ "libName": "CTRE_SimCANCoder",
+ "headerClassifier": "headers",
+ "sharedLibrary": true,
+ "skipInvalidPlatforms": true,
+ "binaryPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "osxuniversal"
+ ],
+ "simMode": "swsim"
+ },
+ {
+ "groupId": "com.ctre.phoenix6.sim",
+ "artifactId": "simProTalonFX",
+ "version": "23.2.2",
+ "libName": "CTRE_SimProTalonFX",
+ "headerClassifier": "headers",
+ "sharedLibrary": true,
+ "skipInvalidPlatforms": true,
+ "binaryPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "osxuniversal"
+ ],
+ "simMode": "swsim"
+ },
+ {
+ "groupId": "com.ctre.phoenix6.sim",
+ "artifactId": "simProCANcoder",
+ "version": "23.2.2",
+ "libName": "CTRE_SimProCANcoder",
+ "headerClassifier": "headers",
+ "sharedLibrary": true,
+ "skipInvalidPlatforms": true,
+ "binaryPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "osxuniversal"
+ ],
+ "simMode": "swsim"
+ },
+ {
+ "groupId": "com.ctre.phoenix6.sim",
+ "artifactId": "simProPigeon2",
+ "version": "23.2.2",
+ "libName": "CTRE_SimProPigeon2",
+ "headerClassifier": "headers",
+ "sharedLibrary": true,
+ "skipInvalidPlatforms": true,
+ "binaryPlatforms": [
+ "windowsx86-64",
+ "linuxx86-64",
+ "osxuniversal"
+ ],
+ "simMode": "swsim"
+ }
+ ]
+}
\ No newline at end of file
diff --git a/vendordeps/WPILibNewCommands.json b/vendordeps/WPILibNewCommands.json
new file mode 100644
index 0000000..bd535bf
--- /dev/null
+++ b/vendordeps/WPILibNewCommands.json
@@ -0,0 +1,37 @@
+{
+ "fileName": "WPILibNewCommands.json",
+ "name": "WPILib-New-Commands",
+ "version": "1.0.0",
+ "uuid": "111e20f7-815e-48f8-9dd6-e675ce75b266",
+ "mavenUrls": [],
+ "jsonUrl": "",
+ "javaDependencies": [
+ {
+ "groupId": "edu.wpi.first.wpilibNewCommands",
+ "artifactId": "wpilibNewCommands-java",
+ "version": "wpilib"
+ }
+ ],
+ "jniDependencies": [],
+ "cppDependencies": [
+ {
+ "groupId": "edu.wpi.first.wpilibNewCommands",
+ "artifactId": "wpilibNewCommands-cpp",
+ "version": "wpilib",
+ "libName": "wpilibNewCommands",
+ "headerClassifier": "headers",
+ "sourcesClassifier": "sources",
+ "sharedLibrary": true,
+ "skipInvalidPlatforms": true,
+ "binaryPlatforms": [
+ "linuxathena",
+ "linuxarm32",
+ "linuxarm64",
+ "windowsx86-64",
+ "windowsx86",
+ "linuxx86-64",
+ "osxuniversal"
+ ]
+ }
+ ]
+}
diff --git a/vendordeps/navx_frc.json b/vendordeps/navx_frc.json
new file mode 100644
index 0000000..dca1d82
--- /dev/null
+++ b/vendordeps/navx_frc.json
@@ -0,0 +1,35 @@
+{
+ "fileName": "navx_frc.json",
+ "name": "KauaiLabs_navX_FRC",
+ "version": "3.1.413",
+ "uuid": "cb311d09-36e9-4143-a032-55bb2b94443b",
+ "mavenUrls": [
+ "https://repo1.maven.org/maven2/"
+ ],
+ "jsonUrl": "https://www.kauailabs.com/dist/frc/2020/navx_frc.json",
+ "javaDependencies": [
+ {
+ "groupId": "com.kauailabs.navx.frc",
+ "artifactId": "navx-java",
+ "version": "3.1.413"
+ }
+ ],
+ "jniDependencies": [],
+ "cppDependencies": [
+ {
+ "groupId": "com.kauailabs.navx.frc",
+ "artifactId": "navx-cpp",
+ "version": "3.1.413",
+ "headerClassifier": "headers",
+ "sourcesClassifier": "sources",
+ "sharedLibrary": false,
+ "libName": "navx_frc",
+ "skipInvalidPlatforms": true,
+ "binaryPlatforms": [
+ "linuxathena",
+ "linuxraspbian",
+ "windowsx86-64"
+ ]
+ }
+ ]
+}
\ No newline at end of file
diff --git a/vendordeps/photonlib.json b/vendordeps/photonlib.json
new file mode 100644
index 0000000..c940b75
--- /dev/null
+++ b/vendordeps/photonlib.json
@@ -0,0 +1,42 @@
+{
+ "fileName": "photonlib.json",
+ "name": "photonlib",
+ "version": "v2024.1.1-beta-3.2",
+ "uuid": "515fe07e-bfc6-11fa-b3de-0242ac130004",
+ "frcYear": "2024",
+ "mavenUrls": [
+ "https://maven.photonvision.org/repository/internal",
+ "https://maven.photonvision.org/repository/snapshots"
+ ],
+ "jsonUrl": "https://maven.photonvision.org/repository/internal/org/photonvision/PhotonLib-json/1.0/PhotonLib-json-1.0.json",
+ "jniDependencies": [],
+ "cppDependencies": [
+ {
+ "groupId": "org.photonvision",
+ "artifactId": "PhotonLib-cpp",
+ "version": "v2024.1.1-beta-3.2",
+ "libName": "Photon",
+ "headerClassifier": "headers",
+ "sharedLibrary": true,
+ "skipInvalidPlatforms": true,
+ "binaryPlatforms": [
+ "windowsx86-64",
+ "linuxathena",
+ "linuxx86-64",
+ "osxuniversal"
+ ]
+ }
+ ],
+ "javaDependencies": [
+ {
+ "groupId": "org.photonvision",
+ "artifactId": "PhotonLib-java",
+ "version": "v2024.1.1-beta-3.2"
+ },
+ {
+ "groupId": "org.photonvision",
+ "artifactId": "PhotonTargeting-java",
+ "version": "v2024.1.1-beta-3.2"
+ }
+ ]
+}
\ No newline at end of file