Plotting data and models with SlimPlotting
title: Overview of SlimPlotting utilities author: Mathias Louboutin date: April 2023 –-
This example script is written using Weave.jl and can be converted to different format for documentation and usage This example is converted to a markdown file for the documentation.
Import SlimPlotting, SegyIO to read seismic data, JLD2 for hdf5-like files
using PythonPlot, SlimPlotting, SegyIO, JLD2conda-forge/linux-64                                        Using cache
conda-forge/noarch                                          Using cache
Transaction
  Prefix: /home/runner/work/SlimPlotting.jl/SlimPlotting.jl/docs/.CondaPkg/
env
  Updating specs:
   - libstdcxx-ng[version='>=3.4,<13.0']
   - matplotlib[version='>=1']
   - openssl[version='>=3, <3.1']
   - conda-forge::python[version='>=3.8,<4',build=*cpython*]
   - uv[version='>=0.4']
  Package         Version  Build               Channel           Size
───────────────────────────────────────────────────────────────────────
  Install:
───────────────────────────────────────────────────────────────────────
  + uv              0.5.8  h0f3a69f_0          conda-forge       10MB
  Reinstall:
───────────────────────────────────────────────────────────────────────
  o python         3.11.0  he550d4f_1_cpython  conda-forge     Cached
  Upgrade:
───────────────────────────────────────────────────────────────────────
  - libgcc         12.4.0  h77fa898_1          conda-forge     Cached
  + libgcc         14.2.0  h77fa898_1          conda-forge      849kB
  - libgomp        12.4.0  h77fa898_1          conda-forge     Cached
  + libgomp        14.2.0  h77fa898_1          conda-forge      461kB
  - libstdcxx      12.4.0  hc0a3c3a_1          conda-forge     Cached
  + libstdcxx      14.2.0  hc0a3c3a_1          conda-forge        4MB
  - libgcc-ng      12.4.0  h69a702a_1          conda-forge     Cached
  + libgcc-ng      14.2.0  h69a702a_1          conda-forge       54kB
  Downgrade:
───────────────────────────────────────────────────────────────────────
  - libstdcxx-ng   12.4.0  h4852527_1          conda-forge     Cached
  + libstdcxx-ng   12.3.0  hc0a3c3a_7          conda-forge        3MB
  Summary:
  Install: 1 packages
  Reinstall: 1 packages
  Upgrade: 4 packages
  Downgrade: 1 packages
  Total download: 19MB
───────────────────────────────────────────────────────────────────────
Transaction starting
Changing libstdcxx-ng-12.4.0-h4852527_1 ==> libstdcxx-ng-12.3.0-hc0a3c3a_7
Changing libgcc-12.4.0-h77fa898_1 ==> libgcc-14.2.0-h77fa898_1
Changing libgomp-12.4.0-h77fa898_1 ==> libgomp-14.2.0-h77fa898_1
Changing libstdcxx-12.4.0-hc0a3c3a_1 ==> libstdcxx-14.2.0-hc0a3c3a_1
Changing libgcc-ng-12.4.0-h69a702a_1 ==> libgcc-ng-14.2.0-h69a702a_1
Linking uv-0.5.8-h0f3a69f_0
Reinstalling python-3.11.0-he550d4f_1_cpython
Transaction finished
To activate this environment, use:
    micromamba activate /home/runner/work/SlimPlotting.jl/SlimPlotting.jl/d
ocs/.CondaPkg/env
Or to execute a single command in this environment, use:
    micromamba run -p /home/runner/work/SlimPlotting.jl/SlimPlotting.jl/doc
s/.CondaPkg/env mycommandInitialize all needed data
Close all figures if any existing
plotclose("all")Python: NonePath to the files and data used for these examples
data_path = dirname(pathof(SlimPlotting)) * "/../data/";Read the data
# Pure array
vp = Float32.(segy_read("$(data_path)2dVP.sgy").data);
dm = diff(vp, dims = 1);
shot = Float32.(segy_read("$(data_path)2dshot.segy").data);
xloc = get_header(segy_read("$(data_path)2dshot.segy"), "GroupX")
fslice = JLD2.load("$(data_path)2dfslice.jld");Create structures to mimic JUDI-like inputs
In the future this should be instead converted into an extension rather than implicit knowledge of the structure
# Dummy structures to check plot with metadata
struct geometry
    xloc::Any
end
struct shotrec
    data::Any
    dt::Any
    geometry::Any
end
struct Phys
    data::Any
    d::Any
end
## Make physical objects
dmp = Phys(dm, (10, 20))
vpp = Phys(vp, (10, 20))
fslicep = Phys(fslice["Freq"][1, :, :], (12.5, 12.5))
shotp = shotrec([shot], 0.008, geometry([xloc]));Model perturbation
We plot here a model perturbation (i.e a Reverse-time Migrated image) and compare a few colormaps:
- The perceptually accurate Greyscolormap from colorcet
- The standard matplotlib Greyscolormap
- Another perceptually accurate Greyscolormap from colorcet
figure(figsize = (10, 10))
subplot(311)
plot_simage(dmp; new_fig = false, name = "Colorcet Greys")
subplot(312)
plot_simage(dm, (10, 20); cmap = "Greys", new_fig = false, name = "Greys")
subplot(313)
plot_simage(dm, (10, 20); cmap = :cet_CET_L2, new_fig = false, name = "Colorcet Greys 2")
tight_layout();
display(gcf());
Velocity
We plot here a velocity model and compare a few colormaps:
- The perceptually accurate jetcolormap from colorcet namedcet_rainbow4
- The ColorSchemes vikcolormap
- The matplotlib jet
figure(figsize = (10, 10))
subplot(311)
plot_velocity(vpp; new_fig = false, name = "colorcet jet", cmap = "cet_rainbow4")
subplot(312)
plot_velocity(vp, (10, 20); cmap = :vik, new_fig = false, name = "ColorSchemes's vik")
subplot(313)
plot_velocity(vp, (10, 20); cmap = "jet", new_fig = false, name = "matplotlib jet")
tight_layout();
display(gcf());
Frequency slice
We plot here a frequency slice for a seismic dataset and compare a few colormaps:
- The perceptually accurate bwrcolormap from colorcet namedcet_CET_D1A
- The standard matplotlib bwrcolormap
- The matplotlib "bwr
# Frequency slice
figure(figsize = (10, 5))
subplot(131)
plot_fslice(fslice["Freq"][1, :, :], (12.5, 12.5); new_fig = false, name = "colorcet bwr")
subplot(132)
plot_fslice(fslicep; cmap = :bwr, new_fig = false, name = "bwr")
subplot(133)
plot_fslice(fslicep; cmap = "bwr", new_fig = false, name = "Matplotlib bwr")
tight_layout();
display(gcf());
Shot record
Seismic blue-white-red
We plot here a frequency slice for a seismic dataset and compare a few colormaps for the bwr colormap:
- The standard matplotlib bwrcolormap
- The perceptually accurate bwrcolormap from colorcet namedcet_CET_D1Aandcet_CET_D1
# Shot record
figure(figsize = (10, 5))
subplot(131)
plot_sdata(shotp; new_fig = false, name = "matplotlib seismic", cmap = "bwr")
subplot(132)
plot_sdata(shot, (12.5, 0.008); cmap = :cet_CET_D1A, new_fig = false, name = "Colorcet bwr")
subplot(133)
plot_sdata(shot, (12.5, 0.008); cmap = :cet_CET_D1, new_fig = false, name = "Colorcet bwr 2")
tight_layout();
display(gcf());
Seismic greys
We plot here a frequency slice for a seismic dataset and compare a few colormaps for the greys colormap:
- The standard matplotlib graycolormap
- The perceptually accurate greyscolormap from colorcet namedcet_CET_L1
# Shot record
figure(figsize = (10, 5))
subplot(121)
plot_sdata(shotp; new_fig = false, name = "colorcet gray", cmap = "cet_CET_L1")
subplot(122)
plot_sdata(shot, (12.5, 0.008); cmap = "gray", new_fig = false, name = "Greys")
tight_layout();
display(gcf());
Compare shot records
One of the main visual representation of FWI inversion is to compare the true shot record with the synthetic data from the current velocity model. A good way to visualize this difference is to overlay the two shot records alternating the traces between each shots with a different colormap to check the alignment of the events. We show below how to do this with the  compare_shots function
figure(figsize = (10, 5))
subplot(131)
compare_shots(shotp, shotp; new_fig = false, name = "Overlap compare")
subplot(132)
compare_shots(
    shotp,
    shotp;
    new_fig = false,
    cmap = ("bwr", "RdBu"),
    name = "Overlap compare custom cmap",
)
subplot(133)
compare_shots(
    shotp,
    shotp;
    side_by_side = true,
    new_fig = false,
    name = "Side by side compare",
)
tight_layout();
display(gcf());
Wiggle traces
We finally show the traditional wiggle plot for a shot record used in seismic.
# Wiggle plot
figure(figsize=(5, 5))
subplot(111)
wiggle_plot(shot[1:5:end, 1:10:end], xloc[1:10:end], 0:0.02:4.6; new_fig=false)
tight_layout();
display(gcf())