R/estimatePSCS.R

Defines functions histosol.surface.tier histosol.control.section estimatePSCS

Documented in estimatePSCS

#estimatePSCS()
#' Estimate boundaries of the U.S Soil Taxonomy Particle Size Control Section
#' 
#' Estimates the upper and lower boundary of the particle size control section
#' for Mineral or Organic soilsby applying a programmatic version of the 
#' particle size control section key from the Keys to Soil Taxonomy (13th edition).
#' See details for assumptions and required data elements.
#' 
#' @details
#'
#' Requires information to identify argillic horizons (clay contents, horizon
#' designations) with \code{getArgillicBounds()} as well as the presence of
#' plow layers and surface organic soil material. Any
#' \code{getArgillicBounds()} arguments may be passed to \code{estimatePSCS}.
#'
#' Also, requires information on taxonomic order to handle Andisols.
#' 
#' In aqp 2.1, particle size control sections of organic soils Histosols and
#' Histels are supported. This requires setting the `"lieutex"` horizon metadata
#' column using `hzmetaname<-()` Horizon designations containing `"f"` or `"W"`
#' are  recognized as permafrost and water layers, respectively, for application 
#' of the organic soils key to control sections. In lieu textures `"SPM"` and
#' `"PEAT"` are used to identify low density organic materials used for surface
#' tier thickness. To avoid using the 160 cm surface tier, set the `"lieutex"` 
#' column to any column that does not contain `"SPM"` or `"PEAT"` values.
#'
#' WARNING: Soils in arenic or grossarenic subgroups, with fragipans, or with
#' strongly contrasting PSCs may not be classified correctly. The author would
#' welcome a dataset to develop this functionality for.
#' 
#' @param p A SoilProfileCollection 
#' @param hzdesgn Name of the horizon attribute containing the horizon
#' designation. Default `hzdesgnname(p, required = TRUE)`
#' @param clay.attr Name of the horizon attribute containing clay contents.
#' Default `hzmetaname(p, "clay", required = TRUE)`
#' @param texcl.attr Name of the horizon attribute containing textural class
#' (used for finding sandy textures). Default `hztexclname(p, required = TRUE)`
#' @param tax_order_field Name of the site attribute containing taxonomic
#' order; for handling PSCS rules for Andisols in lieu of lab data. May be NA
#' or column missing altogether, in which case Andisol PSC possibility is
#' ignored.
#' @param lieutex Optional data element used in addition to the horizon
#' designation to identify kinds of organic soil material for soils with organic 
#' surfaces. Default: `hzmetaname(p, "lieutex")`
#' @param bottom.pattern Regular expression pattern to match a root-restrictive
#' contact. Default matches Cr, R, Cd or m. This argument is passed to both
#' `minDepthOf()` and `getArgillicBounds()`.
#' @param simplify Return a length 2 vector with upper and lower boundary when 
#' p has length 1? Default TRUE.
#' @param ...  additional arguments are passed to getArgillicBounds()
#' @return A numeric vector (when `simplify=TRUE`) containing the top and bottom 
#' depth of the particle
#' size control section. First value is top, second value is bottom. 
#' If `p` contains more than one profile, the result is a data.frame with 
#' profile ID plus PSCS top and bottom depths.
#' @author Andrew Gene Brown
#' @seealso [`getArgillicBounds()`], [`getSurfaceHorizonDepth()`]
#' @references Soil Survey Staff. 2014. Keys to Soil Taxonomy, 12th ed.
#' USDA-Natural Resources Conservation Service, Washington, DC.
#' @keywords manip
#' @export 
#' @examples
#'
#' data(sp1, package = 'aqp')
#' depths(sp1) <- id ~ top + bottom
#' site(sp1) <- ~ group
#'
#' # set required metadata
#' hzdesgnname(sp1) <- 'name'
#' hztexclname(sp1) <- 'texture'
#' hzmetaname(sp1, 'clay') <- 'prop'
#' 
#' x <- estimatePSCS(sp1)
#' x
#' 
#' # change horizon texture and set inlieu texture column to turn
#' # first profile into an organic soil
#' sp1$name[1:6] <- c("Oi1", "Oi2", "Oi3", "Oaf", "Cf1", "Cf2")
#' sp1$texture <- as.character(sp1$texture)
#' sp1$texture[1:6] <- c("PEAT", "PEAT", "PEAT", "MUCK", "GRVLS", "GRVLS")
#' sp1$bottom[6] <- 200
#' hzmetaname(sp1, 'lieutex') <- 'texture'
#' 
#' y <- estimatePSCS(sp1[1, ], simplify = FALSE)
#' 
#' # 74cm lower boundary is 25cm past the upper boundary of permafrost (49cm)
#' # but minimum depth is 100cm unless there is a root-limiting layer
#' y
#' 
estimatePSCS <- function(
    p,
    hzdesgn = hzdesgnname(p, required = TRUE),
    clay.attr = hzmetaname(p, "clay", required = TRUE),
    texcl.attr = hztexclname(p, required = TRUE),
    lieutex = hzmetaname(p, "lieutex"),
    tax_order_field = "tax_order",
    bottom.pattern = 'Cr|R|Cd|m',
    simplify = TRUE,
    ...
) {
  
  .LAST <- NULL
  hz.depths <- horizonDepths(p)

  attr.len <- unlist(lapply(c(hzdesgn, clay.attr, texcl.attr), length))
  if (any(attr.len > 1))
    stop("horizon designation, clay attribute or texture class attribute must have length 1")
  
  if (is.null(hzdesgn) || !hzdesgn %in% horizonNames(p)) {
    stop("Horizon designation column (", hzdesgn, ") does not exist.")
  }
  
  if (is.null(texcl.attr) || !texcl.attr %in% horizonNames(p)) {
    stop("Horizon texture class column (", texcl.attr, ") does not exist.")
  }
  
  if (is.null(clay.attr) | (!clay.attr %in% horizonNames(p))) {
    stop("Horizon clay content column (", clay.attr, ") does not exist.")
  }

  soildepth <- minDepthOf(p, hzdesgn = hzdesgn, pattern = bottom.pattern, simplify = FALSE)[[hz.depths[1]]]
  nobottomidx <- is.na(soildepth)
  if (any(nobottomidx)) {
    soildepth[nobottomidx] <- p[which(nobottomidx), , .LAST][[hz.depths[2]]]
  }
  
  andisols_flag <- rep(FALSE, length(soildepth))
  shallow_flag <- rep(FALSE, length(soildepth))

  # Parts D (argillic starts >100cm  depth) and F (all other mineral soils)
  default_t <- rep(25, length(soildepth))
  default_b <- rep(100, length(soildepth))
  
  odepth <- getMineralSoilSurfaceDepth(p, hzdesgn = hzdesgn, simplify = FALSE)[[hz.depths[2]]]
  othick <- thicknessOf(p, pattern = "O", hzdesgn = hzdesgn)$thickness
  ohzidx <- which(odepth > 0)
  
  # Key part A (soils with restriction in shallow depth)
  lt36idx <- which(soildepth - odepth <= 36)
  if (length(lt36idx) > 0) {
    default_t[lt36idx] <- 0 # O horizon correction applied below
    default_b[lt36idx] <- soildepth[lt36idx]
    shallow_flag[lt36idx] <- TRUE
  }
  
  # Key part B (Andisols)
  if (tax_order_field %in% siteNames(p)) {
    andidx <- grep("[Aa]ndisols", p[[tax_order_field]])
    if (length(andidx) > 0) {
      default_t[andidx] <- 0
      default_b[andidx] <- 100
      andisols_flag[andidx] <- TRUE
    }
  }

  # Adjust PSCS range downward if organic soil material is present at surface (i.e. mineral soil surface depth > 0)
  if (length(ohzidx) > 0) {
    
    default_t[ohzidx] <- default_t[ohzidx] + odepth[ohzidx]
    maxdtruncidx <- which(default_b != soildepth)
    
    if (length(maxdtruncidx) > 0) {
      default_b[maxdtruncidx] <- default_b[maxdtruncidx] + odepth[maxdtruncidx]
    }
  }

  # Key parts C and E (has argillic/kandic/natric WITHIN 100CM)

  if (any(!andisols_flag)) {
    argillic_bounds <- getArgillicBounds(
      p,
      clay.attr = clay.attr,
      texcl.attr = texcl.attr,
      hzdesgn = hzdesgn,
      bottom.pattern = bottom.pattern,
      ...,
      simplify = FALSE # always return data.frame
    )
    
    lt100argidx <- which(argillic_bounds$ubound < 100)
    thinargidx <- argillic_bounds$lbound - argillic_bounds$ubound <= 50
    shallowargidx <- which(argillic_bounds$lbound <= 25)
    
    if (length(lt100argidx) > 0) {
      default_t[lt100argidx] <- argillic_bounds$ubound[lt100argidx]
    }
    
    if (length(which(thinargidx)) > 0) {
      default_b[which(thinargidx)] <- argillic_bounds$lbound[which(thinargidx)]
    }
    
    if (length(which(!thinargidx)) > 0) {
      default_b[which(!thinargidx)] <- argillic_bounds$ubound[which(!thinargidx)] + 50
    }
    
    if (length(shallowargidx) > 0) {
      default_b[shallowargidx] <- 100
    }
  }

  # Adjust PSCS top depth to bottom of plow layer (if appropriate)
  plow_layer_depth <- getPlowLayerDepth(p, hzdesgn, simplify = FALSE)[[hz.depths[2]]]
  plowidx <- which(plow_layer_depth >= 25 + odepth)
  if (length(plowidx) > 0) {
    default_t[plowidx] <- plow_layer_depth[plowidx]
  }
  
  # Adjust PSCS top depth to mineral soil surface for soils <36cm to restriction
  shallowoidx <- which(shallow_flag & default_t != 0)
  if (length(shallowoidx) > 0) {
    default_t[shallowoidx] <- odepth[shallowoidx]
  }
  
  # handle thick o horizons that cause the mineral PSCS to be greater than soil depth
  organic.idx <- which(default_t >= soildepth | odepth >= 40 |
                       (othick >= (soildepth * 2 / 3) & (soildepth - othick) <= 10))
  if (length(organic.idx) > 0) {
    organic_cs <- histosol.control.section(p[organic.idx,], hzdesgn = hzdesgn, lieutex = lieutex)
    default_t[organic.idx] <- organic_cs[[2]]
    default_b[organic.idx] <- organic_cs[[3]]
  }
  
  # Adjust PSCS bottom depth to restriction depth, if appropriate
  restridx <- which(soildepth < default_b)
  if (length(restridx) > 0) { #truncate to restriction
    default_b[restridx] <- soildepth[restridx]
  }
  
  if (length(default_t) == 1 && simplify) {
    return(as.numeric(c(default_t, default_b)))
  }
  
  res <- data.frame(id = profile_id(p), pscs_top = default_t, pscs_bottom = default_b)
  names(res)[1] <- idname(p)
  res
}

# calculate control section based on materials in surface tier, root limiting layers, permafrost and water
histosol.control.section <- function(p,
                                     hzdesgn = hzdesgnname(p, required = TRUE),
                                     lieutex = hzmetaname(p, "lieutex", required = TRUE)) {
  depthz <- horizonDepths(p)
  
  # control section stops at Cr|R|Cd
  rll.depth <- minDepthOf(p, pattern = "Cr|R|Cd|m", hzdesgn = hzdesgn, no.contact.depth = 0, no.contact.assigned = 200, simplify = FALSE)[[depthz[1]]]
  
  # control section stops at permafrost + 25cm (must be at least 100cm)
  permafrost.depth <- minDepthOf(p, "f|ff", hzdesgn = hzdesgn, no.contact.assigned = Inf, simplify = FALSE)[[depthz[1]]]
  
  # control section stops at water bottom depth  greater than either 130 or 160 cm, as applicable
  water.depth <- minDepthOf(p, "W", hzdesgn = hzdesgn, no.contact.assigned = Inf, simplify = FALSE)[[depthz[1]]]
  
  # get surface tier thickness for p
  hst <- histosol.surface.tier(p, hzdesgn = hzdesgn, lieutex = lieutex, rll = rll.depth)$hst_b
  
  data.frame(
    peiid = profile_id(p),
    hcs_t = 0,
    hcs_b = pmin(rll.depth, pmax(permafrost.depth + 25, 100), pmin(water.depth, ifelse(hst == 30, 130, 160)))
  )
  
}

histosol.surface.tier <- function(p,
                                  hzdesgn = hzdesgnname(p, required = TRUE),
                                  lieutex = hzmetaname(p, "lieutex", required = TRUE),
                                  rll = NULL) {
    
  .SD <- NULL; .LAST <- NULL; .BOTTOM <- NULL; .TOP <- NULL; .thk <- NULL
  depthz <- horizonDepths(p)
  
  res <- data.frame(
    peiid = profile_id(p),
    hst_t = 0,
    hst_b = 30
  )
  
  # column "lieutex" used in lieu of texture, and in lieu of horizon designations (Oa, Oe, Oi)
  if (!lieutex %in% names(p)) {
    # this will only trigger if one or pedons appears to meet the organic layer thickness requirements
    stop("The in lieu texture ('lieutex') is required to apply rules for organic soils. 
    Set the 'lieutex' horizon designation metadata with `hzmetaname(p, 'lieutex') <- 'lieutex'`
    Soils with low density fibric materials are indicated using lieutex column values `'PEAT'` or `'SPM'`.")
  }
  
  mss <- getMineralSoilSurfaceDepth(p, hzdesgn = hzdesgn, simplify = FALSE)
  
  # if there is a mineral surface,  no surface tier
  noo.idx <- which(mss[[depthz[2]]] == 0)
  
  psub <- p
  if (length(noo.idx) > 0) {
    psub <- p[-noo.idx, ]
  }
  
  if (is.null(rll)) {
    rll <- minDepthOf(psub, hzdesgn = hzdesgn, pattern = "Cr|Cd|R|m", no.contact.depth = 0, no.contact.assigned = 200, simplify = FALSE)[[depthz[1]]]
  }
  
  p60 <- glom(psub, 0, pmin(rll, 60), truncate = TRUE, drop = FALSE)
  
  # allow for multiple lines of evidence for fibric/low density OM
  #  either lieutex of peat or "i" suffix throughout all horizons in upper 60
  
  hz <- data.table::data.table(horizons(p60))
  idx <- which(grepl("i", hz[[hzdesgn]]) | tolower(hz[[lieutex]]) %in% c("peat", "spm"))
  hz$.thk <- 0
  hz$.thk[idx] <- (hz[[depthz[2]]] - hz[[depthz[1]]])[idx]
  
  # calculate sum of fibric material thickness within upper 60cm, compare to total hz thickness
  fibric.thickness <- hz[, sum(.thk, na.rm = TRUE), by = c(idname(p))]$V1
  total.hz.thickness <- p60[, , .LAST, .BOTTOM] - p60[, 1, .TOP]
  
  # if all fibric in upper 60, the limit is 60
  res$hst_b[fibric.thickness >= 60] <- 60
  
  return(res)
}

Try the aqp package in your browser

Any scripts or data that you put into this service are public.

aqp documentation built on April 4, 2025, 4:15 a.m.