2019-09-19 14:07:42 -04:00
|
|
|
package com.chameleonvision.vision.camera;
|
|
|
|
|
|
2019-10-04 15:55:45 -04:00
|
|
|
import com.chameleonvision.settings.Platform;
|
2019-09-19 14:07:42 -04:00
|
|
|
import com.chameleonvision.vision.Pipeline;
|
2019-09-22 02:49:30 -04:00
|
|
|
import com.chameleonvision.web.ServerHandler;
|
2019-09-20 02:17:22 -04:00
|
|
|
import edu.wpi.cscore.*;
|
|
|
|
|
import edu.wpi.first.cameraserver.CameraServer;
|
|
|
|
|
import org.opencv.core.Mat;
|
2019-09-19 14:07:42 -04:00
|
|
|
|
2019-09-22 02:49:30 -04:00
|
|
|
import java.util.Arrays;
|
2019-09-19 14:07:42 -04:00
|
|
|
import java.util.HashMap;
|
|
|
|
|
import java.util.stream.IntStream;
|
|
|
|
|
|
|
|
|
|
public class Camera {
|
|
|
|
|
|
2019-10-20 10:13:07 +03:00
|
|
|
private static final double DEFAULT_FOV = 60.8;
|
|
|
|
|
private static final StreamDivisor DEFAULT_STREAMDIVISOR = StreamDivisor.none;
|
|
|
|
|
private static final int MINIMUM_FPS = 30;
|
|
|
|
|
private static final int MINIMUM_WIDTH = 320;
|
|
|
|
|
private static final int MINIMUM_HEIGHT = 200;
|
|
|
|
|
private static final int MAX_INIT_MS = 1500;
|
|
|
|
|
|
|
|
|
|
public final String name;
|
|
|
|
|
public final String path;
|
|
|
|
|
|
|
|
|
|
private final UsbCamera UsbCam;
|
|
|
|
|
private final VideoMode[] availableVideoModes;
|
|
|
|
|
|
|
|
|
|
private final CameraServer cs = CameraServer.getInstance();
|
|
|
|
|
private final CvSink cvSink;
|
|
|
|
|
private final Object cvSourceLock = new Object();
|
|
|
|
|
private CvSource cvSource;
|
|
|
|
|
private Double FOV;
|
|
|
|
|
private StreamDivisor streamDivisor;
|
|
|
|
|
private CameraValues camVals;
|
|
|
|
|
private CamVideoMode camVideoMode;
|
|
|
|
|
private int currentPipelineIndex;
|
|
|
|
|
private HashMap<Integer, Pipeline> pipelines;
|
|
|
|
|
|
|
|
|
|
public Camera(String cameraName) {
|
|
|
|
|
this(cameraName, DEFAULT_FOV);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Camera(String cameraName, double fov) {
|
|
|
|
|
this(cameraName, CameraManager.AllUsbCameraInfosByName.get(cameraName), fov);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Camera(String cameraName, UsbCameraInfo usbCameraInfo, double fov) {
|
|
|
|
|
this(cameraName, usbCameraInfo, fov, DEFAULT_STREAMDIVISOR);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Camera(String cameraName, UsbCameraInfo usbCamInfo, double fov, StreamDivisor divisor) {
|
|
|
|
|
this(cameraName, usbCamInfo, fov, new HashMap<>(), 0, divisor);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Camera(String cameraName, double fov, int videoModeIndex, StreamDivisor divisor) {
|
|
|
|
|
this(cameraName, fov, new HashMap<>(), videoModeIndex, divisor);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Camera(String cameraName, double fov, HashMap<Integer, Pipeline> pipelines, int videoModeIndex, StreamDivisor divisor) {
|
|
|
|
|
this(cameraName, CameraManager.AllUsbCameraInfosByName.get(cameraName), fov, pipelines, videoModeIndex, divisor);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Camera(String cameraName, UsbCameraInfo usbCamInfo, double fov, HashMap<Integer, Pipeline> pipelines, int videoModeIndex, StreamDivisor divisor) {
|
|
|
|
|
FOV = fov;
|
|
|
|
|
name = cameraName;
|
|
|
|
|
path = usbCamInfo.path;
|
|
|
|
|
streamDivisor = divisor;
|
|
|
|
|
UsbCam = new UsbCamera(name, path);
|
|
|
|
|
|
|
|
|
|
this.pipelines = pipelines;
|
|
|
|
|
|
|
|
|
|
// set up video modes according to minimums
|
|
|
|
|
if (Platform.getCurrentPlatform() == Platform.WINDOWS_64 && !UsbCam.isConnected()) {
|
|
|
|
|
System.out.print("Waiting on camera... ");
|
|
|
|
|
long initTimeout = System.nanoTime();
|
|
|
|
|
while (!UsbCam.isConnected()) {
|
|
|
|
|
if (((System.nanoTime() - initTimeout) / 1e6) >= MAX_INIT_MS) {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
var initTimeMs = (System.nanoTime() - initTimeout) / 1e6;
|
|
|
|
|
System.out.printf("Camera initialized in %.2fms\n", initTimeMs);
|
|
|
|
|
}
|
|
|
|
|
var trueVideoModes = UsbCam.enumerateVideoModes();
|
|
|
|
|
availableVideoModes = Arrays.stream(trueVideoModes).filter(v -> v.fps >= MINIMUM_FPS && v.width >= MINIMUM_WIDTH && v.height >= MINIMUM_HEIGHT).toArray(VideoMode[]::new);
|
|
|
|
|
if (availableVideoModes.length == 0) {
|
|
|
|
|
System.err.println("Camera not supported!");
|
|
|
|
|
throw new RuntimeException(new CameraException(CameraException.CameraExceptionType.BAD_CAMERA));
|
|
|
|
|
}
|
|
|
|
|
if (videoModeIndex <= availableVideoModes.length - 1) {
|
|
|
|
|
setCamVideoMode(videoModeIndex, false);
|
|
|
|
|
} else {
|
|
|
|
|
setCamVideoMode(0, false);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
cvSink = cs.getVideo(UsbCam);
|
|
|
|
|
cvSource = cs.putVideo(name, camVals.ImageWidth, camVals.ImageHeight);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
VideoMode[] getAvailableVideoModes() {
|
|
|
|
|
return availableVideoModes;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public int getStreamPort() {
|
|
|
|
|
var s = (MjpegServer) cs.getServer("serve_" + name);
|
|
|
|
|
return s.getPort();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void setCamVideoMode(int videoMode, boolean updateCvSource) {
|
|
|
|
|
setCamVideoMode(new CamVideoMode(availableVideoModes[videoMode]), updateCvSource);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void setCamVideoMode(CamVideoMode newVideoMode, boolean updateCvSource) {
|
|
|
|
|
var prevVideoMode = this.camVideoMode;
|
|
|
|
|
this.camVideoMode = newVideoMode;
|
|
|
|
|
UsbCam.setVideoMode(newVideoMode.getActualPixelFormat(), newVideoMode.width, newVideoMode.height, newVideoMode.fps);
|
|
|
|
|
|
|
|
|
|
// update camera values
|
|
|
|
|
camVals = new CameraValues(this);
|
|
|
|
|
if (prevVideoMode != null && !prevVideoMode.equals(newVideoMode) && updateCvSource) { // if resolution changed
|
|
|
|
|
synchronized (cvSourceLock) {
|
|
|
|
|
cvSource = cs.putVideo(name, newVideoMode.width, newVideoMode.height);
|
|
|
|
|
}
|
|
|
|
|
ServerHandler.sendFullSettings();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void addPipeline() {
|
|
|
|
|
addPipeline(pipelines.size());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void addPipeline(int pipelineNumber) {
|
|
|
|
|
if (pipelines.containsKey(pipelineNumber)) return;
|
|
|
|
|
pipelines.put(pipelineNumber, new Pipeline());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void deleteCurrentPipeline() {
|
|
|
|
|
pipelines.remove(getCurrentPipelineIndex());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Pipeline getCurrentPipeline() {
|
|
|
|
|
return pipelines.get(currentPipelineIndex);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public int getCurrentPipelineIndex() {
|
|
|
|
|
return currentPipelineIndex;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void setCurrentPipelineIndex(int pipelineNumber) {
|
|
|
|
|
if (pipelineNumber - 1 > pipelines.size()) return;
|
|
|
|
|
currentPipelineIndex = pipelineNumber;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public StreamDivisor getStreamDivisor() {
|
|
|
|
|
return streamDivisor;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void setStreamDivisor(int divisor) {
|
|
|
|
|
streamDivisor = StreamDivisor.values()[divisor];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public HashMap<Integer, Pipeline> getPipelines() {
|
|
|
|
|
return pipelines;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public CamVideoMode getVideoMode() {
|
|
|
|
|
return camVideoMode;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public int getVideoModeIndex() {
|
|
|
|
|
return IntStream.range(0, availableVideoModes.length)
|
|
|
|
|
.filter(i -> camVideoMode.equals(availableVideoModes[i]))
|
|
|
|
|
.findFirst()
|
|
|
|
|
.orElse(-1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public double getFOV() {
|
|
|
|
|
return FOV;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void setFOV(Number fov) {
|
|
|
|
|
FOV = fov.doubleValue();
|
|
|
|
|
camVals = new CameraValues(this);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public int getBrightness() {
|
|
|
|
|
return getCurrentPipeline().brightness;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void setBrightness(int brightness) {
|
|
|
|
|
getCurrentPipeline().brightness = brightness;
|
|
|
|
|
UsbCam.setBrightness(brightness);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void setExposure(int exposure) {
|
|
|
|
|
getCurrentPipeline().exposure = exposure;
|
|
|
|
|
UsbCam.setExposureManual(exposure);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public long grabFrame(Mat image) {
|
|
|
|
|
return cvSink.grabFrame(image);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public CameraValues getCamVals() {
|
|
|
|
|
return camVals;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void putFrame(Mat image) {
|
|
|
|
|
synchronized (cvSourceLock) {
|
|
|
|
|
cvSource.putFrame(image);
|
|
|
|
|
}
|
|
|
|
|
}
|
2019-09-19 14:07:42 -04:00
|
|
|
}
|