Skip to content

Commit

Permalink
Merge remote-tracking branch 'origin/master'
Browse files Browse the repository at this point in the history
  • Loading branch information
KonstiDE committed Nov 9, 2024
2 parents 292b1b2 + 84ddf87 commit 51a94d8
Show file tree
Hide file tree
Showing 19 changed files with 1,073 additions and 106 deletions.
7 changes: 4 additions & 3 deletions DESCRIPTION
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
Package: RStoolbox
Type: Package
Title: Remote Sensing Data Analysis
Version: 1.0.0
Date: 2024-04-24
Version: 1.0.1
Date: 2024-08-26
Authors@R: c(
person("Benjamin", "Leutner", role= "aut", email="[email protected]", comment = c(ORCID = "0000-0002-6893-2002")),
person("Ned", "Horning", role ="aut", email="[email protected]"),
Expand Down Expand Up @@ -34,6 +34,7 @@ Imports:
magrittr
Suggests:
randomForest,
lattice,
kernlab,
e1071,
gridExtra,
Expand All @@ -42,5 +43,5 @@ Suggests:
LinkingTo: Rcpp,
RcppArmadillo
License: GPL (>=3)
RoxygenNote: 7.3.1
RoxygenNote: 7.3.2
LazyData: true
15 changes: 12 additions & 3 deletions NEWS.md
Original file line number Diff line number Diff line change
@@ -1,5 +1,14 @@
# RStoolbox 1.0.1
Changes for publication in MEE

## New:
* `spectralIndices()` can now be customized. We made the indices available as an editable option in the RStoolbox package as suggested by a reviewer. Users now can append a custom index to it that will be then calculated via the C++ plus within spectralIndices.cpp.

## Changes:
* `mesma()` now does not throw an error anymore when given no models and the default value was too high

# RStoolbox 1.0.0
Updating MESMA, minor changes before major release
Minor changes before major release

## New:
* `mesma()` now better differentiates SMA and MESMA: For single endmember unmixing, each supplied endmember represents a class to unmix (row by row). For multiple endmemeber unmixing, the column `class` can be used to group endmembers by class. If multiple endmembers per class are provided, `mesma()` will compute a number of SMA (determined through the new argument `n_models`) for multiple endmember combinations drawn from endmembers and will select the best fit per pixel based on the lowest RMSE. See `?mesma` for details (fixes #57, reported by @ytarazona)
Expand All @@ -11,11 +20,11 @@ Updating MESMA, minor changes before major release
# RStoolbox 0.4.0
Rewrite of `RStoolbox`, migration from `raster` to `terra` and `sp` to `sf`

## New:
# New:
* RStoolbox moved on from the outdated `sp` and `raster` packages to `sf` and `terra` to ensure long term support of the tools.
* Thrown out unnecessary libraries

## Fixes:
# Fixes:
* `rasterPCA()`: Fixed a bug that caused the method and its unit tests to fail on Linux due to a corrupted covariance matrix calculated previously with `terra::layerCor()`
* `superClass()` unable to predict when there is NA in raster data (closes #102, reported by @bappa10085)

Expand Down
4 changes: 2 additions & 2 deletions R/RcppExports.R
Original file line number Diff line number Diff line change
Expand Up @@ -49,7 +49,7 @@ specSimC <- function(x, em) {
.Call('_RStoolbox_specSimC', PACKAGE = 'RStoolbox', x, em)
}

spectralIndicesCpp <- function(x, indices, redBand, blueBand, greenBand, nirBand, redEdge1Band, redEdge2Band, redEdge3Band, swir1Band, swir2Band, swir3Band, maskLayer, maskValue, L, s, G, C1, C2, Levi, swir2ccc, swir2cdiff, sf) {
.Call('_RStoolbox_spectralIndicesCpp', PACKAGE = 'RStoolbox', x, indices, redBand, blueBand, greenBand, nirBand, redEdge1Band, redEdge2Band, redEdge3Band, swir1Band, swir2Band, swir3Band, maskLayer, maskValue, L, s, G, C1, C2, Levi, swir2ccc, swir2cdiff, sf)
spectralIndicesCpp <- function(x, indices, redBand, blueBand, greenBand, nirBand, redEdge1Band, redEdge2Band, redEdge3Band, swir1Band, swir2Band, swir3Band, maskLayer, maskValue, L, s, G, C1, C2, Levi, swir2ccc, swir2cdiff, sf, formulas) {
.Call('_RStoolbox_spectralIndicesCpp', PACKAGE = 'RStoolbox', x, indices, redBand, blueBand, greenBand, nirBand, redEdge1Band, redEdge2Band, redEdge3Band, swir1Band, swir2Band, swir3Band, maskLayer, maskValue, L, s, G, C1, C2, Levi, swir2ccc, swir2cdiff, sf, formulas)
}

89 changes: 89 additions & 0 deletions R/internalFunctions.R
Original file line number Diff line number Diff line change
Expand Up @@ -255,7 +255,84 @@
#' On package startup
#' @noRd
.onLoad <- function(libname, pkgname){

L <- 0.5
G <- 2.5
L_evi <- 1
C1 <- 6
C2 <- 7.5
s <- 1
swir2ccc <- NULL
swir2coc <- NULL
NDVI <- NULL

.IDXDB <- list(
CLG = list(c("Gitelson2003", "Green-band Chlorophyll Index"),
function(redEdge3, green) {redEdge3/green - 1}),
CLRE = list(c("Gitelson2003", "Red-edge-band Chlorophyll Index"),
function(redEdge3, redEdge1) {redEdge3/redEdge1 - 1}),
CTVI = list(c("Perry1984", "Corrected Transformed Vegetation Index"),
function(red, nir) {(NDVI+.5)/sqrt(abs(NDVI+.5))} ),
DVI = list(c("Richardson1977", "Difference Vegetation Index"),
function(red, nir) {s*nir-red}),
EVI = list(c("Huete1999", "Enhanced Vegetation Index"),
function(red, nir, blue) {G * ((nir - red) / (nir + C1 * red - C2 * blue + L_evi))}),
EVI2 = list(c("Jiang 2008", "Two-band Enhanced Vegetation Index"),
function(red, nir) {G * (nir-red)/(nir + 2.4*red +1)}),
GEMI = list(c("Pinty1992", "Global Environmental Monitoring Index"),
function(red, nir) {(((nir^2 - red^2) * 2 + (nir * 1.5) + (red * 0.5) ) / (nir + red + 0.5)) * (1 - ((((nir^2 - red^2) * 2 + (nir * 1.5) + (red * 0.5) ) / (nir + red + 0.5)) * 0.25)) - ((red - 0.125) / (1 - red))}),
GNDVI = list(c("Gitelson1998", "Green Normalised Difference Vegetation Index" ),
function(green, nir) {(nir-green)/(nir+green)}),
KNDVI = list(c("Camps-Valls2021", "Kernel Normalised Difference Vegetation Index"),
function(red, nir) {tanh(((nir-red)/(nir+red)))^2}),
MCARI = list(c("Daughtery2000", "Modified Chlorophyll Absorption Ratio Index" ),
function(green, red, redEdge1) {((redEdge1-red)-(redEdge1-green))*(redEdge1/red)}),
MNDWI = list(c("Xu2006", "Modified Normalised Difference Water Index"),
function(green, swir2) {(green-swir2) / (green+swir2)}),
MSAVI = list(c("Qi1994", "Modified Soil Adjusted Vegetation Index" ),
function(red, nir) {nir + 0.5 - (0.5 * sqrt((2 * nir + 1)^2 - 8 * (nir - (2 * red))))}),
MSAVI2 = list(c("Qi1994", "Modified Soil Adjusted Vegetation Index 2" ),
function(red, nir) {(2 * (nir + 1) - sqrt((2 * nir + 1)^2 - 8 * (nir - red))) / 2}),
MTCI = list(c("DashAndCurran2004", "MERIS Terrestrial Chlorophyll Index"),
function(red, redEdge1, redEdge2) {(redEdge2-redEdge1)/(redEdge1-red)}),
NBRI = list(c("Garcia1991", "Normalised Burn Ratio Index"),
function(nir, swir3) { (nir - swir3) / (nir + swir3)}),
NDREI1 = list(c("GitelsonAndMerzlyak1994", "Normalised Difference Red Edge Index 1"),
function(redEdge2, redEdge1) {(redEdge2-redEdge1)/(redEdge2+redEdge1)}),
NDREI2 = list(c("Barnes2000", "Normalised Difference Red Edge Index 2"),
function(redEdge3, redEdge1) {(redEdge3-redEdge1)/(redEdge3+redEdge1)}),
NDVI = list(c("Rouse1974", "Normalised Difference Vegetation Index"),
function(red, nir) {(nir-red)/(nir+red)}),
NDVIC = list(c("Nemani1993", "Corrected Normalised Difference Vegetation Index" ),
function(red, nir, swir2) {(nir-red)/(nir+red)*(1-((swir2 - swir2ccc)/(swir2coc-swir2ccc)))}),
NDWI = list(c("McFeeters1996", "Normalised Difference Water Index"),
function(green, nir) {(green - nir)/(green + nir)}),
NDWI2 = list(c("Gao1996", "Normalised Difference Water Index"),
function(nir, swir2) {(nir - swir2)/(nir + swir2)}),
NRVI = list(c("Baret1991", "Normalised Ratio Vegetation Index" ),
function(red, nir) {(red/nir - 1)/(red/nir + 1)}),
REIP = list(c("GuyotAndBarnet1988", "Red Edge Inflection Point"),
function(red, redEdge1, redEdge2, redEdge3) {0.705+0.35*((red+redEdge3)/(2-redEdge1))/(redEdge2-redEdge1)}),
RVI = list(c("", "Ratio Vegetation Index"),
function(red, nir) {red/nir}),
SATVI = list(c("Marsett2006", "Soil Adjusted Total Vegetation Index"),
function(red, swir2, swir3) {(swir2 - red) / (swir2 + red + L) * (1 + L) - (swir3 / 2)}),
SAVI = list(c("Huete1988", "Soil Adjusted Vegetation Index"),
function(red, nir) {(nir - red) * (1+L) / (nir + red + L)}),
SLAVI = list(c("Lymburger2000", "Specific Leaf Area Vegetation Index"),
function(red, nir, swir2) {nir / (red + swir2)}),
SR = list(c("Birth1968", "Simple Ratio Vegetation Index"),
function(red, nir) {nir / red}),
TTVI = list(c("Thiam1997", "Thiam's Transformed Vegetation Index"),
function(red, nir) {sqrt(abs((nir-red)/(nir+red) + 0.5))}),
TVI = list(c("Deering1975", "Transformed Vegetation Index"),
function(red, nir) {sqrt((nir-red)/(nir+red)+0.5)}),
WDVI = list(c("Richardson1977", "Weighted Difference Vegetation Index"),
function(red, nir) {nir - s * red})
)

if(is.null(getOption("RStoolbox.verbose"))) options(RStoolbox.verbose = FALSE)
if(is.null(getOption("RStoolbox.idxdb"))) options(RStoolbox.idxdb = .IDXDB)
}

#' Init verbosity within functions
Expand All @@ -270,6 +347,18 @@
options(RStoolbox.verbose = verbose)
}


#' Init spectralIndices within the index DB
#'
#' will restore global options after function has been called
#' @param spectralIndices List
#' @keywords internal
#' @noRd
.initIDXdb <- function(idxdb){
idxbg <- force(getOption("RStoolbox.idxdb"))
do.call("on.exit", list(substitute(options(RStoolbox.idxdb = idxbg))), envir=parent.frame())
options(RStoolbox.idxdb = idxdb)
}


#' Clean up on package unload
Expand Down
15 changes: 13 additions & 2 deletions R/mesma.R
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@
#' }
#' @param iterate Integer. Set maximum iteration per pixel. Processing time could increase the more iterations are made possible. Default is 400.
#' @param tolerance Numeric. Tolerance limit representing a nearly zero minimal number. Default is 1e-8.
#' @param n_models Logical. Only applies if \code{em} contains column \code{class}. Defines how many endmember combinations should be picked. Maximum is the minimum number of endmembers of a class. Defaults to 5.
#' @param n_models Logical. Only applies if \code{em} contains column \code{class}. Defines how many endmember combinations should be picked. Maximum is the minimum number of endmembers of a class. Defaults to the amount of rows of em.
#' @param sum_to_one Logical. Defines whether a sum-to-one constraint should be applied so that probabilities of endmember classes sum to one (a constraint not covered by NNLS) to be interpretable as fractions. Defaults to \code{TRUE}. To get actual NNLS results, change to \code{FALSE}.
#' @param verbose Logical. Prints progress messages during execution.
#' @param ... further arguments passed to \link[terra]{writeRaster}.
Expand Down Expand Up @@ -72,7 +72,7 @@
#' @importFrom terra which.min rast selectRange nlyr
#' @export
#'
mesma <- function(img, em, method = "NNLS", iterate = 400, tolerance = 0.00000001, n_models = 5, sum_to_one = TRUE, ..., verbose){
mesma <- function(img, em, method = "NNLS", iterate = 400, tolerance = 0.00000001, n_models = NULL, sum_to_one = TRUE, ..., verbose){
img <- .toTerra(img)
## messages
if(!missing("verbose")) .initVerbose(verbose)
Expand All @@ -89,6 +89,12 @@ mesma <- function(img, em, method = "NNLS", iterate = 400, tolerance = 0.0000000
if(anyNA(em)){
stop("'em' is not allowed to contain NA values. Spectra must be consistent.")
}

# Check for n_models type
if(!is.null(n_models) && !is.numeric(n_models)){
stop("'n_models' needs to be an 'numeric' object.")
}


method <- toupper(method)
meth_avail <- c("NNLS") #available methods
Expand All @@ -115,6 +121,11 @@ mesma <- function(img, em, method = "NNLS", iterate = 400, tolerance = 0.0000000
classes <- unique(em$class)
# check n_models
n_em_cl <- min(sapply(classes, function(x) nrow(em[em$class == x,]), USE.NAMES = F))

if(is.null(n_models)){
n_models <- n_em_cl
}

if(n_em_cl < n_models){
warning(paste0("'n_models' cannot be larger than the minimum number of provided endmembers per class. Setting 'n_models' to ", n_em_cl, "."))
n_models <- n_em_cl
Expand Down
12 changes: 9 additions & 3 deletions R/rsOpts.R
Original file line number Diff line number Diff line change
Expand Up @@ -3,14 +3,20 @@
#' shortcut to options(RStoolbox.*)
#'
#' @param verbose Logical. If \code{TRUE} many functions will print status messages about the current processing step. By default verbose mode is disabled.
#' @param idxdb List. The list conatins the formal calculation of spectral indices. Modify this list to pipe your own spectral index through the internal C++ calculation of RStoolbox.
#' @export
#' @return
#' No return, just a setter for the verbosiness of the RStoolbox package
#' No return, just a setter for the verbosiness and the index-database of the RStoolbox package. For latter, see the example of Rstoolbox::spectralIndices()
#' @examples
#' rsOpts(verbose=TRUE)
#'
rsOpts <- function(verbose){
options(RStoolbox.verbose=verbose)
rsOpts <- function(verbose=NULL, idxdb=NULL){
if(!is.null(verbose)){
options(RStoolbox.verbose=verbose)
}
if(!is.null(idxdb)){
options(RStoolbox.idxdb=idxdb)
}
}


Loading

0 comments on commit 51a94d8

Please sign in to comment.