souplesse/lib/Track.hs

118 lines
3.0 KiB
Haskell
Raw Normal View History

2024-10-28 23:35:36 +00:00
{-# LANGUAGE OverloadedStrings #-}
2024-10-29 19:21:25 +00:00
module Track (
Track,
Pos(..),
pos,
elevation,
2024-10-30 17:17:48 +00:00
cadence,
power,
2024-10-30 17:44:40 +00:00
time,
2024-10-29 19:21:25 +00:00
parse,
Track.length
) where
2024-10-27 23:13:39 +00:00
import Data.Time
2024-10-30 17:44:40 +00:00
import qualified Data.Time.ISO8601
2024-10-27 23:13:39 +00:00
import qualified Data.List
2024-10-28 23:35:36 +00:00
import Text.XML
import Text.XML.Cursor as Cursor
import qualified Data.Text
import Data.Text.Lazy as T
import Debug.Trace (trace, traceShow)
import Data.List as List
import Data.Map as Map
import Control.Exception
data Pos = Pos Float Float deriving (Show, Eq)
2024-10-28 23:35:36 +00:00
type Power = Maybe Int
type Cadence = Maybe Int
type HeartRate = Maybe Int
data Point = Point {
pos :: Pos,
elevation :: Maybe Float,
2024-10-28 23:35:36 +00:00
time :: UTCTime,
cadence :: Cadence,
2024-10-30 17:17:48 +00:00
power :: Power,
2024-10-28 23:35:36 +00:00
heartRate :: HeartRate
} deriving (Show)
2024-10-27 23:13:39 +00:00
2024-10-29 19:35:17 +00:00
-- TODO do we even need this type?
2024-10-27 23:13:39 +00:00
type Track = [Point]
2024-10-30 17:17:48 +00:00
gpxNS localName =
Name localName (Just "http://www.topografix.com/GPX/1/1") Nothing
tpxNS localName =
Name localName (Just "http://www.garmin.com/xmlschemas/TrackPointExtension/v2") Nothing
2024-10-30 13:30:33 +00:00
2024-10-28 23:35:36 +00:00
mkPoint pos =
Point
pos
2024-10-29 21:22:49 +00:00
Nothing
2024-10-28 23:35:36 +00:00
(UTCTime (toEnum 60631) 43200)
2024-10-30 17:17:48 +00:00
Nothing
Nothing
Nothing
2024-10-28 23:35:36 +00:00
elToPoint :: Cursor -> Point
2024-10-29 21:22:49 +00:00
elToPoint c =
case node c of
2024-10-28 23:35:36 +00:00
NodeElement (Element _ attrs _) ->
let
2024-10-29 19:35:17 +00:00
lat = getAttr "lat"
2024-10-28 23:35:36 +00:00
lon = getAttr "lon"
2024-10-30 17:17:48 +00:00
ele = child c >>= element (gpxNS "ele") >>= child >>= content
2024-10-30 17:44:40 +00:00
ts = child c >>= element (gpxNS "time") >>= child >>= content
2024-10-30 17:17:48 +00:00
gpxtpx = child c >>=
element (gpxNS "extensions")
>>= child
>>= element (tpxNS "TrackPointExtension") >>= child
cadence = gpxtpx >>= element (tpxNS "cad")
>>= child >>= content
power = gpxtpx >>= element (Name "PowerInWatts" (Just "http://www.garmin.com/xmlschemas/PowerExtension/v1") Nothing)
>>= child >>= content
2024-10-29 21:22:49 +00:00
in Point (Pos lat lon)
(case ele of
2024-10-30 14:25:54 +00:00
[e] -> Just $ asFloat e
2024-10-29 21:22:49 +00:00
_ -> Nothing)
2024-10-30 17:44:40 +00:00
(case ts of
[e] -> case Data.Time.ISO8601.parseISO8601 (Data.Text.unpack e) of
Just utime -> utime
_ -> UTCTime (toEnum 0) 0
_ -> UTCTime (toEnum 0) 0)
2024-10-30 17:17:48 +00:00
(case cadence of
[e] -> Just (asInt e)
_ -> Nothing)
(case power of
[e] -> Just (asInt e)
_ -> Nothing)
Nothing
2024-10-29 21:22:49 +00:00
where
asFloat v = (read (Data.Text.unpack v) :: Float)
2024-10-30 17:17:48 +00:00
asInt v = (read (Data.Text.unpack v) :: Int)
2024-10-29 21:22:49 +00:00
getAttr name =
2024-10-30 14:25:54 +00:00
case Map.lookup name attrs of
2024-10-29 21:22:49 +00:00
Just v -> asFloat v
_ -> 0
_ -> mkPoint (Pos 0 0)
2024-10-28 23:35:36 +00:00
getPoints :: Cursor -> [Point]
getPoints c =
let
trkpts =
2024-10-30 17:17:48 +00:00
element (gpxNS "gpx") c >>= child >>=
element (gpxNS "trk") >>= descendant >>=
element (gpxNS "trkpt")
2024-10-28 23:35:36 +00:00
in
2024-10-30 14:25:54 +00:00
List.map elToPoint trkpts
2024-10-28 23:35:36 +00:00
parse :: String -> Either SomeException [Point]
parse str = do
gpx <- parseText def (T.pack str)
return (getPoints (fromDocument gpx))
2024-10-28 23:35:36 +00:00
2024-10-27 23:13:39 +00:00
length :: Track -> Int
2024-10-30 14:25:54 +00:00
length = Data.List.length