Skip to content

Commit

Permalink
Add insets tutorial
Browse files Browse the repository at this point in the history
  • Loading branch information
joa-quim committed Jun 15, 2024
1 parent b4a0abc commit 8ce28aa
Show file tree
Hide file tree
Showing 8 changed files with 242 additions and 9 deletions.
2 changes: 0 additions & 2 deletions documentation/modules/basemap.md
Original file line number Diff line number Diff line change
Expand Up @@ -18,8 +18,6 @@ At least one of the options \myreflink{frame}, **map_scale**, or **rose** must b

\textinput{common_opts/opt_R}

\textinput{common_opts/opt_R_3D}


- **A** or **polygon** : *polygon=true* or *polygon=fname*\
No plotting is performed. Instead, we determine the geographical coordinates of the polygon outline
Expand Down
9 changes: 8 additions & 1 deletion documentation/modules/colorbar.md
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,14 @@ Optional Arguments

- **B** or **axes** or **frame**\
Set annotation, tick, and gridline interval for the colorbar. More at \myreflink{frame}
The x-axis label will plot beneath a horizontal bar (or vertically to the right of a vertical bar), except when using the **move\_annot** modifier of the **pos** option. As an option, use the y-axis label to plot the data unit to the right of a horizontal bar (and above a vertical bar). When using **frame=(annot=:auto,)** or **frame=(annot=:auto,ticks=:auto)** annotation and/or minor tick intervals are chosen automatically. If **frame** is omitted, or no annotation intervals are provided, the default is to annotate every color level based on the numerical entries in the CPT (which may be overridden by ULB flags in the CPT). To specify custom text annotations for intervals, you must append ; *annotation* to each z-slice in the CPT.
The x-axis label will plot beneath a horizontal bar (or vertically to the right of a vertical bar), except when using
the **move\_annot** modifier of the **pos** option. As an option, use the y-axis label to plot the data unit to the right
of a horizontal bar (and above a vertical bar). When using **frame=(annot=:auto,)** or **frame=(annot=:auto,ticks=:auto)**
annotation and/or minor tick intervals are chosen automatically. If **frame** is omitted, or no annotation intervals
are provided, the default is to annotate every color level based on the numerical entries in the CPT (which may be
overridden by ULB flags in the CPT). The exception to this rule is for CPT files that were scaled to fit the range
of a grid exactly and thus have arbitrary color levels; these will trigger an automatic **frame=(annot=:auto,ticks=:auto)**
setting. To specify custom text annotations for intervals, you must append ; *annotation* to each z-slice in the CPT.

- **C** or **color** or **cmap** : *color=cpt*\
Note: If not given, the module will check if we have a valid *current\_cpt* stored in memory (*makecpt* does store it). If there is one, fine, otherwise **GMT** will error.
Expand Down
70 changes: 67 additions & 3 deletions documentation/modules/inset.md
Original file line number Diff line number Diff line change
Expand Up @@ -29,7 +29,7 @@ region and projection that will be used to draw in the inset, then give these ar
Required Arguments
------------------

- **D** or **inset** or **inset_box** or **insetbox** : -- *inset_box=(map=true, inside=true, outside=true, norm=true, paper=true, anchor=XX, size=XX, width=XX, justify=code, offset=XX)*\
- **D** or **pos** or **position**or **inset_box** or **insetbox** : -- *position=(map=true, inside=true, outside=true, norm=true, paper=true, anchor=XX, size=XX, width=XX, justify=code, offset=XX)*\
Define the map inset rectangle on the map. Specify the rectangle in one of three ways:

1. Use `map=(lon,lat)` for map coordinates. Requires both **region** and **proj** to be set.
Expand All @@ -43,7 +43,7 @@ Required Arguments
by using **justify=??** where *??* stands for a 2-char justification code *justify* (see \myreflink{text}).
Note: with the default **outside=true**, the *justify* defaults to the same as **anchor**, if **inside=true** is used then *justify* defaults to the mirror opposite of **anchor**. Specify inset box attributes via the **box** option [outline only].

Alternatively, use **inset_box="west/east/south/north"** of geographic rectangle bounded by parallels and
Alternatively, use **position="west/east/south/north"** of geographic rectangle bounded by parallels and
meridians; append **+r** if the coordinates instead are the lower left and upper right corners of the desired
rectangle. (Or, give *xmin/xmax/ymin/ymax* of bounding rectangle in projected coordinates and optionally
append **+u**unit [Default coordinate unit is meter (e)]. NOTE that this form requires passing the options
Expand Down Expand Up @@ -94,6 +94,47 @@ The **inset(:end)** command finalizes the current inset, which returns the plott
prior to the start of the inset. The previous region and map projection will be in effect going forward.


The nested call mode
--------------------

The options described above respect the _pure_ use of `inset` as a _modern_ mode function. But we can also use it
in a mix mode in _one-liners_ commands. It is mixed because the functioning relies in mixing the _classic_ and
_modern_ modes (in a way transparent to the user). And, as just said, this mix mode consists in calling the `inset`
function as an option to the `plot`, `basemap` and `grdimage` functions. Since we are doing a nested call, we need
to pass all options as argument to `inset` and this ofc reduces the number of possibilities but still, it offers
quite nice features that allow creating elaborated figures with very short commands. The \myreflink{Figure insets}
shows several examples of this usage.

The inset windows are located according to an algorithm that tries to avoid overlapping lines in line plots (with
a moderate success), or in some corner position for insets with images. Inset windows sizes are also automatically
estimated from image sizes and projections (when they are geographical). However, user can manually control this
wth **position** option explained above.

- **inset** -- *inset=(data, zoom=(...), coast=(...), position=(...), box=(...), clearance=(...))*\

- **data** An image, a grid a table (GMTdataset) or a file name that can be automatically read by
\myreflink{gmtread}. Depending on the data type an _x,y_ plot or an image is displayed inside the inset window.

- **zoom** This refers to an area of the main window that we wish to make a zoom of. Its arguments depend
on whether we are zooming an _x,y_ plot or an image. In the first case we pass an _x_ location and a half-width.
For example `zoom=(10,2)` means that the zoom window covers the abscissa [8 12] range. The _y_ extent is whatever
the line has between those _x_ limits. But for the case of images we need one more argument because now we have 2
dimensions. The syntax now is `zoom=(x0,y0,delta)`, an extension of the previous concept, which requires that the
units of _x_ and _y_ are the same and we get a square zoom window. The alternative is to use the usual way of GMT
specifying region limits. That is `zoom=(x_min, x_max, y_min, y_max)`

- **pzoom** When adding insets to images we can provide an external image (or file name) that will be displayed
in the inset. But since in this case the inset does not have to share the same coordinates with main figure, we
only provide here the point coordinates of the interesting area. The syntax is hence `pzoom=(x0,y0)`.

- **coast** With this argument we can call the \myreflink{coast} command with all of its normal arguments. And
furthermore, we can even call the \myreflink{plot} (or any of its avatars) to add line/symbol plots over the
inset map. A further option to this form of calling `coast` is the option `rect=?`. If `?` is `true`, it will
plot a 0.75 pt blue rectangle showing the main window limits. Alternatives is `rect=number`, `rect=color` or
`rect=(number, color)`, where `color` is a color name and `number` is the rectangle line thickness in points

- **position, box, clearance** Have the same meaning/usage as explained above.

Examples
--------

Expand All @@ -104,11 +145,34 @@ To make a simple basemap plot that demonstrates the inset module, try
using GMT
gmtbegin()
basemap(region=(0,40,20,60), proj=:merc, frame=(annot=:afg, fill=:lightgreen))
inset(inset_box=(anchor=:TR, width=6.4, offset=0.5), box=(fill=:pink, pen=0.5), margins=0.6)
inset(position=(anchor=:TR, width=6.4, offset=0.5), box=(fill=:pink, pen=0.5), margins=0.6)
basemap(region=:global360, proj=(name=:laea, center=[20,20]), figsize=5, frame=:afg)
text(text="INSET", font=18, region_justify=:TR, offset=(away=true, shift=-0.4), noclip=true)
inset(:end)
text(text="MAP", font=18, region_justify=:BL, offset=(away=true, shift=0.5))
gmtend(:show)
```
\end{examplefig}

Make a zoom over a region of a synthetic plot.

\begin{examplefig}{}
```julia
using GMT

t = 0:0.01:2pi;
plot(t, cos.(t).+rand(length(t))*0.1, inset=(zoom=(pi,pi/4), box=(fill=:lightblue,)), show=true)
```
\end{examplefig}


Add an inset to basemap image with a rectangle in the inset taken from main image limits.

\begin{examplefig}{}
```julia
using GMT

basemap(region=(-48,-43,-26,-20), J=:merc,
inset=(coast, R="-80/-28/-43/10", J=:merc, shore=true, rect=(2,:red)), show=true)
```
\end{examplefig}
6 changes: 3 additions & 3 deletions gallery/ex45.md
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ using GMT
resetGMT() # hide

# Basic LS line y = a + bx
model = trend1d("@MaunaLoa_CO2.txt", output=:xm, n_model=:p1)
model = trend1d("@MaunaLoa_CO2.txt", output=:xm, model=:p1)
plot("@MaunaLoa_CO2.txt", region=(1958,2016,310,410), frame=(axes=:WSen, bg=:azure1),
xaxis=(annot=:auto, ticks=:auto), yaxis=(annot=:auto, ticks=:auto, suffix=" ppm"),
marker=:circle, ms=0.05, fill=:red, figsize=(15,5), xshift=4)
Expand All @@ -18,14 +18,14 @@ text!(mat2ds("m@-2@-(t) = a + b@~\\327@~t"), font=12, region_justify=:TL,
offset=(away=true, shift=0.25), fill=:lightyellow)

# Basic LS line y = a + bx + cx^2
model = trend1d("@MaunaLoa_CO2.txt", output=:xm, n_model=:p2)
model = trend1d("@MaunaLoa_CO2.txt", output=:xm, model=:p2)
plot!("@MaunaLoa_CO2.txt", frame=:same, ms=0.05, fill=:red, yshift=6)
plot!(model, pen=(0.5,:blue))
text!(mat2ds("m@-3@-(t) = a + b@~\\327@~t + c@~\\327@~t@+2@+"), font=12,
region_justify=:TL, offset=(away=true, shift=0.25), fill=:lightyellow)

# Basic LS line y = a + bx + cx^2 + seasonal change
model = trend1d("@MaunaLoa_CO2.txt", output=:xmr, n_model="p2,f1+o1958+l1")
model = trend1d("@MaunaLoa_CO2.txt", output=:xmr, model="p2,f1+o1958+l1")
plot!("@MaunaLoa_CO2.txt", frame=:same, ms=0.05, fill=:red, yshift=6)
plot!(model, pen=(0.25,:blue))
text!(mat2ds("m@-5@-(t) = a + b@~\\327@~t + c@~\\327@~t@+2@+ + d@~\\327@~cos(2@~p@~t) + e@~\\327@~sin(2@~p@~t)"),
Expand Down
11 changes: 11 additions & 0 deletions tutorials.md
Original file line number Diff line number Diff line change
Expand Up @@ -100,4 +100,15 @@
~~~</a>~~~
@@

@@box
~~~<a class="boxlink" href="insets/insets/">~~~
@@title Insets in figures@@
@@box-content
~~~
<img src="/tutorials/insets/tilelogo.jpg">
~~~
@@
~~~</a>~~~
@@

@@
3 changes: 3 additions & 0 deletions tutorials/insets.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
# Figure insets

{{list_folder_with_images insets}}
150 changes: 150 additions & 0 deletions tutorials/insets/insets.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,150 @@
# Figure insets

The \myreflink{inset} function can be used as an option to the `plot`, `basemap` and `grdimage`
functions. While not all the possibilities offered by the `inset` function are available in this case,
because all functionalities must be requested via a nested call, it offers quite nice features that
allow creating elaborated figures with very short commands.

First example shows how to make a zoom over a region of a synthetic plot. Notice how we use here
the `inset` command as if it was just another option of the `plot` function. The location of the inset
window is found with a heuristic that tries to place them above the zoom window (to avoid overlapping
the plotted line) or in one corner when previous condition is not possible.

\begin{examplefig}{}
```julia
using GMT

t = 0:0.01:2pi;
plot(t, cos.(t).+rand(length(t))*0.1, inset=(zoom=(pi,pi/4), box=(fill=:lightblue,)), show=true)
```
\end{examplefig}

If we select the zooming interval over a region where the zoom window has a different aspect ratio
(such that as in this case it would overlap the inset located above it) the inset location jumps
to another location. In case of need the inset size is scaled down such that it doesn't go out of
the main figure dimensions. But all of this can be manually controlled by using the `pos` or `position`
sub-options.

\begin{examplefig}{}
```julia
using GMT

t = 0:0.01:2pi;
plot(t, cos.(t).+rand(length(t))*0.1, inset=(zoom=(pi/2,pi/6), box=(fill=:lightblue,)), show=true)
```
\end{examplefig}


We can make zooms on more than one line but for it to work we must plot them as a multi-column dataset.
That is, we must plot all the lines in one command like the next example. Lines existing in the figure
resulting from previous plot commands (case in which we should have used the `plot!` form) are ignored.

\begin{examplefig}{}
```julia
using GMT

t = 0:0.1:2pi;
plot([t cos.(t) cos.(t.+0.1)], multi=true, inset=(zoom=(pi,pi/4), box=(fill=:lightblue,)), show=true)
```
\end{examplefig}


If we want to plot another curve instead of zooming a part of the one in the main figure we pass
the data of that second curve as argument to the `inset` option. Here, for simplicity, we will
show a sinusoid over the same domain.


\begin{examplefig}{}
```julia
using GMT

t = 0:0.1:2pi;
plot(t, cos.(t), inset=([t sin.(t)]), lc=:red, ls=:dash, show=true)
```
\end{examplefig}


In the above examples, selecting the zoom coordinates was simple but when the _xx_ axis represents time,
things get a bit more complicated because time is in fact stored as a number of seconds since a reference
epoch. And there are _lots_ of seconds in an interval. So, when _xx_ axis has time, we must provide the zooming
interval in a date or date-time format. Currently the program accepts time in the ISO "yyyy-mm-ddTHH:MM:SS",
"dd-o-yyyy" or "dd-o-yy" formats. When using the ISO format the _time_, that is the part that includes and
follows the `T....`, is optional


\begin{examplefig}{}
```julia
using GMT

D = gmtread("@RHAT_price.csv", o="0,2", par=(FORMAT_DATE_IN="dd-o-yy",))
viz(D, inset=(zoom=("2004-01-01","2005-01-01"),))
```
\end{examplefig}


For an inset with a rectangle showing the zone that is displayed in main figure, use the `rect` (or `rectangle`)
option. Arguments in this example mean a line thickness of 2 points and a red line color. `rect=true` would
have plotted a thinner (0.75 pt) blue rectangle.

\begin{examplefig}{}
```julia
using GMT

G = grdcut("@earth_relief_04m", R="-48/-43/-26/-20");
viz(G, cmap=:world, J=:merc, inset=(coast, R="-80/-28/-43/10", J=:merc, shore=true, ocean=:blue, rect=(2,:red)))
```
\end{examplefig}


But if instead of a rectangle we want to plot a symbol, let’s say, a red circle, we can call the `plot` function
inside `inset`. In next example we also ask to plot the inset at bottom right (:BR) instead of the default
top right (:TR) default position. If wished, things can be made more complex by passing a full featured GMTdataset
to the `plot` function. _e.g._, `plot=(data=D, ...)`

\begin{examplefig}{}
```julia
using GMT

G = grdcut("@earth_relief_04m", R="-48/-43/-26/-20");
viz(G, J=:merc,
inset=(coast, R="-80/-28/-43/10", J=:merc, shore=true, pos=(anchor=:BR,),
plot=(data=[-45.5 -23], marker=:circ, fill=:red))
)
```
\end{examplefig}


Next examples show how we can zoom out a region of an image and show it in the inset. We will use
as example the image containing the "Finger of God" Bok globule in the Carina Nebula. We download
the images directly from the Wikipedia site. The first example lets the inset location algorithm
choose where to put the inset window. In the second we decide that ourselves.

\begin{examplefig}{}
```julia
using GMT

im = gmtread("https://upload.wikimedia.org/wikipedia/commons/thumb/f/f8/NGC_3372a-full.jpg/2560px-NGC_3372a-full.jpg")
viz(im, inset=(zoom=(910,980,350,430),))
```
\end{examplefig}


Hmm, the above figure is nice, but the finger is a bit fuzzy. The issue is the image is somewhat low resolution.
The Wikipedia page has a high-resolution image but it's a 200 MB jpeg compressed image. Too big to use in these
examples. But we can download only the _finger_ and add it as the inset. In this case we don't have a rectangle
to zoom out (because we are using a different image in the inset) but still want to indicate what zone of the
base image is shown in higher resolution. For that, we use the `pzoom` option (that stands for _pseudo-zoom_),
which expects a _x,y_ location only. Since these images are not referenced, we use _rows_ and _column_ numbers
(counting from lower left corner) as coordinates.


\begin{examplefig}{}
```julia
using GMT

im = gmtread("https://upload.wikimedia.org/wikipedia/commons/thumb/f/f8/NGC_3372a-full.jpg/2560px-NGC_3372a-full.jpg")
dedo = gdaltranslate("https://upload.wikimedia.org/wikipedia/commons/f/f8/NGC_3372a-full.jpg", srcwin=1,
limits=(10580, 11360, 9410, 10270));
viz(im, inset=(dedo, pzoom=(945, 390), pos=(anchor=:MR,)))
```
\end{examplefig}
Binary file added tutorials/insets/tilelogo.jpg
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.

0 comments on commit 8ce28aa

Please sign in to comment.