===============
Alignment Model
===============
An alignment model defines the protocols for pre-transformation and alignment scoring.
.. contents:: Contents
:local:
:depth: 1
Alignment Workflows
===================
There are three types of alignment workflows.
Single template
---------------
If the alignment model is created with a single template image, the workflow is very simple.
It masks and pre-transforms both the sub-volume and the template, and then aligns them.
.. mermaid::
graph LR
vol(sub-volume\nat i-th molecule)
tmp[[template image]]
vol_t(transformed\nsub volume)
tmp_t[[transformed\ntemplate]]
aln{alignment}
result[alignment results]
vol--masking &
pre-transformation-->vol_t-->aln
tmp--masking &
pre-transformation-->tmp_t-->aln
aln-->result
Multiple templates
------------------
If the alignment model is created with multiple template images, masking, pre-transformation
and alignment are performed for each template separately. The alignment result with the best
score will be considered as the optimal result.
.. mermaid::
graph LR
subgraph Subvolumes
vol(sub-volume\nat i-th molecule)
vol_t(transformed\nsub volume)
vol--masking &
pre-transformation-->vol_t
end
subgraph Templates
tmp0[[template image A]]
tmp1[[template image B]]
tmp2[[template image C]]
tmp0_t[[transformed\ntemplate A]]
tmp1_t[[transformed\ntemplate B]]
tmp2_t[[transformed\ntemplate C]]
tmp0--masking &
pre-transformation-->tmp0_t
tmp1--masking &
pre-transformation-->tmp1_t
tmp2--masking &
pre-transformation-->tmp2_t
end
subgraph Alignment
aln0{alignment}
aln1{alignment}
aln2{alignment}
end
result[best alignment results]
vol_t-->aln0
vol_t-->aln1
vol_t-->aln2
tmp0_t-->aln0
tmp1_t-->aln1
tmp2_t-->aln2
aln0-->result
aln1-->result
aln2-->result
Single template with rotation
-----------------------------
Many alignment methods do not search for the optimal rotation of the template image. In this
case, rotated templates will be generated and used for alignment. Essentially, it is the same
as the multiple-template workflow.
.. mermaid::
graph LR
subgraph Subvolumes
vol(sub-volume\nat i-th molecule)
vol_t(transformed\nsub volume)
vol--masking &
pre-transformation-->vol_t
end
subgraph Templates
tmp[[template image]]
tmp_t[[transformed\ntemplate]]
tmp--masking &
pre-transformation-->tmp_t
tmp0[[template image A]]
tmp1[[template image B]]
tmp2[[template image C]]
rot{image rotation}
tmp_t-->rot
rot-->tmp0
rot-->tmp1
rot-->tmp2
end
subgraph Alignment
aln0{alignment}
aln1{alignment}
aln2{alignment}
end
result[best alignment results]
vol_t-->aln0
vol_t-->aln1
vol_t-->aln2
tmp0-->aln0
tmp1-->aln1
tmp2-->aln2
aln0-->result
aln1-->result
aln2-->result
Ready-to-use Models
===================
There are now two alignment models that can be used directly.
1. :class:`ZNCCAlignment`
Model that align subvolumes using ZNCC (Zero-mean Normalized Cross Correlation) score.
2. :class:`PCCAlignment`
Model that align subvolumes using PCC (Phase Cross Correlation) score.
Both models are implemented with low-pass filtering, template rotation and missing
wedge masking.
Model construction
------------------
.. code-block:: python
from acryo.alignment import ZNCCAlignment
model = ZNCCAlignment(
template, # template image
mask, # mask image
rotations=[(10, 5), (4, 2), (8, 4)],
cutoff=0.5,
tilt=(-60, 60),
)
- Shape of ``mask`` must be the same as ``template``. ``template * mask`` and
``subvolume * mask`` will be used for alignment.
- ``rotations`` can be three tuples or a :class:`scipy.spatial.transform.Rotation` object.
- If three tuples are given, each tuple defines the maximum rotation angle and the increment
around z, y or x (external) axis. The unit is degree. For example, the first ``(10, 5)``
means that the rotation angles -10, -5, 0, 5, 10 will be used for the rotation around z axis.
- If a :class:`scipy.spatial.transform.Rotation` object is given, all the rotations in the
object will be used for alignment. Make sure that the identity rotation is included.
- ``cutoff`` is the relative cutoff frequency for low-pass filtering. The Nyquist frequency is
:math:`0.5 \times \sqrt{3} = 0.866` for 3D images.
- ``tilt`` is the tilt series model.
- If a (float, float) tuple is given, it will be interpreted as the minimum and maximum tilt
angle in degree. The rotation axis is assumed to be the y axis.
- You can use classes provided in :mod:`acryo.tilt` to define more complex tilt series
models. For instance, :meth:`single_axis` for single-axis tilt series and
:meth:`dual_axis` for dual-axis tilt series.
Align images
------------
The :meth:`align` method is used to align a sub-volume to the template image of the model.
Note that this method does not actually transform the sub-volume to the template. It only
calculate the optimal shift/rotation parameters. To transform the sub-volume, use :meth:`fit`.
.. code-block:: python
result = model.align(
img,
max_shifts,
quaternion,
pos,
backend,
)
- ``img`` is the sub-volume to be aligned. It must be a 3D array with the same shape
as the template.
- ``max_shifts`` is a tuple of maximum shifts in z, y and x direction. The unit is pixel but
it can be a float number.
- ``quaternion`` is the rotation of the sub-volume in the original tomogram. It must be a (4,)
:class:`numpy.ndarray` object of quaternion. If you are using :class:`acryo.Molecules`,
its quaternions can directly be used here. This is basically used to mask the missing wedge.
- ``pos`` is the position of the sub-volume in the original tomogram. It must be a (3,)
:class:`numpy.ndarray` object. Default alignment models does not use this parameter.
- ``backend`` is the array API backend. It can be ``"numpy"`` or ``"cupy"``.
The return value ``result`` is a named-tuple :class:`AlignmentResult` object. It contains the
following fields.
.. code-block:: python
class AlignmentResult(NamedTuple):
label: int
shift: NDArray[np.float32]
quat: NDArray[np.float32]
score: float
- ``label`` is the integer label of the best alignment if multiple templates are used.
- ``shift`` is the optimal shift in z, y and x direction.
- ``quat`` is the optimal rotation in quaternion.
- ``score`` is the alignment score of the best alignment.
Fit images
----------
The :meth:`fit` method is used to transform the sub-volume to fit the template image of the
model. It is essentially the same as calling :meth:`align` for every rotation and then
Affine transform the sub-volume to the best alignment result, but :meth:`fit` is faster
because it parallelizes the rotation and alignment processes.
.. code-block:: python
result = model.fit(
img,
max_shifts,
cval=0.0,
backend=None,
)
- ``img`` and ``max_shifts`` is the same as :meth:`align`.
- ``cval`` is the constant value used for Affine transformations. 1% percentile will be used
by default.
Correlation landscape
---------------------
The word "correlation landscape" came from "energy landscape" in the context of protein
folding. It is a 3D array of the correlation scores between the sub-volume and the template
image.
.. code-block:: python
arr = model.landscape(
img,
max_shifts: tuple[float, float, float],
quaternion: NDArray[np.float32] | None = None,
pos: NDArray[np.float32] | None = None,
upsample: int = 1,
backend: Backend | None = None,
)
- ``img`` is the sub-volume to be aligned. It must be a 3D array with the same shape
as the template.
- ``max_shifts`` is a tuple of maximum shifts in z, y and x direction. The unit is pixel but
it can be a float number.
- ``quaternion`` is the rotation of the sub-volume in the original tomogram. It must be a (4,)
:class:`numpy.ndarray` object of quaternion. If you are using :class:`acryo.Molecules`,
its quaternions can directly be used here. This is basically used to mask the missing wedge.
- ``pos`` is the position of the sub-volume in the original tomogram. It must be a (3,)
:class:`numpy.ndarray` object. Default alignment models does not use this parameter.
Define Custom Alignment Model
=============================
In :mod:`acryo.alignment`, there are several abstract base classes that can be used to
efficiently define custom alignment models.
- :class:`BaseAlignmentModel` ... The most basic one that provides the minimum interface.
Need to override :meth:`_optimize` and :meth:`pre_transform`.
- :class:`RotationImplemented` ... Rotated templates will be generated even if the
optimization algorithm does not optimize the rotation. Need to override :meth:`_optimize`
and :meth:`pre_transform`.
- :class:`TomographyInput` ... Rotation, low-pass filtering and missing wedge masking is
already implemented. Only need to override :meth:`_optimize`.
When you override methods, the following should be noted.
- :meth:`pre_transform`
This method must have the following signature.
.. code-block:: python
def pre_transform(
self,
image: NDArray[np.float32],
backend=None,
) -> NDArray[np.complex64]:
...
The input image could be either the sub-volume or the template image. It is masked by
the input mask image but is not masked by the missing wedge mask in :class:`TomographyInput`.
The output image will be directly passed to the :meth:`_optimize` method, so the data
type depends on the implementation. ``backend`` is the backend array API object. You don't
have to use it unless you want to implement same method for GPU.
- :meth:`_optimize`
This method must have the following signature.
.. code-block:: python
def _optimize(
self,
subvolume: NDArray[T],
template: NDArray[T],
max_shifts: tuple[float, float, float],
quaternion: NDArray[np.float32],
pos: NDArray[np.float32],
backend=None,
) -> tuple[NDArray[np.float32], NDArray[np.float32], float]:
...
This method is called for every set of sub-volume and template images.
- ``subvolume`` and ``template`` is the images *after* pre-transformation.
Thus, they could be Fourier transformed.
- ``max_shift`` is directly passed from :meth:`align` or :meth:`fit` method.
- ``quaternion`` is the rotation of the sub-volume. This parameter can be used
to mask the missing wedge.
- ``pos`` is the position of the sub-volume in the original tomogram. Its
unit is pixel. This parameter can be used for CTF correction of defocusing.
- ``backend`` is the backend array API object. You don't have to use it unless
you want to implement same method for GPU.
- The return value must be a tuple of ``(shift, rotation, score)``.
- ``shift`` is the optimal shift in z, y and x direction. More precisely,
``ndi.shift(img, -shift)`` will properly align the image to the template.
- ``rotation`` is the optimal rotation in quaternion. If the alignment model
does not optimize the rotation, this value should be ``array([0, 0, 0, 1])``.
- ``score`` is the score of the alignment. Larger score means better alignment.