.. DO NOT EDIT. .. THIS FILE WAS AUTOMATICALLY GENERATED BY SPHINX-GALLERY. .. TO MAKE CHANGES, EDIT THE SOURCE PYTHON FILE: .. "examples/compute_head_direction.py" .. LINE NUMBERS ARE GIVEN BELOW. .. only:: html .. note:: :class: sphx-glr-download-link-note :ref:`Go to the end ` to download the full example code. or to run this example in your browser via Binder .. rst-class:: sphx-glr-example-title .. _sphx_glr_examples_compute_head_direction.py: Compute head direction ========================= Compute the head direction vector and angle using different methods. .. GENERATED FROM PYTHON SOURCE LINES 8-10 Imports ------- .. GENERATED FROM PYTHON SOURCE LINES 10-25 .. code-block:: Python # For interactive plots: install ipympl with `pip install ipympl` and uncomment # the following line in your notebook # %matplotlib widget import numpy as np from matplotlib import pyplot as plt from movement import sample_data from movement.kinematics import ( compute_forward_vector, compute_forward_vector_angle, ) from movement.plots import plot_centroid_trajectory from movement.utils.vector import cart2pol, pol2cart .. GENERATED FROM PYTHON SOURCE LINES 26-30 Load sample dataset ------------------- In this tutorial, we will use a sample dataset with a single individual (a mouse) and six keypoints. .. GENERATED FROM PYTHON SOURCE LINES 30-38 .. code-block:: Python ds = sample_data.fetch_dataset("DLC_single-mouse_EPM.predictions.h5") print(ds) print("-----------------------------") print(f"Individuals: {ds.individuals.values}") print(f"Keypoints: {ds.keypoints.values}") .. rst-class:: sphx-glr-script-out .. code-block:: none Size: 4MB Dimensions: (time: 18485, space: 2, keypoints: 8, individuals: 1) Coordinates: * time (time) float64 148kB 0.0 0.03333 0.06667 ... 616.1 616.1 616.1 * space (space) ` function can help you visualise the trajectory of any keypoint in the data. Passing a list of keypoints, in this case ``["left_ear", "right_ear"]``, will plot the centroid (midpoint) of the selected keypoints. By default, the first individual in the dataset is shown. .. GENERATED FROM PYTHON SOURCE LINES 61-89 .. code-block:: Python # Create figure and axis fig, ax = plt.subplots(1, 1) # Plot a single frame from the dataset (its path is stored as an attribute) frame = plt.imread(ds.frame_path) ax.imshow(frame) # Plot the trajectory of the head centre plot_centroid_trajectory( ds.position, keypoints=["left_ear", "right_ear"], ax=ax, # arguments forwarded to plt.scatter s=10, cmap="viridis", marker="o", alpha=0.05, ) # Adjust title ax.set_title("Head trajectory") ax.set_ylim(frame.shape[0], 0) # match y-axis limits to image coordinates ax.set_xlabel("x (pixels)") ax.set_ylabel("y (pixels)") ax.collections[0].colorbar.set_label("Time (seconds)") fig.show() .. image-sg:: /examples/images/sphx_glr_compute_head_direction_001.png :alt: Head trajectory :srcset: /examples/images/sphx_glr_compute_head_direction_001.png :class: sphx-glr-single-img .. GENERATED FROM PYTHON SOURCE LINES 90-95 We can see that most of the head trajectory data is within a cruciform shape, because the mouse is moving on an `Elevated Plus Maze `_. The plot suggests the mouse spends most of its time in the covered arms of the maze (the vertical arms). .. GENERATED FROM PYTHON SOURCE LINES 97-101 Compute the head-to-snout vector -------------------------------- We can define the head direction as the vector from the midpoint between the ears to the snout. .. GENERATED FROM PYTHON SOURCE LINES 101-114 .. code-block:: Python # Compute the head centre as the midpoint between the ears midpoint_ears = position.sel(keypoints=["left_ear", "right_ear"]).mean( dim="keypoints" ) # Snout position # (`drop=True` removes the keypoints dimension, which is now redundant) snout = position.sel(keypoints="snout", drop=True) # Compute the head vector as the difference vector between the snout position # and the head-centre position. head_to_snout = snout - midpoint_ears .. GENERATED FROM PYTHON SOURCE LINES 115-130 .. admonition:: Vector subtraction :class: note Note that any position data point can be seen as a point :math:`U` in the 2D plane, or as a 2D vector :math:`\vec{u}` that goes from the image coordinate system origin (by default, the centre of the top-left pixel) to the point :math:`U` (see left subplot). The vector that goes from point :math:`U` to point :math:`V` can be computed as the difference :math:`\vec{v} - \vec{u}` (see right subplot). .. image:: ../_static/Vector-Subtraction.png :width: 600 :alt: Schematic showing vector subtraction .. GENERATED FROM PYTHON SOURCE LINES 132-135 Let's validate our computation by plotting the head-to-snout vector alongside the midpoint between the ears and the snout position. We will do this for a small time window to make the plot more readable. .. GENERATED FROM PYTHON SOURCE LINES 135-183 .. code-block:: Python # Time window to restrict the plot time_window = slice(54.9, 55.1) # seconds fig, ax = plt.subplots() # Plot the computed head-to-snout vector originating from the ears midpoint ax.quiver( midpoint_ears.sel(space="x", time=time_window), midpoint_ears.sel(space="y", time=time_window), head_to_snout.sel(space="x", time=time_window), head_to_snout.sel(space="y", time=time_window), color="gray", angles="xy", scale=1, scale_units="xy", headwidth=4, headlength=5, headaxislength=5, label="Head-to-snout vector", ) # Plot midpoint between the ears within the time window plot_centroid_trajectory( midpoint_ears.sel(time=time_window), ax=ax, s=60, label="ears midpoint", ) # Plot the snout position within the time window plot_centroid_trajectory( snout.sel(time=time_window), ax=ax, s=60, marker="*", label="snout", ) # Calling plot_centroid_trajectory twice will add 2 identical colorbars # so we remove 1 ax.collections[2].colorbar.remove() ax.set_title("Zoomed in head-to-snout vectors") ax.invert_yaxis() # invert y-axis to match image coordinates ax.legend(loc="upper left") .. image-sg:: /examples/images/sphx_glr_compute_head_direction_002.png :alt: Zoomed in head-to-snout vectors :srcset: /examples/images/sphx_glr_compute_head_direction_002.png :class: sphx-glr-single-img .. rst-class:: sphx-glr-script-out .. code-block:: none .. GENERATED FROM PYTHON SOURCE LINES 184-190 Head-to-snout vector in polar coordinates ----------------------------------------- Now that we have the head-to-snout vector, we can compute its angle in 2D space. A convenient way to achieve that is to convert the vector from cartesian to polar coordinates using the :func:`cart2pol()` function. .. GENERATED FROM PYTHON SOURCE LINES 190-195 .. code-block:: Python head_to_snout_polar = cart2pol(head_to_snout) print(head_to_snout_polar) .. rst-class:: sphx-glr-script-out .. code-block:: none Size: 296kB 1.637 1.995 1.624 1.99 1.67 1.994 1.681 ... 22.11 0.937 22.12 0.9768 20.81 1.011 Coordinates: * time (time) float64 148kB 0.0 0.03333 0.06667 ... 616.1 616.1 616.1 individuals `_ convention). The ``phi`` angle is positive if the rotation from the positive x-axis to the vector is in the same direction as the rotation from the positive x-axis to the positive y-axis. In the default image coordinate system, this means ``phi`` will be positive if the rotation is clockwise, and negative if the rotation is anti-clockwise. .. image:: ../_static/Cartesian-vs-Polar.png :width: 600 :alt: Schematic comparing cartesian and polar coordinates .. GENERATED FROM PYTHON SOURCE LINES 222-226 ``movement`` also provides a ``pol2cart`` function to transform data in polar coordinates to cartesian. Note that the resulting ``head_to_snout_cart`` array has a ``space`` dimension with two coordinates: ``x`` and ``y``. .. GENERATED FROM PYTHON SOURCE LINES 226-230 .. code-block:: Python head_to_snout_cart = pol2cart(head_to_snout_polar) print(head_to_snout_cart) .. rst-class:: sphx-glr-script-out .. code-block:: none Size: 296kB -0.6741 1.492 -0.661 1.484 -0.6864 1.522 ... 13.09 17.82 12.38 18.33 11.05 17.64 Coordinates: * time (time) float64 148kB 0.0 0.03333 0.06667 ... 616.1 616.1 616.1 individuals ` function, which takes a different approach to the one we used above: it accepts a pair of bilaterally symmetric keypoints and computes the vector that originates at the midpoint between the keypoints and is perpendicular to the line connecting them. Here we will use the two ears to find the head direction vector. We may prefer this method if we expect the snout detection to be unreliable (e.g., because it's often occluded in a top-down camera view). .. GENERATED FROM PYTHON SOURCE LINES 243-252 .. code-block:: Python forward_vector = compute_forward_vector( position, left_keypoint="left_ear", right_keypoint="right_ear", camera_view="top_down", ) print(forward_vector) .. rst-class:: sphx-glr-script-out .. code-block:: none Size: 296kB -0.01196 0.9999 -0.01182 0.9999 -0.0153 ... 0.9058 0.4111 0.9116 0.3797 0.9251 Coordinates: * space (space) `, which makes the intent of the function clearer. .. GENERATED FROM PYTHON SOURCE LINES 281-291 Compute head direction angle ---------------------------- We may want to explicitly compute the orientation of the animal's head as an angle, rather than as a vector. We can compute this angle from the forward vector as we did with the head-to-snout vector, i.e., by converting the vector to polar coordinates and extracting the ``phi`` coordinate. However, it's more convenient to use the :func:`compute_forward_vector_angle()\ ` function, which by default would return the same ``phi`` angle. .. GENERATED FROM PYTHON SOURCE LINES 291-303 .. code-block:: Python forward_vector_angle = compute_forward_vector_angle( position, left_keypoint="left_ear", right_keypoint="right_ear", # Optional parameters: reference_vector=(1, 0), # positive x-axis camera_view="top_down", in_degrees=False, # set to True for degrees ) print(forward_vector_angle) .. rst-class:: sphx-glr-script-out .. code-block:: none Size: 148kB 1.583 1.583 1.586 1.588 1.585 1.585 ... 1.088 1.171 1.258 1.133 1.147 1.181 Coordinates: * time (time) float64 148kB 0.0 0.03333 0.06667 ... 616.1 616.1 616.1 individuals ` .. container:: sphx-glr-download sphx-glr-download-python :download:`Download Python source code: compute_head_direction.py ` .. container:: sphx-glr-download sphx-glr-download-zip :download:`Download zipped: compute_head_direction.zip ` .. only:: html .. rst-class:: sphx-glr-signature `Gallery generated by Sphinx-Gallery `_