我喜欢在特定区域内绘制ggmap
的街道。我通过立交桥api从osm获取数据。对geom_path
的大多数街道来说,它非常有用。然而,一些街道搞砸了。任何提示都表示赞赏。
请查看http://overpass-turbo.eu/以获取所需的输出。您可以在下面的R代码中找到查询。
library(httr)
library(tidyverse)
#> ── Attaching packages ─────────────────────────────────────────────────────────────────────────────────────────── tidyverse 1.2.1 ──
#> ✔ ggplot2 2.2.1 ✔ purrr 0.2.4
#> ✔ tibble 1.4.2 ✔ dplyr 0.7.4
#> ✔ tidyr 0.8.0 ✔ stringr 1.3.0
#> ✔ readr 1.1.1 ✔ forcats 0.3.0
#> ── Conflicts ────────────────────────────────────────────────────────────────────────────────────────────── tidyverse_conflicts() ──
#> ✖ dplyr::filter() masks stats::filter()
#> ✖ dplyr::lag() masks stats::lag()
library(ggmap)
library(rlist)
# Get data from overpass api
query <- paste('[out:json];',
'(way["highway"~"primary|residential"](53.5970, 9.9010, 53.6050, 9.9080););',
'(._;>;);',
'out body;')
url <- 'http://overpass-api.de/api/interpreter'
r <- POST(url = url, body = query, encode = 'json')
# Tidy data
nodes <- content(r)$elements %>% list.filter(type == 'node')
ways <- content(r)$elements %>% list.filter(type == 'way')
df_nodes <- nodes %>%
list.select(type, id, lat, lon) %>%
bind_rows()
df_ways <- ways %>%
lapply(function(x) list.append(x, street = x$tags$name)) %>%
list.select(street, nodes)
df_ways <- map(df_ways, function(x) x %>% as_tibble) %>%
bind_rows() %>%
mutate(id = unlist(nodes))
df <- df_ways %>% left_join(df_nodes, by = 'id')
head(df)
#> # A tibble: 6 x 6
#> street nodes id type lat lon
#> <chr> <list> <dbl> <chr> <dbl> <dbl>
#> 1 Reichsbahnstraße <int [1]> 38893884 node 53.6 9.91
#> 2 Reichsbahnstraße <int [1]> 55079985 node 53.6 9.91
#> 3 Reichsbahnstraße <int [1]> 38893882 node 53.6 9.91
#> 4 Reichsbahnstraße <int [1]> 38893881 node 53.6 9.91
#> 5 Reichsbahnstraße <int [1]> 380820539 node 53.6 9.91
#> 6 Reichsbahnstraße <int [1]> 38893879 node 53.6 9.91
# Get map
lat <- (max(df$lat)+min(df$lat))/2
lon <- (max(df$lon)+min(df$lon))/2
hamburg <- get_map(location = c(lon = lon, lat = lat), zoom = 16)
#> Map from URL : http://maps.googleapis.com/maps/api/staticmap?center=53.601726,9.90531&zoom=16&size=640x640&scale=2&maptype=terrain&language=en-EN&sensor=false
# Plot
ggmap(hamburg) +
geom_path(data = df, aes(x = lon, y = lat, color = street), size = 2)
#> Warning: Removed 3 rows containing missing values (geom_path).
答案 0 :(得分:1)
每当街道有很多分支时,你就会遇到这个问题,因为geom_path()
只会用直线连接每两个连续的点。
我们以'Reichsbahnstraße'为例:
lon <- df %>% filter(street == "Reichsbahnstraße") %>% .$lon
lat <- df %>% filter(street == "Reichsbahnstraße") %>% .$lat
lat[1] == lat[41] & lon[1] == lon[41]
# returns TRUE
geom_path()
从第1点(交叉点,见下图)开始,绘制街道的一部分(向东北方向),然后再次返回交叉点(索引41)以绘制下一个分支。
在绘制其他分支之前,您可以做的是避免这种情况,然后通过相同的道路返回交叉点(例如,而不是c(7, 8, 9, 10, 7, 6)
执行c(7, 8, 9, 10, 9, 8, 7, 6)
)。有点像你试图用铅笔画街而不把它从纸页上拿下来你会怎么做。