# -*- coding: utf-8 -*-
# -----------------------------------------------
# Filename: mopad_wrapper.py
# Purpose: Wrapper for mopad
# Author: Tobias Megies, Moritz Beyreuther
# Email: megies@geophysik.uni-muenchen.de
#
# Copyright (C) 2008-2012 ObsPy Development Team
# -----------------------------------------------
"""
ObsPy wrapper to the *Moment tensor Plotting and Decomposition tool* (MoPaD)
written by Lars Krieger and Sebastian Heimann.
.. seealso:: [Krieger2012]_
.. warning:: The MoPaD wrapper does not yet provide the full functionality of
MoPaD. Please consider using the command line script ``obspy-mopad`` for
now if you need the full power of MoPaD.
:copyright:
The ObsPy Development Team (devs@obspy.org)
:license:
GNU Lesser General Public License, Version 3
(https://www.gnu.org/copyleft/lesser.html)
"""
import numpy as np
from obspy.imaging.beachball import xy2patch
from obspy.imaging.scripts.mopad import BeachBall as mopad_BeachBall
from obspy.imaging.scripts.mopad import MomentTensor as mopad_MomentTensor
from obspy.imaging.scripts.mopad import epsilon
# seems the base system we (gmt) are using is called "USE" in mopad
KWARG_MAP = {
'size': ['plot_size', 'plot_aux_plot_size'],
'linewidth': ['plot_nodalline_width', 'plot_outerline_width'],
'facecolor': ['plot_tension_colour'],
'edgecolor': ['plot_outerline_colour'],
'bgcolor': [],
'alpha': ['plot_total_alpha'],
'width': [],
'outfile': ['plot_outfile'],
'format': ['plot_outfile_format'],
'nofill': ['plot_only_lines']
}
[docs]
def _normalize_focmec(fm):
"""
Improve stability of plots by normalizing the moment tensors. The scale
does not matter for the beachballs.
"""
# Only normalize 6 component tensors.
if len(fm) != 6:
return fm
fm = np.array(fm, dtype=np.float64)
fm /= np.linalg.norm(fm)
return fm
[docs]
def beach(fm, linewidth=2, facecolor='b', bgcolor='w', edgecolor='k',
alpha=1.0, xy=(0, 0), width=200, size=100, nofill=False,
zorder=100, mopad_basis='USE', axes=None):
"""
Return a beach ball as a collection which can be connected to an
current matplotlib axes instance (ax.add_collection). Based on MoPaD.
S1, D1, and R1, the strike, dip and rake of one of the focal planes, can
be vectors of multiple focal mechanisms.
:param fm: Focal mechanism that is either number of mechanisms (NM) by 3
(strike, dip, and rake) or NM x 6 (M11, M22, M33, M12, M13, M23 - the
six independent components of the moment tensor, where the coordinate
system is 1,2,3 = Up,South,East which equals r,theta,phi -
Harvard/Global CMT convention). The relation to Aki and Richards
x,y,z equals North,East,Down convention is as follows: Mrr=Mzz,
Mtt=Mxx, Mpp=Myy, Mrt=Mxz, Mrp=-Myz, Mtp=-Mxy.
The strike is of the first plane, clockwise relative to north.
The dip is of the first plane, defined clockwise and perpendicular to
strike, relative to horizontal such that 0 is horizontal and 90 is
vertical. The rake is of the first focal plane solution. 90 moves the
hanging wall up-dip (thrust), 0 moves it in the strike direction
(left-lateral), -90 moves it down-dip (normal), and 180 moves it
opposite to strike (right-lateral).
:param facecolor: Color to use for quadrants of tension; can be a string,
e.g. ``'r'``, ``'b'`` or three component color vector, [R G B].
Defaults to ``'b'`` (blue).
:param bgcolor: The background color. Defaults to ``'w'`` (white).
:param edgecolor: Color of the edges. Defaults to ``'k'`` (black).
:param alpha: The alpha level of the beach ball. Defaults to ``1.0``
(opaque).
:param xy: Origin position of the beach ball as tuple. Defaults to
``(0, 0)``.
:type width: int
:param width: Symbol size of beach ball. Defaults to ``200``.
:param size: Controls the number of interpolation points for the
curves. Minimum is automatically set to ``100``.
:param nofill: Do not fill the beach ball, but only plot the planes.
:param zorder: Set zorder. Artists with lower zorder values are drawn
first.
:param mopad_basis: The basis system. Defaults to ``'USE'``. See the
`Supported Basis Systems`_ section below for a full list of supported
systems.
:type axes: :class:`matplotlib.axes.Axes`
:param axes: Used to make beach balls circular on non-scaled axes. Also
maintains the aspect ratio when resizing the figure. Will not add
the returned collection to the axes instance.
.. rubric:: _`Supported Basis Systems`
========= =================== =============================================
Short Basis vectors Usage
========= =================== =============================================
``'NED'`` North, East, Down Jost and Herrmann 1989
``'USE'`` Up, South, East Global CMT Catalog, Larson et al. 2010
``'XYZ'`` East, North, Up General formulation, Jost and Herrmann 1989
``'RT'`` Radial, Transverse, psmeca (GMT), Wessel and Smith 1999
Tangential
``'NWU'`` North, West, Up Stein and Wysession 2003
========= =================== =============================================
"""
import matplotlib.collections as mpl_collections
from matplotlib import patches, transforms
fm = _normalize_focmec(fm)
# initialize beachball
mt = mopad_MomentTensor(fm, system=mopad_basis)
bb = mopad_BeachBall(mt, npoints=size)
bb._setup_BB(unit_circle=False)
# extract the coordinates and colors of the lines
radius = width / 2.0
neg_nodalline = bb._nodalline_negative_final_US
pos_nodalline = bb._nodalline_positive_final_US
tension_colour = facecolor
pressure_colour = bgcolor
if nofill:
tension_colour = 'none'
pressure_colour = 'none'
# based on mopads _setup_plot_US() function
# collect patches for the selection
coll = [None, None, None]
coll[0] = patches.Circle(xy, radius=radius)
coll[1] = xy2patch(neg_nodalline[0, :], neg_nodalline[1, :], radius, xy)
coll[2] = xy2patch(pos_nodalline[0, :], pos_nodalline[1, :], radius, xy)
# set the color of the three parts
fc = [None, None, None]
if bb._plot_clr_order > 0:
fc[0] = pressure_colour
fc[1] = tension_colour
fc[2] = tension_colour
if bb._plot_curve_in_curve != 0:
fc[0] = tension_colour
if bb._plot_curve_in_curve < 1:
fc[1] = pressure_colour
fc[2] = tension_colour
else:
coll = [coll[i] for i in (0, 2, 1)]
fc[1] = pressure_colour
fc[2] = tension_colour
else:
fc[0] = tension_colour
fc[1] = pressure_colour
fc[2] = pressure_colour
if bb._plot_curve_in_curve != 0:
fc[0] = pressure_colour
if bb._plot_curve_in_curve < 1:
fc[1] = tension_colour
fc[2] = pressure_colour
else:
coll = [coll[i] for i in (0, 2, 1)]
fc[1] = tension_colour
fc[2] = pressure_colour
if bb._pure_isotropic:
if abs(np.trace(bb._M)) > epsilon:
# use the circle as the most upper layer
coll = [coll[0]]
if bb._plot_clr_order < 0:
fc = [tension_colour]
else:
fc = [pressure_colour]
# transform the patches to a path collection and set
# the appropriate attributes
collection = mpl_collections.PatchCollection(coll, match_original=False)
collection.set_facecolors(fc)
# Use the given axes to maintain the aspect ratio of beachballs on figure
# resize.
if axes is not None:
# This is what holds the aspect ratio (but breaks the positioning)
collection.set_transform(transforms.Affine2D(np.identity(3)))
# Next is a dirty hack to fix the positioning:
# 1. Need to bring the all patches to the origin (0, 0).
for p in collection._paths:
p.vertices -= xy
# 2. Then use the offset property of the collection to position the
# patches
collection.set_offsets(xy)
try:
collection.set_offset_transform(axes.transData)
except AttributeError:
# compatibility for matplotlib 3.3 (and maybe 3.4 too?)
collection._transOffset = axes.transData
collection.set_edgecolors(edgecolor)
collection.set_alpha(alpha)
collection.set_linewidth(linewidth)
collection.set_zorder(zorder)
return collection
[docs]
def beachball(fm, linewidth=2, facecolor='b', bgcolor='w', edgecolor='k',
alpha=1.0, xy=(0, 0), width=200, size=100, nofill=False,
zorder=100, mopad_basis='USE', outfile=None, format=None,
fig=None):
"""
Draws a beach ball diagram of an earthquake focal mechanism. Based on
MoPaD.
S1, D1, and R1, the strike, dip and rake of one of the focal planes, can
be vectors of multiple focal mechanisms.
:param fm: Focal mechanism that is either number of mechanisms (NM) by 3
(strike, dip, and rake) or NM x 6 (M11, M22, M33, M12, M13, M23 - the
six independent components of the moment tensor, where the coordinate
system is 1,2,3 = Up,South,East which equals r,theta,phi). The strike
is of the first plane, clockwise relative to north.
The dip is of the first plane, defined clockwise and perpendicular to
strike, relative to horizontal such that 0 is horizontal and 90 is
vertical. The rake is of the first focal plane solution. 90 moves the
hanging wall up-dip (thrust), 0 moves it in the strike direction
(left-lateral), -90 moves it down-dip (normal), and 180 moves it
opposite to strike (right-lateral).
:param facecolor: Color to use for quadrants of tension; can be a string,
e.g. ``'r'``, ``'b'`` or three component color vector, [R G B].
Defaults to ``'b'`` (blue).
:param bgcolor: The background color. Defaults to ``'w'`` (white).
:param edgecolor: Color of the edges. Defaults to ``'k'`` (black).
:param alpha: The alpha level of the beach ball. Defaults to ``1.0``
(opaque).
:param xy: Origin position of the beach ball as tuple. Defaults to
``(0, 0)``.
:type width: int
:param width: Symbol size of beach ball. Defaults to ``200``.
:param size: Controls the number of interpolation points for the
curves. Minimum is automatically set to ``100``.
:param nofill: Do not fill the beach ball, but only plot the planes.
:param zorder: Set zorder. Artists with lower zorder values are drawn
first.
:param mopad_basis: The basis system. Defaults to ``'USE'``. See the
`Supported Basis Systems`_ section below for a full list of supported
systems.
:param outfile: Output file string. Also used to automatically
determine the output format. Supported file formats depend on your
matplotlib backend. Most backends support png, pdf, ps, eps and
svg. Defaults to ``None``.
:param format: Format of the graph picture. If no format is given the
outfile parameter will be used to try to automatically determine
the output format. If no format is found it defaults to png output.
If no outfile is specified but a format is, than a binary
imagestring will be returned.
Defaults to ``None``.
:param fig: Give an existing figure instance to plot into. New Figure if
set to ``None``.
.. rubric:: _`Supported Basis Systems`
========= =================== =============================================
Short Basis vectors Usage
========= =================== =============================================
``'NED'`` North, East, Down Jost and Herrmann 1989
``'USE'`` Up, South, East Global CMT Catalog, Larson et al. 2010
``'XYZ'`` East, North, Up General formulation, Jost and Herrmann 1989
``'RT'`` Radial, Transverse, psmeca (GMT), Wessel and Smith 1999
Tangential
``'NWU'`` North, West, Up Stein and Wysession 2003
========= =================== =============================================
.. rubric:: Examples
(1) Using basis system ``'NED'``.
>>> from obspy.imaging.mopad_wrapper import beachball
>>> mt = [1, 2, 3, -4, -5, -10]
>>> beachball(mt, mopad_basis='NED') #doctest: +SKIP
.. plot::
from obspy.imaging.mopad_wrapper import beachball
mt = [1, 2, 3, -4, -5, -10]
beachball(mt, mopad_basis='NED')
"""
fm = _normalize_focmec(fm)
mopad_kwargs = {}
loc = locals()
# map to kwargs used in mopad
for key in KWARG_MAP:
value = loc[key]
for mopad_key in KWARG_MAP[key]:
mopad_kwargs[mopad_key] = value
# convert from points to size in cm
for key in ['plot_aux_plot_size', 'plot_size']:
# 100.0 is matplotlib's default DPI for savefig
mopad_kwargs[key] = mopad_kwargs[key] / 100.0 * 2.54
# use nofill kwarg
mt = mopad_MomentTensor(fm, system=mopad_basis)
bb = mopad_BeachBall(mt, npoints=size)
# show plot in a window
if outfile is None:
bb.ploBB(mopad_kwargs)
# save plot to file
else:
# no format specified, parse it from outfile name
if mopad_kwargs['plot_outfile_format'] is None:
mopad_kwargs['plot_outfile_format'] = \
mopad_kwargs['plot_outfile'].split(".")[-1]
else:
# append file format if not already at end of outfile
if not mopad_kwargs['plot_outfile'].endswith(
mopad_kwargs['plot_outfile_format']):
mopad_kwargs['plot_outfile'] += "." + \
mopad_kwargs['plot_outfile_format']
bb.save_BB(mopad_kwargs)