# Repository: https://gitlab.com/quantify-os/quantify-core
# Licensed according to the LICENCE file on the main branch
# pylint: disable=invalid-name # disabled because of capital SI in module name
"""
Utilities for managing SI units with plotting systems.
"""
import re
import string
from typing import Optional, Tuple, Union
import lmfit
import matplotlib
import matplotlib.pyplot as plt
import numpy as np
import uncertainties
golden_mean = (np.sqrt(5) - 1.0) / 2.0 # Aesthetic ratio
single_col_figsize = (3.39, golden_mean * 3.39)
double_col_figsize = (6.9, golden_mean * 6.9)
thesis_col_figsize = (12.2 / 2.54, golden_mean * 12.2 / 2.54)
[docs]def set_xlabel(axis, label, unit=None, **kw):
"""
Add a unit aware x-label to an axis object.
Parameters
----------
axis
matplotlib axis object to set label on
label
the desired label
unit
the unit
**kw
keyword argument to be passed to matplotlib.set_xlabel
"""
if unit is not None and unit != "":
xticks = axis.get_xticks()
scale_factor, unit = SI_prefix_and_scale_factor(val=max(abs(xticks)), unit=unit)
formatter = matplotlib.ticker.FuncFormatter(
lambda x, pos: f"{x * scale_factor:.4g}"
)
axis.xaxis.set_major_formatter(formatter)
axis.set_xlabel(label + f" [{unit}]", **kw)
else:
axis.set_xlabel(label, **kw)
return axis
[docs]def set_ylabel(axis, label, unit=None, **kw):
"""
Add a unit aware y-label to an axis object.
Parameters
----------
axis
matplotlib axis object to set label on
label
the desired label
unit
the unit
**kw
keyword argument to be passed to matplotlib.set_ylabel
"""
if unit is not None and unit != "":
yticks = axis.get_yticks()
scale_factor, unit = SI_prefix_and_scale_factor(val=max(abs(yticks)), unit=unit)
formatter = matplotlib.ticker.FuncFormatter(
lambda x, pos: f"{x * scale_factor:.6g}"
)
axis.yaxis.set_major_formatter(formatter)
axis.set_ylabel(label + f" [{unit}]", **kw)
else:
axis.set_ylabel(label, **kw)
return axis
[docs]def set_cbarlabel(cbar, label, unit=None, **kw):
"""
Add a unit aware z-label to a colorbar object
Parameters
----------
cbar
colorbar object to set label on
label
the desired label
unit
the unit
**kw
keyword argument to be passed to cbar.set_label
"""
if unit is not None and unit != "":
zticks = cbar.get_ticks()
scale_factor, unit = SI_prefix_and_scale_factor(val=max(abs(zticks)), unit=unit)
formatter = matplotlib.ticker.FuncFormatter(
lambda x, pos: f"{x * scale_factor:.6g}"
)
cbar.ax.yaxis.set_major_formatter(formatter)
cbar.set_label(label + f" [{unit}]")
else:
cbar.set_label(label, **kw)
return cbar
# pylint: disable=invalid-name
[docs]def adjust_axeslabels_SI(ax) -> None:
"""
Auto adjust the labels of a plot generated by xarray to SI-unit aware labels.
"""
xlabel = ax.get_xlabel()
idxl = xlabel.find("[")
idxr = xlabel.find("]")
# only update the label if brackets are present
if idxl != -1 and idxr != -1:
# extract unit
xunit = xlabel[idxl + 1 : idxr]
xlabel = xlabel[: -(len(xunit) + 3)]
# replace by a unit aware label formatter
set_xlabel(ax, xlabel, xunit)
ylabel = ax.get_ylabel()
idxl = ylabel.find("[")
idxr = ylabel.find("]")
# only update the label if brackets are present
if idxl != -1 and idxr != -1:
yunit = ylabel[idxl + 1 : idxr]
ylabel = ylabel[: -(len(yunit) + 3)]
# replace by a unit aware label formatter
set_ylabel(ax, ylabel, yunit)
SI_PREFIXES = dict(zip(range(-24, 25, 3), "yzafpnμm kMGTPEZY"))
SI_PREFIXES[0] = ""
# N.B. not all of these are SI units, however, all of these support SI prefixes
SI_UNITS = (
"SI_PREFIX_ONLY,m,s,g,W,J,V,A,F,T,Hz,Ohm,S,N,C,px,b,B,K,Bar,"
r"Vpeak,Vpp,Vp,Vrms,A/s,$\Phi_0$".split(",")
) # noqa: W605
_SI_PREFIX_TO_FACTOR_MAPPING = {v: 10**key for key, v in SI_PREFIXES.items()}
_SI_PREFIX_TO_FACTOR_MAPPING["u"] = 10**-6
_prefix_regexp = "(" + "|".join(list("yzafpnμmkMGTPEZY")) + ")"
_si_regex = "(" + "|".join(map(re.escape, SI_UNITS)) + ")"
_prefixed_si_regex = re.compile(f"{_prefix_regexp}{_si_regex}$")
[docs]def SI_prefix_and_scale_factor(
val: float, unit: Optional[str] = None
) -> Tuple[float, str]:
"""
Takes in a value and unit, returns a scale factor and scaled unit.
It returns a scale factor to convert the input value to a value in the
range [1.0, 1000.0), plus the corresponding scaled SI unit (e.g. 'mT', 'kV'),
deduced from the input unit, to represent the input value in those scaled units.
The scaling is only applied if the unit is an unscaled or scaled unit present in
the variable :data::`SI_UNITS`.
If the unit is None, no scaling is done.
If the unit is "SI_PREFIX_ONLY", the value is scaled and an SI prefix is applied
without a base unit.
Parameters
----------
val : float
the value
unit : str
the unit of the value
Returns
-------
scale_factor : float
scale_factor needed to convert value
scaled_unit : str
unit including the prefix
"""
if unit and val is not None and (match := _prefixed_si_regex.match(unit)):
scale_part = match.group(1)
unit_part = match.group(2)
plus_scale = _SI_PREFIX_TO_FACTOR_MAPPING[scale_part]
scale_factor, scaled_unit = SI_prefix_and_scale_factor(
plus_scale * val, unit_part
)
return plus_scale * scale_factor, scaled_unit
if unit in SI_UNITS:
try:
with np.errstate(all="ignore"):
prefix_power = np.log10(abs(val)) // 3 * 3
prefix = SI_PREFIXES[prefix_power]
# Greek symbols not supported in tex
if plt.rcParams["text.usetex"] and prefix == "μ":
prefix = r"$\mu$"
if unit == "SI_PREFIX_ONLY":
scale_factor, scaled_unit = 10**-prefix_power, prefix
else:
scale_factor, scaled_unit = 10**-prefix_power, prefix + unit
# this exception can be triggered in the pyqtgraph multi processing
except (KeyError, TypeError):
scale_factor, scaled_unit = 1, unit
elif unit is None:
scale_factor, scaled_unit = 1, ""
else:
scale_factor, scaled_unit = 1, unit
return scale_factor, scaled_unit
[docs]def SI_val_to_msg_str(val: Union[float, int], unit: str = None, return_type=str):
"""
Takes in a value with optional unit and returns a string tuple consisting of
(value_str, unit) where the value and unit are rescaled according to SI prefixes,
IF the unit is an SI unit (according to the comprehensive list of
SI units in this file ;).
the value_str is of the type specified in return_type (str) by default.
"""
sc, new_unit = SI_prefix_and_scale_factor(val, unit)
try:
new_val = sc * val
except TypeError:
return return_type(val), unit
# Convert floats to int if possible and if inital value was int
if isinstance(val, int):
if isinstance(new_val, float) and new_val.is_integer():
new_val = int(new_val)
return return_type(new_val), new_unit
class SafeFormatter(string.Formatter):
def __init__(self, missing: str = "~~", bad_fmt: str = "!!"):
# pylint: disable=line-too-long
"""
A formatter that replaces "missing" values and "bad_fmt" to prevent unexpected
Exceptions being raised.
Parameters
----------
missing
Replaces missing values with specified string.
bad_fmt
Replaces values that cannot be formatted with specified string.
Based on https://stackoverflow.com/questions/20248355/how-to-get-python-to-gracefully-format-none-and-non-existing-fields
"""
self.missing, self.bad_fmt = missing, bad_fmt
def get_field(self, field_name, args, kwargs):
# Handle a key not found
try:
val = super().get_field(field_name, args, kwargs)
# Python 3, 'super().get_field(field_name, args, kwargs)' works
except (KeyError, AttributeError):
val = None, field_name
return val
def format_field(self, value, format_spec):
# handle an invalid format
if value is None:
return self.missing
try:
return super().format_field(value, format_spec)
except ValueError as e:
if self.bad_fmt is not None:
return self.bad_fmt
raise e
[docs]def value_precision(val, stderr=None) -> Tuple[str, str]:
"""
Calculate the precision to which a parameter is to be specified, according to
its standard error. Returns the appropriate format specifier string.
If there is no stderr, use 5 significant figures.
If there is a standard error use a precision one order of magnitude more precise
than the size of the error and display the stderr itself to two significant figures
in standard index notation in the same units as the value.
Parameters
----------
val: float
the nominal value of the parameter
stderr: float
the standard error on the parameter
Returns
----------
val_format_specifier: str
python format specifier which sets the precision of the parameter value
err_format_specifier: str
python format specifier which set the precision of the error
"""
if stderr is None or stderr == 0:
return "{:.5g}", "{:.1f}"
value_mag = np.floor(np.log10(abs(val))) + 1
err_mag = np.floor(np.log10(abs(stderr))) + 1
# pylint: disable=no-else-return
if err_mag == 2:
return "{:.0f}", "{:.0f}"
elif err_mag == 1:
return "{:.1f}", "{:.1f}"
else:
sig_figs = int(
max(value_mag - err_mag + 2, 2)
) # If the error is the same size as the value or larger, use 2 sig figs
return "{:#." + "{:d}".format(sig_figs) + "g}", "{:#.2g}"