import imageio import numpy as np import torch from PIL import Image from scipy.interpolate import UnivariateSpline from scipy.interpolate import interp1d def load_video(video_path): reader = imageio.get_reader(video_path) total_frames = reader.count_frames() frames = [] for i in range(total_frames): frame = reader.get_data(i) frames.append(Image.fromarray(frame)) reader.close() return frames def points_padding(points): padding = torch.ones_like(points)[..., 0:1] points = torch.cat([points, padding], dim=-1) return points def np_points_padding(points): padding = np.ones_like(points)[..., 0:1] points = np.concatenate([points, padding], axis=-1) return points def txt_interpolation(input_list, n, mode='smooth'): x = np.linspace(0, 1, len(input_list)) if mode == 'smooth': f = UnivariateSpline(x, input_list, k=3) elif mode == 'linear': f = interp1d(x, input_list) else: raise KeyError(f"Invalid txt interpolation mode: {mode}") xnew = np.linspace(0, 1, n) ynew = f(xnew) return ynew def traj_map(traj_type): # pre-defined trajectories if traj_type == "free1": # Zoom out and rotate to the upper left cam_traj = "free" x_offset = 0.0 y_offset = 0.0 z_offset = 0.0 d_theta = -15.0 d_phi = 45.0 d_r = 1.6 elif traj_type == "free2": # Rotate to the right horizontally cam_traj = "free" x_offset = -0.05 y_offset = 0.0 z_offset = 0.0 d_theta = 0.0 d_phi = -60.0 d_r = 1.0 elif traj_type == "free3": # Move back to the left cam_traj = "free" x_offset = -0.25 y_offset = 0.0 z_offset = 0.0 d_theta = 0.0 d_phi = 0.0 d_r = 1.7 elif traj_type == "free4": # Rotate and approach to the upper right cam_traj = "free" x_offset = 0.0 y_offset = 0.0 z_offset = 0.0 d_theta = -15.0 d_phi = -60.0 d_r = 0.75 elif traj_type == "free5": # Large-angle camera movement to the upper right cam_traj = "free" x_offset = 0.0 y_offset = 0.0 z_offset = 0.0 d_theta = -15.0 d_phi = -120.0 d_r = 1.6 elif traj_type == "swing1": # Swing shot 1 cam_traj = "swing1" x_offset = 0.0 y_offset = 0.0 z_offset = 0.0 d_theta = 0.0 d_phi = 0.0 d_r = 1.0 elif traj_type == "swing2": # Swing shot 2 cam_traj = "swing2" x_offset = 0.0 y_offset = 0.0 z_offset = 0.0 d_theta = 0.0 d_phi = 0.0 d_r = 1.0 elif traj_type == "orbit": # 360-degree counterclockwise rotation cam_traj = "free" x_offset = 0.0 y_offset = 0.0 z_offset = 0.0 d_theta = 0.0 d_phi = -360.0 d_r = 1.0 else: raise NotImplementedError return cam_traj, x_offset, y_offset, z_offset, d_theta, d_phi, d_r def set_initial_camera(start_elevation, radius): c2w_0 = torch.tensor([[1, 0, 0, 0], [0, 1, 0, 0], [0, 0, 1, -radius], [0, 0, 0, 1]], dtype=torch.float32) elevation_rad = np.deg2rad(start_elevation) R_elevation = torch.tensor([[1, 0, 0, 0], [0, np.cos(-elevation_rad), -np.sin(-elevation_rad), 0], [0, np.sin(-elevation_rad), np.cos(-elevation_rad), 0], [0, 0, 0, 1]], dtype=torch.float32) c2w_0 = R_elevation @ c2w_0 w2c_0 = c2w_0.inverse() return w2c_0, c2w_0 def build_cameras(cam_traj, w2c_0, c2w_0, intrinsic, nframe, focal_length, d_theta, d_phi, d_r, radius, x_offset, y_offset, z_offset): # build camera viewpoints according to d_theta,d_phi, d_r # return: w2cs:[V,4,4], c2ws:[V,4,4], intrinsic:[V,3,3] if intrinsic.ndim == 2: intrinsic = intrinsic[None].repeat(nframe, 1, 1) c2ws = [c2w_0] w2cs = [w2c_0] d_thetas, d_phis, d_rs = [], [], [] x_offsets, y_offsets, z_offsets = [], [], [] if cam_traj == "free": for i in range(nframe - 1): coef = (i + 1) / (nframe - 1) d_thetas.append(d_theta * coef) d_phis.append(d_phi * coef) d_rs.append(coef * d_r + (1 - coef) * 1.0) x_offsets.append(radius * x_offset * ((i + 1) / nframe)) y_offsets.append(radius * y_offset * ((i + 1) / nframe)) z_offsets.append(radius * z_offset * ((i + 1) / nframe)) elif cam_traj == "swing1": phis__ = [0, -5, -25, -30, -20, -8, 0] thetas__ = [0, -8, -12, -20, -17, -12, -5, -2, 1, 5, 3, 1, 0] rs__ = [0, 0.2] d_phis = txt_interpolation(phis__, nframe, mode='smooth') d_phis[0] = phis__[0] d_phis[-1] = phis__[-1] d_thetas = txt_interpolation(thetas__, nframe, mode='smooth') d_thetas[0] = thetas__[0] d_thetas[-1] = thetas__[-1] d_rs = txt_interpolation(rs__, nframe, mode='linear') d_rs = 1.0 + d_rs elif cam_traj == "swing2": phis__ = [0, 5, 25, 30, 20, 10, 0] thetas__ = [0, -5, -14, -11, 0, 1, 5, 3, 0] rs__ = [0, -0.03, -0.1, -0.2, -0.17, -0.1, 0] d_phis = txt_interpolation(phis__, nframe, mode='smooth') d_phis[0] = phis__[0] d_phis[-1] = phis__[-1] d_thetas = txt_interpolation(thetas__, nframe, mode='smooth') d_thetas[0] = thetas__[0] d_thetas[-1] = thetas__[-1] d_rs = txt_interpolation(rs__, nframe, mode='smooth') d_rs = 1.0 + d_rs else: raise NotImplementedError("Unknown trajectory type...") for i in range(nframe - 1): d_theta_rad = np.deg2rad(d_thetas[i]) R_theta = torch.tensor([[1, 0, 0, 0], [0, np.cos(d_theta_rad), -np.sin(d_theta_rad), 0], [0, np.sin(d_theta_rad), np.cos(d_theta_rad), 0], [0, 0, 0, 1]], dtype=torch.float32) d_phi_rad = np.deg2rad(d_phis[i]) R_phi = torch.tensor([[np.cos(d_phi_rad), 0, np.sin(d_phi_rad), 0], [0, 1, 0, 0], [-np.sin(d_phi_rad), 0, np.cos(d_phi_rad), 0], [0, 0, 0, 1]], dtype=torch.float32) c2w_1 = R_phi @ R_theta @ c2w_0 if i < len(x_offsets) and i < len(y_offsets) and i < len(z_offsets): c2w_1[:3, -1] += torch.tensor([x_offsets[i], y_offsets[i], z_offsets[i]]) c2w_1[:3, -1] *= d_rs[i] w2c_1 = c2w_1.inverse() c2ws.append(c2w_1) w2cs.append(w2c_1) intrinsic[i + 1, :2, :2] = intrinsic[i + 1, :2, :2] * focal_length * ((i + 1) / nframe) + \ intrinsic[i + 1, :2, :2] * ((nframe - (i + 1)) / nframe) w2cs = torch.stack(w2cs, dim=0) c2ws = torch.stack(c2ws, dim=0) return w2cs, c2ws, intrinsic