Files
2024AcrossTheRidgebotiverse/src/main/java/frc4388/robot/subsystems/SwerveModule.java
T

200 lines
7.7 KiB
Java
Raw Normal View History

2024-01-08 11:38:08 -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.
package frc4388.robot.subsystems;
2024-03-15 14:39:35 -06:00
import javax.swing.text.StyleContext.SmallAttributeSet;
import com.ctre.phoenix.ErrorCode;
2024-01-08 11:38:08 -07:00
import com.ctre.phoenix.motorcontrol.FeedbackDevice;
import com.ctre.phoenix.motorcontrol.RemoteSensorSource;
import com.ctre.phoenix.motorcontrol.TalonFXControlMode;
import com.ctre.phoenix.motorcontrol.can.TalonFXConfiguration;
import com.ctre.phoenix.motorcontrol.can.WPI_TalonFX;
import com.ctre.phoenix.sensors.CANCoder;
2024-03-15 14:39:35 -06:00
import com.ctre.phoenix.sensors.SensorInitializationStrategy;
2024-04-04 08:31:23 -06:00
import com.ctre.phoenix6.configs.Slot0Configs;
import com.ctre.phoenix6.hardware.CANcoder;
import com.ctre.phoenix6.hardware.TalonFX;
2024-01-08 11:38:08 -07:00
import edu.wpi.first.math.geometry.Rotation2d;
import edu.wpi.first.math.kinematics.SwerveModulePosition;
import edu.wpi.first.math.kinematics.SwerveModuleState;
import edu.wpi.first.math.util.Units;
2024-03-15 14:39:35 -06:00
import edu.wpi.first.wpilibj.smartdashboard.SmartDashboard;
2024-01-08 11:38:08 -07:00
import edu.wpi.first.wpilibj2.command.SubsystemBase;
import frc4388.robot.Constants.SwerveDriveConstants;
import frc4388.utility.Gains;
import frc4388.utility.configurable.ConfigurableDouble;
2024-01-08 11:38:08 -07:00
public class SwerveModule extends SubsystemBase {
2024-04-04 12:18:11 -06:00
private WPI_TalonFX tal;
2024-04-04 08:31:23 -06:00
private TalonFX driveMotor;
private TalonFX angleMotor;
private CANcoder encoder;
2024-03-15 14:39:35 -06:00
private int selfid;
2024-03-21 09:01:42 -06:00
// private ConfigurableDouble offsetGetter;
private static int swerveId = 0;
2024-01-08 11:38:08 -07:00
public static Gains swerveGains = SwerveDriveConstants.PIDConstants.SWERVE_GAINS;
/** Creates a new SwerveModule. */
2024-04-04 08:31:23 -06:00
public SwerveModule(TalonFX driveMotor, TalonFX angleMotor, CANcoder encoder, double offset) {
2024-01-08 11:38:08 -07:00
this.driveMotor = driveMotor;
this.angleMotor = angleMotor;
this.encoder = encoder;
2024-03-21 09:01:42 -06:00
// this.offsetGetter = new ConfigurableDouble("Swerve id " + swerveId, offset);
2024-03-15 14:39:35 -06:00
this.selfid = swerveId;
swerveId++;
2024-01-08 11:38:08 -07:00
TalonFXConfiguration angleConfig = new TalonFXConfiguration();
angleConfig.slot0.kP = swerveGains.kP;
angleConfig.slot0.kI = swerveGains.kI;
angleConfig.slot0.kD = swerveGains.kD;
// use the CANcoder as the remote sensor for the primary TalonFX PID
2024-04-04 08:31:23 -06:00
new Slot0Configs().
2024-01-08 11:38:08 -07:00
angleConfig.remoteFilter0.remoteSensorDeviceID = encoder.getDeviceID();
angleConfig.remoteFilter0.remoteSensorSource = RemoteSensorSource.CANCoder;
angleConfig.primaryPID.selectedFeedbackSensor = FeedbackDevice.RemoteSensor0;
angleMotor.configAllSettings(angleConfig);
2024-03-15 14:39:35 -06:00
//encoder.configSensorInitializationStrategy(SensorInitializationStrategy.BootToAbsolutePosition);
reset(0);
2024-01-08 11:38:08 -07:00
encoder.configMagnetOffset(offset);
2024-04-04 08:31:23 -06:00
driveMotor.
2024-01-08 11:38:08 -07:00
driveMotor.setSelectedSensorPosition(0);
driveMotor.config_kP(0, 0.2);
}
2024-03-15 14:39:35 -06:00
@Override
public void periodic() {
//encoder.configMagnetOffset(offsetGetter.get());
//SmartDashboard.putString("Error Code: " + selfid, getstuff());
// SmartDashboard.putNumber("Angular Position: " + selfid, getAngle().getDegrees());
// SmartDashboard.putNumber("Angular Velocity: " + selfid, getAngularVel());
// SmartDashboard.putNumber("Drive Position: " + selfid, getDrivePos());
// SmartDashboard.putNumber("Drive Velocity: " + selfid, getDriveVel());
}
2024-01-08 11:38:08 -07:00
/**
* Get the drive motor of the SwerveModule
* @return the drive motor of the SwerveModule
*/
2024-04-04 12:18:11 -06:00
public TalonFX getDriveMotor() {
2024-01-08 11:38:08 -07:00
return this.driveMotor;
}
/**
* Get the angle motor of the SwerveModule
* @return the angle motor of the SwerveModule
*/
2024-04-04 12:18:11 -06:00
public TalonFX getAngleMotor() {
2024-01-08 11:38:08 -07:00
return this.angleMotor;
}
/**
* Get the CANcoder of the SwerveModule
* @return the CANcoder of the SwerveModule
*/
2024-04-04 12:18:11 -06:00
public CANcoder getEncoder() {
2024-01-08 11:38:08 -07:00
return this.encoder;
}
/**
* Get the angle of a SwerveModule as a Rotation2d
* @return the angle of a SwerveModule as a Rotation2d
*/
public Rotation2d getAngle() {
// * Note: This assumes that the CANCoders are setup with the default feedback coefficient and the sensor value reports degrees.
2024-04-04 12:18:11 -06:00
return Rotation2d.fromDegrees(encoder.getAbsolutePosition().getValue());
// return Rotation2d.fromDegrees(tal.get());
2024-01-08 11:38:08 -07:00
}
public double getAngularVel() {
2024-04-04 12:18:11 -06:00
return this.angleMotor.getVelocity().getValue() * 360;
// return this.tal.getSelectedSensorVelocity();
// return this.angleMotor.getSelectedSensorVelocity();
2024-01-08 11:38:08 -07:00
}
public double getDrivePos() {
2024-04-04 12:18:11 -06:00
return this.driveMotor.getPosition().getValue(); // TODO: with drive test, might have to multiply or divide by 2
// return this.tal.getSelectedSensorPosition() / SwerveDriveConstants.Conversions.TICKS_PER_MOTOR_REV;
2024-01-08 11:38:08 -07:00
}
public double getDriveVel() {
2024-04-04 12:18:11 -06:00
return this.driveMotor.getVelocity().getValue() * 360;
// return this.driveMotor.getSelectedSensorVelocity(0);
2024-01-08 11:38:08 -07:00
}
public void stop() {
driveMotor.set(0);
angleMotor.set(0);
}
public void rotateToAngle(double angle) {
angleMotor.set(TalonFXControlMode.Position, angle);
}
/**
* Get state of swerve module
* @return speed in m/s and angle in degrees
*/
public SwerveModuleState getState() {
return new SwerveModuleState(
Units.inchesToMeters(driveMotor.getSelectedSensorVelocity() * SwerveDriveConstants.Conversions.INCHES_PER_TICK) * SwerveDriveConstants.Conversions.TICK_TIME_TO_SECONDS,
getAngle()
);
}
/**
* Returns the current position of the SwerveModule
* @return The current position of the SwerveModule in meters traveled by the driveMotor and the angle of the angleMotor.
*/
public SwerveModulePosition getPosition() {
return new SwerveModulePosition(Units.inchesToMeters(driveMotor.getSelectedSensorPosition() * SwerveDriveConstants.Conversions.INCHES_PER_TICK), getAngle());
}
/**
* Set the speed and rotation of the SwerveModule from a SwerveModuleState object
* @param desiredState a SwerveModuleState representing the desired new state of the module
*/
public void setDesiredState(SwerveModuleState desiredState) {
Rotation2d currentRotation = this.getAngle();
SwerveModuleState state = SwerveModuleState.optimize(desiredState, currentRotation);
// calculate the difference between our current rotational position and our new rotational position
Rotation2d rotationDelta = state.angle.minus(currentRotation);
// calculate the new absolute position of the SwerveModule based on the difference in rotation
double deltaTicks = (rotationDelta.getDegrees() / 360.) * SwerveDriveConstants.Conversions.CANCODER_TICKS_PER_ROTATION;
// convert the CANCoder from its position reading to ticks
double currentTicks = encoder.getPosition() / encoder.configGetFeedbackCoefficient();
angleMotor.set(TalonFXControlMode.Position, currentTicks + deltaTicks);
double feetPerSecond = Units.metersToFeet(state.speedMetersPerSecond);
driveMotor.set((feetPerSecond / SwerveDriveConstants.MAX_SPEED_FEET_PER_SECOND));
}
public void reset(double position) {
encoder.setPositionToAbsolute();
}
public double getCurrent() {
return angleMotor.getSupplyCurrent() + driveMotor.getSupplyCurrent();
}
public double getVoltage() {
return (Math.abs(angleMotor.getMotorOutputVoltage()) + Math.abs(driveMotor.getMotorOutputVoltage()));
}
2024-03-15 14:39:35 -06:00
// public String getstuff() {
// encoder.getPosition();
// return "" + encoder.getLastError().value;
// }
2024-01-08 11:38:08 -07:00
}