Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

georeferenced output #914

Open
potrykus opened this issue Aug 15, 2024 · 13 comments
Open

georeferenced output #914

potrykus opened this issue Aug 15, 2024 · 13 comments

Comments

@potrykus
Copy link

potrykus commented Aug 15, 2024

tmap produces PDFs wih clean vector data. Shouldn't tmap automatically georeference that output - i.e. make its output geo-enabled? Here is a python overview: https://gis.stackexchange.com/questions/49646/is-it-possible-to-georeference-an-existing-un-georeferenced-pdf. This is what I found for the R raster package: https://rpubs.com/Rubio-Polania/1123497.

@Nowosad
Copy link
Member

Nowosad commented Aug 23, 2024

Related discussion: #383

@potrykus
Copy link
Author

Thanks. Avenza is an app for mobile use of geo-referenced PDFs (including in a disconnected back-country hiking use case.) They are an end-user platform for US Bureau of Land Management final products https://www.blm.gov/maps/georeferenced-PDFs (expand "How to download and use"). Right now those two major players might be georeferencing maps manually, and may be locked into an old ArcGIS workflow: Avenza-ready Google Earth imagery georeferenced PDF in QGIS. (QGIS is a GIS system that cannot compete with ArcGIS for feature-completeness.)

Given this technological state, tmap georeferenced output would be a powerful and potentially widely-consumed application of the R sf + tmap architecture.

@mtennekes
Copy link
Member

Thx for this input @potrykus

Just tested, and it is already possible in sf:

library(sf)
library(tmap)

data(World)

st_write(World, "World.pdf", driver = "PDF")

So now the question is how replace the sf plotting mechanism with tmap. Any ideas @edzer or @tim-salabim ?

If someone could help me with the gdal part, I am happy to implement the other part (passing on the map coordinates of the reference objects).

@edzer
Copy link
Contributor

edzer commented Sep 2, 2024

So now the question is how replace the sf plotting mechanism with tmap.

st_write() only writes data, it doesn't plot: there are no visual elements passed on to it (colors, symbols, line properties etc). If you want to create a GeoPDF from a tmap plot, you'd have to add the geo elements to R's pdf driver - I'd guess these are offset & scale for x and y axes, and a coordinate reference system.

@mtennekes
Copy link
Member

Can I use st_write for that purpose? E.g. like this:

tm = tm_shape(World) + tm_poygons()
tmap_save(tm, "World.pdf")

#within tmap_save:
pdf(file)
print(tm)
dev.off()

sf::st_write(?, "World.pdf", offset = , scale = , crs = )

Is so, do you know via which of its arguments these can be passed on (..., dataset_options, layer_options, ... ?

@edzer
Copy link
Contributor

edzer commented Sep 2, 2024

I think st_write() does offset, scale and CRS by itself, as it is an OGR (GDAL) driver; it however takes an sf object, not a tmap object.

@tim-salabim
Copy link

I cannot add much here, as mapshot does only simple pdf rendering via png screenshots of the html. The only thing I found is
https://gdal.org/en/latest/drivers/vector/pdf.html#feature-style-support

@edzer
Copy link
Contributor

edzer commented Sep 2, 2024

Oh, yes, and there's the raster driver too: https://gdal.org/en/latest/drivers/raster/pdf.html#raster-pdf

@tim-salabim
Copy link

Oh, yes, and there's the raster driver too: https://gdal.org/en/latest/drivers/raster/pdf.html#raster-pdf

Yes, but if I understand that correctly, outpu can either be RGB(A) or not. In case it's not I guess it'll be greyscale?

@mdsumner
Copy link

mdsumner commented Nov 5, 2024

would have to get the "plotted" extents of the map for the geotransform, i.e, if it was rendered and written out to file with base graphics the a_ullr for GDAL VRT would be par('usr')[c(1, 4, 2, 3)] .

Tools will complain about coordinates out of range for longlat and potentially other crs (the margin of a plot in Mollweide won't have a valid inverse for example, but I don't think that's a huge problem. I'll explore where the render happens and if the overall frame extent can be gotten. (I wrote something for this elsewhere but I can't find it).

Just to clarify, we're talking about raster graphics yeah? In GDAL terms you need the geotransform, which is a rejig of extent and dimension (shape) mixed together as offset and scale, i.e. equivalent of

f <- function(x, dimension) {
    px <- c(diff(x[1:2])/dimension[1L], -diff(x[3:4])/dimension[2L])
 c(xmin = x[1], xres = px[1], yskew = 0, ymax = x[4], xskew = 0, yres = px[2])   
}
f(par("usr"), dev.size("px"))

but, extent is enough reordered into 'a_ullr' with VRT or vrt::// or write to .wld file even.

@mdsumner
Copy link

mdsumner commented Nov 5, 2024

oops sorry, not "usr", it needs the full expanded plot range - in base might need to use 'plt' to get that (which I think I did before somewhere)

at any rate we need the grid viewport extent but handy to have some more examples I expect

@mdsumner
Copy link

mdsumner commented Nov 5, 2024

grconvertX/Y is your friend for base graphics:

device_extent <- function() {
    c(grconvertX(c(0, 1) , "ndc", "user"), 
      grconvertY(c(0, 1), "ndc", "user"))
}

pdf(tf <- tempfile(fileext = ".pdf"))
maps::map(col = "grey")
## get this information relative to the current plot
dext <- device_extent()
dev.off()
print(dext)

##  (equivalent to a call to gdal_translate {tf} out.vrt -a_ullr ...) since GDAL 3.7
dsn <- sprintf("vrt://%s?a_srs=EPSG:4326&a_ullr=%s", tf, paste0(dext[c(1, 4, 2, 3)], collapse = ","))

terra::plot(terra::rast(dsn))
maps::map(add = TRUE, col = "red")

@mdsumner
Copy link

is it possible to get this information? Surely in the pipeline from a tmap object to a rendered graphic we could get information about the frame within the device coordinates? We need this for getting decent images from web servers in #975 and #502 also. With base graphics you can do

ideal_dim <- function(x, ...) {
  if (!missing(x)) plot(x, ...)
  dcx <- grconvertX(c(0, 1), "npc", "device")
  dcy <- grconvertY(c(0, 1), "npc", "device")
  ## "ideal" dimension then is
  ceiling(c(diff(dcx), diff(dcy[2:1])))
}

to get the right dimensions to fit within an existing plot, the xsize and ysize to provide to GDAL (using non-nearest resampling is also important, for maps with text or other features on them).

I see that tm_rgb() and the tm_lines() and other vector plotting families set up different framing to each other, and to base graphics. I don't know how to get these equivalent values, par("usr") would be enough. I'll explore grid package and see if it's somehow possible to interrogate.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

No branches or pull requests

6 participants