openmaptiles-tools
openmaptiles-tools copied to clipboard
Add LabelCenterline function
Closes #354. The goal is to replace acalcutt/osm-lakelines with a faster SQL function. I've never written anything this complex in SQL before, so any help with efficiency and best practices is appreciated. I'm also new to this project, so I'm not sure about where to hook this in.
The logic:
- Get the exterior polygon (ignoring islands) and use a convex hull if for some reason there's more than one exterior polygon
- Use ST_LineInterpolatePoints to evenly distribute vertices around the polygon
- Get ST_VoronoiLines from the interpolated vertices and covert to edges
- Filter to only edges within the polygon
- Recursively clip the shortest branch edges until only a single line is left
- Simplify, smooth, and return
SFCGAL is not required for this method.
It takes about 40 seconds to calculate lines for the 395 largest lakes of Michigan, which I think is already a big improvement over osm-lakelines. The recursion takes about 75% of this time and can probably be made vastly more efficient.
The results are relatively consistent with osm-lakelines and I think look pretty good, except when there are large islands or bays.
data:image/s3,"s3://crabby-images/4bf0a/4bf0a891111fcd5a56369c9bdd51b39f29a8d447" alt="Screenshot 2023-02-27 at 9 47 15 PM"
data:image/s3,"s3://crabby-images/ab925/ab9256e159d989632a7e5c170255a2c610f48aca" alt="Screenshot 2023-02-27 at 9 47 01 PM"
data:image/s3,"s3://crabby-images/7c01f/7c01fe1f605c425cc092820494126e7f6914d46f" alt="Screenshot 2023-02-27 at 9 45 37 PM"
Rad!
this is really impressive! The sql style is a bit different from other files (e.g. using '
, etc), but no biggie. I will be studying it for a bit for sure
I'll try to give this a detailed review, as it's something I've been interested in.
Among the above examples, I think only this one seems a bit incorrect:
The sql style is a bit different from other files (e.g. using ', etc), but no biggie.
I converted these to PL/pgSQL functions to avoid the block string, but I see that this also enables some language features we might be able to use to speed up the code. I'll look into it at some point.
Among the above examples, I think only this one seems a bit incorrect:
Yes, the more circular the feature, the harder it is to determine a sensical centerline. We might try baking in a threshold where circular stuff just defaults to a horizontal line.
We could also achieve better results by allowing multiple lines in the output to account for islands and bays, perhaps based on the zoom. Lots of of shapes just really aren't modeled sufficiently by a single line. Ex: Manicouagan Reservoir.
data:image/s3,"s3://crabby-images/d3560/d3560a7a643bf883b8cade4c637d456ec2619310" alt="Screenshot 2023-02-28 at 11 17 21 AM"
Okay, I added a number of enhancements to get nicer results in a reasonably fast timeframe. It now takes just 15 seconds to calculate the basic centerlines for the 395 largest lakes of Michigan, down from 40 seconds. Smoothing is now done proportionally to the size of the feature.
data:image/s3,"s3://crabby-images/29d8f/29d8f596fea8826b8edd612820866e103ea3bdac" alt="Screenshot 2023-03-10 at 1 42 49 PM"
Multiple lines over a given length threshold can be included in the output. This allows more accurate labeling at higher zooms.
data:image/s3,"s3://crabby-images/ee902/ee90233aa8f7639a089c634a6cba96a1f2843d8f" alt="Screenshot 2023-03-10 at 1 44 08 PM"
Holes over a given size can be included in the calculation as well. This takes a little longer to calculate, 22 seconds for these lakes.
data:image/s3,"s3://crabby-images/32bf8/32bf89d358e0dd3ae37d89b0b805615d37fc9a4b" alt="Screenshot 2023-03-10 at 1 44 55 PM"
More examples:
There as six parameters to customize these behaviors.