Plotting with Makie.jl
Introduction
Seis comes with two kinds of plotting functionality. The second, experimental kind relies on Makie. To create plots with Seis and Makie.jl, first install one of the Makie backends (see below):
import Pkg
Pkg.add("GLMakie")
Whenever you wish to create plots, do e.g.
julia> using Seis, GLMakie
The other Plots-based function is described here.
Makie backends
The Makie plotting ecosystem relies on the user loading one or more different 'backend' packages which implement the plotting commands in different ways. The Makie backend documentation describes the different choices and how to switch between them. For the purposes of this documentation, however, we will consider two backends:
- GLMakie: Allows interactive plotting; can only save bitmap figures
- CairoMakie: Non-interactive; can save vector graphics for publications and presentation
In your project which uses Seis.jl, you should add then import one or more of these as shown above.
The figures produced here were made using CairoMakie unless otherwise noted.
In the documentation below, wherever you see Makie
, this represents whichever Makie backend module you have loaded. In some cases, Seis.jl extends Makie functions with methods for Seis.jl objects
Plotting functions
New figure or existing axis?
The Makie plotting functions mostly come in two flavours—one which adds a plot into an existing axis (and ends with !
, such as plot_section!
), and another without a !
which creates a returns a new Makie.Figure
object, such as plot_section
. We discuss the new-figure versions below.
Makie.plot
: Default plot
Seis.jl extends Makie.plot
to produce the default kind of plot for whatever Seis.jl types are passed to it. At present, it will forward to plot_traces
for single or array of AbstractTrace
s.
plot_traces
: Single-trace plots
plot_traces
creates a set of single-trace plots where the independent (x) axis is time, and the dependent (y) axis is the value of the trace, similar to simply plotting trace(t)
against times(t)
.
However, plot_traces
when applied to a single Trace
or an AbstractArray{<:Trace}
offers options to scale plots relatively, show pick times, and so on.
(Note that there is no in-place version of plot_traces
because it requires several axes.)
Examples
1: A simple plot for a single trace, showing the picks.
using Seis, CairoMakie
t = sample_data()
plot_traces(t)

2: Plot the vertical components for an earthquake with the distance in km and station code labelled.
t = sample_data(:regional)
z = filter(x -> endswith(x.sta.cha, "z"), t)
labels = z.sta.sta .* ", ∆ = " .* string.(round.(distance_km.(z), digits=1)) .* "km"
plot_traces(z; label=labels, sort=:dist)

3: Same again, but showing the falloff in amplitude with distance by setting all plots to have the same scale with the ylims
option:
plot_traces(z; label=labels, sort=:dist, ylims=:all)

Full docstring
Seis.plot_traces
— Functionplot_traces
Function to plot several traces which can be used once you have loaded a Makie backend (e.g., via using GLMakie
, using CairoMakie
, etc.).
See docstrings below after loading a Makie backend for more information.
plot_section
: Record sections
section
plots a 'record section', or a set of traces where a trace's position on the y-axis is determined by some other information.
Typically record sections show traces against distance, but section
supports arbitrary values or functions to place the traces.
To align traces, pass a set of values to the align
keyword argument.
Examples
1. Simple distance record section
Using the radial components of a regional earthquake.
t = sample_data(:regional)
e, n, z = t[1:3:end], t[2:3:end], t[3:3:end]
rotate_to_gcp!.(e, n)
r, t = e, n
plot_section(r)

2. Normalised section
Same again, but with traces normalise
d so that the amplitudes are more similar.
plot_section(r .|> normalise)

3. Record section aligned on a set of picks
Let us use the picks stored with the key :A
to align our traces. Here we can manually set the plot limits by passing arguments via axis
, which are then passed to Makie.Axis
.
t = sample_data(:array)
plot_section(t; align=:A, axis=(limits=(-10, 20, nothing, nothing),))

4. Section, sorted by distance
We can plot traces sorted by epicentral distance, but equally spaced along the y-axis, where traces are normalize
d to show how well the peaks are aligned, with the trace amplitudes scaled down by half (using the zoom
keyword), and Makie keywords to set the y-axis label. This time we cut the traces relative to the :A
pick rather than setting plot limits.
ts = sort(t; by=distance_deg)
plot_section(cut.(ts, :A, -10, 10) .|> normalize, "index";
align=:A, zoom=0.5, axis=(ylabel="Trace index, increasing distance",))

5. Section aligned on predicted phase arrival time
If you have installed SeisTau.jl, then adding predicted travel times and aligning on these becomes easy.
Here we calculate predicted times for the PKIKP arrival using SeisTau
, called via add_picks!
.
using SeisTau
add_picks!.(t, "PKIKP")
plot_section(cut.(t, :PKIKP, -10, 10); align=:PKIKP)

Interactions
If you are using an interactive Makie backend like GLMakie, when the plot is active you can use the standard Makie interactions to change the plot:
- scroll to zoom
- click and drag to set a new view region
- right-click and drag to move the viewport
- hold down the
x
ory
key while scrolling or dragging to limit the change to be just horizontal or vertical, respectively - Ctrl-click to return the view to the original plot zoom and limits
In addition, with plot_section
you can press:
=
to increase the trace amplitude-
to decrease the trace amplitude
Downsampling
When scrolling and zooming with a plot_section
plot, you will also note that if there are too many data points to plot, the small text 'Downsampled' appears at the bottom right of the figure. plot_section
automatically reloads data into the window at either full resolution or downsampled to keep interactions quick. Use the keyword argument decimate=false
to turn off downsampling.
If using decimate=false
with very large datasets (many millions of points) plotting may become extremely slow or even unstable.
Full docstrings
Seis.plot_section!
— Functionplot_section!
Function to plot several traces as a function of distance or some other variable on the same axis. This can be used once you have loaded a Makie backend (e.g., via using GLMakie
, using CairoMakie
, etc.).
See docstrings below after loading a Makie backend for more information.
Seis.plot_section
— Functionplot_section
Function to plot several traces as a function of distance or some other variable on the same axis. This can be used once you have loaded a Makie backend (e.g., via using GLMakie
, using CairoMakie
, etc.).
See docstrings below after loading a Makie backend for more information.
plot_hodogram
: Particle motion plots
Hododgrams, or particle motion plots, are parametric plots of two or three components of motion through time. These can be plotted in Seis using plot_hodogram
.
Examples
Hodogram of the horizontal components of an earthquake, windowed around the P-wave arrival, showing the backazimuth to the event location, and the particle motion and backazimuth lines set to have different colours.
t = e, n, z = sample_data(:regional)[1:3]
cut!.(t, 50, 70)
plot_hodogram(e, n; backazimuth=true, lines=(color=:blue,), backazimuth_lines=(color=:red, linestyle=:dash))

We can also create a 3D plot of the particle motion across all three components:
plot_hodogram(e, n, z)

Full docstring
Seis.plot_hodogram!
— Functionplot_hodogram!
Function to plot the particle motion of two (in 2D) or three (in 3D) traces. This can be used once you have loaded a Makie backed (e.g., via using GLMakie
, using CairoMakie
, etc.).
See docstrings below after loading a Makie backend for more information.
Seis.plot_hodogram
— Functionplot_hodogram
Function to plot the particle motion of two (in 2D) or three (in 3D) traces. This can be used once you have loaded a Makie backed (e.g., via using GLMakie
, using CairoMakie
, etc.).
See docstrings below after loading a Makie backend for more information.
Layouts
Makie has a powerful set of tools to create 'layouts'—multi-panel plots with multiple axes. (See the Makie layout tutorial) for more.)
Seis.jl's plotting functions compose well with Makie's layout capabilities. All plotting functions can take a Makie.GridPosition
or Makie.GridSubposition
which can be accessed by indexing into a Makie.Figure
object, like fig[1,2]
. Passing a grid position object into the plotting functions creates a new axis (or axes) and creates the plot at that point.
In the example below, we create a complex layout showing three components of ground motion recorded at ELK on the left, zoomed-in plots of the time window around the P wave on the top right, and a particle motion plot of the
t = e, n, z = sample_data(:regional)[1:3]
window = (55, 70)
t_cut = e_cut, n_cut, z_cut = cut.(t, window...)
# Build the figure object
fig = Makie.Figure()
# Create a new grid layout with `fig[1,1]`
axisplots = plot(fig[1,1], t)
# Label the time window
for (ax, pl) in axisplots
vspan!(ax, window...; alpha=0.5)
end
# Create a new grid position at the top right with `fig[1,2][1,1]`
axisplots_zoom = plot(fig[1,2][1,1], [e_cut, n_cut];
ylims=:all, lines=(;color=:blue))
# Add a title to the top trace plot.
ax_zoom, _ = axisplots_zoom[1]
ax_zoom.title = "P-wave arrival"
# Create the final grid position at the bottom right with `fig[1,2][2,1]`
plot_hodogram(fig[1,2][2,1], e_cut, n_cut;
backazimuth=true, axis=(;title="P-wave particle motion"))
fig
