I'm trying to check whether two polygons intersect in R. When plotting, they clearly do not. When checking the intersection, rgeos::gIntersects()
currently returns FALSE
, while sf::intersects()
returns TRUE
. I imagine this is due to the polygons being (1) large and (2) close together, so something about when on a flat surface they don't appear to intersect but on a sphere they would appear to intersect?
Ideally I could keep my workflow all in sf
-- but I'm wondering if there's a way to use sf::intersects()
(or another sf
function) that will return FALSE
here?
Here's an example:
library(sf)
library(rgeos)
library(leaflet)
library(leaflet.extras)
#### Make Polygons
poly_1 <- c(xmin = -124.75961, ymin = 49.53330, xmax = -113.77328, ymax = 56.15249) %>%
st_bbox() %>%
st_as_sfc()
st_crs(poly_1) <- 4326
poly_2 <- c(xmin = -124.73214, ymin = 25.11625, xmax = -66.94889, ymax = 49.38330) %>%
st_bbox() %>%
st_as_sfc()
st_crs(poly_2) <- 4326
#### Plotting
# Visually, the polygons do not intersect
leaflet() %>%
addTiles() %>%
addPolygons(data = poly_1) %>%
addPolygons(data = poly_2)
#### Check Intersection
# returns FALSE
gIntersects(poly_1 %>% as("Spatial"),
poly_2 %>% as("Spatial"))
# returns TRUE
st_intersects(poly_1,
poly_2,
sparse = F)
This is an interesting problem, with the root cause being difference between planar (on a flat surface) and spherical (on a globe) geometry.
On a plane - which is a simplified approach that GEOS takes - the four corners of a polygon are connected by four straight lines, the sum of angles is 360° degrees etc. Geometry works as Euclid taught ages ago.
But, and this is crucial, this is not how the world works. On a globe the four connections of polygon are not straight lines but great circles. Or rather they are straight when drawn on a globe, and curved when rolled flat one a planar surface (such as a map or your computer screen).
Because an example is more than a 1000 words consider this piece of code:
library(sf)
library(dplyr)
# make polygons
poly_1 <- c(xmin = -124.75961, ymin = 49.53330, xmax = -113.77328, ymax = 56.15249) %>%
st_bbox() %>%
st_as_sfc()
st_crs(poly_1) <- 4326
poly_2 <- c(xmin = -124.73214, ymin = 25.11625, xmax = -66.94889, ymax = 49.38330) %>%
st_bbox() %>%
st_as_sfc()
st_crs(poly_2) <- 4326
# this is what you *think* you see (and what GEOS sees, being planar)
# = four corners connected by straight lines
# & no intersecton
mapview::mapview(list(poly_1, poly_2))
# this is what you *should* see (and what {s2} sees, being spherical)
# = four corners connected by great circles
# & an obvious intersection around the lower right corner of the small polygon
poly_1alt <- st_segmentize(poly_1, units::set_units(1, degree))
poly_2alt <- st_segmentize(poly_2, units::set_units(1, degree))
mapview::mapview(list(poly_1alt, poly_2alt))
You have two options:
{s2}
logic.This should be in theory the correct approach, but somewhat counter intuitive.
{sf}
abandon spherical approach to polygons, and force it to apply planar approach (such as GEOS uses).This would be in theory a wrong approach, but consistent both with your planar intuition and previous behaviour of most GIS tools, including {sf}
prior to version 1.0.0
# remedy (of a kind...) = force planar geometry operations
sf::sf_use_s2(FALSE)
st_intersects(poly_1, poly_2, sparse = F)
# [,1]
# [1,] FALSE