2026-01-10 16:52:43 -07:00
|
|
|
package frc4388.robot.subsystems.shooter;
|
|
|
|
|
|
|
|
|
|
import static edu.wpi.first.units.Units.InchesPerSecond;
|
|
|
|
|
import static edu.wpi.first.units.Units.RotationsPerSecond;
|
|
|
|
|
|
|
|
|
|
import com.ctre.phoenix6.controls.VelocityDutyCycle;
|
|
|
|
|
import com.ctre.phoenix6.hardware.TalonFX;
|
|
|
|
|
|
2026-01-27 18:13:51 -07:00
|
|
|
import edu.wpi.first.units.measure.Angle;
|
|
|
|
|
import edu.wpi.first.units.measure.AngularVelocity;
|
2026-02-09 17:18:54 -08:00
|
|
|
import frc4388.robot.subsystems.intake.IntakeConstants;
|
|
|
|
|
import frc4388.utility.configurable.ConfigurableDouble;
|
2026-01-10 16:52:43 -07:00
|
|
|
|
2026-01-27 19:15:37 -07:00
|
|
|
public class ShooterReal implements ShooterIO {
|
2026-01-27 18:13:51 -07:00
|
|
|
|
|
|
|
|
TalonFX m_shooter1Motor;
|
|
|
|
|
TalonFX m_shooter2Motor;
|
|
|
|
|
TalonFX m_indexerMotor;
|
2026-02-09 17:18:54 -08:00
|
|
|
|
2026-01-27 18:13:51 -07:00
|
|
|
VelocityDutyCycle shooter1Velocity = new VelocityDutyCycle(0);
|
|
|
|
|
VelocityDutyCycle shooter2Velocity = new VelocityDutyCycle(0);
|
|
|
|
|
VelocityDutyCycle m_indexerVelocity = new VelocityDutyCycle(0);
|
2026-01-10 16:52:43 -07:00
|
|
|
|
|
|
|
|
|
|
|
|
|
public ShooterReal(
|
2026-01-27 18:13:51 -07:00
|
|
|
TalonFX shooter1Motor,
|
|
|
|
|
TalonFX shooter2Motor,
|
|
|
|
|
TalonFX indexerMotor
|
2026-01-10 16:52:43 -07:00
|
|
|
) {
|
2026-02-09 16:03:48 -08:00
|
|
|
m_shooter1Motor = shooter1Motor;
|
|
|
|
|
m_shooter2Motor = shooter2Motor;
|
|
|
|
|
m_indexerMotor = indexerMotor;
|
2026-02-09 17:18:54 -08:00
|
|
|
|
2026-01-27 19:15:37 -07:00
|
|
|
m_shooter1Motor.getConfigurator().apply(ShooterConstants.SHOOTER_PID);
|
|
|
|
|
m_shooter2Motor.getConfigurator().apply(ShooterConstants.SHOOTER_PID);
|
|
|
|
|
m_indexerMotor.getConfigurator().apply(ShooterConstants.SHOOTER_PID);
|
|
|
|
|
|
2026-01-27 18:13:51 -07:00
|
|
|
m_shooter1Motor.getConfigurator().apply(ShooterConstants.SHOOTER1_MOTOR_CONFIG);
|
|
|
|
|
m_shooter2Motor.getConfigurator().apply(ShooterConstants.SHOOTER2_MOTOR_CONFIG);
|
|
|
|
|
m_indexerMotor.getConfigurator().apply(ShooterConstants.INDEXER_MOTOR_CONFIG);
|
2026-02-09 17:18:54 -08:00
|
|
|
|
|
|
|
|
shooter1Velocity.Slot = 0;
|
|
|
|
|
shooter2Velocity.Slot = 0;
|
|
|
|
|
m_indexerVelocity.Slot = 0;
|
2026-01-10 16:52:43 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private Angle clampAng(Angle x, Angle min, Angle max){
|
|
|
|
|
if(x.gt(max)) {
|
|
|
|
|
return max;
|
|
|
|
|
}else if(x.lt(min)) {
|
|
|
|
|
return min;
|
|
|
|
|
}else{
|
|
|
|
|
return x;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-01-27 18:13:51 -07:00
|
|
|
|
|
|
|
|
|
2026-01-27 19:15:37 -07:00
|
|
|
|
2026-01-27 18:13:51 -07:00
|
|
|
|
2026-01-13 10:08:44 -07:00
|
|
|
// // TODO: Test
|
|
|
|
|
// @Override
|
|
|
|
|
// public void setShooterAngle(ShooterState state, Angle angle) {
|
|
|
|
|
// state.shooterTargetAngle = angle;
|
|
|
|
|
// // Assume that the angle is always accurate, because I think we will use a shaft encoder
|
|
|
|
|
// // Assume that 0 degrees = forwards. Might need an offset here
|
2026-01-10 16:52:43 -07:00
|
|
|
|
2026-01-13 10:08:44 -07:00
|
|
|
// Angle boundedAngle = clampAng(angle, ShooterConstants.ANGLE_LIMIT_LEFT, ShooterConstants.ANGLE_LIMIT_RIGHT);
|
|
|
|
|
// // (REAL_ROT) * (MOTOR_ROT / REAL_ROT) = MOTOR_ROT
|
|
|
|
|
// double motorTargetAngle = boundedAngle.in(Rotations) / ShooterConstants.ANGLE_MOTOR_GEAR_RATIO;
|
|
|
|
|
// PositionDutyCycle posRequest = new PositionDutyCycle(motorTargetAngle);
|
|
|
|
|
// m_angleMotor.setControl(posRequest);
|
|
|
|
|
// }
|
2026-01-10 16:52:43 -07:00
|
|
|
|
|
|
|
|
|
2026-01-13 10:08:44 -07:00
|
|
|
// TODO: Test
|
|
|
|
|
// @Override
|
|
|
|
|
// public void setShooterPitch(ShooterState state, Angle angle) {
|
|
|
|
|
// state.shooterTargetPitch = angle;
|
|
|
|
|
// // TODO: Test
|
|
|
|
|
// // This assumes that the 0 is paralell to the ground. Might need an offset here
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Angle boundedAngle = clampAng(angle, ShooterConstants.PITCH_LIMIT_UPPER, ShooterConstants.PITCH_LIMIT_LOWER);
|
|
|
|
|
// // (REAL_ROT) * (MOTOR_ROT / REAL_ROT) = MOTOR_ROT
|
|
|
|
|
// double motorTargetAngle = boundedAngle.in(Rotations) / ShooterConstants.PITCH_MOTOR_GEAR_RATIO;
|
|
|
|
|
// PositionDutyCycle posRequest = new PositionDutyCycle(motorTargetAngle);
|
|
|
|
|
// m_pitchMotor.setControl(posRequest);
|
|
|
|
|
// }
|
2026-01-10 16:52:43 -07:00
|
|
|
|
2026-01-27 18:13:51 -07:00
|
|
|
|
|
|
|
|
@Override
|
2026-02-09 17:18:54 -08:00
|
|
|
public void setShooterVelocity(ShooterState state, AngularVelocity target) {
|
2026-01-27 19:15:37 -07:00
|
|
|
state.motor1TargetVelocity = target;
|
2026-01-27 18:13:51 -07:00
|
|
|
state.motor2TargetVelocity = target;
|
2026-01-27 19:15:37 -07:00
|
|
|
|
2026-02-09 18:38:55 -08:00
|
|
|
if(target.baseUnitMagnitude() == 0) {
|
|
|
|
|
m_shooter1Motor.set(0);
|
|
|
|
|
m_shooter2Motor.set(0);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2026-02-09 17:18:54 -08:00
|
|
|
AngularVelocity motorRps = target.div(ShooterConstants.INDEXER_GEAR_RATIO);
|
|
|
|
|
|
|
|
|
|
m_shooter1Motor.setControl(shooter1Velocity.withVelocity(motorRps));
|
|
|
|
|
m_shooter2Motor.setControl(shooter2Velocity.withVelocity(motorRps));
|
2026-01-10 16:52:43 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@Override
|
2026-01-27 18:13:51 -07:00
|
|
|
public void setIndexerVelocity(ShooterState state, AngularVelocity target) {
|
|
|
|
|
state.indexerTargetVelocity = target;
|
2026-02-09 17:18:54 -08:00
|
|
|
AngularVelocity motorRps = target.div(ShooterConstants.INDEXER_GEAR_RATIO);
|
|
|
|
|
m_indexerMotor.setControl(m_indexerVelocity.withVelocity(motorRps));
|
2026-01-10 16:52:43 -07:00
|
|
|
}
|
|
|
|
|
|
2026-02-09 17:18:54 -08:00
|
|
|
|
2026-01-10 16:52:43 -07:00
|
|
|
@Override
|
|
|
|
|
public void updateInputs(ShooterState state) {
|
|
|
|
|
|
2026-02-09 17:18:54 -08:00
|
|
|
state.motor1Velocity = m_shooter1Motor.getVelocity().getValue().times(ShooterConstants.SHOOTERMOTOR1_GEAR_RATIO);
|
|
|
|
|
state.motor2Velocity = m_shooter2Motor.getVelocity().getValue().times(ShooterConstants.SHOOTERMOTOR2_GEAR_RATIO);
|
|
|
|
|
state.indexerVelocity = m_indexerMotor.getVelocity().getValue().times(ShooterConstants.INDEXER_GEAR_RATIO);
|
2026-01-10 16:52:43 -07:00
|
|
|
|
2026-02-09 17:18:54 -08:00
|
|
|
// state.motorLinearVelocity = InchesPerSecond.of(m_shooter1Motor.getVelocity().getValue().in(RotationsPerSecond) * ShooterConstants.FEEDER_INCHES_PER_ROT);
|
2026-01-27 19:15:37 -07:00
|
|
|
|
2026-01-27 18:13:51 -07:00
|
|
|
state.motor1Current = m_shooter1Motor.getStatorCurrent().getValue();
|
2026-01-27 19:15:37 -07:00
|
|
|
state.motor2Current = m_shooter2Motor.getStatorCurrent().getValue();
|
|
|
|
|
state.indexerCurrent = m_indexerMotor.getStatorCurrent().getValue();
|
2026-02-09 17:18:54 -08:00
|
|
|
|
|
|
|
|
}
|
2026-02-09 18:38:55 -08:00
|
|
|
|
2026-02-09 17:18:54 -08:00
|
|
|
@Override
|
|
|
|
|
public void updateGains() {
|
|
|
|
|
// TEMPORARY PIDs
|
2026-02-09 18:38:55 -08:00
|
|
|
ShooterConstants.SHOOTER_PID.kP = ShooterConstants.shooter_kP.get();
|
|
|
|
|
ShooterConstants.SHOOTER_PID.kI = ShooterConstants.shooter_kI.get();
|
|
|
|
|
ShooterConstants.SHOOTER_PID.kD = ShooterConstants.shooter_kD.get();
|
2026-02-09 17:18:54 -08:00
|
|
|
m_shooter1Motor.getConfigurator().apply(ShooterConstants.SHOOTER_PID);
|
|
|
|
|
m_shooter2Motor.getConfigurator().apply(ShooterConstants.SHOOTER_PID);
|
|
|
|
|
|
2026-02-09 18:38:55 -08:00
|
|
|
ShooterConstants.INDEXER_PID.kP = ShooterConstants.indexer_kP.get();
|
|
|
|
|
ShooterConstants.INDEXER_PID.kI = ShooterConstants.indexer_kI.get();
|
|
|
|
|
ShooterConstants.INDEXER_PID.kD = ShooterConstants.indexer_kD.get();
|
2026-02-09 17:18:54 -08:00
|
|
|
m_indexerMotor.getConfigurator().apply(ShooterConstants.INDEXER_PID);
|
2026-01-10 16:52:43 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|