Skip to content

Instantly share code, notes, and snippets.

@3noch
Last active January 30, 2016 20:25
Show Gist options
  • Star 1 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save 3noch/e6f0aded09b6aaeb8f48 to your computer and use it in GitHub Desktop.
Save 3noch/e6f0aded09b6aaeb8f48 to your computer and use it in GitHub Desktop.
Reflex-DOM Geolocation
module Geoposition where
import Control.Concurrent (forkIO)
import Control.Monad.Exception (catch, throw)
import Control.Monad.IO.Class (liftIO)
import Reflex (Event)
import Reflex.Dom (MonadWidget)
import Reflex.Dom.Class (performEventAsync)
import GHCJS.DOM (currentWindow)
import GHCJS.DOM.PositionError (PositionException(..), PositionErrorCode(..))
import qualified GHCJS.DOM.Coordinates as Coord
import GHCJS.DOM.Geolocation (getCurrentPosition)
import GHCJS.DOM.Geoposition (getCoords)
import GHCJS.DOM.Navigator (getGeolocation)
import GHCJS.DOM.Window (getNavigator)
defaultPosException :: PositionException
defaultPosException = PositionException PositionUnavailable "Failed to get geolocation"
data GeopositionInfo = GeopositionInfo
{ geoLatitude :: Double
, geoLongitude :: Double
, geoAltitudeMeters :: Maybe Double
, geoAccuracyMeters :: Double
, geoAltitudeAccuracyMeters :: Maybe Double
, geoHeadingDegrees :: Maybe Double
, geoSpeedMetersPerSec :: Maybe Double
} deriving (Show, Eq)
getGeopositionInfo :: IO (Either PositionException GeopositionInfo)
getGeopositionInfo = (Right <$> getInfo) `catch` (pure . Left)
where
getInfo = do
window <- currentWindow >>= orBombOut
nav <- getNavigator window >>= orBombOut
geoloc <- getGeolocation nav >>= orBombOut
geopos <- getCurrentPosition geoloc Nothing
coord <- getCoords geopos >>= orBombOut
GeopositionInfo
<$> Coord.getLatitude coord
<*> Coord.getLongitude coord
<*> Coord.getAltitude coord
<*> Coord.getAccuracy coord
<*> Coord.getAltitudeAccuracy coord
<*> Coord.getHeading coord
<*> Coord.getSpeed coord
orBombOut = maybe (throw defaultPosException) pure
attachGeoposition :: MonadWidget t m => Event t a -> m (Event t (Either PositionException GeopositionInfo, a))
attachGeoposition event = performEventAsync (fetchInfoAsync <$> event)
where
fetchInfoAsync a callback = liftIO $ do
_ <- forkIO $ do
info <- getGeopositionInfo
callback (info, a)
pure ()
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment