Beginnings of a different solution for the PhotonVision cameras

This commit is contained in:
2026-02-08 11:23:27 -05:00
parent b68d7b7399
commit ee4955485c
3 changed files with 76 additions and 47 deletions

View File

@@ -1,6 +1,9 @@
package frc.robot.constants; package frc.robot.constants;
import java.util.List;
import edu.wpi.first.math.geometry.Transform3d; import edu.wpi.first.math.geometry.Transform3d;
import frc.robot.utilities.PhotonVisionConfig;
public class PhotonConstants { public class PhotonConstants {
public static final String kCamera1Name = "pv1"; public static final String kCamera1Name = "pv1";
@@ -14,4 +17,9 @@ public class PhotonConstants {
public static final double kCamera1PitchRadians = 0; public static final double kCamera1PitchRadians = 0;
public static final double kCamera2HeightMeters = 0; public static final double kCamera2HeightMeters = 0;
public static final double kCamera2PitchRadians = 0; public static final double kCamera2PitchRadians = 0;
public static final List<PhotonVisionConfig> configs = List.of(
new PhotonVisionConfig(kCamera1Name, kCamera1RobotToCam, kCamera1HeightMeters, kCamera1PitchRadians),
new PhotonVisionConfig(kCamera2Name, kCamera2RobotToCam, kCamera2HeightMeters, kCamera2PitchRadians)
);
} }

View File

@@ -1,9 +1,11 @@
package frc.robot.utilities; package frc.robot.subsystems;
import java.io.IOException; import java.io.IOException;
import java.util.ArrayList;
import java.util.List; import java.util.List;
import java.util.Optional; import java.util.Optional;
import java.util.OptionalDouble; import java.util.OptionalDouble;
import java.util.function.Consumer;
import org.photonvision.EstimatedRobotPose; import org.photonvision.EstimatedRobotPose;
import org.photonvision.PhotonCamera; import org.photonvision.PhotonCamera;
@@ -15,65 +17,74 @@ import org.photonvision.targeting.PhotonTrackedTarget;
import edu.wpi.first.math.geometry.Transform3d; import edu.wpi.first.math.geometry.Transform3d;
import edu.wpi.first.math.util.Units; import edu.wpi.first.math.util.Units;
import edu.wpi.first.wpilibj2.command.SubsystemBase;
import edu.wpi.first.wpilibj2.command.button.Trigger;
import frc.robot.constants.CompetitionConstants; import frc.robot.constants.CompetitionConstants;
import frc.robot.constants.PhotonConstants;
import frc.robot.interfaces.IAprilTagProvider; import frc.robot.interfaces.IAprilTagProvider;
import frc.robot.interfaces.IVisualPoseProvider; import frc.robot.interfaces.IVisualPoseProvider;
import frc.robot.interfaces.IVisualPoseProvider.VisualPose;
import frc.robot.utilities.PhotonVisionConfig;
public class PhotonVision implements IAprilTagProvider,IVisualPoseProvider { public class PhotonVision extends SubsystemBase implements IAprilTagProvider {
private PhotonCamera[] cameras;
private PhotonPoseEstimator[] estimators;
private PhotonPipelineResult[] latestResults;
private final PhotonCamera camera; private ArrayList<Consumer<VisualPose>> poseEstimateConsumers;
private final PhotonPoseEstimator photonPoseEstimator; public PhotonVision() {
cameras = new PhotonCamera[PhotonConstants.configs.size()];
estimators = new PhotonPoseEstimator[PhotonConstants.configs.size()];
latestResults = new PhotonPipelineResult[PhotonConstants.configs.size()];
private final double cameraHeightMeters; for(int i = 0; i < PhotonConstants.configs.size(); i++) {
private final double cameraPitchRadians; cameras[i] = new PhotonCamera(PhotonConstants.configs.get(i).cameraName());
estimators[i] = new PhotonPoseEstimator(
private PhotonPipelineResult latestResult; CompetitionConstants.kTagLayout,
PoseStrategy.MULTI_TAG_PNP_ON_COPROCESSOR,
public PhotonVision(String cameraName, Transform3d robotToCam, double cameraHeightMeters, double cameraPitchRadians) throws IOException { PhotonConstants.configs.get(i).robotToCamera()
camera = new PhotonCamera(cameraName); );
latestResults[i] = null;
photonPoseEstimator = new PhotonPoseEstimator(
CompetitionConstants.kTagLayout,
PoseStrategy.MULTI_TAG_PNP_ON_COPROCESSOR,
robotToCam
);
this.cameraHeightMeters = cameraHeightMeters;
this.cameraPitchRadians = cameraPitchRadians;
this.latestResult = null;
}
// TODO This periodic method has to be called from somewhere, even though the cameras
// could be used in multiple other subsystems
public void periodic() {
// TODO Do we care about missed results? Probably not, if we're taking long enough to miss results something else is wrong
List<PhotonPipelineResult> results = camera.getAllUnreadResults();
if(!results.isEmpty()) {
latestResult = results.get(results.size() - 1);
} }
poseEstimateConsumers = new ArrayList<Consumer<VisualPose>>();
} }
@Override @Override
public Optional<VisualPose> getVisualPose() { public void periodic() {
if(latestResult == null) { for(int i = 0; i < cameras.length; i++) {
return Optional.empty(); List<PhotonPipelineResult> results = cameras[i].getAllUnreadResults();
if(!results.isEmpty()) {
latestResults[i] = results.get(results.size() - 1);
}
Optional<EstimatedRobotPose> pose = estimators[i].update(latestResults[i]);
if(!pose.isEmpty()) {
VisualPose visualPose = new VisualPose(
pose.get().estimatedPose.toPose2d(),
pose.get().timestampSeconds
);
for(Consumer<VisualPose> consumer: poseEstimateConsumers) {
consumer.accept(visualPose);
}
}
} }
}
Optional<EstimatedRobotPose> pose = photonPoseEstimator.update(latestResult); public Trigger tagPrescenseTrigger(int targetTag) {
return new Trigger(() -> {
if (pose.isEmpty()) { return List.of(latestResults).stream()
return Optional.empty(); .filter((p) -> p != null)
} .anyMatch((p) -> {
return p.getTargets().stream().map(PhotonTrackedTarget::getFiducialId).anyMatch((i) -> {
return Optional.of( return i == targetTag;
new VisualPose( });
pose.get().estimatedPose.toPose2d(), });
pose.get().timestampSeconds });
)
);
} }
@Override @Override

View File

@@ -0,0 +1,10 @@
package frc.robot.utilities;
import edu.wpi.first.math.geometry.Transform3d;
public record PhotonVisionConfig (
String cameraName,
Transform3d robotToCamera,
double cameraHeightMeters,
double cameraPitchRadians
) {}