-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathaugmented_stacking.py
210 lines (159 loc) · 9.13 KB
/
augmented_stacking.py
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
210
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
"""
Created on Thu Feb 26 15:20:00 2022
CLI main file for Augmented Stacking:
The main file is used to run the augmented stacking algorithm to guide an operator in
correctly placing stones following a stacking algorithm and an AR system composed by a
projector and 3D camera (ZEDi2)
"""
import os
import sys
import subprocess
from threading import Thread
# # The package opencv is not found in virtual env, solve
# # the issue without importing the sys.path here. With this it works
sys.path.append('/usr/local/lib/python3.8/dist-packages')
import cv2
import dataset_IO
import stacking_algorithm
import distance_map
from util import terminal
import camera_capture
import matplotlib.pyplot as plt
import numpy as np
import open3d as o3d
import colorama as cr
# Path where the landscape is overwritten
_name_landscape = './landscape.ply'
# Set a target of mesh faces for the low-res mesh
_faces_target_stone_mesh = 500
# Set a target of max vertices for the captured scene mesh
_vertices_target_low_res_scene = 2000
# Set dir to store validated stone and landscape of each captured
DIR_AS_BUILT_ASSEMBLY = './as_built_status/'
# Stone poses recorder path
stone_pose_record_path = './temp/placedStones.txt'
def main():
# -----------------------------------------------------------------------
# [0] Decorator
# -----------------------------------------------------------------------
terminal.cat('./logo/logoascii')
terminal.cat('./logo/credits.txt')
print('Before to start make sure that:')
print('-> You have done a calibration')
print('-> You have a second screen with the same resolution as your main machine')
# -----------------------------------------------------------------------
# [1] Create as-built dir + start 3D visualizer
# -----------------------------------------------------------------------
# Create sub-dir for current session
dir_as_built = dataset_IO.create_record_session_subdir(DIR_AS_BUILT_ASSEMBLY)
# Set stone stacked counter for recording
STONE_COUNTER = 0
# Launch non-blocking visualizer
subprocess.Popen(['python', './util/non_block_visualizer.py'])
# ADD-ON: clean out preavious stone center record
record_stone_file = './temp/placedStones.txt'
dataset_IO.delete_file(record_stone_file)
terminal.custom_print("[INFO]: stone record pose delted.")
while(True): # For each stone - loop
while(True): # Guide the pose - loop
while(True): # Find the pose - loop
# -----------------------------------------------------------------------
# [2] Download the stone mesh from dataset
# -----------------------------------------------------------------------
# Ask for stone label & download mesh and open it
name_stone_mesh, stone_mesh = dataset_IO.download_github_raw_file()
# Check the n faces for the download mesh if not downsample
faces_stone_mesh = len(np.asarray(stone_mesh.triangles))
if (faces_stone_mesh > _faces_target_stone_mesh):
terminal.custom_print(f"The mesh needs to be decimate for the stacking algorithms.\n"
f" Number of vertices for downloaded mesh: {faces_stone_mesh}")
stone_mesh = stone_mesh.simplify_quadric_decimation(
target_number_of_triangles=_faces_target_stone_mesh)
new_faces_stone_mesh = len(np.asarray(stone_mesh.triangles))
terminal.custom_print(f" Number of vertices of the decimated mesh: {new_faces_stone_mesh}")
# Check if the mesh is out of scale or not watertight, and confirm
try:
if stone_mesh.get_volume() > 1.0: # 1m3
stone_mesh = stone_mesh.scale(scale=1/1000,
center=stone_mesh.get_center())
break
except:
terminal.error_print("ERROR: the imported mesh is corrupted or not water tight.\n"
"Choose another stone from dataset ...")
continue
# Write out the mesh (for algorithm to read)
o3d.io.write_triangle_mesh(name_stone_mesh, stone_mesh)
# -----------------------------------------------------------------------
# [3] Capture the scene mesh + compute the mesh 6dof pose
# -----------------------------------------------------------------------
# Capture meshed scene
landscape_mesh = camera_capture.get_mesh_scene(
_vertices_target_low_res_scene)
# # Save meshed scene
terminal.custom_print("Writing out the captured mesh from 3d camera")
o3d.io.write_triangle_mesh(
_name_landscape, landscape_mesh, write_ascii=True)
terminal.custom_print(f"Mesh out with name {_name_landscape}")
# Compute stacking pose
stacking_algorithm.compute(path_exec='./stacking_algorithm/build/main',
path_mesh=name_stone_mesh,
path_landscape=_name_landscape, # TODO: param this
config_file='./stacking_algorithm/data/input.txt',
name_output='pose',
dir_output='./temp')
# Delete donwloaded mesh
dataset_IO.delete_file(name_stone_mesh)
# read the 6dof pose and store it as numpy array
pose_matrix = dataset_IO.read_pose_6dof('./temp/pose.txt')
print("Computation done: here's the computed 4x4 Pose Matrix:\n\n", pose_matrix)
# Transform the low-res mesh for visualization
stone_mesh.transform(pose_matrix)
# Temporary store landscape and stone for visualization
terminal.custom_print("Temporary storing the landscape and stone mesh before validation ...")
STONE_COUNTER, path_stone_mesh, path_landscape_mesh = dataset_IO.save_current_built_layer(landscape_mesh=landscape_mesh,
stone_mesh=stone_mesh,
stone_label=name_stone_mesh,
sub_dir=dir_as_built,
stone_counter=STONE_COUNTER)
# Merge transformed stone and landscape mesh
merged_landscape = stone_mesh + landscape_mesh
break
# -----------------------------------------------------------------------
# [4] Augmented feedback loop
# -----------------------------------------------------------------------
# First open the camera and close at the end
zed, point_cloud = camera_capture.set_up_zed()
# Create the 3D space model
Live_3D_space = camera_capture.Live_3D_space(rock_mesh=stone_mesh,zed=zed,point_cloud=point_cloud)
# Create image object
image_drawer = camera_capture.Image_drawer(Live_3D_space=Live_3D_space)
# Start live 2D projection to guide user to stone placement
terminal.custom_print(f"When the stone is placed correnctly, Press <Escape> to close the tkinter window.")
live_stream = camera_capture.Live_stream(Live_3D_space=Live_3D_space,image_drawer=image_drawer)
live_stream.run()
# Now that the loop is closed, close the camera
zed.close()
# Ask to validate the stone and store current layer results
i_is_pose_stone_valid = None
while not i_is_pose_stone_valid in ['Y','y','n','N']:
i_is_pose_stone_valid = terminal.user_input('Do you validate the stone position? (y/n)\n>>> ')
if i_is_pose_stone_valid not in ['Y', 'y']:
# If the stones are not validated erase them from recording folder
terminal.custom_print("Erasing the stone and scene from recording folder")
os.remove(path_stone_mesh)
os.remove(path_landscape_mesh)
# Erase stone's pose record
dataset_IO.erase_last_pose_stone(stone_pose_record_path)
terminal.custom_print("[INFO]: The current stone pose is erased from the algorithm pose recorder")
# Ask to place another stone or terminate CLI
i_continue_stacking = None
while not i_continue_stacking in ['Y','y','n','N']:
i_continue_stacking = terminal.user_input('Do you want to stack another stone? (y/n)\n>>> ')
if i_continue_stacking in ['N', 'n']:
terminal.custom_print("The augmented stacking is shuting down ...")
exit()
terminal.custom_print("Validating the stone and scene in recording folder ...")
if __name__ == "__main__":
main()