boofcv.io.geo.CodecBundleAdjustmentInTheLarge Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of boofcv-io Show documentation
Show all versions of boofcv-io Show documentation
BoofCV is an open source Java library for real-time computer vision and robotics applications.
/*
* Copyright (c) 2011-2020, Peter Abeles. All Rights Reserved.
*
* This file is part of BoofCV (http://boofcv.org).
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package boofcv.io.geo;
import boofcv.abst.geo.bundle.SceneObservations;
import boofcv.abst.geo.bundle.SceneObservations.View;
import boofcv.abst.geo.bundle.SceneStructureMetric;
import boofcv.alg.geo.bundle.cameras.BundlePinholeSnavely;
import boofcv.io.UtilIO;
import boofcv.struct.geo.PointIndex2D_F64;
import georegression.geometry.ConvertRotation3D_F64;
import georegression.struct.point.Point3D_F64;
import georegression.struct.se.Se3_F64;
import georegression.struct.so.Rodrigues_F64;
import java.io.*;
/**
* Reading and writing data in the Bundle Adjustment in the Large format.
*
* @author Peter Abeles
*/
public class CodecBundleAdjustmentInTheLarge {
public SceneStructureMetric scene;
public SceneObservations observations;
public void parse( File file ) throws IOException {
InputStream stream = UtilIO.openStream(file.getPath());
BufferedReader reader = new BufferedReader(new InputStreamReader(stream));
String words[] = reader.readLine().split("\\s+");
if( words.length != 3 )
throw new IOException("Unexpected number of words on first line");
int numCameras = Integer.parseInt(words[0]);
int numPoints = Integer.parseInt(words[1]);
int numObservations = Integer.parseInt(words[2]);
scene = new SceneStructureMetric(false);
scene.initialize(numCameras,numCameras,numPoints);
observations = new SceneObservations();
observations.initialize(numCameras);
for (int i = 0; i < numObservations; i++) {
words = reader.readLine().split("\\s+");
if (words.length != 4)
throw new IOException("Unexpected number of words in obs");
int cameraID = Integer.parseInt(words[0]);
int pointID = Integer.parseInt(words[1]);
float pixelX = Float.parseFloat(words[2]);
float pixelY = Float.parseFloat(words[3]);
if( pointID >= numPoints ) {
throw new RuntimeException("Out of bounds pointID");
}
if( cameraID >= numCameras ) {
throw new RuntimeException("Out of bounds cameraID");
}
observations.getView(cameraID).add(pointID,pixelX,pixelY);
}
Se3_F64 worldToCameraGL = new Se3_F64();
Rodrigues_F64 rod = new Rodrigues_F64();
for (int i = 0; i < numCameras; i++) {
rod.unitAxisRotation.x = Double.parseDouble(reader.readLine());
rod.unitAxisRotation.y = Double.parseDouble(reader.readLine());
rod.unitAxisRotation.z = Double.parseDouble(reader.readLine());
rod.theta = rod.unitAxisRotation.norm();
if( rod.theta != 0 )
rod.unitAxisRotation.divide(rod.theta);
worldToCameraGL.T.x = Double.parseDouble(reader.readLine());
worldToCameraGL.T.y = Double.parseDouble(reader.readLine());
worldToCameraGL.T.z = Double.parseDouble(reader.readLine());
ConvertRotation3D_F64.rodriguesToMatrix(rod,worldToCameraGL.R);
BundlePinholeSnavely camera = new BundlePinholeSnavely();
camera.f = Double.parseDouble(reader.readLine());
camera.k1 = Double.parseDouble(reader.readLine());
camera.k2 = Double.parseDouble(reader.readLine());
scene.setCamera(i,false,camera);
scene.setView(i,false,worldToCameraGL);
scene.connectViewToCamera(i,i);
}
Point3D_F64 P = new Point3D_F64();
for (int i = 0; i < numPoints; i++) {
P.x = Float.parseFloat(reader.readLine());
P.y = Float.parseFloat(reader.readLine());
P.z = Float.parseFloat(reader.readLine());
// GeometryMath_F64.mult(glToCv.R,P,P);
scene.setPoint(i,P.x,P.y,P.z);
}
for (int i = 0; i < observations.views.size; i++) {
View v = observations.getView(i);
for (int j = 0; j < v.point.size; j++) {
scene.connectPointToView(v.getPointId(j),i);
}
}
reader.close();
observations.checkOneObservationPerView();
}
public void save( File file ) throws IOException {
PrintStream writer = new PrintStream(file);
writer.println(scene.views.size+" "+scene.points.size+" "+observations.getObservationCount());
PointIndex2D_F64 o = new PointIndex2D_F64();
for (int viewIdx = 0; viewIdx < observations.views.size; viewIdx++) {
SceneObservations.View view = observations.views.data[viewIdx];
for (int obsIdx = 0; obsIdx < view.size(); obsIdx++) {
view.get(obsIdx,o);
writer.printf("%d %d %.8f %.8f\n",viewIdx,o.index,o.x,o.y);
}
}
Rodrigues_F64 axisAngle = new Rodrigues_F64();
for (int viewIdx = 0; viewIdx < scene.views.size; viewIdx++) {
SceneStructureMetric.View view = scene.views.data[viewIdx];
BundlePinholeSnavely camera = scene.cameras.get(view.camera).getModel();
ConvertRotation3D_F64.matrixToRodrigues(view.worldToView.R,axisAngle);
double axisX = axisAngle.unitAxisRotation.x*axisAngle.theta;
double axisY = axisAngle.unitAxisRotation.y*axisAngle.theta;
double axisZ = axisAngle.unitAxisRotation.z*axisAngle.theta;
writer.printf("%.10f\n%.10f\n%.10f\n",axisX,axisY,axisZ);
writer.printf("%.10f\n%.10f\n%.10f\n",view.worldToView.T.x,view.worldToView.T.y,view.worldToView.T.z);
writer.printf("%.10f\n%.10f\n%.10f\n",camera.f,camera.k1,camera.k2);
}
for (int pointId = 0; pointId < scene.points.size; pointId++) {
SceneStructureMetric.Point p = scene.points.data[pointId];
writer.printf("%.10f\n%.10f\n%.10f\n",p.coordinate[0],p.coordinate[1],p.coordinate[2]);
}
writer.close();
}
public static void main(String[] args) throws IOException {
CodecBundleAdjustmentInTheLarge alg = new CodecBundleAdjustmentInTheLarge();
alg.parse(new File("data/bundle_adjustment/ladybug/problem-49-7776-pre.txt"));
System.out.println("Done!");
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy