-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathpointcloud.py
186 lines (156 loc) · 6.62 KB
/
pointcloud.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
import moderngl
import numpy as np
from moderngl_window.opengl.vao import VAO
from PIL import Image
class PointCloud:
EMPTY = 2**32 - 1
def __init__(
self, points: np.ndarray, colors: np.ndarray = None, point_size: float = 1.0
) -> None:
# to provide a uniform camera experience
self._points = normalize(points)
self._point_size = point_size
if colors is None:
colors = np.random.rand(points.shape[0], 3).astype(np.float32)
self._colors = colors
self._vao = None
@property
def point_size(self) -> float:
return self._point_size
@property
def vao(self) -> VAO:
if self._vao is None:
self._vao = self._create_pc()
return self._vao
def get_va_from(
self, ctx: moderngl.Context, program: moderngl.Program
) -> moderngl.VertexArray:
"""Create a vertex array from the points and colors."""
vbo_points = ctx.buffer(self._points.astype("f4").tobytes())
vbo_colors = ctx.buffer(self._colors.astype("f4").tobytes())
va = ctx.vertex_array(
program, [(vbo_points, "3f", "in_position"), (vbo_colors, "3f", "in_color")]
)
return va
def _create_pc(self) -> VAO:
_vao = VAO(mode=moderngl.POINTS)
vbo = self._points.astype("f4").tobytes()
_vao.buffer(vbo, "3f", "in_position")
vbo = self._colors.astype("f4").tobytes()
_vao.buffer(vbo, "3f", "in_color")
return _vao
def set_color(self, ids: np.array, colors: np.array) -> None:
"""Set the color of all points with ids in the ids set."""
self._colors[ids] = colors
self._vao = None
def set_pcd(self, points: np.ndarray, colors: np.ndarray):
self._points = normalize(points)
self._colors = colors
self._vao = None
def filter(self, u_ids: np.array) -> None:
"""Discard all points except those with ids in the ids array."""
self._points = self._points[u_ids]
self._colors = self._colors[u_ids]
self._vao = None
class SDPointCloud:
"""A point cloud wrapper that provides additional retexturing functionality for StableScan."""
def __init__(self, pcd: PointCloud, debug=False) -> None:
self.original_points = pcd._points.copy()
self.original_colors = pcd._colors.copy()
self.pcd = pcd
self.debug = debug
self.retextured_points = set()
@property
def retextured_point_ids(self) -> np.ndarray:
return np.array(list(self.retextured_points))
def retexture(self, texture: Image, ids: np.ndarray) -> None:
"""Given a texture and a 2D array of ids, retexture the point cloud."""
assert texture.width == ids.shape[1]
assert texture.height == ids.shape[0]
retexture_ids = []
retexture_colors = []
for y in range(texture.height):
for x in range(texture.width):
id = ids[y, x]
if id == PointCloud.EMPTY:
continue
if id in self.retextured_points:
continue
color = texture.getpixel((x, y))
color = np.array(color, dtype=np.float32) / 255.0
retexture_ids.append(id)
retexture_colors.append(color)
retexture_ids = np.array(retexture_ids)
retexture_colors = np.array(retexture_colors)
self.pcd.set_color(retexture_ids, retexture_colors)
self.retextured_points.update(retexture_ids)
if self.debug:
print(f"Retextured {len(retexture_ids)} points.")
def flag(self, ids: np.ndarray) -> None:
"""Flag all points with ids in the ids set."""
self.pcd.set_color(ids, np.array([1.0, 0.0, 0.0], dtype=np.float32))
def filter(self, ids: np.ndarray) -> None:
"""Discard all points except those with ids in the ids array."""
self.pcd.filter(ids)
def save(self, filename: str) -> None:
"""Save the change on the point cloud to two .npy files."""
ids = np.array(list(self.retextured_points))
colors = self.pcd._colors[ids]
np.save(filename + "_ids.npy", ids)
np.save(filename + "_colors.npy", colors)
def load(self, filename: str) -> None:
"""Load two .npy files and retexture the point cloud."""
self.reset()
ids = np.load(filename + "_ids.npy")
colors = np.load(filename + "_colors.npy")
self.pcd.set_color(ids, colors)
self.retextured_points.update(ids)
def reset(self) -> None:
"""Reset the point cloud to its original state."""
self.pcd.set_pcd(self.original_points.copy(), self.original_colors.copy())
self.retextured_points.clear()
def mask_retextured(self, ids: np.ndarray) -> np.ndarray:
"""Given a 2D ids array, mask seen ids with 1, and unseen ids with 0."""
mask = np.zeros_like(ids, dtype=np.uint8)
debug = np.zeros((*ids.shape, 3), dtype=np.uint8)
mask_count = 0
for x in range(ids.shape[1]):
for y in range(ids.shape[0]):
id = ids[y, x]
if id == PointCloud.EMPTY:
debug[y, x] = [0, 0, 0]
mask[y, x] = 1
continue
if id in self.retextured_points:
debug[y, x] = [0, 255, 0]
mask[y, x] = 0
mask_count += 1
else:
debug[y, x] = [255, 0, 0]
mask[y, x] = 1
if self.debug:
print(f"Keeping the color of {mask_count} points.")
Image.fromarray(debug, mode="RGB").show()
mask_ratio = mask_count / (ids.shape[0] * ids.shape[1])
# TODO(memben): breaks if mask is completely white
if mask_ratio < 0.02:
print(f"Warning: Mask ratio is {mask_ratio}.")
print(
"To ensure consistent retexturing, use an area with more retexured points."
)
print("Proceeding without a mask.")
mask = None
return mask
# TODO(memben): Slighly shifts the point cloud one pixel to the bottom and right.
def normalize(points: np.ndarray) -> np.ndarray:
min_coords = np.min(points, axis=0)
max_coords = np.max(points, axis=0)
ranges = max_coords - min_coords
scaling_factor = 2.0 / np.max(ranges)
normalized_points = (points - min_coords) * scaling_factor - 1.0
return normalized_points
def flatten_and_filter(ids: np.ndarray) -> np.ndarray:
"""Given a 2D array of ids, flatten it and remove all empty ids."""
ids = ids.flatten()
ids = ids[ids != PointCloud.EMPTY]
return ids