Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Modular coreg.py revision #71

Merged
merged 17 commits into from
Apr 8, 2021
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
17 commits
Select commit Hold shift + click to select a range
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
174 changes: 145 additions & 29 deletions docs/source/coregistration.rst
Original file line number Diff line number Diff line change
Expand Up @@ -2,17 +2,23 @@

DEM Coregistration
==================

.. contents:: Contents
:local:

Introduction
^^^^^^^^^^^^

Coregistration of a DEM is performed when it needs to be compared to a reference, but the DEM does not align with the reference perfectly.
There are many reasons for why this might be, for example: poor georeferencing, unknown coordinate system transforms or vertical datums, and instrument- or processing-induced distortion.

A main principle of all coregistration approaches is the assumption that all or parts of the portrayed terrain are unchanged between the reference and the DEM to be aligned.
This *stable ground* can be extracted by masking out features that are assumed to be unstable.
Then, the DEM to be aligned is translated, rotated and/or bent to fit the stable surfaces of the reference DEM as well as possible.
In mountainous environments, unstable areas could be: glaciers, landslides, vegetation, dead-ice terrain and human settlements.
In mountainous environments, unstable areas could be: glaciers, landslides, vegetation, dead-ice terrain and human structures.
Unless the entire terrain is assumed to be stable, a mask layer is required.
``xdem`` supports either shapefiles or rasters to specify the masked (unstable) areas.

There are multiple methods for coregistration, and each have their own strengths and weaknesses.
There are multiple approaches for coregistration, and each have their own strengths and weaknesses.
Below is a summary of how each method works, and when it should (and should not) be used.

**Example data**
Expand All @@ -28,18 +34,42 @@ Examples are given using data close to Longyearbyen on Svalbard. These can be lo
# Download the necessary data. This may take a few minutes.
xdem.examples.download_longyearbyen_examples(overwrite=False)

### Load the data using xdem and geoutils (could be with rasterio and geopandas instead)
# Load a reference DEM from 2009
reference_dem = xdem.DEM(xdem.examples.FILEPATHS["longyearbyen_ref_dem"])
# Load a moderately well aligned DEM from 1990
dem_to_be_aligned = xdem.DEM(xdem.examples.FILEPATHS["longyearbyen_tba_dem"])
dem_to_be_aligned = xdem.DEM(xdem.examples.FILEPATHS["longyearbyen_tba_dem"]).resample(reference_dem)
# Load glacier outlines from 1990. This will act as the unstable ground.
glacier_outlines = gu.Vector(xdem.examples.FILEPATHS["longyearbyen_glacier_outlines"])

.. contents:: Contents
:local:

# Prepare the inputs for coregistration.
ref_data = reference_dem.data # This is a numpy 2D array/masked_array
tba_data = dem_to_be_aligned.data # This is a numpy 2D array/masked_array
mask = glacier_outlines.create_mask(reference_dem) == 255 # This is a boolean numpy 2D array
transform = reference_dem.transform # This is a rio.transform.Affine object.



The Coreg object
^^^^^^^^^^^^^^^^^^^^
:class:`xdem.coreg.Coreg`

Each of the coregistration approaches in ``xdem`` inherit their interface from the ``Coreg`` class.
It is written in a style that should resemble that of ``scikit-learn`` (see their `LinearRegression <https://scikit-learn.org/stable/modules/generated/sklearn.linear_model.LinearRegression.html#sklearn-linear-model-linearregression>`_ class for example).
Each coregistration approach has the methods:

* ``.fit()`` for estimating the transform.
* ``.apply()`` for applying the transform to a DEM.
* ``.apply_pts()`` for applying the transform to a set of 3D points.
* ``.to_matrix()`` to convert the transform to a 4x4 transformation matrix, if possible.

First, ``.fit()`` is called to estimate the transform, and then this transform can be used or exported using the subsequent methods.

Nuth and Kääb (2011)
^^^^^^^^^^^^^^^^^^^^
:class:`xdem.coreg.NuthKaab`

- **Performs:** translation and bias corrections.
- **Supports weights** (soon)
- **Recommended for:** Noisy data with low rotational differences.
Expand All @@ -63,15 +93,17 @@ Example
*******
.. code-block:: python

aligned_dem, error = xdem.coreg.coregister(
reference_dem,
dem_to_be_aligned,
method="nuth_kaab",
mask=glacier_outlines
)
nuth_kaab = coreg.NuthKaab()
# Fit the data to a suitable x/y/z offset.
nuth_kaab.fit(ref_data, tba_data, transform=transform, mask=mask)

# Apply the transformation to the data (or any other data)
aligned_dem = nuth_kaab.apply(tba_data, transform=transform)

Deramping
^^^^^^^^^
:class:`xdem.coreg.Deramp`

- **Performs:** Bias, linear or nonlinear height corrections.
- **Supports weights** (soon)
- **Recommended for:** Data with no horizontal offset and low to moderate rotational differences.
Expand All @@ -91,17 +123,51 @@ Example
*******
.. code-block:: python

# Apply a 1st order deramping correction.
deramped_dem, error = xdem.coreg.coregister(
reference_dem,
dem_to_be_aligned,
method="deramp",
deramp_degree=1,
mask=glacier_outlines
)
# Instantiate a 1st order deramping object.
deramp = coreg.Deramp(degree=1)
# Fit the data to a suitable polynomial solution.
deramp.fit(ref_data, tba_data, transform=transform, mask=mask)

# Apply the transformation to the data (or any other data)
deramped_dem = deramp.apply(dem_to_be_aligned.data, transform=dem_to_be_aligned.transform)


Bias correction
^^^^^^^^^^^^^^^
:class:`xdem.coreg.BiasCorr`

- **Performs:** (Weighted) bias correction using the mean, median or anything else
- **Supports weights** (soon)
- **Recommended for:** A precursor step to e.g. ICP.

``BiasCorr`` has very similar functionality to ``Deramp(degree=0)`` or the z-component of `Nuth and Kääb (2011)`_.
This function is more customizable, for example allowing changing of the bias algorithm (from weighted average to e.g. median).
It should also be faster, since it is a single function call.

Limitations
***********
Only performs vertical corrections, so it should be combined with another approach.

Example
*******
.. code-block:: python

bias_corr = coreg.BiasCorr()
# Note that the transform argument is not needed, since it is a simple vertical correction.
bias_corr.fit(ref_data, tba_data, mask=mask)

# Apply the bias to a DEM
corrected_dem = bias_corr.apply(tba_data)

# Use median bias instead
bias_median = coreg.BiasCorr(bias_func=np.median)

bias_median.fit(... # etc.

ICP
^^^
:class:`xdem.coreg.ICP`

- **Performs:** Rigid transform correction (translation + rotation).
- **Does not support weights**
- **Recommended for:** Data with low noise and a high relative rotation.
Expand All @@ -117,8 +183,7 @@ This may improve results on noisy data significantly, but care should still be t

Limitations
***********
ICP is notoriously bad on noisy data.
TODO: Add references for ICP being bad on noisy data.
ICP often works poorly on noisy data.
The outlier removal functionality of the opencv implementation is a step in the right direction, but it still does not compete with other coregistration approaches when the relative rotation is small.
In cases of high rotation, ICP is the only approach that can account for this properly, but results may need refinement, for example with the `Nuth and Kääb (2011)`_ approach.

Expand All @@ -128,12 +193,63 @@ Example
*******
.. code-block:: python

# Use the opencv ICP implementation. For PDAL, use "icp_pdal")
aligned_dem, error = xdem.coreg.coregister(
reference_dem,
dem_to_be_aligned,
method="icp",
mask=glacier_outlines
)
# Instantiate the object with default parameters
icp = coreg.ICP()
# Fit the data to a suitable transformation.
icp.fit(ref_data, tba_data, transform=transform, mask=mask)

# Apply the transformation matrix to the data (or any other data)
aligned_dem = icp.apply(tba_data, transform=transform)


The CoregPipeline object
^^^^^^^^^^^^^^^^^^^^^^^^^^^^
:class:`xdem.coreg.CoregPipeline`

Often, more than one coregistration approach is necessary to obtain the best results.
For example, ICP works poorly with large initial biases, so a ``CoregPipeline`` can be constructed to perform both sequentially:

.. code-block:: python

pipeline = coreg.CoregPipeline([coreg.BiasCorr(), coreg.ICP()])

pipeline.fit(... # etc.

# This works identically to the syntax above
pipeline2 = coreg.BiasCorr() + coreg.ICP()

The ``CoregPipeline`` object exposes the same interface as the ``Coreg`` object.
The results of a pipeline can be used in other programs by exporting the combined transformation matrix:

.. code-block:: python

pipeline.to_matrix()


This class is heavily inspired by the `Pipeline <https://scikit-learn.org/stable/modules/generated/sklearn.pipeline.Pipeline.html#sklearn-pipeline-pipeline>`_ and `make_pipeline() <https://scikit-learn.org/stable/modules/generated/sklearn.pipeline.make_pipeline.html#sklearn.pipeline.make_pipeline>`_ functionalities in ``scikit-learn``.

Suggested pipelines
*******************

For sub-pixel accuracy, the `Nuth and Kääb (2011)`_ approach should almost always be used.
The approach does not account for rotations in the dataset, however, so a combination is often necessary.
For small rotations, a 1st degree deramp could be used:

.. code-block:: python

coreg.NuthKaab() + coreg.Deramp(degree=1)

For larger rotations, ICP is the only reliable approach (but does not outperform in sub-pixel accuracy):

.. code-block:: python

coreg.ICP() + coreg.NuthKaab()


For large biases, rotations and high amounts of noise:

.. code-block:: python

coreg.BiasCorr() + coreg.ICP() + coreg.NuthKaab()


3 changes: 2 additions & 1 deletion setup.py
Original file line number Diff line number Diff line change
Expand Up @@ -14,7 +14,8 @@
packages=['xdem'],
install_requires=['numpy', 'scipy', 'rasterio', 'geopandas',
'pyproj', 'tqdm', 'geoutils @ https://github.com/GlacioHack/GeoUtils/tarball/master', 'scikit-gstat'],
extras_require={'rioxarray': ['rioxarray'], 'richdem': ['richdem'], 'pdal': ['pdal'], 'opencv': ['opencv']},
extras_require={'rioxarray': ['rioxarray'], 'richdem': ['richdem'], 'pdal': [
'pdal'], 'opencv': ['opencv'], "pytransform3d": ["pytransform3d"]},
scripts=[],
zip_safe=False)

Expand Down
Loading