photoprism/internal/meta/gps.go

89 lines
1.9 KiB
Go

package meta
import (
"regexp"
"strconv"
"github.com/dsoprea/go-exif/v3"
)
var GpsCoordsRegexp = regexp.MustCompile("[0-9\\.]+")
var GpsRefRegexp = regexp.MustCompile("[NSEW]+")
var GpsFloatRegexp = regexp.MustCompile("[+\\-]?(?:(?:0|[1-9]\\d*)(?:\\.\\d*)?|\\.\\d+)")
// GpsToLatLng returns the GPS latitude and longitude as float point number.
func GpsToLatLng(s string) (lat, lng float32) {
if s == "" {
return 0, 0
}
// Floating point numbers?
if fl := GpsFloatRegexp.FindAllString(s, -1); len(fl) == 2 {
lat, _ := strconv.ParseFloat(fl[0], 64)
lng, _ := strconv.ParseFloat(fl[1], 64)
return float32(lat), float32(lng)
}
// Parse human readable strings.
co := GpsCoordsRegexp.FindAllString(s, -1)
re := GpsRefRegexp.FindAllString(s, -1)
if len(co) != 6 || len(re) != 2 {
return 0, 0
}
latDeg := exif.GpsDegrees{
Orientation: re[0][0],
Degrees: GpsCoord(co[0]),
Minutes: GpsCoord(co[1]),
Seconds: GpsCoord(co[2]),
}
lngDeg := exif.GpsDegrees{
Orientation: re[1][0],
Degrees: GpsCoord(co[3]),
Minutes: GpsCoord(co[4]),
Seconds: GpsCoord(co[5]),
}
return float32(latDeg.Decimal()), float32(lngDeg.Decimal())
}
// GpsToDecimal returns the GPS latitude or longitude as decimal float point number.
func GpsToDecimal(s string) float32 {
if s == "" {
return 0
}
co := GpsCoordsRegexp.FindAllString(s, -1)
re := GpsRefRegexp.FindAllString(s, -1)
if len(co) != 3 || len(re) != 1 {
return 0
}
latDeg := exif.GpsDegrees{
Orientation: re[0][0],
Degrees: GpsCoord(co[0]),
Minutes: GpsCoord(co[1]),
Seconds: GpsCoord(co[2]),
}
return float32(latDeg.Decimal())
}
// GpsCoord returns a single GPS coordinate value as floating point number (degree, minute or second).
func GpsCoord(s string) float64 {
if s == "" {
return 0
}
result, err := strconv.ParseFloat(s, 64)
if err != nil {
log.Debugf("metadata: failed parsing GPS coordinate '%s'", s)
return 0
}
return result
}