Source code for pygmt.src.subplot

"""
subplot - Manage modern mode figure subplot configuration and selection.
"""
import contextlib

import numpy as np
from pygmt.clib import Session
from pygmt.helpers import build_arg_string, fmt_docstring, kwargs_to_strings, use_alias


@fmt_docstring
@contextlib.contextmanager
@use_alias(
    Ff="figsize",
    Fs="subsize",
    A="autolabel",
    B="frame",
    C="clearance",
    M="margins",
    S="layout",
    T="title",
)
@kwargs_to_strings(Ff="sequence", Fs="sequence", M="sequence")
def subplot(self, nrows=1, ncols=1, **kwargs):
    """
    The **subplot** module is used to split the current figure into a
    rectangular layout of subplots that each may contain a single self-
    contained figure.

    Begin by defining the layout of the entire multi-panel illustration.
    Several options are available to specify the systematic layout, labeling,
    dimensions, and more for the subplots.

    Full option list at :gmt-docs:`subplot.html#synopsis-begin-mode`

    {aliases}

    Parameters
    ----------
    nrows : int
        Number of vertical rows of the subplot grid.
    ncols : int
        Number of horizontal columns of the subplot grid.
    figsize : tuple
        Specify the final figure dimensions as ``(width, height)``.
    subsize : tuple
        Specify the dimensions of each subplot directly as ``(width, height)``.

    autolabel : bool or str
        ``[autolabel][+cdx[/dy]][+gfill][+j|Jrefpoint][+odx[/dy]][+ppen][+r|R]
        [+v]``.
        Specify automatic tagging of each subplot. Append either a number or
        letter [a]. This sets the tag of the first, top-left subplot and others
        follow sequentially. Surround the number or letter by parentheses on
        any side if these should be typeset as part of the tag. Use
        **+j|J**\\ *refpoint* to specify where the tag should be placed in the
        subplot [TL]. Note: **+j** sets the justification of the tag to
        *refpoint* (suitable for interior tags) while **+J** instead selects
        the mirror opposite (suitable for exterior tags). Append
        **+c**\\ *dx*[/*dy*] to set the clearance between the tag and a
        surrounding text box requested via **+g** or **+p** [3p/3p, i.e., 15%
        of the FONT_TAG size dimension]. Append **+g**\\ *fill* to paint the
        tag's text box with *fill* [no painting]. Append
        **+o**\\ *dx*\\ [/*dy*] to offset the tag's reference point in the
        direction implied by the justification [4p/4p, i.e., 20% of the
        FONT_TAG size]. Append **+p**\\ *pen* to draw the outline of the tag's
        text box using selected *pen* [no outline]. Append **+r** to typeset
        your tag numbers using lowercase Roman numerals; use **+R** for
        uppercase Roman numerals [Arabic numerals]. Append **+v** to increase
        tag numbers vertically down columns [horizontally across rows].
    clearance : str
        ``[side]clearance``.
        Reserve a space of dimension *clearance* between the margin and the
        subplot on the specified side, using *side* values from **w**, **e**,
        **s**, or **n**, or **x** for both **w** and **e** or **y** for both
        **s** and **n**.  No *side* means all sides. The option is repeatable
        to set aside space on more than one side. Such space will be left
        untouched by the main map plotting but can be accessed by modules that
        plot scales, bars, text, etc.  Settings specified under **begin**
        directive apply to all subplots, while settings under **set** only
        apply to the selected (active) subplot. **Note**: Common options
        **x_offset** and **y_offset* are not available during subplots; use
        **clearance** instead.
    margins : tuple
        This is margin space that is added between neighboring subplots (i.e.,
        the interior margins) in addition to the automatic space added for tick
        marks, annotations, and labels. The margins can be specified as either:

        - a single value (for same margin on all sides). E.g. '5c'.
        - a pair of values (for setting separate horizontal and vertical
          margins). E.g. ['5c', '3c'].
        - a set of four values (for setting separate left, right, bottom, and
          top margins). E.g. ['1c', '2c', '3c', '4c'].

        The actual gap created is always a sum of the margins for the two
        opposing sides (e.g., east plus west or south plus north margins)
        [Default is half the primary annotation font size, giving the full
        annotation font size as the default gap].
    layout : str or list
        Set subplot layout for shared axes. May be set separately for rows
        (**R**) and columns (**C**). E.g. ``layout=['Rl', 'Cb']``.
        Considerations for **C**: Use when all subplots in a **C**\\ olumn
        share a common *x*-range. The first (i.e., **t**\\ op) and the last
        (i.e., **b**\\ ottom) rows will have *x* annotations; append **t** or
        **b** to select only one of those two rows [both]. Append **+l** if
        annotated *x*-axes should have a label [none]; optionally append the
        label if it is the same for the entire subplot. Append **+t** to make
        space for subplot titles for each row; use **+tc** for top row titles
        only [no subplot titles]. Labels and titles that depends on which row
        or column are specified as usual via a subplot's own **frame** setting.
        Considerations for **R**: Use when all subplots in a **R**\\ ow share a
        common *y*-range. The first (i.e., **l**\\ eft) and the last (i.e.,
        **r**\\ ight) columns will have *y*-annotations; append **l** or **r**
        to select only one of those two columns [both]. Append **+l** if
        annotated *y*-axes will have a label [none]; optionally, append the
        label if it is the same for the entire subplot. Append **+p** to make
        all annotations axis-parallel [horizontal]; if not used you may have to
        set **clearance** to secure extra space for long horizontal
        annotations. Append **+w** to draw horizontal and vertical lines
        between interior panels using selected pen [no lines].
    title : str
        Overarching heading for the entire figure. Font is determined by
        setting ``FONT_HEADING``.

    Yields
    ------
    axs : numpy.ndarray
        Array of Axes objects.
    """
    kwargs = self._preprocess(**kwargs)  # pylint: disable=protected-access

    with Session() as lib:
        try:
            arg_str = " ".join(["begin", f"{nrows}x{ncols}", build_arg_string(kwargs)])
            lib.call_module(module="subplot", args=arg_str)
            # Setup matplotlib-like Axes
            axs = np.empty(shape=(nrows, ncols), dtype=object)
            for index in range(nrows * ncols):
                i = index // ncols  # row
                j = index % ncols  # column
                axs[i, j] = index
            yield axs
        finally:
            v_arg = build_arg_string(kwargs.fromkeys("V"))
            lib.call_module("subplot", f"end {v_arg}".strip())


@fmt_docstring
@contextlib.contextmanager
@use_alias(A="fixedlabel", C="clearance", V="verbose")
def sca(self, ax=None, **kwargs):
    r"""
    Set the current Axes instance to *ax*.

    Before you start plotting you must first select the active subplot. Note:
    If any *projection* option is passed with **?** as scale or width when
    plotting subplots, then the dimensions of the map are automatically
    determined by the subplot size and your region. For Cartesian plots: If you
    want the scale to apply equally to both dimensions then you must specify
    ``projection="x"`` [The default ``projection="X"`` will fill the subplot by
    using unequal scales].

    {aliases}

    Parameters
    ----------
    ax : str
        *row,col*\|\ *index*.
        Sets the current subplot until further notice.  **Note**: First *row*
        or *col* is 0, not 1. If not given we go to the next subplot by order
        specified via **autolabel**.  As an alternative, you may bypass the
        **sca** mode and instead supply the common option **ax**\ [*row,col*]
        to the first plot command you issue in that subplot. GMT maintains
        information about the current figure and subplot. Also, you may give
        the one-dimensional *index* instead which starts at 0 and follows the
        row or column order set via **autolabel**.

    fixedlabel : str
        Overrides the automatic labeling with the given string. No modifiers
        are allowed. Placement, justification, etc. are all inherited from how
        **autolabel** was specified by the initial **subplot** command.

    clearance : str
        [*side*]\ *clearance*.
        Reserve a space of dimension *clearance* between the margin and the
        subplot on the specified side, using *side* values from **w**, **e**,
        **s**, or **n**.  The option is repeatable to set aside space on more
        than one side.  Such space will be left untouched by the main map
        plotting but can be accessed by modules that plot scales, bars, text,
        etc.  This setting overrides the common clearances set by **clearance**
        in the initial **subplot** call.

    {V}
    """
    kwargs = self._preprocess(**kwargs)  # pylint: disable=protected-access

    with Session() as lib:
        arg_str = " ".join(["set", f"{ax}", build_arg_string(kwargs)])
        lib.call_module(module="subplot", args=arg_str)
        yield