Creating a new conda environment or a virtual environment with Python 3.10+.
conda create -y -n PynamicMesh -c conda-forge python=3.11
conda activate PynamicMeshClone the repo and install dependencies:
git clone https://github.com/MMV-Lab/PynamicMesh
cd PynamicMesh
# Instalation with only CPU support
pip install .
# Instalation with GPU support depend in your drivers options
pip install ".[gpu-12x]"
pip install ".[gpu-13x]"
pip install ".[rocm-7-0]"
# Instalation with editable mood for developers
pip install -e .All the following examples can be replicated using the meshes and the generated example files provided here, and all the code usage syntax is summarized in the provided code.
The provided tools were developed independently into pure computations and visualization/graphical tools in order to keep the flexibility of running the computations on a pure non-graphic node or high-performance computing cluster.
Given a family of meshes
PynamicMesh offers a full general range of pipelines based on Topology, Differential Geometry, and Physics in order to model the complex dynamics encoded in the transformation, allowing the extraction of features that help to characterize and understand the dynamical process.
For a detailed and applied understanding of meshes as Manifolds and triangulations, the following Jupyter Notebook might interest you.
Global Geometry
This Basic analysis offer a time tracking of the global geometry features of the mesh.
Generatibg a csv with the metric values and the plot of each one.
from PynamicMesh.core.pipelines import run_pipeline
run_pipeline(**args)In order to track the Global Geometry, run:
from PynamicMesh.core.pipelines import run_pipeline
run_pipeline(
path_str='base/path'
compute_basicGeo = True,
metrics = 'all',
plot_basicGeo = True
)Or you can set your parameters on the yaml file, and within the PynamicMesh enviroment run on the comand line:
run_pynamic --config /path/to/the/config.yaml
Global Geometry Parameters
Path to the root folder that contains the scenes:
path_str (str) Flag to indicate the execution:
compute_basicGeo (bool)Flag to indicate the plot of the metrics:
plot_basicGeo (bool)Desired metrics to track:
metrics (str) | (list)Available metrics
Compute and report all the available metrics :
metrics (str) : 'all'Compute just the selected set of metrics :
metrics (list) : ['metric_1',...,'metric_n']Mesh Detail : n_vertices and n_faces captures the structural resolution of the mesh (the total number of points and connecting triangles), constant values mean the object is changing shape or moving without altering its basic blueprint.
Changing values mean the model is actively gaining or losing detail (e.g., tearing, merging, or adaptive rewriting).
Surface Area area captures the total amount of outside covering for the mesh, tracks stretching and compression. If the surface area spikes while the overall size remains constant, it indicates that the object is wrinkling, crumpling, or becoming highly textured.
Volume volume captures the total relative amount of physical space enclosed inside the object, tracks inflation and deflation. A steady volume means the object is maintaining its physical mass/size while it moves or bends.
Sphericity phericity captures the roundness score from 0 to 1, evaluating how closely the object resembles a perfect ball (1 being a perfect sphere). A rising score means the object is compacting or pulling itself together into a ball shape.
A falling score means it is stretching out, flattening, or growing irregular limbs.
Convexity convexity captures a "bulginess" score measuring how many hollows, indents, or valleys the object has. A high score indicates a smooth, rounded object. A decreasing score means the object is actively folding in on itself, developing deep cavities, or sprouting appendages.
Surface Curvature mean_gaussian_curvature captures the average texture profile of the surface, distinguishing between dome-like features and saddle-like curves.
Serves as a global bumpiness tracker. Rapid fluctuations indicate that the smooth skin of the object is rapidly turning wavy, creased, or jagged.
Relative Movement Speed center_mass captures the straight-line distance traveled by the object's center of gravity from one time frame to the next. Tracks overall speed. A flat line near zero means the object is stationary (even if it is spinning or shaking in place).
Sudden spikes indicate a sudden leap or fast global movement across space.
Functional Map
General Overview
The functional map
In order to understand the dynamics of the deformation we can compute this matrix and vector in every time step
At the end we can use this representations to have a lot of features, those are defined and explained on the correspondig Functional Map Implementation Usage and Analysis section.
Mathematical Construction Details
Given two consecutive time step meshes
The goal is to find a representation of the unknown bijective transformation function
We can use a scalar function defined over each mesh
In our case we are goind to use the Dirichlet energy on each point of the mesh for the wave propagation function, the heat diffusion function, or the sum of both:
The composition
As the linear function spaces
If we apply the Finite Element Method (FEM) to the Laplace-Beltrami equation of a function
Means that we want to compute the gradient of a function defined on a triangle, but locally the function varies linearly within each triangle
We can contruct the Connectivity matrix or the Cotan-Laplace operator
The "connectivity" is encoded in the adjacency of the mesh. The Laplacian matrix
$$L_{ij} = \left{ \begin{array}{cl} -w_{ij} & : v_i\to v_j \text{conected}\ 0 & : \text{ no conexion} \ \end{array} \right.$$
For the diagona the sum of weights of all edges connected to
$$ L_{ii} = \sum w_ii $$
This matrix
Then for every vertex
$$W_{ij} = \left{ \begin{array}{cl} A_i & : i=j\ 0 & \text{ other case} \ \end{array} \right.$$
This matrix essentially encode the surface area contribution of each vertex. Because a mesh is made of triangles, the "area" of a vertex is defined by the triangles that share it.
Then we can solve the generalized eigenvalue decomposition for a matrix
Only for the first
Once solved, each column
We can obtain this matrix for the
In theory this basis allows to express our fucntional as a linear combination:
$$\mathcal{F}{\varphi_n}(f) = \sum_k\sum_j a_jc{jk}\phi_k^{M_{t_{i}}}$$
This provide a matrix representation
We can express this coeficents using a inner product to project the tranformation represented on the domain base into the codomain base:
$$\displaystyle c_{jk}= \left\langle \mathcal{F}{\varphi_n}(\phi_k^{M{t_{i}}}) , \phi_j ^{M_{t_{i-1}}} \right\rangle$$
But now we have a matrix representation
We can use our Dirichlet energy descriptors in order of get a clue:
We can project the Dirichlet energy on mesh
$$ \Psi^{t_{i-1}} = \Phi^{T} W E[\Psi] \hspace{6mm} \Psi^{t_{i}} = \Phi^{T} W E[\Psi] $$
The functional map matrix $\mathcal{F}{\varphi_n} = C{t_{i-1} \to t_{i}} \in \mathbb{M}_{k \times k}(\mathbb{R})$ that we seek now is given for the one that minimizes the following objective function:
$$ \min_{C} E(C) = \underbrace{\sum_{m=1}^{n_f} | C \Psi^{t_{i-1}}m - \Psi^{t{i}}m |^2}{E_{desc}} + \underbrace{\lambda_{reg} | C \Lambda_1 - \Lambda_2 C |^2}{E{reg}} + \underbrace{\lambda_{land} \sum_{l=1}^L | C \Phi_1(x_l, :)^\top - \Phi_2(y_l, :)^\top |^2}{E{land}} $$
Were:
This optimal matrix
We need to take the basis representation of a point
And then transform it to the spectral domain of
Take the vertex
Functional Map Implementation Usage and Analysis
The computations are executed and managed through the syntax:
from PynamicMesh.core.pipelines import run_pipeline
run_pipeline(**args)In order to compute the Functional Map transformations, run:
from PynamicMesh.core.pipelines import run_pipeline
run_pipeline(
path_str='base/path',
matrix_tranformation=True,
diagonal_analysis=True,
isometric_analysis=True,
k_eigenfunctions=(10,10),
k_eigenvalues=100,
descriptor='WKS+HKS',
landmarks='precomputed',
compute_physic_fields=True,
)Or you can set your parameters on the yaml file, and within the PynamicMesh enviroment run on the comand line:
run_pynamic --config /path/to/the/config.yamlFunctional Map Parameters
Path to the root folder that contains the scenes:
path_str (str) Flag to indicate the model execution:
matrix_tranformation (bool)Flag to indicate the isometry analysis execution within the loop:
isometric_analysis (bool)Flag to indicate the diagonal analysis execution within the loop:
diagonal_analysis (bool)Flag to indicate the computation and storage of the physical fields (once); if it is false, the visualizer will compute them during execution time every time.
compute_physic_fields (bool)Energy function used on the pipeline: WKS (wave propagation kernel), HKS (heat diffusion kernel), or WKS+HKS (both).
descriptor (str): 'HKS' | 'WKS' | 'WKS+HKS'Number of eigenfunctions used on the base matrix
k_eigenfunctions (tuple) (int,int)Number of low frecuence eigenvalues
k_eigenvalues (int) Vertex indices indicators for symmetry restriction:
landmark (list|str):Landmark Options
No symmetry restrictions applied:
landmark (str) : NoneA priori known indices of the
landmark (list) : [1,2,3,4,..,n] -> (n,)A priori known indices of the symmetrical vertices (one per considered transformation). If the list contains fewer sets of vertices than the pairs of meshes, the remaining computations will perform without restrictions.
landmark (list) : [[1,..,n1],..,[1,..,nk]] -> (n,m)A priori known pair indices of the symmetrical vertices; here
landmark (list) : [[1,2],...,[j,k]] -> (2,n)A priori known pair indices of the symmetrical vertices (one set of relations considered per transformation). If the list contains fewer sets of vertex relations than the pairs of meshes, the remaining computations will perform without restrictions.
landmark (list) : [[[1,2],...,[j,k]],...,[[1,2],...,[l,m]]]] -> (m,2,n)Note:
The independent function precompute_landmarks(root_path,'FM') can be used along with the graphical tool to click over the vertex selection for every scene in the project. Alternatively, the function visual_selection_edition(path_to_meshes,'FM') is included to precompute or edit existing landmarks for a specific scene. Both functions will generate a landmark.npy file with the corresponding selected vertex relations, and this option will check for this precomputed .npy file during execution.
landmark (str) : 'precomputed'Landmarks Graphical Selection
For the precomputed landmarks:
from PynamicMesh.core.custom_fm import precompute_landmarks
precompute_landmarks("./PynamicMesh/Mesh_models",'FM')Vertices are chosen by clicking over them and unmarked by clicking again over the selected vertex. When the selection is ready, just close the window in order to pass to the next mesh and the pipeline is the same.
The vertex selection should be performed in the same order for each mesh so the right related pairs are formed. The selected vertices should roughly correspond to the same related regions of the meshes.
At the end, the .npy file with our vertex selection on each frame will be stored in the path ./PynamicMesh/Results/scene1/landmark.npy.
If we use the other function:
from PynamicMesh.core.custom_fm import visual_selection_edition
visual_selection_edition("./PynamicMesh/Mesh_models/scene1",'FM')The visualizer is going to show the specific mesh dynamics and the landmarks created previously, giving the chance to edit them or also create them if they do not exist.
In this case, in order to change among meshes in time, use the arrow keys on the keyboard. When the edition is finished, just close the window and it will be stored.
Physical Features and Map Transformation Models
We can run our matrix transformation computing; at the end of the run, the pipeline will save one matrix per transformation (FMC_ij.npy) and the corresponding vector of vertex transformations (FMV_ji.npy) within the folder ./PynamicMesh/Results/scene1/Transform_Matrices/.
Let's run first the pipeline without landmarks to see the results.
from PynamicMesh.core.pipelines import run_pipeline
run_pipeline(base_mesh_path, matrix_tranformation=True, descriptor='WKS+HKS', landmarks=None, k_eigenfunctions=(10,10),k_eigenvalues=100)Compute the descriptors and visualize them with:
from PynamicMesh.core.physic_model import visualize_physics
visualize_physics("./PynamicMesh/Mesh_models/", "./PynamicMesh/Results/scene1/Transform_Matrices/", on_time=False)The parameter on_time indicates if it needs to compute the physical field during execution or just look for the precomputed and stored results created during run_pipeline.
on_time(bool)The program will compute and deploy the following:
$\Delta$-Color transfer:
Showing with different colors which region of the mesh
$\Delta\vec{v}$ Vertex velocity displacement:
Showing the map over the mesh about the velocity ratios of change with respect to the vertex mapping (euclidean velocity of displacement) $||\vec{v}{t{i-1}}-\vec{v}{t{i}}||$; that means coloring the regions of faster change.
Linear (edge) strain:
Plot the result of the finite elements pipeline to compute the zones of stretch and compression with respect to the edges (1D), that is where the coefficient
Area (Face) strain:
Plot the result of the finite elements pipeline to compute the zones of stretch and compression with respect to the faces (2D), that is where the coefficient
Normal protrusion flow $\vec{v}|| \vec{N}$:
Describes vertex-wise normals of the mesh geometry. That is the "outward" or "inward" direction relative to the surface. (Areas where the mesh is actively pushing out or pulling in.)
Tangent flow $\vec{v} \hookrightarrow \vec{T}$:
Subtracts the normal component from the total displacement to isolate the lateral movement, representing the lateral or crawling flow across the surface. This captures how the mesh material slides or shifts across the surface without necessarily changing the local thickness or volume. (Surface flow, where the mesh might be rearranging its surface area laterally.)
Note:
The Camel is almost symmetric by the middle; due to this, the
The used landmarks in these examples are provided here.
from PynamicMesh.core.pipelines import run_pipeline
from PynamicMesh.core.physic_model import visualize_physics
run_pipeline(base_mesh_path, matrix_tranformation=True, descriptor='WKS+HKS', landmarks='precomputed', k_eigenfunctions=(10,10),k_eigenvalues=100)
visualize_physics("./PynamicMesh/Mesh_models/", "./PynamicMesh/Results/scene1/Transform_Matrices/", on_time=False)
Isometry Transformation Tracking
Beside the animations and computations, within the folder (./PynamicMesh/Results/scene1/Diagonal_analysis) the Heatmap of the matrix representation in each time step will be reported. This heatmap
In our camel example, changes to the topology are not too aggressive (mostly just the legs changing positions), so the matrices stay close to the diagonal. In order to track the diagonality of the heatmaps over time, we track three metrics and report the results in a CSV file:
Moment of Inertia Metric:
where
If
If
Note: This metric is highly sensitive to far-away outliers, meaning even a small amount of energy in the far corners will cause this metric to drop significantly.
Exponential Decay Metric:
If
If
Cumulative Distribution Function Bandwidth:
The function loops through every possible bandwidth radius
Exact percentage of total energy sitting directly on the core main diagonal line.
If the plotted curve shoots up vertically and hits 1.0 at a very low bandwidth (
If the curve scales up gradually as a slow diagonal line, it indicates that the spectral energy is leaking into wide off-diagonal frequencies.
Heatmaps Similarity Metrics
A comparison among
Jensen-Shannon Divergence:
Treats the squared matrix as an "energy distribution" and measures how much the allocation of energy changes between the two mappings. A spike in JSD indicates a sudden phase shift in the physical deformation.
For example, if a mesh was smoothly expanding over time, but suddenly starts twisting or turning, the energy distribution across the matrix will dramatically change, and JSD will spike.
Pearson & Spearman Correlation:
Measures how linearly aligned (Pearson) and structurally ranked (Spearman) the cells of
High correlation means the "nature" or "pattern" of the deformation is steady and consistent. If a mesh is undergoing a continuous, prolonged stretch in one direction over several frames, the FMs will look structurally identical. A drop in correlation means the mesh has started a new, different movement.
Manhattan $L_1$ and Euclidean $L_2$ Distances:
Measures the raw geometric difference between the specific coefficient values of the two matrices.
This acts as a measure of acceleration or intensity change. If the deformation is speeding up or becoming more drastic between frames, the coordinate distances will increase, even if the general shape of the matrix (the correlation) stays roughly the same.
Reeb Graph
General Overview
The Reeb Graph (
This is possible by assigning a vertex in the graph to each level curve, which generates a graph based on the local geometric structures.
In order to understand the dynamics of the deformation we can compute this graophs in every time step
At the end we can use this representations to have a lot of features, those are defined and explained on the correspondig Reeb Graph Usage and Analysis section
Mathematical Construction Details
Given a real scalar field over the mesh
Then the Reeb graph is the topological quotient space induced by the relation, endowed with the quotient topology
Even when the definition can be abstract, the idea is very intuitive. Think about the scalar field
The condition
Defining the equivalence relation means that we need to look at how many points of the surface
Saying that the graph has the quotient topology means that the graph captures the topology relationships of the mesh.
The idea is easy to follow graphically:
Reeb Graph Usage and Analysis
The computations are executed and managed through the syntax:
from PynamicMesh.core.pipelines import run_pipeline
run_pipeline(**args)In order to compute the Reeb Graph run:
from PynamicMesh.core.pipelines import run_pipeline
run_pipeline(
path_str='base/path',
compute_reeb=True,
time_graph_analysis=True,
reeb_scalar="geodesic",
bins=30,
**scalar_fields_args
)Or you can set your parameters on the yaml file, and within the PynamicMesh enviroment run on the comand line:
run_pynamic --config /path/to/the/config.yamlReeb Graph Parameters
Path to the root folder that contains the scenes:
path_str (str) Flag to indicate the model execution:
compute_reeb (bool)
Number of level sets used for the graph computing:
bins (int)Scalar field used to compute the graph:
reeb_scalar (str)Parameters related to the selection of Scalar Fields param_name = param_value:
**scalar_fields_args (dict)Flag to indicate if the analysis should be run within the loop; this will run the analysis over the raw computed graphs. If you need to run the analysis on the graphs after a modification, you can run it independently over the modified graphs.
time_graph_analysis (bool)Available Scalar Fields
Spatial Based
$(x,y,z)$-Level sets:
reeb_scalar="x" | reeb_scalar="y" | reeb_scalar="z"Distance from the center of mass:
reeb_scalar="dist_centroid"Signed distance relative to parallel planes crossing the centroid:
reeb_scalar="signed_dist_x" | reeb_scalar="signed_dist_y" | reeb_scalar="signed_dist_z"Absolute distance to specified axis:
reeb_scalar="dist_x_axis" | reeb_scalar="dist_y_axis" | reeb_scalar="dist_z_axis" Geometric/Topology based
Geometric mean curvature:
reeb_scalar="mean_curvature"Gaussian curvature:
reeb_scalar="gaussian_curvature"Shape base index:
reeb_scalar="shape_index"Curve base index:
reeb_scalar="curvedness"Protrusion mapping based on mesh $M_{t-1}$:
reeb_scalar="normal_displacement"Spectral mapping based on $n$ Laplace-Beltrami eigenfunctions:
reeb_scalar="lb_eigen_n" Geodesic path to the center of mass:
reeb_scalar="mass_center_geodesic"Multi scalar field maps combination through Mapper Lens construction (PCA feature extraction):
reeb_scalar="multi_pca", fields=["f1","f2",..,"fn"] Spectral mapping based on Heat diffusion, source point (vertex index $i$) $v_i$ and time $t$:
reeb_scalar="heat_diffusion", source_idx=i, t=t source_idx options
source_idx (str|list|int)
t(int) While the time source_idx can codify different options:
Apply the same heat source (
source_idx (int): nApply the heat source source_idx=0 will be applied:
source_idx (list): [0,1,2,3,...,n]Apply the heat source source_idx=0 will be applied:
source_idx (list): [0,None,None,3,...,n]Visual selection, in the same fashion as in the case of the functional map (see section Landmarks graphical selection for functional maps):
source_idx (str): "precomputed"Spectral mapping based on Harmonic with boundary conditions injection flow in vertex $v_i$ and leaving flow in vertex $v_t$:
reeb_scalar="harmonic", source_idx=i, sink_idx=t source_idx options
While in the case of source_idx=i, sink_idx=j we refer to the boundary conditions injection flow in vertex source_idx can codify different options:
source_idx (str|list|int)
sink_idx (int) Use the pair
source_idx (int)
sink_idx (int) Use every pair source_idx=i, sink_idx=j for each mesh. If there are fewer pairs than meshes, for the rest the default value source_idx=min(index), sink_idx=max(index) will be applied.
source_idx (list): [[1,2],[3,4],...,[i,j]]Use every pair source_idx=i, sink_idx=j for each mesh. For the None positions, the default value source_idx=min(index), sink_idx=max(index) will be applied.
source_idx (list): [[1,2],None,...,[i,j]]Visual selection, in the same fashion as in the case of the functional map (see section Landmarks graphical selection for functional maps).
source_idx (str): "precomputed"Geodesic mapping based on vertex landmarks $[v_0,...,v_n] \in M_t$:
reeb_scalar="geodesic", vertex_ref_index=[0,1,2,n] vertex_ref_index options
The parameter vertex_ref_index can codify different options:
vertex_ref_index (str|list)Apply the set of reference vertices
vertex_ref_index (list): [0,1,2,3,...,n] Apply each set of reference vertices vertex_ref_index=[0] will be applied.
vertex_ref_index (list): [[0,...,n1],[0,...,n2],...,[0,...,nk]] Apply each set of reference vertices vertex_ref_index=[0] will be applied.
vertex_ref_index (list): [[0,...,n1],None,...,[0,...,nk]]Visual selection, in the same fashion as in the case of the functional map (see section Landmarks graphical selection for functional maps).
vertex_ref_index (str): "precomputed"Visual reference notes
In each case of the Heat diffusion, Harmonic, and Geodesic based scalar maps, the optional reference point can be selected graphically with the execution of the corresponding code:
from PynamicMesh.core.custom_fm import visual_selection_edition, precompute_landmarks
################################ Sources Vertex index precompute visual tools for 'heat_diffusion' in Reebs #############################################################
print('Visualizing or editing Sources Vertex index for RG...')
visual_selection_edition(mesh_path,'heat_diffusion')
print('Precomputing Sources Vertex index for RG...')
precompute_landmarks(base_mesh_path,'heat_diffusion')
################################ Source-sink Vertex index precompute visual tools for 'harmonic' in Reebs #############################################################
print('Visualizing or editing Source-sink Vertex index for RG...')
visual_selection_edition(mesh_path,'harmonic')
print('Precomputing Source-sink Vertex index for RG...')
precompute_landmarks(base_mesh_path,'harmonic')
################################# Vertex index reference precompute visual tools for 'geodesic' in Reebs #############################################################
print('Visualizing or editing Vertex index for RG...')
visual_selection_edition(mesh_path,'geodesic')
print('Precomputing Vertex index for RG...')
precompute_landmarks(base_mesh_path,'geodesic') Each one will generate and save the respective sources.npy, source_sink.npy, vert_ref_geo.npy files within the folder ./PynamicMesh/Results/scene1.
Note:
If during the pipeline execution the corresponding 'precomputed' function is used but no .npy files are found for a certain folder, the default values will be used.
Reeb Graph Visualization
After the modeling pipeline execution, the files Reeb_Ti.pkl and Scalar_Ti.npy (one for each time ./PynamicMesh/Results/scene1/Reeb_Graphs.
With these files, we can visualize the evolution of the field and the graph over time:
from PynamicMesh.core.pipelines import run_pipeline
from PynamicMesh.core.reeb_graph import graph_time_analysis, visualize_reeb_graphs
print('Executing modeling ...')
run_pipeline(base_mesh_path, compute_reeb=True, bins=30 , reeb_scalar='geodesic', vertex_ref_index=[4896])
print('Reeb visualizations...')
visualize_reeb_graphs(mesh_path, reeb_path)
Reeb Graph Edition Tool
When the graphs are created, we can use the graphical tool to edit the created graph:
- Click over an existing vertex (on the graph side) to delete the vertex and all the connected edges to it.
- Click on the surface mesh vertex (on the mesh side) to create a vertex, and click on an existing graph vertex (on the graph side) to create the edge among them.
- Press key 'i' to activate INNER mode; when you click on a vertex, it moves orthogonally into the mesh (press again to deactivate).
- Press key 'o' to activate OUTER mode; when you click on a vertex, it moves orthogonally closer to the mesh surface (press again to deactivate mode).
- Press key 'c' to activate LINK mode; when you click on two existing vertices on the graph side, the edge among them is created (press again to deactivate mode).
- Space Bar to Undo the last change.
- Use 's' and 'w' keys to activate/deactivate the visible layer on the mesh.
- Use the arrow keys to change the graph in time.
When the edition is ready, just close the window. Only the corresponding modified graphs will be saved.
Note:
The original computed graphs are not overridden. The modified graphs will be stored within the folder ./PynamicMesh/Results/scene1/Reeb_graph_manual_edit.
Time Graph analysis and plotting
When the desired graphs are ready and saved, we can run a temporal analysis and generate the plot of the results and the CSV with the data:
from PynamicMesh.core.reeb_graph import graph_time_analysis, plot_dynamic_graph_analysis
print('Graph path analysis...')
graph_time_analysis(reeb_path)
print('Plotting dynamic graph analysis...')
plot_dynamic_graph_analysis(csv_file_path)Structural Complexity (Nodes & Edges)
Encodes the raw size of the Reeb graph skeleton. This tracks how "complex" or "branchy" the shape is.
A spike in nodes and edges indicates the mesh is growing new appendages, fragmenting, or wrinkling in time.
A drop indicates the mesh is smoothing out, shrinking, or parts are merging together in time.
The "Intensity" of Deformation (The Distance Metrics)
We calculate three distance metrics (Wasserstein, Spectral Laplacian, and Graph Edit Distance) between consecutive time steps. Together, these act as an "earthquake seismograph" for the meshes.
Encodes how drastically the skeleton shifted from
Smooth, low values mean that the mesh is experiencing stable, continuous deformation (e.g., simply moving or slowly expanding).
Sudden spikes indicate a critical topological event in the system. The mesh just underwent a sudden structural change, such as breaking apart (fission), colliding/merging (fusion), or suddenly collapsing.
The Graph Edit Distance highlights direct physical breakages/additions of branches, while Spectral Distance highlights global warping of the overall shape.
Holes, Loops, and Fusions (Betti-1 Cycles)
The Betti-1 number counts the number of 1D loops or cycles in the graph. It detects when the shape folds back on itself to create a hole or a tunnel (like a donut).
An increase in cycles means appendages have touched and fused together, creating a closed loop.
Stretching and Elongation (LCC Diameter)
The "Diameter" of the Largest Connected Component represents the longest shortest-path across the graph's skeleton. It measures the maximum spatial span of the object. If the diameter steadily increases while the number of nodes stays the same, it means your mesh is being stretched or elongated (like pulling a piece of taffy).
This analysis allows us to automatically pinpoint exactly when and how your 3D meshes undergo major structural changes without having to manually watch the 3D animation. It converts visual shape evolution into a dashboard of growth (size), drastic events (distances), stretching (diameter), and fusions (cycles).
Batch Analysis Notes
The described methods on the previous sections allow us to configure specific sets of parameters for computing each mesh within a timelapse, or alternatively, to apply a consistent set of parameters across different timelapses. Naturally, it may be necessary to use unique parameter sets for distinct timelapses for instance, when applying different scalar fields for Reeb graphs for example or when processing timelapses of varying natures or sample types.
To better organize and manage these multi-parameter configurations, it is possible to define a unique set of parameters for each timelapse within the yaml configuration file and execute the following instruction within the conda environment.
run_pynamic --config /path/to/the/config_batch.yaml --batchOr execute programatically as
from PynamicMesh.utils.batch import run_batch
from PynamicMesh.utils.tools import extract_yaml
config_path = '\PynamicMesh\examples\config_batch.yaml'
config = extract_yaml(config_path)
data_cfg = config.get("Data", {})
path_str = data_cfg.get("path_str")
run_batch(config,path_str)







