imposm3/geom/geom.go

150 lines
2.9 KiB
Go
Raw Permalink Normal View History

2013-04-21 18:37:05 +04:00
package geom
import (
"errors"
"math"
2014-08-04 17:19:35 +04:00
"github.com/omniscale/imposm3/element"
"github.com/omniscale/imposm3/geom/geos"
2013-04-21 18:37:05 +04:00
)
2013-05-10 12:57:06 +04:00
type GeomError struct {
message string
level int
}
type Geometry struct {
Geom *geos.Geom
Wkb []byte
}
2013-05-10 12:57:06 +04:00
func (e *GeomError) Error() string {
return e.message
}
func (e *GeomError) Level() int {
return e.level
}
2015-04-30 11:05:22 +03:00
func newGeomError(message string, level int) *GeomError {
2013-05-10 12:57:06 +04:00
return &GeomError{message, level}
}
2013-05-10 12:08:31 +04:00
var (
2015-04-30 11:05:22 +03:00
ErrorOneNodeWay = newGeomError("need at least two separate nodes for way", 0)
ErrorNoRing = newGeomError("linestrings do not form ring", 0)
2013-05-10 12:08:31 +04:00
)
func Point(g *geos.Geos, node element.Node) (*geos.Geom, error) {
2013-07-30 10:01:15 +04:00
geom := g.Point(node.Long, node.Lat)
if geom == nil {
2015-04-30 11:05:22 +03:00
return nil, newGeomError("couldn't create point", 1)
2013-05-06 18:17:07 +04:00
}
2013-05-15 11:49:38 +04:00
g.DestroyLater(geom)
return geom, nil
2013-05-06 18:17:07 +04:00
}
func nodesEqual(a, b element.Node) bool {
if d := a.Long - b.Long; math.Abs(d) < 1e-9 {
if d := a.Lat - b.Lat; math.Abs(d) < 1e-9 {
return true
}
}
return false
}
func unduplicateNodes(nodes []element.Node) []element.Node {
if len(nodes) < 2 {
return nodes
}
foundDup := false
for i := 1; i < len(nodes); i++ {
if nodesEqual(nodes[i-1], nodes[i]) {
foundDup = true
break
}
}
if !foundDup {
return nodes
}
result := make([]element.Node, 0, len(nodes))
result = append(result, nodes[0])
for i := 1; i < len(nodes); i++ {
if nodesEqual(nodes[i-1], nodes[i]) {
continue
}
result = append(result, nodes[i])
}
return result
}
func LineString(g *geos.Geos, nodes []element.Node) (*geos.Geom, error) {
nodes = unduplicateNodes(nodes)
2013-05-10 12:08:31 +04:00
if len(nodes) < 2 {
return nil, ErrorOneNodeWay
}
2013-05-06 21:24:49 +04:00
coordSeq, err := g.CreateCoordSeq(uint32(len(nodes)), 2)
2013-04-24 00:02:27 +04:00
if err != nil {
return nil, err
}
// coordSeq inherited by LineString
2013-04-21 18:37:05 +04:00
for i, nd := range nodes {
2013-05-06 21:24:49 +04:00
coordSeq.SetXY(g, uint32(i), nd.Long, nd.Lat)
2013-04-21 18:37:05 +04:00
}
2013-05-06 21:24:49 +04:00
geom, err := coordSeq.AsLineString(g)
2013-05-28 13:10:19 +04:00
if err != nil {
// coordSeq gets Destroy by GEOS
return nil, err
}
g.DestroyLater(geom)
return geom, nil
2013-05-16 14:17:21 +04:00
}
func Polygon(g *geos.Geos, nodes []element.Node) (*geos.Geom, error) {
nodes = unduplicateNodes(nodes)
if len(nodes) < 4 {
return nil, ErrorNoRing
}
2013-05-06 21:24:49 +04:00
coordSeq, err := g.CreateCoordSeq(uint32(len(nodes)), 2)
2013-04-24 00:02:27 +04:00
if err != nil {
return nil, err
}
2013-05-28 13:10:19 +04:00
// coordSeq inherited by LinearRing, no destroy
2013-04-24 00:02:27 +04:00
for i, nd := range nodes {
2013-05-06 21:24:49 +04:00
err := coordSeq.SetXY(g, uint32(i), nd.Long, nd.Lat)
2013-04-24 00:02:27 +04:00
if err != nil {
return nil, err
}
}
ring, err := coordSeq.AsLinearRing(g)
2013-04-24 00:02:27 +04:00
if err != nil {
2013-05-28 13:10:19 +04:00
// coordSeq gets Destroy by GEOS
2013-04-24 00:02:27 +04:00
return nil, err
}
// ring inherited by Polygon, no destroy
2013-05-16 14:17:21 +04:00
geom := g.Polygon(ring, nil)
if geom == nil {
g.Destroy(ring)
return nil, errors.New("unable to create polygon")
2013-04-24 00:02:27 +04:00
}
g.DestroyLater(geom)
2013-05-16 14:17:21 +04:00
return geom, nil
2013-04-24 00:02:27 +04:00
}
func AsGeomElement(g *geos.Geos, geom *geos.Geom) (Geometry, error) {
wkb := g.AsEwkbHex(geom)
if wkb == nil {
return Geometry{}, errors.New("could not create wkb")
}
return Geometry{
Wkb: wkb,
Geom: geom,
}, nil
}