5  What are transforms?

5.1 Motivation

In Part I of the book, we learned that our GeoTable representation of geospatial data provides the data access pattern of the DataFrame, a feature that is very convenient for data science. To recap, let’s consider the following geotable with four random variables:

N = 10000
a = [2randn(N÷2) .+ 6; randn(N÷2)]
b = [3randn(N÷2); 2randn(N÷2)]
c = randn(N)
d = c .+ 0.6randn(N)

table = (; a, b, c, d)

gt = georef(table, CartesianGrid(100, 100))
10000×5 GeoTable over 100×100 CartesianGrid
a b c d geometry
Continuous Continuous Continuous Continuous Quadrangle
[NoUnits] [NoUnits] [NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
6.54032 -0.0834871 -0.0540702 -0.110909 Quadrangle((x: 0.0 m, y: 0.0 m), ..., (x: 0.0 m, y: 1.0 m))
3.27624 1.03753 0.991226 0.955863 Quadrangle((x: 1.0 m, y: 0.0 m), ..., (x: 1.0 m, y: 1.0 m))
5.05776 3.30281 -1.10131 -2.07675 Quadrangle((x: 2.0 m, y: 0.0 m), ..., (x: 2.0 m, y: 1.0 m))
8.73168 3.77482 2.55862 3.10735 Quadrangle((x: 3.0 m, y: 0.0 m), ..., (x: 3.0 m, y: 1.0 m))
7.97004 -3.26781 -0.382793 -0.125731 Quadrangle((x: 4.0 m, y: 0.0 m), ..., (x: 4.0 m, y: 1.0 m))
5.56696 -0.430157 -1.29843 -0.857643 Quadrangle((x: 5.0 m, y: 0.0 m), ..., (x: 5.0 m, y: 1.0 m))
8.70301 -4.03265 0.847914 0.315138 Quadrangle((x: 6.0 m, y: 0.0 m), ..., (x: 6.0 m, y: 1.0 m))
6.85746 -1.53107 -0.439342 0.241203 Quadrangle((x: 7.0 m, y: 0.0 m), ..., (x: 7.0 m, y: 1.0 m))
2.93672 1.14326 -1.87917 -1.71375 Quadrangle((x: 8.0 m, y: 0.0 m), ..., (x: 8.0 m, y: 1.0 m))
5.19611 1.06549 -0.0499859 -0.283416 Quadrangle((x: 9.0 m, y: 0.0 m), ..., (x: 9.0 m, y: 1.0 m))

We can easily retrieve the “a” column of the geotable as a vector, and plot its histogram:

Mke.hist(gt.a, color = "gray80")
┌ Warning: Found `resolution` in the theme when creating a `Scene`. The `resolution` keyword for `Scene`s and `Figure`s has been deprecated. Use `Figure(; size = ...` or `Scene(; size = ...)` instead, which better reflects that this is a unitless size and not a pixel resolution. The key could also come from `set_theme!` calls or related theming functions.
└ @ Makie ~/.julia/packages/Makie/We6MY/src/scenes.jl:227

We can compute the cross-correlation between columns “a” and “b”:

cor(gt.a, gt.b)
-0.014604971746462575

And inspect bivariate distributions of the values of the geotable with PairPlots.jl by Thompson (2023):

using PairPlots

pairplot(values(gt))
┌ Warning: Found `resolution` in the theme when creating a `Scene`. The `resolution` keyword for `Scene`s and `Figure`s has been deprecated. Use `Figure(; size = ...` or `Scene(; size = ...)` instead, which better reflects that this is a unitless size and not a pixel resolution. The key could also come from `set_theme!` calls or related theming functions.
└ @ Makie ~/.julia/packages/Makie/We6MY/src/scenes.jl:227

This pattern is useful to answer geoscientific questions via marginal analysis (i.e. entire columns treated as measurements of a single random variable). However, the answers to many questions in geosciences depend on where the measurements were made.

Attempting to answer geoscientific questions with basic access to rows and columns can be very frustrating. In particular, this approach is prone to unintentional removal of geospatial information:

gt.a
10000-element Vector{Float64}:
  6.540315418024154
  3.2762372524069274
  5.057760528049224
  8.731683492287079
  7.970035271688275
  5.5669641549702185
  8.703012123999418
  6.857460874307793
  2.93671554332138
  5.196110196190582
  9.362841449058013
  6.57115655559497
  6.00114864121228
  ⋮
  0.005630923500863075
 -0.4542748260257103
 -0.2262826116737986
 -1.4005257370107842
  1.0654900684914104
 -0.4224004807827854
  0.03772155490762509
 -0.2811034344799006
 -0.3379979213170261
 -0.3365214001539591
 -0.4400644372890852
  0.4650158741545599
Note

Any script that is written in terms of direct column access has the potential to discard the special geometry column, and become unreadable very quickly with the use of auxiliary indices for rows.

We propose a new approach to geospatial data science with the concept of transforms, which we introduce in three classes with practical examples:

  1. Feature transforms
  2. Geometric transforms
  3. Geospatial transforms

5.2 Feature transforms

A feature transform is a function that takes the values of the geotable and produces a new set of values over the same geospatial domain. The framework provides over 30 such transforms, ranging from basic selection of columns, to data cleaning, to advanced multivariate statistical transforms.

5.2.1 Basic

Let’s start with two basic and important transforms, Select and Reject. The Select transform can be used to select columns of interest from a geotable:

gt |> Select("a", "b") # select columns "a" and "b"
10000×3 GeoTable over 100×100 CartesianGrid
a b geometry
Continuous Continuous Quadrangle
[NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
6.54032 -0.0834871 Quadrangle((x: 0.0 m, y: 0.0 m), ..., (x: 0.0 m, y: 1.0 m))
3.27624 1.03753 Quadrangle((x: 1.0 m, y: 0.0 m), ..., (x: 1.0 m, y: 1.0 m))
5.05776 3.30281 Quadrangle((x: 2.0 m, y: 0.0 m), ..., (x: 2.0 m, y: 1.0 m))
8.73168 3.77482 Quadrangle((x: 3.0 m, y: 0.0 m), ..., (x: 3.0 m, y: 1.0 m))
7.97004 -3.26781 Quadrangle((x: 4.0 m, y: 0.0 m), ..., (x: 4.0 m, y: 1.0 m))
5.56696 -0.430157 Quadrangle((x: 5.0 m, y: 0.0 m), ..., (x: 5.0 m, y: 1.0 m))
8.70301 -4.03265 Quadrangle((x: 6.0 m, y: 0.0 m), ..., (x: 6.0 m, y: 1.0 m))
6.85746 -1.53107 Quadrangle((x: 7.0 m, y: 0.0 m), ..., (x: 7.0 m, y: 1.0 m))
2.93672 1.14326 Quadrangle((x: 8.0 m, y: 0.0 m), ..., (x: 8.0 m, y: 1.0 m))
5.19611 1.06549 Quadrangle((x: 9.0 m, y: 0.0 m), ..., (x: 9.0 m, y: 1.0 m))

In the example above, we selected the columns “a” and “b” explicitly, but Select has various methods for more flexible column selection:

gt |> Select(1:3) # select columns 1 to 3
10000×4 GeoTable over 100×100 CartesianGrid
a b c geometry
Continuous Continuous Continuous Quadrangle
[NoUnits] [NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
6.54032 -0.0834871 -0.0540702 Quadrangle((x: 0.0 m, y: 0.0 m), ..., (x: 0.0 m, y: 1.0 m))
3.27624 1.03753 0.991226 Quadrangle((x: 1.0 m, y: 0.0 m), ..., (x: 1.0 m, y: 1.0 m))
5.05776 3.30281 -1.10131 Quadrangle((x: 2.0 m, y: 0.0 m), ..., (x: 2.0 m, y: 1.0 m))
8.73168 3.77482 2.55862 Quadrangle((x: 3.0 m, y: 0.0 m), ..., (x: 3.0 m, y: 1.0 m))
7.97004 -3.26781 -0.382793 Quadrangle((x: 4.0 m, y: 0.0 m), ..., (x: 4.0 m, y: 1.0 m))
5.56696 -0.430157 -1.29843 Quadrangle((x: 5.0 m, y: 0.0 m), ..., (x: 5.0 m, y: 1.0 m))
8.70301 -4.03265 0.847914 Quadrangle((x: 6.0 m, y: 0.0 m), ..., (x: 6.0 m, y: 1.0 m))
6.85746 -1.53107 -0.439342 Quadrangle((x: 7.0 m, y: 0.0 m), ..., (x: 7.0 m, y: 1.0 m))
2.93672 1.14326 -1.87917 Quadrangle((x: 8.0 m, y: 0.0 m), ..., (x: 8.0 m, y: 1.0 m))
5.19611 1.06549 -0.0499859 Quadrangle((x: 9.0 m, y: 0.0 m), ..., (x: 9.0 m, y: 1.0 m))
gt |> Select(r"[bcd]") # columns matching regular expression
10000×4 GeoTable over 100×100 CartesianGrid
b c d geometry
Continuous Continuous Continuous Quadrangle
[NoUnits] [NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
-0.0834871 -0.0540702 -0.110909 Quadrangle((x: 0.0 m, y: 0.0 m), ..., (x: 0.0 m, y: 1.0 m))
1.03753 0.991226 0.955863 Quadrangle((x: 1.0 m, y: 0.0 m), ..., (x: 1.0 m, y: 1.0 m))
3.30281 -1.10131 -2.07675 Quadrangle((x: 2.0 m, y: 0.0 m), ..., (x: 2.0 m, y: 1.0 m))
3.77482 2.55862 3.10735 Quadrangle((x: 3.0 m, y: 0.0 m), ..., (x: 3.0 m, y: 1.0 m))
-3.26781 -0.382793 -0.125731 Quadrangle((x: 4.0 m, y: 0.0 m), ..., (x: 4.0 m, y: 1.0 m))
-0.430157 -1.29843 -0.857643 Quadrangle((x: 5.0 m, y: 0.0 m), ..., (x: 5.0 m, y: 1.0 m))
-4.03265 0.847914 0.315138 Quadrangle((x: 6.0 m, y: 0.0 m), ..., (x: 6.0 m, y: 1.0 m))
-1.53107 -0.439342 0.241203 Quadrangle((x: 7.0 m, y: 0.0 m), ..., (x: 7.0 m, y: 1.0 m))
1.14326 -1.87917 -1.71375 Quadrangle((x: 8.0 m, y: 0.0 m), ..., (x: 8.0 m, y: 1.0 m))
1.06549 -0.0499859 -0.283416 Quadrangle((x: 9.0 m, y: 0.0 m), ..., (x: 9.0 m, y: 1.0 m))

A convenient method is also provided to select and rename columns:

gt |> Select("a" => "A", "b" => "B")
10000×3 GeoTable over 100×100 CartesianGrid
A B geometry
Continuous Continuous Quadrangle
[NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
6.54032 -0.0834871 Quadrangle((x: 0.0 m, y: 0.0 m), ..., (x: 0.0 m, y: 1.0 m))
3.27624 1.03753 Quadrangle((x: 1.0 m, y: 0.0 m), ..., (x: 1.0 m, y: 1.0 m))
5.05776 3.30281 Quadrangle((x: 2.0 m, y: 0.0 m), ..., (x: 2.0 m, y: 1.0 m))
8.73168 3.77482 Quadrangle((x: 3.0 m, y: 0.0 m), ..., (x: 3.0 m, y: 1.0 m))
7.97004 -3.26781 Quadrangle((x: 4.0 m, y: 0.0 m), ..., (x: 4.0 m, y: 1.0 m))
5.56696 -0.430157 Quadrangle((x: 5.0 m, y: 0.0 m), ..., (x: 5.0 m, y: 1.0 m))
8.70301 -4.03265 Quadrangle((x: 6.0 m, y: 0.0 m), ..., (x: 6.0 m, y: 1.0 m))
6.85746 -1.53107 Quadrangle((x: 7.0 m, y: 0.0 m), ..., (x: 7.0 m, y: 1.0 m))
2.93672 1.14326 Quadrangle((x: 8.0 m, y: 0.0 m), ..., (x: 8.0 m, y: 1.0 m))
5.19611 1.06549 Quadrangle((x: 9.0 m, y: 0.0 m), ..., (x: 9.0 m, y: 1.0 m))

The Reject transform can be used to reject columns from a geotable that are not relevant for a given analysis. It supports the same column specification of Select:

gt |> Reject("b") # reject column "b"
10000×4 GeoTable over 100×100 CartesianGrid
a c d geometry
Continuous Continuous Continuous Quadrangle
[NoUnits] [NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
6.54032 -0.0540702 -0.110909 Quadrangle((x: 0.0 m, y: 0.0 m), ..., (x: 0.0 m, y: 1.0 m))
3.27624 0.991226 0.955863 Quadrangle((x: 1.0 m, y: 0.0 m), ..., (x: 1.0 m, y: 1.0 m))
5.05776 -1.10131 -2.07675 Quadrangle((x: 2.0 m, y: 0.0 m), ..., (x: 2.0 m, y: 1.0 m))
8.73168 2.55862 3.10735 Quadrangle((x: 3.0 m, y: 0.0 m), ..., (x: 3.0 m, y: 1.0 m))
7.97004 -0.382793 -0.125731 Quadrangle((x: 4.0 m, y: 0.0 m), ..., (x: 4.0 m, y: 1.0 m))
5.56696 -1.29843 -0.857643 Quadrangle((x: 5.0 m, y: 0.0 m), ..., (x: 5.0 m, y: 1.0 m))
8.70301 0.847914 0.315138 Quadrangle((x: 6.0 m, y: 0.0 m), ..., (x: 6.0 m, y: 1.0 m))
6.85746 -0.439342 0.241203 Quadrangle((x: 7.0 m, y: 0.0 m), ..., (x: 7.0 m, y: 1.0 m))
2.93672 -1.87917 -1.71375 Quadrangle((x: 8.0 m, y: 0.0 m), ..., (x: 8.0 m, y: 1.0 m))
5.19611 -0.0499859 -0.283416 Quadrangle((x: 9.0 m, y: 0.0 m), ..., (x: 9.0 m, y: 1.0 m))
Note

Unlike direct column access, the Select and Reject transforms preserve geospatial information.

Tip for all users

The Select transform can be used in conjunction with the viewer to quickly visualize a specific variable:

gt |> Select("a") |> viewer
┌ Warning: Found `resolution` in the theme when creating a `Scene`. The `resolution` keyword for `Scene`s and `Figure`s has been deprecated. Use `Figure(; size = ...` or `Scene(; size = ...)` instead, which better reflects that this is a unitless size and not a pixel resolution. The key could also come from `set_theme!` calls or related theming functions.
└ @ Makie ~/.julia/packages/Makie/We6MY/src/scenes.jl:227

The Rename transform can be used to rename specific columns of a geotable. It preserves all other columns that are not part of the column specification:

gt |> Rename("a" => "A", "b" => "B")
10000×5 GeoTable over 100×100 CartesianGrid
A B c d geometry
Continuous Continuous Continuous Continuous Quadrangle
[NoUnits] [NoUnits] [NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
6.54032 -0.0834871 -0.0540702 -0.110909 Quadrangle((x: 0.0 m, y: 0.0 m), ..., (x: 0.0 m, y: 1.0 m))
3.27624 1.03753 0.991226 0.955863 Quadrangle((x: 1.0 m, y: 0.0 m), ..., (x: 1.0 m, y: 1.0 m))
5.05776 3.30281 -1.10131 -2.07675 Quadrangle((x: 2.0 m, y: 0.0 m), ..., (x: 2.0 m, y: 1.0 m))
8.73168 3.77482 2.55862 3.10735 Quadrangle((x: 3.0 m, y: 0.0 m), ..., (x: 3.0 m, y: 1.0 m))
7.97004 -3.26781 -0.382793 -0.125731 Quadrangle((x: 4.0 m, y: 0.0 m), ..., (x: 4.0 m, y: 1.0 m))
5.56696 -0.430157 -1.29843 -0.857643 Quadrangle((x: 5.0 m, y: 0.0 m), ..., (x: 5.0 m, y: 1.0 m))
8.70301 -4.03265 0.847914 0.315138 Quadrangle((x: 6.0 m, y: 0.0 m), ..., (x: 6.0 m, y: 1.0 m))
6.85746 -1.53107 -0.439342 0.241203 Quadrangle((x: 7.0 m, y: 0.0 m), ..., (x: 7.0 m, y: 1.0 m))
2.93672 1.14326 -1.87917 -1.71375 Quadrangle((x: 8.0 m, y: 0.0 m), ..., (x: 8.0 m, y: 1.0 m))
5.19611 1.06549 -0.0499859 -0.283416 Quadrangle((x: 9.0 m, y: 0.0 m), ..., (x: 9.0 m, y: 1.0 m))

The Identity transform can be used as a placeholder to forward the geotable without modifications to the next transform:

gt |> Identity()
10000×5 GeoTable over 100×100 CartesianGrid
a b c d geometry
Continuous Continuous Continuous Continuous Quadrangle
[NoUnits] [NoUnits] [NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
6.54032 -0.0834871 -0.0540702 -0.110909 Quadrangle((x: 0.0 m, y: 0.0 m), ..., (x: 0.0 m, y: 1.0 m))
3.27624 1.03753 0.991226 0.955863 Quadrangle((x: 1.0 m, y: 0.0 m), ..., (x: 1.0 m, y: 1.0 m))
5.05776 3.30281 -1.10131 -2.07675 Quadrangle((x: 2.0 m, y: 0.0 m), ..., (x: 2.0 m, y: 1.0 m))
8.73168 3.77482 2.55862 3.10735 Quadrangle((x: 3.0 m, y: 0.0 m), ..., (x: 3.0 m, y: 1.0 m))
7.97004 -3.26781 -0.382793 -0.125731 Quadrangle((x: 4.0 m, y: 0.0 m), ..., (x: 4.0 m, y: 1.0 m))
5.56696 -0.430157 -1.29843 -0.857643 Quadrangle((x: 5.0 m, y: 0.0 m), ..., (x: 5.0 m, y: 1.0 m))
8.70301 -4.03265 0.847914 0.315138 Quadrangle((x: 6.0 m, y: 0.0 m), ..., (x: 6.0 m, y: 1.0 m))
6.85746 -1.53107 -0.439342 0.241203 Quadrangle((x: 7.0 m, y: 0.0 m), ..., (x: 7.0 m, y: 1.0 m))
2.93672 1.14326 -1.87917 -1.71375 Quadrangle((x: 8.0 m, y: 0.0 m), ..., (x: 8.0 m, y: 1.0 m))
5.19611 1.06549 -0.0499859 -0.283416 Quadrangle((x: 9.0 m, y: 0.0 m), ..., (x: 9.0 m, y: 1.0 m))

The RowTable and ColTable transforms change the underlying table representation of the values of the geotable as discussed in the first chapter of the book:

rt = gt |> RowTable()
10000×5 GeoTable over 100×100 CartesianGrid
a b c d geometry
Continuous Continuous Continuous Continuous Quadrangle
[NoUnits] [NoUnits] [NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
6.54032 -0.0834871 -0.0540702 -0.110909 Quadrangle((x: 0.0 m, y: 0.0 m), ..., (x: 0.0 m, y: 1.0 m))
3.27624 1.03753 0.991226 0.955863 Quadrangle((x: 1.0 m, y: 0.0 m), ..., (x: 1.0 m, y: 1.0 m))
5.05776 3.30281 -1.10131 -2.07675 Quadrangle((x: 2.0 m, y: 0.0 m), ..., (x: 2.0 m, y: 1.0 m))
8.73168 3.77482 2.55862 3.10735 Quadrangle((x: 3.0 m, y: 0.0 m), ..., (x: 3.0 m, y: 1.0 m))
7.97004 -3.26781 -0.382793 -0.125731 Quadrangle((x: 4.0 m, y: 0.0 m), ..., (x: 4.0 m, y: 1.0 m))
5.56696 -0.430157 -1.29843 -0.857643 Quadrangle((x: 5.0 m, y: 0.0 m), ..., (x: 5.0 m, y: 1.0 m))
8.70301 -4.03265 0.847914 0.315138 Quadrangle((x: 6.0 m, y: 0.0 m), ..., (x: 6.0 m, y: 1.0 m))
6.85746 -1.53107 -0.439342 0.241203 Quadrangle((x: 7.0 m, y: 0.0 m), ..., (x: 7.0 m, y: 1.0 m))
2.93672 1.14326 -1.87917 -1.71375 Quadrangle((x: 8.0 m, y: 0.0 m), ..., (x: 8.0 m, y: 1.0 m))
5.19611 1.06549 -0.0499859 -0.283416 Quadrangle((x: 9.0 m, y: 0.0 m), ..., (x: 9.0 m, y: 1.0 m))
rt |> values |> typeof
Vector{@NamedTuple{a::Float64, b::Float64, c::Float64, d::Float64}} (alias for Array{@NamedTuple{a::Float64, b::Float64, c::Float64, d::Float64}, 1})

The Functional transform can be used to apply a function to columns of a geotable in place:

gt |> Functional(cos) |> values |> pairplot
┌ Warning: Found `resolution` in the theme when creating a `Scene`. The `resolution` keyword for `Scene`s and `Figure`s has been deprecated. Use `Figure(; size = ...` or `Scene(; size = ...)` instead, which better reflects that this is a unitless size and not a pixel resolution. The key could also come from `set_theme!` calls or related theming functions.
└ @ Makie ~/.julia/packages/Makie/We6MY/src/scenes.jl:227

gt |> Functional("a" => cos, "b" => sin) |> values |> pairplot
┌ Warning: Found `resolution` in the theme when creating a `Scene`. The `resolution` keyword for `Scene`s and `Figure`s has been deprecated. Use `Figure(; size = ...` or `Scene(; size = ...)` instead, which better reflects that this is a unitless size and not a pixel resolution. The key could also come from `set_theme!` calls or related theming functions.
└ @ Makie ~/.julia/packages/Makie/We6MY/src/scenes.jl:227

The Map transform can be used to create new columns from existing columns in the geotable. It takes a column specification, calls a function on the selected columns row-by-row, and returns the result as a new column:

gt |> Map("a" => sin, "b" => cos => "cos(b)")
10000×7 GeoTable over 100×100 CartesianGrid
a b c d sin_a cos(b) geometry
Continuous Continuous Continuous Continuous Continuous Continuous Quadrangle
[NoUnits] [NoUnits] [NoUnits] [NoUnits] [NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
6.54032 -0.0834871 -0.0540702 -0.110909 0.254306 0.996517 Quadrangle((x: 0.0 m, y: 0.0 m), ..., (x: 0.0 m, y: 1.0 m))
3.27624 1.03753 0.991226 0.955863 -0.134238 0.50835 Quadrangle((x: 1.0 m, y: 0.0 m), ..., (x: 1.0 m, y: 1.0 m))
5.05776 3.30281 -1.10131 -2.07675 -0.94095 -0.987032 Quadrangle((x: 2.0 m, y: 0.0 m), ..., (x: 2.0 m, y: 1.0 m))
8.73168 3.77482 2.55862 3.10735 0.638921 -0.806124 Quadrangle((x: 3.0 m, y: 0.0 m), ..., (x: 3.0 m, y: 1.0 m))
7.97004 -3.26781 -0.382793 -0.125731 0.993273 -0.992046 Quadrangle((x: 4.0 m, y: 0.0 m), ..., (x: 4.0 m, y: 1.0 m))
5.56696 -0.430157 -1.29843 -0.857643 -0.656539 0.9089 Quadrangle((x: 5.0 m, y: 0.0 m), ..., (x: 5.0 m, y: 1.0 m))
8.70301 -4.03265 0.847914 0.315138 0.660711 -0.628591 Quadrangle((x: 6.0 m, y: 0.0 m), ..., (x: 6.0 m, y: 1.0 m))
6.85746 -1.53107 -0.439342 0.241203 0.543227 0.03972 Quadrangle((x: 7.0 m, y: 0.0 m), ..., (x: 7.0 m, y: 1.0 m))
2.93672 1.14326 -1.87917 -1.71375 0.203447 0.414633 Quadrangle((x: 8.0 m, y: 0.0 m), ..., (x: 8.0 m, y: 1.0 m))
5.19611 1.06549 -0.0499859 -0.283416 -0.88527 0.484074 Quadrangle((x: 9.0 m, y: 0.0 m), ..., (x: 9.0 m, y: 1.0 m))
gt |> Map([2, 3] => ((b, c) -> 2b + c) => "f(b, c)")
10000×6 GeoTable over 100×100 CartesianGrid
a b c d f(b, c) geometry
Continuous Continuous Continuous Continuous Continuous Quadrangle
[NoUnits] [NoUnits] [NoUnits] [NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
6.54032 -0.0834871 -0.0540702 -0.110909 -0.221044 Quadrangle((x: 0.0 m, y: 0.0 m), ..., (x: 0.0 m, y: 1.0 m))
3.27624 1.03753 0.991226 0.955863 3.06628 Quadrangle((x: 1.0 m, y: 0.0 m), ..., (x: 1.0 m, y: 1.0 m))
5.05776 3.30281 -1.10131 -2.07675 5.50432 Quadrangle((x: 2.0 m, y: 0.0 m), ..., (x: 2.0 m, y: 1.0 m))
8.73168 3.77482 2.55862 3.10735 10.1083 Quadrangle((x: 3.0 m, y: 0.0 m), ..., (x: 3.0 m, y: 1.0 m))
7.97004 -3.26781 -0.382793 -0.125731 -6.9184 Quadrangle((x: 4.0 m, y: 0.0 m), ..., (x: 4.0 m, y: 1.0 m))
5.56696 -0.430157 -1.29843 -0.857643 -2.15875 Quadrangle((x: 5.0 m, y: 0.0 m), ..., (x: 5.0 m, y: 1.0 m))
8.70301 -4.03265 0.847914 0.315138 -7.21738 Quadrangle((x: 6.0 m, y: 0.0 m), ..., (x: 6.0 m, y: 1.0 m))
6.85746 -1.53107 -0.439342 0.241203 -3.50147 Quadrangle((x: 7.0 m, y: 0.0 m), ..., (x: 7.0 m, y: 1.0 m))
2.93672 1.14326 -1.87917 -1.71375 0.407344 Quadrangle((x: 8.0 m, y: 0.0 m), ..., (x: 8.0 m, y: 1.0 m))
5.19611 1.06549 -0.0499859 -0.283416 2.081 Quadrangle((x: 9.0 m, y: 0.0 m), ..., (x: 9.0 m, y: 1.0 m))

The name of the resulting column can be provided or omitted. If the name is omitted like in the example above with the column “a”, it is created by concatenation of column and function names.

Note

The Map transform mimics the behavior of the transform function in DataFrames.jl, except that it always broadcasts the functions to the rows of the selected columns and always produces a single column for each function.

To filter rows in the geotable based on a given predicate (i.e., a function that returns true or false), we can use the Filter transform:

gt |> Filter(row -> row.a < 0 && row.b > 0)
1261×5 GeoTable over 1261 view(::CartesianGrid, [1631, 1993, 3133, 4133, ..., 9991, 9996, 9998, 9999])
a b c d geometry
Continuous Continuous Continuous Continuous Quadrangle
[NoUnits] [NoUnits] [NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
-0.280453 0.789803 -0.112671 0.42756 Quadrangle((x: 30.0 m, y: 16.0 m), ..., (x: 30.0 m, y: 17.0 m))
-0.440897 2.06155 0.187134 1.74653 Quadrangle((x: 92.0 m, y: 19.0 m), ..., (x: 92.0 m, y: 20.0 m))
-1.79787 0.175035 -0.808858 -1.50545 Quadrangle((x: 32.0 m, y: 31.0 m), ..., (x: 32.0 m, y: 32.0 m))
-0.489696 4.13609 2.34051 1.67604 Quadrangle((x: 32.0 m, y: 41.0 m), ..., (x: 32.0 m, y: 42.0 m))
-0.198169 1.20125 0.489804 -0.376225 Quadrangle((x: 2.0 m, y: 50.0 m), ..., (x: 2.0 m, y: 51.0 m))
-0.403846 1.65043 1.28477 1.49168 Quadrangle((x: 12.0 m, y: 50.0 m), ..., (x: 12.0 m, y: 51.0 m))
-0.490289 0.60561 -0.699498 -0.432979 Quadrangle((x: 14.0 m, y: 50.0 m), ..., (x: 14.0 m, y: 51.0 m))
-0.94797 0.546192 0.699063 0.699101 Quadrangle((x: 19.0 m, y: 50.0 m), ..., (x: 19.0 m, y: 51.0 m))
-0.135501 3.76098 -0.989691 -0.805745 Quadrangle((x: 22.0 m, y: 50.0 m), ..., (x: 22.0 m, y: 51.0 m))
-0.242961 1.11395 0.527797 -0.375361 Quadrangle((x: 24.0 m, y: 50.0 m), ..., (x: 24.0 m, y: 51.0 m))

To sort rows based on specific columns we can use the Sort transform:

gt |> Sort("a", "b")
10000×5 GeoTable over 10000 view(::CartesianGrid, [7532, 8517, 8335, 5574, ..., 1047, 3870, 2581, 4259])
a b c d geometry
Continuous Continuous Continuous Continuous Quadrangle
[NoUnits] [NoUnits] [NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
-3.40243 -0.479528 -0.218977 -0.686963 Quadrangle((x: 31.0 m, y: 75.0 m), ..., (x: 31.0 m, y: 76.0 m))
-3.02565 4.75279 1.86366 2.1012 Quadrangle((x: 16.0 m, y: 85.0 m), ..., (x: 16.0 m, y: 86.0 m))
-3.01129 -3.84292 1.6387 3.36433 Quadrangle((x: 34.0 m, y: 83.0 m), ..., (x: 34.0 m, y: 84.0 m))
-2.98885 -1.15004 -0.664941 -0.892982 Quadrangle((x: 73.0 m, y: 55.0 m), ..., (x: 73.0 m, y: 56.0 m))
-2.88009 -1.31162 0.32931 0.394791 Quadrangle((x: 47.0 m, y: 94.0 m), ..., (x: 47.0 m, y: 95.0 m))
-2.81365 1.0065 1.14315 1.74841 Quadrangle((x: 45.0 m, y: 80.0 m), ..., (x: 45.0 m, y: 81.0 m))
-2.78805 0.5721 -0.692597 0.217301 Quadrangle((x: 77.0 m, y: 67.0 m), ..., (x: 77.0 m, y: 68.0 m))
-2.77232 -2.35126 1.13484 0.321422 Quadrangle((x: 31.0 m, y: 81.0 m), ..., (x: 31.0 m, y: 82.0 m))
-2.70335 -0.610621 1.6432 2.70654 Quadrangle((x: 42.0 m, y: 57.0 m), ..., (x: 42.0 m, y: 58.0 m))
-2.70226 -0.474055 -2.38766 -2.33585 Quadrangle((x: 91.0 m, y: 60.0 m), ..., (x: 91.0 m, y: 61.0 m))

This transform accepts all options of the sortperm function in Julia, including the option to sort in reverse order:

gt |> Sort("a", "b", rev=true)
10000×5 GeoTable over 10000 view(::CartesianGrid, [4259, 2581, 3870, 1047, ..., 5574, 8335, 8517, 7532])
a b c d geometry
Continuous Continuous Continuous Continuous Quadrangle
[NoUnits] [NoUnits] [NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
13.0081 -3.19902 0.611242 1.59763 Quadrangle((x: 58.0 m, y: 42.0 m), ..., (x: 58.0 m, y: 43.0 m))
12.8954 4.59833 -1.69455 -1.38286 Quadrangle((x: 80.0 m, y: 25.0 m), ..., (x: 80.0 m, y: 26.0 m))
12.7384 -2.98058 -0.269874 -0.319404 Quadrangle((x: 69.0 m, y: 38.0 m), ..., (x: 69.0 m, y: 39.0 m))
12.5818 -2.74374 1.28193 1.66729 Quadrangle((x: 46.0 m, y: 10.0 m), ..., (x: 46.0 m, y: 11.0 m))
12.5283 -0.510797 0.636088 1.0342 Quadrangle((x: 87.0 m, y: 40.0 m), ..., (x: 87.0 m, y: 41.0 m))
12.5103 0.0563021 1.0759 2.31588 Quadrangle((x: 11.0 m, y: 38.0 m), ..., (x: 11.0 m, y: 39.0 m))
12.3746 3.12621 0.408167 -0.0783823 Quadrangle((x: 73.0 m, y: 12.0 m), ..., (x: 73.0 m, y: 13.0 m))
12.1005 -2.1437 0.358305 -0.467963 Quadrangle((x: 16.0 m, y: 33.0 m), ..., (x: 16.0 m, y: 34.0 m))
12.0777 -2.41779 2.53087 3.04657 Quadrangle((x: 51.0 m, y: 27.0 m), ..., (x: 51.0 m, y: 28.0 m))
12.0317 3.76954 0.063239 -0.398588 Quadrangle((x: 18.0 m, y: 17.0 m), ..., (x: 18.0 m, y: 18.0 m))

5.2.2 Cleaning

Some feature transforms are used to clean the data before geostatistical analysis. For example, the StdNames transform can be used to standardize variable names that are not very readable due to file format limitations. To illustrate this transform, let’s create a geotable with unreadable variable names:

ut = gt |> Select("a" => "aBc De-F", "b" => "b_2 (1)")
10000×3 GeoTable over 100×100 CartesianGrid
aBc De-F b_2 (1) geometry
Continuous Continuous Quadrangle
[NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
6.54032 -0.0834871 Quadrangle((x: 0.0 m, y: 0.0 m), ..., (x: 0.0 m, y: 1.0 m))
3.27624 1.03753 Quadrangle((x: 1.0 m, y: 0.0 m), ..., (x: 1.0 m, y: 1.0 m))
5.05776 3.30281 Quadrangle((x: 2.0 m, y: 0.0 m), ..., (x: 2.0 m, y: 1.0 m))
8.73168 3.77482 Quadrangle((x: 3.0 m, y: 0.0 m), ..., (x: 3.0 m, y: 1.0 m))
7.97004 -3.26781 Quadrangle((x: 4.0 m, y: 0.0 m), ..., (x: 4.0 m, y: 1.0 m))
5.56696 -0.430157 Quadrangle((x: 5.0 m, y: 0.0 m), ..., (x: 5.0 m, y: 1.0 m))
8.70301 -4.03265 Quadrangle((x: 6.0 m, y: 0.0 m), ..., (x: 6.0 m, y: 1.0 m))
6.85746 -1.53107 Quadrangle((x: 7.0 m, y: 0.0 m), ..., (x: 7.0 m, y: 1.0 m))
2.93672 1.14326 Quadrangle((x: 8.0 m, y: 0.0 m), ..., (x: 8.0 m, y: 1.0 m))
5.19611 1.06549 Quadrangle((x: 9.0 m, y: 0.0 m), ..., (x: 9.0 m, y: 1.0 m))

We can standardize the names with:

ut |> StdNames()
10000×3 GeoTable over 100×100 CartesianGrid
ABC_DE_F B_2_1 geometry
Continuous Continuous Quadrangle
[NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
6.54032 -0.0834871 Quadrangle((x: 0.0 m, y: 0.0 m), ..., (x: 0.0 m, y: 1.0 m))
3.27624 1.03753 Quadrangle((x: 1.0 m, y: 0.0 m), ..., (x: 1.0 m, y: 1.0 m))
5.05776 3.30281 Quadrangle((x: 2.0 m, y: 0.0 m), ..., (x: 2.0 m, y: 1.0 m))
8.73168 3.77482 Quadrangle((x: 3.0 m, y: 0.0 m), ..., (x: 3.0 m, y: 1.0 m))
7.97004 -3.26781 Quadrangle((x: 4.0 m, y: 0.0 m), ..., (x: 4.0 m, y: 1.0 m))
5.56696 -0.430157 Quadrangle((x: 5.0 m, y: 0.0 m), ..., (x: 5.0 m, y: 1.0 m))
8.70301 -4.03265 Quadrangle((x: 6.0 m, y: 0.0 m), ..., (x: 6.0 m, y: 1.0 m))
6.85746 -1.53107 Quadrangle((x: 7.0 m, y: 0.0 m), ..., (x: 7.0 m, y: 1.0 m))
2.93672 1.14326 Quadrangle((x: 8.0 m, y: 0.0 m), ..., (x: 8.0 m, y: 1.0 m))
5.19611 1.06549 Quadrangle((x: 9.0 m, y: 0.0 m), ..., (x: 9.0 m, y: 1.0 m))

By default the transform, uses the :uppersnake naming convention. Other conventions can be specified depending on personal preference:

ut |> StdNames(:uppercamel)
10000×3 GeoTable over 100×100 CartesianGrid
AbcDeF B21 geometry
Continuous Continuous Quadrangle
[NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
6.54032 -0.0834871 Quadrangle((x: 0.0 m, y: 0.0 m), ..., (x: 0.0 m, y: 1.0 m))
3.27624 1.03753 Quadrangle((x: 1.0 m, y: 0.0 m), ..., (x: 1.0 m, y: 1.0 m))
5.05776 3.30281 Quadrangle((x: 2.0 m, y: 0.0 m), ..., (x: 2.0 m, y: 1.0 m))
8.73168 3.77482 Quadrangle((x: 3.0 m, y: 0.0 m), ..., (x: 3.0 m, y: 1.0 m))
7.97004 -3.26781 Quadrangle((x: 4.0 m, y: 0.0 m), ..., (x: 4.0 m, y: 1.0 m))
5.56696 -0.430157 Quadrangle((x: 5.0 m, y: 0.0 m), ..., (x: 5.0 m, y: 1.0 m))
8.70301 -4.03265 Quadrangle((x: 6.0 m, y: 0.0 m), ..., (x: 6.0 m, y: 1.0 m))
6.85746 -1.53107 Quadrangle((x: 7.0 m, y: 0.0 m), ..., (x: 7.0 m, y: 1.0 m))
2.93672 1.14326 Quadrangle((x: 8.0 m, y: 0.0 m), ..., (x: 8.0 m, y: 1.0 m))
5.19611 1.06549 Quadrangle((x: 9.0 m, y: 0.0 m), ..., (x: 9.0 m, y: 1.0 m))
ut |> StdNames(:upperflat)
10000×3 GeoTable over 100×100 CartesianGrid
ABCDEF B21 geometry
Continuous Continuous Quadrangle
[NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
6.54032 -0.0834871 Quadrangle((x: 0.0 m, y: 0.0 m), ..., (x: 0.0 m, y: 1.0 m))
3.27624 1.03753 Quadrangle((x: 1.0 m, y: 0.0 m), ..., (x: 1.0 m, y: 1.0 m))
5.05776 3.30281 Quadrangle((x: 2.0 m, y: 0.0 m), ..., (x: 2.0 m, y: 1.0 m))
8.73168 3.77482 Quadrangle((x: 3.0 m, y: 0.0 m), ..., (x: 3.0 m, y: 1.0 m))
7.97004 -3.26781 Quadrangle((x: 4.0 m, y: 0.0 m), ..., (x: 4.0 m, y: 1.0 m))
5.56696 -0.430157 Quadrangle((x: 5.0 m, y: 0.0 m), ..., (x: 5.0 m, y: 1.0 m))
8.70301 -4.03265 Quadrangle((x: 6.0 m, y: 0.0 m), ..., (x: 6.0 m, y: 1.0 m))
6.85746 -1.53107 Quadrangle((x: 7.0 m, y: 0.0 m), ..., (x: 7.0 m, y: 1.0 m))
2.93672 1.14326 Quadrangle((x: 8.0 m, y: 0.0 m), ..., (x: 8.0 m, y: 1.0 m))
5.19611 1.06549 Quadrangle((x: 9.0 m, y: 0.0 m), ..., (x: 9.0 m, y: 1.0 m))

The Replace transform can be used to replace specific values in the geotable by new values that are meaningful to the analysis. For example, we can replace the values -999 and NaN that are used to represent missing values in some file formats:

rt = georef((a=[1,-999,3], b=[NaN,5,6]))
3×3 GeoTable over 3 CartesianGrid
a b geometry
Categorical Continuous Segment
[NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
1 NaN Segment((x: 0.0 m), (x: 1.0 m))
-999 5.0 Segment((x: 1.0 m), (x: 2.0 m))
3 6.0 Segment((x: 2.0 m), (x: 3.0 m))
rt |> Replace(-999 => missing, NaN => missing)
3×3 GeoTable over 3 CartesianGrid
a b geometry
Categorical Continuous Segment
[NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
1 missing Segment((x: 0.0 m), (x: 1.0 m))
missing 5.0 Segment((x: 1.0 m), (x: 2.0 m))
3 6.0 Segment((x: 2.0 m), (x: 3.0 m))

or replace all negative values using a predicate function:

rt |> Replace(<(0) => missing)
3×3 GeoTable over 3 CartesianGrid
a b geometry
Categorical Continuous Segment
[NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
1 NaN Segment((x: 0.0 m), (x: 1.0 m))
missing 5.0 Segment((x: 1.0 m), (x: 2.0 m))
3 6.0 Segment((x: 2.0 m), (x: 3.0 m))
Note

In Julia, the expression <(0) is equivalent to the predicate function x -> x < 0.

Although Replace could be used to replace missing values by new values, there is a specific transform for this purpose named Coalesce:

ct = georef((a=[1,missing,3], b=[4,5,6])) |> Coalesce(value=2)
3×3 GeoTable over 3 CartesianGrid
a b geometry
Categorical Categorical Segment
[NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
1 4 Segment((x: 0.0 m), (x: 1.0 m))
2 5 Segment((x: 1.0 m), (x: 2.0 m))
3 6 Segment((x: 2.0 m), (x: 3.0 m))
Note

Unlike Replace, the Coalesce transform also changes the column type to make sure that no missing values can be stored in the future:

typeof(ct.a)
Vector{Int64} (alias for Array{Int64, 1})

In many applications, it is enough to simply drop all rows for which the selected column values are missing. This is the purpose of the DropMissing transform:

georef((a=[1,missing,3], b=[4,5,6])) |> DropMissing()
2×3 GeoTable over 2 view(::CartesianGrid, [1, 3])
a b geometry
Categorical Categorical Segment
[NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
1 4 Segment((x: 0.0 m), (x: 1.0 m))
3 6 Segment((x: 2.0 m), (x: 3.0 m))

The DropNaN is an alternative to drop all rows for which the selected column values are NaN.

5.2.3 Statistical

The framework provides various feature transforms for statistical analysis. We will cover some of these transforms in more detail in Part V of the book with real data. In the following examples we illustrate the most basic statistical transforms with synthetic data.

The Sample transform can be used to sample rows of the geotable at random, with or without replacement depending on the replace option. Other options are available such as rng to set the random number generator and ordered to preserve the order of rows in the original geotable:

gt |> Sample(1000, replace=false) |> viewer
┌ Warning: Found `resolution` in the theme when creating a `Scene`. The `resolution` keyword for `Scene`s and `Figure`s has been deprecated. Use `Figure(; size = ...` or `Scene(; size = ...)` instead, which better reflects that this is a unitless size and not a pixel resolution. The key could also come from `set_theme!` calls or related theming functions.
└ @ Makie ~/.julia/packages/Makie/We6MY/src/scenes.jl:227

Note

Similar to Filter and Sort, the Sample transform is lazy. It simply stores the indices of sampled rows for future construction of the new geotable.

The Center and Scale transforms can be used to standardize the range of values in a geotable. Aliases are provided for specific types of Scale such as MinMax and Interquartile. We can use the describe function to visualize basic statistics before and after the transforms:

gt |> describe
Table with 6 columns and 4 rows:
     variable  mean         minimum   median       maximum  nmissing
   ┌────────────────────────────────────────────────────────────────
 1 │ a         3.01096      -3.40243  2.08518      13.0081  0
 2 │ b         -0.0589702   -11.5306  -0.0499244   11.7694  0
 3 │ c         -0.00188243  -3.87011  -0.0015644   3.51286  0
 4 │ d         0.000235056  -4.38454  -0.00393728  4.4025   0
gt |> Center("a") |> describe
Table with 6 columns and 4 rows:
     variable  mean         minimum   median       maximum  nmissing
   ┌────────────────────────────────────────────────────────────────
 1 │ a         0.0          -6.4134   -0.925785    9.99709  0
 2 │ b         -0.0589702   -11.5306  -0.0499244   11.7694  0
 3 │ c         -0.00188243  -3.87011  -0.0015644   3.51286  0
 4 │ d         0.000235056  -4.38454  -0.00393728  4.4025   0
gt |> MinMax() |> describe
Table with 6 columns and 4 rows:
     variable  mean      minimum  median    maximum  nmissing
   ┌─────────────────────────────────────────────────────────
 1 │ a         0.390811  0.0      0.334397  1.0      0
 2 │ b         0.492346  0.0      0.492735  1.0      0
 3 │ c         0.523939  0.0      0.523982  1.0      0
 4 │ d         0.499005  0.0      0.49853   1.0      0

The ZScore transform is similar to the Scale transform, but it uses the mean and the standard deviation to standardize the range:

gt |> ZScore() |> describe
Table with 6 columns and 4 rows:
     variable  mean          minimum   median       maximum  nmissing
   ┌─────────────────────────────────────────────────────────────────
 1 │ a         9.09495e-17   -1.88517  -0.272127    2.93857  0
 2 │ b         2.55795e-17   -4.56363  0.00359858   4.70551  0
 3 │ c         1.20792e-17   -3.8713   0.000318283  3.51753  0
 4 │ d         -1.13687e-17  -3.77367  -0.00359085  3.78872  0

Another important univariate transform is the Quantile transform, which can be used to convert empirical distribution in a column of the geotable to any given distribution from Distributions.jl by Lin et al. (2023). Selected columns are converted to a Normal distribution by default, but more than 60 distributions are available:

gt |> Quantile() |> values |> pairplot
┌ Warning: Found `resolution` in the theme when creating a `Scene`. The `resolution` keyword for `Scene`s and `Figure`s has been deprecated. Use `Figure(; size = ...` or `Scene(; size = ...)` instead, which better reflects that this is a unitless size and not a pixel resolution. The key could also come from `set_theme!` calls or related theming functions.
└ @ Makie ~/.julia/packages/Makie/We6MY/src/scenes.jl:227

In data science, scientific traits are used to link data types to adequate statistical algorithms. The most popular scientific traits encountered in geoscientific applications are the Continuous and the Categorical scientific traits. To convert (or coerce) the scientific traits of columns in a geotable, we can use the Coerce transform:

st = georef((a=[1,2,2,2,3,3], b=[1,2,3,4,5,6])) |> Coerce("b" => Continuous)
6×3 GeoTable over 6 CartesianGrid
a b geometry
Categorical Continuous Segment
[NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
1 1.0 Segment((x: 0.0 m), (x: 1.0 m))
2 2.0 Segment((x: 1.0 m), (x: 2.0 m))
2 3.0 Segment((x: 2.0 m), (x: 3.0 m))
2 4.0 Segment((x: 3.0 m), (x: 4.0 m))
3 5.0 Segment((x: 4.0 m), (x: 5.0 m))
3 6.0 Segment((x: 5.0 m), (x: 6.0 m))
eltype(st.b)
Float64
Note

All scientific traits are documented in the DataScienceTraits.jl module, and can be used to select variables:

st |> Only(Continuous)
6×2 GeoTable over 6 CartesianGrid
b geometry
Continuous Segment
[NoUnits] 🖈 Cartesian{NoDatum}
1.0 Segment((x: 0.0 m), (x: 1.0 m))
2.0 Segment((x: 1.0 m), (x: 2.0 m))
3.0 Segment((x: 2.0 m), (x: 3.0 m))
4.0 Segment((x: 3.0 m), (x: 4.0 m))
5.0 Segment((x: 4.0 m), (x: 5.0 m))
6.0 Segment((x: 5.0 m), (x: 6.0 m))

The Levels transform can be used to adjust the categories (or levels) of Categorical columns in case the sampling process does not include all possible values:

st = st |> Levels("a" => [1,2,3,4])
6×3 GeoTable over 6 CartesianGrid
a b geometry
Categorical Continuous Segment
[NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
1 1.0 Segment((x: 0.0 m), (x: 1.0 m))
2 2.0 Segment((x: 1.0 m), (x: 2.0 m))
2 3.0 Segment((x: 2.0 m), (x: 3.0 m))
2 4.0 Segment((x: 3.0 m), (x: 4.0 m))
3 5.0 Segment((x: 4.0 m), (x: 5.0 m))
3 6.0 Segment((x: 5.0 m), (x: 6.0 m))
levels(st.a)
4-element Vector{Int64}:
 1
 2
 3
 4

Another popular transform in statistical learning is the OneHot transform. It converts a Categorical column into multiple columns of true/false values, one column for each level:

st |> OneHot("a")
6×6 GeoTable over 6 CartesianGrid
a_1 a_2 a_3 a_4 b geometry
Categorical Categorical Categorical Categorical Continuous Segment
[NoUnits] [NoUnits] [NoUnits] [NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
true false false false 1.0 Segment((x: 0.0 m), (x: 1.0 m))
false true false false 2.0 Segment((x: 1.0 m), (x: 2.0 m))
false true false false 3.0 Segment((x: 2.0 m), (x: 3.0 m))
false true false false 4.0 Segment((x: 3.0 m), (x: 4.0 m))
false false true false 5.0 Segment((x: 4.0 m), (x: 5.0 m))
false false true false 6.0 Segment((x: 5.0 m), (x: 6.0 m))

A similar transform for Continuous columns is the Indicator transform. It converts the column into multiple columns based on threshold values on the support of the data. By default, the threshold values are computed on a quantile scale:

st |> Indicator("b", k=3, scale=:quantile)
6×5 GeoTable over 6 CartesianGrid
a b_1 b_2 b_3 geometry
Categorical Categorical Categorical Categorical Segment
[NoUnits] [NoUnits] [NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
1 true true true Segment((x: 0.0 m), (x: 1.0 m))
2 true true true Segment((x: 1.0 m), (x: 2.0 m))
2 false true true Segment((x: 2.0 m), (x: 3.0 m))
2 false true true Segment((x: 3.0 m), (x: 4.0 m))
3 false false true Segment((x: 4.0 m), (x: 5.0 m))
3 false false true Segment((x: 5.0 m), (x: 6.0 m))

More advanced statistical transforms such as EigenAnalysis, PCA, DRS, SDS, ProjectionPursuit for multivariate data analysis and Remainder, Closure, LogRatio, ALR, CLR, ILR for compositional data analysis will be covered in future chapters.

5.3 Geometric transforms

While feature transforms operate on the values of the geotable, geometric transforms operate on the geospatial domain. The framework provides various geometric transforms for 2D and 3D space.

5.3.1 Coordinate

A coordinate transform is a geometric transform that modifies the coordinates of all points in the domain without any advanced topological modification (i.e., connectivities are preserved). The most prominent examples of coordinate transforms are Translate, Rotate and Scale.

Let’s load an additional geotable to see these transforms in action:

using GeoIO

bt = GeoIO.load("data/beethoven.ply")

viz(bt.geometry)
┌ Warning: Found `resolution` in the theme when creating a `Scene`. The `resolution` keyword for `Scene`s and `Figure`s has been deprecated. Use `Figure(; size = ...` or `Scene(; size = ...)` instead, which better reflects that this is a unitless size and not a pixel resolution. The key could also come from `set_theme!` calls or related theming functions.
└ @ Makie ~/.julia/packages/Makie/We6MY/src/scenes.jl:227

The Beethoven domain has been saved in the .ply file in a position that is not ideal for visualization. We can rotate this domain with any active rotation specification from Rotations.jl by Koolen et al. (2023) to improve the visualization. For example, we can specify that we want to rotate all points in the mesh by analogy with a rotation between coordinates (0, 1, 0) and coordinates (0, 0, 1):

rt = bt |> Rotate((0, 1, 0), (0, 0, 1))

viz(rt.geometry)
┌ Warning: Found `resolution` in the theme when creating a `Scene`. The `resolution` keyword for `Scene`s and `Figure`s has been deprecated. Use `Figure(; size = ...` or `Scene(; size = ...)` instead, which better reflects that this is a unitless size and not a pixel resolution. The key could also come from `set_theme!` calls or related theming functions.
└ @ Makie ~/.julia/packages/Makie/We6MY/src/scenes.jl:227

Beethoven is now standing up, but still facing the wall. Let’s rotate it once again by analogy between coordinates (1, 0, 0) and (-1, 1, 0):

rt = rt |> Rotate((1, 0, 0), (-1, 1, 0))

viz(rt.geometry)
┌ Warning: Found `resolution` in the theme when creating a `Scene`. The `resolution` keyword for `Scene`s and `Figure`s has been deprecated. Use `Figure(; size = ...` or `Scene(; size = ...)` instead, which better reflects that this is a unitless size and not a pixel resolution. The key could also come from `set_theme!` calls or related theming functions.
└ @ Makie ~/.julia/packages/Makie/We6MY/src/scenes.jl:227

Rotation specifications are also available in 2D space. As an example, we can rotate the 2D grid of our synthetic geotable by the counter clockwise angle π/4:

gt |> Rotate(Angle2d(π/4)) |> viewer
┌ Warning: Found `resolution` in the theme when creating a `Scene`. The `resolution` keyword for `Scene`s and `Figure`s has been deprecated. Use `Figure(; size = ...` or `Scene(; size = ...)` instead, which better reflects that this is a unitless size and not a pixel resolution. The key could also come from `set_theme!` calls or related theming functions.
└ @ Makie ~/.julia/packages/Makie/We6MY/src/scenes.jl:227

In GIS, this new geotable would be called a rotated “raster”. As another example, let’s translate the geotable to the origin of the coordinate system with the Translate transform:

c = centroid(gt.geometry)

gt |> Translate(-to(c)...) |> viewer
┌ Warning: Found `resolution` in the theme when creating a `Scene`. The `resolution` keyword for `Scene`s and `Figure`s has been deprecated. Use `Figure(; size = ...` or `Scene(; size = ...)` instead, which better reflects that this is a unitless size and not a pixel resolution. The key could also come from `set_theme!` calls or related theming functions.
└ @ Makie ~/.julia/packages/Makie/We6MY/src/scenes.jl:227

and scale it with a positive factor for each dimension:

gt |> Scale(0.1, 0.2) |> viewer
┌ Warning: Found `resolution` in the theme when creating a `Scene`. The `resolution` keyword for `Scene`s and `Figure`s has been deprecated. Use `Figure(; size = ...` or `Scene(; size = ...)` instead, which better reflects that this is a unitless size and not a pixel resolution. The key could also come from `set_theme!` calls or related theming functions.
└ @ Makie ~/.julia/packages/Makie/We6MY/src/scenes.jl:227

The StdCoords transform combines Translate and Scale to standardize the coordinates of the domain to the interval [-0.5, 0.5]:

gt |> StdCoords() |> viewer
┌ Warning: Found `resolution` in the theme when creating a `Scene`. The `resolution` keyword for `Scene`s and `Figure`s has been deprecated. Use `Figure(; size = ...` or `Scene(; size = ...)` instead, which better reflects that this is a unitless size and not a pixel resolution. The key could also come from `set_theme!` calls or related theming functions.
└ @ Makie ~/.julia/packages/Makie/We6MY/src/scenes.jl:227

In GIS, another very important coordinate transform is the Proj transform. We will cover this transform in the next chapter because it depends on the concept of map projection, which deserves more attention.

Note

In our framework, the Proj transform is just another coordinate transform. It is implemented with the same code optimizations, and can be used in conjunction with many other transforms that are not available elsewhere.

5.3.2 Advanced

Advanced geometric transforms are provided that change the topology of the domain besides the coordinates of points. Some of these transforms can be useful to repair problematic geometries acquired from sensors in the real world.

The Repair transform is parameterized by an integer K that identifies the repair to be performed. For example, Repair{0}() is a transform that removes duplicated vertices and faces in a domain represented by a mesh. The Repair{9}() on the other hand fixes the orientation of rings in polygonal areas so that the external boundary is oriented counter clockwise and the inner boundaries are oriented clockwise. The list of available repairs will continue to grow with the implementation of new geometric algorithms in the framework.

To understand why geometric transforms are more general than coordinate transforms, let’s consider the following polygonal area with holes:

outer = [(0.0, 0.0), (1.0, 0.0), (1.0, 1.0), (0.0, 1.0)]
hole1 = [(0.2, 0.2), (0.2, 0.4), (0.4, 0.4), (0.4, 0.2)]
hole2 = [(0.6, 0.2), (0.6, 0.4), (0.8, 0.4), (0.8, 0.2)]
poly  = PolyArea([outer, hole1, hole2])

viz(poly)
┌ Warning: Found `resolution` in the theme when creating a `Scene`. The `resolution` keyword for `Scene`s and `Figure`s has been deprecated. Use `Figure(; size = ...` or `Scene(; size = ...)` instead, which better reflects that this is a unitless size and not a pixel resolution. The key could also come from `set_theme!` calls or related theming functions.
└ @ Makie ~/.julia/packages/Makie/We6MY/src/scenes.jl:227

We can connect the holes with the external boundary (or ring) using the Bridge transform:

poly |> Bridge(0.01) |> viz
┌ Warning: Found `resolution` in the theme when creating a `Scene`. The `resolution` keyword for `Scene`s and `Figure`s has been deprecated. Use `Figure(; size = ...` or `Scene(; size = ...)` instead, which better reflects that this is a unitless size and not a pixel resolution. The key could also come from `set_theme!` calls or related theming functions.
└ @ Makie ~/.julia/packages/Makie/We6MY/src/scenes.jl:227

By looking at the visualization, we observe that the number of vertices changed to accommodate the so called “bridges” between the rings. The topology also changed as there are no holes in the resulting geometry.

As a final example of advanced geometric transform, we illustrate the TaubinSmoothing transform, which gradually removes sharp boundaries of a manifold mesh:

st = bt |> TaubinSmoothing(30)

fig = Mke.Figure()
viz(fig[1,1], bt.geometry)
viz(fig[1,2], st.geometry)
fig
┌ Warning: Found `resolution` in the theme when creating a `Scene`. The `resolution` keyword for `Scene`s and `Figure`s has been deprecated. Use `Figure(; size = ...` or `Scene(; size = ...)` instead, which better reflects that this is a unitless size and not a pixel resolution. The key could also come from `set_theme!` calls or related theming functions.
└ @ Makie ~/.julia/packages/Makie/We6MY/src/scenes.jl:227

For more advanced geometric transforms, please consult the official documentation.

5.4 Geospatial transforms

Geospatial transforms are those transforms that operate on both the values and the domain of the geotable. They are common in geostatistical workflows that need to remove geospatial “trends” or workflows that need to extract geometries from domains.

As an example, let’s consider the following geotable with a variable z that made of a trend component μ and a noise component ϵ:

# quadratic + noise
r = range(-1, stop=1, length=100)
μ = [x^2 + y^2 for x in r, y in r]
ϵ = 0.1rand(100, 100)
t = georef((z=μ+ϵ,))

viewer(t)
┌ Warning: Found `resolution` in the theme when creating a `Scene`. The `resolution` keyword for `Scene`s and `Figure`s has been deprecated. Use `Figure(; size = ...` or `Scene(; size = ...)` instead, which better reflects that this is a unitless size and not a pixel resolution. The key could also come from `set_theme!` calls or related theming functions.
└ @ Makie ~/.julia/packages/Makie/We6MY/src/scenes.jl:227

We can use the Detrend transform to remove a trend of polynomial degree 2:

t |> Detrend(degree=2) |> viewer
┌ Warning: Found `resolution` in the theme when creating a `Scene`. The `resolution` keyword for `Scene`s and `Figure`s has been deprecated. Use `Figure(; size = ...` or `Scene(; size = ...)` instead, which better reflects that this is a unitless size and not a pixel resolution. The key could also come from `set_theme!` calls or related theming functions.
└ @ Makie ~/.julia/packages/Makie/We6MY/src/scenes.jl:227

The remaining component can then be modeled with geostatistical models of geospatial correlation, which will be covered in Part IV of the book.

Models of geospatial correlation such as variograms (Hoffimann and Zadrozny 2019) require unique coordinates in the geotable and that is the purpose of the UniqueCoords transform. It removes duplicate points in the geotable and aggregates the values with custom aggregation functions.

Let’s consider the following geotable stored in a .png file to illustrate another geospatial transform:

letters = GeoIO.load("data/letters.png")
44255×2 GeoTable over 265×167 TransformedGrid
color geometry
Colorful Quadrangle
[NoUnits] 🖈 Cartesian{NoDatum}
Gray{N0f8}(1.0) Quadrangle((x: -1.62266e-14 m, y: 265.0 m), ..., (x: 1.0 m, y: 265.0 m))
Gray{N0f8}(1.0) Quadrangle((x: -1.61653e-14 m, y: 264.0 m), ..., (x: 1.0 m, y: 264.0 m))
Gray{N0f8}(1.0) Quadrangle((x: -1.61041e-14 m, y: 263.0 m), ..., (x: 1.0 m, y: 263.0 m))
Gray{N0f8}(1.0) Quadrangle((x: -1.60429e-14 m, y: 262.0 m), ..., (x: 1.0 m, y: 262.0 m))
Gray{N0f8}(1.0) Quadrangle((x: -1.59816e-14 m, y: 261.0 m), ..., (x: 1.0 m, y: 261.0 m))
Gray{N0f8}(1.0) Quadrangle((x: -1.59204e-14 m, y: 260.0 m), ..., (x: 1.0 m, y: 260.0 m))
Gray{N0f8}(1.0) Quadrangle((x: -1.58592e-14 m, y: 259.0 m), ..., (x: 1.0 m, y: 259.0 m))
Gray{N0f8}(1.0) Quadrangle((x: -1.57979e-14 m, y: 258.0 m), ..., (x: 1.0 m, y: 258.0 m))
Gray{N0f8}(1.0) Quadrangle((x: -1.57367e-14 m, y: 257.0 m), ..., (x: 1.0 m, y: 257.0 m))
Gray{N0f8}(1.0) Quadrangle((x: -1.56755e-14 m, y: 256.0 m), ..., (x: 1.0 m, y: 256.0 m))

The Potrace transform can be used to extract complex geometries from a geotable over a 2D Grid. It transforms the Grid domain into a GeometrySet based on any column that contains a discrete set of marker values. In this example, we use the color as the column with markers:

Ab = letters |> Potrace("color", ϵ=0.8)
2×2 GeoTable over 2 GeometrySet
color geometry
Colorful MultiPolygon
[NoUnits] 🖈 Cartesian{NoDatum}
Gray{N0f8}(1.0) Multi(4×PolyArea)
Gray{N0f8}(0.0) Multi(2×PolyArea)

The option ϵ controls the deviation tolerance used to simplify the boundaries of the geometries. The higher is the tolerance, the less is the number of segments in the boundary:

viz(Ab.geometry[2], color = "black")
┌ Warning: Found `resolution` in the theme when creating a `Scene`. The `resolution` keyword for `Scene`s and `Figure`s has been deprecated. Use `Figure(; size = ...` or `Scene(; size = ...)` instead, which better reflects that this is a unitless size and not a pixel resolution. The key could also come from `set_theme!` calls or related theming functions.
└ @ Makie ~/.julia/packages/Makie/We6MY/src/scenes.jl:227

In the reverse direction, we have the Rasterize transform, which takes a geotable over a GeometrySet and assigns the geometries to a Grid. In this transform, we can either provide an external grid for the the assignments, or request a grid size to discretize the boundingbox of all geometries:

A = [1, 2, 3, 4, 5]
B = [1.1, 2.2, 3.3, 4.4, 5.5]
p1 = Triangle((2, 0), (6, 2), (2, 2))
p2 = Triangle((0, 6), (3, 8), (0, 10))
p3 = Quadrangle((3, 6), (9, 6), (9, 9), (6, 9))
p4 = Quadrangle((7, 0), (10, 0), (10, 4), (7, 4))
p5 = Pentagon((1, 3), (5, 3), (6, 6), (3, 8), (0, 6))
gt = georef((; A, B), [p1, p2, p3, p4, p5])
5×3 GeoTable over 5 GeometrySet
A B geometry
Categorical Continuous Ngon
[NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
1 1.1 Triangle((x: 2.0 m, y: 0.0 m), (x: 6.0 m, y: 2.0 m), (x: 2.0 m, y: 2.0 m))
2 2.2 Triangle((x: 0.0 m, y: 6.0 m), (x: 3.0 m, y: 8.0 m), (x: 0.0 m, y: 10.0 m))
3 3.3 Quadrangle((x: 3.0 m, y: 6.0 m), ..., (x: 6.0 m, y: 9.0 m))
4 4.4 Quadrangle((x: 7.0 m, y: 0.0 m), ..., (x: 7.0 m, y: 4.0 m))
5 5.5 Pentagon((x: 1.0 m, y: 3.0 m), ..., (x: 0.0 m, y: 6.0 m))
gt |> viewer
┌ Warning: Found `resolution` in the theme when creating a `Scene`. The `resolution` keyword for `Scene`s and `Figure`s has been deprecated. Use `Figure(; size = ...` or `Scene(; size = ...)` instead, which better reflects that this is a unitless size and not a pixel resolution. The key could also come from `set_theme!` calls or related theming functions.
└ @ Makie ~/.julia/packages/Makie/We6MY/src/scenes.jl:227

nt = gt |> Rasterize(20, 20)
400×3 GeoTable over 20×20 CartesianGrid
A B geometry
Categorical Continuous Quadrangle
[NoUnits] [NoUnits] 🖈 Cartesian{NoDatum}
missing missing Quadrangle((x: 0.0 m, y: 0.0 m), ..., (x: 0.0 m, y: 0.5 m))
missing missing Quadrangle((x: 0.5 m, y: 0.0 m), ..., (x: 0.5 m, y: 0.5 m))
missing missing Quadrangle((x: 1.0 m, y: 0.0 m), ..., (x: 1.0 m, y: 0.5 m))
1 1.1 Quadrangle((x: 1.5 m, y: 0.0 m), ..., (x: 1.5 m, y: 0.5 m))
1 1.1 Quadrangle((x: 2.0 m, y: 0.0 m), ..., (x: 2.0 m, y: 0.5 m))
missing missing Quadrangle((x: 2.5 m, y: 0.0 m), ..., (x: 2.5 m, y: 0.5 m))
missing missing Quadrangle((x: 3.0 m, y: 0.0 m), ..., (x: 3.0 m, y: 0.5 m))
missing missing Quadrangle((x: 3.5 m, y: 0.0 m), ..., (x: 3.5 m, y: 0.5 m))
missing missing Quadrangle((x: 4.0 m, y: 0.0 m), ..., (x: 4.0 m, y: 0.5 m))
missing missing Quadrangle((x: 4.5 m, y: 0.0 m), ..., (x: 4.5 m, y: 0.5 m))
nt |> viewer
┌ Warning: Found `resolution` in the theme when creating a `Scene`. The `resolution` keyword for `Scene`s and `Figure`s has been deprecated. Use `Figure(; size = ...` or `Scene(; size = ...)` instead, which better reflects that this is a unitless size and not a pixel resolution. The key could also come from `set_theme!` calls or related theming functions.
└ @ Makie ~/.julia/packages/Makie/We6MY/src/scenes.jl:227

The values of the variables are aggregated at geometric intersections using a default aggregation function, which can be overwritten with an option. Once the geotable is defined over a Grid, it is possible to refine or coarsen the grid with the Downscale and Upscale transforms.

The Transfer of values to a new geospatial domain is another very useful geospatial transform. The Aggregate transform is related, but aggregates the values with given aggregation functions.

5.5 Remarks

In this chapter we learned the important concept of transforms, and saw examples of the concept in action with synthetic data. In order to leverage the large number of transforms implemented in the framework, all we need to do is load our geospatial data as a geotable using georef or GeoIO.jl.

Some additional remarks:

  • One of the major advantages of transforms compared to traditional row/column access in data science is that they preserve geospatial information. There is no need to keep track of indices in arrays to repeatedly reattach values to geometries.
  • Transforms can be organized into three classes—feature, geometric and geospatial—depending on how they operate with the values and the domain of the geotable:
    • Feature transforms operate on the values. They include column selection, data cleaning, statistical analysis and any transform designed for traditional Tables.jl.
    • Geometric transforms operate on the domain. They include coordinate transforms that simply modify the coordinates of points as well as more advanced transforms that can change the topology of the domain.
    • Geospatial transforms operate on both the values and domain. They include geostatistical transforms and transforms that use other columns besides the geometry column to produce new columns and geometries.

In the next chapters, we will review map projections with the Proj coordinate transform, and will introduce one of the greatest features of the framework known as transform pipelines.