-
-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy pathoutbox.qmd
219 lines (166 loc) · 9.92 KB
/
outbox.qmd
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
```{r,echo=FALSE,message=FALSE,warning=FALSE}
library(lidR)
library(ggplot2)
library(rgl)
library(stars)
r3dDefaults = rgl::r3dDefaults
m = structure(c(0.921, -0.146, 0.362, 0, 0.386, 0.482, -0.787, 0,
-0.06, 0.864, 0.5, 0, 0, 0, 0, 1), .Dim = c(4L, 4L))
r3dDefaults$FOV = 50
r3dDefaults$userMatrix = m
r3dDefaults$zoom = 0.75
options(lidR.progress = FALSE)
knitr::opts_chunk$set(
comment = "#>",
collapse = TRUE,
fig.align = "center")
rgl::setupKnitr(autoprint = TRUE)
```
# Thinking outside the box {#sec-outbox}
We have presented many common processes through these tutorials. Most examples presented could have been done with other softwares, but what we have seen is only the tip of the `lidR` iceberg! In fact, `lidR` was never designed with common processes in mind, but instead to explore new ones and push innovation in this field. Some examples might be found here and there in the book. The following section presents some applications that are likely to be impossible to reproduce in other softwares, demonstrating how we can go from a novel idea to a fully functional tool using `lidR`.
## New complex metrics in ABA {#sec-outbox-custom-metrics}
### Distance between returns {#sec-outbox-distance-returns}
In forestry, simple metrics can be derived from first returns. But what about metrics that estimate the average distance between first and last returns? Is it a valuable metric for characterizing of forest stucture? Can we refine the prediction with such metric? Nobody knows, but we can try.
```{r, echo = FALSE, message=FALSE}
library(lidR)
LASfile <- "data/ENGINE/catalog/tiles_338000_5239000_1.laz"
las <- readLAS(LASfile)
```
First, we need to retrieve each emitted pulse and keep only the first and last point of each sequence excluding sequences with only one return.
```{r}
las <- retrieve_pulses(las)
las <- filter_firstlast(las)
las <- filter_poi(las, NumberOfReturns > 1)
```
Using the [`data.table` syntax](https://cran.r-project.org/web/packages/data.table/vignettes/datatable-intro.html) we can compute the distance between first and last returns for each pulse. It could be done with `dplyr` as well using `group_by() %>% mutate()` but we are proponent of using `data.table()`.
``` r
las@data[, d := Z[1]-Z[2], by = pulseID]
```
```{r, echo = FALSE}
u <- las@data[, .(d = Z[1]-Z[2]), by = pulseID]
las@data <- las@data[u, on = "pulseID"]
```
At this stage our data set contains distances twice. **1** associated with the first return and **2** with the last return. We only need a single value, so lets keep only first returns. Some distances are `NA` and correspond to pulses at the very edge of the file. We can discard those case for the moment.
```{r}
las <- filter_first(las)
las <- filter_poi(las, !is.na(d))
```
We can now plot the first returns colored by their distance to the last return.
```{r plot-las-dreturns, rgl = TRUE}
plot(las, bg = "white", color = "d", pal = viridis::magma(50), size = 4)
```
Now we can compute the average in an ABA (see @sec-aba).
```{r plot-raster-avg-dreturns, fig.width=6, fig.height=5}
d_first_last <- pixel_metrics(las, ~mean(d), 15)
plot(d_first_last, col = height.colors(50))
```
Now we have a working method, we can extend the idea to apply it over a broad coverage using the `LAScatalog` processing engine (@sec-engine). First we create a function that does this job. This is not mandatory but factorizing the code is good practice.
```{r}
grid_distance_first_last <- function(las, res) {
las <- retrieve_pulses(las)
las <- filter_firstlast(las)
las <- filter_poi(las, NumberOfReturns > 1)
las@data[, d := Z[1]-Z[2], by = pulseID]
las <- filter_first(las)
las <- filter_poi(las, !is.na(d))
return(pixel_metrics(las, ~mean(d), res))
}
```
To finish we use the `LAScatalog` processing engine with `catalog_map()`. We need to process with a buffer to avoid the case where pulses are separated in two files, so we use `opt_chunk_buffer(ctg) <- 2`. In addition, with the output being a raster, we ensure alignment of the chunk with the raster.
``` r
ctg <- readLAScatalog("folder/")
opt_chunk_buffer(ctg) <- 2
options <- list(raster_alignment = 15)
metrics <- catalog_map(ctg, grid_distance_first_last, res = 15, .options = options)
```
And we are done. Our innovative metric is computing with real-time monitoring. It can run in parallel on a single machine or in parallel on several machines. At the end, the output is a wall-to-wall raster map of the average distance between first and last returns for beams that return multiple returns. In 19 lines of code!
``` r
plot(metrics, col = height.colors(50))
```
```{r displaymap, echo=FALSE, fig.width=6.2, fig.height=6.5}
m <- raster::raster("data/chap12/distancefl.grd")
plot(m, col = height.colors(50))
raster::scalebar(500, type = "bar")
```
### Rumple index {#sec-outbox-rumple-index}
Another interesting metric that could be computed is the rumple index (see [Kane et al. (2008)](https://gis.ess.washington.edu/keck/ESS_421_additional_reading/Kane_Interpretation_and_topographic.pdf)).
To compute a rumple index we can triangulate points, measure the surface created, and divide by the projected surface. Rumple index is however expected to measure surface roughness of the canopy and it doesn't makes sense to triangulate every point. An option is to triangulate each first return but yet first returns are not all representative of the canopy. In the following example we are using the surface points every 1 m.
```{r}
grid_rumple_index <- function(las, res) { # user-defined function
las <- filter_surfacepoints(las, 1)
return(pixel_metrics(las, ~rumple_index(X, Y, Z), res))
}
```
Then the code is pretty much the same than in previous section
``` r
ctg <- readLAScatalog("folder/")
opt_chunk_buffer(ctg) <- 1
options <- list(raster_alignment = 20)
metrics <- catalog_map(ctg, grid_rumple_index, res = 20, .options = options)
plot(metrics, col = height.colors(50))
```
```{r, displaymap2, echo=FALSE, fig.width=6.2, fig.height=6.5}
m <- raster::raster("data/chap12/rumple.grd")
plot(m, col = height.colors(50))
raster::scalebar(500, type = "bar")
```
## Multi-spectral coloring {#sec-outbox-multispectral-coloring}
We have already seen a multi-spectral coloring example in @sec-pba-applications-coloring). This is truly a good example of how to think outside the box with `lidR`, so we show several variants here. The goal is to use multi-spectral data to generate false coloring and see how false coloring may be used for further analyses.
Multi-spectral ALS data are sampled with 3 devices each emitting a different wavelength. The point cloud is usually provided in the form of 3 `LAS` files, each file corresponding to a spectral wavelength. No matter the actual wavelength, we can consider the first band as blue, the second as red, and the third as green, and thus consider that each point has a pure color.
```{r plot-las-multispectral-otb, rgl = TRUE, message=FALSE}
f1 <- "data/chap11/PR1107_c1_ar_c.laz"
f2 <- "data/chap11/PR1107_c2_ar_c.laz"
f3 <- "data/chap11/PR1107_c3_ar_c.laz"
las <- readMSLAS(f1, f2, f3, filter = "-keep_z_below 300")
plot(las, color = "ScannerChannel", size = 6)
```
Each channel's returns vary in intensities as seen in the figure below because of the wavelength reflectance properties of the targets. A first step could be to normalize the intensities. For this example we won't do that to keep things simple.
```{r ggplot-hist-intensities}
library(ggplot2)
ggplot(las@data) +
aes(x = Intensity, fill = as.factor(ScannerChannel)) +
geom_density(alpha = 0.5) +
theme_minimal() +
theme(legend.position = c(.9, .90),
legend.title = element_blank())
```
In this specific example, notice that only few intensities are above 255, So intensity ranges from 0 to almost 255. This is very convenient because we will be able to use the raw values as color without transformation. We will simply force values above 255 to be 255. This is a dirty solution but fair enough for the example which is not about intensity normalization.
```{r}
las@data[Intensity > 255L, Intensity := 255L]
```
To finish, we define a function that takes the intensities and the associated channel as inputs. This function decomposes the 3 channels, computes the average intensity of each channel, and returns these 3 values that will later be interpreted as RGB. Also if one color is missing we force RGB to be NA.
```{r}
colorize <- function(intensity, channel)
{
# Split the intensities of each channel
i1 <- intensity[channel == 1L]
i2 <- intensity[channel == 2L]
i3 <- intensity[channel == 3L]
# If one channel is missing return RGB = NA
if (length(i1) == 0 | length(i2) == 0 | length(i3) == 0)
return(list(R = NA_integer_, G = NA_integer_, B = NA_integer_))
i1 <- as.integer(mean(i1))
i2 <- as.integer(mean(i2))
i3 <- as.integer(mean(i3))
return(list(R = i1, G = i2, B = i3))
}
```
Now we can use this function with different levels of regularization. We can use `colorize()` with `grid_metrics()` in an ABA. It returns a multi-layer raster that can be interpreted as RGB
```{r plot-raster-multispectral-rgb,fig.width=3}
rgb <- pixel_metrics(las, ~colorize(Intensity, ScannerChannel), 2)
terra::plotRGB(rgb)
```
We can also attribute an RGB color per voxel, discarding the voxels where `RGB = NA` by using `colorize()` with `voxel_metrics()` (see @sec-vba).
```{r plot-voxel-multispectral-rgb, rgl = TRUE, warning=FALSE}
rgb <- voxel_metrics(las, ~colorize(Intensity, ScannerChannel), 2)
rgb <- rgb[!is.na(R)] # Remove NAs
rgb <- LAS(rgb, las@header) # Convert to LAS for display
plot(rgb, color = "RGB", nbits = 8, size = 5)
```
To finish we can `colorize()` with `point_metrics()` (see @sec-pba). This has the advantage of maintaining much more points compared to the voxel-based version and to preserve coordinates.
```{r plot-las-multispectral-rgb-otb, rgl = TRUE}
rgb <- point_metrics(las, ~colorize(Intensity, ScannerChannel), r = 0.5)
las <- add_lasrgb(las, rgb$R, rgb$G, rgb$B)
colored <- filter_poi(las, !is.na(R)) # Remove NAs
plot(colored, color = "RGB", size = 4)
```