# yapf: disable import copy import glob import os import os.path as osp import shutil import warnings from functools import partial from pathlib import Path from typing import List, Optional, Tuple, Union import mmcv import numpy as np import torch import torch.nn as nn from colormap import Color from detrsmpl.core.cameras import ( WeakPerspectiveCameras, compute_orbit_cameras, ) from detrsmpl.core.cameras.builder import build_cameras from detrsmpl.core.conventions.cameras.convert_convention import \ convert_camera_matrix # prevent yapf isort conflict from detrsmpl.core.conventions.segmentation import body_segmentation from detrsmpl.core.renderer.torch3d_renderer import render_runner from detrsmpl.core.renderer.torch3d_renderer.meshes import \ ParametricMeshes # noqa: E501 from detrsmpl.core.renderer.torch3d_renderer.render_smpl_config import ( RENDER_CONFIGS, ) from detrsmpl.core.renderer.torch3d_renderer.smpl_renderer import SMPLRenderer from detrsmpl.core.renderer.torch3d_renderer.utils import \ align_input_to_padded # noqa: E501 from detrsmpl.models.body_models.builder import build_body_model from detrsmpl.utils.demo_utils import ( convert_bbox_to_intrinsic, convert_crop_cam_to_orig_img, convert_kp2d_to_bbox, get_default_hmr_intrinsic, get_different_colors, ) from detrsmpl.utils.ffmpeg_utils import ( check_input_path, images_to_array, prepare_output_path, vid_info_reader, video_to_array, video_to_images, ) from detrsmpl.utils.mesh_utils import save_meshes_as_objs, save_meshes_as_plys from detrsmpl.utils.path_utils import check_path_suffix # yapf: enable try: from typing import Literal except ImportError: from typing_extensions import Literal def _prepare_background(image_array, frame_list, origin_frames, output_path, start, end, img_format, overwrite, num_frames, read_frames_batch): """Compare among `image_array`, `frame_list` and `origin_frames` and decide whether to save the temp background images.""" if num_frames > 300: read_frames_batch = True frames_folder = None remove_folder = False if isinstance(image_array, np.ndarray): image_array = torch.Tensor(image_array) if image_array is not None: if image_array.ndim == 3: image_array = image_array[None] if image_array.shape[0] == 1: image_array = image_array.repeat(num_frames, 1, 1, 1) frame_list = None origin_frames = None image_array = image_array[start:end] # check the output path and get the image_array if output_path is not None: prepare_output_path(output_path=output_path, allowed_suffix=['.mp4', 'gif', '.png', '.jpg','.jpeg'], tag='output video', path_type='auto', overwrite=overwrite) if image_array is None: # choose in frame_list or origin_frames # if all None, will use pure white background if frame_list is None and origin_frames is None: print( 'No background provided, will use pure white background.') elif frame_list is not None and origin_frames is not None: warnings.warn('Redundant input, will only use frame_list.') origin_frames = None # read the origin frames as array if any. if frame_list is None and origin_frames is not None: check_input_path(input_path=origin_frames, allowed_suffix=['.mp4', '.gif', ''], tag='origin frames', path_type='auto') # if origin_frames is a video, write it as a folder of images # if read_frames_batch is True, else read directly as an array. if Path(origin_frames).is_file(): if read_frames_batch: frames_folder = osp.join( Path(output_path).parent, Path(output_path).name + '_input_temp') os.makedirs(frames_folder, exist_ok=True) video_to_images(origin_frames, frames_folder, img_format=img_format, start=start, end=end) remove_folder = True else: remove_folder = False frames_folder = None image_array = video_to_array(origin_frames, start=start, end=end) # if origin_frames is a folder, write it as a folder of images # read the folder as an array if read_frames_batch is True # else return frames_folder for reading during rendering. else: if read_frames_batch: frames_folder = origin_frames remove_folder = False image_array = None else: image_array = images_to_array(origin_frames, img_format=img_format, start=start, end=end) remove_folder = False frames_folder = origin_frames # if frame_list is not None, move the images into a folder # read the folder as an array if read_frames_batch is True # else return frames_folder for reading during rendering. elif frame_list is not None and origin_frames is None: frames_folder = osp.join( Path(output_path).parent, Path(output_path).name + '_input_temp') os.makedirs(frames_folder, exist_ok=True) for frame_idx, frame_path in enumerate(frame_list): if check_path_suffix( path_str=frame_path, allowed_suffix=['.jpg', '.png', '.jpeg']): shutil.copy( frame_path, os.path.join(frames_folder, '%06d.png' % frame_idx)) img_format = '%06d.png' if not read_frames_batch: image_array = images_to_array(frames_folder, img_format=img_format, remove_raw_files=True) frames_folder = None remove_folder = False else: image_array = None remove_folder = True return image_array, remove_folder, frames_folder def _prepare_body_model(body_model, body_model_config): """Prepare `body_model` from `body_model_config` or existing `body_model`.""" if body_model is None: if body_model_config is not None: body_model_config = copy.deepcopy(body_model_config) model_path = body_model_config.get('model_path', None) model_type = body_model_config.get('type').lower() if model_type not in ['smpl', 'smplx']: raise ValueError(f'Do not support {model_type}, please choose' f' in `smpl` or `smplx.') if model_path and osp.isdir(model_path): model_path = osp.join(model_path, model_type) body_model_config.update(model_path=model_path) body_model = build_body_model(body_model_config) assert os.path.isdir(model_path) else: raise FileNotFoundError('Wrong model_path.' ' File or directory does not exist.') else: raise ValueError('Please input body_model_config.') else: if body_model_config is not None: warnings.warn('Redundant input, will take body_model directly' 'and ignore body_model_config.') return body_model def _prepare_input_pose(verts, poses, betas, transl): """Prepare input pose data as tensor and ensure correct temporal slice.""" if verts is None and poses is None: raise ValueError('Please input valid poses or verts.') elif (verts is not None) and (poses is not None): warnings.warn('Redundant input, will take verts and ignore poses & ' 'betas & transl.') poses = None transl = None betas = None elif isinstance(poses, dict): transl = poses.get('transl', transl) betas = poses.get('betas', betas) if isinstance(verts, np.ndarray): verts = torch.Tensor(verts) num_frames = verts.shape[0] elif isinstance(verts, torch.Tensor): num_frames = verts.shape[0] if isinstance(poses, np.ndarray): poses = torch.Tensor(poses) num_frames = poses.shape[0] elif isinstance(poses, torch.Tensor): num_frames = poses.shape[0] elif isinstance(poses, dict): for k, v in poses.items(): if isinstance(v, np.ndarray): poses[k] = torch.tensor(v) num_frames = poses['body_pose'].shape[0] if isinstance(betas, np.ndarray): betas = torch.Tensor(betas) if betas is not None: if betas.shape[0] != num_frames: times = num_frames // betas.shape[0] if betas.ndim == 2: betas = betas.repeat(times, 1)[:num_frames] elif betas.ndim == 3: betas = betas.repeat(times, 1, 1)[:num_frames] print(f'betas will be repeated by dim 0 for {times} times.') if isinstance(transl, np.ndarray): transl = torch.Tensor(transl) return verts, poses, betas, transl def _prepare_mesh(poses, betas, transl, verts, start, end, body_model): """Prepare the mesh info for rendering.""" NUM_JOINTS = body_model.NUM_JOINTS NUM_BODY_JOINTS = body_model.NUM_BODY_JOINTS NUM_DIM = 3 * (NUM_JOINTS + 1) body_pose_keys = body_model.body_pose_keys joints = None if poses is not None: if isinstance(poses, dict): if not body_pose_keys.issubset(poses): raise KeyError( f'{str(poses.keys())}, Please make sure that your ' f'input dict has all of {", ".join(body_pose_keys)}') num_frames = poses['body_pose'].shape[0] _, num_person, _ = poses['body_pose'].view( num_frames, -1, NUM_BODY_JOINTS * 3).shape full_pose = body_model.dict2tensor(poses) full_pose = full_pose[start:end] elif isinstance(poses, torch.Tensor): if poses.shape[-1] != NUM_DIM: raise ValueError( f'Please make sure your poses is {NUM_DIM} dims in' f'the last axis. Your input shape: {poses.shape}') poses = poses.view(poses.shape[0], -1, (NUM_JOINTS + 1) * 3) num_frames, num_person, _ = poses.shape full_pose = poses[start:end] else: raise ValueError('Wrong pose type, should be `dict` or `tensor`.') # multi person check if num_person > 1: if betas is not None: num_betas = betas.shape[-1] betas = betas.view(num_frames, -1, num_betas) if betas.shape[1] == 1: betas = betas.repeat(1, num_person, 1) warnings.warn( 'Only one betas for multi-person, will all be the ' 'same body shape.') elif betas.shape[1] > num_person: betas = betas[:, :num_person] warnings.warn( f'Betas shape exceed, will be sliced as {betas.shape}.' ) elif betas.shape[1] == num_person: pass else: raise ValueError( f'Odd betas shape: {betas.shape}, inconsistent' f'with poses in num_person: {poses.shape}.') else: warnings.warn('None betas for multi-person, will all be the ' 'default body shape.') if transl is not None: transl = transl.view(poses.shape[0], -1, 3) if transl.shape[1] == 1: transl = transl.repeat(1, num_person, 1) warnings.warn( 'Only one transl for multi-person, will all be the ' 'same translation.') elif transl.shape[1] > num_person: transl = transl[:, :num_person] warnings.warn(f'Transl shape exceed, will be sliced as' f'{transl.shape}.') elif transl.shape[1] == num_person: pass else: raise ValueError( f'Odd transl shape: {transl.shape}, inconsistent' f'with poses in num_person: {poses.shape}.') else: warnings.warn('None transl for multi-person, will all be the ' 'default translation.') # slice the input poses, betas, and transl. betas = betas[start:end] if betas is not None else None transl = transl[start:end] if transl is not None else None pose_dict = body_model.tensor2dict(full_pose=full_pose, betas=betas, transl=transl) # get new num_frames num_frames = full_pose.shape[0] model_output = body_model(**pose_dict) vertices = model_output['vertices'] joints = model_output['joints'][0] # hardcode here elif verts is not None: if isinstance(verts, np.ndarray): verts = torch.Tensor(verts) verts = verts[start:end] pose_dict = body_model.tensor2dict(torch.zeros(1, (NUM_JOINTS + 1) * 3)) if verts.ndim == 3: joints = torch.einsum('bik,ji->bjk', [verts, body_model.J_regressor]) elif verts.ndim == 4: joints = torch.einsum('fpik,ji->fpjk', [verts, body_model.J_regressor]) num_verts = body_model.NUM_VERTS assert verts.shape[-2] == num_verts, 'Wrong input verts shape.' num_frames = verts.shape[0] vertices = verts.view(num_frames, -1, num_verts, 3) num_joints = joints.shape[-2] joints = joints.view(num_frames, -1, num_joints, 3) num_person = vertices.shape[1] else: raise ValueError('Poses and verts are all None.') return vertices, joints, num_frames, num_person def _prepare_colors(palette, render_choice, num_person, num_verts, model_type): """Prepare the `color` as a tensor of shape (num_person, num_verts, 3) according to `palette`. This is to make the identity in video clear. """ if not len(palette) == num_person: raise ValueError('Please give the right number of palette.') body_segger = body_segmentation(model_type) if render_choice == 'silhouette': colors = torch.ones(num_person, num_verts, 3) elif render_choice == 'part_silhouette': colors = torch.zeros(num_person, num_verts, 3) for i, k in enumerate(body_segger.keys()): colors[:, body_segger[k]] = i + 1 else: if isinstance(palette, torch.Tensor): if palette.max() > 1: palette = palette / 255.0 palette = torch.clip(palette, min=0, max=1) colors = palette.view(num_person, 3).unsqueeze(1).repeat(1, num_verts, 1) elif isinstance(palette, list): colors = [] for person_idx in range(num_person): if palette[person_idx] == 'random': color_person = get_different_colors( num_person, int_dtype=False)[person_idx] color_person = torch.FloatTensor(color_person) color_person = torch.clip(color_person * 1.5, min=0.6, max=1) color_person = color_person.view(1, 1, 3).repeat( 1, num_verts, 1) elif palette[person_idx] == 'segmentation': verts_labels = torch.zeros(num_verts) color_person = torch.ones(1, num_verts, 3) color_part = get_different_colors(len(body_segger), int_dtype=False) for part_idx, k in enumerate(body_segger.keys()): index = body_segger[k] verts_labels[index] = part_idx color_person[:, index] = torch.FloatTensor( color_part[part_idx]) elif palette[person_idx] in Color.color_names: color_person = torch.FloatTensor( Color(palette[person_idx]).rgb).view(1, 1, 3).repeat( 1, num_verts, 1) else: raise ValueError('Wrong palette string. ' 'Please choose in the pre-defined range.') colors.append(color_person) colors = torch.cat(colors, 0) assert colors.shape == (num_person, num_verts, 3) # the color passed to renderer will be (num_person, num_verts, 3) else: raise ValueError( 'Palette should be tensor, array or list of strs.') return colors def render_smpl( # smpl parameters poses: Optional[Union[torch.Tensor, np.ndarray, dict]] = None, betas: Optional[Union[torch.Tensor, np.ndarray]] = None, transl: Optional[Union[torch.Tensor, np.ndarray]] = None, verts: Optional[Union[torch.Tensor, np.ndarray]] = None, body_model: Optional[nn.Module] = None, body_model_config: Optional[dict] = None, # camera parameters R: Optional[Union[torch.Tensor, np.ndarray]] = None, T: Optional[Union[torch.Tensor, np.ndarray]] = None, K: Optional[Union[torch.Tensor, np.ndarray]] = None, orig_cam: Optional[Union[torch.Tensor, np.ndarray]] = None, Ks: Optional[Union[torch.Tensor, np.ndarray]] = None, in_ndc: bool = True, convention: str = 'pytorch3d', projection: Literal['weakperspective', 'perspective', 'fovperspective', 'orthographics', 'fovorthographics'] = 'perspective', orbit_speed: Union[float, Tuple[float, float]] = 0.0, # render choice parameters render_choice: Literal['lq', 'mq', 'hq', 'silhouette', 'depth', 'normal', 'pointcloud', 'part_silhouette'] = 'hq', palette: Union[List[str], str, np.ndarray, torch.Tensor] = 'white', texture_image: Union[torch.Tensor, np.ndarray] = None, resolution: Optional[Union[List[int], Tuple[int, int]]] = None, start: int = 0, end: Optional[int] = None, alpha: float = 1.0, no_grad: bool = True, batch_size: int = 10, device: Union[torch.device, str] = 'cuda', # file io parameters return_tensor: bool = False, output_path: str = None, origin_frames: Optional[str] = None, frame_list: Optional[List[str]] = None, image_array: Optional[Union[np.ndarray, torch.Tensor]] = None, img_format: str = '%06d.png', overwrite: bool = False, mesh_file_path: Optional[str] = None, read_frames_batch: bool = False, # visualize keypoints plot_kps: bool = False, kp3d: Optional[Union[np.ndarray, torch.Tensor]] = None, mask: Optional[Union[np.ndarray, List[int]]] = None, vis_kp_index: bool = False, verbose: bool = False) -> Union[None, torch.Tensor]: """Render SMPL or SMPL-X mesh or silhouette into differentiable tensors, and export video or images. Args: # smpl parameters: poses (Union[torch.Tensor, np.ndarray, dict]): 1). `tensor` or `array` and ndim is 2, shape should be (frame, 72). 2). `tensor` or `array` and ndim is 3, shape should be (frame, num_person, 72/165). num_person equals 1 means single-person. Rendering predicted multi-person should feed together with multi-person weakperspective cameras. meshes would be computed and use an identity intrinsic matrix. 3). `dict`, standard dict format defined in smplx.body_models. will be treated as single-person. Lower priority than `verts`. Defaults to None. betas (Optional[Union[torch.Tensor, np.ndarray]], optional): 1). ndim is 2, shape should be (frame, 10). 2). ndim is 3, shape should be (frame, num_person, 10). num_person equals 1 means single-person. If poses are multi-person, betas should be set to the same person number. None will use default betas. Defaults to None. transl (Optional[Union[torch.Tensor, np.ndarray]], optional): translations of smpl(x). 1). ndim is 2, shape should be (frame, 3). 2). ndim is 3, shape should be (frame, num_person, 3). num_person equals 1 means single-person. If poses are multi-person, transl should be set to the same person number. Defaults to None. verts (Optional[Union[torch.Tensor, np.ndarray]], optional): 1). ndim is 3, shape should be (frame, num_verts, 3). 2). ndim is 4, shape should be (frame, num_person, num_verts, 3). num_person equals 1 means single-person. Higher priority over `poses` & `betas` & `transl`. Defaults to None. body_model (nn.Module, optional): body_model created from smplx.create. Higher priority than `body_model_config`. If `body_model` is not None, it will override `body_model_config`. Should not both be None. Defaults to None. body_model_config (dict, optional): body_model_config for build_model. Lower priority than `body_model`. Should not both be None. Defaults to None. # camera parameters: K (Optional[Union[torch.Tensor, np.ndarray]], optional): shape should be (frame, 4, 4) or (frame, 3, 3), frame could be 1. if (4, 4) or (3, 3), dim 0 will be added automatically. Will be default `FovPerspectiveCameras` intrinsic if None. Lower priority than `orig_cam`. R (Optional[Union[torch.Tensor, np.ndarray]], optional): shape should be (frame, 3, 3), If f equals 1, camera will have identical rotation. If `K` and `orig_cam` is None, will be generated by `look_at_view`. If have `K` or `orig_cam` and `R` is None, will be generated by `convert_camera_matrix`. Defaults to None. T (Optional[Union[torch.Tensor, np.ndarray]], optional): shape should be (frame, 3). If f equals 1, camera will have identical translation. If `K` and `orig_cam` is None, will be generated by `look_at_view`. If have `K` or `orig_cam` and `T` is None, will be generated by `convert_camera_matrix`. Defaults to None. orig_cam (Optional[Union[torch.Tensor, np.ndarray]], optional): shape should be (frame, 4) or (frame, num_person, 4). If f equals 1, will be repeated to num_frames. num_person should be 1 if single person. Usually for HMR, VIBE predicted cameras. Higher priority than `K` & `R` & `T`. Defaults to None. Ks (Optional[Union[torch.Tensor, np.ndarray]], optional): shape should be (frame, 4, 4). This is for HMR or SPIN multi-person demo. in_ndc (bool, optional): . Defaults to True. convention (str, optional): If want to use an existing convention, choose in ['opengl', 'opencv', 'pytorch3d', 'pyrender', 'open3d', 'maya', 'blender', 'unity']. If want to use a new convention, define your convention in (CAMERA_CONVENTION_FACTORY)[mmhuman3d/core/conventions/cameras/ __init__.py] by the order of right, front and up. Defaults to 'pytorch3d'. projection (Literal[, optional): projection mode of camers. Choose in ['orthographics, fovperspective', 'perspective', 'weakperspective', 'fovorthographics'] Defaults to 'perspective'. orbit_speed (float, optional): orbit speed for viewing when no `K` provided. `float` for only azim speed and Tuple for `azim` and `elev`. # render choice parameters: render_choice (Literal[, optional): choose in ['lq', 'mq', 'hq', 'silhouette', 'depth', 'normal', 'pointcloud', 'part_silhouette'] . `lq`, `mq`, `hq` would output (frame, h, w, 4) FloatTensor. `lq` means low quality, `mq` means medium quality, h`q means high quality. `silhouette` would output (frame, h, w) soft binary FloatTensor. `part_silhouette` would output (frame, h, w, 1) LongTensor. Every pixel stores a class index. `depth` will output a depth map of (frame, h, w, 1) FloatTensor and 'normal' will output a normal map of (frame, h, w, 1). `pointcloud` will output a (frame, h, w, 4) FloatTensor. Defaults to 'mq'. palette (Union[List[str], str, np.ndarray], optional): color theme str or list of color str or `array`. 1). If use str to represent the color, should choose in ['segmentation', 'random'] or color from Colormap https://en.wikipedia.org/wiki/X11_color_names. If choose 'segmentation', will get a color for each part. 2). If you have multi-person, better give a list of str or all will be in the same color. 3). If you want to define your specific color, use an `array` of shape (3,) for single person and (N, 3) for multiple persons. If (3,) for multiple persons, all will be in the same color. Your `array` should be in range [0, 255] for 8 bit color. Defaults to 'white'. texture_image (Union[torch.Tensor, np.ndarray], optional): Texture image to be wrapped on the smpl mesh. If not None, the `palette` will be ignored, and the `body_model` is required to have `uv_param_path`. Should pass list or tensor of shape (num_person, H, W, 3). The color channel should be `RGB`. Defaults to None. resolution (Union[Iterable[int], int], optional): 1). If iterable, should be (height, width) of output images. 2). If int, would be taken as (resolution, resolution). Defaults to (1024, 1024). This will influence the overlay results when render with backgrounds. The output video will be rendered following the size of background images and finally resized to resolution. start (int, optional): start frame index. Defaults to 0. end (int, optional): end frame index. Exclusive. Could be positive int or negative int or None. None represents include all the frames. Defaults to None. alpha (float, optional): Transparency of the mesh. Range in [0.0, 1.0] Defaults to 1.0. no_grad (bool, optional): Set to True if do not need differentiable render. Defaults to False. batch_size (int, optional): Batch size for render. Related to your gpu memory. Defaults to 10. # file io parameters: return_tensor (bool, optional): Whether return the result tensors. Defaults to False, will return None. output_path (str, optional): output video or gif or image folder. Defaults to None, pass export procedure. # background frames, priority: image_array > frame_list > origin_frames origin_frames (Optional[str], optional): origin background frame path, could be `.mp4`, `.gif`(will be sliced into a folder) or an image folder. Defaults to None. frame_list (Optional[List[str]], optional): list of origin background frame paths, element in list each should be a image path like `*.jpg` or `*.png`. Use this when your file names is hard to sort or you only want to render a small number frames. Defaults to None. image_array: (Optional[Union[np.ndarray, torch.Tensor]], optional): origin background frame `tensor` or `array`, use this when you want your frames in memory as array or tensor. overwrite (bool, optional): whether overwriting the existing files. Defaults to False. mesh_file_path (bool, optional): the directory path to store the `.ply` or '.ply' files. Will be named like 'frame_idx_person_idx.ply'. Defaults to None. read_frames_batch (bool, optional): Whether read frames by batch. Set it as True if your video is large in size. Defaults to False. # visualize keypoints plot_kps (bool, optional): whether plot keypoints on the output video. Defaults to False. kp3d (Optional[Union[np.ndarray, torch.Tensor]], optional): the keypoints of any convention, should pass `mask` if have any none-sense points. Shape should be (frame, ) Defaults to None. mask (Optional[Union[np.ndarray, List[int]]], optional): Mask of keypoints existence. Defaults to None. vis_kp_index (bool, optional): Whether plot keypoint index number on human mesh. Defaults to False. # visualize render progress verbose (bool, optional): Whether print the progress bar for rendering. Returns: Union[None, torch.Tensor]: return the rendered image tensors or None. """ # initialize the device device = torch.device(device) if isinstance(device, str) else device if isinstance(resolution, int): resolution = (resolution, resolution) elif isinstance(resolution, list): resolution = tuple(resolution) verts, poses, betas, transl = _prepare_input_pose(verts, poses, betas, transl) body_model = _prepare_body_model(body_model, body_model_config) model_type = body_model.name().replace('-', '').lower() assert model_type in ['smpl', 'smplx'] vertices, joints, num_frames, num_person = _prepare_mesh( poses, betas, transl, verts, start, end, body_model) end = num_frames if end is None else end vertices = vertices.view(num_frames, num_person, -1, 3) num_verts = vertices.shape[-2] if not plot_kps: joints = None if kp3d is not None: warnings.warn('`plot_kps` is False, `kp3d` will be set as None.') kp3d = None image_array, remove_folder, frames_folder = _prepare_background( image_array, frame_list, origin_frames, output_path, start, end, img_format, overwrite, num_frames, read_frames_batch) render_resolution = None if image_array is not None: render_resolution = (image_array.shape[1], image_array.shape[2]) elif frames_folder is not None: frame_path_list = glob.glob(osp.join( frames_folder, '*.jpg')) + glob.glob( osp.join(frames_folder, '*.png')) + glob.glob( osp.join(frames_folder, '*.jpeg')) vid_info = vid_info_reader(frame_path_list[0]) render_resolution = (int(vid_info['height']), int(vid_info['width'])) if resolution is not None: if render_resolution is not None: if render_resolution != resolution: warnings.warn( f'Size of background: {render_resolution} !=' f' resolution: {resolution}, the output video will be ' f'resized as {resolution}') final_resolution = resolution elif render_resolution is None: render_resolution = final_resolution = resolution elif resolution is None: if render_resolution is None: render_resolution = final_resolution = (1024, 1024) elif render_resolution is not None: final_resolution = render_resolution if isinstance(kp3d, np.ndarray): kp3d = torch.Tensor(kp3d) if kp3d is not None: if mask is not None: map_index = np.where(np.array(mask) != 0)[0] kp3d = kp3d[map_index.tolist()] kp3d = kp3d[start:end] kp3d = kp3d.view(num_frames, -1, 3) # prepare render_param_dict render_param_dict = copy.deepcopy(RENDER_CONFIGS[render_choice.lower()]) if model_type == 'smpl': render_param_dict.update(num_class=24) elif model_type == 'smplx': render_param_dict.update(num_class=27) if render_choice not in [ 'hq', 'mq', 'lq', 'silhouette', 'part_silhouette', 'depth', 'pointcloud', 'normal' ]: raise ValueError('Please choose the right render_choice.') # body part colorful visualization should use flat shader to be sharper. if texture_image is None: if isinstance(palette, str): palette = [palette] * num_person elif isinstance(palette, np.ndarray): palette = torch.Tensor(palette) palette = palette.view(-1, 3) if palette.shape[0] != num_person: _times = num_person // palette.shape[0] palette = palette.repeat(_times, 1)[:num_person] if palette.shape[0] == 1: print(f'Same color for all the {num_person} people') else: print('Repeat palette for multi-person.') else: raise ValueError('Wrong input palette type. ' 'Palette should be tensor, array or list of strs') colors_all = _prepare_colors(palette, render_choice, num_person, num_verts, model_type) colors_all = colors_all.view(-1, num_person * num_verts, 3) # verts of ParametricMeshes should be in (N, V, 3) vertices = vertices.view(num_frames, -1, 3) meshes = ParametricMeshes( body_model=body_model, verts=vertices, N_individual_overdide=num_person, model_type=model_type, texture_image=texture_image, use_nearest=bool(render_choice == 'part_silhouette'), vertex_color=colors_all) # write .ply or .obj files if mesh_file_path is not None: mmcv.mkdir_or_exist(mesh_file_path) for person_idx in range(meshes.shape[1]): mesh_person = meshes[:, person_idx] if texture_image is None: ply_paths = [ f'{mesh_file_path}/frame{frame_idx}_' f'person{person_idx}.ply' for frame_idx in range(num_frames) ] save_meshes_as_plys(meshes=mesh_person, files=ply_paths) else: obj_paths = [ f'{mesh_file_path}/frame{frame_idx}_' f'person{person_idx}.obj' for frame_idx in range(num_frames) ] save_meshes_as_objs(meshes=mesh_person, files=obj_paths) vertices = meshes.verts_padded().view(num_frames, num_person, -1, 3) # prepare camera matrixs if Ks is not None: projection = 'perspective' orig_cam = None if isinstance(Ks, np.ndarray): Ks = torch.Tensor(Ks) Ks = Ks.view(-1, num_person, 3, 3) Ks = Ks[start:end] Ks = Ks.view(-1, 3, 3) K = K.repeat(num_frames * num_person, 1, 1) Ks = K.inverse() @ Ks @ K vertices = vertices.view(num_frames * num_person, -1, 3) if T is None: T = torch.zeros(num_frames, num_person, 1, 3) elif isinstance(T, np.ndarray): T = torch.Tensor(T) T = T[start:end] T = T.view(num_frames * num_person, 1, 3) vertices = torch.einsum('blc,bvc->bvl', Ks, vertices + T) R = None T = None vertices = vertices.view(num_frames, num_person, -1, 3) if orig_cam is not None: if isinstance(orig_cam, np.ndarray): orig_cam = torch.Tensor(orig_cam) projection = 'weakperspective' r = render_resolution[1] / render_resolution[0] orig_cam = orig_cam[start:end] orig_cam = orig_cam.view(num_frames, num_person, 4) # if num_person > 1: sx, sy, tx, ty = torch.unbind(orig_cam, -1) vertices[..., 0] += tx.view(num_frames, num_person, 1) vertices[..., 1] += ty.view(num_frames, num_person, 1) vertices[..., 0] *= sx.view(num_frames, num_person, 1) vertices[..., 1] *= sy.view(num_frames, num_person, 1) orig_cam = torch.tensor([1.0, 1.0, 0.0, 0.0]).view(1, 4).repeat(num_frames, 1) K, R, T = WeakPerspectiveCameras.convert_orig_cam_to_matrix( orig_cam=orig_cam, znear=torch.min(vertices[..., 2] - 1), aspect_ratio=r) if num_person > 1: vertices = vertices.reshape(num_frames, -1, 3) else: vertices = vertices.view(num_frames, -1, 3) meshes = meshes.update_padded(new_verts_padded=vertices) # orig_cam and K are None, use look_at_view if K is None: projection = 'fovperspective' K, R, T = compute_orbit_cameras(at=(torch.mean(vertices.view(-1, 3), 0)).detach().cpu(), orbit_speed=orbit_speed, batch_size=num_frames, convention=convention) convention = 'pytorch3d' if isinstance(R, np.ndarray): R = torch.Tensor(R).view(-1, 3, 3) elif isinstance(R, torch.Tensor): R = R.view(-1, 3, 3) elif isinstance(R, list): R = torch.Tensor(R).view(-1, 3, 3) elif R is None: pass else: raise ValueError(f'Wrong type of R: {type(R)}!') if R is not None: if len(R) > num_frames: R = R[start:end] if isinstance(T, np.ndarray): T = torch.Tensor(T).view(-1, 3) elif isinstance(T, torch.Tensor): T = T.view(-1, 3) elif isinstance(T, list): T = torch.Tensor(T).view(-1, 3) elif T is None: pass else: raise ValueError(f'Wrong type of T: {type(T)}!') if T is not None: if len(T) > num_frames: T = T[start:end] if isinstance(K, np.ndarray): K = torch.Tensor(K).view(-1, K.shape[-2], K.shape[-1]) elif isinstance(K, torch.Tensor): K = K.view(-1, K.shape[-2], K.shape[-1]) elif isinstance(K, list): K = torch.Tensor(K) K = K.view(-1, K.shape[-2], K.shape[-1]) else: raise ValueError(f'Wrong type of K: {type(K)}!') if K is not None: if len(K) > num_frames: K = K[start:end] assert projection in [ 'perspective', 'weakperspective', 'orthographics', 'fovorthographics', 'fovperspective' ], f'Wrong camera projection: {projection}' if projection in ['fovperspective', 'perspective']: is_perspective = True elif projection in [ 'fovorthographics', 'weakperspective', 'orthographics' ]: is_perspective = False if projection in ['fovperspective', 'fovorthographics', 'weakperspective']: assert in_ndc K, R, T = convert_camera_matrix(convention_dst='pytorch3d', K=K, R=R, T=T, is_perspective=is_perspective, convention_src=convention, resolution_src=render_resolution, in_ndc_src=in_ndc, in_ndc_dst=in_ndc) # initialize the renderer. renderer = SMPLRenderer(resolution=render_resolution, device=device, output_path=output_path, return_tensor=return_tensor, alpha=alpha, read_img_format=img_format, render_choice=render_choice, frames_folder=frames_folder, plot_kps=plot_kps, vis_kp_index=vis_kp_index, final_resolution=final_resolution, **render_param_dict) cameras = build_cameras( dict(type=projection, in_ndc=in_ndc, device=device, K=K, R=R, T=T, resolution=render_resolution)) if image_array is not None: image_array = torch.Tensor(image_array) image_array = align_input_to_padded(image_array, ndim=4, batch_size=num_frames, padding_mode='ones') # prepare the render data. render_data = dict( images=image_array, meshes=meshes, cameras=cameras, joints=joints, joints_gt=kp3d, ) results = render_runner.render(renderer=renderer, device=device, batch_size=batch_size, output_path=output_path, return_tensor=return_tensor, no_grad=no_grad, verbose=verbose, **render_data) if remove_folder: if Path(frames_folder).is_dir(): shutil.rmtree(frames_folder) if return_tensor: return results else: return None def visualize_smpl_calibration( K, R, T, resolution, **kwargs, ) -> None: """Visualize a smpl mesh which has opencv calibration matrix defined in screen.""" assert K is not None, '`K` is required.' assert resolution is not None, '`resolution`(h, w) is required.' func = partial(render_smpl, projection='perspective', convention='opencv', orig_cam=None, in_ndc=False) for k in func.keywords.keys(): if k in kwargs: kwargs.pop(k) return func(K=K, R=R, T=T, resolution=resolution, **kwargs) def visualize_smpl_hmr(cam_transl, bbox=None, kp2d=None, focal_length=5000, det_width=224, det_height=224, bbox_format='xyxy', **kwargs) -> None: """Simplest way to visualize HMR or SPIN or Smplify pred smpl with origin frames and predicted cameras.""" if kp2d is not None: bbox = convert_kp2d_to_bbox(kp2d, bbox_format=bbox_format) Ks = convert_bbox_to_intrinsic(bbox, bbox_format=bbox_format) K = torch.Tensor( get_default_hmr_intrinsic(focal_length=focal_length, det_height=det_height, det_width=det_width)) func = partial( render_smpl, projection='perspective', convention='opencv', in_ndc=False, K=None, R=None, orig_cam=None, ) if isinstance(cam_transl, np.ndarray): cam_transl = torch.Tensor(cam_transl) T = torch.cat([ cam_transl[..., [1]], cam_transl[..., [2]], 2 * focal_length / (det_width * cam_transl[..., [0]] + 1e-9) ], -1) for k in func.keywords.keys(): if k in kwargs: kwargs.pop(k) return func(Ks=Ks, K=K, T=T, **kwargs) def visualize_smpl_vibe(orig_cam=None, pred_cam=None, bbox=None, output_path='sample.mp4', resolution=None, aspect_ratio=1.0, bbox_scale_factor=1.25, bbox_format='xyxy', **kwargs) -> None: """Simplest way to visualize pred smpl with origin frames and predicted cameras.""" assert resolution is not None if pred_cam is not None and bbox is not None: orig_cam = torch.Tensor( convert_crop_cam_to_orig_img(pred_cam, bbox, resolution[1], resolution[0], aspect_ratio, bbox_scale_factor, bbox_format)) assert orig_cam is not None, '`orig_cam` is required.' func = partial( render_smpl, projection='weakperspective', convention='opencv', in_ndc=True, ) for k in func.keywords.keys(): if k in kwargs: kwargs.pop(k) return func(orig_cam=orig_cam, output_path=output_path, resolution=resolution, **kwargs) def visualize_T_pose(num_frames, body_model_config=None, body_model=None, orbit_speed=1.0, **kwargs) -> None: """Simplest way to visualize a sequence of T pose.""" assert num_frames > 0, '`num_frames` is required.' assert body_model_config is not None or body_model is not None model_type = body_model_config[ 'type'] if body_model_config is not None else body_model.name( ).replace('-', '').lower() if model_type == 'smpl': poses = torch.zeros(num_frames, 72) else: poses = torch.zeros(num_frames, 165) func = partial(render_smpl, betas=None, transl=None, verts=None, convention='pytorch3d', projection='fovperspective', K=None, R=None, T=None, origin_frames=None) for k in func.keywords.keys(): if k in kwargs: kwargs.pop(k) return func(poses=poses, body_model_config=body_model_config, body_model=body_model, orbit_speed=orbit_speed, **kwargs) def visualize_smpl_pose(poses=None, verts=None, **kwargs) -> None: """Simplest way to visualize a sequence of smpl pose. Cameras will focus on the center of smpl mesh. `orbit speed` is recommended. """ assert (poses is not None) or (verts is not None), 'Pass either `poses` or `verts`.' func = partial(render_smpl, convention='opencv', projection='fovperspective', K=None, R=None, T=None, in_ndc=True, origin_frames=None, frame_list=None, image_array=None) for k in func.keywords.keys(): if k in kwargs: kwargs.pop(k) return func(poses=poses, verts=verts, **kwargs)