Metadata-Version: 2.4
Name: volVIC
Version: 1.0.1
Summary: VOLume-based Virtual Image Correlation (volVIC) library for Python
Author-email: Dorian Bichet <dbichet@insa-toulouse.fr>
License-Expression: CECILL-2.1
Project-URL: Homepage, https://github.com/Dorian210/volVIC
Project-URL: Repository, https://github.com/Dorian210/volVIC
Project-URL: Issues, https://github.com/Dorian210/volVIC/issues
Classifier: Programming Language :: Python :: 3
Classifier: Programming Language :: Python :: 3 :: Only
Classifier: Operating System :: OS Independent
Classifier: Topic :: Scientific/Engineering
Classifier: Intended Audience :: Science/Research
Requires-Python: >=3.9
Description-Content-Type: text/markdown
License-File: LICENSE.txt
Requires-Dist: numpy>=1.20
Requires-Dist: numba>=0.55
Requires-Dist: scipy>=1.8
Requires-Dist: pyvista>=0.41
Requires-Dist: meshio>=5.0
Requires-Dist: matplotlib>=3.5
Requires-Dist: tqdm>=4.60
Requires-Dist: interpylate>=1.0.2
Requires-Dist: treeIDW>=1.0.1
Requires-Dist: bsplyne>=1.0.6
Requires-Dist: IGA_for_bsplyne>=1.0.3
Provides-Extra: dev
Requires-Dist: build; extra == "dev"
Requires-Dist: twine; extra == "dev"
Requires-Dist: pytest; extra == "dev"
Provides-Extra: docs
Requires-Dist: wkhtmltopdf; extra == "docs"
Requires-Dist: pdoc>=12.0; extra == "docs"
Dynamic: license-file

# volVIC — VOLume-based Virtual Image Correlation

<p align="center">
  <img src="https://raw.githubusercontent.com/Dorian210/volVIC/master/docs/logo.png" width="500" />
</p>

**volVIC** is a Python library for surface shape measurement from 3D volumetric images (e.g. CT scans) using the **Virtual Image Correlation (VIC)** method. It fits a parametric B-spline surface mesh to a grayscale volume by minimizing an energy functional that compares a synthetic "virtual image" — built from the surface geometry — to the actual acquired image.

The method is particularly suited to problems in experimental mechanics, where one wants to track the shape or deformation of a material surface from tomographic data.

---

## Table of Contents

- [Method overview](#method-overview)
- [Installation](#installation)
- [Quick start](#quick-start)
- [Module overview](#module-overview)
- [License](#license)

---

## Method overview

The VIC approach replaces classical pixel-to-pixel image correlation with a physics-motivated model. Given a B-spline surface mesh and a volumetric image:

1. A **virtual image** is constructed analytically from the surface geometry. It models the gray-level transition (background ↔ foreground) along the surface normal, parameterized by a scalar distance `rho`.
2. The **VIC energy** measures the discrepancy between the virtual image and the actual acquired image, integrated over a normal neighborhood of the surface.
3. A **Gauss–Newton solver** iteratively minimizes the total energy (VIC + membrane regularization) with respect to the B-spline control point displacements and `rho`.
4. **C¹ continuity constraints** across patches and **Dirichlet boundary conditions** are enforced through a constraint matrix.

---

## Installation

### 1. High-Performance Solvers (Optional but Recommended)

**volVIC** relies on [IGA_for_bsplyne](https://github.com/Dorian210/IGA_for_bsplyne) for its linear system solvers. This library can leverage `SuiteSparse` (via `scikit-sparse` and `sparseqr`) for significant speedups on large-scale problems.

* **Recommended (Conda)**:
  ```bash
  conda install -c conda-forge scikit-sparse
  pip install sparseqr
  ```
  *Note: `sparseqr` is not on Conda, but it will successfully link to the SuiteSparse libraries installed by `scikit-sparse`.*

* **Alternative (system package manager)**:
  * **macOS**: `brew install suitesparse && pip install scikit-sparse sparseqr`
  * **Ubuntu/Debian**: `sudo apt-get install libsuitesparse-dev && pip install scikit-sparse sparseqr`

* **Fallback**: If SuiteSparse is not available, the library automatically falls back to standard `scipy.sparse` solvers, ensuring full compatibility at the cost of performance.

### 2. Install volVIC

```bash
pip install volVIC
```

**Requirements:** Python ≥ 3.9

**Key dependencies:** `numpy`, `scipy`, `numba`, `bsplyne`, `IGA_for_bsplyne`, `pyvista`, `meshio`, `interpylate`, `treeIDW`, `matplotlib`, `tqdm`

---

## Quick start

```python
import numpy as np
from volVIC import Mesh, Problem

# Load your B-spline surface mesh and volumetric image
mesh = Mesh.Mesh.load("my_mesh.pkl")
image = np.load("my_ct_scan.npy")   # dtype: uint16, shape: (Z, Y, X)

# Set up and solve the VIC problem
problem = Problem(
    mesh=mesh,
    image=image,
    ICP_init=True,         # Rigid-body alignment before solving
    C1_mode="auto",        # Automatic C¹ continuity between patches
    membrane_weight=None,  # Automatic regularization weight estimation
)
u_field, rho = problem.solve(max_iter=20)

# Visualize results
problem.plot_results(u_field)

# Export to ParaView
problem.save_paraview(u_field, folder="results/", name="vic_output")
```

---

## Module overview

### `volVIC.Problem` — High-level solver

The main entry point. The `Problem` class encapsulates the full VIC pipeline.

**Constructor** `Problem(mesh, image, ...)`:
- Estimates foreground/background gray levels from the image histogram (Otsu or interpolation-based).
- Optionally performs a rigid-body ICP alignment of the mesh to the image isosurface.
- Builds the virtual image energy elements, membrane stiffness matrix, and C¹ constraint matrix.
- Calibrates the membrane regularization weight automatically.

**Key methods:**

| Method | Description |
|---|---|
| `solve(max_iter, ...)` | Run the Gauss–Newton optimization loop. Returns `(u_field, rho)`. |
| `plot_results(u_field, ...)` | Visualize signed distance fields with PyVista. |
| `save_paraview(u_field, folder, name)` | Export displacement and distance fields to VTK for ParaView. |
| `propagate_displacement_to_volume_mesh(u_field, volume_mesh)` | Propagate the surface displacement to a volumetric B-spline mesh. |

---

### `volVIC.Mesh` — B-spline mesh management

Provides the `Mesh` class wrapping multi-patch B-spline surfaces.

**Key features:**
- Load/save mesh from `.pkl` (pickle) files (`Mesh.load`, `Mesh.save`).
- Evaluate the mesh geometry and fields at arbitrary parametric points.
- Convert fields between per-patch and unique-node representations.
- Extract border patches, mesh subsets, and orientation fields.
- Visualize with PyVista (`plot`, `plot_in_image`, `plot_orientation`).
- Export time series to ParaView with `save_paraview`.
- Register mesh to a target triangle surface with `ICP_rigid_body_transform`.
- Propagate scalar or vector fields from a submesh to a parent mesh.

---

### `volVIC.virtual_image_correlation_energy` — VIC energy

The `VirtualImageCorrelationEnergyElem` class handles the energy computation for a single B-spline patch. It:
- Builds integration grids in both parametric and normal directions.
- Constructs the sparse small rotation displacement operator mapping control points to integration points in the image.
- Evaluates the VIC energy, gradient, and Gauss-Newton Hessian w.r.t. displacements and `rho`.

The module also provides utilities:
- `make_image_energies(mesh, h, ...)` — initialize energy elements for a full mesh.
- `compute_image_energy_operators(...)` — assemble global gradient and Hessian.
- `compute_distance_field(...)` — compute the signed distance field on each patch.

---

### `volVIC.virtual_image` — Virtual image model

Defines the virtual image function used in the VIC energy:

```python
from volVIC.virtual_image import g_slide

g, dg_drho = g_slide(xi, eta, gamma, rho, bg=0.0, fg=1.0)
```

`g_slide` models a smooth transition from background to foreground gray level along the surface normal (`gamma`), parameterized by the half-width `rho`. It returns both the image value `g` and its derivative w.r.t. `rho`.

---

### `volVIC.solve` — Gauss–Newton iteration

Low-level function performing one VIC iteration:

```python
from volVIC.solve import iteration

du_field, drho = iteration(u_field, rho, mesh, image_energies, image,
                           membrane_K, membrane_weight, C)
```

Assembles and solves the linearized system `H_tot @ Δu = -grad_tot`, with constraint enforcement via the matrix `C`. The `rho` increment is computed independently by a Newton step on the scalar energy.

---

### `volVIC.membrane_stiffness` — Regularization

Assembles the membrane (in-plane) stiffness matrix for B-spline surface patches, used as a regularization term in the VIC energy:

```python
from volVIC.membrane_stiffness import make_membrane_stifness, make_membrane_weight

K = make_membrane_stifness(mesh)
w = make_membrane_weight(mesh, image_energies, K, rho=1.5, image_std=5000)
```

The `make_membrane_weight` function calibrates the regularization weight by matching the expected membrane energy to the expected VIC energy under a probabilistic displacement model.

---

### `volVIC.C1_triplets` — Smoothness constraints

Generates C¹ continuity constraints between neighboring patches:

```python
from volVIC.C1_triplets import make_C1_eqs

C = make_C1_eqs(mesh, C1_inds="auto", threshold=0.1)
```

Each constraint takes the form `A - 2B + C = 0` for a triplet of control nodes `(A, B, C)`. The `"auto"` mode selects only geometrically consistent triplets.

---

### `volVIC.image_utils` — Image preprocessing

Utilities for gray-level analysis of CT images:

```python
from volVIC.image_utils import find_fg_bg, find_sigma_hat

fg, bg = find_fg_bg(image, method="otsu")   # or "interp"
sigma  = find_sigma_hat(image, fg, bg)
```

- `find_fg_bg` estimates foreground and background gray levels from the image histogram.
- `find_sigma_hat` estimates the noise standard deviation from single-phase voxels.
- Lower-level functions `hist`, `otsu_threshold`, and `interp_fg_bg` are also exposed.

---

### `volVIC.marching_cubes` — Isosurface extraction

Extracts a triangle isosurface from a 3D scalar field:

```python
from volVIC.marching_cubes import marching_cubes

surface = marching_cubes(volume, threshold=0.5)  # returns meshio.Mesh
```

Uses a Numba-compiled implementation for performance, with vertex deduplication and zero-area triangle prevention.

---

### `volVIC.integration_space_image` — Parametric integration

Computes integration points and weights in the parametric space of a B-spline surface such that mapped distances between points are approximately uniform:

```python
from volVIC.integration_space_image import linspace_for_VIC_elem

(xi, eta), (dxi, deta) = linspace_for_VIC_elem(spline, ctrl_pts, dist=1.0)
```

---

## Documentation

The full API documentation is available on the [Online Portal](https://dorian210.github.io/volVIC/).

---

## License

This project is licensed under the **CeCILL-2.1** license — see [https://cecill.info](https://cecill.info) for details.

---

*Author: Dorian Bichet — [dbichet@insa-toulouse.fr](mailto:dbichet@insa-toulouse.fr)*  
*Repository: [https://github.com/Dorian210/volVIC](https://github.com/Dorian210/volVIC)*
