File size: 7,978 Bytes
37cca23 |
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 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 |
import os
import time
import shutil
from pathlib import Path
from typing import Union
import atexit
import spaces
from concurrent.futures import ThreadPoolExecutor
import open3d as o3d
import trimesh
import gradio as gr
from gradio_imageslider import ImageSlider
import cv2
import numpy as np
import click
import imageio
from promptda.promptda import PromptDA
from promptda.utils.io_wrapper import load_image, load_depth
from promptda.utils.depth_utils import visualize_depth, unproject_depth
model = PromptDA.from_pretrained('depth-anything/promptda_vitl').to("cuda").eval()
thread_pool_executor = ThreadPoolExecutor(max_workers=1)
def delete_later(path: Union[str, os.PathLike], delay: int = 300):
print(f"Deleting file: {path}")
def _delete():
try:
if os.path.isfile(path):
os.remove(path)
print(f"Deleted file: {path}")
elif os.path.isdir(path):
shutil.rmtree(path)
print(f"Deleted directory: {path}")
except:
pass
def _wait_and_delete():
time.sleep(delay)
_delete(path)
thread_pool_executor.submit(_wait_and_delete)
atexit.register(_delete)
@spaces.GPU
def run_with_gpu(image, prompt_depth):
depth = model.predict(image, prompt_depth)
depth = depth[0, 0].detach().cpu().numpy()
return depth
def check_is_stray_scanner_app_capture(input_dir):
assert os.path.exists(os.path.join(input_dir, 'rgb.mp4')), 'rgb.mp4 not found'
pass
def run(input_file, resolution):
import ipdb; ipdb.set_trace()
# unzip zip file
input_file = input_file.name
root_dir = os.path.dirname(input_file)
scene_name = input_file.split('/')[-1].split('.')[0]
input_dir = os.path.join(root_dir, scene_name)
cmd = f'unzip -o {input_file} -d {root_dir}'
os.system(cmd)
check_is_stray_scanner_app_capture(input_dir)
# extract rgb images
os.makedirs(os.path.join(input_dir, 'rgb'), exist_ok=True)
cmd = f'ffmpeg -i {input_dir}/rgb.mp4 -start_number 0 -frames:v 10 -q:v 2 {input_dir}/rgb/%06d.jpg'
os.system(cmd)
# Loading & Inference
image_path = os.path.join(input_dir, 'rgb', '000000.jpg')
image = load_image(image_path)
prompt_depth_path = os.path.join(input_dir, 'depth/000000.png')
prompt_depth = load_depth(prompt_depth_path)
depth = run_with_gpu(image, prompt_depth)
color = (image[0].permute(1,2,0).cpu().numpy() * 255.).astype(np.uint8)
# Visualization file
vis_depth, depth_min, depth_max = visualize_depth(depth, ret_minmax=True)
vis_prompt_depth = visualize_depth(prompt_depth[0, 0].detach().cpu().numpy(), depth_min=depth_min, depth_max=depth_max)
vis_prompt_depth = cv2.resize(vis_prompt_depth, (vis_depth.shape[1], vis_depth.shape[0]), interpolation=cv2.INTER_NEAREST)
# PLY File
ixt_path = os.path.join(input_dir, f'camera_matrix.csv')
ixt = np.loadtxt(ixt_path, delimiter=',')
orig_max = 1920
now_max = max(color.shape[1], color.shape[0])
scale = orig_max / now_max
ixt[:2] = ixt[:2] / scale
pcd = unproject_depth(depth, ixt=ixt, color=color, ret_pcd=True)
ply_path = os.path.join(input_dir, f'pointcloud.ply')
o3d.io.write_point_cloud(ply_path, pcd)
glb_path = os.path.join(input_dir, f'pointcloud.glb')
scene_3d = trimesh.Scene()
glb_colors = np.asarray(pcd.colors).astype(np.float32)
glb_colors = np.concatenate([glb_colors, np.ones_like(glb_colors[:, :1])], axis=1)
# glb_colors = (np.asarray(pcd.colors) * 255).astype(np.uint8)
pcd_data = trimesh.PointCloud(
vertices=np.asarray(pcd.points) * np.array([[1, -1, -1]]),
colors=glb_colors.astype(np.float64),
)
scene_3d.add_geometry(pcd_data)
scene_3d.export(file_obj=glb_path)
# o3d.io.write_point_cloud(glb_path, pcd)
# Depth Map Original Value
depth_path = os.path.join(input_dir, f'depth.png')
output_depth = (depth * 1000).astype(np.uint16)
imageio.imwrite(depth_path, output_depth)
delete_later(Path(input_dir))
delete_later(Path(input_file))
return color, (vis_depth, vis_prompt_depth), Path(glb_path), Path(ply_path).as_posix(), Path(depth_path).as_posix()
DESCRIPTION = """
# Estimate accurate and high-resolution depth maps from your iPhone capture.
## Requirements:
1. iPhone 12 Pro or later Pro models, iPad 2020 Pro or later Pro models
2. Free iOS App: [Stray Scanner App](https://apps.apple.com/us/app/stray-scanner/id1557051662)
## Testing Steps:
1. Capture a scene with the Stray Scanner App.
2. Use the iPhone [Files App](https://apps.apple.com/us/app/files/id1232058109) to compress it into a zip file and transfer it to your computer. (Long press the capture folder to compress)
3. Upload the zip file and click "Submit" to get the depth map of the first frame.
Note:
- Currently, this demo only supports inference for the first frame. If you need to obtain all depth frames, please refer to our [GitHub repo](https://github.com/DepthAnything/PromptDA).
- The depth map is stored as uint16, with a unit of millimeters.
"""
@click.command()
@click.option('--share', is_flag=True, help='Whether to run the app in shared mode.')
def main(share: bool):
with gr.Blocks(theme=gr.themes.Soft()) as demo:
gr.Markdown(DESCRIPTION)
with gr.Row():
input_file = gr.File(type="filepath", label="Upload a stray scanner app capture zip file")
resolution = gr.Dropdown(choices=['756x1008', '1428x1904'], value='756x1008', label="Inference resolution")
submit_btn = gr.Button("Submit")
gr.Examples(examples=[
["data/assets/example0_chair.zip", "756x1008"]
],
inputs=[input_file, resolution],
# outputs=[output_rgb, output_depths, output_3d_model, output_ply, output_depth_map],
label="Examples",
)
with gr.Row():
with gr.Column():
output_rgb = gr.Image(type="numpy", label="RGB Image")
with gr.Column():
output_depths = ImageSlider(label="Depth map / prompt depth", position=0.5)
with gr.Row():
with gr.Column():
output_3d_model = gr.Model3D(label="3D Viewer", display_mode='solid', clear_color=[1.0, 1.0, 1.0, 1.0])
with gr.Column():
output_ply = gr.File(type="filepath", label="Download the unprojected point cloud as .ply file")
output_depth_map = gr.File(type="filepath", label="Download the depth map as .png file")
outputs = [
output_rgb,
output_depths,
output_3d_model,
output_ply,
output_depth_map,
]
submit_btn.click(run,
inputs=[input_file, resolution],
outputs=outputs)
demo.launch(share=share, debug=True)
# def main(share: bool):
# gr.Interface(
# fn=run,
# inputs=[
# gr.File(type="filepath", label="Upload a stray scanner app capture zip file"),
# gr.Dropdown(choices=['756x1008', '1428x1904'], value='756x1008', label="Inference resolution")
# ],
# outputs=[
# gr.Image(type="numpy", label="RGB Image"),
# ImageSlider(label="Depth map / prompt depth", position=0.5),
# gr.Model3D(label="3D Viewer", display_mode='solid', clear_color=[1.0, 1.0, 1.0, 1.0]),
# gr.File(type="filepath", label="Download the unprojected point cloud as .ply file"),
# gr.File(type="filepath", label="Download the depth map as .png file"),
# ],
# title=None,
# description=DESCRIPTION,
# clear_btn=None,
# allow_flagging="never",
# theme=gr.themes.Soft(),
# examples=[
# ["data/assets/8b98276b0a.zip"]
# ]
# ).launch(share=True)
if __name__ == '__main__':
main() |