This project centers on discrete geometry processing algorithms,drawing on:
- Digital Geometry Processing USTC course
- Games102 course
- Polygon Mesh Processing book
the mesh can be viewed as a graph, where the vertices of the mesh correspond to the nodes in the graph and the mesh topology corresponds to the connections of the graph.
- input: mesh, id of 2 vertices
- output: the shortest path on the mesh connecting the input vertices
- Mesh topology -> Graph connections: The mesh's topology defines the neighboring relations between its vertices. This is represented as edges connecting the nodes in the graph.
- Mesh vertices -> Graph nodes: Each vertex in the mesh corresponds to a node in the graph, with the edge weight typically representing the geometric distance or some other metric between vertices.
- Shortest path on the mesh -> Shortest path on a graph without negative weights: In this case, the edges of the graph have no negative weights, making it suitable to use the classic Dijkstra's algorithm to find the shortest path.
- d(v): The distance from the starting point
sp
to vertexv
. - inS(v): Boolean indicating whether vertex
v
is included in the setS
(the set of processed vertices). - N: The total number of vertices in the grid
M
. - Length(u, v): The length of the edge connecting vertices
u
andv
. - H: A heap data structure used to efficiently find the next vertex to process.
- pre(v): Stores the previous vertex in the shortest path leading to vertex
v
.
//initialize
for i in [1, N] do
d(i) ← +∞, inS(i) ← false
end
d(sp) ← 0, pre(sp) ← sp
H.push((sp, d(sp))
//main loop
while H ≠ ø do
tmp ← H.top()
if inS(tmp) = true then
continue
end
if tmp = ep then
break
else
inS(tmp) ← true
for each uv ∈ 1-ring of tmp do
if d(tmp) + Length(tmp, v) < d(v) then
d(vv) ← d(tmp) + Length(tmp, v)
H.push((uv, d(vv)))
pre(uv) ← tmp
end
end
end
end
//search path backward
while ep ≠ pre(ep) do
Epath.push((ep, pre(ep)))
Vpath.push(ep)
ep ← pre(ep)
end
Vpath.push(ep)
- Computing the shortest paths between all pairs of vertices, resulting in a complete graph of the input vertices over the mesh.
- Finding the MST of this complete graph, which serves as an approximate solution to Steiner's Tree.
Input:
M
: Input mesh.Lmk
: Input set of landmarks.
Output:
MST
: Approximate Steiner's Tree.
Data:
complete_graph
: A heap data structure.spanning_tree_current
: A disjoint-set (Union-Find) structure.
-
Initialize and Compute the Complete Graph:
- For each landmark
i
:- Calculate the modified Dijkstra algorithm to find the shortest paths from
Lmk[i]
to all other landmarks. - Store the paths in the
complete_graph
heap.
- Calculate the modified Dijkstra algorithm to find the shortest paths from
Lsize ← Lmk.size() for i ∈ [1, Lsize] do count ← Lsize - i # Modified Dijkstra Algorithm # while count ≠ 0 do if inS(tmp) = true then continue end if tmp = Lmk[u] and u > i then count ← count - 1 end inS(tmp) ← true end for j ∈ [i + 1, Lsize] do calculate Vpath complete_graph.push(Vpath) end end
- For each landmark
-
Compute the Minimum Spanning Tree:
- Initialize the spanning tree.
- While there are still vertices to process, extract the top element of the
complete_graph
and add the edge to the MST if it does not form a cycle.
index ← 0 for i ∈ [1, Lsize] do spanning_tree_current(i) ← i end while index < Lsize do E ← complete_graph.top() m, n ← the two endpoints of E m_id, n_id ← m, n in Lmk if m_id < n_id then swap(m_id, n_id) end if m_id ≠ n_id then MST.push(E) index ← index + 1 for i ∈ [1, Lsize] do if spanning_tree_current(i) = n_id then spanning_tree_current(i) ← m_id end end end end
- vertexLocalAveRegion: computes the average region for each vertex based on the areas of the adjacent faces. If an obtuse triangle exists, the area is divided accordingly to its vertices.
- mean curvature: calculates the mean curvature for each vertex using the Laplace-Beltrami operator. The cotangent of angles between edges in the vertex's 1-ring neighborhood is used in the calculation.
- Gaussian curvature: The angular deficit, which is the difference between 2π and the sum of angles at a vertex, is divided by the average area associated with the vertex to determine the Gaussian curvature.
- M: The input 3D mesh with faces and vertices.
- σ: Parameter controlling the influence of spatial proximity (distance between neighboring faces).
- ω: Parameter controlling the influence of normal similarity (difference between normals of neighboring faces).
- normal_iter: Number of iterations to perform for normal smoothing.
- vertex_iter: Number of iterations to perform for updating vertex positions.
- normal field
- updated vertex
-
Initialization of Normals:
- The normals of the mesh faces are initialized based on the original face normals.
-
Bilateral Filtering on raw Normals:
- For each face in the mesh, its normal is updated using the normals of its neighboring faces, weighted by both the spatial distance between face centers and the normal differences.
-
Vertex Position Update based on filterd normals:
- After filtering the normals, the positions of the vertices are adjusted based on the smoothed normals of the neighboring faces, iteratively updating the vertex positions(Gauss-Seidel iteration per vertex per iteration , other vertex fixed).
-
Iterative Process:
- The algorithm iteratively refines the normals for a specified number of iterations (
normal_iter
). - Update vertex positions one by one, Fixed the positions of the remaining vertices. The reason for this is to make the loss function a quadratic function of one variable $$ x_{\text{new}, i}= x_i + \sum_s{j \in \text{neighborhood}(x_i)} n_j \cdot( n_j^T (c_j - x_i)) $$
- The algorithm iteratively refines the normals for a specified number of iterations (
Given a triangulated surface homeomorphic to a disk, if the(u,v)coordinates at the boundary vertices lie on a convex polygon inorder, and if the coordinates of the internal vertices are a convex combination of their neighbors, then the (u, v) coordinates form a valid parameterization (without self-intersections, bijective)
- boundary: Artificially put the boundary point UV On a convex polygon
- interior: Uniform laplacian, Linear system
- Read into a grid, identify the boundaries, place them in order on a polygon (bisected N bisected circles),
- The inner points are convex combinations in it's one-ring, N points with N equations
- Init
-
1.1 Init parameterization: Tutte's Embedding
-
1.2 localCooridnate: for each face, compute local cooridnate:
#pragma omp parallel for for (int i = 0; i < n_faces; ++i){ auto face = mesh->polyface(i); auto normal = face->normal(); auto fv_iter = mesh->fv_iter(face); auto v0 = (*fv_iter)->position(); ++fv_iter; auto v1 = (*fv_iter)->position(); ++fv_iter; auto v2 = (*fv_iter)->position(); MVector3 e1 = v1 - v0; MVector3 e2 = v2 - v0; MVector3 x_ = e1.normalized(); MVector3 y_ = cross(x_, normal); localCoord.row(i) << 0, 0, e1.norm(), 0, dot(e2, x_), dot(e2, y_); /* face_0: (edge0_x, edge0_y, edge1_x, edge1_y, ... ) face_1: (edge0_x, edge0_y, edge1_x, edge1_y, ... ) ... face_n */ }
- Optimatize L_t step:
- 2.1 Fixing uv also fixes J(uv), finding the optimal approximation in the function space(rotation transformation space)
- Optimatize J(uv) step:
- 3.1 Fixing targeted L_t computed from 2, find the optimal approximation UV to appoximate such rigid transformation.
- 3.2 if get UV, LocalCoordinate: we deduce the map 's J(uv) (map: localcooridnate to uv)