#' @title Resolve a Britsh or Irish National Grid Reference.
#' @description Convert a Britsh or Irish National Grid Reference to a coordinate pair.
#' @param ngr A character string with a British National Grid Reference
#' @param centre Return the coordinate of the origin of the local square (default) or the centre.
#' @param grid Input grid system (currently British or Irish)
#' @return A numeric coordinate pair
#' @examples
#' ngrResolve("SJ456782")
#' @export
ngrResolve <- function(ngr, centre=FALSE, grid="British"){
if (!is.na(ngr)){
ngr <- gsub(pattern="[[:space:]]+", replacement="", ngr)
lets <- substr(ngr,1,2)
if (grid=="British"){
nums <- substr(ngr,3,nchar(ngr))
prefixes <- data.frame(L=c("HP", "HT", "HU", "HY", "HZ", "NA", "NB", "NC", "ND", "NF", "NG", "NH", "NJ", "NK", "NL", "NM", "NN", "NO", "NR", "NS", "NT", "NU", "NW", "NX", "NY", "NZ", "SC", "SD", "SE", "SH", "SJ", "SK", "SM", "SN", "SO", "SP", "SR", "SS", "ST", "SU", "SV", "SW", "SX", "SY", "SZ", "TA", "TF", "TG", "TL", "TM", "TQ", "TR", "TV"), X=c("4", "3", "4", "3", "4", "0", "1", "2", "3", "0", "1", "2", "3", "4", "0", "1", "2", "3", "1", "2", "3", "4", "1", "2", "3", "4", "2", "3", "4", "2", "3", "4", "1", "2", "3", "4", "1", "2", "3", "4", "0", "1", "2", "3", "4", "5", "5", "6", "5", "6", "5", "6", "5"), Y=c("12", "11", "11", "10", "10", "9", "9", "9", "9", "8", "8", "8", "8", "8", "7", "7", "7", "7", "6", "6", "6", "6", "5", "5", "5", "5", "4", "4", "4", "3", "3", "3", "2", "2", "2", "2", "1", "1", "1", "1", "0", "0", "0", "0", "0", "4", "3", "3", "2", "2", "1", "1", "0"))
} else if (grid=="Irish"){
nums <- substr(ngr,2,nchar(ngr))
nums[isOdd(nchar(nums))] <- paste(0,nums[isOdd(nchar(nums))],sep="")
northings <- data.frame(L=c("A","B","C","D","E","F","G","H","J","K","L","M","N","O","P","Q","R","S","T","U","V","W","X","Y","Z"),Y=rep(c(4:0),each=5))
eastings <- data.frame(L=c("A","B","C","D","E","F","G","H","J","K","L","M","N","O","P","Q","R","S","T","U","V","W","X","Y","Z"),X=rep(c(0:4),times=5))
prefixes <- merge(eastings, northings, by="L")
}
epre <- as.character(prefixes$X[prefixes$L==lets])
npre <- as.character(prefixes$Y[prefixes$L==lets])
w <- nchar(nums) / 2
e <- substr(nums,1,w)
n <- substr(nums, w+1, nchar(nums))
e <- paste(epre,e,sep="")
n <- paste(npre,n,sep="")
if (centre) mv <- "5" else mv <- "0"
if (w==1){
e <- as.numeric(paste(e,mv,"000",sep=""))
n <- as.numeric(paste(n,mv,"000",sep=""))
} else if (w==2){
e <- as.numeric(paste(e,mv,"00",sep=""))
n <- as.numeric(paste(n,mv,"00",sep=""))
} else if (w==3){
# six fig ref
e <- as.numeric(paste(e,mv,"0",sep=""))
n <- as.numeric(paste(n,mv,"0",sep=""))
} else if (w==4){
e <- as.numeric(paste(e,mv,sep=""))
n <- as.numeric(paste(n,mv,sep=""))
} else if (w==5){
e <- as.numeric(paste(e,mv,sep=""))/10
n <- as.numeric(paste(n,mv,sep=""))/10
} else if (w==6){
e <- as.numeric(paste(e,mv,sep=""))/100
n <- as.numeric(paste(n,mv,sep=""))/100
} else if (w==7){
e <- as.numeric(paste(e,mv,sep=""))/1000
n <- as.numeric(paste(n,mv,sep=""))/1000
} else if (w==8){
e <- as.numeric(paste(e,mv,sep=""))/10000
n <- as.numeric(paste(n,mv,sep=""))/10000
} else {
e <- NA
n <- NA
}
return(c(e,n))
} else {
return(NA)
}
}
#' @title Change the resolution of a British National Grid Reference.
#' @description Increase of decrease the resolution of a British National Grid Reference.
#' @param x A character string with a British National Grid Reference
#' @param figures The desired precision (an even whole number)
#' @return A character string with a new British National Grid Reference
#' @examples
#' x <- "SJ456782"
#' ngrResolution(x,4)
#' ngrResolution(x,8)
#' ngrResolution(x,7)
#' @export
ngrResolution <- function(x, figures){
if (figures%%1!=0 | figures < 0 | figures %% 2 != 0){
stop("The figures argument must be an even, whole number greater than or equal to 0.")
}
prefix <- substr(x,1,2)
suffix <- substr(x,3,nchar(x))
if (nchar(suffix) %% 2 == 0){
xcomp <- substr(suffix,1,nchar(suffix)/2)
ycomp <- substr(suffix,(nchar(suffix)/2)+1,nchar(suffix))
if (figures <= nchar(suffix)){
newxcomp <- substr(xcomp,1, figures/2)
newycomp <- substr(ycomp,1, figures/2)
newngr <- paste(prefix, newxcomp, newycomp, sep="")
} else {
add0s <- (figures - nchar(suffix))/2
add0s <- paste0(rep("0",add0s), collapse="")
newngr <- paste(prefix, xcomp, add0s, ycomp, add0s, sep="")
}
} else {
stop("Grid reference has an incorrect (odd) number of figures.")
}
return(newngr)
}
#' @title Remove trailing zeroes in National Grid Reference
#' @description Remove trailing zeroes in National Grid Reference when present in both x and y
#' @param x A string containing a valid National Grid Reference
#' @return A string containing a valid National Grid Reference
#' @examples
#' ngrZeroStrip("ST46409000")
#' ngrZeroStrip("ST46409001")
#' @export
ngrZeroStrip <- function(x){
prefix <- substr(x,1,2)
suffix <- substr(x,3,nchar(x))
if (nchar(suffix) %% 2 == 0){
xcomp <- substr(suffix,1,nchar(suffix)/2)
ycomp <- substr(suffix,(nchar(suffix)/2)+1,nchar(suffix))
xtolose <- nchar(xcomp) - nchar(sub("0+$", "", xcomp))
ytolose <- nchar(ycomp) - nchar(sub("0+$", "", ycomp))
tolose <- min(xtolose, ytolose)
newxcomp <- substr(xcomp,1,nchar(xcomp)-tolose)
newycomp <- substr(ycomp,1,nchar(ycomp)-tolose)
newngr <- paste(prefix, newxcomp, newycomp, sep="")
} else {
stop("Grid reference has an incorrect (odd) number of figures.")
}
return(newngr)
}
#' @title Convert a coordinate pair to a different coordinate system
#' @description Converts a single coordinate pair to a different coordinate system
#' @param coords A vector of length 2 with the coordinates
#' @param proj1 The CRS of the input coordinates
#' @param proj2 The target CRS of the output coordinates
#' @return A vector of length 2 with the converted coordinate pair.
#' @examples
#' bng <- CRS("+init=epsg:27700") # British National Grid
#' lonlat <- CRS("+init=epsg:4326") # LatLon WGS84
#' xyTransform(c(345600,378200), bng, lonlat)
#' @export
xyTransform <- function(coords, proj1, proj2){
if (as.character(class(proj1))!="CRS" | as.character(class(proj2))!="CRS"){
stop("One or both of proj1 and proj2 are not correct CRS class definitions.")
}
tmp <- SpatialPoints(t(as.matrix(as.numeric(coords))), proj4string=proj1)
tmp <- spTransform(tmp,proj2)
coordinates(tmp)
res <- as.numeric(coordinates(tmp))
return(res)
}
#' @title Convert an alphahull to a SpatialPolygonsDataFrame
#'
#' @description Convert an ahull object from the alphahull package to a SpatialPolygonsDataFramefor wider use.
#'
#' @param x An object of class ahull from the alphahull package
#' @param increment A single numeric value for the number of increments with which to describe the circumference of a circle.
#' @param rnd A single numeric value controlling the precision of the estimate of the arcs (the default is usually sensible).
#' @param proj4string Projection parameters if the alphahull is based on geographic locations.
#'
#' @return An object of class SpatialPolygonsDataFrame
#' @examples
#' ##Example point set and its alphahull
#' n <- 300
#' theta <- runif(n, 0,2 * pi)
#' r <- sqrt(runif(n, 0.25^2, 0.5^2))
#' x <- cbind(0.5 + r*cos(theta), 0.5 + r*sin(theta))
#' ahullobj <- ahull(x, alpha=0.1)
#' ahobj <- ah2sp(ahullobj)
#' ## Plot
#' par(mfrow=c(1,2), pty="s")
#' plot(ahullobj, main="ahull object")
#' plot(ahobj, col="lightgrey", pbg="white", axes=TRUE, main="sp object")
#' points(x)
#' @import alphahull
#' @import maptools
#' @export
#'
ah2sp <- function(x, increment=360, rnd=10, proj4string=CRS(as.character(NA))){
if (class(x) != "ahull"){
stop("x needs to be an ahull class object")
}
xdf <- as.data.frame(x$arcs)
xdf <- subset(xdf,xdf$r > 0)
res <- NULL
if (nrow(xdf) > 0){
linesj <- list()
prevx<-NULL
prevy<-NULL
j<-1
for (i in 1:nrow(xdf)){
rowi <- xdf[i,]
v <- c(rowi$v.x, rowi$v.y)
theta <- rowi$theta
r <- rowi$r
cc <- c(rowi$c1, rowi$c2)
#Arcs need to be redefined as strings of points. Work out the number of points to allocate in this arc segment.
ipoints <- 2 + round(increment * (rowi$theta / 2),0)
angles <- anglesArc(v, theta)
seqang <- seq(angles[1], angles[2], length=ipoints)
x <- round(cc[1] + r * cos(seqang),rnd)
y <- round(cc[2] + r * sin(seqang),rnd)
if (is.null(prevx)){
prevx <- x
prevy <- y
} else if (x[1] == round(prevx[length(prevx)],rnd) && y[1] == round(prevy[length(prevy)],rnd)){
if (i == nrow(xdf)){
prevx <- append(prevx, x[2:ipoints])
prevy <- append(prevy, y[2:ipoints])
prevx[length(prevx)] <- prevx[1]
prevy[length(prevy)] <- prevy[1]
coordsj <- cbind(prevx, prevy)
colnames(coordsj) <- NULL
linej <- Line(coordsj)
linesj[[j]] <- Lines(linej, ID = as.character(j))
} else {
prevx <- append(prevx, x[2:ipoints])
prevy <- append(prevy, y[2:ipoints])
}
} else {
prevx[length(prevx)] <- prevx[1]
prevy[length(prevy)] <- prevy[1]
coordsj<-cbind(prevx, prevy)
colnames(coordsj) <- NULL
linej <- Line(coordsj)
linesj[[j]] <- Lines(linej, ID=as.character(j))
j <- j+1
prevx <- NULL
prevy <- NULL
}
}
lspl <- SpatialLines(linesj)
lns <- slot(lspl, "lines")
polys <- sapply(lns, function(x) {
crds <- slot(slot(x, "Lines")[[1]], "coords")
identical(crds[1, ], crds[nrow(crds), ])
})
polyssl <- lspl[polys]
list_of_Lines <- slot(polyssl, "lines")
sppolys <- SpatialPolygons(list(Polygons(lapply(list_of_Lines, function(x) { Polygon(slot(slot(x, "Lines")[[1]], "coords")) }), ID="1")), proj4string=proj4string)
hid <- sapply(slot(sppolys, "polygons"), function(x) slot(x, "ID"))
areas <- sapply(slot(sppolys, "polygons"), function(x) slot(x, "area"))
df <- data.frame(hid, areas)
names(df) <- c("HID", "Area")
rownames(df) <- df$HID
res <- SpatialPolygonsDataFrame(sppolys, data=df)
res <- res[which(res@data$Area > 0),]
}
return(res)
}
##
Add the following code to your website.
For more information on customizing the embed code, read Embedding Snippets.