forked from nypl-spacetime/map-vectorizer
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsimplify_map.R
91 lines (91 loc) · 3.26 KB
/
simplify_map.R
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
library(rgdal)
library(alphahull)
library(igraph)
library(shapefiles)
args <- commandArgs(trailingOnly = TRUE)
filepath = args[1]
layer = args[2]
finalout = args[3]
directory = args[4]
polysubset = as.numeric(args[5])
myPolygons = readOGR(filepath,layer=layer)
alpha=2 # 1.6 is another option, dependant on polygon areas that are being extracted (see minarea/maxarea below)
numsample=1000
minarea=20 # square meters (minimum polygon size to include)
maxarea=3000 # square meters (maximum polygon size to include)
num = length(myPolygons)
poly.list = vector('list')
logfile <- file(paste(directory,"/r-log.txt", sep=""), "a") # log file
for (i in 1: num) {
poly = myPolygons@polygons[[i]]
if (poly@area > minarea && poly@area < maxarea) {
types = c("hexagonal","regular","nonaligned","stratified")
attempts = 1
x.as = 0
error = 1
# try to find a functional ashape with any of the sampling types above
while (attempts <= length(types) && error > 0) {
pts = spsample(poly, n=numsample, type=types[attempts])
x.coords = coordinates(pts)
x.as = tryCatch(
{
ashape(x.coords,alpha=alpha)
}
,warning = function (w) {}
,error = function (e) {}
)
# cat("Attempts: ", attempts, " class: ", class(x.as), "\n")
if ( class(x.as)=="ashape" ) {
x.asg = graph.edgelist(cbind(as.character(x.as$edges[, "ind1"]), as.character(x.as$edges[, "ind2"])), directed = FALSE)
error = 0
# try to fix circularity
if (error == 0 && any(degree(x.asg) != 2)) {
x.asg <- delete.vertices(x.asg, which(degree(x.asg) == 1))
}
# try to fix multiple circles (when shape has big holes inside)
if (error == 0 && clusters(x.asg)$no > 1) {
# get the components
components = decompose.graph(x.asg, min.vertices = 3, max.comps = 1)
# we are assuming the FIRST cluster is the LARGEST (external container)
x.asg = components[[1]]
# cat(poly@ID, ": Graph composed of more than one circle", "\n")
}
if (error == 0 && any(degree(x.asg) != 2)) {
# cat(poly@ID, ": Graph not circular", "\n")
error = 2
}
if (!is.connected(x.asg)) {
# cat(poly@ID, ": Graph not connected", "\n")
error = 1
}
if (error == 0) {
# print("Added new polygon")
cutg = x.asg - E(x.asg)[1]
# find chain end points
ends = names(which(degree(cutg) == 1))
path = get.shortest.paths(cutg, ends[1], ends[2])[[1]]
# this is an index into the points
pathX = as.numeric(V(x.asg)[path]$name)
# join the ends
pathX = c(pathX, pathX[1])
p = Polygon(x.as$x[pathX, ])
# now we simplify the polygon (reduce edge count)
temp <- as.data.frame(p@coords)
names(temp) <- c("x","y")
simple = dp(temp,0.5)
p@coords <- as.matrix(cbind(simple$x, simple$y))
# finished simplifying
ps = Polygons(list(p),1)
sps = SpatialPolygons(list(ps), , myPolygons@proj4string)
data = data.frame(DN=(((polysubset-1) * 50000) + i)) # the polygon ID
output = SpatialPolygonsDataFrame(sps, data)
writeOGR(output, paste(finalout, (((polysubset-1) * 50000) + i), "-polygon.shp", sep=""), layer, driver="ESRI Shapefile")
}
}
attempts = attempts + 1
}
if (error > 0) {
writeLines(paste((((polysubset-1) * 50000) + i)," could not be written: ", error, sep=""), logfile)
}
}
}