Skip to content

Commit

Permalink
Document limitations of GMT xarray accessors (#2375)
Browse files Browse the repository at this point in the history
Co-authored-by: Yvonne Fröhlich <94163266+yvonnefroehlich@users.noreply.github.com>
Co-authored-by: Wei Ji <23487320+weiji14@users.noreply.github.com>
  • Loading branch information
3 people authored Feb 22, 2023
1 parent 0c21c65 commit 8aec841
Showing 1 changed file with 88 additions and 17 deletions.
105 changes: 88 additions & 17 deletions pygmt/accessors.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
"""
GMT accessor methods.
GMT accessor for :class:`xarray.DataArray`.
"""
from pathlib import Path

Expand All @@ -11,15 +11,36 @@
@xr.register_dataarray_accessor("gmt")
class GMTDataArrayAccessor:
"""
GMT extension for :class:`xarray.DataArray`.
GMT accessor for :class:`xarray.DataArray`.
The extension provides easy ways to access and change the GMT specific
properties about grids. Currently, two properties are available:
The accessor extends :class:`xarray.DataArray` to store GMT-specific
properties about grids, which are important for PyGMT to correctly process
and plot the grids.
- ``registration``: Gridline (0) or Pixel (1) registration
- ``gtype``: Cartesian (0) or Geographic (1) coordinate system
Attributes
----------
You can access these GMT specific properties about your grid as follows:
registration : int
Registration type of the grid, either 0 (Gridline) or 1 (Pixel).
gtype : int
Coordinate system type of the grid, either 0 (Cartesian) or 1
(Geographic).
Notes
-----
Due to the limitations of xarray accessors, the GMT accessors are created
once per :class:`xarray.DataArray` instance. You may lose these
GMT-specific properties when manipulating grids (e.g., arithmetic and slice
operations) or when accessing a :class:`xarray.DataArray` from a
:class:`xarray.Dataset`. In these cases, you need to manually set these
properties before passing the grid to PyGMT.
Examples
--------
For GMT's built-in remote datasets, these GMT-specific properties are
automatically determined and you can access them as follows:
>>> from pygmt.datasets import load_earth_relief
>>> # Use the global Earth relief grid with 1 degree spacing
Expand All @@ -31,7 +52,10 @@ class GMTDataArrayAccessor:
>>> grid.gmt.gtype
1
You can also set the GMT specific properties for grids created by yourself:
For :class:`xarray.DataArray` grids created by yourself, grid properties
``registration`` and ``gtype`` default to 0 (i.e., a gridline-registered,
Cartesian grid). You need to set the correct properties before
passing it to PyGMT functions:
>>> import numpy as np
>>> import pygmt
Expand All @@ -45,9 +69,58 @@ class GMTDataArrayAccessor:
>>> grid = xr.DataArray(
... data, coords=[("latitude", lat), ("longitude", lon)]
... )
>>> # default to a gridline-registrated Cartesian grid
>>> grid.gmt.registration, grid.gmt.gtype
(0, 0)
>>> # set it to a gridline-registered geographic grid
>>> grid.gmt.registration = 0
>>> grid.gmt.gtype = 1
>>> grid.gmt.registration, grid.gmt.gtype
(0, 1)
Note that the accessors are created once per :class:`xarray.DataArray`
instance, so you may lose these GMT-specific properties after manipulating
your grid.
Inplace assignment operators like ``*=`` don't create new instances, so the
properties are still kept:
>>> grid *= 2.0
>>> grid.gmt.registration, grid.gmt.gtype
(0, 1)
Other grid operations (e.g., arithmetic or slice operations) create new
instances, so the properties will be lost:
>>> # grid2 is a slice of the original grid
>>> grid2 = grid[0:30, 50:80]
>>> # properties are reset to the default values for new instance
>>> grid2.gmt.registration, grid2.gmt.gtype
(0, 0)
>>> # need to set these properties before passing the grid to PyGMT
>>> grid2.gmt.registration = grid.gmt.registration
>>> grid2.gmt.gtype = grid.gmt.gtype
>>> grid2.gmt.registration, grid2.gmt.gtype
(0, 1)
Accesing a :class:`xarray.DataArray` from a :class:`xarray.Dataset` always
creates new instances, so these properties are always lost. The workaround
is to assign the :class:`xarray.DataArray` into a variable:
>>> ds = xr.Dataset({"zval": grid})
>>> ds.zval.gmt.registration, ds.zval.gmt.gtype
(0, 0)
>>> # manually set these properties won't work as expected
>>> ds.zval.gmt.registration, ds.zval.gmt.gtype = 0, 1
>>> ds.zval.gmt.registration, ds.zval.gmt.gtype
(0, 0)
>>> # workaround: assign the DataArray into a variable
>>> zval = ds.zval
>>> zval.gmt.registration, zval.gmt.gtype
(0, 0)
>>> zval.gmt.registration, zval.gmt.gtype = 0, 1
>>> zval.gmt.registration, zval.gmt.gtype
(0, 1)
"""

def __init__(self, xarray_obj):
Expand All @@ -72,34 +145,32 @@ def __init__(self, xarray_obj):
@property
def registration(self):
"""
Registration type of the grid, either Gridline (0) or Pixel (1).
Registration type of the grid, either 0 (Gridline) or 1 (Pixel).
"""
return self._registration

@registration.setter
def registration(self, value):
if value in (0, 1):
self._registration = value
else:
if value not in (0, 1):
raise GMTInvalidInput(
f"Invalid grid registration value: {value}, should be either "
"0 for Gridline registration or 1 for Pixel registration."
)
self._registration = value

@property
def gtype(self):
"""
Coordinate system type of the grid, either Cartesian (0) or Geographic
(1).
Coordinate system type of the grid, either 0 (Cartesian) or 1
(Geographic).
"""
return self._gtype

@gtype.setter
def gtype(self, value):
if value in (0, 1):
self._gtype = value
else:
if value not in (0, 1):
raise GMTInvalidInput(
f"Invalid coordinate system type: {value}, should be "
"either 0 for Cartesian or 1 for Geographic."
)
self._gtype = value

0 comments on commit 8aec841

Please sign in to comment.