# Geodesic Calculus on Implicitly Defined Latent Manifolds

This repository provides code to compute geodesic paths and exponential maps on latent manifolds.

Given a latent manifold represented by a point cloud of latent samples, the method enables interpolation paths that remain on the manifold and respect its geometry. The latent manifold is modeled implicitly via a learned representation function.
The approach is independent of the specific autoencoder model and supports different choices of Riemannian metrics on the latent space.

For theoretical background and methodological details see our paper.
# Installation
Main dependencies (see [pyproject.toml](pyproject.toml))
- torch >=2.9.1
- scipy
- numpy
- scikit-learn
- matplotlib (examples only)
## Manual installation
You can manually install all dependencies using your preferred package manager. 
## Using uv
1. install uv
  ```bash 
  curl -LsSf https://astral.sh/uv/install.sh | sh
  ```

2. install dependies from .toml file
  ```bash 
  uv sync
  ```
  to install optional dependencies use 
  ```bash 
    uv sync --extra training
  ```
3. activate virtual environment
  ```bash 
  source .venv/bin/activate
  ```  
4. install local project
  ```bash 
 uv pip install -e .
  ```  
5. (Optional) register a Jupyter kernel to run notebooks in VS Code
  ```bash 
  uv run python -m ipykernel install --user --name latentgeodesics --display-name "latentgeodesics"
  ```    


If an error occurs during installation it is probably due to torch.
Install a suitable torch cuda version for your machine using `uv pip install` with correct versions, and adjust the version in .toml file if necessary.

# Usage

We provide different toy examples that give an overview over the main functionalities in [examples](examples).
- [example_gt_surface](examples/GT_example/example_gt_surface.ipynb) shows how to compute geodesics on ground truth implicit manifolds (e.g. a torus) where the implicit function is known.
- [VAE_example](examples/VAE_example/VAE_example.ipynb) gives an example of a latent manifold learned with a variational autoencoder and illustrates the use of different metrics on the latent manifold (euclidean, pullback metric, KL divergence)
- [Point_Cloud_example](examples/Point_Cloud_example/point_cloud_example.ipynb) provides an example of computing geodesics and exponential maps on a generated data distribution (noisy halfcircle data).
## Training a representation function
If you want to apply the framework to your latent manifold, the first step is to train a representation function.

  `python scripts/train_latent_representation.py --config examples/your_example/your_args.json` 

We provide a [template args file](scripts/args_template.json). You have to adapt the path to your latent samples and the latent dimension. The sampling parameter std refers to the standard deviation for the denosing functional used for learning the projection.
The larger it is the stronger is the denoising property of the learned representation.
It depends on the sampling density and the noise of your latent manifold. 
A good rule of thumb is that a ball of radius std should contain at least 10 points. For more details on this parameter see our paper. If your latent manifold has a large extent (>5) set normalization to true.

## Computing geodesics
Once you have trained the representation function, you can load the network and initialize the solver

```python
latentprojector = setup_latent_network(args_path).to(device)
latentprojector.load_state_dict(network_weights)

#representation function
phi = latentprojector.phi()

Solver = GeodesicSolverTorch(latent_dim, phi)
```
and compute a geodesic with resolution $K$ between start point $xA$ and $xB$.

```python
geodesic = Solver.AugLagrangeMinimize(K, xA, xB)
```

To compute the exponential map, you have to specify a start point $xA$ and an initial direction $v$
```python
dphi = latentprojector.dphi()
exp = KExponentialTorch(phi, dphi, xA, v, K)
```
# Attribution

Please cite our paper when using this code for research purposes. 