File size: 2,250 Bytes
4bde5d3 |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 |
import argparse
from pathlib import Path
import numpy as np
import scipy.spatial
from . import logger
from .pairs_from_retrieval import pairs_from_score_matrix
from .utils.read_write_model import read_images_binary
DEFAULT_ROT_THRESH = 30 # in degrees
def get_pairwise_distances(images):
ids = np.array(list(images.keys()))
Rs = []
ts = []
for id_ in ids:
image = images[id_]
R = image.qvec2rotmat()
t = image.tvec
Rs.append(R)
ts.append(t)
Rs = np.stack(Rs, 0)
ts = np.stack(ts, 0)
# Invert the poses from world-to-camera to camera-to-world.
Rs = Rs.transpose(0, 2, 1)
ts = -(Rs @ ts[:, :, None])[:, :, 0]
dist = scipy.spatial.distance.squareform(scipy.spatial.distance.pdist(ts))
# Instead of computing the angle between two camera orientations,
# we compute the angle between the principal axes, as two images rotated
# around their principal axis still observe the same scene.
axes = Rs[:, :, -1]
dots = np.einsum("mi,ni->mn", axes, axes, optimize=True)
dR = np.rad2deg(np.arccos(np.clip(dots, -1.0, 1.0)))
return ids, dist, dR
def main(model, output, num_matched, rotation_threshold=DEFAULT_ROT_THRESH):
logger.info("Reading the COLMAP model...")
images = read_images_binary(model / "images.bin")
logger.info(f"Obtaining pairwise distances between {len(images)} images...")
ids, dist, dR = get_pairwise_distances(images)
scores = -dist
invalid = dR >= rotation_threshold
np.fill_diagonal(invalid, True)
pairs = pairs_from_score_matrix(scores, invalid, num_matched)
pairs = [(images[ids[i]].name, images[ids[j]].name) for i, j in pairs]
logger.info(f"Found {len(pairs)} pairs.")
with open(output, "w") as f:
f.write("\n".join(" ".join(p) for p in pairs))
if __name__ == "__main__":
parser = argparse.ArgumentParser()
parser.add_argument("--model", required=True, type=Path)
parser.add_argument("--output", required=True, type=Path)
parser.add_argument("--num_matched", required=True, type=int)
parser.add_argument(
"--rotation_threshold", default=DEFAULT_ROT_THRESH, type=float
)
args = parser.parse_args()
main(**args.__dict__)
|