Skip to content

Commit 072f48d

Browse files
committed
added downscale when in trouble, adjusted tests to output visuals and corrected catcells
1 parent e01d500 commit 072f48d

7 files changed

+197
-131
lines changed

irregular_object_packing/mesh/sampling.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -100,7 +100,7 @@ def resample_mesh_by_triangle_area(example_mesh: pv.PolyData, target_mesh: pv.Po
100100
target_avg_area = compute_average_triangle_area(target_mesh)
101101

102102
# Calculate the desired number of triangles in the target mesh
103-
target_num_triangles = int(target_mesh.n_faces * (example_avg_area / target_avg_area))
103+
target_num_triangles = int(target_mesh.n_faces * (target_avg_area / example_avg_area))
104104

105105
# Use the decimation algorithm to reduce the number of triangles in the target mesh
106106
resampled_mesh = resample_pyvista_mesh(target_mesh, target_faces=target_num_triangles)

irregular_object_packing/packing/chordal_axis_transform.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -235,8 +235,8 @@ def create_faces_3(data: CatData, occ, tet_points: list[TetPoint]):
235235
bc_point = data.point_id((most[1].vertex + least[0].vertex) / 2)
236236
bd_point = data.point_id((most[1].vertex + least[1].vertex) / 2)
237237

238-
most_face_c = [acd_point, ac_point, bcd_point, bc_point]
239-
most_face_d = [acd_point, ad_point, bcd_point, bd_point]
238+
most_face_c = [acd_point, bcd_point, bc_point, ac_point]
239+
most_face_d = [acd_point, bcd_point, bd_point, ad_point]
240240
n_mfc = compute_face_unit_normal(
241241
data.get_face(most_face_c), most[0].vertex
242242
) # can be both most[0] and most[1]

irregular_object_packing/packing/growth_based_optimisation.py

+29-9
Original file line numberDiff line numberDiff line change
@@ -161,10 +161,10 @@ def setup(self):
161161
self.log(
162162
f"Skipped {skipped} points to avoid overlap with container", LOG_LVL_DEBUG
163163
)
164+
self.log(f"Setup with settings: \n{self.settings}", LOG_LVL_INFO)
164165
self.log(f"Number of objects: {self.n_objs}", LOG_LVL_INFO)
165166

166167
object_rotations = np.random.uniform(-np.pi, np.pi, (self.n_objs, 3))
167-
self.padding = 0.01
168168

169169
# SET TRANSFORM DATA
170170
self.tf_arrays = np.empty((self.n_objs, 7))
@@ -250,6 +250,9 @@ def resample_meshes(self, scale_factor):
250250
self.curr_sample_rate = self.sample_rate_mesh(scale_factor)
251251
self.shape = resample_pyvista_mesh(self.shape0, self.curr_sample_rate)
252252
self.container = resample_mesh_by_triangle_area(self.shape, self.container0)
253+
assert self.shape.is_manifold
254+
assert self.container.is_manifold
255+
253256
self.log(f"container: n_faces: {self.container.n_faces}[sampled]/{self.container0.n_faces}[original]", LOG_LVL_INFO)
254257
self.log(f"mesh: n_faces: {self.curr_sample_rate}[sampled]/{self.shape0.n_faces}[original]", LOG_LVL_INFO)
255258
# self.padding = avg_mesh_area**(0.5) / 4 # This doenst help
@@ -353,8 +356,17 @@ def process_iteration(self, ib, i):
353356
cat_viols, con_viols, collisions = compute_all_collisions(p_meshes, cat_meshes, self.container, set_contacts=False)
354357
self.log_violations((cat_viols, con_viols, collisions))
355358

359+
violating_ids = set([i for i, v in cat_viols] + [i for i, v in con_viols])
360+
361+
self.log("reducing scale for violating objects: " + str(violating_ids), LOG_LVL_INFO)
362+
for id in violating_ids:
363+
self.reduce_scale(id)
364+
356365
self.update_data(ib, i, (cat_viols, con_viols, collisions))
357366

367+
def reduce_scale(self, id):
368+
self.tf_arrays[id][0] *= 0.95
369+
358370
def check_closed_cells(self):
359371
cat_cells = [
360372
PolyData(*cat.face_coord_to_points_and_faces(self.cat_data, obj_id))
@@ -379,7 +391,7 @@ def log_violations(self, violations):
379391
def default_setup() -> "Optimizer":
380392
DATA_FOLDER = "./../../data/mesh/"
381393

382-
mesh_volume = 0.3
394+
mesh_volume = 0.2
383395
container_volume = 10
384396

385397
loaded_mesh = pv.read(DATA_FOLDER + "RBC_normal.stl")
@@ -401,7 +413,7 @@ def default_setup() -> "Optimizer":
401413
# padding=1e-3,
402414
# sample_rate=1000,
403415
dynamic_simplification=True,
404-
alpha=0.25,
416+
alpha=0.2,
405417
beta=0.5,
406418
)
407419
plotter = None
@@ -459,23 +471,25 @@ def simple_shapes_setup() -> "Optimizer":
459471

460472
# %%
461473

474+
reload(plots)
475+
462476

463477
def plot_step(optimizer, step):
464478
plotter = pv.Plotter()
465479
meshes, cat_meshes, container = optimizer.recreate_scene(step)
466-
plots.plot_simulation_scene(plotter, meshes, cat_meshes, container)
480+
plots.plot_simulation_scene(plotter, meshes, cat_meshes, container, c_kwargs={"show_edges": True, "edge_color": "purple"})
467481
plotter.add_text(optimizer.status(step).table_str, position="upper_left")
468482

469483
plotter.show()
470484
return plotter
471485

472486

473-
plot_step(optimizer, 6)
487+
plot_step(optimizer, 33)
474488

475489

476490
# %%
477-
obj_i, step = 10, 5
478-
meshes, cat_meshes, container = optimizer.recreate_scene(step)
491+
obj_i, step = 6, 31
492+
meshes_before, meshes_after, cat_meshes, container = optimizer.recreate_scene(step)
479493
# plots.plot_step_comparison(
480494
# optimizer.mesh_before,
481495
# optimizer.mesh_after(step, obj_i),
@@ -484,8 +498,13 @@ def plot_step(optimizer, step):
484498
# )
485499
# %%
486500
reload(plots)
487-
488-
plotter = plots.plot_step_single(meshes[obj_i], cat_meshes[obj_i], cat_opacity=0.7, mesh_opacity=1 , clipped=True, title="cat overlap")
501+
obj_i = 6
502+
plotter = plots.plot_step_single(
503+
meshes_before[obj_i], cat_meshes[obj_i], container=container, cat_opacity=0.7, mesh_opacity=1 , clipped=True, title="cat overlap",
504+
c_kwargs={"show_edges": True, "edge_color": "purple", "show_vertices": True, "point_size": 10},
505+
m_kwargs={"show_edges": True, "show_vertices": True, "point_size": 10, },
506+
cat_kwargs={"show_edges": True, "show_vertices": True, "point_size": 5, },
507+
)
489508

490509
# %%
491510
# store cat mesh in file
@@ -534,4 +553,5 @@ def profile_optimizer():
534553
print(f"{ai} {bi}`")
535554
ax.plot(mesh_simplification_condition(x, ai, bi), label=f"a:{ai:.2f}, b:{bi:.2f}")
536555
ax.legend()
556+
537557
# %%

irregular_object_packing/packing/nlc_optimisation.py

+18-15
Original file line numberDiff line numberDiff line change
@@ -149,28 +149,27 @@ def local_constraint_for_point_with_padding(
149149
face_coords = [np.array(points[p_id]) - obj_coord for p_id in face_p_ids]
150150

151151
# NOTE: Any point on the surface of the object can be used as q_j
152-
q_j = face_coords[0] # q_j = np.mean(face_coords, axis=0) -> not necessary
152+
q_j = face_coords[0] # q_j = np.mean(face_coords, axis=0) -> not necessary
153153

154154
# NOTE: The normal vector has unit length [./utils.py:196], no need to divide
155155
condition = np.dot(transformed_v_i - q_j, n_face)
156156

157-
158157
# Return negative value if point is inside surface plus margin,
159158
dist = condition - padding
160159
constraints.append(dist)
161160

162-
# NOTE: alternative implementation: (currently allows point inside margin only)
163-
# MARGIN: Compute distance to surface
164-
# dist = abs(condition) - padding
165-
# if dist < 0:
166-
# dist = 0
167-
# # positive value otherwise
168-
# if condition < 0:
169-
# constraints.append(-dist)
170-
# else:
171-
# constraints.append(dist)
172-
# # constraints.append(condition)
173161
return constraints
162+
# NOTE: alternative implementation: (currently allows point inside margin only)
163+
# MARGIN: Compute distance to surface
164+
# dist = abs(condition) - padding
165+
# if dist < 0:
166+
# dist = 0
167+
# # positive value otherwise
168+
# if condition < 0:
169+
# constraints.append(-dist)
170+
# else:
171+
# constraints.append(dist)
172+
# # constraints.append(condition)
174173

175174

176175
def local_constraint_for_point(
@@ -270,6 +269,8 @@ def local_constraints_from_cat(
270269

271270
# [(vi, [facet_j, facet_j+1, ...]), (vi+1, [facet_k, facet_k+1, ...)]
272271
items = cat_data.cat_faces[obj_id].items()
272+
cat_cell = cat_data.cat_cells[obj_id]
273+
_all_faces = [cat_cell] * len(items)
273274

274275
v, sets_of_faces = [*zip(*items, strict=True)]
275276
return local_constraint_multiple_points(
@@ -284,6 +285,8 @@ def local_constraints_from_cat(
284285
# -----------------------------------------------------------------------------
285286
# Visual Tests
286287
# -----------------------------------------------------------------------------
288+
289+
287290
def test_nlcp_facets():
288291
# Define points
289292
points = {
@@ -341,7 +344,7 @@ def get_face_coords(facet, points):
341344
objective, x0, method="SLSQP", bounds=bounds, constraints=constraint_dict
342345
)
343346
T = construct_transform_matrix(res.x)
344-
## %%
347+
# %%
345348
# Print the results
346349
print("Optimal solution:")
347350
print_transform_array(res.x)
@@ -351,7 +354,7 @@ def get_face_coords(facet, points):
351354
print(transform_v(points[10], T))
352355
print(transform_v(points[11], T))
353356

354-
## %%
357+
# %%
355358
# Create a 3D plot
356359
import matplotlib.pyplot as plt
357360
from mpl_toolkits.mplot3d.art3d import Poly3DCollection

irregular_object_packing/packing/optimizer_data.py

+10-5
Original file line numberDiff line numberDiff line change
@@ -188,11 +188,13 @@ def reconstruct_delaunay(self, iteration: int):
188188

189189
def recreate_scene(self, iteration: int):
190190
"""Recreate the scene at the given iteration."""
191-
meshes = self.meshes_after(iteration)
191+
meshes_before = self.meshes_before(iteration)
192+
meshes_after = self.meshes_after(iteration)
192193
cat_meshes = self.cat_meshes(iteration)
193-
compute_all_collisions(meshes, cat_meshes, self.container0, set_contacts=True)
194+
compute_all_collisions(meshes_before, cat_meshes, self.container0, set_contacts=True)
195+
compute_all_collisions(meshes_after, cat_meshes, self.container0, set_contacts=True)
194196

195-
return meshes, cat_meshes, self.container0
197+
return meshes_before, meshes_after, cat_meshes, self.container0
196198

197199
def final_meshes_after(self):
198200
"""Get the meshes of all objects with the most recent transformation."""
@@ -238,8 +240,11 @@ def _report(self, iteration=None):
238240
)
239241

240242
def print_report(self, notebook=False):
241-
tabulate(
243+
table = tabulate(
242244
[self.status(i) for i in range(self.idx)], # type: ignore
243245
headers=[field.name for field in fields(IterationData)],
244-
tablefmt="html" if notebook else "grid"
246+
tablefmt="html" if notebook else "grid",
247+
showindex=True,
245248
)
249+
print(table)
250+
return table

irregular_object_packing/packing/plots.py

+17-11
Original file line numberDiff line numberDiff line change
@@ -121,11 +121,13 @@ def plot_step_comparison(
121121
def plot_step_single(
122122
mesh,
123123
cat_cell_mesh_1,
124+
container=None,
124125
mesh_opacity=0.9,
125126
cat_opacity=0.4,
126127
plotter=None,
127128
clipped=False,
128129
title="Title",
130+
c_kwargs={}, m_kwargs={}, cat_kwargs={},
129131
):
130132
if plotter is None:
131133
plotter = pv.Plotter()
@@ -139,16 +141,19 @@ def plot_step_single(
139141
)
140142
if clipped:
141143
try:
142-
plotter.add_mesh_clip_plane(mesh, opacity=mesh_opacity, color="red", cmap='bwr', scalars="collisions", point_size=10, show_vertices=True, show_edges=True)
144+
plotter.add_mesh_clip_plane(mesh, opacity=mesh_opacity, color="red", cmap='bwr', scalars="collisions", **m_kwargs)
143145
except KeyError:
144146
plotter.add_mesh(mesh, opacity=mesh_opacity, color="blue")
145-
plotter.add_mesh_clip_plane(cat_cell_mesh_1, opacity=cat_opacity, color="yellow", show_edges=True, show_vertices=True, point_size=5)
147+
plotter.add_mesh_clip_plane(cat_cell_mesh_1, opacity=cat_opacity, color="yellow", **cat_kwargs)
146148
else:
147149
try:
148-
plotter.add_mesh(mesh, opacity=mesh_opacity, color="red", cmap='bwr', scalars="collisions", point_size=10, show_vertices=True, show_edges=True)
150+
plotter.add_mesh(mesh, opacity=mesh_opacity, color="red", cmap='bwr', scalars="collisions", **m_kwargs)
149151
except KeyError:
150-
plotter.add_mesh(mesh, opacity=mesh_opacity, color="blue")
151-
plotter.add_mesh(cat_cell_mesh_1, opacity=cat_opacity, color="yellow", show_edges=True, show_vertices=True, point_size=5)
152+
plotter.add_mesh(mesh, opacity=mesh_opacity, color="blue", **m_kwargs)
153+
plotter.add_mesh(cat_cell_mesh_1, opacity=cat_opacity, color="yellow", **cat_kwargs)
154+
155+
if container is not None:
156+
plotter.add_mesh(container, opacity=0.3, **c_kwargs)
152157
# plotter.add_mesh(open_edges, color="black", line_width=1, opacity=0.8)
153158
plotter.show()
154159

@@ -225,15 +230,16 @@ def create_packed_scene(
225230
return plotter
226231

227232

228-
def plot_simulation_scene(plotter, meshes, cat_meshes, container):
233+
def plot_simulation_scene(plotter, meshes, cat_meshes, container, c_kwargs={}, m_kwargs={}, cat_kwargs={},
234+
):
229235
for i in range(len(meshes)):
230236
try:
231-
plotter.add_mesh(meshes[i], opacity=0.95, color="red", cmap='bwr', scalars="collisions")
232-
plotter.add_mesh(cat_meshes[i], opacity=0.6, color="yellow")
237+
plotter.add_mesh(meshes[i], opacity=0.95, color="red", cmap='bwr', scalars="collisions", **m_kwargs)
238+
plotter.add_mesh(cat_meshes[i], opacity=0.6, color="yellow", **cat_kwargs)
233239
except KeyError:
234-
plotter.add_mesh(meshes[i], opacity=0.8, color="blue")
235-
plotter.add_mesh(cat_meshes[i], opacity=0.4, color="greenyellow")
236-
plotter.add_mesh(container, color="white", opacity=0.3)
240+
plotter.add_mesh(meshes[i], opacity=0.8, color="blue", **m_kwargs)
241+
plotter.add_mesh(cat_meshes[i], opacity=0.4, color="greenyellow", **cat_kwargs)
242+
plotter.add_mesh(container, color="white", opacity=0.3, **c_kwargs)
237243

238244

239245
def generate_gif(optimizer, save_path, title="Optimization"):

0 commit comments

Comments
 (0)