"""
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