papermap is a Python library and CLI tool for creating ready-to-print paper maps.
Project description
papermap
papermap is a Python library and CLI tool for creating ready-to-print paper maps.
Installation
papermap is available on PyPI. Install with uv or your package manager of choice:
uv add papermap
Documentation
Check out the papermap documentation for the User's Guide, API Reference and CLI Reference.
Usage
papermap can be used both in your own applications as a package, as well as a CLI tool.
As a Library
Basic Usage
Create a simple portrait-oriented, A4-sized map at scale 1:25000:
>>> from papermap import PaperMap
>>> pm = PaperMap(13.75889, 100.49722) # Bangkok, Thailand
>>> pm.render()
>>> pm.save("Bangkok.pdf")
Custom Size and Orientation
Create a landscape-oriented, A3-sized map with grid overlay:
>>> from papermap import PaperMap
>>> pm = PaperMap(
... lat=40.416775,
... lon=-3.703790, # Madrid, Spain
... tile_provider_key="stadia-stamenterrain",
... size="a3",
... landscape=True,
... scale=50_000,
... add_grid=True,
... )
>>> pm.render()
>>> pm.save("Madrid.pdf")
Satellite Imagery
Create a map using satellite imagery:
>>> from papermap import PaperMap
>>> pm = PaperMap(
... lat=51.5074,
... lon=-0.1278, # London, UK
... tile_provider_key="esri-worldimagery",
... size="a4",
... scale=10_000,
... )
>>> pm.render()
>>> pm.save("London_Satellite.pdf")
Topographic Maps
Create a topographic map for hiking:
>>> from papermap import PaperMap
>>> pm = PaperMap(
... lat=46.5197,
... lon=7.9577, # Mürren, Switzerland
... tile_provider_key="opentopomap",
... size="a3",
... landscape=True,
... scale=25_000,
... add_grid=True,
... grid_size=500, # 500m grid for easier navigation
... )
>>> pm.render()
>>> pm.save("Murren_Topo.pdf")
High-Resolution Printing
Create a high-resolution map for professional printing:
>>> from papermap import PaperMap
>>> pm = PaperMap(
... lat=35.6762,
... lon=139.6503, # Tokyo, Japan
... tile_provider_key="openstreetmap",
... size="a0", # Large format
... landscape=True,
... scale=15_000,
... dpi=600, # High resolution
... add_grid=True,
... )
>>> pm.render()
>>> pm.save("Tokyo_HighRes.pdf")
Using UTM Coordinates
Create a map using UTM coordinates instead of latitude/longitude:
>>> from papermap import PaperMap
>>> pm = PaperMap.from_utm(
... easting=500000,
... northing=4649776,
... zone=30,
... hemisphere="N", # Northern hemisphere
... tile_provider_key="openstreetmap",
... size="a4",
... scale=25_000,
... add_grid=True,
... )
>>> pm.render()
>>> pm.save("UTM_Map.pdf")
Custom Margins
Create a map with custom margins for binding:
>>> from papermap import PaperMap
>>> pm = PaperMap(
... lat=-33.8688,
... lon=151.2093, # Sydney, Australia
... tile_provider_key="openstreetmap",
... size="letter",
... margin_left=20, # Extra margin for binding
... margin_top=10,
... margin_right=10,
... margin_bottom=10,
... scale=20_000,
... )
>>> pm.render()
>>> pm.save("Sydney_Binding.pdf")
Using API Keys
Some tile providers require API keys. Here's how to use them:
>>> from papermap import PaperMap
>>> pm = PaperMap(
... lat=37.7749,
... lon=-122.4194, # San Francisco, USA
... tile_provider_key="thunderforest-outdoors",
... api_key="your_api_key_here", # Get from thunderforest.com
... size="a4",
... landscape=True,
... scale=25_000,
... add_grid=True,
... )
>>> pm.render()
>>> pm.save("SF_Outdoors.pdf")
Adding Features (i.e., Markers, Lines and Polygons)
Overlay GeoJSON-style geometries (points, lines, polygons) on top of the base map. Features are styled per call, rendered above the base map but below the grid, and clipped to the map area. You can also add a raw GeoJSON dict (or any object implementing the __geo_interface__ protocol) via add_geojson().
>>> from papermap import PaperMap
>>> pm = PaperMap(lat=40.7128, lon=-74.0060, scale=25_000)
>>> # A red dot at the map centre.
>>> pm.add_circle_marker(40.7128, -74.0060, radius=3, fill_color="#f00")
>>> # A blue route.
>>> pm.add_line(
... [(40.7100, -74.0100), (40.7150, -74.0050), (40.7200, -74.0000)],
... stroke_color="#00f",
... stroke_width=1.0
... )
>>> # A semi-transparent green region from a GeoJSON Polygon.
>>> pm.add_geojson(
... {
... "type": "Polygon",
... "coordinates": [
... [
... [-74.020, 40.710],
... [-74.000, 40.710],
... [-74.000, 40.720],
... [-74.020, 40.720],
... [-74.020, 40.710]
... ]
... ]
... },
... style={"fill_color": "#0f0", "opacity": 0.3}
... )
>>> pm.render()
>>> pm.save("NYC_Annotated.pdf")
Creating a Map From Features
To fit the map around features you already have, use PaperMap.from_features(). It centres the map on the features' bounding box and pre-populates them so they will be drawn on render(). PaperMap.from_geojson() and PaperMap.from_gpx() do the same for a GeoJSON file/dict or a GPX file (or any object exposing __geo_interface__, such as a gpx.GPX instance). GeoJSON Point/MultiPoint become CircleMarker, LineString/MultiLineString become Line, and Polygon/MultiPolygon become Polygon; GPX waypoints become circle markers, routes become lines, and each segment of each track becomes its own line. An optional style dict provides default styling for every parsed feature, overridden by per-feature simplestyle-spec properties.
Reading GPX files from disk requires the optional gpx package; install it via pip install papermap[gpx].
>>> from papermap import PaperMap
>>> from papermap.features import CircleMarker, Line
>>> pm = PaperMap.from_features(
... CircleMarker(40.7484, -73.9857, fill_color="#f00"), # Empire State Building
... CircleMarker(40.7128, -74.0060, fill_color="#f00"), # NYC City Hall
... Line(
... [(40.7484, -73.9857), (40.7128, -74.0060)],
... stroke_color="#00f",
... stroke_width=1.0,
... ),
... scale=25_000,
... )
>>> pm.render()
>>> pm.save("NYC_Landmarks.pdf")
Auto-scaling to Features
Pass auto_scale=True to from_features(), from_geojson(), or from_gpx() to skip picking a scale by hand: the scale is computed so the features fit within the printable area, then snapped up to the nearest common cartographic scale (1:1 000, 1:2 500, 1:5 000, 1:10 000, 1:25 000, 1:50 000, …). Use padding (mm per side, default 5.0) to control how much breathing room is left around the features:
>>> from papermap import PaperMap
>>> pm = PaperMap.from_gpx("hike.gpx", auto_scale=True, padding=10)
>>> pm.render()
>>> pm.save("Hike.pdf")
For more options and details, see the API Reference.
As a CLI Tool
Basic Usage
Create a simple portrait-oriented, A4-sized map:
$ papermap latlon -- 13.75889 100.49722 Bangkok.pdf
Custom Size and Orientation
Create a landscape-oriented, A3-sized map with grid overlay:
$ papermap latlon \
--tile-provider stadia-stamenterrain \
--paper-size a3 \
--landscape \
--scale 50000 \
--grid \
-- 40.416775 -3.703790 Madrid.pdf
Satellite Imagery
Create a map using satellite imagery:
$ papermap latlon \
--tile-provider esri-worldimagery \
--scale 10000 \
-- 51.5074 -0.1278 London_Satellite.pdf
Topographic Maps
Create a topographic map for hiking:
$ papermap latlon \
--tile-provider opentopomap \
--paper-size a3 \
--landscape \
--scale 25000 \
--grid \
--grid-size 500 \
-- 46.5197 7.9577 Murren_Topo.pdf
High-Resolution Printing
Create a high-resolution map for professional printing:
$ papermap latlon \
--tile-provider openstreetmap \
--paper-size a0 \
--landscape \
--scale 15000 \
--dpi 600 \
--grid \
-- 35.6762 139.6503 Tokyo_HighRes.pdf
Using UTM Coordinates
Create a map using UTM coordinates:
$ papermap utm \
--tile-provider openstreetmap \
--paper-size a4 \
--scale 25000 \
--grid \
-- 500000 4649776 30 N UTM_Map.pdf
From a GPX File
Render a paper map from a .gpx file, auto-scaled to fit the waypoints, routes, and tracks (requires pip install papermap[gpx]):
$ papermap gpx --auto-scale --padding 10 -- hike.gpx Hike.pdf
Custom Margins
Create a map with custom margins for binding:
$ papermap latlon \
--tile-provider openstreetmap \
--paper-size letter \
--margin-left 20 \
--margin-top 10 \
--margin-right 10 \
--margin-bottom 10 \
--scale 20000 \
-- -33.8688 151.2093 Sydney_Binding.pdf
Using API Keys
Use tile providers that require API keys:
$ papermap latlon \
--tile-provider thunderforest-outdoors \
--api-key "your_api_key_here" \
--paper-size a4 \
--landscape \
--scale 25000 \
--grid \
-- 37.7749 -122.4194 SF_Outdoors.pdf
For more options and details, see the CLI Reference.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file papermap-2026.2.0.tar.gz.
File metadata
- Download URL: papermap-2026.2.0.tar.gz
- Upload date:
- Size: 78.0 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: uv/0.11.14 {"installer":{"name":"uv","version":"0.11.14","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
5da9913923ed6493bcc55692d3f54343dfef9ce42fc8a042f6beaf86de022c6d
|
|
| MD5 |
d8b04357cf3d7b35b0cc824567fd5f9e
|
|
| BLAKE2b-256 |
409d349dc083d126ec736ee11037bcb7f33f61d2b711c978bb5091a86a14d7f0
|
File details
Details for the file papermap-2026.2.0-py3-none-any.whl.
File metadata
- Download URL: papermap-2026.2.0-py3-none-any.whl
- Upload date:
- Size: 77.8 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: uv/0.11.14 {"installer":{"name":"uv","version":"0.11.14","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Ubuntu","version":"24.04","id":"noble","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":true}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
5b120c6638236e0539b1d6fb4eb64a44c74354315dcc96537b8038a73a181e17
|
|
| MD5 |
d7586322e22b52f832d97f232c15853f
|
|
| BLAKE2b-256 |
466fffd40d566c0c21b4e0b67f7751fa5096f997ea774bc54a42a685112bb73d
|