R/sp_from_data_frame.R

Defines functions matrix_to_sp_polygon data_frame_to_polygons data_frame_to_lines data_frame_to_points sp_from_data_frame

Documented in sp_from_data_frame

#' Function to promote a data frame to a SpatialPointsDataFrame, 
#' SpatialLinesDataFrame, or SpatialPolygonsDataFrame.
#' 
#' @param df Data frame to be converted into spatial data frame.  
#' 
#' @param type Type of geometry. Type must be one of \code{"points"}, 
#' \code{"lines"}, or \code{"polygons"}. 
#' 
#' @param latitude \code{df}'s latitude variable name. 
#' 
#' @param longitude \code{df}'s longitude variable name.
#' 
#' @param projection \code{df}'s latitude and longitude projection system. 
#' Default is WGS84.
#' 
#' @param keep For when type is \code{"points"}, should latitude and longitude
#' be kept in the SpatialPointsDataFrame's data slot? 
#' 
#' @param id Variable in \code{df} which codes for spatial object's id. This is
#' used when a data frame contains many separate geometries. \code{id} is not 
#' used for points because each point is a separate geometry. 
#' 
#' @param warn Should the function raise a warning when observations are 
#' removed? 
#' 
#' @author Stuart K. Grange
#' 
#' @return A SpatialPointsDataFrame, SpatialLinesDataFrame, or 
#' SpatialPolygonsDataFrame.
#' 
#' @seealso \code{\link{sp_from_wkt}}
#' 
#' @examples 
#' \dontrun{
#' 
#' # Promote to different geometry types
#' # Points
#' sp_points <- sp_from_data_frame(data_drawn, type = "points")
#' 
#' # Lines
#' sp_lines <- sp_from_data_frame(data_drawn, type = "lines")
#' # Lines with seperate geometries
#' sp_lines <- sp_from_data_frame(data_drawn, type = "lines", id = "name")
#' 
#' # Polygons
#' sp_polygons <- sp_from_data_frame(data_drawn, type = "polygons")
#' # Polygons with seperate geometries
#' sp_polygons <- sp_from_data_frame(data_drawn, type = "polygons", id = "name")
#' 
#' }
#' 
#' @export
sp_from_data_frame <- function(df, type = "points", latitude = "latitude", 
                               longitude = "longitude", 
                               projection = projection_wgs84(), 
                               keep = FALSE, id = NA, warn = TRUE) {
  
  #  Check and parse
  if (length(type) != 1) stop("'type' must have a length of one.", call. = FALSE)
  
  # Make plurals
  type <- stringr::str_trim(type)
  type <- stringr::str_to_lower(type)
  type <- if_else(type == "point", "points", type)
  type <- if_else(type == "line", "lines", type)
  type <- if_else(type == "polygon", "polygons", type)
  
  # Check
  if (!grepl("points|lines|polygons", type)) {
    stop("`type` must be one of `points`, `lines`, or `polygons`.", call. = FALSE)
  }
  
  # Promote to spatial data
  if (type == "points") {
    sp <- data_frame_to_points(df, latitude, longitude, projection, keep, warn)
  } else if (type == "lines") {
    sp <- data_frame_to_lines(df, latitude, longitude, projection, id, warn)
  } else if (type == "polygons") {
    sp <- data_frame_to_polygons(df, latitude, longitude, projection, id, warn)
  }
  
  # Ensure data slot is a data.frame, not a tibble
  if (grepl("Data", sp_class(sp)) && tibble::is_tibble(sp@data)) {
    sp@data <- data.frame(sp@data)
  }
  
  return(sp)
  
}


data_frame_to_points <- function(df, latitude, longitude, projection, 
                                 keep = FALSE, warn) {
  
  # NA check, if NAs drop them
  if (any(is.na(c(df[, latitude, drop = TRUE], df[, longitude, drop = TRUE])))) {
    
    # Remove NAs
    df <- df[!(is.na(df[, latitude, drop = TRUE]) | is.na(df[, longitude, drop = TRUE])), ]
    
    # Raise a warning
    if (warn) {
      warning(
        "Missing coordinates were detected and have been removed...", 
        call. = FALSE
      )
    }
    
    # Check if all observations have been lost
    if (nrow(df) == 0) stop("There are no valid coordinates.", call. = FALSE)
    
  }
  
  # Reassign, df is used again
  sp <- df
  
  # Make sp points object
  coordinates(sp) <- c(longitude, latitude)
  
  # Put vectors back in data slot
  if (keep) {
    sp@data[, latitude] <- df[, latitude, drop = TRUE]
    sp@data[, longitude] <- df[, longitude, drop = TRUE]
  }
  
  # Give the object a projection
  if (!is.na(projection)) sp <- sp_transform(sp, projection, warn = FALSE)
  
  return(sp)
  
}


data_frame_to_lines <- function(df, latitude, longitude, projection, id, warn) {
  
  # Make an identifier variable for lines
  if (is.na(id)) {
    # Single line object, no grouping
    df[, "id"] <- 1
  } else {
    # Use input variable
    df[, "id"] <- df[, id]
  }
  
  # Get data part for the SpatialLinesDataFrame
  data_extras <- dplyr::distinct(df, id, .keep_all = TRUE)
  
  # Drop the coordinates from extra data frame
  data_extras[, c(latitude, longitude)] <- NULL
  
  # Make sp points object
  sp_object <- data_frame_to_points(
    df, 
    latitude, 
    longitude, 
    projection,
    keep = FALSE,
    warn = warn
  )
  
  # From
  # http://stackoverflow.com/questions/24284356/convert-spatialpointsdataframe-
  # to-spatiallinesdataframe-in-r
  # Generate lines for each id
  lines <- lapply(
    split(sp_object, sp_object$id), 
    function(x) Lines(list(Line(coordinates(x))), x$id[1L])
  )
  
  # Drop
  if (!is.na(id)) data_extras[, "id"] <- NULL
  
  # Create SpatialLines
  sp <- SpatialLines(lines)
  
  # Make SpatialLinesDataFrame
  sp <- SpatialLinesDataFrame(sp, data_extras, match.ID = FALSE)
  
  # Give the object a projection
  if (!is.na(projection)) sp <- sp_transform(sp, projection, warn = FALSE)
  
  return(sp)
  
}


data_frame_to_polygons <- function(df, latitude, longitude, projection, id,
                                   warn) {
  
  # Make an identifier variable for lines
  if (is.na(id)) {
    # Single line object, no grouping
    df[, "id"] <- 1
  } else {
    # Use input variable
    df[, "id"] <- df[, id]
  }
  
  # Get data part for the SpatialLinesDataFrame
  data_extras <- dplyr::distinct(df, id, .keep_all = TRUE)
  
  # Drop the coordinates from extra data frame
  data_extras[, c(latitude, longitude)] <- NULL
  
  # Make sp points object
  sp <- data_frame_to_points(
    df, 
    latitude, 
    longitude, 
    projection,
    keep = FALSE,
    warn = warn
  )
  
  # A list element will represent each group within a feature 
  # Longitude latitude order is important
  coordinates <- plyr::dlply(
    df, 
    "id", 
    function(x) data.matrix(x[, c(longitude, latitude)])
  )
  
  # Make polygons
  sp <- lapply(
    seq_along(coordinates), 
    function(x) matrix_to_sp_polygon(coordinates[x], x)
  )
  
  # Bind geometries
  sp <- do.call(rbind, sp)
  
  # Make sp data frame
  sp <- SpatialPolygonsDataFrame(sp, data_extras)
  
  # Give the object a projection
  if (!is.na(projection)) sp <- sp_transform(sp, projection, warn = FALSE)
  
  return(sp)
  
}


matrix_to_sp_polygon <- function(matrix, id) {
  
  # Matrix to polygon
  polygon <- Polygon(matrix)
  
  # Polygon to polygons
  polygon <- Polygons(list(polygon), id)
  
  # To spatial polygons
  sp <- SpatialPolygons(list(polygon))
  
  return(sp)
  
}
skgrange/gissr documentation built on Feb. 24, 2024, 2:55 p.m.